Improve keyboard shortcuts list with resizable headers

This commit is contained in:
David Capello 2017-10-24 10:20:21 -03:00
parent bfc2ecb890
commit 0ee5dbea7a
5 changed files with 154 additions and 45 deletions

View File

@ -478,6 +478,9 @@
<text color="disabled" align="left" state="disabled" />
</style>
<style id="mini_label" extends="label" font="mini" />
<style id="list_header_label" padding="2">
<text color="text" align="left" x="2" />
</style>
<style id="link">
<text color="link_text" align="left" />
<text color="link_hover" align="left" state="mouse" />
@ -687,7 +690,7 @@
<style id="tab">
<background part="tab_normal" align="middle" />
<background part="tab_active" align="middle" state="focus" />
<text color="tab_normal_text" align="left" valign="middle" padding-left="4" padding-top="2" />
<text color="tab_normal_text" align="left" valign="middle" />
<text color="tab_active_text" state="focus" />
</style>
<style id="tab_text">

View File

@ -47,7 +47,60 @@ using namespace skin;
using namespace tools;
using namespace ui;
static int g_sep = 0;
namespace {
class HeaderSplitter : public Splitter {
public:
HeaderSplitter() : Splitter(Splitter::ByPixel, HORIZONTAL) {
}
void onPositionChange() override {
Splitter::onPositionChange();
Widget* p = parent();
while (p && p->type() != kViewWidget)
p = p->parent();
if (p)
p->layout();
}
};
class HeaderItem : public ListItem {
public:
HeaderItem()
: m_actionLabel("Action")
, m_keyLabel("Key")
, m_contextLabel("Context") {
setBorder(gfx::Border(0));
auto theme = SkinTheme::instance();
m_actionLabel.setStyle(theme->styles.listHeaderLabel());
m_keyLabel.setStyle(theme->styles.listHeaderLabel());
m_contextLabel.setStyle(theme->styles.listHeaderLabel());
m_splitter1.setPosition(ui::display_w()*3/4 * 4/10);
m_splitter2.setPosition(ui::display_w()*3/4 * 2/10);
addChild(&m_splitter1);
m_splitter1.addChild(&m_actionLabel);
m_splitter1.addChild(&m_splitter2);
m_splitter2.addChild(&m_keyLabel);
m_splitter2.addChild(&m_contextLabel);
}
int keyXPos() const {
return m_keyLabel.bounds().x - bounds().x;
}
int contextXPos() const {
return m_contextLabel.bounds().x - bounds().x;
}
private:
HeaderSplitter m_splitter1, m_splitter2;
Label m_actionLabel;
Label m_keyLabel;
Label m_contextLabel;
};
class KeyItem : public ListItem {
@ -65,14 +118,19 @@ class KeyItem : public ListItem {
};
public:
KeyItem(const std::string& text, Key* key, AppMenuItem* menuitem, int level)
KeyItem(const std::string& text,
Key* key,
AppMenuItem* menuitem,
const int level,
HeaderItem* headerItem)
: ListItem(text)
, m_key(key)
, m_keyOrig(key ? new Key(*key): nullptr)
, m_menuitem(menuitem)
, m_level(level)
, m_hotAccel(-1)
, m_lockButtons(false) {
, m_lockButtons(false)
, m_headerItem(headerItem) {
gfx::Border border = this->border();
border.top(0);
border.bottom(0);
@ -186,6 +244,14 @@ private:
size.w = size.w + border().width();
size.h = size.h + border().height() + 4*guiscale();
if (m_key && m_key->keycontext() != KeyContext::Any) {
int w =
m_headerItem->contextXPos() +
Graphics::measureUITextLength(
convertKeyContextToUserFriendlyString(m_key->keycontext()), font());
size.w = MAX(size.w, w);
}
if (m_key && !m_key->accels().empty()) {
size_t combos = m_key->accels().size();
if (combos > 1)
@ -212,34 +278,43 @@ private:
g->fillRect(bg, bounds);
int y = bounds.y + 2*guiscale();
const int th = textSize().h;
// Position of the second and third columns
const int keyXPos = bounds.x + m_headerItem->keyXPos();
const int contextXPos = bounds.x + m_headerItem->contextXPos();
bounds.shrink(border());
g->drawUIText(text(), fg, bg,
gfx::Point(
bounds.x + m_level*16 * guiscale(),
bounds.y + 2*guiscale()), 0);
{
int x = bounds.x + m_level*16 * guiscale();
IntersectClip clip(g, gfx::Rect(x, y, keyXPos - x, th));
if (clip)
g->drawUIText(text(), fg, bg, gfx::Point(x, y), 0);
}
if (m_key && !m_key->accels().empty()) {
std::string buf;
int y = bounds.y;
int dh = textSize().h + 4*guiscale();
int i = 0;
if (m_key->keycontext() != KeyContext::Any) {
g->drawText(
convertKeyContextToUserFriendlyString(m_key->keycontext()), fg, bg,
gfx::Point(contextXPos, y));
}
IntersectClip clip(g, gfx::Rect(keyXPos, y, contextXPos - keyXPos, th));
if (clip) {
int dh = th + 4*guiscale();
int i = 0;
for (const Accelerator& accel : m_key->accels()) {
if (i != m_hotAccel || !m_changeButton) {
std::string s = accel.toString();
if (m_key->keycontext() != KeyContext::Any) {
s += fmt::format(
" (Context: {0})", convertKeyContextToUserFriendlyString(m_key->keycontext()));
g->drawText(
accel.toString(), fg, bg,
gfx::Point(keyXPos, y));
}
g->drawText(s, fg, bg,
gfx::Point(bounds.x + g_sep, y + 2*guiscale()));
}
y += dh;
++i;
}
}
}
}
void onResize(ResizeEvent& ev) override {
ListItem::onResize(ev);
@ -268,7 +343,7 @@ private:
int w = Graphics::measureUITextLength(
(accels && i < (int)accels->size() ? (*accels)[i].toString().c_str(): ""),
font());
gfx::Rect itemBounds(bounds.x + g_sep, y, w, dh);
gfx::Rect itemBounds(bounds.x + m_headerItem->keyXPos(), y, w, dh);
itemBounds = itemBounds.enlarge(
gfx::Border(
4*guiscale(), 0,
@ -369,6 +444,7 @@ private:
obs::scoped_connection m_addConn;
int m_hotAccel;
bool m_lockButtons;
HeaderItem* m_headerItem;
};
class KeyboardShortcutsWindow : public app::gen::KeyboardShortcuts {
@ -382,6 +458,11 @@ public:
section()->addChild(new ListItem("Tools"));
section()->addChild(new ListItem("Action Modifiers"));
m_listBoxes.push_back(menus());
m_listBoxes.push_back(commands());
m_listBoxes.push_back(tools());
m_listBoxes.push_back(actions());
search()->Change.connect(base::Bind<void>(&KeyboardShortcutsWindow::onSearchChange, this));
section()->Change.connect(base::Bind<void>(&KeyboardShortcutsWindow::onSectionChange, this));
importButton()->Click.connect(base::Bind<void>(&KeyboardShortcutsWindow::onImport, this));
@ -420,8 +501,8 @@ private:
deleteAllKeyItems();
// Load keyboard shortcuts
fillList(this->menus(), AppMenus::instance()->getRootMenu(), 0);
fillList(this->tools(), App::instance()->toolBox());
fillList(menus(), AppMenus::instance()->getRootMenu(), 0);
fillList(tools(), App::instance()->toolBox());
for (Key* key : *app::KeyboardShortcuts::instance()) {
if (key->type() == KeyType::Tool ||
key->type() == KeyType::Quicktool) {
@ -442,7 +523,7 @@ private:
+ ": " + text;
break;
}
KeyItem* keyItem = new KeyItem(text, key, NULL, 0);
KeyItem* keyItem = new KeyItem(text, key, nullptr, 0, &m_headerItem);
ListBox* listBox = nullptr;
switch (key->type()) {
@ -461,15 +542,18 @@ private:
}
}
this->commands()->sortItems();
this->tools()->sortItems();
this->actions()->sortItems();
commands()->sortItems();
tools()->sortItems();
actions()->sortItems();
section()->selectIndex(0);
updateViews();
}
void deleteList(Widget* listbox) {
if (m_headerItem.parent() == listbox)
listbox->removeChild(&m_headerItem);
while (listbox->lastChild()) {
Widget* item = listbox->lastChild();
listbox->removeChild(item);
@ -487,9 +571,8 @@ private:
MatchWords match(search);
ListBox* listBoxes[] = { menus(), commands(), tools(), actions() };
int sectionIdx = 0; // index 0 is menus
for (auto listBox : listBoxes) {
for (auto listBox : m_listBoxes) {
Separator* group = nullptr;
for (auto item : listBox->children()) {
@ -507,7 +590,8 @@ private:
KeyItem* copyItem =
new KeyItem(itemText,
keyItem->key(),
keyItem->menuitem(), 0);
keyItem->menuitem(), 0,
&m_headerItem);
m_allKeyItems.push_back(copyItem);
searchList()->addChild(copyItem);
@ -541,12 +625,20 @@ private:
}
void updateViews() {
int section = this->section()->getSelectedIndex();
searchView()->setVisible(section < 0);
menusView()->setVisible(section == 0);
commandsView()->setVisible(section == 1);
toolsView()->setVisible(section == 2);
actionsView()->setVisible(section == 3);
int s = section()->getSelectedIndex();
searchView()->setVisible(s < 0);
menusView()->setVisible(s == 0);
commandsView()->setVisible(s == 1);
toolsView()->setVisible(s == 2);
actionsView()->setVisible(s == 3);
if (m_headerItem.parent())
m_headerItem.parent()->removeChild(&m_headerItem);
if (s < 0)
searchList()->insertChild(0, &m_headerItem);
else
m_listBoxes[s]->insertChild(0, &m_headerItem);
layout();
}
@ -596,7 +688,8 @@ private:
KeyItem* keyItem = new KeyItem(
menuItem->text().c_str(),
menuItem->key(), menuItem, level);
menuItem->key(), menuItem, level,
&m_headerItem);
m_allKeyItems.push_back(keyItem);
listbox->addChild(keyItem);
@ -612,22 +705,28 @@ private:
std::string text = tool->getText();
Key* key = app::KeyboardShortcuts::instance()->tool(tool);
KeyItem* keyItem = new KeyItem(text, key, nullptr, 0);
KeyItem* keyItem = new KeyItem(text, key, nullptr, 0,
&m_headerItem);
m_allKeyItems.push_back(keyItem);
listbox->addChild(keyItem);
text += " (quick)";
key = app::KeyboardShortcuts::instance()->quicktool(tool);
keyItem = new KeyItem(text, key, nullptr, 0);
keyItem = new KeyItem(text, key, nullptr, 0,
&m_headerItem);
m_allKeyItems.push_back(keyItem);
listbox->addChild(keyItem);
}
}
std::vector<ListBox*> m_listBoxes;
std::vector<KeyItem*> m_allKeyItems;
bool m_searchChange;
HeaderItem m_headerItem;
};
} // anonymous namespace
class KeyboardShortcutsCommand : public Command {
public:
KeyboardShortcutsCommand();
@ -663,7 +762,6 @@ void KeyboardShortcutsCommand::onExecute(Context* context)
KeyboardShortcutsWindow window(neededSearchCopy);
window.setBounds(gfx::Rect(0, 0, ui::display_w()*3/4, ui::display_h()*3/4));
g_sep = window.bounds().w / 2;
window.centerWindow();
window.setVisible(true);

View File

@ -14,7 +14,7 @@ namespace ui {
class ListItem : public Widget {
public:
ListItem(const std::string& text = "");
ListItem(const std::string& text = std::string());
const std::string& getValue() const { return m_value; }

View File

@ -40,6 +40,7 @@ void Splitter::setPosition(double pos)
{
m_pos = pos;
limitPos();
onPositionChange();
invalidate();
}
@ -119,7 +120,7 @@ bool Splitter::onProcessMessage(Message* msg)
}
limitPos();
layout();
onPositionChange();
return true;
}
break;
@ -309,6 +310,11 @@ void Splitter::onSaveLayout(SaveLayoutEvent& ev)
ev.stream() << pos;
}
void Splitter::onPositionChange()
{
layout();
}
Widget* Splitter::panel1() const
{
const WidgetsList& list = children();

View File

@ -30,6 +30,8 @@ namespace ui {
void onLoadLayout(LoadLayoutEvent& ev) override;
void onSaveLayout(SaveLayoutEvent& ev) override;
virtual void onPositionChange();
private:
Widget* panel1() const;
Widget* panel2() const;