A11Y: Update bulk selection keyboard shortcuts (#26069)

* A11Y: Update bulk selection keyboard shortcuts

Still a draft, but in current state this:

- adds `shift+b` as a keyboard shortcut to toggle bulk select
- adds `shift+d` as a keyboard shortcut to dismiss selected topic(s) (this
replaces `x r` and `x t` shortcuts)
- adds `x` as a keyboard shortcut to toggle selection (while in bulk select mode)
- fixes a bug with the `shift+a` shortcut, which was not working properly

Note that there is a breaking change here. Previously we had:

- `x r` to dismiss new topics
- `x t` to dismiss unread topics

However, this meant that we couldn't use `x` for selection, because the
itsatrap library does not allow the same character to be used both as a
single character shortcut and as the start of a sequence. The proposed
solution here is more consistent with other apps (Gmail, Github) that use
`x` to toggle selection.

Also, we never show both "Dismiss New" and "Dismiss Unread" in the same
screen, hence it makes sense to consolidate both actions under `shift+d`.

* Address review
This commit is contained in:
Penar Musaraj 2024-03-08 09:54:10 -05:00 committed by GitHub
parent 766e0f7b36
commit 32e1eda3fa
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 104 additions and 22 deletions

View File

@ -88,11 +88,14 @@ export default class KeyboardShortcutsHelp extends Component {
keysDelimiter: PLUS, keysDelimiter: PLUS,
}), }),
help: buildShortcut("application.help", { keys1: ["?"] }), help: buildShortcut("application.help", { keys1: ["?"] }),
dismiss_new: buildShortcut("application.dismiss_new", { bulk_select: buildShortcut("application.toggle_bulk_select", {
keys1: ["x", "r"], keys1: [SHIFT, "b"],
}), }),
dismiss_topics: buildShortcut("application.dismiss_topics", { dismiss: buildShortcut("application.dismiss", {
keys1: ["x", "t"], keys1: [SHIFT, "d"],
}),
x: buildShortcut("application.x", {
keys1: ["x"],
}), }),
log_out: buildShortcut("application.log_out", { log_out: buildShortcut("application.log_out", {
keys1: [SHIFT, "z"], keys1: [SHIFT, "z"],

View File

@ -116,12 +116,14 @@ const DEFAULT_BINDINGS = {
"shift+f11": { handler: "fullscreenComposer", global: true }, "shift+f11": { handler: "fullscreenComposer", global: true },
"shift+u": { handler: "deferTopic" }, "shift+u": { handler: "deferTopic" },
"shift+a": { handler: "toggleAdminActions" }, "shift+a": { handler: "toggleAdminActions" },
"shift+b": { handler: "toggleBulkSelect" },
t: { postAction: "replyAsNewTopic" }, t: { postAction: "replyAsNewTopic" },
u: { handler: "goBack", anonymous: true }, u: { handler: "goBack", anonymous: true },
"x r": { x: { handler: "bulkSelectItem" },
click: "#dismiss-new-bottom,#dismiss-new-top", "shift+d": {
}, // dismiss new click:
"x t": { click: "#dismiss-topics-bottom,#dismiss-topics-top" }, // dismiss topics "#dismiss-new-bottom, #dismiss-new-top, #dismiss-topics-bottom, #dismiss-topics-top",
}, // dismiss new or unread
}; };
const animationDuration = 100; const animationDuration = 100;
@ -412,6 +414,13 @@ export default {
this._moveSelection({ direction: -1, scrollWithinPosts: true }); this._moveSelection({ direction: -1, scrollWithinPosts: true });
}, },
bulkSelectItem() {
const elem = document.querySelector(
".selected input.bulk-select, .selected .select-post"
);
elem?.click();
},
goBack() { goBack() {
history.back(); history.back();
}, },
@ -895,7 +904,17 @@ export default {
}, },
toggleAdminActions() { toggleAdminActions() {
this.appEvents.trigger("topic:toggle-actions"); document.querySelector(".toggle-admin-menu")?.click();
},
toggleBulkSelect() {
const bulkSelect = document.querySelector("button.bulk-select");
if (bulkSelect) {
bulkSelect.click();
} else {
getOwner(this).lookup("controller:topic").send("toggleMultiSelect");
}
}, },
toggleArchivePM() { toggleArchivePM() {

View File

@ -153,8 +153,7 @@ acceptance("Keyboard Shortcuts - Authenticated Users", function (needs) {
exists("#dismiss-topics-top"), exists("#dismiss-topics-top"),
"dismiss unread top button is present" "dismiss unread top button is present"
); );
await triggerKeyEvent(document, "keypress", "X"); await triggerKeyEvent(document, "keydown", "D", { shiftKey: true });
await triggerKeyEvent(document, "keypress", "T");
assert.ok( assert.ok(
exists("#dismiss-read-confirm"), exists("#dismiss-read-confirm"),
"confirmation modal to dismiss unread is present" "confirmation modal to dismiss unread is present"
@ -183,8 +182,8 @@ acceptance("Keyboard Shortcuts - Authenticated Users", function (needs) {
exists("#dismiss-topics-bottom"), exists("#dismiss-topics-bottom"),
"dismiss unread bottom button is hidden" "dismiss unread bottom button is hidden"
); );
await triggerKeyEvent(document, "keypress", "X");
await triggerKeyEvent(document, "keypress", "T"); await triggerKeyEvent(document, "keydown", "D", { shiftKey: true });
assert.ok( assert.ok(
exists("#dismiss-read-confirm"), exists("#dismiss-read-confirm"),
"confirmation modal to dismiss unread is present" "confirmation modal to dismiss unread is present"
@ -212,8 +211,8 @@ acceptance("Keyboard Shortcuts - Authenticated Users", function (needs) {
document.getElementById("ember-testing-container").scrollTop = 0; document.getElementById("ember-testing-container").scrollTop = 0;
await visit("/new"); await visit("/new");
assert.ok(exists("#dismiss-new-top"), "dismiss new top button is present"); assert.ok(exists("#dismiss-new-top"), "dismiss new top button is present");
await triggerKeyEvent(document, "keypress", "X");
await triggerKeyEvent(document, "keypress", "R"); await triggerKeyEvent(document, "keydown", "D", { shiftKey: true });
assert.strictEqual(resetNewCalled, 1); assert.strictEqual(resetNewCalled, 1);
// we get rid of all but one topic so the bottom dismiss button doesn't // we get rid of all but one topic so the bottom dismiss button doesn't
@ -229,8 +228,8 @@ acceptance("Keyboard Shortcuts - Authenticated Users", function (needs) {
exists("#dismiss-new-bottom"), exists("#dismiss-new-bottom"),
"dismiss new bottom button has been hidden" "dismiss new bottom button has been hidden"
); );
await triggerKeyEvent(document, "keypress", "X");
await triggerKeyEvent(document, "keypress", "R"); await triggerKeyEvent(document, "keydown", "D", { shiftKey: true });
assert.strictEqual(resetNewCalled, 2); assert.strictEqual(resetNewCalled, 2);
// restore the original topic list // restore the original topic list
@ -252,8 +251,7 @@ acceptance("Keyboard Shortcuts - Authenticated Users", function (needs) {
"dismiss new bottom button is present" "dismiss new bottom button is present"
); );
await triggerKeyEvent(document, "keypress", "X"); await triggerKeyEvent(document, "keydown", "D", { shiftKey: true });
await triggerKeyEvent(document, "keypress", "R");
assert.strictEqual(resetNewCalled, 1); assert.strictEqual(resetNewCalled, 1);
}); });

View File

@ -4254,8 +4254,9 @@ en:
search: "%{shortcut} Search" search: "%{shortcut} Search"
filter_sidebar: "%{shortcut} Filter sidebar" filter_sidebar: "%{shortcut} Filter sidebar"
help: "%{shortcut} Open keyboard help" help: "%{shortcut} Open keyboard help"
dismiss_new: "%{shortcut} Dismiss New" toggle_bulk_select: "%{shortcut} Toggle bulk select"
dismiss_topics: "%{shortcut} Dismiss Topics" dismiss: "%{shortcut} Dismiss selected topic(s)"
x: "%{shortcut} Toggle selection (in bulk select mode)"
log_out: "%{shortcut} Log Out" log_out: "%{shortcut} Log Out"
composing: composing:
title: "Composing" title: "Composing"

View File

@ -46,6 +46,14 @@ module PageObjects
page.has_no_css?("#{topic_list_item_unread_badge(topic)}") page.has_no_css?("#{topic_list_item_unread_badge(topic)}")
end end
def has_checkbox_selected_on_row?(n)
page.has_css?("#{TOPIC_LIST_ITEM_SELECTOR}:nth-child(#{n}) input.bulk-select:checked")
end
def has_no_checkbox_selected_on_row?(n)
page.has_no_css?("#{TOPIC_LIST_ITEM_SELECTOR}:nth-child(#{n}) input.bulk-select:checked")
end
def click_topic_checkbox(topic) def click_topic_checkbox(topic)
find("#{topic_list_item_class(topic)} input#bulk-select-#{topic.id}").click find("#{topic_list_item_class(topic)} input#bulk-select-#{topic.id}").click
end end

View File

@ -50,6 +50,10 @@ module PageObjects
find("#topic-bulk-action-options__silent").click find("#topic-bulk-action-options__silent").click
end end
def click_dismiss_read_confirm
find("#dismiss-read-confirm").click
end
private private
def bulk_select_dropdown_item(name) def bulk_select_dropdown_item(name)

View File

@ -87,5 +87,37 @@ describe "Topic bulk select", type: :system do
visit("/latest") visit("/latest")
expect(topic_list).to have_no_unread_badge(topics.first) expect(topic_list).to have_no_unread_badge(topics.first)
end end
it "works with keyboard shortcuts" do
sign_in(admin)
visit("/latest")
send_keys([:shift, "b"])
send_keys("j")
send_keys("x") # toggle select
expect(topic_list).to have_checkbox_selected_on_row(1)
send_keys("x") # toggle deselect
expect(topic_list).to have_no_checkbox_selected_on_row(1)
# watch topic and add a reply so we have something in /unread
topic = topics.first
visit("/t/#{topic.slug}/#{topic.id}")
topic_page.watch_topic
expect(topic_page).to have_read_post(1)
Fabricate(:post, topic: topic)
visit("/unread")
expect(topic_list).to have_topics
send_keys([:shift, "b"])
send_keys("j")
send_keys("x")
send_keys([:shift, "d"])
topic_list_header.click_dismiss_read_confirm
expect(topic_list).to have_no_topics
end
end end
end end

View File

@ -2,6 +2,7 @@
describe "Topic page", type: :system do describe "Topic page", type: :system do
fab!(:topic) fab!(:topic)
fab!(:admin)
before { Fabricate(:post, topic: topic, cooked: <<~HTML) } before { Fabricate(:post, topic: topic, cooked: <<~HTML) }
<h2 dir="ltr" id="toc-h2-testing" data-d-toc="toc-h2-testing" class="d-toc-post-heading"> <h2 dir="ltr" id="toc-h2-testing" data-d-toc="toc-h2-testing" class="d-toc-post-heading">
@ -58,7 +59,7 @@ describe "Topic page", type: :system do
PostDestroyer.new(Discourse.system_user, post2).destroy PostDestroyer.new(Discourse.system_user, post2).destroy
PostDestroyer.new(Discourse.system_user, post3).destroy PostDestroyer.new(Discourse.system_user, post3).destroy
sign_in Fabricate(:admin) sign_in admin
end end
it "displays the gap to admins, and allows them to expand it" do it "displays the gap to admins, and allows them to expand it" do
@ -69,4 +70,20 @@ describe "Topic page", type: :system do
expect(page).to have_css(".topic-post", count: 4) expect(page).to have_css(".topic-post", count: 4)
end end
end end
it "supports shift+a kbd shortcut to toggle admin menu" do
sign_in admin
visit("/t/#{topic.slug}/#{topic.id}")
expect(".topic-admin-menu-button-container").to be_present
send_keys([:shift, "a"])
expect(page).to have_css(".topic-admin-popup-menu")
send_keys([:shift, "a"])
expect(page).to have_no_css(".topic-admin-popup-menu")
end
end end