/********************************************************************** Audacity: A Digital Audio Editor KeyConfigPrefs.cpp Brian Gunlogson Dominic Mazzoni James Crook *******************************************************************//*! \class KeyConfigPrefs \brief A PrefsPanel for keybindings. The code for displaying keybindings is similar to code in MousePrefs. It would be nice to create a new 'Bindings' class which both KeyConfigPrefs and MousePrefs use. *//*********************************************************************/ #include "../Audacity.h" #include #include #include #include #include #include "../Prefs.h" #include "../Project.h" #include "../commands/CommandManager.h" #include "../commands/Keyboard.h" #include "../xml/XMLFileReader.h" #include "../Internat.h" #include "../ShuttleGui.h" #include "KeyConfigPrefs.h" #include "FileDialog.h" #if defined(EXPERIMENTAL_KEY_VIEW) #include "../widgets/KeyView.h" // // KeyConfigPrefs // #define AssignDefaultsButtonID 17001 #define CurrentComboID 17002 #define SetButtonID 17003 #define ClearButtonID 17004 #define CommandsListID 17005 #define ExportButtonID 17006 #define ImportButtonID 17007 #define FilterID 17008 #define ViewByTreeID 17009 #define ViewByNameID 17010 #define ViewByKeyID 17011 #define FilterTimerID 17012 BEGIN_EVENT_TABLE(KeyConfigPrefs, PrefsPanel) EVT_BUTTON(AssignDefaultsButtonID, KeyConfigPrefs::OnDefaults) EVT_BUTTON(SetButtonID, KeyConfigPrefs::OnSet) EVT_BUTTON(ClearButtonID, KeyConfigPrefs::OnClear) EVT_BUTTON(ExportButtonID, KeyConfigPrefs::OnExport) EVT_BUTTON(ImportButtonID, KeyConfigPrefs::OnImport) EVT_LISTBOX(CommandsListID, KeyConfigPrefs::OnSelected) EVT_RADIOBUTTON(ViewByTreeID, KeyConfigPrefs::OnViewBy) EVT_RADIOBUTTON(ViewByNameID, KeyConfigPrefs::OnViewBy) EVT_RADIOBUTTON(ViewByKeyID, KeyConfigPrefs::OnViewBy) EVT_TIMER(FilterTimerID, KeyConfigPrefs::OnFilterTimer) END_EVENT_TABLE() KeyConfigPrefs::KeyConfigPrefs(wxWindow * parent) : PrefsPanel(parent, _("Keyboard")), mView(NULL), mFilter(NULL), mKey(NULL), mFilterTimer(this, FilterTimerID), mFilterPending(false) { Populate(); } KeyConfigPrefs::~KeyConfigPrefs() { if (mKey) { mKey->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(KeyConfigPrefs::OnHotkeyKeyDown), NULL, this); mKey->Disconnect(wxEVT_CHAR, wxKeyEventHandler(KeyConfigPrefs::OnHotkeyChar), NULL, this); mKey->Disconnect(wxEVT_KILL_FOCUS, wxFocusEventHandler(KeyConfigPrefs::OnHotkeyKillFocus), NULL, this); } if (mFilter) { mKey->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(KeyConfigPrefs::OnFilterKeyDown), NULL, this); mKey->Disconnect(wxEVT_CHAR, wxKeyEventHandler(KeyConfigPrefs::OnFilterChar), NULL, this); } } void KeyConfigPrefs::Populate() { ShuttleGui S(this, eIsCreatingFromPrefs); AudacityProject *project = GetActiveProject(); if (!project) { S.StartVerticalLay(true); { S.StartStatic(wxEmptyString, true); { S.AddTitle(_("Keyboard preferences currently unavailable.")); S.AddTitle(_("Open a new project to modify keyboard shortcuts.")); } S.EndStatic(); } S.EndVerticalLay(); return; } PopulateOrExchange(S); mCommandSelected = wxNOT_FOUND; mManager = project->GetCommandManager(); RefreshBindings(); if (mViewByTree->GetValue()) { mViewType = ViewByTree; } else if (mViewByName->GetValue()) { mViewType = ViewByName; } else if (mViewByKey->GetValue()) { mViewType = ViewByKey; mFilterLabel->SetLabel(_("&Hotkey:")); mFilter->SetName(wxStripMenuCodes(mFilterLabel->GetLabel())); } mView->SetView(mViewType); } /// Normally in classes derived from PrefsPanel this function /// is used both to populate the panel and to exchange data with it. /// With KeyConfigPrefs all the exchanges are handled specially, /// so this is only used in populating the panel. void KeyConfigPrefs::PopulateOrExchange(ShuttleGui & S) { S.SetBorder(2); S.StartStatic(_("Key Bindings"), 1); { S.StartMultiColumn(3, wxEXPAND); { S.SetStretchyCol(1); S.StartHorizontalLay(wxALIGN_LEFT|wxALIGN_CENTER_VERTICAL, 0); { S.AddTitle(_("View by:")); S.StartRadioButtonGroup(wxT("/Prefs/KeyConfig/ViewBy"), wxT("tree")); { mViewByTree = S.Id(ViewByTreeID).TieRadioButton(_("&Tree"), wxT("tree")); mViewByTree->SetName(_("View by tree")); mViewByName = S.Id(ViewByNameID).TieRadioButton(_("&Name"), wxT("name")); mViewByName->SetName(_("View by name")); mViewByKey = S.Id(ViewByKeyID).TieRadioButton(_("&Key"), wxT("key")); mViewByKey->SetName(_("View by key")); } S.EndRadioButtonGroup(); } S.EndHorizontalLay(); S.StartHorizontalLay(wxALIGN_CENTER|wxALIGN_CENTER_VERTICAL, 0); { // just a spacer } S.EndHorizontalLay(); S.StartHorizontalLay(wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL, 0); { mFilterLabel = S.AddVariableText(_("Searc&h:")); if (!mFilter) { mFilter = new wxTextCtrl(this, FilterID, wxT(""), wxDefaultPosition, #if defined(__WXMAC__) wxSize(300, -1), #else wxSize(210, -1), #endif wxTE_PROCESS_ENTER); mFilter->SetName(wxStripMenuCodes(mFilterLabel->GetLabel())); mFilter->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(KeyConfigPrefs::OnFilterKeyDown), NULL, this); mFilter->Connect(wxEVT_CHAR, wxKeyEventHandler(KeyConfigPrefs::OnFilterChar), NULL, this); } S.AddWindow(mFilter, wxALIGN_RIGHT); } S.EndHorizontalLay(); } S.EndThreeColumn(); S.AddSpace(-1, 2); S.StartHorizontalLay(wxEXPAND, 1); { if (!mView) { mView = new KeyView(this, CommandsListID); mView->SetName(_("Bindings")); } S.Prop(true); S.AddWindow(mView, wxEXPAND); } S.EndHorizontalLay(); S.StartThreeColumn(); { if (!mKey) { mKey = new wxTextCtrl(this, CurrentComboID, wxT(""), wxDefaultPosition, #if defined(__WXMAC__) wxSize(300, -1), #else wxSize(210, -1), #endif wxTE_PROCESS_ENTER); mKey->SetName(_("Short cut")); mKey->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(KeyConfigPrefs::OnHotkeyKeyDown), NULL, this); mKey->Connect(wxEVT_CHAR, wxKeyEventHandler(KeyConfigPrefs::OnHotkeyChar), NULL, this); mKey->Connect(wxEVT_KILL_FOCUS, wxFocusEventHandler(KeyConfigPrefs::OnHotkeyKillFocus), NULL, this); } S.AddWindow(mKey); /* i18n-hint: (verb)*/ mSet = S.Id(SetButtonID).AddButton(_("&Set")); mClear = S.Id(ClearButtonID).AddButton(_("Cl&ear")); } S.EndThreeColumn(); #if defined(__WXMAC__) S.AddFixedText(_("Note: Pressing Cmd+Q will quit. All other keys are valid.")); #endif S.StartThreeColumn(); { S.Id(ImportButtonID).AddButton(_("&Import...")); S.Id(ExportButtonID).AddButton(_("&Export...")); S.Id(AssignDefaultsButtonID).AddButton(_("&Defaults")); } S.EndThreeColumn(); } S.EndStatic(); // Need to layout so that the KeyView is properly sized before populating. // Otherwise, the initial selection is not scrolled into view. Layout(); } void KeyConfigPrefs::RefreshBindings() { wxArrayString Labels; wxArrayString Categories; wxArrayString Prefixes; mNames.Clear(); mKeys.Clear(); mDefaultKeys.Clear(); mManager->GetAllCommandData( mNames, mKeys, mDefaultKeys, Labels, Categories, Prefixes, true); // True to include effects (list items), false otherwise. mView->RefreshBindings(mNames, Categories, Prefixes, Labels, mKeys); mView->ExpandAll(); mNewKeys = mKeys; } void KeyConfigPrefs::OnImport(wxCommandEvent & WXUNUSED(event)) { wxString file = wxT("Audacity-keys.xml"); wxString path = gPrefs->Read(wxT("/DefaultOpenPath"), ::wxGetCwd()); file = FileSelector(_("Select an XML file containing Audacity keyboard shortcuts..."), path, file, wxT(""), _("XML files (*.xml)|*.xml|All files|*"), wxRESIZE_BORDER, this); if (!file) { return; } path = wxPathOnly(file); gPrefs->Write(wxT("/DefaultOpenPath"), path); gPrefs->Flush(); XMLFileReader reader; if (!reader.Parse(mManager, file)) { wxMessageBox(reader.GetErrorStr(), _("Error Importing Keyboard Shortcuts"), wxOK | wxCENTRE, this); } RefreshBindings(); } void KeyConfigPrefs::OnExport(wxCommandEvent & WXUNUSED(event)) { wxString file = wxT("Audacity-keys.xml"); wxString path = gPrefs->Read(wxT("/DefaultExportPath"), ::wxGetCwd()); file = FileSelector(_("Export Keyboard Shortcuts As:"), path, file, wxT("xml"), _("XML files (*.xml)|*.xml|All files|*"), wxFD_SAVE | wxFD_OVERWRITE_PROMPT | wxRESIZE_BORDER, this); if (!file) { return; } path = wxPathOnly(file); gPrefs->Write(wxT("/DefaultExportPath"), path); gPrefs->Flush(); XMLFileWriter prefFile; try { prefFile.Open(file, wxT("wb")); mManager->WriteXML(prefFile); prefFile.Close(); } catch (XMLFileWriterException* pException) { wxMessageBox(_("Couldn't write to file: ") + file, _("Error Exporting Keyboard Shortcuts"), wxOK | wxCENTRE, this); delete pException; } } void KeyConfigPrefs::OnDefaults(wxCommandEvent & WXUNUSED(event)) { mNewKeys = mDefaultKeys; for (size_t i = 0; i < mNewKeys.GetCount(); i++) { mManager->SetKeyFromIndex(i, mNewKeys[i]); } RefreshBindings(); } void KeyConfigPrefs::OnHotkeyKeyDown(wxKeyEvent & e) { wxTextCtrl *t = (wxTextCtrl *)e.GetEventObject(); // Make sure we can navigate away from the hotkey textctrl. // On Linux and OSX, it an get stuck, but it doesn't hurt // to do it for Windows as well. // // Mac note: Don't waste time trying to figure out why the // focus goes back to the prefs tree. Unless Voiceover is // active, buttons on the Mac do not accept focus and all the // controls between this one and the tree control are buttons. if (e.GetKeyCode() == WXK_TAB) { wxNavigationKeyEvent nevent; nevent.SetWindowChange(e.ControlDown()); nevent.SetDirection(!e.ShiftDown()); nevent.SetEventObject(t); nevent.SetCurrentFocus(t); t->GetParent()->GetEventHandler()->ProcessEvent(nevent); return; } t->SetValue(KeyStringDisplay(KeyEventToKeyString(e))); } void KeyConfigPrefs::OnHotkeyChar(wxKeyEvent & WXUNUSED(e)) { // event.Skip() not performed, so event will not be processed further. } void KeyConfigPrefs::OnHotkeyKillFocus(wxFocusEvent & e) { if (mKey->GetValue().IsEmpty() && mCommandSelected != wxNOT_FOUND) { mKey->AppendText(mView->GetKey(mCommandSelected)); } e.Skip(); } void KeyConfigPrefs::OnFilterTimer(wxTimerEvent & WXUNUSED(e)) { // The filter timer has expired, so set the filter if (mFilterPending) { // Do not reset mFilterPending here...possible race mView->SetFilter(mFilter->GetValue()); } } void KeyConfigPrefs::OnFilterKeyDown(wxKeyEvent & e) { wxTextCtrl *t = (wxTextCtrl *)e.GetEventObject(); int keycode = e.GetKeyCode(); // Make sure we can navigate away from the hotkey textctrl. // On Linux and OSX, it an get stuck, but it doesn't hurt // to do it for Windows as well. if (keycode == WXK_TAB) { wxNavigationKeyEvent nevent; nevent.SetWindowChange(e.ControlDown()); nevent.SetDirection(!e.ShiftDown()); nevent.SetEventObject(t); nevent.SetCurrentFocus(t); t->GetParent()->GetEventHandler()->ProcessEvent(nevent); return; } if (mViewType == ViewByKey) { wxString key = KeyStringDisplay(KeyEventToKeyString(e)); t->SetValue(key); if (key != wxEmptyString) { mView->SetFilter(t->GetValue()); } } else { if (keycode == WXK_RETURN) { mFilterPending = false; mView->SetFilter(t->GetValue()); } else { mFilterPending = true; mFilterTimer.Start(500, wxTIMER_ONE_SHOT); e.Skip(); } } } void KeyConfigPrefs::OnFilterChar(wxKeyEvent & e) { if (mViewType != ViewByKey) { e.Skip(); } } // Given a hotkey combination, returns the name (description) of the // corresponding command, or the empty string if none is found. wxString KeyConfigPrefs::NameFromKey(const wxString & key) { return mView->GetNameByKey(key); } // Sets the selected command to have this key // This is not yet a committed change, which will happen on a save. void KeyConfigPrefs::SetKeyForSelected(const wxString & key) { wxString name = mView->GetName(mCommandSelected); if (!mView->CanSetKey(mCommandSelected)) { wxMessageBox(_("You may not assign a key to this entry"), _("Error"), wxICON_ERROR | wxCENTRE, this); return; } mView->SetKey(mCommandSelected, key); mManager->SetKeyFromName(name, key); mNewKeys[mNames.Index(name)] = key; } void KeyConfigPrefs::OnSet(wxCommandEvent & WXUNUSED(event)) { if (mCommandSelected == wxNOT_FOUND) { wxMessageBox(_("You must select a binding before assigning a shortcut"), _("Error"), wxICON_WARNING | wxCENTRE, this); return; } wxString key = mKey->GetValue(); wxString oldname = mView->GetNameByKey(key); wxString newname = mView->GetName(mCommandSelected); // Just ignore it if they are the same if (oldname == newname) { return; } // Prevent same hotkey combination being used twice. if (!oldname.IsEmpty()) { wxString oldlabel = mManager->GetCategoryFromName(oldname) + wxT(" - ") + mManager->GetPrefixedLabelFromName(oldname); wxString newlabel = mManager->GetCategoryFromName(newname) + wxT(" - ") + mManager->GetPrefixedLabelFromName(newname); if (wxMessageBox( wxString::Format( _("The keyboard shortcut '%s' is already assigned to:\n\n\t'%s'\n\nClick OK to assign the shortcut to\n\n\t'%s'\n\ninstead. Otherwise, click Cancel."), key.c_str(), oldlabel.c_str(), newlabel.c_str()), _("Error"), wxOK | wxCANCEL | wxICON_STOP | wxCENTRE, this) == wxCANCEL) { return; } mView->SetKeyByName(oldname, wxEmptyString); mManager->SetKeyFromName(oldname, wxEmptyString); mNewKeys[mNames.Index(oldname)].Empty(); } SetKeyForSelected(key); } void KeyConfigPrefs::OnClear(wxCommandEvent& WXUNUSED(event)) { mKey->Clear(); if (mCommandSelected != wxNOT_FOUND) { SetKeyForSelected(wxEmptyString); } } void KeyConfigPrefs::OnSelected(wxCommandEvent & WXUNUSED(e)) { mCommandSelected = mView->GetSelected(); mKey->Clear(); if (mCommandSelected != wxNOT_FOUND) { bool canset = mView->CanSetKey(mCommandSelected); if (canset) { mKey->AppendText(mView->GetKey(mCommandSelected)); } mKey->Enable(canset); mSet->Enable(canset); mClear->Enable(canset); } } void KeyConfigPrefs::OnViewBy(wxCommandEvent & e) { switch (e.GetId()) { case ViewByTreeID: mViewType = ViewByTree; mFilterLabel->SetLabel(_("Searc&h:")); break; case ViewByNameID: mViewType = ViewByName; mFilterLabel->SetLabel(_("Searc&h:")); break; case ViewByKeyID: mViewType = ViewByKey; mFilterLabel->SetLabel(_("&Hotkey:")); break; } mView->SetView(mViewType); mFilter->SetName(wxStripMenuCodes(mFilterLabel->GetLabel())); } bool KeyConfigPrefs::Apply() { // On the Mac, preferences may be changed without any active // projects. This means that the CommandManager isn't availabe // either. So we can't attempt to save preferences, otherwise // NULL ptr dereferences will happen in ShuttleGui because the // radio buttons are never created. (See Populate() above.) if (!GetActiveProject()) { return true; } ShuttleGui S(this, eIsSavingToPrefs); PopulateOrExchange(S); for (size_t i = 0; i < mNames.GetCount(); i++) { wxString dkey = KeyStringNormalize(mDefaultKeys[i]); wxString name = wxT("/NewKeys/") + mNames[i]; wxString key = KeyStringNormalize(mNewKeys[i]); if (gPrefs->HasEntry(name)) { if (key != KeyStringNormalize(gPrefs->Read(name, key))) { gPrefs->Write(name, key); } if (key == dkey) { gPrefs->DeleteEntry(name); } } else { if (key != dkey) { gPrefs->Write(name, key); } } } return gPrefs->Flush(); } void KeyConfigPrefs::Cancel() { // Restore original key values for (size_t i = 0; i < mNames.GetCount(); i++) { mManager->SetKeyFromIndex(i, mKeys[i]); } return; } #else // // KeyConfigPrefs // #define AssignDefaultsButtonID 17001 #define CurrentComboID 17002 #define SetButtonID 17003 #define ClearButtonID 17004 #define CommandsListID 17005 #define ExportButtonID 17006 #define ImportButtonID 17007 #define CategoryID 17008 // The numbers of the columns of the mList. enum { CommandColumn, KeyComboColumn }; BEGIN_EVENT_TABLE(KeyConfigPrefs, PrefsPanel) EVT_BUTTON(AssignDefaultsButtonID, KeyConfigPrefs::OnDefaults) EVT_BUTTON(SetButtonID, KeyConfigPrefs::OnSet) EVT_BUTTON(ClearButtonID, KeyConfigPrefs::OnClear) EVT_BUTTON(ExportButtonID, KeyConfigPrefs::OnExport) EVT_BUTTON(ImportButtonID, KeyConfigPrefs::OnImport) EVT_CHOICE(CategoryID, KeyConfigPrefs::OnCategory) EVT_LIST_ITEM_SELECTED(CommandsListID, KeyConfigPrefs::OnItemSelected) EVT_LIST_KEY_DOWN(CommandsListID, KeyConfigPrefs::OnKeyDown) END_EVENT_TABLE() KeyConfigPrefs::KeyConfigPrefs(wxWindow * parent) : PrefsPanel(parent, _("Keyboard")), mKey(NULL) { Populate(); } KeyConfigPrefs::~KeyConfigPrefs() { if (mKey) { mKey->Disconnect(wxEVT_KEY_DOWN, wxKeyEventHandler(KeyConfigPrefs::OnCaptureKeyDown)); mKey->Disconnect(wxEVT_CHAR, wxKeyEventHandler(KeyConfigPrefs::OnCaptureChar)); } } void KeyConfigPrefs::Populate() { ShuttleGui S(this, eIsCreatingFromPrefs); AudacityProject *project = GetActiveProject(); if (!project) { S.StartVerticalLay(true); { S.StartStatic(wxEmptyString, true); { S.AddTitle(_("Keyboard preferences currently unavailable.")); S.AddTitle(_("Open a new project to modify keyboard shortcuts.")); } S.EndStatic(); } S.EndVerticalLay(); return; } mManager = project->GetCommandManager(); mManager->GetCategories(mCats); mCats.Insert(_("All"), 0); PopulateOrExchange(S); CreateList(); mCommandSelected = -1; } /// Normally in classes derived from PrefsPanel this function /// is used both to populate the panel and to exchange data with it. /// With KeyConfigPrefs all the exchanges are handled specially, /// so this is only used in populating the panel. void KeyConfigPrefs::PopulateOrExchange(ShuttleGui & S) { S.SetBorder(2); S.StartStatic(_("Key Bindings"), 1); { S.StartHorizontalLay(wxALIGN_CENTRE, false); { S.Id(CategoryID); mCat = S.AddChoice(_("C&ategory:"), mCats[0], &mCats); } S.EndHorizontalLay(); mList = S.Id(CommandsListID).AddListControlReportMode(); mList->SetName(_("Key Bindings")); S.StartThreeColumn(); { if (!mKey) { mKey = new wxTextCtrl(this, CurrentComboID, wxT(""), wxDefaultPosition, #if defined(__WXMAC__) wxSize(300, -1)); #else wxSize(210, -1)); #endif mKey->Connect(wxEVT_KEY_DOWN, wxKeyEventHandler(KeyConfigPrefs::OnCaptureKeyDown)); mKey->Connect(wxEVT_CHAR, wxKeyEventHandler(KeyConfigPrefs::OnCaptureChar)); } S.AddWindow(mKey); /* i18n-hint: (verb)*/ S.Id(SetButtonID).AddButton(_("Set")); S.Id(ClearButtonID).AddButton(_("Cl&ear")); } S.EndThreeColumn(); #if defined(__WXMAC__) S.AddFixedText(_("Note: Pressing Cmd+Q will quit. All other keys are valid.")); #endif S.StartThreeColumn(); { S.Id(ImportButtonID).AddButton(_("&Import...")); S.Id(ExportButtonID).AddButton(_("&Export...")); S.Id(AssignDefaultsButtonID).AddButton(_("&Defaults")); } S.EndThreeColumn(); } S.EndStatic(); } /// Sets up mList with the right number of columns, titles, /// fills the contents and sets column widths. void KeyConfigPrefs::CreateList() { mList->InsertColumn(CommandColumn, _("Command"), wxLIST_FORMAT_LEFT); mList->InsertColumn(KeyComboColumn, _("Key Combination"), wxLIST_FORMAT_LEFT); RepopulateBindingsList(); mList->SetColumnWidth(CommandColumn, wxLIST_AUTOSIZE); mList->SetColumnWidth(KeyComboColumn, 250); } static int wxCALLBACK SortCallback(long item1, long item2, long sortData) { wxArrayString *names = (wxArrayString *) sortData; if (names->Item(item1) < names->Item(item2)) { return -1; } if (names->Item(item1) > names->Item(item2)) { return 1; } return 0; } void KeyConfigPrefs::RepopulateBindingsList() { wxString cat = mCat->GetStringSelection(); mList->DeleteAllItems(); // Delete contents, but not the column headers. mNames.Clear(); mDefaultKeys.Clear(); wxArrayString Keys,Labels,Categories; mManager->GetAllCommandData( mNames, Keys, mDefaultKeys, Labels, Categories, // True to include effects (list items), false otherwise. true ); bool save = (mKeys.GetCount() == 0); size_t ndx = 0; int color = 0; for (size_t i = 0; i < mNames.GetCount(); i++) { wxString name = mNames[i]; wxString key = KeyStringDisplay(Keys[i]); // Save the original key value to support canceling if (save) { mKeys.Add(key); // mNewKeys is what mKeys will change to. mNewKeys.Add(key); } else mNewKeys[i] = key; // Make sure mNewKeys is updated. // if (cat != _("All") && ! Categories[i].StartsWith(cat)) { if (cat != _("All") && ! (Categories[i]== cat)) { continue; } wxString label; // Labels for undo and redo change according to the last command // which can be undone/redone, so give them a special check in order // not to confuse users if (name == wxT("Undo")) { label = _("Undo"); } else if (name == wxT("Redo")) { label = _("Redo"); } else { label = mManager->GetPrefixedLabelFromName(name); } label = wxMenuItem::GetLabelFromText(label.BeforeFirst(wxT('\t'))); mList->InsertItem(ndx, label); mList->SetItem(ndx, KeyComboColumn, key); mList->SetItemData(ndx, i); mList->SetItemBackgroundColour(ndx, color ? wxColour(240, 240, 240) : *wxWHITE); color = 1 - color; ndx++; } // mList->SortItems(SortCallback, (long) &mNames); } void KeyConfigPrefs::OnImport(wxCommandEvent & WXUNUSED(event)) { wxString file = wxT("Audacity-keys.xml"); wxString path = gPrefs->Read(wxT("/DefaultOpenPath"), ::wxGetCwd()); file = FileSelector(_("Select an XML file containing Audacity keyboard shortcuts..."), path, file, wxT(""), _("XML files (*.xml)|*.xml|All files|*"), wxRESIZE_BORDER, this); if (!file) { return; } path = wxPathOnly(file); gPrefs->Write(wxT("/DefaultOpenPath"), path); gPrefs->Flush(); XMLFileReader reader; if (!reader.Parse(mManager, file)) { wxMessageBox(reader.GetErrorStr(), _("Error Importing Keyboard Shortcuts"), wxOK | wxCENTRE, this); } RepopulateBindingsList(); } void KeyConfigPrefs::OnExport(wxCommandEvent & WXUNUSED(event)) { wxString file = wxT("Audacity-keys.xml"); wxString path = gPrefs->Read(wxT("/DefaultExportPath"), ::wxGetCwd()); file = FileSelector(_("Export Keyboard Shortcuts As:"), path, file, wxT("xml"), _("XML files (*.xml)|*.xml|All files|*"), wxFD_SAVE | wxFD_OVERWRITE_PROMPT | wxRESIZE_BORDER, this); if (!file) { return; } path = wxPathOnly(file); gPrefs->Write(wxT("/DefaultExportPath"), path); gPrefs->Flush(); XMLFileWriter prefFile; try { prefFile.Open(file, wxT("wb")); mManager->WriteXML(prefFile); prefFile.Close(); } catch (XMLFileWriterException* pException) { wxMessageBox(_("Couldn't write to file: ") + file, _("Error Exporting Keyboard Shortcuts"), wxOK | wxCENTRE, this); delete pException; } } void KeyConfigPrefs::OnDefaults(wxCommandEvent & WXUNUSED(event)) { for (size_t i = 0; i < mNames.GetCount(); i++) { mManager->SetKeyFromIndex(i,mDefaultKeys[i]); mNewKeys[i]=mDefaultKeys[i]; } RepopulateBindingsList(); } void KeyConfigPrefs::OnCaptureKeyDown(wxKeyEvent & e) { wxTextCtrl *t = (wxTextCtrl *)e.GetEventObject(); #if defined(__WXMAC__) || defined(__WXGTK__) if (e.GetKeyCode() == WXK_TAB) { wxNavigationKeyEvent nevent; nevent.SetWindowChange(e.ControlDown()); nevent.SetDirection(!e.ShiftDown()); nevent.SetEventObject(t); nevent.SetCurrentFocus(t); t->GetParent()->ProcessEvent(nevent); return; } #endif t->SetValue(KeyStringDisplay(KeyEventToKeyString(e))); } void KeyConfigPrefs::OnCaptureChar(wxKeyEvent & WXUNUSED(event)) { } // Given a hotkey combination, returns the name (description) of the // corresponding command, or the empty string if none is found. wxString KeyConfigPrefs::NameFromKey( const wxString & key ) { int i; i=mNewKeys.Index( key ); if( i== wxNOT_FOUND ) return wxT(""); return mNames[i]; } // Sets the selected command to have this key // This is not yet a committed change, which will happen on a save. void KeyConfigPrefs::SetKeyForSelected( const wxString & key ) { int i = mList->GetItemData(mCommandSelected); wxString name = mNames[i]; mList->SetItem(mCommandSelected, KeyComboColumn, key); mManager->SetKeyFromIndex(i, key); #if 0 int i=mNames.Index( name ); if( i!=wxNOT_FOUND ) mNewKeys[i]=key; #endif mNewKeys[i]=key; } void KeyConfigPrefs::OnSet(wxCommandEvent & WXUNUSED(event)) { if ( mCommandSelected >= (int)mNames.GetCount()) return; wxString newKey = mKey->GetValue(); wxString alreadyAssignedName = NameFromKey( newKey ); // Prevent same hotkey combination being used twice. if( !alreadyAssignedName.IsEmpty() ) { wxMessageBox( wxString::Format( _("The keyboard shortcut '%s' is already assigned to:\n\n'%s'"), newKey.c_str(), alreadyAssignedName.c_str()), _("Error"), wxICON_STOP | wxCENTRE, this); return; } SetKeyForSelected( newKey ); } void KeyConfigPrefs::OnClear(wxCommandEvent& WXUNUSED(event)) { mKey->Clear(); if (mCommandSelected < 0 || mCommandSelected >= (int)mNames.GetCount()) { return; } SetKeyForSelected( wxT("") ); } void KeyConfigPrefs::OnKeyDown(wxListEvent & e) { // the code in this function allows the user to seek to the next // command which begins with the letter that is pressed #ifdef __WXMAC__ // I (Ed) have no way of telling what code will work on // the Mac but the following code does not return; #endif #ifdef __WXMSW__ // Windows seems to have this built-in // and does not need the following code return; #else // The following code seems to work well on at least some versions of Linux int keycode = e.GetKeyCode(); int selected = mList->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED); int cnt = mList->GetItemCount(); wxListItem item; bool found = false; item.SetColumn(CommandColumn); item.SetMask(wxLIST_MASK_TEXT); for (int i = selected + 1; i < cnt; i++) { item.SetId(i); mList->GetItem(item); if (item.GetText().Left(1).IsSameAs(keycode, false)) { mList->SetItemState(e.GetIndex(), 0, wxLIST_STATE_FOCUSED | wxLIST_STATE_SELECTED); mList->SetItemState(i, wxLIST_STATE_FOCUSED | wxLIST_STATE_SELECTED, wxLIST_STATE_FOCUSED | wxLIST_STATE_SELECTED); mList->EnsureVisible(i); found = true; break; } } if (!found) { for (int i = 0; i < selected; i++) { item.SetId(i); mList->GetItem(item); if (item.GetText().Left(1).IsSameAs(keycode, false)) { mList->SetItemState(e.GetIndex(), 0, wxLIST_STATE_FOCUSED | wxLIST_STATE_SELECTED); mList->SetItemState(i, wxLIST_STATE_FOCUSED | wxLIST_STATE_SELECTED, wxLIST_STATE_FOCUSED | wxLIST_STATE_SELECTED); mList->EnsureVisible(i); break; } } } #endif } void KeyConfigPrefs::OnCategory(wxCommandEvent & WXUNUSED(event)) { RepopulateBindingsList(); } void KeyConfigPrefs::OnItemSelected(wxListEvent & e) { mCommandSelected = e.GetIndex(); if (mCommandSelected < 0 || mCommandSelected >= (int)mNames.GetCount()) { mKey->SetLabel(wxT("")); return; } wxListItem item; item.SetColumn(KeyComboColumn); item.SetMask(wxLIST_MASK_TEXT); item.SetId(mCommandSelected); mList->GetItem(item); mKey->Clear(); mKey->AppendText(item.GetText()); } bool KeyConfigPrefs::Apply() { for (size_t i = 0; i < mNames.GetCount(); i++) { // wxString dkey = KeyStringNormalize(mManager->GetDefaultKeyFromName(mNames[i])); wxString dkey = KeyStringNormalize(mDefaultKeys[i]); wxString name = wxT("/NewKeys/") + mNames[i]; // wxString key = KeyStringNormalize(mManager->GetKeyFromName(mNames[i])); wxString key = KeyStringNormalize(mNewKeys[i]); if (gPrefs->HasEntry(name)) { if (key != KeyStringNormalize(gPrefs->Read(name, key))) { gPrefs->Write(name, key); } if (key == dkey) { gPrefs->DeleteEntry(name); } } else { if (key != dkey) { gPrefs->Write(name, key); } } gPrefs->Flush(); } return gPrefs->Flush(); } void KeyConfigPrefs::Cancel() { // Restore original key values for (size_t i = 0; i < mNames.GetCount(); i++) { mManager->SetKeyFromIndex(i, mKeys[i]); } return; } #endif