/* Copyright (c) 2015 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 "ftnoir_protocol_mouse.h"

#include "api/plugin-api.hpp"
#include "compat/math.hpp"

#include <cmath>
#include <algorithm>

#include <windows.h>

#ifndef MOUSEEVENTF_MOVE_NOCOALESCE
#   define MOUSEEVENTF_MOVE_NOCOALESCE 0x2000
#endif

static const double invert[] = {
    1., 1., 1.,
    1., -1., 1.
};

void mouse::pose(const double* headpose, const double*)
{
    const int axis_x = s.mouse_x - 1;
    const int axis_y = s.mouse_y - 1;

    int mouse_x = 0, mouse_y = 0;

    if (axis_x == clamp(axis_x, Axis_MIN, Axis_MAX))
        mouse_x = get_value(headpose[axis_x] * invert[axis_x],
                            *s.sensitivity_x,
                            axis_x >= 3);

    if (axis_y == clamp(axis_y, Axis_MIN, Axis_MAX))
        mouse_y = get_value(headpose[axis_y] * invert[axis_y],
                            *s.sensitivity_y,
                            axis_y >= 3);

    int dx = get_delta(mouse_x, last_x), dy = mouse_y - last_y;

    last_x = mouse_x; last_y = mouse_y;

    if (dx || dy)
    {
        switch (s.input_method)
        {
        default:
            eval_once(qDebug() << "proto/mouse: invalid input method");
            [[fallthrough]];
        case input_direct:
        {
            INPUT input;
            input.type = INPUT_MOUSE;
            MOUSEINPUT& mi = input.mi;
            mi = {};
            mi.dx = dx;
            mi.dy = dy;
            mi.dwFlags = MOUSEEVENTF_MOVE | MOUSEEVENTF_MOVE_NOCOALESCE;

            (void)SendInput(1, &input, sizeof(input));

            break;
        }
        case input_legacy:
        {
            POINT pt{};
            (void)GetCursorPos(&pt);
            (void)SetCursorPos(pt.x + dx, pt.y + dy);

            break;
        }
        }
    }
}

QString mouse::game_name()
{
    return tr("Mouse tracker");
}

int mouse::get_delta(int val, int prev)
{
    const int a = std::abs(val - prev), b = std::abs(val + prev);
    if (b < a)
        return val + prev;
    else
        return val - prev;
}

int mouse::get_value(double val, double sensitivity, bool is_rotation)
{
    constexpr double c[] = { 1e-3, 1e-1 };

    return iround(val * sensitivity * c[unsigned(is_rotation)]);
}

OPENTRACK_DECLARE_PROTOCOL(mouse, MOUSEControls, mouseDll)