/*
 * 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