aboutsummaryrefslogtreecommitdiff
path: root/Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp
diff options
context:
space:
mode:
authorJean-Francois Mauguit <jfmauguit@mac.com>2024-09-24 09:03:25 -0400
committerGitHub <noreply@github.com>2024-09-24 09:03:25 -0400
commitbab614c421ed7ae329d26bf028c4a3b1d2450f5a (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp
parent4bde6044fddf053f31795b9eaccdd2a5a527d21f (diff)
parent20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (diff)
downloadwinamp-bab614c421ed7ae329d26bf028c4a3b1d2450f5a.tar.gz
Merge pull request #5 from WinampDesktop/community
Merge to main
Diffstat (limited to 'Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp')
-rw-r--r--Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp2027
1 files changed, 2027 insertions, 0 deletions
diff --git a/Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp b/Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp
new file mode 100644
index 00000000..8fe4f98f
--- /dev/null
+++ b/Src/external_dependencies/openmpt-trunk/mptrack/CommandSet.cpp
@@ -0,0 +1,2027 @@
+/*
+ * CommandSet.cpp
+ * --------------
+ * Purpose: Implementation of custom key handling.
+ * Notes : (currently none)
+ * Authors: OpenMPT Devs
+ * The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
+ */
+
+
+#include "stdafx.h"
+#include "CommandSet.h"
+#include "resource.h"
+#include "Mptrack.h" // For ErrorBox
+#include "../soundlib/mod_specifications.h"
+#include "../soundlib/Tables.h"
+#include "../mptrack/Reporting.h"
+#include "../common/mptFileIO.h"
+#include <sstream>
+#include "TrackerSettings.h"
+
+
+OPENMPT_NAMESPACE_BEGIN
+
+namespace
+{
+// Version of the .mkb format
+constexpr int KEYMAP_VERSION = 1;
+
+constexpr std::tuple<InputTargetContext, CommandID, CommandID> NoteContexts[] =
+{
+ {kCtxViewPatternsNote, kcVPStartNotes, kcVPStartNoteStops},
+ {kCtxViewSamples, kcSampStartNotes, kcSampStartNoteStops},
+ {kCtxViewInstruments, kcInstrumentStartNotes, kcInstrumentStartNoteStops},
+ {kCtxViewTree, kcTreeViewStartNotes, kcTreeViewStartNoteStops},
+ {kCtxInsNoteMap, kcInsNoteMapStartNotes, kcInsNoteMapStartNoteStops},
+ {kCtxVSTGUI, kcVSTGUIStartNotes, kcVSTGUIStartNoteStops},
+ {kCtxViewComments, kcCommentsStartNotes, kcCommentsStartNoteStops},
+};
+
+}; // namespace
+
+#ifdef MPT_ALL_LOGGING
+#define MPT_COMMANDSET_LOGGING
+#endif
+
+#ifdef MPT_COMMANDSET_LOGGING
+#define LOG_COMMANDSET(x) MPT_LOG_GLOBAL(LogDebug, "CommandSet", x)
+#else
+#define LOG_COMMANDSET(x) do { } while(0)
+#endif
+
+
+CCommandSet::CCommandSet()
+{
+ // Which key binding rules to enforce?
+ m_enforceRule[krPreventDuplicate] = true;
+ m_enforceRule[krDeleteOldOnConflict] = true;
+ m_enforceRule[krAllowNavigationWithSelection] = true;
+ m_enforceRule[krAllowSelectionWithNavigation] = true;
+ m_enforceRule[krAllowSelectCopySelectCombos] = true;
+ m_enforceRule[krLockNotesToChords] = true;
+ m_enforceRule[krNoteOffOnKeyRelease] = true;
+ m_enforceRule[krPropagateNotes] = true;
+ m_enforceRule[krReassignDigitsToOctaves] = false;
+ m_enforceRule[krAutoSelectOff] = true;
+ m_enforceRule[krAutoSpacing] = true;
+ m_enforceRule[krCheckModifiers] = true;
+ m_enforceRule[krPropagateSampleManipulation] = true;
+// enforceRule[krCheckContextHierarchy] = true;
+
+ SetupCommands();
+ SetupContextHierarchy();
+}
+
+
+// Setup
+
+KeyCommand::KeyCommand(uint32 uid, const TCHAR *message, std::vector<KeyCombination> keys)
+ : kcList{std::move(keys)}
+ , Message{message}
+ , UID{uid}
+{
+}
+
+static constexpr struct
+{
+ uint32 uid = 0; // ID | Hidden | Dummy
+ CommandID cmd = kcNull;
+ const TCHAR *description = nullptr;
+} CommandDefinitions[] =
+{
+ {1001, kcPatternRecord, _T("Enable Recording")},
+ {1002, kcPatternPlayRow, _T("Play Row")},
+ {1003, kcCursorCopy, _T("Quick Copy")},
+ {1004, kcCursorPaste, _T("Quick Paste")},
+ {1005, kcChannelMute, _T("Mute Current Channel")},
+ {1006, kcChannelSolo, _T("Solo Current Channel")},
+ {1007, kcTransposeUp, _T("Transpose +1")},
+ {1008, kcTransposeDown, _T("Transpose -1")},
+ {1009, kcTransposeOctUp, _T("Transpose +1 Octave")},
+ {1010, kcTransposeOctDown, _T("Transpose -1 Octave")},
+ {1011, kcSelectChannel, _T("Select Channel / Select All")},
+ {1012, kcPatternAmplify, _T("Amplify selection")},
+ {1013, kcPatternSetInstrument, _T("Apply current instrument")},
+ {1014, kcPatternInterpolateVol, _T("Interpolate Volume")},
+ {1015, kcPatternInterpolateEffect, _T("Interpolate Effect")},
+ {1016, kcPatternVisualizeEffect, _T("Open Effect Visualizer")},
+ {1017, kcPatternJumpDownh1, _T("Jump down by measure")},
+ {1018, kcPatternJumpUph1, _T("Jump up by measure")},
+ {1019, kcPatternSnapDownh1, _T("Snap down to measure")},
+ {1020, kcPatternSnapUph1, _T("Snap up to measure")},
+ {1021, kcViewGeneral, _T("View General")},
+ {1022, kcViewPattern, _T("View Pattern")},
+ {1023, kcViewSamples, _T("View Samples")},
+ {1024, kcViewInstruments, _T("View Instruments")},
+ {1025, kcViewComments, _T("View Comments")},
+ {1026, kcPlayPatternFromCursor, _T("Play Pattern from Cursor")},
+ {1027, kcPlayPatternFromStart, _T("Play Pattern from Start")},
+ {1028, kcPlaySongFromCursor, _T("Play Song from Cursor")},
+ {1029, kcPlaySongFromStart, _T("Play Song from Start")},
+ {1030, kcPlayPauseSong, _T("Play Song / Pause Song")},
+ {1031, kcPauseSong, _T("Pause Song")},
+ {1032, kcPrevInstrument, _T("Previous Instrument")},
+ {1033, kcNextInstrument, _T("Next Instrument")},
+ {1034, kcPrevOrder, _T("Previous Order")},
+ {1035, kcNextOrder, _T("Next Order")},
+ {1036, kcPrevOctave, _T("Previous Octave")},
+ {1037, kcNextOctave, _T("Next Octave")},
+ {1038, kcNavigateDown, _T("Navigate down by 1 row")},
+ {1039, kcNavigateUp, _T("Navigate up by 1 row")},
+ {1040, kcNavigateLeft, _T("Navigate left")},
+ {1041, kcNavigateRight, _T("Navigate right")},
+ {1042, kcNavigateNextChan, _T("Navigate to next channel")},
+ {1043, kcNavigatePrevChan, _T("Navigate to previous channel")},
+ {1044, kcHomeHorizontal, _T("Go to first channel")},
+ {1045, kcHomeVertical, _T("Go to first row")},
+ {1046, kcHomeAbsolute, _T("Go to first row of first channel")},
+ {1047, kcEndHorizontal, _T("Go to last channel")},
+ {1048, kcEndVertical, _T("Go to last row")},
+ {1049, kcEndAbsolute, _T("Go to last row of last channel")},
+ {1050, kcSelect, _T("Selection key")},
+ {1051, kcCopySelect, _T("Copy select key")},
+ {KeyCommand::Hidden, kcSelectOff, _T("Deselect")},
+ {KeyCommand::Hidden, kcCopySelectOff, _T("Copy deselect key")},
+ {1054, kcNextPattern, _T("Next Pattern")},
+ {1055, kcPrevPattern, _T("Previous Pattern")},
+ //{1056, kcClearSelection, _T("Wipe selection")},
+ {1057, kcClearRow, _T("Clear row")},
+ {1058, kcClearField, _T("Clear field")},
+ {1059, kcClearRowStep, _T("Clear row and step")},
+ {1060, kcClearFieldStep, _T("Clear field and step")},
+ {1061, kcDeleteRow, _T("Delete Row(s)")},
+ {1062, kcShowNoteProperties, _T("Show Note Properties")},
+ {1063, kcShowEditMenu, _T("Show Context (Right-Click) Menu")},
+ {1064, kcVPNoteC_0, _T("Base octave C")},
+ {1065, kcVPNoteCS0, _T("Base octave C#")},
+ {1066, kcVPNoteD_0, _T("Base octave D")},
+ {1067, kcVPNoteDS0, _T("Base octave D#")},
+ {1068, kcVPNoteE_0, _T("Base octave E")},
+ {1069, kcVPNoteF_0, _T("Base octave F")},
+ {1070, kcVPNoteFS0, _T("Base octave F#")},
+ {1071, kcVPNoteG_0, _T("Base octave G")},
+ {1072, kcVPNoteGS0, _T("Base octave G#")},
+ {1073, kcVPNoteA_1, _T("Base octave A")},
+ {1074, kcVPNoteAS1, _T("Base octave A#")},
+ {1075, kcVPNoteB_1, _T("Base octave B")},
+ {1076, kcVPNoteC_1, _T("Base octave +1 C")},
+ {1077, kcVPNoteCS1, _T("Base octave +1 C#")},
+ {1078, kcVPNoteD_1, _T("Base octave +1 D")},
+ {1079, kcVPNoteDS1, _T("Base octave +1 D#")},
+ {1080, kcVPNoteE_1, _T("Base octave +1 E")},
+ {1081, kcVPNoteF_1, _T("Base octave +1 F")},
+ {1082, kcVPNoteFS1, _T("Base octave +1 F#")},
+ {1083, kcVPNoteG_1, _T("Base octave +1 G")},
+ {1084, kcVPNoteGS1, _T("Base octave +1 G#")},
+ {1085, kcVPNoteA_2, _T("Base octave +1 A")},
+ {1086, kcVPNoteAS2, _T("Base octave +1 A#")},
+ {1087, kcVPNoteB_2, _T("Base octave +1 B")},
+ {1088, kcVPNoteC_2, _T("Base octave +2 C")},
+ {1089, kcVPNoteCS2, _T("Base octave +2 C#")},
+ {1090, kcVPNoteD_2, _T("Base octave +2 D")},
+ {1091, kcVPNoteDS2, _T("Base octave +2 D#")},
+ {1092, kcVPNoteE_2, _T("Base octave +2 E")},
+ {1093, kcVPNoteF_2, _T("Base octave +2 F")},
+ {1094, kcVPNoteFS2, _T("Base octave +2 F#")},
+ {1095, kcVPNoteG_2, _T("Base octave +2 G")},
+ {1096, kcVPNoteGS2, _T("Base octave +2 G#")},
+ {1097, kcVPNoteA_3, _T("Base octave +2 A")},
+ {KeyCommand::Hidden, kcVPNoteStopC_0, _T("Stop base octave C")},
+ {KeyCommand::Hidden, kcVPNoteStopCS0, _T("Stop base octave C#")},
+ {KeyCommand::Hidden, kcVPNoteStopD_0, _T("Stop base octave D")},
+ {KeyCommand::Hidden, kcVPNoteStopDS0, _T("Stop base octave D#")},
+ {KeyCommand::Hidden, kcVPNoteStopE_0, _T("Stop base octave E")},
+ {KeyCommand::Hidden, kcVPNoteStopF_0, _T("Stop base octave F")},
+ {KeyCommand::Hidden, kcVPNoteStopFS0, _T("Stop base octave F#")},
+ {KeyCommand::Hidden, kcVPNoteStopG_0, _T("Stop base octave G")},
+ {KeyCommand::Hidden, kcVPNoteStopGS0, _T("Stop base octave G#")},
+ {KeyCommand::Hidden, kcVPNoteStopA_1, _T("Stop base octave +1 A")},
+ {KeyCommand::Hidden, kcVPNoteStopAS1, _T("Stop base octave +1 A#")},
+ {KeyCommand::Hidden, kcVPNoteStopB_1, _T("Stop base octave +1 B")},
+ {KeyCommand::Hidden, kcVPNoteStopC_1, _T("Stop base octave +1 C")},
+ {KeyCommand::Hidden, kcVPNoteStopCS1, _T("Stop base octave +1 C#")},
+ {KeyCommand::Hidden, kcVPNoteStopD_1, _T("Stop base octave +1 D")},
+ {KeyCommand::Hidden, kcVPNoteStopDS1, _T("Stop base octave +1 D#")},
+ {KeyCommand::Hidden, kcVPNoteStopE_1, _T("Stop base octave +1 E")},
+ {KeyCommand::Hidden, kcVPNoteStopF_1, _T("Stop base octave +1 F")},
+ {KeyCommand::Hidden, kcVPNoteStopFS1, _T("Stop base octave +1 F#")},
+ {KeyCommand::Hidden, kcVPNoteStopG_1, _T("Stop base octave +1 G")},
+ {KeyCommand::Hidden, kcVPNoteStopGS1, _T("Stop base octave +1 G#")},
+ {KeyCommand::Hidden, kcVPNoteStopA_2, _T("Stop base octave +2 A")},
+ {KeyCommand::Hidden, kcVPNoteStopAS2, _T("Stop base octave +2 A#")},
+ {KeyCommand::Hidden, kcVPNoteStopB_2, _T("Stop base octave +2 B")},
+ {KeyCommand::Hidden, kcVPNoteStopC_2, _T("Stop base octave +2 C")},
+ {KeyCommand::Hidden, kcVPNoteStopCS2, _T("Stop base octave +2 C#")},
+ {KeyCommand::Hidden, kcVPNoteStopD_2, _T("Stop base octave +2 D")},
+ {KeyCommand::Hidden, kcVPNoteStopDS2, _T("Stop base octave +2 D#")},
+ {KeyCommand::Hidden, kcVPNoteStopE_2, _T("Stop base octave +2 E")},
+ {KeyCommand::Hidden, kcVPNoteStopF_2, _T("Stop base octave +2 F")},
+ {KeyCommand::Hidden, kcVPNoteStopFS2, _T("Stop base octave +2 F#")},
+ {KeyCommand::Hidden, kcVPNoteStopG_2, _T("Stop base octave +2 G")},
+ {KeyCommand::Hidden, kcVPNoteStopGS2, _T("Stop base octave +2 G#")},
+ {KeyCommand::Hidden, kcVPNoteStopA_3, _T("Stop base octave +3 A")},
+ {KeyCommand::Hidden, kcVPChordC_0, _T("Base octave chord C")},
+ {KeyCommand::Hidden, kcVPChordCS0, _T("Base octave chord C#")},
+ {KeyCommand::Hidden, kcVPChordD_0, _T("Base octave chord D")},
+ {KeyCommand::Hidden, kcVPChordDS0, _T("Base octave chord D#")},
+ {KeyCommand::Hidden, kcVPChordE_0, _T("Base octave chord E")},
+ {KeyCommand::Hidden, kcVPChordF_0, _T("Base octave chord F")},
+ {KeyCommand::Hidden, kcVPChordFS0, _T("Base octave chord F#")},
+ {KeyCommand::Hidden, kcVPChordG_0, _T("Base octave chord G")},
+ {KeyCommand::Hidden, kcVPChordGS0, _T("Base octave chord G#")},
+ {KeyCommand::Hidden, kcVPChordA_1, _T("Base octave +1 chord A")},
+ {KeyCommand::Hidden, kcVPChordAS1, _T("Base octave +1 chord A#")},
+ {KeyCommand::Hidden, kcVPChordB_1, _T("Base octave +1 chord B")},
+ {KeyCommand::Hidden, kcVPChordC_1, _T("Base octave +1 chord C")},
+ {KeyCommand::Hidden, kcVPChordCS1, _T("Base octave +1 chord C#")},
+ {KeyCommand::Hidden, kcVPChordD_1, _T("Base octave +1 chord D")},
+ {KeyCommand::Hidden, kcVPChordDS1, _T("Base octave +1 chord D#")},
+ {KeyCommand::Hidden, kcVPChordE_1, _T("Base octave +1 chord E")},
+ {KeyCommand::Hidden, kcVPChordF_1, _T("Base octave +1 chord F")},
+ {KeyCommand::Hidden, kcVPChordFS1, _T("Base octave +1 chord F#")},
+ {KeyCommand::Hidden, kcVPChordG_1, _T("Base octave +1 chord G")},
+ {KeyCommand::Hidden, kcVPChordGS1, _T("Base octave +1 chord G#")},
+ {KeyCommand::Hidden, kcVPChordA_2, _T("Base octave +2 chord A")},
+ {KeyCommand::Hidden, kcVPChordAS2, _T("Base octave +2 chord A#")},
+ {KeyCommand::Hidden, kcVPChordB_2, _T("Base octave +2 chord B")},
+ {KeyCommand::Hidden, kcVPChordC_2, _T("Base octave +2 chord C")},
+ {KeyCommand::Hidden, kcVPChordCS2, _T("Base octave +2 chord C#")},
+ {KeyCommand::Hidden, kcVPChordD_2, _T("Base octave +2 chord D")},
+ {KeyCommand::Hidden, kcVPChordDS2, _T("Base octave +2 chord D#")},
+ {KeyCommand::Hidden, kcVPChordE_2, _T("Base octave +2 chord E")},
+ {KeyCommand::Hidden, kcVPChordF_2, _T("Base octave +2 chord F")},
+ {KeyCommand::Hidden, kcVPChordFS2, _T("Base octave +2 chord F#")},
+ {KeyCommand::Hidden, kcVPChordG_2, _T("Base octave +2 chord G")},
+ {KeyCommand::Hidden, kcVPChordGS2, _T("Base octave +2 chord G#")},
+ {KeyCommand::Hidden, kcVPChordA_3, _T("Base octave chord +3 A")},
+ {KeyCommand::Hidden, kcVPChordStopC_0, _T("Stop base octave chord C")},
+ {KeyCommand::Hidden, kcVPChordStopCS0, _T("Stop base octave chord C#")},
+ {KeyCommand::Hidden, kcVPChordStopD_0, _T("Stop base octave chord D")},
+ {KeyCommand::Hidden, kcVPChordStopDS0, _T("Stop base octave chord D#")},
+ {KeyCommand::Hidden, kcVPChordStopE_0, _T("Stop base octave chord E")},
+ {KeyCommand::Hidden, kcVPChordStopF_0, _T("Stop base octave chord F")},
+ {KeyCommand::Hidden, kcVPChordStopFS0, _T("Stop base octave chord F#")},
+ {KeyCommand::Hidden, kcVPChordStopG_0, _T("Stop base octave chord G")},
+ {KeyCommand::Hidden, kcVPChordStopGS0, _T("Stop base octave chord G#")},
+ {KeyCommand::Hidden, kcVPChordStopA_1, _T("Stop base octave +1 chord A")},
+ {KeyCommand::Hidden, kcVPChordStopAS1, _T("Stop base octave +1 chord A#")},
+ {KeyCommand::Hidden, kcVPChordStopB_1, _T("Stop base octave +1 chord B")},
+ {KeyCommand::Hidden, kcVPChordStopC_1, _T("Stop base octave +1 chord C")},
+ {KeyCommand::Hidden, kcVPChordStopCS1, _T("Stop base octave +1 chord C#")},
+ {KeyCommand::Hidden, kcVPChordStopD_1, _T("Stop base octave +1 chord D")},
+ {KeyCommand::Hidden, kcVPChordStopDS1, _T("Stop base octave +1 chord D#")},
+ {KeyCommand::Hidden, kcVPChordStopE_1, _T("Stop base octave +1 chord E")},
+ {KeyCommand::Hidden, kcVPChordStopF_1, _T("Stop base octave +1 chord F")},
+ {KeyCommand::Hidden, kcVPChordStopFS1, _T("Stop base octave +1 chord F#")},
+ {KeyCommand::Hidden, kcVPChordStopG_1, _T("Stop base octave +1 chord G")},
+ {KeyCommand::Hidden, kcVPChordStopGS1, _T("Stop base octave +1 chord G#")},
+ {KeyCommand::Hidden, kcVPChordStopA_2, _T("Stop base octave +2 chord A")},
+ {KeyCommand::Hidden, kcVPChordStopAS2, _T("Stop base octave +2 chord A#")},
+ {KeyCommand::Hidden, kcVPChordStopB_2, _T("Stop base octave +2 chord B")},
+ {KeyCommand::Hidden, kcVPChordStopC_2, _T("Stop base octave +2 chord C")},
+ {KeyCommand::Hidden, kcVPChordStopCS2, _T("Stop base octave +2 chord C#")},
+ {KeyCommand::Hidden, kcVPChordStopD_2, _T("Stop base octave +2 chord D")},
+ {KeyCommand::Hidden, kcVPChordStopDS2, _T("Stop base octave +2 chord D#")},
+ {KeyCommand::Hidden, kcVPChordStopE_2, _T("Stop base octave +2 chord E")},
+ {KeyCommand::Hidden, kcVPChordStopF_2, _T("Stop base octave +2 chord F")},
+ {KeyCommand::Hidden, kcVPChordStopFS2, _T("Stop base octave +2 chord F#")},
+ {KeyCommand::Hidden, kcVPChordStopG_2, _T("Stop base octave +2 chord G")},
+ {KeyCommand::Hidden, kcVPChordStopGS2, _T("Stop base octave +2 chord G#")},
+ {KeyCommand::Hidden, kcVPChordStopA_3, _T("Stop base octave +3 chord A")},
+ {1200, kcNoteCut, _T("Note Cut")},
+ {1201, kcNoteOff, _T("Note Off")},
+ {1202, kcSetIns0, _T("Set instrument digit 0")},
+ {1203, kcSetIns1, _T("Set instrument digit 1")},
+ {1204, kcSetIns2, _T("Set instrument digit 2")},
+ {1205, kcSetIns3, _T("Set instrument digit 3")},
+ {1206, kcSetIns4, _T("Set instrument digit 4")},
+ {1207, kcSetIns5, _T("Set instrument digit 5")},
+ {1208, kcSetIns6, _T("Set instrument digit 6")},
+ {1209, kcSetIns7, _T("Set instrument digit 7")},
+ {1210, kcSetIns8, _T("Set instrument digit 8")},
+ {1211, kcSetIns9, _T("Set instrument digit 9")},
+ {1212, kcSetOctave0, _T("Set octave 0")},
+ {1213, kcSetOctave1, _T("Set octave 1")},
+ {1214, kcSetOctave2, _T("Set octave 2")},
+ {1215, kcSetOctave3, _T("Set octave 3")},
+ {1216, kcSetOctave4, _T("Set octave 4")},
+ {1217, kcSetOctave5, _T("Set octave 5")},
+ {1218, kcSetOctave6, _T("Set octave 6")},
+ {1219, kcSetOctave7, _T("Set octave 7")},
+ {1220, kcSetOctave8, _T("Set octave 8")},
+ {1221, kcSetOctave9, _T("Set octave 9")},
+ {1222, kcSetVolume0, _T("Set volume digit 0")},
+ {1223, kcSetVolume1, _T("Set volume digit 1")},
+ {1224, kcSetVolume2, _T("Set volume digit 2")},
+ {1225, kcSetVolume3, _T("Set volume digit 3")},
+ {1226, kcSetVolume4, _T("Set volume digit 4")},
+ {1227, kcSetVolume5, _T("Set volume digit 5")},
+ {1228, kcSetVolume6, _T("Set volume digit 6")},
+ {1229, kcSetVolume7, _T("Set volume digit 7")},
+ {1230, kcSetVolume8, _T("Set volume digit 8")},
+ {1231, kcSetVolume9, _T("Set volume digit 9")},
+ {1232, kcSetVolumeVol, _T("Volume Command - Volume")},
+ {1233, kcSetVolumePan, _T("Volume Command - Panning")},
+ {1234, kcSetVolumeVolSlideUp, _T("Volume Command - Volume Slide Up")},
+ {1235, kcSetVolumeVolSlideDown, _T("Volume Command - Volume Slide Down")},
+ {1236, kcSetVolumeFineVolUp, _T("Volume Command - Fine Volume Slide Up")},
+ {1237, kcSetVolumeFineVolDown, _T("Volume Command - Fine Volume Slide Down")},
+ {1238, kcSetVolumeVibratoSpd, _T("Volume Command - Vibrato Speed")},
+ {1239, kcSetVolumeVibrato, _T("Volume Command - Vibrato Depth")},
+ {1240, kcSetVolumeXMPanLeft, _T("Volume Command - XM Pan Slide Left")},
+ {1241, kcSetVolumeXMPanRight, _T("Volume Command - XM Pan Slide Right")},
+ {1242, kcSetVolumePortamento, _T("Volume Command - Tone Portamento")},
+ {1243, kcSetVolumeITPortaUp, _T("Volume Command - Portamento Up")},
+ {1244, kcSetVolumeITPortaDown, _T("Volume Command - Portamento Down")},
+ {KeyCommand::Hidden, kcSetVolumeITUnused, _T("Volume Command - Unused")},
+ {1246, kcSetVolumeITOffset, _T("Volume Command - Offset")},
+ {1247, kcSetFXParam0, _T("Effect Parameter Digit 0")},
+ {1248, kcSetFXParam1, _T("Effect Parameter Digit 1")},
+ {1249, kcSetFXParam2, _T("Effect Parameter Digit 2")},
+ {1250, kcSetFXParam3, _T("Effect Parameter Digit 3")},
+ {1251, kcSetFXParam4, _T("Effect Parameter Digit 4")},
+ {1252, kcSetFXParam5, _T("Effect Parameter Digit 5")},
+ {1253, kcSetFXParam6, _T("Effect Parameter Digit 6")},
+ {1254, kcSetFXParam7, _T("Effect Parameter Digit 7")},
+ {1255, kcSetFXParam8, _T("Effect Parameter Digit 8")},
+ {1256, kcSetFXParam9, _T("Effect Parameter Digit 9")},
+ {1257, kcSetFXParamA, _T("Effect Parameter Digit A")},
+ {1258, kcSetFXParamB, _T("Effect Parameter Digit B")},
+ {1259, kcSetFXParamC, _T("Effect Parameter Digit C")},
+ {1260, kcSetFXParamD, _T("Effect Parameter Digit D")},
+ {1261, kcSetFXParamE, _T("Effect Parameter Digit E")},
+ {1262, kcSetFXParamF, _T("Effect Parameter Digit F")},
+ {KeyCommand::Hidden, kcSetFXarp, _T("FX Arpeggio")},
+ {KeyCommand::Hidden, kcSetFXportUp, _T("FX Portamento Up")},
+ {KeyCommand::Hidden, kcSetFXportDown, _T("FX Portamento Down")},
+ {KeyCommand::Hidden, kcSetFXport, _T("FX Tone Portamento")},
+ {KeyCommand::Hidden, kcSetFXvibrato, _T("FX Vibrato")},
+ {KeyCommand::Hidden, kcSetFXportSlide, _T("FX Portamento + Volume Slide")},
+ {KeyCommand::Hidden, kcSetFXvibSlide, _T("FX Vibrato + Volume Slide")},
+ {KeyCommand::Hidden, kcSetFXtremolo, _T("FX Tremolo")},
+ {KeyCommand::Hidden, kcSetFXpan, _T("FX Pan")},
+ {KeyCommand::Hidden, kcSetFXoffset, _T("FX Offset")},
+ {KeyCommand::Hidden, kcSetFXvolSlide, _T("FX Volume Slide")},
+ {KeyCommand::Hidden, kcSetFXgotoOrd, _T("FX Pattern Jump")},
+ {KeyCommand::Hidden, kcSetFXsetVol, _T("FX Set Volume")},
+ {KeyCommand::Hidden, kcSetFXgotoRow, _T("FX Break To Row")},
+ {KeyCommand::Hidden, kcSetFXretrig, _T("FX Retrigger")},
+ {KeyCommand::Hidden, kcSetFXspeed, _T("FX Set Speed")},
+ {KeyCommand::Hidden, kcSetFXtempo, _T("FX Set Tempo")},
+ {KeyCommand::Hidden, kcSetFXtremor, _T("FX Tremor")},
+ {KeyCommand::Hidden, kcSetFXextendedMOD, _T("FX Extended MOD Commands")},
+ {KeyCommand::Hidden, kcSetFXextendedS3M, _T("FX Extended S3M Commands")},
+ {KeyCommand::Hidden, kcSetFXchannelVol, _T("FX Set Channel Volume")},
+ {KeyCommand::Hidden, kcSetFXchannelVols, _T("FX Channel Volume Slide")},
+ {KeyCommand::Hidden, kcSetFXglobalVol, _T("FX Set Global Volume")},
+ {KeyCommand::Hidden, kcSetFXglobalVols, _T("FX Global Volume Slide")},
+ {KeyCommand::Hidden, kcSetFXkeyoff, _T("FX Key Off (XM)")},
+ {KeyCommand::Hidden, kcSetFXfineVib, _T("FX Fine Vibrato")},
+ {KeyCommand::Hidden, kcSetFXpanbrello, _T("FX Panbrello")},
+ {KeyCommand::Hidden, kcSetFXextendedXM, _T("FX Extended XM Commands")},
+ {KeyCommand::Hidden, kcSetFXpanSlide, _T("FX Pan Slide")},
+ {KeyCommand::Hidden, kcSetFXsetEnvPos, _T("FX Set Envelope Position (XM)")},
+ {KeyCommand::Hidden, kcSetFXmacro, _T("FX MIDI Macro")},
+ {1294, kcSetFXmacroSlide, _T("Smooth MIDI Macro Slide")},
+ {1295, kcSetFXdelaycut, _T("Combined Note Delay and Note Cut")},
+ {KeyCommand::Hidden, kcPatternJumpDownh1Select, _T("kcPatternJumpDownh1Select")},
+ {KeyCommand::Hidden, kcPatternJumpUph1Select, _T("kcPatternJumpUph1Select")},
+ {KeyCommand::Hidden, kcPatternSnapDownh1Select, _T("kcPatternSnapDownh1Select")},
+ {KeyCommand::Hidden, kcPatternSnapUph1Select, _T("kcPatternSnapUph1Select")},
+ {KeyCommand::Hidden, kcNavigateDownSelect, _T("kcNavigateDownSelect")},
+ {KeyCommand::Hidden, kcNavigateUpSelect, _T("kcNavigateUpSelect")},
+ {KeyCommand::Hidden, kcNavigateLeftSelect, _T("kcNavigateLeftSelect")},
+ {KeyCommand::Hidden, kcNavigateRightSelect, _T("kcNavigateRightSelect")},
+ {KeyCommand::Hidden, kcNavigateNextChanSelect, _T("kcNavigateNextChanSelect")},
+ {KeyCommand::Hidden, kcNavigatePrevChanSelect, _T("kcNavigatePrevChanSelect")},
+ {KeyCommand::Hidden, kcHomeHorizontalSelect, _T("kcHomeHorizontalSelect")},
+ {KeyCommand::Hidden, kcHomeVerticalSelect, _T("kcHomeVerticalSelect")},
+ {KeyCommand::Hidden, kcHomeAbsoluteSelect, _T("kcHomeAbsoluteSelect")},
+ {KeyCommand::Hidden, kcEndHorizontalSelect, _T("kcEndHorizontalSelect")},
+ {KeyCommand::Hidden, kcEndVerticalSelect, _T("kcEndVerticalSelect")},
+ {KeyCommand::Hidden, kcEndAbsoluteSelect, _T("kcEndAbsoluteSelect")},
+ {KeyCommand::Hidden, kcSelectWithNav, _T("kcSelectWithNav")},
+ {KeyCommand::Hidden, kcSelectOffWithNav, _T("kcSelectOffWithNav")},
+ {KeyCommand::Hidden, kcCopySelectWithNav, _T("kcCopySelectWithNav")},
+ {KeyCommand::Hidden, kcCopySelectOffWithNav, _T("kcCopySelectOffWithNav")},
+ {1316 | KeyCommand::Dummy, kcChordModifier, _T("Chord Modifier")},
+ {1317 | KeyCommand::Dummy, kcSetSpacing, _T("Set edit step on note entry")},
+ {KeyCommand::Hidden, kcSetSpacing0, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing1, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing2, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing3, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing4, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing5, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing6, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing7, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing8, _T("")},
+ {KeyCommand::Hidden, kcSetSpacing9, _T("")},
+ {KeyCommand::Hidden, kcCopySelectWithSelect, _T("kcCopySelectWithSelect")},
+ {KeyCommand::Hidden, kcCopySelectOffWithSelect, _T("kcCopySelectOffWithSelect")},
+ {KeyCommand::Hidden, kcSelectWithCopySelect, _T("kcSelectWithCopySelect")},
+ {KeyCommand::Hidden, kcSelectOffWithCopySelect, _T("kcSelectOffWithCopySelect")},
+ /*
+ {1332, kcCopy, _T("Copy pattern data")},
+ {1333, kcCut, _T("Cut pattern data")},
+ {1334, kcPaste, _T("Paste pattern data")},
+ {1335, kcMixPaste, _T("Mix-paste pattern data")},
+ {1336, kcSelectAll, _T("Select all pattern data")},
+ {CommandStruct::Hidden, kcSelectCol, _T("Select Channel / Select All")},
+ */
+ {1338, kcPatternJumpDownh2, _T("Jump down by beat")},
+ {1339, kcPatternJumpUph2, _T("Jump up by beat")},
+ {1340, kcPatternSnapDownh2, _T("Snap down to beat")},
+ {1341, kcPatternSnapUph2, _T("Snap up to beat")},
+ {KeyCommand::Hidden, kcPatternJumpDownh2Select, _T("kcPatternJumpDownh2Select")},
+ {KeyCommand::Hidden, kcPatternJumpUph2Select, _T("kcPatternJumpUph2Select")},
+ {KeyCommand::Hidden, kcPatternSnapDownh2Select, _T("kcPatternSnapDownh2Select")},
+ {KeyCommand::Hidden, kcPatternSnapUph2Select, _T("kcPatternSnapUph2Select")},
+ {1346, kcFileOpen, _T("File/Open")},
+ {1347, kcFileNew, _T("File/New")},
+ {1348, kcFileClose, _T("File/Close")},
+ {1349, kcFileSave, _T("File/Save")},
+ {1350, kcFileSaveAs, _T("File/Save As")},
+ {1351, kcFileSaveAsWave, _T("File/Stream Export")},
+ {1352 | KeyCommand::Hidden, kcFileSaveAsMP3, _T("File/Stream Export")}, // Legacy
+ {1353, kcFileSaveMidi, _T("File/Export as MIDI")},
+ {1354, kcFileImportMidiLib, _T("File/Import MIDI Library")},
+ {1355, kcFileAddSoundBank, _T("File/Add Sound Bank")},
+ {1359, kcEditUndo, _T("Undo")},
+ {1360, kcEditCut, _T("Cut")},
+ {1361, kcEditCopy, _T("Copy")},
+ {1362, kcEditPaste, _T("Paste")},
+ {1363, kcEditMixPaste, _T("Mix Paste")},
+ {1364, kcEditSelectAll, _T("Select All")},
+ {1365, kcEditFind, _T("Find / Replace")},
+ {1366, kcEditFindNext, _T("Find Next")},
+ {1367, kcViewMain, _T("Toggle Main Toolbar")},
+ {1368, kcViewTree, _T("Toggle Tree View")},
+ {1369, kcViewOptions, _T("View Options")},
+ {1370, kcHelp, _T("Help")},
+ /*
+ {1370, kcWindowNew, _T("New Window")},
+ {1371, kcWindowCascade, _T("Cascade Windows")},
+ {1372, kcWindowTileHorz, _T("Tile Windows Horizontally")},
+ {1373, kcWindowTileVert, _T("Tile Windows Vertically")},
+ */
+ {1374, kcEstimateSongLength, _T("Estimate Song Length")},
+ {1375, kcStopSong, _T("Stop Song")},
+ {1376, kcMidiRecord, _T("Toggle MIDI Record")},
+ {1377, kcDeleteWholeRow, _T("Delete Row(s) (All Channels)")},
+ {1378, kcInsertRow, _T("Insert Row(s)")},
+ {1379, kcInsertWholeRow, _T("Insert Row(s) (All Channels)")},
+ {1380, kcSampleTrim, _T("Trim sample around loop points")},
+ {1381, kcSampleReverse, _T("Reverse Sample")},
+ {1382, kcSampleDelete, _T("Delete Sample Selection")},
+ {1383, kcSampleSilence, _T("Silence Sample Selection")},
+ {1384, kcSampleNormalize, _T("Normalize Sample")},
+ {1385, kcSampleAmplify, _T("Amplify Sample")},
+ {1386, kcSampleZoomUp, _T("Zoom In")},
+ {1387, kcSampleZoomDown, _T("Zoom Out")},
+ {1660, kcPatternGrowSelection, _T("Grow selection")},
+ {1661, kcPatternShrinkSelection, _T("Shrink selection")},
+ {1662, kcTogglePluginEditor, _T("Toggle channel's plugin editor")},
+ {1663, kcToggleFollowSong, _T("Toggle follow song")},
+ {1664, kcClearFieldITStyle, _T("Clear field (IT Style)")},
+ {1665, kcClearFieldStepITStyle, _T("Clear field and step (IT Style)")},
+ {1666, kcSetFXextension, _T("Parameter Extension Command")},
+ {1667 | KeyCommand::Hidden, kcNoteCutOld, _T("Note Cut")}, // Legacy
+ {1668 | KeyCommand::Hidden, kcNoteOffOld, _T("Note Off")}, // Legacy
+ {1669, kcViewAddPlugin, _T("View Plugin Manager")},
+ {1670, kcViewChannelManager, _T("View Channel Manager")},
+ {1671, kcCopyAndLoseSelection, _T("Copy and lose selection")},
+ {1672, kcNewPattern, _T("Insert new pattern")},
+ {1673, kcSampleLoad, _T("Load Sample")},
+ {1674, kcSampleSave, _T("Save Sample")},
+ {1675, kcSampleNew, _T("New Sample")},
+ //{CommandStruct::Hidden, kcSampleCtrlLoad, _T("Load Sample")},
+ //{CommandStruct::Hidden, kcSampleCtrlSave, _T("Save Sample")},
+ //{CommandStruct::Hidden, kcSampleCtrlNew, _T("New Sample")},
+ {KeyCommand::Hidden, kcInstrumentLoad, _T("Load Instrument")},
+ {KeyCommand::Hidden, kcInstrumentSave, _T("Save Instrument")},
+ {KeyCommand::Hidden, kcInstrumentNew, _T("New Instrument")},
+ {KeyCommand::Hidden, kcInstrumentCtrlLoad, _T("Load Instrument")},
+ {KeyCommand::Hidden, kcInstrumentCtrlSave, _T("Save Instrument")},
+ {KeyCommand::Hidden, kcInstrumentCtrlNew, _T("New Instrument")},
+ {1685, kcSwitchToOrderList, _T("Switch to Order List")},
+ {1686, kcEditMixPasteITStyle, _T("Mix Paste (IT Style)")},
+ {1687, kcApproxRealBPM, _T("Show approx. real BPM")},
+ {KeyCommand::Hidden, kcNavigateDownBySpacingSelect, _T("kcNavigateDownBySpacingSelect")},
+ {KeyCommand::Hidden, kcNavigateUpBySpacingSelect, _T("kcNavigateUpBySpacingSelect")},
+ {1691, kcNavigateDownBySpacing, _T("Navigate down by spacing")},
+ {1692, kcNavigateUpBySpacing, _T("Navigate up by spacing")},
+ {1693, kcPrevDocument, _T("Previous Document")},
+ {1694, kcNextDocument, _T("Next Document")},
+ {1763, kcVSTGUIPrevPreset, _T("Previous Plugin Preset")},
+ {1764, kcVSTGUINextPreset, _T("Next Plugin Preset")},
+ {1765, kcVSTGUIRandParams, _T("Randomize Plugin Parameters")},
+ {1766, kcPatternGoto, _T("Go to row/channel/...")},
+ {KeyCommand::Hidden, kcPatternOpenRandomizer, _T("Pattern Randomizer")}, // while there's not randomizer yet, let's just disable it for now
+ {1768, kcPatternInterpolateNote, _T("Interpolate Note")},
+ {KeyCommand::Hidden, kcViewGraph, _T("View Graph")}, // while there's no graph yet, let's just disable it for now
+ {1770, kcToggleChanMuteOnPatTransition, _T("(Un)mute channel on pattern transition")},
+ {1771, kcChannelUnmuteAll, _T("Unmute all channels")},
+ {1772, kcShowPatternProperties, _T("Show Pattern Properties")},
+ {1773, kcShowMacroConfig, _T("View Zxx Macro Configuration")},
+ {1775, kcViewSongProperties, _T("View Song Properties")},
+ {1776, kcChangeLoopStatus, _T("Toggle Loop Pattern")},
+ {1777, kcFileExportCompat, _T("File/Compatibility Export")},
+ {1778, kcUnmuteAllChnOnPatTransition, _T("Unmute all channels on pattern transition")},
+ {1779, kcSoloChnOnPatTransition, _T("Solo channel on pattern transition")},
+ {1780, kcTimeAtRow, _T("Show playback time at current row")},
+ {1781, kcViewMIDImapping, _T("View MIDI Mapping")},
+ {1782, kcVSTGUIPrevPresetJump, _T("Plugin preset backward jump")},
+ {1783, kcVSTGUINextPresetJump, _T("Plugin preset forward jump")},
+ {1784, kcSampleInvert, _T("Invert Sample Phase")},
+ {1785, kcSampleSignUnsign, _T("Signed / Unsigned Conversion")},
+ {1786, kcChannelReset, _T("Reset Channel")},
+ {1787, kcToggleOverflowPaste, _T("Toggle overflow paste")},
+ {1788, kcNotePC, _T("Parameter Control")},
+ {1789, kcNotePCS, _T("Parameter Control (smooth)")},
+ {1790, kcSampleRemoveDCOffset, _T("Remove DC Offset")},
+ {1791, kcNoteFade, _T("Note Fade")},
+ {1792 | KeyCommand::Hidden, kcNoteFadeOld, _T("Note Fade")}, // Legacy
+ {1793, kcEditPasteFlood, _T("Paste Flood")},
+ {1794, kcOrderlistNavigateLeft, _T("Previous Order")},
+ {1795, kcOrderlistNavigateRight, _T("Next Order")},
+ {1796, kcOrderlistNavigateFirst, _T("First Order")},
+ {1797, kcOrderlistNavigateLast, _T("Last Order")},
+ {KeyCommand::Hidden, kcOrderlistNavigateLeftSelect, _T("kcOrderlistNavigateLeftSelect")},
+ {KeyCommand::Hidden, kcOrderlistNavigateRightSelect, _T("kcOrderlistNavigateRightSelect")},
+ {KeyCommand::Hidden, kcOrderlistNavigateFirstSelect, _T("kcOrderlistNavigateFirstSelect")},
+ {KeyCommand::Hidden, kcOrderlistNavigateLastSelect, _T("kcOrderlistNavigateLastSelect")},
+ {1802, kcOrderlistEditDelete, _T("Delete Order")},
+ {1803, kcOrderlistEditInsert, _T("Insert Order")},
+ {1804, kcOrderlistEditPattern, _T("Edit Pattern")},
+ {1805, kcOrderlistSwitchToPatternView, _T("Switch to pattern editor")},
+ {1806, kcDuplicatePattern, _T("Duplicate Pattern")},
+ {1807, kcOrderlistPat0, _T("Pattern index digit 0")},
+ {1808, kcOrderlistPat1, _T("Pattern index digit 1")},
+ {1809, kcOrderlistPat2, _T("Pattern index digit 2")},
+ {1810, kcOrderlistPat3, _T("Pattern index digit 3")},
+ {1811, kcOrderlistPat4, _T("Pattern index digit 4")},
+ {1812, kcOrderlistPat5, _T("Pattern index digit 5")},
+ {1813, kcOrderlistPat6, _T("Pattern index digit 6")},
+ {1814, kcOrderlistPat7, _T("Pattern index digit 7")},
+ {1815, kcOrderlistPat8, _T("Pattern index digit 8")},
+ {1816, kcOrderlistPat9, _T("Pattern index digit 9")},
+ {1817, kcOrderlistPatPlus, _T("Increase pattern index ")},
+ {1818, kcOrderlistPatMinus, _T("Decrease pattern index")},
+ {1819, kcShowSplitKeyboardSettings, _T("Split Keyboard Settings dialog")},
+ {1820, kcEditPushForwardPaste, _T("Push Forward Paste (Insert)")},
+ {1821, kcInstrumentEnvelopePointMoveLeft, _T("Move envelope point left")},
+ {1822, kcInstrumentEnvelopePointMoveRight, _T("Move envelope point right")},
+ {1823, kcInstrumentEnvelopePointMoveUp, _T("Move envelope point up")},
+ {1824, kcInstrumentEnvelopePointMoveDown, _T("Move envelope point down")},
+ {1825, kcInstrumentEnvelopePointPrev, _T("Select previous envelope point")},
+ {1826, kcInstrumentEnvelopePointNext, _T("Select next envelope point")},
+ {1827, kcInstrumentEnvelopePointInsert, _T("Insert Envelope Point")},
+ {1828, kcInstrumentEnvelopePointRemove, _T("Remove Envelope Point")},
+ {1829, kcInstrumentEnvelopeSetLoopStart, _T("Set Loop Start")},
+ {1830, kcInstrumentEnvelopeSetLoopEnd, _T("Set Loop End")},
+ {1831, kcInstrumentEnvelopeSetSustainLoopStart, _T("Set sustain loop start")},
+ {1832, kcInstrumentEnvelopeSetSustainLoopEnd, _T("Set sustain loop end")},
+ {1833, kcInstrumentEnvelopeToggleReleaseNode, _T("Toggle release node")},
+ {1834, kcInstrumentEnvelopePointMoveUp8, _T("Move envelope point up (Coarse)")},
+ {1835, kcInstrumentEnvelopePointMoveDown8, _T("Move envelope point down (Coarse)")},
+ {1836, kcPatternEditPCNotePlugin, _T("Toggle PC Event/instrument plugin editor")},
+ {1837, kcInstrumentEnvelopeZoomIn, _T("Zoom In")},
+ {1838, kcInstrumentEnvelopeZoomOut, _T("Zoom Out")},
+ {1839, kcVSTGUIToggleRecordParams, _T("Toggle Parameter Recording")},
+ {1840, kcVSTGUIToggleSendKeysToPlug, _T("Pass Key Presses to Plugin")},
+ {1841, kcVSTGUIBypassPlug, _T("Bypass Plugin")},
+ {1842, kcInsNoteMapTransposeDown, _T("Transpose -1 (Note Map)")},
+ {1843, kcInsNoteMapTransposeUp, _T("Transpose +1 (Note Map)")},
+ {1844, kcInsNoteMapTransposeOctDown, _T("Transpose -1 Octave (Note Map)")},
+ {1845, kcInsNoteMapTransposeOctUp, _T("Transpose +1 Octave (Note Map)")},
+ {1846, kcInsNoteMapCopyCurrentNote, _T("Map all notes to selected note")},
+ {1847, kcInsNoteMapCopyCurrentSample, _T("Map all notes to selected sample")},
+ {1848, kcInsNoteMapReset, _T("Reset Note Mapping")},
+ {1849, kcInsNoteMapEditSample, _T("Edit Current Sample")},
+ {1850, kcInsNoteMapEditSampleMap, _T("Edit Sample Map")},
+ {1851, kcInstrumentCtrlDuplicate, _T("Duplicate Instrument")},
+ {1852, kcPanic, _T("Panic")},
+ {1853, kcOrderlistPatIgnore, _T("Separator (+++) Index")},
+ {1854, kcOrderlistPatInvalid, _T("Stop (---) Index")},
+ {1855, kcViewEditHistory, _T("View Edit History")},
+ {1856, kcSampleQuickFade, _T("Quick Fade")},
+ {1857, kcSampleXFade, _T("Crossfade Sample Loop")},
+ {1858, kcSelectBeat, _T("Select Beat")},
+ {1859, kcSelectMeasure, _T("Select Measure")},
+ {1860, kcFileSaveTemplate, _T("File/Save As Template")},
+ {1861, kcIncreaseSpacing, _T("Increase Edit Step")},
+ {1862, kcDecreaseSpacing, _T("Decrease Edit Step")},
+ {1863, kcSampleAutotune, _T("Tune Sample to given Note")},
+ {1864, kcFileCloseAll, _T("File/Close All")},
+ {KeyCommand::Hidden, kcSetOctaveStop0, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop1, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop2, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop3, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop4, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop5, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop6, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop7, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop8, _T("")},
+ {KeyCommand::Hidden, kcSetOctaveStop9, _T("")},
+ {1875, kcOrderlistLockPlayback, _T("Lock Playback to Selection")},
+ {1876, kcOrderlistUnlockPlayback, _T("Unlock Playback")},
+ {1877, kcChannelSettings, _T("Quick Channel Settings")},
+ {1878, kcChnSettingsPrev, _T("Previous Channel")},
+ {1879, kcChnSettingsNext, _T("Next Channel")},
+ {1880, kcChnSettingsClose, _T("Switch to Pattern Editor")},
+ {1881, kcTransposeCustom, _T("Transpose Custom")},
+ {1882, kcSampleZoomSelection, _T("Zoom into Selection")},
+ {1883, kcChannelRecordSelect, _T("Channel Record Select")},
+ {1884, kcChannelSplitRecordSelect, _T("Channel Split Record Select")},
+ {1885, kcDataEntryUp, _T("Data Entry +1")},
+ {1886, kcDataEntryDown, _T("Data Entry -1")},
+ {1887, kcSample8Bit, _T("Convert to 8-bit / 16-bit")},
+ {1888, kcSampleMonoMix, _T("Convert to Mono (Mix)")},
+ {1889, kcSampleMonoLeft, _T("Convert to Mono (Left Channel)")},
+ {1890, kcSampleMonoRight, _T("Convert to Mono (Right Channel)")},
+ {1891, kcSampleMonoSplit, _T("Convert to Mono (Split Sample)")},
+ {1892, kcQuantizeSettings, _T("Quantize Settings")},
+ {1893, kcDataEntryUpCoarse, _T("Data Entry Up (Coarse)")},
+ {1894, kcDataEntryDownCoarse, _T("Data Entry Down (Coarse)")},
+ {1895, kcToggleNoteOffRecordPC, _T("Toggle Note Off record (PC keyboard)")},
+ {1896, kcToggleNoteOffRecordMIDI, _T("Toggle Note Off record (MIDI)")},
+ {1897, kcFindInstrument, _T("Pick up nearest instrument number")},
+ {1898, kcPlaySongFromPattern, _T("Play Song from Pattern Start")},
+ {1899, kcVSTGUIToggleRecordMIDIOut, _T("Record MIDI Out to Pattern Editor")},
+ {1900, kcToggleClipboardManager, _T("Toggle Clipboard Manager")},
+ {1901, kcClipboardPrev, _T("Cycle to Previous Clipboard")},
+ {1902, kcClipboardNext, _T("Cycle to Next Clipboard")},
+ {1903, kcSelectRow, _T("Select Row")},
+ {1904, kcSelectEvent, _T("Select Event")},
+ {1905, kcEditRedo, _T("Redo")},
+ {1906, kcFileAppend, _T("File/Append Module")},
+ {1907, kcSampleTransposeUp, _T("Transpose +1")},
+ {1908, kcSampleTransposeDown, _T("Transpose -1")},
+ {1909, kcSampleTransposeOctUp, _T("Transpose +1 Octave")},
+ {1910, kcSampleTransposeOctDown, _T("Transpose -1 Octave")},
+ {1911, kcPatternInterpolateInstr, _T("Interpolate Instrument")},
+ {1912, kcDummyShortcut, _T("Dummy Shortcut")},
+ {1913, kcSampleUpsample, _T("Upsample")},
+ {1914, kcSampleDownsample, _T("Downsample")},
+ {1915, kcSampleResample, _T("Resample")},
+ {1916, kcSampleCenterLoopStart, _T("Center loop start in view")},
+ {1917, kcSampleCenterLoopEnd, _T("Center loop end in view")},
+ {1918, kcSampleCenterSustainStart, _T("Center sustain loop start in view")},
+ {1919, kcSampleCenterSustainEnd, _T("Center sustain loop end in view")},
+ {1920, kcInstrumentEnvelopeLoad, _T("Load Envelope")},
+ {1921, kcInstrumentEnvelopeSave, _T("Save Envelope")},
+ {1922, kcChannelTranspose, _T("Transpose Channel")},
+ {1923, kcChannelDuplicate, _T("Duplicate Channel")},
+ // Reserved range 1924...1949 for kcStartSampleCues...kcEndSampleCues (generated below)
+ {1950, kcOrderlistEditCopyOrders, _T("Copy Orders")},
+ {KeyCommand::Hidden, kcTreeViewStopPreview, _T("Stop sample preview")},
+ {1952, kcSampleDuplicate, _T("Duplicate Sample")},
+ {1953, kcSampleSlice, _T("Slice at cue points")},
+ {1954, kcInstrumentEnvelopeScale, _T("Scale Envelope Points")},
+ {1955, kcInsNoteMapRemove, _T("Remove All Samples")},
+ {1956, kcInstrumentEnvelopeSelectLoopStart, _T("Select Envelope Loop Start")},
+ {1957, kcInstrumentEnvelopeSelectLoopEnd, _T("Select Envelope Loop End")},
+ {1958, kcInstrumentEnvelopeSelectSustainStart, _T("Select Envelope Sustain Start")},
+ {1959, kcInstrumentEnvelopeSelectSustainEnd, _T("Select Envelope Sustain End")},
+ {1960, kcInstrumentEnvelopePointMoveLeftCoarse, _T("Move envelope point left (Coarse)")},
+ {1961, kcInstrumentEnvelopePointMoveRightCoarse, _T("Move envelope point right (Coarse)")},
+ {1962, kcSampleCenterSampleStart, _T("Zoom into sample start")},
+ {1963, kcSampleCenterSampleEnd, _T("Zoom into sample end")},
+ {1964, kcSampleTrimToLoopEnd, _T("Trim to loop end")},
+ {1965, kcLockPlaybackToRows, _T("Lock Playback to Rows")},
+ {1966, kcSwitchToInstrLibrary, _T("Switch To Instrument Library")},
+ {1967, kcPatternSetInstrumentNotEmpty, _T("Apply current instrument to existing only")},
+ {1968, kcSelectColumn, _T("Select Column")},
+ {1969, kcSampleStereoSep, _T("Change Stereo Separation")},
+ {1970, kcTransposeCustomQuick, _T("Transpose Custom (Quick)")},
+ {1971, kcPrevEntryInColumn, _T("Jump to previous entry in column")},
+ {1972, kcNextEntryInColumn, _T("Jump to next entry in column")},
+ {1973, kcViewTempoSwing, _T("View Global Tempo Swing Settings")},
+ {1974, kcChordEditor, _T("Show Chord Editor")},
+ {1975, kcToggleLoopSong, _T("Toggle Loop Song")},
+ {1976, kcInstrumentEnvelopeSwitchToVolume, _T("Switch to Volume Envelope")},
+ {1977, kcInstrumentEnvelopeSwitchToPanning, _T("Switch to Panning Envelope")},
+ {1978, kcInstrumentEnvelopeSwitchToPitch, _T("Switch to Pitch / Filter Envelope")},
+ {1979, kcInstrumentEnvelopeToggleVolume, _T("Toggle Volume Envelope")},
+ {1980, kcInstrumentEnvelopeTogglePanning, _T("Toggle Panning Envelope")},
+ {1981, kcInstrumentEnvelopeTogglePitch, _T("Toggle Pitch Envelope")},
+ {1982, kcInstrumentEnvelopeToggleFilter, _T("Toggle Filter Envelope")},
+ {1983, kcInstrumentEnvelopeToggleLoop, _T("Toggle Envelope Loop")},
+ {1984, kcInstrumentEnvelopeToggleSustain, _T("Toggle Envelope Sustain Loop")},
+ {1985, kcInstrumentEnvelopeToggleCarry, _T("Toggle Envelope Carry")},
+ {1986, kcSampleInitializeOPL, _T("Initialize OPL Instrument")},
+ {1987, kcFileSaveCopy, _T("File/Save Copy")},
+ {1988, kcMergePatterns, _T("Merge Patterns")},
+ {1989, kcSplitPattern, _T("Split Pattern")},
+ {1990, kcSampleToggleDrawing, _T("Toggle Sample Drawing")},
+ {1991, kcSampleResize, _T("Add Silence / Create Sample")},
+ {1992, kcSampleGrid, _T("Configure Sample Grid")},
+ {1993, kcLoseSelection, _T("Lose Selection")},
+ {1994, kcCutPatternChannel, _T("Cut to Pattern Channel Clipboard")},
+ {1995, kcCutPattern, _T("Cut to Pattern Clipboard")},
+ {1996, kcCopyPatternChannel, _T("Copy to Pattern Channel Clipboard")},
+ {1997, kcCopyPattern, _T("Copy to Pattern Clipboard")},
+ {1998, kcPastePatternChannel, _T("Paste from Pattern Channel Clipboard")},
+ {1999, kcPastePattern, _T("Paste from Pattern Clipboard")},
+ {2000, kcToggleSmpInsList, _T("Toggle between lists")},
+ {2001, kcExecuteSmpInsListItem, _T("Open item in editor")},
+ {2002, kcDeleteRowGlobal, _T("Delete Row(s) (Global)")},
+ {2003, kcDeleteWholeRowGlobal, _T("Delete Row(s) (All Channels, Global)")},
+ {2004, kcInsertRowGlobal, _T("Insert Row(s) (Global)")},
+ {2005, kcInsertWholeRowGlobal, _T("Insert Row(s) (All Channels, Global)")},
+ {2006, kcPrevSequence, _T("Previous Sequence")},
+ {2007, kcNextSequence, _T("Next Sequence")},
+ {2008, kcChnColorFromPrev , _T("Pick Color from Previous Channel")},
+ {2009, kcChnColorFromNext , _T("Pick Color from Next Channel") },
+ {2010, kcChannelMoveLeft, _T("Move Channels to Left")},
+ {2011, kcChannelMoveRight, _T("Move Channels to Right")},
+ {2012, kcSampleConvertPingPongLoop, _T("Convert Ping-Pong Loop to Unidirectional") },
+ {2013, kcSampleConvertPingPongSustain, _T("Convert Ping-Pong Sustain Loop to Unidirectional") },
+ {2014, kcChannelAddBefore, _T("Add Channel Before Current")},
+ {2015, kcChannelAddAfter, _T("Add Channel After Current") },
+ {2016, kcChannelRemove, _T("Remove Channel") },
+ {2017, kcSetFXFinetune, _T("Finetune") },
+ {2018, kcSetFXFinetuneSmooth, _T("Finetune (Smooth)")},
+ {2019, kcOrderlistEditInsertSeparator, _T("Insert Separator") },
+ {2020, kcTempoIncrease, _T("Increase Tempo")},
+ {2021, kcTempoDecrease, _T("Decrease Tempo")},
+ {2022, kcTempoIncreaseFine, _T("Increase Tempo (Fine)")},
+ {2023, kcTempoDecreaseFine, _T("Decrease Tempo (Fine)")},
+ {2024, kcSpeedIncrease, _T("Increase Ticks per Row")},
+ {2025, kcSpeedDecrease, _T("Decrease Ticks per Row")},
+ {2026, kcRenameSmpInsListItem, _T("Rename Item")},
+ {2027, kcShowChannelCtxMenu, _T("Show Channel Context (Right-Click) Menu")},
+ {2028, kcShowChannelPluginCtxMenu, _T("Show Channel Plugin Context (Right-Click) Menu")},
+ {2029, kcViewToggle, _T("Toggle Between Upper / Lower View") },
+ {2030, kcFileSaveOPL, _T("File/Export OPL Register Dump") },
+ {2031, kcSampleLoadRaw, _T("Load Raw Sample")},
+ {2032, kcTogglePatternPlayRow, _T("Toggle row playback when navigating")},
+ {2033, kcInsNoteMapTransposeSamples, _T("Transpose Samples / Reset Map") },
+ {KeyCommand::Hidden, kcPrevEntryInColumnSelect, _T("kcPrevEntryInColumnSelect")},
+ {KeyCommand::Hidden, kcNextEntryInColumnSelect, _T("kcNextEntryInColumnSelect")},
+};
+
+// Get command descriptions etc.. loaded up.
+void CCommandSet::SetupCommands()
+{
+ for(const auto &def : CommandDefinitions)
+ {
+ m_commands[def.cmd] = {def.uid, def.description};
+ }
+
+ for(int j = kcStartSampleCues; j <= kcEndSampleCues; j++)
+ {
+ CString s = MPT_CFORMAT("Preview Sample Cue {}")(j - kcStartSampleCues + 1);
+ m_commands[j] = {static_cast<uint32>(1924 + j - kcStartSampleCues), s};
+ }
+ static_assert(1924 + kcEndSampleCues - kcStartSampleCues < 1950);
+
+ // Automatically generated note entry keys in non-pattern contexts
+ for(const auto ctx : NoteContexts)
+ {
+ const auto contextStartNotes = std::get<1>(ctx);
+ const auto contextStopNotes = std::get<2>(ctx);
+ if(contextStartNotes == kcVPStartNotes)
+ continue;
+
+ for(int i = kcVPStartNotes; i <= kcVPEndNotes; i++)
+ {
+ m_commands[i - kcVPStartNotes + contextStartNotes] = {KeyCommand::Hidden, m_commands[i].Message};
+ }
+ for(int i = kcVPStartNoteStops; i <= kcVPEndNoteStops; i++)
+ {
+ m_commands[i - kcVPStartNoteStops + contextStopNotes] = {KeyCommand::Hidden, m_commands[i].Message};
+ }
+ }
+
+#ifdef MPT_BUILD_DEBUG
+ // Ensure that every visible command has a unique ID
+ for(size_t i = 0; i < kcNumCommands; i++)
+ {
+ if(m_commands[i].ID() != 0 || !m_commands[i].IsHidden())
+ {
+ for(size_t j = i + 1; j < kcNumCommands; j++)
+ {
+ if(m_commands[i].ID() == m_commands[j].ID())
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Duplicate or unset command UID: {}\n")(m_commands[i].ID()));
+ MPT_ASSERT_NOTREACHED();
+ }
+ }
+ }
+ }
+#endif // MPT_BUILD_DEBUG
+}
+
+
+// Command Manipulation
+
+
+CString CCommandSet::Add(KeyCombination kc, CommandID cmd, bool overwrite, int pos, bool checkEventConflict)
+{
+ auto &kcList = m_commands[cmd].kcList;
+
+ // Avoid duplicate
+ if(mpt::contains(kcList, kc))
+ {
+ return CString();
+ }
+
+ // Check that this keycombination isn't already assigned (in this context), except for dummy keys
+ CString report;
+ if(auto conflictCmd = IsConflicting(kc, cmd, checkEventConflict); conflictCmd.first != kcNull)
+ {
+ if (!overwrite)
+ {
+ return CString();
+ } else
+ {
+ if (IsCrossContextConflict(kc, conflictCmd.second))
+ {
+ report += _T("The following commands may conflict:\r\n >") + GetCommandText(conflictCmd.first) + _T(" in ") + conflictCmd.second.GetContextText() + _T("\r\n >") + GetCommandText(cmd) + _T(" in ") + kc.GetContextText() + _T("\r\n\r\n");
+ LOG_COMMANDSET(mpt::ToUnicode(report));
+ } else
+ {
+ //if(!TrackerSettings::Instance().MiscAllowMultipleCommandsPerKey)
+ // Remove(conflictCmd.second, conflictCmd.first);
+ report += _T("The following commands in same context share the same key combination:\r\n >") + GetCommandText(conflictCmd.first) + _T(" in ") + conflictCmd.second.GetContextText() + _T("\r\n\r\n");
+ LOG_COMMANDSET(mpt::ToUnicode(report));
+ }
+ }
+ }
+
+ kcList.insert((pos < 0) ? kcList.end() : (kcList.begin() + pos), kc);
+
+ //enfore rules on CommandSet
+ report += EnforceAll(kc, cmd, true);
+ return report;
+}
+
+
+std::pair<CommandID, KeyCombination> CCommandSet::IsConflicting(KeyCombination kc, CommandID cmd, bool checkEventConflict) const
+{
+ if(m_commands[cmd].IsDummy()) // no need to search if we are adding a dummy key
+ return {kcNull, KeyCombination()};
+
+ for(int pass = 0; pass < 2; pass++)
+ {
+ // In the first pass, only look for conflicts in the same context, since
+ // such conflicts are errors. Cross-context conflicts only emit warnings.
+ for(int curCmd = 0; curCmd < kcNumCommands; curCmd++)
+ {
+ if(m_commands[curCmd].IsDummy())
+ continue;
+
+ for(auto &curKc : m_commands[curCmd].kcList)
+ {
+ if(pass == 0 && curKc.Context() != kc.Context())
+ continue;
+
+ if(KeyCombinationConflict(curKc, kc, checkEventConflict))
+ {
+ return {(CommandID)curCmd, curKc};
+ }
+ }
+ }
+ }
+
+ return std::make_pair(kcNull, KeyCombination());
+}
+
+
+CString CCommandSet::Remove(int pos, CommandID cmd)
+{
+ if (pos>=0 && (size_t)pos<m_commands[cmd].kcList.size())
+ {
+ return Remove(m_commands[cmd].kcList[pos], cmd);
+ }
+
+ LOG_COMMANDSET(U_("Failed to remove a key: keychoice out of range."));
+ return _T("");
+}
+
+
+CString CCommandSet::Remove(KeyCombination kc, CommandID cmd)
+{
+ auto &kcList = m_commands[cmd].kcList;
+ auto index = std::find(kcList.begin(), kcList.end(), kc);
+ if (index != kcList.end())
+ {
+ kcList.erase(index);
+ LOG_COMMANDSET(U_("Removed a key"));
+ return EnforceAll(kc, cmd, false);
+ } else
+ {
+ LOG_COMMANDSET(U_("Failed to remove a key as it was not found"));
+ return CString();
+ }
+
+}
+
+
+CString CCommandSet::EnforceAll(KeyCombination inKc, CommandID inCmd, bool adding)
+{
+ //World's biggest, most confusing method. :)
+ //Needs refactoring. Maybe make lots of Rule subclasses, each with their own Enforce() method?
+ KeyCombination curKc; // for looping through key combinations
+ KeyCombination newKc; // for adding new key combinations
+ CString report;
+
+ if(m_enforceRule[krAllowNavigationWithSelection])
+ {
+ // When we get a new navigation command key, we need to
+ // make sure this navigation will work when any selection key is pressed
+ if(inCmd >= kcStartPatNavigation && inCmd <= kcEndPatNavigation)
+ {//Check that it is a nav cmd
+ CommandID cmdNavSelection = (CommandID)(kcStartPatNavigationSelect + (inCmd-kcStartPatNavigation));
+ for(auto &kc :m_commands[kcSelect].kcList)
+ {//for all selection modifiers
+ newKc = inKc;
+ newKc.Modifier(kc); //Add selection modifier's modifiers to this command
+ if(adding)
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - adding key with modifier:{} to command: {}")(newKc.Modifier().GetRaw(), cmdNavSelection));
+ Add(newKc, cmdNavSelection, false);
+ } else
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - removing key with modifier:{} to command: {}")(newKc.Modifier().GetRaw(), cmdNavSelection));
+ Remove(newKc, cmdNavSelection);
+ }
+ }
+ }
+ // Same applies for orderlist navigation
+ else if(inCmd >= kcStartOrderlistNavigation && inCmd <= kcEndOrderlistNavigation)
+ {//Check that it is a nav cmd
+ CommandID cmdNavSelection = (CommandID)(kcStartOrderlistNavigationSelect+ (inCmd-kcStartOrderlistNavigation));
+ for(auto &kc : m_commands[kcSelect].kcList)
+ {//for all selection modifiers
+ newKc = inKc;
+ newKc.AddModifier(kc); //Add selection modifier's modifiers to this command
+ if(adding)
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - adding key with modifier:{} to command: {}")(newKc.Modifier().GetRaw(), cmdNavSelection));
+ Add(newKc, cmdNavSelection, false);
+ } else
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - removing key with modifier:{} to command: {}")(newKc.Modifier().GetRaw(), cmdNavSelection));
+ Remove(newKc, cmdNavSelection);
+ }
+ }
+ }
+ // When we get a new selection key, we need to make sure that
+ // all navigation commands will work with this selection key pressed
+ else if(inCmd == kcSelect)
+ {
+ // check that is is a selection
+ for(int curCmd=kcStartPatNavigation; curCmd<=kcEndPatNavigation; curCmd++)
+ {
+ // for all nav commands
+ for(auto &kc : m_commands[curCmd].kcList)
+ {
+ // for all keys for this command
+ CommandID cmdNavSelection = (CommandID)(kcStartPatNavigationSelect + (curCmd-kcStartPatNavigation));
+ newKc = kc; // get all properties from the current nav cmd key
+ newKc.AddModifier(inKc); // and the new selection modifier
+ if(adding)
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - adding key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), cmdNavSelection));
+ Add(newKc, cmdNavSelection, false);
+ } else
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - removing key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), cmdNavSelection));
+ Remove(newKc, cmdNavSelection);
+ }
+ }
+ } // end all nav commands
+ for(int curCmd = kcStartOrderlistNavigation; curCmd <= kcEndOrderlistNavigation; curCmd++)
+ {// for all nav commands
+ for(auto &kc : m_commands[curCmd].kcList)
+ {// for all keys for this command
+ CommandID cmdNavSelection = (CommandID)(kcStartOrderlistNavigationSelect+ (curCmd-kcStartOrderlistNavigation));
+ newKc = kc; // get all properties from the current nav cmd key
+ newKc.AddModifier(inKc); // and the new selection modifier
+ if(adding)
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - adding key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), cmdNavSelection));
+ Add(newKc, cmdNavSelection, false);
+ } else
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowNavigationWithSelection - removing key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), cmdNavSelection));
+ Remove(newKc, cmdNavSelection);
+ }
+ }
+ } // end all nav commands
+ }
+ } // end krAllowNavigationWithSelection
+
+ if(m_enforceRule[krAllowSelectionWithNavigation])
+ {
+ KeyCombination newKcSel;
+
+ // When we get a new navigation command key, we need to ensure
+ // all selection keys will work even when this new selection key is pressed
+ if(inCmd >= kcStartPatNavigation && inCmd <= kcEndPatNavigation)
+ {//if this is a navigation command
+ for(auto &kc : m_commands[kcSelect].kcList)
+ {//for all deselection modifiers
+ newKcSel = kc; // get all properties from the selection key
+ newKcSel.AddModifier(inKc); // add modifiers from the new nav command
+ if(adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krAllowSelectionWithNavigation: adding removing kcSelectWithNav and kcSelectOffWithNav"));
+ Add(newKcSel, kcSelectWithNav, false);
+ } else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krAllowSelectionWithNavigation: removing kcSelectWithNav and kcSelectOffWithNav"));
+ Remove(newKcSel, kcSelectWithNav);
+ }
+ }
+ }
+ // Same for orderlist navigation
+ if(inCmd >= kcStartOrderlistNavigation && inCmd <= kcEndOrderlistNavigation)
+ {//if this is a navigation command
+ for(auto &kc : m_commands[kcSelect].kcList)
+ {//for all deselection modifiers
+ newKcSel = kc; // get all properties from the selection key
+ newKcSel.AddModifier(inKc); // add modifiers from the new nav command
+ if(adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krAllowSelectionWithNavigation: adding removing kcSelectWithNav and kcSelectOffWithNav"));
+ Add(newKcSel, kcSelectWithNav, false);
+ } else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krAllowSelectionWithNavigation: removing kcSelectWithNav and kcSelectOffWithNav"));
+ Remove(newKcSel, kcSelectWithNav);
+ }
+ }
+ }
+ // When we get a new selection key, we need to ensure it will work even when
+ // any navigation key is pressed
+ else if(inCmd == kcSelect)
+ {
+ for(int curCmd = kcStartPatNavigation; curCmd <= kcEndPatNavigation; curCmd++)
+ {//for all nav commands
+ for(auto &kc : m_commands[curCmd].kcList)
+ {// for all keys for this command
+ newKcSel = inKc; // get all properties from the selection key
+ newKcSel.AddModifier(kc); //add the nav keys' modifiers
+ if(adding)
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowSelectionWithNavigation - adding key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), kcSelectWithNav));
+ Add(newKcSel, kcSelectWithNav, false);
+ } else
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowSelectionWithNavigation - removing key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), kcSelectWithNav));
+ Remove(newKcSel, kcSelectWithNav);
+ }
+ }
+ } // end all nav commands
+
+ for(int curCmd = kcStartOrderlistNavigation; curCmd <= kcEndOrderlistNavigation; curCmd++)
+ {//for all nav commands
+ for(auto &kc : m_commands[curCmd].kcList)
+ {// for all keys for this command
+ newKcSel=inKc; // get all properties from the selection key
+ newKcSel.AddModifier(kc); //add the nav keys' modifiers
+ if(adding)
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowSelectionWithNavigation - adding key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), kcSelectWithNav));
+ Add(newKcSel, kcSelectWithNav, false);
+ } else
+ {
+ LOG_COMMANDSET(MPT_UFORMAT("Enforcing rule krAllowSelectionWithNavigation - removing key:{} with modifier:{} to command: {}")(curCmd, inKc.Modifier().GetRaw(), kcSelectWithNav));
+ Remove(newKcSel, kcSelectWithNav);
+ }
+ }
+ } // end all nav commands
+ }
+
+ }
+
+ // if we add a selector or a copy selector, we need it to switch off when we release the key.
+ if (m_enforceRule[krAutoSelectOff])
+ {
+ KeyCombination newKcDeSel;
+ CommandID cmdOff = kcNull;
+ switch (inCmd)
+ {
+ case kcSelect: cmdOff = kcSelectOff; break;
+ case kcSelectWithNav: cmdOff = kcSelectOffWithNav; break;
+ case kcCopySelect: cmdOff = kcCopySelectOff; break;
+ case kcCopySelectWithNav: cmdOff = kcCopySelectOffWithNav; break;
+ case kcSelectWithCopySelect: cmdOff = kcSelectOffWithCopySelect; break;
+ case kcCopySelectWithSelect: cmdOff = kcCopySelectOffWithSelect; break;
+ }
+
+ if(cmdOff != kcNull)
+ {
+ newKcDeSel = inKc;
+ newKcDeSel.EventType(kKeyEventUp);
+
+ // Register key-up when releasing any of the modifiers.
+ // Otherwise, select key combos might get stuck. Example:
+ // [Ctrl Down] [Alt Down] [Ctrl Up] [Alt Up] After this action, copy select (Ctrl+Drag) would still be activated without this code.
+ const UINT maxMod = TrackerSettings::Instance().MiscDistinguishModifiers ? MaxMod : (MaxMod & ~(HOTKEYF_RSHIFT | HOTKEYF_RCONTROL | HOTKEYF_RALT));
+ for(UINT i = 0; i <= maxMod; i++)
+ {
+ // Avoid Windows key, so that it won't detected as being actively used
+ if(i & HOTKEYF_EXT)
+ continue;
+ newKcDeSel.Modifier(static_cast<Modifiers>(i));
+ //newKcDeSel.mod&=~CodeToModifier(inKc.code); //<-- Need to get rid of right modifier!!
+
+ if (adding)
+ Add(newKcDeSel, cmdOff, false);
+ else
+ Remove(newKcDeSel, cmdOff);
+ }
+ }
+
+ }
+ // Allow combinations of copyselect and select
+ if(m_enforceRule[krAllowSelectCopySelectCombos])
+ {
+ KeyCombination newKcSel, newKcCopySel;
+ if(inCmd==kcSelect)
+ {
+ // On getting a new selection key, make this selection key work with all copy selects' modifiers
+ // On getting a new selection key, make all copyselects work with this key's modifiers
+ for(auto &kc : m_commands[kcCopySelect].kcList)
+ {
+ newKcSel=inKc;
+ newKcSel.AddModifier(kc);
+ newKcCopySel = kc;
+ newKcCopySel.AddModifier(inKc);
+ LOG_COMMANDSET(U_("Enforcing rule krAllowSelectCopySelectCombos"));
+ if(adding)
+ {
+ Add(newKcSel, kcSelectWithCopySelect, false);
+ Add(newKcCopySel, kcCopySelectWithSelect, false);
+ } else
+ {
+ Remove(newKcSel, kcSelectWithCopySelect);
+ Remove(newKcCopySel, kcCopySelectWithSelect);
+ }
+ }
+ }
+ if(inCmd == kcCopySelect)
+ {
+ // On getting a new copyselection key, make this copyselection key work with all selects' modifiers
+ // On getting a new copyselection key, make all selects work with this key's modifiers
+ for(auto &kc : m_commands[kcSelect].kcList)
+ {
+ newKcSel = kc;
+ newKcSel.AddModifier(inKc);
+ newKcCopySel = inKc;
+ newKcCopySel.AddModifier(kc);
+ LOG_COMMANDSET(U_("Enforcing rule krAllowSelectCopySelectCombos"));
+ if(adding)
+ {
+ Add(newKcSel, kcSelectWithCopySelect, false);
+ Add(newKcCopySel, kcCopySelectWithSelect, false);
+ } else
+ {
+ Remove(newKcSel, kcSelectWithCopySelect);
+ Remove(newKcCopySel, kcCopySelectWithSelect);
+ }
+ }
+ }
+ }
+
+
+ // Lock Notes to Chords
+ if (m_enforceRule[krLockNotesToChords])
+ {
+ if (inCmd>=kcVPStartNotes && inCmd<=kcVPEndNotes)
+ {
+ int noteOffset = inCmd - kcVPStartNotes;
+ for(auto &kc : m_commands[kcChordModifier].kcList)
+ {//for all chord modifier keys
+ newKc = inKc;
+ newKc.AddModifier(kc);
+ if (adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krLockNotesToChords: auto adding in a chord command"));
+ Add(newKc, (CommandID)(kcVPStartChords+noteOffset), false);
+ }
+ else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krLockNotesToChords: auto removing a chord command"));
+ Remove(newKc, (CommandID)(kcVPStartChords+noteOffset));
+ }
+ }
+ }
+ if (inCmd==kcChordModifier)
+ {
+ int noteOffset;
+ for (int curCmd=kcVPStartNotes; curCmd<=kcVPEndNotes; curCmd++)
+ {//for all notes
+ for(auto kc : m_commands[curCmd].kcList)
+ {//for all keys for this note
+ noteOffset = curCmd - kcVPStartNotes;
+ kc.AddModifier(inKc);
+ if (adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krLockNotesToChords: auto adding in a chord command"));
+ Add(kc, (CommandID)(kcVPStartChords+noteOffset), false);
+ }
+ else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krLockNotesToChords: auto removing a chord command"));
+ Remove(kc, (CommandID)(kcVPStartChords+noteOffset));
+ }
+ }
+ }
+ }
+ }
+
+
+ // Auto set note off on release
+ if (m_enforceRule[krNoteOffOnKeyRelease])
+ {
+ if (inCmd>=kcVPStartNotes && inCmd<=kcVPEndNotes)
+ {
+ int noteOffset = inCmd - kcVPStartNotes;
+ newKc=inKc;
+ newKc.EventType(kKeyEventUp);
+ if (adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krNoteOffOnKeyRelease: adding note off command"));
+ Add(newKc, (CommandID)(kcVPStartNoteStops+noteOffset), false);
+ }
+ else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krNoteOffOnKeyRelease: removing note off command"));
+ Remove(newKc, (CommandID)(kcVPStartNoteStops+noteOffset));
+ }
+ }
+ if (inCmd>=kcVPStartChords && inCmd<=kcVPEndChords)
+ {
+ int noteOffset = inCmd - kcVPStartChords;
+ newKc=inKc;
+ newKc.EventType(kKeyEventUp);
+ if (adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krNoteOffOnKeyRelease: adding Chord off command"));
+ Add(newKc, (CommandID)(kcVPStartChordStops+noteOffset), false);
+ }
+ else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krNoteOffOnKeyRelease: removing Chord off command"));
+ Remove(newKc, (CommandID)(kcVPStartChordStops+noteOffset));
+ }
+ }
+ if(inCmd >= kcSetOctave0 && inCmd <= kcSetOctave9)
+ {
+ int noteOffset = inCmd - kcSetOctave0;
+ newKc=inKc;
+ newKc.EventType(kKeyEventUp);
+ if (adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krNoteOffOnKeyRelease: adding Chord off command"));
+ Add(newKc, (CommandID)(kcSetOctaveStop0+noteOffset), false);
+ }
+ else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krNoteOffOnKeyRelease: removing Chord off command"));
+ Remove(newKc, (CommandID)(kcSetOctaveStop0+noteOffset));
+ }
+ }
+ }
+
+ // Reassign freed number keys to octaves
+ if (m_enforceRule[krReassignDigitsToOctaves] && !adding)
+ {
+ if ( (inKc.Modifier() == ModNone) && //no modifier
+ ( (inKc.Context() == kCtxViewPatternsNote) || (inKc.Context() == kCtxViewPatterns) ) && //note scope or pattern scope
+ ( ('0'<=inKc.KeyCode() && inKc.KeyCode()<='9') || (VK_NUMPAD0<=inKc.KeyCode() && inKc.KeyCode()<=VK_NUMPAD9) ) ) //is number key
+ {
+ newKc = KeyCombination(kCtxViewPatternsNote, ModNone, inKc.KeyCode(), kKeyEventDown);
+ int offset = ('0'<=inKc.KeyCode() && inKc.KeyCode()<='9') ? newKc.KeyCode()-'0' : newKc.KeyCode()-VK_NUMPAD0;
+ Add(newKc, (CommandID)(kcSetOctave0 + (newKc.KeyCode()-offset)), false);
+ }
+ }
+ // Add spacing
+ if (m_enforceRule[krAutoSpacing])
+ {
+ if (inCmd == kcSetSpacing && adding)
+ {
+ newKc = KeyCombination(kCtxViewPatterns, inKc.Modifier(), 0, kKeyEventDown);
+ for (char i = 0; i <= 9; i++)
+ {
+ newKc.KeyCode('0' + i);
+ Add(newKc, (CommandID)(kcSetSpacing0 + i), false);
+ newKc.KeyCode(VK_NUMPAD0 + i);
+ Add(newKc, (CommandID)(kcSetSpacing0 + i), false);
+ }
+ }
+ else if (!adding && (inCmd < kcSetSpacing || inCmd > kcSetSpacing9))
+ {
+ // Re-add combinations that might have been overwritten by another command
+ if(('0' <= inKc.KeyCode() && inKc.KeyCode() <= '9') || (VK_NUMPAD0 <= inKc.KeyCode() && inKc.KeyCode() <= VK_NUMPAD9))
+ {
+ for(const auto &spacing : m_commands[kcSetSpacing].kcList)
+ {
+ newKc = KeyCombination(kCtxViewPatterns, spacing.Modifier(), inKc.KeyCode(), spacing.EventType());
+ if('0' <= inKc.KeyCode() && inKc.KeyCode() <= '9')
+ Add(newKc, (CommandID)(kcSetSpacing0 + inKc.KeyCode() - '0'), false);
+ else if(VK_NUMPAD0 <= inKc.KeyCode() && inKc.KeyCode() <= VK_NUMPAD9)
+ Add(newKc, (CommandID)(kcSetSpacing0 + inKc.KeyCode() - VK_NUMPAD0), false);
+ }
+ }
+
+ }
+ }
+ if (m_enforceRule[krPropagateNotes])
+ {
+ if((inCmd >= kcVPStartNotes && inCmd <= kcVPEndNotes) || (inCmd >= kcVPStartNoteStops && inCmd <= kcVPEndNoteStops))
+ {
+ const bool areNoteStarts = (inCmd >= kcVPStartNotes && inCmd <= kcVPEndNotes);
+ const auto startNote = areNoteStarts ? kcVPStartNotes : kcVPStartNoteStops;
+ const auto noteOffset = inCmd - startNote;
+ for(const auto ctx : NoteContexts)
+ {
+ const auto context = std::get<0>(ctx);
+ const auto contextStartNote = areNoteStarts ? std::get<1>(ctx) : std::get<2>(ctx);
+
+ if(contextStartNote == startNote)
+ continue;
+
+ newKc = inKc;
+ newKc.Context(context);
+
+ if(adding)
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krPropagateNotes: adding Note on/off"));
+ Add(newKc, static_cast<CommandID>(contextStartNote + noteOffset), false);
+ } else
+ {
+ LOG_COMMANDSET(U_("Enforcing rule krPropagateNotes: removing Note on/off"));
+ Remove(newKc, static_cast<CommandID>(contextStartNote + noteOffset));
+ }
+ }
+ } else if(inCmd == kcNoteCut || inCmd == kcNoteOff || inCmd == kcNoteFade)
+ {
+ // Stop preview in instrument browser
+ KeyCombination newKcTree = inKc;
+ newKcTree.Context(kCtxViewTree);
+ if(adding)
+ {
+ Add(newKcTree, kcTreeViewStopPreview, false);
+ } else
+ {
+ Remove(newKcTree, kcTreeViewStopPreview);
+ }
+ }
+ }
+ if (m_enforceRule[krCheckModifiers])
+ {
+ // for all commands that must be modifiers
+ for (auto curCmd : { kcSelect, kcCopySelect, kcChordModifier, kcSetSpacing })
+ {
+ //for all of this command's key combinations
+ for (auto &kc : m_commands[curCmd].kcList)
+ {
+ if ((!kc.Modifier()) || (kc.KeyCode()!=VK_SHIFT && kc.KeyCode()!=VK_CONTROL && kc.KeyCode()!=VK_MENU && kc.KeyCode()!=0 &&
+ kc.KeyCode()!=VK_LWIN && kc.KeyCode()!=VK_RWIN )) // Feature: use Windows keys as modifier keys
+ {
+ report += _T("Error! ") + GetCommandText((CommandID)curCmd) + _T(" must be a modifier (shift/ctrl/alt), but is currently ") + inKc.GetKeyText() + _T("\r\n");
+ //replace with dummy
+ kc.Modifier(ModShift);
+ kc.KeyCode(0);
+ kc.EventType(kKeyEventNone);
+ }
+ }
+
+ }
+ }
+ if (m_enforceRule[krPropagateSampleManipulation])
+ {
+ static constexpr CommandID propagateCmds[] = {kcSampleLoad, kcSampleSave, kcSampleNew};
+ static constexpr CommandID translatedCmds[] = {kcInstrumentLoad, kcInstrumentSave, kcInstrumentNew};
+ if(const auto propCmd = std::find(std::begin(propagateCmds), std::end(propagateCmds), inCmd); propCmd != std::end(propagateCmds))
+ {
+ //propagate to InstrumentView
+ const auto newCmd = translatedCmds[std::distance(std::begin(propagateCmds), propCmd)];
+ m_commands[newCmd].kcList.reserve(m_commands[inCmd].kcList.size());
+ for(auto kc : m_commands[inCmd].kcList)
+ {
+ kc.Context(kCtxViewInstruments);
+ m_commands[newCmd].kcList.push_back(kc);
+ }
+ }
+
+ }
+
+/* if (enforceRule[krFoldEffectColumnAnd])
+ {
+ if (inKc.ctx == kCtxViewPatternsFX) {
+ KeyCombination newKc = inKc;
+ newKc.ctx = kCtxViewPatternsFXparam;
+ if (adding) {
+ Add(newKc, inCmd, false);
+ } else {
+ Remove(newKc, inCmd);
+ }
+ }
+ if (inKc.ctx == kCtxViewPatternsFXparam) {
+ KeyCombination newKc = inKc;
+ newKc.ctx = kCtxViewPatternsFX;
+ if (adding) {
+ Add(newKc, inCmd, false);
+ } else {
+ Remove(newKc, inCmd);
+ }
+ }
+ }
+*/
+ return report;
+}
+
+
+//Generate a keymap from a command set
+void CCommandSet::GenKeyMap(KeyMap &km)
+{
+ std::vector<KeyEventType> eventTypes;
+ std::vector<InputTargetContext> contexts;
+
+ km.clear();
+
+ const bool allowDupes = TrackerSettings::Instance().MiscAllowMultipleCommandsPerKey;
+
+ // Copy commandlist content into map:
+ for(UINT cmd = 0; cmd < kcNumCommands; cmd++)
+ {
+ if(m_commands[cmd].IsDummy())
+ continue;
+
+ for(auto curKc : m_commands[cmd].kcList)
+ {
+ eventTypes.clear();
+ contexts.clear();
+
+ // Handle keyEventType mask.
+ if(curKc.EventType() & kKeyEventDown)
+ eventTypes.push_back(kKeyEventDown);
+ if(curKc.EventType() & kKeyEventUp)
+ eventTypes.push_back(kKeyEventUp);
+ if(curKc.EventType() & kKeyEventRepeat)
+ eventTypes.push_back(kKeyEventRepeat);
+ //ASSERT(eventTypes.GetSize()>0);
+
+ // Handle super-contexts (contexts that represent a set of sub contexts)
+ if(curKc.Context() == kCtxViewPatterns)
+ contexts.insert(contexts.end(), {kCtxViewPatternsNote, kCtxViewPatternsIns, kCtxViewPatternsVol, kCtxViewPatternsFX, kCtxViewPatternsFXparam});
+ else if(curKc.Context() == kCtxCtrlPatterns)
+ contexts.push_back(kCtxCtrlOrderlist);
+ else
+ contexts.push_back(curKc.Context());
+
+ for(auto ctx : contexts)
+ {
+ for(auto event : eventTypes)
+ {
+ KeyCombination kc(ctx, curKc.Modifier(), curKc.KeyCode(), event);
+ if(!allowDupes)
+ {
+ KeyMapRange dupes = km.equal_range(kc);
+ km.erase(dupes.first, dupes.second);
+ }
+ km.insert(std::make_pair(kc, static_cast<CommandID>(cmd)));
+ }
+ }
+ }
+ }
+}
+
+
+void CCommandSet::Copy(const CCommandSet *source)
+{
+ m_oldSpecs = nullptr;
+ std::copy(std::begin(source->m_commands), std::end(source->m_commands), std::begin(m_commands));
+}
+
+
+// Export
+
+bool CCommandSet::SaveFile(const mpt::PathString &filename)
+{
+
+/* Layout:
+//----( Context1 Text (id) )----
+ctx:UID:Description:Modifier:Key:EventMask
+ctx:UID:Description:Modifier:Key:EventMask
+...
+//----( Context2 Text (id) )----
+...
+*/
+
+ mpt::SafeOutputFile sf(filename, std::ios::out, mpt::FlushModeFromBool(TrackerSettings::Instance().MiscFlushFileBuffersOnSave));
+ mpt::ofstream& f = sf;
+ if(!f)
+ {
+ ErrorBox(IDS_CANT_OPEN_FILE_FOR_WRITING);
+ return false;
+ }
+ f << "//----------------- OpenMPT key binding definition file ---------------\n"
+ "//- Format is: -\n"
+ "//- Context:Command ID:Modifiers:Key:KeypressEventType //Comments -\n"
+ "//----------------------------------------------------------------------\n"
+ "version:" << KEYMAP_VERSION << "\n";
+
+ std::vector<HKL> layouts(GetKeyboardLayoutList(0, nullptr));
+ GetKeyboardLayoutList(static_cast<int>(layouts.size()), layouts.data());
+
+ for(int ctx = 0; ctx < kCtxMaxInputContexts; ctx++)
+ {
+ f << "\n//----( " << mpt::ToCharset(mpt::Charset::UTF8, KeyCombination::GetContextText((InputTargetContext)ctx)) << " )------------\n";
+
+ for(int cmd = kcFirst; cmd < kcNumCommands; cmd++)
+ {
+ if(m_commands[cmd].IsHidden())
+ continue;
+
+ for(const auto kc : m_commands[cmd].kcList)
+ {
+ if(kc.Context() != ctx)
+ continue; // Sort by context
+
+ f << ctx << ":"
+ << m_commands[cmd].ID() << ":"
+ << static_cast<int>(kc.Modifier().GetRaw()) << ":"
+ << kc.KeyCode();
+ if(cmd >= kcVPStartNotes && cmd <= kcVPEndNotes)
+ {
+ UINT sc = 0, vk = kc.KeyCode();
+ for(auto i = layouts.begin(); i != layouts.end() && sc == 0; i++)
+ {
+ sc = MapVirtualKeyEx(vk, MAPVK_VK_TO_VSC, *i);
+ }
+ f << "/" << sc;
+ }
+ f << ":"
+ << static_cast<int>(kc.EventType().GetRaw()) << "\t\t//"
+ << mpt::ToCharset(mpt::Charset::UTF8, GetCommandText((CommandID)cmd)) << ": "
+ << mpt::ToCharset(mpt::Charset::UTF8, kc.GetKeyText()) << " ("
+ << mpt::ToCharset(mpt::Charset::UTF8, kc.GetKeyEventText()) << ")\n";
+ }
+ }
+ }
+
+ return true;
+}
+
+
+static std::string GetDefaultKeymap()
+{
+ return mpt::make_basic_string(mpt::byte_cast<mpt::span<const char>>(GetResource(MAKEINTRESOURCE(IDR_DEFAULT_KEYBINDINGS), TEXT("KEYBINDINGS"))));
+}
+
+
+bool CCommandSet::LoadFile(std::istream &iStrm, const mpt::ustring &filenameDescription, const bool fillExistingSet)
+{
+ KeyCombination kc;
+ char s[1024];
+ std::string curLine;
+ std::vector<std::string> tokens;
+ int l = 0;
+ m_oldSpecs = nullptr; // After clearing the key set, need to fix effect letters
+
+ if(!fillExistingSet)
+ {
+ for(auto &cmd : m_commands)
+ cmd.kcList.clear();
+ }
+
+ CString errText;
+ int errorCount = 0;
+
+ std::vector<HKL> layouts(GetKeyboardLayoutList(0, nullptr));
+ GetKeyboardLayoutList(static_cast<int>(layouts.size()), layouts.data());
+
+ const std::string whitespace(" \n\r\t");
+ while(iStrm.getline(s, std::size(s)))
+ {
+ curLine = s;
+ l++;
+
+ // Cut everything after a //, trim whitespace
+ auto pos = curLine.find("//");
+ if(pos != std::string::npos)
+ curLine.resize(pos);
+ pos = curLine.find_first_not_of(whitespace);
+ if(pos == std::string::npos)
+ continue;
+ curLine.erase(0, pos);
+ pos = curLine.find_last_not_of(whitespace);
+ if(pos != std::string::npos)
+ curLine.resize(pos + 1);
+
+ if (curLine.empty())
+ continue;
+
+ tokens = mpt::String::Split<std::string>(curLine, ":");
+ if(tokens.size() == 2 && !mpt::CompareNoCaseAscii(tokens[0], "version"))
+ {
+ // This line indicates the version of this keymap file (e.g. "version:1")
+ int fileVersion = ConvertStrTo<int>(tokens[1]);
+ if(fileVersion > KEYMAP_VERSION)
+ {
+ errText.AppendFormat(_T("File version is %d, but your version of OpenMPT only supports loading files up to version %d.\n"), fileVersion, KEYMAP_VERSION);
+ }
+ continue;
+ }
+
+ // Format: ctx:UID:Description:Modifier:Key:EventMask
+ CommandID cmd = kcNumCommands;
+ if(tokens.size() >= 5)
+ {
+ kc.Context(static_cast<InputTargetContext>(ConvertStrTo<int>(tokens[0])));
+ cmd = FindCmd(ConvertStrTo<uint32>(tokens[1]));
+
+ // Modifier
+ kc.Modifier(static_cast<Modifiers>(ConvertStrTo<int>(tokens[2])));
+
+ // Virtual Key code / Scan code
+ UINT vk = 0;
+ auto scPos = tokens[3].find('/');
+ if(scPos != std::string::npos)
+ {
+ // Scan code present
+ UINT sc = ConvertStrTo<UINT>(tokens[3].substr(scPos + 1));
+ for(auto i = layouts.begin(); i != layouts.end() && vk == 0; i++)
+ {
+ vk = MapVirtualKeyEx(sc, MAPVK_VSC_TO_VK, *i);
+ }
+ }
+ if(vk == 0)
+ {
+ vk = ConvertStrTo<UINT>(tokens[3]);
+ }
+ kc.KeyCode(vk);
+
+ // Event
+ kc.EventType(static_cast<KeyEventType>(ConvertStrTo<int>(tokens[4])));
+
+ if(fillExistingSet && cmd != kcNull && GetKeyListSize(cmd) != 0)
+ {
+ // Do not map shortcuts that already have custom keys assigned.
+ // In particular with default note keys, this can create awkward keymaps when loading
+ // e.g. an IT-style keymap and it contains two keys mapped to the same notes.
+ continue;
+ }
+ }
+
+ // Error checking
+ if(cmd < 0 || cmd >= kcNumCommands || kc.Context() >= kCtxMaxInputContexts || tokens.size() < 4)
+ {
+ errorCount++;
+ if (errorCount < 10)
+ {
+ if(tokens.size() < 4)
+ errText.AppendFormat(_T("Line %d was not understood.\n"), l);
+ else
+ errText.AppendFormat(_T("Line %d contained an unknown command.\n"), l);
+ } else if (errorCount == 10)
+ {
+ errText += _T("Too many errors detected, not reporting any more.\n");
+ }
+ } else
+ {
+ Add(kc, cmd, !fillExistingSet, -1, !fillExistingSet);
+ }
+ }
+
+ if(!fillExistingSet)
+ {
+ // Add the default command set to our freshly loaded command set.
+ std::istringstream ss{ GetDefaultKeymap() };
+ LoadFile(ss, mpt::ustring(), true);
+ } else
+ {
+ // We were just adding stuff to an existing command set - don't delete it!
+ return true;
+ }
+
+ // Fix up old keymaps containing legacy commands that have been merged into other commands
+ static constexpr std::pair<CommandID, CommandID> MergeCommands[] =
+ {
+ {kcFileSaveAsMP3, kcFileSaveAsWave},
+ {kcNoteCutOld, kcNoteCut},
+ {kcNoteOffOld, kcNoteOff},
+ {kcNoteFadeOld, kcNoteFade},
+ };
+ for(const auto [from, to] : MergeCommands)
+ {
+ m_commands[to].kcList.insert(m_commands[to].kcList.end(), m_commands[from].kcList.begin(), m_commands[from].kcList.end());
+ m_commands[from].kcList.clear();
+ }
+
+ if(!errText.IsEmpty())
+ {
+ Reporting::Warning(MPT_CFORMAT("The following problems have been encountered while trying to load the key binding file {}:\n{}")
+ (mpt::ToCString(filenameDescription), errText));
+ }
+
+ m_oldSpecs = nullptr;
+ return true;
+}
+
+
+bool CCommandSet::LoadFile(const mpt::PathString &filename)
+{
+ mpt::ifstream fin(filename);
+ if(fin.fail())
+ {
+ Reporting::Warning(MPT_TFORMAT("Can't open key bindings file {} for reading. Default key bindings will be used.")(filename));
+ return false;
+ } else
+ {
+ return LoadFile(fin, filename.ToUnicode());
+ }
+}
+
+
+bool CCommandSet::LoadDefaultKeymap()
+{
+ std::istringstream ss{ GetDefaultKeymap() };
+ return LoadFile(ss, U_("\"executable resource\""));
+}
+
+
+CommandID CCommandSet::FindCmd(uint32 uid) const
+{
+ for(int i = 0; i < kcNumCommands; i++)
+ {
+ if(m_commands[i].ID() == uid)
+ return static_cast<CommandID>(i);
+ }
+ return kcNull;
+}
+
+
+CString KeyCombination::GetContextText(InputTargetContext ctx)
+{
+ switch(ctx)
+ {
+ case kCtxAllContexts: return _T("Global Context");
+ case kCtxViewGeneral: return _T("General Context [bottom]");
+ case kCtxViewPatterns: return _T("Pattern Context [bottom]");
+ case kCtxViewPatternsNote: return _T("Pattern Context [bottom] - Note Col");
+ case kCtxViewPatternsIns: return _T("Pattern Context [bottom] - Ins Col");
+ case kCtxViewPatternsVol: return _T("Pattern Context [bottom] - Vol Col");
+ case kCtxViewPatternsFX: return _T("Pattern Context [bottom] - FX Col");
+ case kCtxViewPatternsFXparam: return _T("Pattern Context [bottom] - Param Col");
+ case kCtxViewSamples: return _T("Sample Context [bottom]");
+ case kCtxViewInstruments: return _T("Instrument Context [bottom]");
+ case kCtxViewComments: return _T("Comments Context [bottom]");
+ case kCtxCtrlGeneral: return _T("General Context [top]");
+ case kCtxCtrlPatterns: return _T("Pattern Context [top]");
+ case kCtxCtrlSamples: return _T("Sample Context [top]");
+ case kCtxCtrlInstruments: return _T("Instrument Context [top]");
+ case kCtxCtrlComments: return _T("Comments Context [top]");
+ case kCtxCtrlOrderlist: return _T("Orderlist");
+ case kCtxVSTGUI: return _T("Plugin GUI Context");
+ case kCtxChannelSettings: return _T("Quick Channel Settings Context");
+ case kCtxUnknownContext:
+ default: return _T("Unknown Context");
+ }
+}
+
+
+CString KeyCombination::GetKeyEventText(FlagSet<KeyEventType> event)
+{
+ CString text;
+
+ bool first = true;
+ if (event & kKeyEventDown)
+ {
+ first=false;
+ text.Append(_T("KeyDown"));
+ }
+ if (event & kKeyEventRepeat)
+ {
+ if (!first) text.Append(_T("|"));
+ text.Append(_T("KeyHold"));
+ first=false;
+ }
+ if (event & kKeyEventUp)
+ {
+ if (!first) text.Append(_T("|"));
+ text.Append(_T("KeyUp"));
+ }
+
+ return text;
+}
+
+
+CString KeyCombination::GetModifierText(FlagSet<Modifiers> mod)
+{
+ CString text;
+ if (mod[ModShift]) text.Append(_T("Shift+"));
+ if (mod[ModCtrl]) text.Append(_T("Ctrl+"));
+ if (mod[ModAlt]) text.Append(_T("Alt+"));
+ if (mod[ModRShift]) text.Append(_T("RShift+"));
+ if (mod[ModRCtrl]) text.Append(_T("RCtrl+"));
+ if (mod[ModRAlt]) text.Append(_T("RAlt+"));
+ if (mod[ModWin]) text.Append(_T("Win+")); // Feature: use Windows keys as modifier keys
+ if (mod[ModMidi]) text.Append(_T("MIDI"));
+ return text;
+}
+
+
+CString KeyCombination::GetKeyText(FlagSet<Modifiers> mod, UINT code)
+{
+ CString keyText = GetModifierText(mod);
+ if(mod[ModMidi])
+ {
+ if(code < 0x80)
+ keyText.AppendFormat(_T(" CC %u"), code);
+ else
+ keyText += MPT_CFORMAT(" {}{}")(mpt::ustring(NoteNamesSharp[(code & 0x7F) % 12]), (code & 0x7F) / 12);
+ } else
+ {
+ keyText.Append(CHotKeyCtrl::GetKeyName(code, IsExtended(code)));
+ }
+ //HACK:
+ if (keyText == _T("Ctrl+CTRL")) keyText = _T("Ctrl");
+ else if (keyText == _T("Alt+ALT")) keyText = _T("Alt");
+ else if (keyText == _T("Shift+SHIFT")) keyText = _T("Shift");
+ else if (keyText == _T("RCtrl+CTRL")) keyText = _T("RCtrl");
+ else if (keyText == _T("RAlt+ALT")) keyText = _T("RAlt");
+ else if (keyText == _T("RShift+SHIFT")) keyText = _T("RShift");
+
+ return keyText;
+}
+
+
+CString CCommandSet::GetKeyTextFromCommand(CommandID c, UINT key) const
+{
+ if (key < m_commands[c].kcList.size())
+ return m_commands[c].kcList[0].GetKeyText();
+ else
+ return CString();
+}
+
+
+// Quick Changes - modify many commands with one call.
+
+bool CCommandSet::QuickChange_NotesRepeat(bool repeat)
+{
+ for (CommandID cmd = kcVPStartNotes; cmd <= kcVPEndNotes; cmd=(CommandID)(cmd + 1)) //for all notes
+ {
+ for(auto &kc : m_commands[cmd].kcList)
+ {
+ if(repeat)
+ kc.EventType(kc.EventType() | kKeyEventRepeat);
+ else
+ kc.EventType(kc.EventType() & ~kKeyEventRepeat);
+ }
+ }
+ return true;
+}
+
+
+bool CCommandSet::QuickChange_SetEffects(const CModSpecifications &modSpecs)
+{
+ // Is this already the active key configuration?
+ if(&modSpecs == m_oldSpecs)
+ {
+ return false;
+ }
+ m_oldSpecs = &modSpecs;
+
+ int choices = 0;
+ KeyCombination kc(kCtxViewPatternsFX, ModNone, 0, kKeyEventDown | kKeyEventRepeat);
+
+ for(CommandID cmd = kcFixedFXStart; cmd <= kcFixedFXend; cmd = static_cast<CommandID>(cmd + 1))
+ {
+ // Remove all old choices
+ choices = GetKeyListSize(cmd);
+ for(int p = choices; p >= 0; --p)
+ {
+ Remove(p, cmd);
+ }
+
+ char effect = modSpecs.GetEffectLetter(static_cast<ModCommand::COMMAND>(cmd - kcSetFXStart + 1));
+ if(effect >= 'A' && effect <= 'Z')
+ {
+ // VkKeyScanEx needs lowercase letters
+ effect = effect - 'A' + 'a';
+ } else if(effect < '0' || effect > '9')
+ {
+ // Don't map effects that use "weird" effect letters (such as # or \)
+ effect = '?';
+ }
+
+ if(effect != '?')
+ {
+ // Hack for situations where a non-latin keyboard layout without A...Z key code mapping may the current layout (e.g. Russian),
+ // but a latin layout (e.g. EN-US) is installed as well.
+ std::vector<HKL> layouts(GetKeyboardLayoutList(0, nullptr));
+ GetKeyboardLayoutList(static_cast<int>(layouts.size()), layouts.data());
+ SHORT codeNmod = -1;
+ for(auto i = layouts.begin(); i != layouts.end() && codeNmod == -1; i++)
+ {
+ codeNmod = VkKeyScanEx(effect, *i);
+ }
+ if(codeNmod != -1)
+ {
+ kc.KeyCode(LOBYTE(codeNmod));
+ // Don't add modifier keys, since on French keyboards, numbers are input using Shift.
+ // We don't really want that behaviour here, and I'm sure we don't want that in other cases on other layouts as well.
+ kc.Modifier(ModNone);
+ Add(kc, cmd, true);
+ }
+
+ if (effect >= '0' && effect <= '9') // For numbers, ensure numpad works too
+ {
+ kc.KeyCode(VK_NUMPAD0 + (effect - '0'));
+ Add(kc, cmd, true);
+ }
+ }
+ }
+
+ return true;
+}
+
+// Stupid MFC crap: for some reason VK code isn't enough to get correct string with GetKeyName.
+// We also need to figure out the correct "extended" bit.
+bool KeyCombination::IsExtended(UINT code)
+{
+ if (code==VK_SNAPSHOT) //print screen
+ return true;
+ if (code>=VK_PRIOR && code<=VK_DOWN) //pgup, pg down, home, end, cursor keys,
+ return true;
+ if (code>=VK_INSERT && code<=VK_DELETE) // ins, del
+ return true;
+ if (code>=VK_LWIN && code<=VK_APPS) //winkeys & application key
+ return true;
+ if (code==VK_DIVIDE) //Numpad '/'
+ return true;
+ if (code==VK_NUMLOCK) //print screen
+ return true;
+ if (code>=0xA0 && code<=0xA5) //attempt for RL mods
+ return true;
+
+ return false;
+}
+
+
+void CCommandSet::SetupContextHierarchy()
+{
+ // For now much be fully expanded (i.e. don't rely on grandparent relationships).
+ m_isParentContext[kCtxAllContexts].set(kCtxViewGeneral);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewPatterns);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewPatternsNote);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewPatternsIns);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewPatternsVol);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewPatternsFX);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewPatternsFXparam);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewSamples);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewInstruments);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewComments);
+ m_isParentContext[kCtxAllContexts].set(kCtxViewTree);
+ m_isParentContext[kCtxAllContexts].set(kCtxInsNoteMap);
+ m_isParentContext[kCtxAllContexts].set(kCtxVSTGUI);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlGeneral);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlPatterns);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlSamples);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlInstruments);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlComments);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlSamples);
+ m_isParentContext[kCtxAllContexts].set(kCtxCtrlOrderlist);
+ m_isParentContext[kCtxAllContexts].set(kCtxChannelSettings);
+
+ m_isParentContext[kCtxViewPatterns].set(kCtxViewPatternsNote);
+ m_isParentContext[kCtxViewPatterns].set(kCtxViewPatternsIns);
+ m_isParentContext[kCtxViewPatterns].set(kCtxViewPatternsVol);
+ m_isParentContext[kCtxViewPatterns].set(kCtxViewPatternsFX);
+ m_isParentContext[kCtxViewPatterns].set(kCtxViewPatternsFXparam);
+ m_isParentContext[kCtxCtrlPatterns].set(kCtxCtrlOrderlist);
+
+}
+
+
+bool CCommandSet::KeyCombinationConflict(KeyCombination kc1, KeyCombination kc2, bool checkEventConflict) const
+{
+ bool modConflict = (kc1.Modifier()==kc2.Modifier());
+ bool codeConflict = (kc1.KeyCode()==kc2.KeyCode());
+ bool eventConflict = ((kc1.EventType()&kc2.EventType()));
+ bool ctxConflict = (kc1.Context() == kc2.Context());
+ bool crossCxtConflict = IsCrossContextConflict(kc1, kc2);
+
+ bool conflict = modConflict && codeConflict && (eventConflict || !checkEventConflict) &&
+ (ctxConflict || crossCxtConflict);
+
+ return conflict;
+}
+
+
+bool CCommandSet::IsCrossContextConflict(KeyCombination kc1, KeyCombination kc2) const
+{
+ return m_isParentContext[kc1.Context()][kc2.Context()] || m_isParentContext[kc2.Context()][kc1.Context()];
+}
+
+OPENMPT_NAMESPACE_END