shortcut editor, replace the shortcut catcher by the one used in an object inspector

This commit is contained in:
Basile Burg 2017-01-26 02:28:16 +01:00
parent dedab16951
commit a0425c4e9a
No known key found for this signature in database
GPG Key ID: 1868039F415CB8CF
4 changed files with 116 additions and 120 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 81 KiB

View File

@ -1 +1,18 @@
---
title: Options - shortcuts
---
{% include xstyle.css %}
### Shortcuts editor
This category exposes all the shortcuts, sorted by category.
![](img/options/shortcuts.png)
Each item is editable.
- <img src="{%include icurl%}other/clean.png" class="tlbric"/>: Deletes the current assignation.
- <img src="{%include icurl%}other/keyboard_pencil.png" class="tlbric"/>: Displays the editor. The shortcut can also be selected in the combo menu.
After an assignation attempt and if the selected combination is already used the new shortcut is not validated and a warning is displayed.

View File

@ -62,54 +62,42 @@ object CEShortcutEditor: TCEShortcutEditor
ClientHeight = 26 ClientHeight = 26
ClientWidth = 420 ClientWidth = 420
TabOrder = 2 TabOrder = 2
object schrtText: TStaticText
Left = 0
Height = 22
Top = 2
Width = 247
Align = alClient
Alignment = taCenter
BorderSpacing.Top = 2
BorderSpacing.Bottom = 2
BorderStyle = sbsSunken
TabOrder = 0
end
object shortcutCatcher: TEdit
Left = 299
Height = 22
Hint = 'type a key combination while active'
Top = 2
Width = 120
Align = alRight
BorderSpacing.Top = 1
BorderSpacing.Bottom = 1
BorderSpacing.Around = 1
Enabled = False
OnExit = shortcutCatcherExit
OnKeyDown = LabeledEdit1KeyDown
OnMouseLeave = shortcutCatcherMouseLeave
TabOrder = 1
end
object btnActivate: TSpeedButton
Left = 274
Height = 22
Hint = 'enable shortcut capture'
Top = 2
Width = 23
Align = alRight
BorderSpacing.Around = 2
OnClick = btnActivateClick
end
object btnClear: TSpeedButton object btnClear: TSpeedButton
Left = 249 Left = 2
Height = 22 Height = 22
Hint = 'clear shortcut' Hint = 'clear shortcut'
Top = 2 Top = 2
Width = 23 Width = 23
Align = alRight Align = alLeft
BorderSpacing.Around = 2 BorderSpacing.Around = 2
Flat = True
OnClick = btnClearClick OnClick = btnClearClick
end end
object propedit: TTIPropertyGrid
Left = 52
Height = 26
Top = 0
Width = 368
Align = alClient
CheckboxForBoolean = False
DefaultValueFont.Color = clWindowText
Filter = [tkInteger, tkChar, tkEnumeration, tkFloat, tkSet, tkMethod, tkSString, tkLString, tkAString, tkWString, tkVariant, tkArray, tkRecord, tkInterface, tkClass, tkObject, tkWChar, tkBool, tkInt64, tkQWord, tkDynArray, tkInterfaceRaw, tkProcVar, tkUString, tkUChar, tkHelper, tkFile, tkClassRef, tkPointer]
Indent = 10
NameFont.Color = clWindowText
OnModified = propeditModified
ValueFont.Color = clMaroon
end
object btnEdit: TSpeedButton
Left = 27
Height = 22
Hint = 'edit selected shortcut'
Top = 2
Width = 23
Align = alLeft
BorderSpacing.Around = 2
Flat = True
OnClick = btnEditClick
end
end end
end end
end end

View File

