1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef CommandSet_DEFINED
#define CommandSet_DEFINED
#include "SkString.h"
#include "Window.h"
#include <functional>
#include <vector>
class SkCanvas;
namespace sk_app {
/**
* Helper class used by applications that want to hook keypresses to trigger events.
*
* An app can simply store an instance of CommandSet and then use it as follows:
* 1) Attach to the Window at initialization time.
* 2) Register commands to be executed for characters or keys. Each command needs a Group and a
* description (both just strings). Commands attached to Keys (rather than characters) also need
* a displayable name for the Key. Finally, a function to execute when the key or character is
* pressed must be supplied. The easiest option to is pass in a lambda that captures [this]
* (your application object), and performs whatever action is desired.
* 3) Register key and char handlers with the Window, and - depending on your state - forward those
* events to the CommandSet's onKey, onChar, and onSoftKey.
* 4) At the end of your onPaint, call drawHelp, and pass in the application's canvas.
* The CommandSet always binds 'h' to cycle through two different help screens. The first shows
* all commands, organized by Group (with headings for each Group). The second shows all commands
* alphabetically by key/character.
*/
class CommandSet {
public:
CommandSet();
void attach(Window* window);
bool onKey(sk_app::Window::Key key, sk_app::Window::InputState state, uint32_t modifiers);
bool onChar(SkUnichar, uint32_t modifiers);
bool onSoftkey(const SkString& softkey);
void addCommand(SkUnichar c, const char* group, const char* description,
std::function<void(void)> function);
void addCommand(Window::Key k, const char* keyName, const char* group, const char* description,
std::function<void(void)> function);
void drawHelp(SkCanvas* canvas);
std::vector<SkString> getCommandsAsSoftkeys() const;
private:
struct Command {
enum CommandType {
kChar_CommandType,
kKey_CommandType,
};
Command(SkUnichar c, const char* group, const char* description,
std::function<void(void)> function)
: fType(kChar_CommandType)
, fChar(c)
, fKeyName(' ' == c ? SkString("Space") : SkStringPrintf("%c", c))
, fGroup(group)
, fDescription(description)
, fFunction(function) {}
Command(Window::Key k, const char* keyName, const char* group, const char* description,
std::function<void(void)> function)
: fType(kKey_CommandType)
, fKey(k)
, fKeyName(keyName)
, fGroup(group)
, fDescription(description)
, fFunction(function) {}
CommandType fType;
// For kChar_CommandType
SkUnichar fChar;
// For kKey_CommandType
Window::Key fKey;
// Common to all command types
SkString fKeyName;
SkString fGroup;
SkString fDescription;
std::function<void(void)> fFunction;
SkString getSoftkeyString() const {
return SkStringPrintf("%s (%s)", fKeyName.c_str(), fDescription.c_str());
}
};
static bool compareCommandKey(const Command& first, const Command& second);
static bool compareCommandGroup(const Command& first, const Command& second);
enum HelpMode {
kNone_HelpMode,
kGrouped_HelpMode,
kAlphabetical_HelpMode,
};
Window* fWindow;
SkTArray<Command> fCommands;
HelpMode fHelpMode;
};
} // namespace sk_app
#endif
|