"""!

@brief CCORE Wrapper for oscillatory neural network for pattern recognition (syncpr).

@authors Andrei Novikov (pyclustering@yandex.ru)
@date 2014-2020
@copyright BSD-3-Clause

"""

from pyclustering.core.wrapper import *;
from pyclustering.core.pyclustering_package import pyclustering_package, package_extractor, package_builder;


def pack_pattern(pattern):
    return package_builder(pattern, c_int).create();


def syncpr_create(num_osc, increase_strength1, increase_strength2):
    ccore = ccore_library.get();
    
    ccore.syncpr_create.restype = POINTER(c_void_p);
    pointer_network = ccore.syncpr_create(c_uint(num_osc), c_double(increase_strength1), c_double(increase_strength2));
    
    return pointer_network;


def syncpr_destroy(pointer_network):
    ccore = ccore_library.get();
    ccore.syncpr_destroy(pointer_network);


def syncpr_get_size(pointer_network):
    ccore = ccore_library.get();
    ccore.syncpr_get_size.restype = c_size_t;
    return ccore.syncpr_get_size(pointer_network);
    

def syncpr_train(pointer_network, patterns):
    c_patterns = package_builder(patterns, c_int).create();
    
    ccore = ccore_library.get();
    ccore.syncpr_train(pointer_network, c_patterns);
    
    
def syncpr_simulate_static(pointer_network, steps, time, pattern, solution, collect_dynamic):
    package_pattern = pack_pattern(pattern);
    
    ccore = ccore_library.get();
    ccore.syncpr_simulate_static.restype = POINTER(c_void_p);
    return ccore.syncpr_simulate_static(pointer_network, c_uint(steps), c_double(time), package_pattern, c_uint(solution), c_bool(collect_dynamic));


def syncpr_simulate_dynamic(pointer_network, pattern, order, solution, collect_dynamic, step):
    package_pattern = pack_pattern(pattern);
    
    ccore = ccore_library.get();
    ccore.syncpr_simulate_dynamic.restype = POINTER(c_void_p);
    return ccore.syncpr_simulate_dynamic(pointer_network, package_pattern, c_double(order), c_uint(solution), c_bool(collect_dynamic), c_double(step));


def syncpr_memory_order(pointer_network, pattern):
    package_pattern = pack_pattern(pattern);
    
    ccore = ccore_library.get();
    
    ccore.syncpr_memory_order.restype = c_double;
    return ccore.syncpr_memory_order(pointer_network, package_pattern);


def syncpr_dynamic_get_size(pointer_dynamic):
    ccore = ccore_library.get();
    ccore.syncpr_dynamic_get_size.restype = c_uint;
    return ccore.syncpr_dynamic_get_size(pointer_dynamic);


def syncpr_dynamic_destroy(pointer_dynamic):
    ccore = ccore_library.get();
    ccore.syncpr_dynamic_destroy(pointer_dynamic);


def syncpr_dynamic_allocate_sync_ensembles(pointer_dynamic, tolerance):
    ccore = ccore_library.get();
    
    ccore.syncpr_dynamic_allocate_sync_ensembles.restype = POINTER(pyclustering_package);
    package = ccore.syncpr_dynamic_allocate_sync_ensembles(pointer_dynamic, c_double(tolerance));
    
    result = package_extractor(package).extract();
    ccore.free_pyclustering_package(package);
    
    return result;


def syncpr_dynamic_get_output(pointer_dynamic):
    ccore = ccore_library.get();
    
    ccore.syncpr_dynamic_get_output.restype = POINTER(pyclustering_package);
    package = ccore.syncpr_dynamic_get_output(pointer_dynamic);
    
    result = package_extractor(package).extract();
    ccore.free_pyclustering_package(package);
    
    return result;


def syncpr_dynamic_get_time(pointer_dynamic):
    ccore = ccore_library.get();
    
    ccore.syncpr_dynamic_get_time.restype = POINTER(pyclustering_package);
    package = ccore.syncpr_dynamic_get_time(pointer_dynamic);
    
    result = package_extractor(package).extract();
    ccore.free_pyclustering_package(package);
    
    return result;