/* Copyright (c) 2016, Stanislaw Halik <sthalik@misaki.pl>

 * Permission to use, copy, modify, and/or distribute this
 * software for any purpose with or without fee is hereby granted,
 * provided that the above copyright notice and this permission
 * notice appear in all copies.
 */

#include "compat/util.hpp"
#include "connector.hpp"
#include "value.hpp"

#include <utility>

namespace options {
namespace detail {

static bool generic_is_equal(const QVariant& val1, const QVariant& val2)
{
    return val1 == val2;
}

connector::~connector() {}

bool connector::is_equal(const QString& name, const QVariant& val1, const QVariant& val2) const
{
    QMutexLocker l(get_mtx());

    auto it = connected_values.find(name);

    if (it != connected_values.cend() && std::get<0>((*it).second).size() != 0)
        return std::get<1>((*it).second)(val1, val2);
    else
        return generic_is_equal(val1, val2);
}

bool connector::on_value_destructed_impl(const QString& name, const base_value* val)
{
    QMutexLocker l(get_mtx());

    const bool ok = progn(
        auto it = connected_values.find(name);

        if (it != connected_values.end())
        {
            std::vector<const base_value*>& values = std::get<0>((*it).second);
            for (auto it = values.begin(); it != values.end(); it++)
            {
                if (*it == val)
                {
                    values.erase(it);
                    return true;
                }
            }
        }
        return false;
    );

    return ok;
}



void connector::on_value_destructed(const QString& name, const base_value* val)
{
    if (!name.size())
        return;

    const bool ok = on_value_destructed_impl(name, val);

    if (!ok)
        qWarning() << "options/connector: value destructed without creating;"
                   << "bundle"
                   << val->b->name()
                   << "value-name" << name
                   << "value-ptr" << quintptr(val);
}

void connector::on_value_created(const QString& name, const base_value* val)
{
    if (!name.size())
        return;

    QMutexLocker l(get_mtx());

    if (on_value_destructed_impl(name, val))
    {
        qWarning() << "options/connector: value created twice;"
                   << "bundle"
                   << val->b->name()
                   << "value-name" << name
                   << "value-ptr" << quintptr(val);
    }

    auto it = connected_values.find(name);

    if (it != connected_values.end())
    {
        tt& tmp = (*it).second;
        std::type_index& typeidx = std::get<2>(tmp);
        std::vector<const base_value*>& values = std::get<0>(tmp);

        if (typeidx != val->type_index)
            std::get<1>((*it).second) = generic_is_equal;
        values.push_back(val);
    }
    else
    {
        std::vector<const base_value*> vec;
        vec.push_back(val);
        connected_values.emplace(name, tt(vec, val->cmp, val->type_index));
    }
}

void connector::notify_values(const QString& name) const
{
    auto it = connected_values.find(name);
    if (it != connected_values.cend())
    {
        for (const base_value* val : std::get<0>((*it).second))
        {
            val->bundle_value_changed();
        }
    }
}

void connector::notify_all_values() const
{
    for (auto& pair : connected_values)
        for (const base_value* val : std::get<0>(pair.second))
            val->bundle_value_changed();
}

connector::connector()
{
}

}

}