@ -6,9 +6,9 @@ interface
uses uses
Classes, SysUtils, FileUtil, TreeFilterEdit, Forms, Controls, Menus, Graphics, Classes, SysUtils, FileUtil, TreeFilterEdit, Forms, Controls, Menus, Graphics,
ExtCtrls, LCLProc, ComCtrls, StdCtrls, Buttons, LCLType, PropEdits, strutils, ExtCtrls, LCLProc, ComCtrls, Buttons, LCLType, PropEdits, RTTIGrids,
ce_sharedres, ce_observer, ce_interfaces, ce_common, ce_writableComponent, strutils, ce_sharedres, ce_observer, ce_interfaces, ce_common,
ce_dialogs, EditBtn; ce_writableComponent, ce_dialogs, EditBtn;
type type
@ -46,29 +46,36 @@ type
property item[index: Integer]: TShortcutItem read getItem; default; property item[index: Integer]: TShortcutItem read getItem; default;
end; end;
TEditableShortcut = class(TPersistent)
public
value: TShortCut;
published
property shortcut: TShortCut read value write value;
end;
{ TCEShortcutEditor } { TCEShortcutEditor }
TCEShortcutEditor = class(TFrame, ICEEditableOptions) TCEShortcutEditor = class(TFrame, ICEEditableOptions)
btnClear: TSpeedButton; btnClear: TSpeedButton;
shortcutCatcher: TEdit; btnEdit: TSpeedButton;
Panel1: TPanel; Panel1: TPanel;
fltItems: TTreeFilterEdit; fltItems: TTreeFilterEdit;
Panel2: TPanel; Panel2: TPanel;
schrtText: TStaticText; propedit: TTIPropertyGrid;
btnActivate: TSpeedButton;
tree: TTreeView; tree: TTreeView;
procedure btnActivateClick(Sender: TObject);
procedure btnClearClick(Sender: TObject); procedure btnClearClick(Sender: TObject);
procedure btnEditClick(Sender: TObject);
function fltItemsFilterItem(Item: TObject; out Done: Boolean): Boolean; function fltItemsFilterItem(Item: TObject; out Done: Boolean): Boolean;
procedure LabeledEdit1KeyDown(Sender: TObject; var Key: Word;Shift: TShiftState);
procedure shortcutCatcherExit(Sender: TObject); procedure shortcutCatcherExit(Sender: TObject);
procedure shortcutCatcherMouseLeave(Sender: TObject); procedure shortcutCatcherMouseLeave(Sender: TObject);
procedure propeditModified(Sender: TObject);
procedure treeSelectionChanged(Sender: TObject); procedure treeSelectionChanged(Sender: TObject);
private private
fObservers: TCEEditableShortCutSubject; fObservers: TCEEditableShortCutSubject;
fShortcuts: TShortCutCollection; fShortcuts: TShortCutCollection;
fBackup: TShortCutCollection; fBackup: TShortCutCollection;
fHasChanged: boolean; fHasChanged: boolean;
propvalue: TEditableShortcut;
// //
function optionedWantCategory(): string; function optionedWantCategory(): string;
function optionedWantEditorKind: TOptionEditorKind; function optionedWantEditorKind: TOptionEditorKind;
@ -82,8 +89,7 @@ type
procedure receiveShortcuts; procedure receiveShortcuts;
procedure updateEditCtrls; procedure updateEditCtrls;
procedure sendShortcuts; procedure sendShortcuts;
protected function anItemIsSelected: boolean;
procedure UpdateShowing; override;
public public
constructor create(TheOwner: TComponent); override; constructor create(TheOwner: TComponent); override;
destructor destroy; override; destructor destroy; override;
@ -175,25 +181,29 @@ end;
constructor TCEShortcutEditor.create(TheOwner: TComponent); constructor TCEShortcutEditor.create(TheOwner: TComponent);
begin begin
inherited; inherited;
propvalue := TEditableShortcut.Create;
fObservers := TCEEditableShortCutSubject.create; fObservers := TCEEditableShortCutSubject.create;
fShortcuts := TShortCutCollection.create(self); fShortcuts := TShortCutCollection.create(self);
fBackup := TShortCutCollection.create(self); fBackup := TShortCutCollection.create(self);
AssignPng(btnClear, 'CLEAN'); AssignPng(btnClear, 'CLEAN');
AssignPng(btnEdit, 'KEYBOARD_PENCIL');
EntitiesConnector.addObserver(self); EntitiesConnector.addObserver(self);
propedit.TIObject := propvalue;
propedit.PropertyEditorHook.AddHandlerModified(@propeditModified);
end; end;
destructor TCEShortcutEditor.destroy; destructor TCEShortcutEditor.destroy;
begin begin
propvalue.Free;
fObservers.Free; fObservers.Free;
inherited; inherited;
end; end;
procedure TCEShortcutEditor.UpdateShowing; function TCEShortcutEditor.anItemIsSelected: boolean;
begin begin
inherited; result := true;
if not visible then exit; if tree.Selected.isNil or (tree.Selected.Level = 0) or tree.Selected.Data.isNil then
// result := false;
AssignPng(btnActivate, 'KEYBOARD_PENCIL');
end; end;
{$ENDREGION} {$ENDREGION}
@ -247,43 +257,62 @@ end;
procedure TCEShortcutEditor.shortcutCatcherExit(Sender: TObject); procedure TCEShortcutEditor.shortcutCatcherExit(Sender: TObject);
begin begin
shortcutCatcher.Enabled := false;
updateEditCtrls; updateEditCtrls;
end; end;
procedure TCEShortcutEditor.shortcutCatcherMouseLeave(Sender: TObject); procedure TCEShortcutEditor.shortcutCatcherMouseLeave(Sender: TObject);
begin begin
shortcutCatcher.Enabled := false;
updateEditCtrls; updateEditCtrls;
end; end;
procedure TCEShortcutEditor.btnActivateClick(Sender: TObject); procedure TCEShortcutEditor.propeditModified(Sender: TObject);
var
i: integer;
sh: TShortCut;
sht: string;
dup: TShortcutItem = nil;
const
msg = '"%s" is already assigned in the same category by "%s". The new shortcut will be ignored';
begin begin
if tree.Selected.isNil then exit; if not anItemIsSelected then
if tree.Selected.Level = 0 then exit; exit;
if tree.Selected.Data.isNil then exit; sh := propvalue.value;
// sht := shortCutToText(sh);
shortcutCatcher.Enabled := not shortcutCatcher.Enabled; if sht.isEmpty then
if shortcutCatcher.Enabled then exit;
shortcutCatcher.SetFocus; for i:= 0 to tree.Selected.Parent.Count-1 do
if i <> tree.Selected.Index then
if TShortcutItem(tree.Selected.Parent.Items[i].Data).data = sh then
dup := TShortcutItem(tree.Selected.Parent.Items[i].Data);
if dup.isNotNil then
dlgOkInfo(format(msg,[ShortCutToText(sh), dup.identifier]))
else if TShortcutItem(tree.Selected.Data).data <> sh then
begin
TShortcutItem(tree.Selected.Data).data := sh;
fHasChanged := true;
end;
updateEditCtrls;
end; end;
procedure TCEShortcutEditor.btnClearClick(Sender: TObject); procedure TCEShortcutEditor.btnClearClick(Sender: TObject);
begin begin
if tree.Selected.isNil then exit; if not anItemIsSelected then
if tree.Selected.Level = 0 then exit; exit;
if tree.Selected.Data.isNil then exit;
//
if TShortcutItem(tree.Selected.Data).data <> 0 then if TShortcutItem(tree.Selected.Data).data <> 0 then
begin begin
TShortcutItem(tree.Selected.Data).data := 0; TShortcutItem(tree.Selected.Data).data := 0;
fHasChanged := true; fHasChanged := true;
end; end;
//
shortcutCatcher.Enabled := false;
updateEditCtrls; updateEditCtrls;
end; end;
procedure TCEShortcutEditor.btnEditClick(Sender: TObject);
begin
if not anItemIsSelected then
exit;
propedit.Rows[0].Editor.Edit;
end;
function TCEShortcutEditor.fltItemsFilterItem(Item: TObject; out Done: Boolean): Boolean; function TCEShortcutEditor.fltItemsFilterItem(Item: TObject; out Done: Boolean): Boolean;
var var
shc: TShortcutItem; shc: TShortcutItem;
@ -306,56 +335,18 @@ begin
end; end;
end; end;
procedure TCEShortcutEditor.LabeledEdit1KeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
var
i: integer;
sh: TShortCut;
sht: string;
dup: TShortcutItem = nil;
const
msg = '"%s" is already assigned in the same category by "%s". The new shortcut will be ignored';
begin
if tree.Selected.isNil then exit;
if tree.Selected.Level = 0 then exit;
if tree.Selected.Data.isNil then exit;
//
if Key = VK_RETURN then
shortcutCatcher.Enabled := false
else
begin
sh := Shortcut(Key, Shift);
sht := shortCutToText(sh);
if sht.isEmpty then
exit;
for i:= 0 to tree.Selected.Parent.Count-1 do
if i <> tree.Selected.Index then
if TShortcutItem(tree.Selected.Parent.Items[i].Data).data = sh then
dup := TShortcutItem(tree.Selected.Parent.Items[i].Data);
if dup.isNotNil then
dlgOkInfo(format(msg,[ShortCutToText(sh), dup.identifier]))
else if TShortcutItem(tree.Selected.Data).data <> sh then
begin
TShortcutItem(tree.Selected.Data).data := sh;
fHasChanged := true;
end;
end;
//
updateEditCtrls;
end;
procedure TCEShortcutEditor.updateEditCtrls; procedure TCEShortcutEditor.updateEditCtrls;
var
shc: TShortcutItem;
begin begin
schrtText.Caption := ''; if not anItemIsSelected then
shortcutCatcher.TextHint := '(Enter shortcut)'; exit;
// shc := TShortcutItem(tree.Selected.Data);
if tree.Selected.isNil then exit; if propvalue.value <> shc.data then
if tree.Selected.Level = 0 then exit; begin
if tree.Selected.Data.isNil then exit; propvalue.value := shc.data;
// propedit.Update;
schrtText.Caption := TShortcutItem(tree.Selected.Data).combination; end;
shortcutCatcher.Text := '';
if schrtText.Caption <> '' then
shortcutCatcher.TextHint := '(' + schrtText.Caption + ')';
end; end;
function TCEShortcutEditor.findCategory(const aName: string; aData: Pointer): TTreeNode; function TCEShortcutEditor.findCategory(const aName: string; aData: Pointer): TTreeNode;