require "spec_helper"
require "tempfile"
describe Karabiner do
let!(:config) { Tempfile.new(".karabiner") }
let(:xml_dir) { "/tmp" }
let(:xml_path) { File.join(xml_dir, Karabiner::XML_FILE_NAME) }
let(:result) { File.read(xml_path) }
before do
stub_const("Karabiner::XML_DIR", xml_dir)
allow(Karabiner::CLI).to receive(:reload_xml)
# Silence stdout
allow_any_instance_of(Kernel).to receive(:puts)
end
after do
config.close!
end
def prepare_karabiner(karabiner)
config.write(karabiner)
config.rewind
end
def expect_result(expected_result)
karabiner = Karabiner.new(config.path)
karabiner.apply_configuration
expect(result).to eq(expected_result)
end
it "accepts blank config" do
prepare_karabiner("")
expect_result(<<-EOS.unindent)
EOS
end
it "accepts cmd combination" do
prepare_karabiner(<<-EOS)
item "Command+A to Command+B" do
remap "Cmd-A", to: "Cmd-B"
end
EOS
expect_result(<<-EOS.unindent)
-
Command+A to Command+B
remap.command_a_to_command_b
__KeyToKey__ KeyCode::A, VK_COMMAND, KeyCode::B, VK_COMMAND
EOS
end
it "accepts multiple remaps" do
prepare_karabiner(<<-EOS)
item "multiple remaps" do
remap "Cmd-A", to: "Cmd-B"
remap "Shift-A", to: "Shift-B"
end
EOS
expect_result(<<-EOS.unindent)
-
multiple remaps
remap.multiple_remaps
__KeyToKey__ KeyCode::A, VK_COMMAND, KeyCode::B, VK_COMMAND
__KeyToKey__ KeyCode::A, VK_SHIFT, KeyCode::B, VK_SHIFT
EOS
end
it "accepts multiple items" do
prepare_karabiner(<<-EOS)
item "first item" do
remap "Cmd-C-A", to: "Cmd-M-B"
end
item "second item" do
remap "Shift-Opt-A", to: "Shift-Cmd-B"
end
EOS
expect_result(<<-EOS.unindent)
-
first item
remap.first_item
__KeyToKey__ KeyCode::A, VK_COMMAND | VK_CONTROL, KeyCode::B, VK_COMMAND | VK_OPTION
-
second item
remap.second_item
__KeyToKey__ KeyCode::A, VK_SHIFT | VK_OPTION, KeyCode::B, VK_SHIFT | VK_COMMAND
EOS
end
it "accepts appdef and app option" do
prepare_karabiner(<<-EOS)
appdef "CHROME", equal: "com.google.Chrome"
item "Command+K to Command+L", only: "CHROME" do
remap "Cmd-K", to: "Cmd-L"
end
EOS
expect_result(<<-EOS.unindent)
CHROME
com.google.Chrome
-
Command+K to Command+L
remap.command_k_to_command_l
CHROME
__KeyToKey__ KeyCode::K, VK_COMMAND, KeyCode::L, VK_COMMAND
EOS
end
it "accepts config and show_message" do
prepare_karabiner(<<-EOS)
item "CapsLock ON", config_not: "notsave.private_capslock_on" do
remap "Cmd-L", to: ["capslock", "VK_CONFIG_FORCE_ON_notsave_private_capslock_on"]
end
item "CapsLock OFF", config_only: "notsave.private_capslock_on" do
remap "Cmd-L", to: ["capslock", "VK_CONFIG_FORCE_OFF_notsave_private_capslock_on"]
end
item "CapsLock Mode" do
identifier "notsave.private_capslock_on", vk_config: "true"
show_message "CapsLock"
end
EOS
expect_result(<<-EOS.unindent)
-
CapsLock ON
remap.capslock_on
notsave.private_capslock_on
__KeyToKey__ KeyCode::L, VK_COMMAND, KeyCode::CAPSLOCK, KeyCode::VK_CONFIG_FORCE_ON_notsave_private_capslock_on
-
CapsLock OFF
remap.capslock_off
notsave.private_capslock_on
__KeyToKey__ KeyCode::L, VK_COMMAND, KeyCode::CAPSLOCK, KeyCode::VK_CONFIG_FORCE_OFF_notsave_private_capslock_on
-
CapsLock Mode
notsave.private_capslock_on
__ShowStatusMessage__ CapsLock
EOS
end
it "accepts implicit autogen selection" do
prepare_karabiner(<<-EOS)
item "Control+LeftClick to Command+LeftClick" do
autogen "__PointingButtonToPointingButton__ PointingButton::LEFT, MODIFIERFLAG_EITHER_LEFT_OR_RIGHT_CONTROL, PointingButton::LEFT, MODIFIERFLAG_EITHER_LEFT_OR_RIGHT_COMMAND"
end
EOS
expect_result(<<-EOS.unindent)
-
Control+LeftClick to Command+LeftClick
remap.control_leftclick_to_command_leftclick
__PointingButtonToPointingButton__ PointingButton::LEFT, MODIFIERFLAG_EITHER_LEFT_OR_RIGHT_CONTROL, PointingButton::LEFT, MODIFIERFLAG_EITHER_LEFT_OR_RIGHT_COMMAND
EOS
end
it "application invoking" do
prepare_karabiner(<<-EOS)
item "Application shortcuts" do
remap "C-o", to: invoke("YoruFukurou")
remap "C-u", to: invoke("Google Chrome")
remap "C-h", to: invoke("iTerm")
end
item "duplicate app" do
remap "C-a", to: invoke("YoruFukurou")
end
EOS
expect_result(<<-EOS.unindent)
-
Application shortcuts
remap.application_shortcuts
__KeyToKey__ KeyCode::O, VK_CONTROL, KeyCode::VK_OPEN_URL_APP_YoruFukurou
__KeyToKey__ KeyCode::U, VK_CONTROL, KeyCode::VK_OPEN_URL_APP_Google_Chrome
__KeyToKey__ KeyCode::H, VK_CONTROL, KeyCode::VK_OPEN_URL_APP_iTerm
-
duplicate app
remap.duplicate_app
__KeyToKey__ KeyCode::A, VK_CONTROL, KeyCode::VK_OPEN_URL_APP_YoruFukurou
KeyCode::VK_OPEN_URL_APP_YoruFukurou
/Applications/YoruFukurou.app
KeyCode::VK_OPEN_URL_APP_Google_Chrome
/Applications/Google Chrome.app
KeyCode::VK_OPEN_URL_APP_iTerm
/Applications/iTerm.app
EOS
end
it "accepts group items" do
prepare_karabiner(<<-EOS)
group "Option" do
item "First" do
identifier "option.option_first"
end
item "Second" do
identifier "option.option_second"
end
end
EOS
expect_result(<<-EOS.unindent)
-
Option
-
First
option.option_first
-
Second
option.option_second
EOS
end
context "when items are surrounded by config" do
it "accepts cmd combination" do
prepare_karabiner(<<-EOS)
config "Default" do
item "Command+A to Command+B" do
remap "Cmd-A", to: "Cmd-B"
end
end
EOS
expect_result(<<-EOS.unindent)
-
Command+A to Command+B
remap.command_a_to_command_b
__KeyToKey__ KeyCode::A, VK_COMMAND, KeyCode::B, VK_COMMAND
EOS
end
it "accepts group items" do
prepare_karabiner(<<-EOS)
config "Original" do
group "Option" do
item "First" do
identifier "option.option_first"
end
item "Second" do
identifier "option.option_second"
end
end
end
EOS
expect_result(<<-EOS.unindent)
-
Option
-
First
option.option_first
-
Second
option.option_second
EOS
end
end
end