#include "../ethos.hpp" #include "hotkeys.cpp" InputManager* inputManager = nullptr; void AbstractInput::bind() { inputList.reset(); lstring list = mapping.split(","); for(auto& mapping : list) { Input::Type type; if(mapping.endsWith(".Up")) type = Input::Type::HatUp; else if(mapping.endsWith(".Down")) type = Input::Type::HatDown; else if(mapping.endsWith(".Left")) type = Input::Type::HatLeft; else if(mapping.endsWith(".Right")) type = Input::Type::HatRight; else if(mapping.endsWith(".Lo")) type = Input::Type::AxisLo; else if(mapping.endsWith(".Hi")) type = Input::Type::AxisHi; else if(mapping.beginsWith("JP") && mapping.find("Axis")) type = Input::Type::Axis; else if(mapping.beginsWith("JP") && mapping.endsWith("Rumble")) type = Input::Type::Rumble; else if(mapping.beginsWith("MS") && mapping.endsWith("axis")) type = Input::Type::MouseAxis; else if(mapping.beginsWith("MS")) type = Input::Type::MouseButton; else type = Input::Type::Button; if(type == Input::Type::Rumble) { unsigned joypad = mapping[2] - '0'; inputList.append({type, joypad}); } else { string decode = mapping; if(auto position = decode.find(".")) decode.resize(position()); unsigned scancode = Scancode::decode(decode); inputList.append({type, scancode}); } } } bool AbstractInput::append(string encode) { if(mapping.find(encode)) return true; //mapping already bound if(mapping.empty() || mapping == "None") mapping = encode; //remove "None" else mapping.append(",", encode); //add to existing mapping list bind(); return true; } AbstractInput::AbstractInput() : state(false) { } // bool DigitalInput::bind(unsigned scancode, int16_t value) { using nall::Keyboard; using nall::Mouse; if(scancode == Scancode::None || scancode == keyboard(0)[Keyboard::Escape]) { inputList.reset(); mapping = "None"; return true; } string encode = Scancode::encode(scancode); if(Keyboard::isAnyKey(scancode) || Keyboard::isAnyModifier(scancode) || Joypad::isAnyButton(scancode)) { if(value == 0) return false; return append(encode); } if(Mouse::isAnyButton(scancode)) { if(value == 0) return false; return append(encode); } if(Joypad::isAnyHat(scancode)) { if(value & Joypad::HatUp ) { encode.append(".Up" ); return append(encode); } if(value & Joypad::HatDown ) { encode.append(".Down" ); return append(encode); } if(value & Joypad::HatLeft ) { encode.append(".Left" ); return append(encode); } if(value & Joypad::HatRight) { encode.append(".Right"); return append(encode); } } if(Joypad::isAnyAxis(scancode)) { if(value < -12288) { encode.append(".Lo"); return append(encode); } if(value > +24576) { encode.append(".Hi"); return append(encode); } } return false; } int16_t DigitalInput::poll() { if(program->focused() == false) return 0; bool result = logic; for(auto& item : inputList) { int16_t value = inputManager->poll(item.scancode); bool output = logic; switch(item.type) { case Input::Type::Button: output = value; break; case Input::Type::MouseButton: output = value & input.acquired(); break; case Input::Type::HatUp: output = value & Joypad::HatUp; break; case Input::Type::HatDown: output = value & Joypad::HatDown; break; case Input::Type::HatLeft: output = value & Joypad::HatLeft; break; case Input::Type::HatRight: output = value & Joypad::HatRight; break; case Input::Type::AxisLo: output = value < -16384; break; case Input::Type::AxisHi: output = value > +16384; break; } if(logic == 0) result |= output; if(logic == 1) result &= output; } return result; } // bool RelativeInput::bind(unsigned scancode, int16_t value) { using nall::Keyboard; using nall::Mouse; if(scancode == Scancode::None || scancode == keyboard(0)[Keyboard::Escape]) { inputList.reset(); mapping = "None"; return true; } string encode = Scancode::encode(scancode); if(Mouse::isAnyAxis(scancode)) return append(encode); if(Joypad::isAnyAxis(scancode)) return append(encode); return false; } int16_t RelativeInput::poll() { if(program->focused() == false) return 0; int16_t result = 0; for(auto& item : inputList) { int16_t value = inputManager->poll(item.scancode); switch(item.type) { case Input::Type::MouseAxis: value = input.acquired() ? value : 0; break; case Input::Type::Axis: value = value; break; } result += value; } return result; } // bool AbsoluteInput::bind(unsigned scancode, int16_t value) { using nall::Keyboard; using nall::Mouse; if(scancode == Scancode::None || scancode == keyboard(0)[Keyboard::Escape]) { inputList.reset(); mapping = "None"; return true; } string encode = Scancode::encode(scancode); if(Mouse::isAnyAxis(scancode)) { //only one input can be assigned for absolute positioning inputList.reset(); mapping = encode; return true; } return false; } int16_t AbsoluteInput::poll() { if(program->focused() == false) return -32768; int16_t result = -32768; //offscreen value using nall::Mouse; Position position = phoenix::Mouse::position(); Geometry geometry = presentation->geometry(); if(position.x < geometry.x || position.y < geometry.y || position.x >= geometry.x + geometry.width || position.y >= geometry.y + geometry.height) { //cursor is offscreen position.x = -32768; position.y = -32768; } else { //convert from screen to viewport coordinates double x = position.x - geometry.x; double y = position.y - geometry.y; //scale coordinate range to -0.5 to +0.5 (0.0 = center) x = x * 1.0 / geometry.width - 0.5; y = y * 1.0 / geometry.height - 0.5; //scale coordinates to -32767 to +32767 signed px = (signed)(x * 65535.0); signed py = (signed)(y * 65535.0); //clamp to valid range position.x = max(-32767, min(+32767, px)); position.y = max(-32767, min(+32767, py)); } for(auto& item : inputList) { if(item.scancode == mouse(0)[Mouse::Xaxis]) { result = position.x; } if(item.scancode == mouse(0)[Mouse::Yaxis]) { result = position.y; } } return result; } bool RumbleInput::bind(unsigned scancode, int16_t value) { using nall::Keyboard; if(scancode == Scancode::None || scancode == keyboard(0)[Keyboard::Escape]) { inputList.reset(); mapping = "None"; return true; } string encode = Scancode::encode(scancode); if(Joypad::isAnyButton(scancode)) { if(value == 0) return false; if(auto position = encode.find("::")) encode.resize(position()); encode.append("::Rumble"); return append(encode); } return false; } int16_t RumbleInput::poll() { return false; } void RumbleInput::rumble(bool enable) { if(program->focused() == false) return; for(auto& item : inputList) { input.rumble(item.scancode, enable); } } // HotkeyInput::HotkeyInput() { logic = 1; //AND inputManager->hotkeyMap.append(this); } // void InputManager::bind() { for(auto& input : inputMap) input->bind(); for(auto& input : hotkeyMap) input->bind(); } void InputManager::poll() { using nall::Keyboard; activeScancode = !activeScancode; if(input.poll(scancode[activeScancode]) == false) return; for(unsigned n = 0; n < Scancode::Limit; n++) { if(scancode[0][n] != scancode[1][n]) { if(settings->focused()) { inputSettings->inputEvent(n, scancode[activeScancode][n]); hotkeySettings->inputEvent(n, scancode[activeScancode][n]); } } } if(presentation->focused()) pollHotkeys(); } int16_t InputManager::poll(unsigned scancode) { return this->scancode[activeScancode][scancode]; } void InputManager::saveConfiguration() { config.save(program->path("input.bml")); } InputManager::InputManager() { inputManager = this; scancode[0] = new int16_t[Scancode::Limit](); scancode[1] = new int16_t[Scancode::Limit](); activeScancode = 0; bootstrap(); } InputManager::~InputManager() { delete[] scancode[0]; delete[] scancode[1]; } void InputManager::bootstrap() { unsigned guid = 0; for(auto& emulator : program->emulator) { Configuration::Node emulatorNode; for(auto& port : emulator->port) { Configuration::Node portNode; for(auto& device : port.device) { Configuration::Node deviceNode; for(auto& number : device.order) { auto& input = device.input[number]; AbstractInput* abstract = nullptr; if(input.type == 0) abstract = new DigitalInput; if(input.type == 1) abstract = new RelativeInput; if(input.type == 2) abstract = new AbsoluteInput; if(input.type == 3) abstract = new RumbleInput; if(abstract == nullptr) continue; abstract->name = string{input.name}.replace(" ", ""); abstract->mapping = "None"; abstract->logic = 0; //OR input.guid = guid++; inputMap.append(abstract); deviceNode.append(abstract->mapping, abstract->name); } portNode.append(deviceNode, string{device.name}.replace(" ", "")); } emulatorNode.append(portNode, string{port.name}.replace(" ", "")); } config.append(emulatorNode, string{emulator->information.name}.replace(" ", "")); } appendHotkeys(); config.load(program->path("input.bml")); config.save(program->path("input.bml")); bind(); }