diff --git a/Machines/Apple/AppleII/AppleII.cpp b/Machines/Apple/AppleII/AppleII.cpp index 34d8ad874..0c5d2d811 100644 --- a/Machines/Apple/AppleII/AppleII.cpp +++ b/Machines/Apple/AppleII/AppleII.cpp @@ -804,38 +804,62 @@ template class ConcreteMachine: m6502_.run_for(cycles); } - void reset_all_keys(Inputs::Keyboard *) final { + void reset_all_keys() final { open_apple_is_pressed_ = closed_apple_is_pressed_ = key_is_down_ = false; } + bool prefers_logical_input() final { + return true; + } + bool set_key_pressed(Key key, char value, bool is_pressed) final { + // If no ASCII value is supplied, look for a few special cases. switch(key) { - default: break; - case Key::F12: - m6502_.set_reset_line(is_pressed); - return true; + case Key::Left: value = 0x08; break; + case Key::Right: value = 0x15; break; + case Key::Down: value = 0x0a; break; + case Key::Up: value = 0x0b; break; + case Key::Backspace: value = 0x7f; break; + case Key::Enter: value = 0x0d; break; + case Key::Tab: value = '\t'; break; + case Key::Escape: value = 0x1b; break; + case Key::LeftOption: + case Key::RightMeta: open_apple_is_pressed_ = is_pressed; return true; + case Key::RightOption: + case Key::LeftMeta: closed_apple_is_pressed_ = is_pressed; return true; - } - // If no ASCII value is supplied, look for a few special cases. - if(!value) { - switch(key) { - case Key::Left: value = 0x08; break; - case Key::Right: value = 0x15; break; - case Key::Down: value = 0x0a; break; - case Key::Up: value = 0x0b; break; - case Key::Backspace: value = 0x7f; break; - default: return false; - } - } + case Key::F1: case Key::F2: case Key::F3: case Key::F4: + case Key::F5: case Key::F6: case Key::F7: case Key::F8: + case Key::F9: case Key::F10: case Key::F11: case Key::F12: + case Key::PrintScreen: + case Key::ScrollLock: + case Key::Pause: + case Key::Insert: + case Key::Home: + case Key::PageUp: + case Key::PageDown: + case Key::End: + // Accept a bunch non-symbolic other keys, as + // reset, in the hope that the user can find + // at least one usable key. + m6502_.set_reset_line(is_pressed); + return true; - // Prior to the IIe, the keyboard could produce uppercase only. - if(!is_iie()) value = char(toupper(value)); + default: + if(!value) { + return false; + } + + // Prior to the IIe, the keyboard could produce uppercase only. + if(!is_iie()) value = char(toupper(value)); + break; + } if(is_pressed) { keyboard_input_ = uint8_t(value | 0x80); diff --git a/Machines/KeyboardMachine.hpp b/Machines/KeyboardMachine.hpp index 6ed5a8fc8..81c0918b1 100644 --- a/Machines/KeyboardMachine.hpp +++ b/Machines/KeyboardMachine.hpp @@ -32,6 +32,11 @@ struct KeyActions { Instructs that all keys should now be treated as released. */ virtual void clear_all_keys() {} + + /*! + Indicates whether a machine most naturally accepts logical rather than physical input. + */ + virtual bool prefers_logical_input() { return false; } }; /*! diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 01b08feff..6bf23cd99 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -5175,6 +5175,7 @@ "$(USER_LIBRARY_DIR)/Frameworks", ); GCC_C_LANGUAGE_STANDARD = gnu11; + MACOSX_DEPLOYMENT_TARGET = 10.13; PRODUCT_NAME = "$(TARGET_NAME)"; }; name = Debug; @@ -5197,6 +5198,8 @@ GCC_C_LANGUAGE_STANDARD = gnu11; GCC_OPTIMIZATION_LEVEL = 2; GCC_PREPROCESSOR_DEFINITIONS = NDEBUG; + MACOSX_DEPLOYMENT_TARGET = 10.13; + ONLY_ACTIVE_ARCH = YES; PRODUCT_NAME = "$(TARGET_NAME)"; }; name = Release; diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 9891a6c49..3f83509e5 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -31,7 +31,7 @@ + + + + diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index d95ba699a..497ef6004 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -783,8 +783,10 @@ int main(int argc, char *argv[]) { } } - // Check whether a 'logical' keyboard has been requested. - const bool logical_keyboard = arguments.selections.find("logical-keyboard") != arguments.selections.end(); + // Check whether a 'logical' keyboard has been requested, or the machine would prefer one anyway. + const bool logical_keyboard = + (arguments.selections.find("logical-keyboard") != arguments.selections.end()) || + (machine->keyboard_machine() && machine->keyboard_machine()->prefers_logical_input()); if(logical_keyboard) { SDL_StartTextInput(); } @@ -1162,13 +1164,8 @@ int main(int argc, char *argv[]) { } else { // This is a slightly terrible way of obtaining a symbol for the key, e.g. for letters it will always return // the capital letter version, at least empirically. But it'll have to do for now. - // - // TODO: ideally have a keyboard machine declare whether it wants either key events or text input? But that - // doesn't match machines like the IIe that, to some extent, expose both. So then eliding as attempted above, - // and keeping ephemeral track of which symbols have been tied to which keys for the benefit of future key up - // events is probably the way forward? const char *key_name = SDL_GetKeyName(keypress.keycode); - if(keyboard_machine->get_keyboard().set_key_pressed(key, key_name[0], keypress.is_down)) { + if(keyboard_machine->get_keyboard().set_key_pressed(key, (strlen(key_name) == 1) ? key_name[0] : 0, keypress.is_down)) { continue; } } diff --git a/Outputs/OpenGL/ScanTargetGLSLFragments.cpp b/Outputs/OpenGL/ScanTargetGLSLFragments.cpp index 7a55329a0..d83f6814a 100644 --- a/Outputs/OpenGL/ScanTargetGLSLFragments.cpp +++ b/Outputs/OpenGL/ScanTargetGLSLFragments.cpp @@ -194,8 +194,9 @@ std::vector ScanTarget::bindings(ShaderType type) const { std::string ScanTarget::sampling_function() const { std::string fragment_shader; const auto modals = BufferingScanTarget::modals(); + const bool is_svideo = modals.display_type == DisplayType::SVideo; - if(modals.display_type == DisplayType::SVideo) { + if(is_svideo) { fragment_shader += "vec2 svideo_sample(vec2 coordinate, float angle) {"; } else { @@ -203,7 +204,6 @@ std::string ScanTarget::sampling_function() const { "float composite_sample(vec2 coordinate, float angle) {"; } - const bool is_svideo = modals.display_type == DisplayType::SVideo; switch(modals.input_data_type) { case InputDataType::Luminance1: case InputDataType::Luminance8: