/*
www.sourceforge.net/projects/dfhack
Copyright (c) 2009 Petr Mrázek (peterix), Kenneth Ferland (Impaler[WrG]), dorf

This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any
damages arising from the use of this software.

Permission is granted to anyone to use this software for any
purpose, including commercial applications, and to alter it and
redistribute it freely, subject to the following restrictions:

1. The origin of this software must not be misrepresented; you must
not claim that you wrote the original software. If you use this
software in a product, an acknowledgment in the product documentation
would be appreciated but is not required.

2. Altered source versions must be plainly marked as such, and
must not be misrepresented as being the original software.

3. This notice may not be removed or altered from any source
distribution.
*/

#include "Internal.h"
#include "PlatformInternal.h"
#include <vector>

using namespace std;
#include "ContextShared.h"
#include "dfhack/modules/WindowIO.h"
#include "dfhack/DFProcess.h"
#include "ModuleFactory.h"

using namespace DFHack;

Module* DFHack::createWindowIO(DFContextShared * d)
{
    return new WindowIO(d);
}

// should always reflect the enum in DFkeys.h
const static int ksTable[NUM_SPECIALS]=
{
    VK_RETURN,
    VK_SPACE,
    VK_BACK,
    VK_TAB,
    VK_CAPITAL,
    VK_LSHIFT,
    VK_RSHIFT,
    VK_LCONTROL,
    VK_RCONTROL,
    VK_MENU,
    0, //XK_VoidSymbol, // WAIT
    VK_ESCAPE,
    VK_UP,
    VK_DOWN,
    VK_LEFT,
    VK_RIGHT,
    VK_F1,
    VK_F2,
    VK_F3,
    VK_F4,
    VK_F5,
    VK_F6,
    VK_F7,
    VK_F8,
    VK_F9,
    VK_F10,
    VK_F11,
    VK_F12,
    VK_PRIOR,
    VK_NEXT,
    VK_INSERT,
    VK_DELETE,
    VK_HOME,
    VK_END,
    VK_DIVIDE,
    VK_MULTIPLY,
    VK_SUBTRACT,
    VK_ADD,
    VK_RETURN,
    VK_NUMPAD0,
    VK_NUMPAD1,
    VK_NUMPAD2,
    VK_NUMPAD3,
    VK_NUMPAD4,
    VK_NUMPAD5,
    VK_NUMPAD6,
    VK_NUMPAD7,
    VK_NUMPAD8,
    VK_NUMPAD9,
    VK_SEPARATOR
};
//Windows key handlers
struct window
{
    HWND windowHandle;
    uint32_t pid;
};

BOOL CALLBACK EnumWindowsProc (HWND hwnd, LPARAM lParam)
{
    uint32_t pid;
    
    GetWindowThreadProcessId (hwnd, (LPDWORD) &pid);
    if (pid == ( (window *) lParam)->pid)
    {
        ( (window *) lParam)->windowHandle = hwnd;
        return FALSE;
    }
        return TRUE;
}

class WindowIO::Private
{
    public:
        Private(Process * _p)
        {
            p=_p;
        };
        ~Private(){};
        // our parent process
        Process * p;
};

// ctor
WindowIO::WindowIO (DFContextShared * schr)
{
    d = new Private(schr->p);
}

// dtor
WindowIO::~WindowIO ()
{
    delete d;
}

// TODO: also investigate possible problems with UIPI on Vista and 7
void WindowIO::TypeStr (const char *input, int delay, bool useShift)
{
    //sendmessage needs a window handle HWND, so have to get that from the process HANDLE
    window myWindow;
    myWindow.pid = d->p->getPID();
    EnumWindows (EnumWindowsProc, (LPARAM) &myWindow);
    char cChar;
    DWORD dfProcess = GetWindowThreadProcessId(myWindow.windowHandle,NULL);
    DWORD currentProcess = GetCurrentThreadId();
    AttachThreadInput(currentProcess,dfProcess,TRUE); //The two threads have to have attached input in order to change the keyboard state, which is needed to set the shift state
    while ( (cChar = *input++)) // loops through chars
    {
        short vk = VkKeyScan (cChar); // keycode of char
        if (useShift || (vk >> 8) &1)   // char is capital, so need to hold down shift
        {
            vk = vk & 0xFF; // remove the shift state from the virtual key code
            BYTE keybstate[256] = {0};
            BYTE keybstateOrig[256] = {0};
            GetKeyboardState((LPBYTE)&keybstateOrig);
            GetKeyboardState((LPBYTE)&keybstate);
            keybstate[VK_SHIFT] |= 0x80; //Set shift state to on in variable
            SetKeyboardState((LPBYTE)&keybstate); //set the current state to the variable
            SendMessage(myWindow.windowHandle,WM_KEYDOWN,vk,0);
            SendMessage(myWindow.windowHandle,WM_KEYUP,vk,0);
            SetKeyboardState((LPBYTE)&keybstateOrig); // reset the shift state to the original state
        }
        else
        {
            SendMessage(myWindow.windowHandle,WM_KEYDOWN,vk,0);
            SendMessage(myWindow.windowHandle,WM_KEYUP,vk,0);
        }
    }
    AttachThreadInput(currentProcess,dfProcess,FALSE); //detach the threads
    Sleep (delay);
}

void WindowIO::TypeSpecial (t_special command, int count, int delay)
{
    if (command != WAIT)
    {
        HWND currentWindow = GetForegroundWindow();
        window myWindow;
        myWindow.pid = d->p->getPID();
        EnumWindows (EnumWindowsProc, (LPARAM) &myWindow);

        for (int i = 0; i < count;i++)
        {
            SendMessage(myWindow.windowHandle,WM_KEYDOWN,ksTable[command],0);
            SendMessage(myWindow.windowHandle,WM_KEYUP,ksTable[command],0);
        }
        Sleep (delay);
    }
    else
    {
        Sleep (delay*count);
    }
}