Fix keyboard accessibility in conflict-resolution dialog
It's now possible to tab through the panes and the buttons, and left/right-arrow also change the selection, so you can use Left/Right + Return to move through multiple conflicts. Fixes #3395
This commit is contained in:
parent
ad676adbbb
commit
f688183878
3 changed files with 40 additions and 12 deletions
|
@ -37,6 +37,7 @@
|
||||||
this.hideEmptyFields = false;
|
this.hideEmptyFields = false;
|
||||||
this.clickByRow = false;
|
this.clickByRow = false;
|
||||||
this.clickByItem = false;
|
this.clickByItem = false;
|
||||||
|
this.preventFocus = false;
|
||||||
|
|
||||||
this.clickHandler = null;
|
this.clickHandler = null;
|
||||||
this.blurHandler = null;
|
this.blurHandler = null;
|
||||||
|
@ -1005,10 +1006,12 @@
|
||||||
}
|
}
|
||||||
td.appendChild(addButton);
|
td.appendChild(addButton);
|
||||||
|
|
||||||
for (const domEl of [th, toggleButton, removeButton, addButton]) {
|
if (!this.preventFocus) {
|
||||||
domEl.setAttribute('tabindex', '0');
|
for (const domEl of [th, toggleButton, removeButton, addButton]) {
|
||||||
domEl.addEventListener('keypress', this.handleKeyPress.bind(this));
|
domEl.setAttribute('tabindex', '0');
|
||||||
domEl.addEventListener('focusin', this.updateLastFocused.bind(this));
|
domEl.addEventListener('keypress', this.handleKeyPress.bind(this));
|
||||||
|
domEl.addEventListener('focusin', this.updateLastFocused.bind(this));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this._creatorCount++;
|
this._creatorCount++;
|
||||||
|
|
|
@ -51,6 +51,18 @@
|
||||||
this._leftPane = this._id('left-pane');
|
this._leftPane = this._id('left-pane');
|
||||||
this._rightPane = this._id('right-pane');
|
this._rightPane = this._id('right-pane');
|
||||||
this._mergePane = this._id('merge-pane');
|
this._mergePane = this._id('merge-pane');
|
||||||
|
|
||||||
|
// Select pane with left/right arrow key
|
||||||
|
this.addEventListener('keypress', (event) => {
|
||||||
|
if (event.key == "ArrowRight" && !this._rightPane.hasAttribute("selected")) {
|
||||||
|
this.choosePane(this._rightPane);
|
||||||
|
this.rightPane.groupbox.focus();
|
||||||
|
}
|
||||||
|
else if (event.key == "ArrowLeft" && !this._leftPane.hasAttribute("selected")) {
|
||||||
|
this.choosePane(this._leftPane);
|
||||||
|
this._leftPane.groupbox.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
get data() {
|
get data() {
|
||||||
|
@ -267,12 +279,12 @@
|
||||||
this.append(document.importNode(this.content, true));
|
this.append(document.importNode(this.content, true));
|
||||||
|
|
||||||
this.parent = document.querySelector('merge-group');
|
this.parent = document.querySelector('merge-group');
|
||||||
|
this.isLeftPane = this.id == 'left-pane';
|
||||||
|
this.isRightPane = this.id == 'right-pane';
|
||||||
this.isMergePane = this.id == 'merge-pane';
|
this.isMergePane = this.id == 'merge-pane';
|
||||||
|
|
||||||
if (!this.isMergePane) {
|
if (!this.isMergePane) {
|
||||||
this.box.onclick = function () {
|
this.groupbox.onclick = this.handleClick.bind(this);
|
||||||
this.parent.choosePane(this);
|
|
||||||
}.bind(this);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -280,7 +292,7 @@
|
||||||
return this.parent.type;
|
return this.parent.type;
|
||||||
}
|
}
|
||||||
|
|
||||||
get box() {
|
get groupbox() {
|
||||||
return this.querySelector('groupbox');
|
return this.querySelector('groupbox');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -326,7 +338,7 @@
|
||||||
var button = this._class('choose-button');
|
var button = this._class('choose-button');
|
||||||
button.label = Zotero.getString('sync.conflict.chooseThisVersion');
|
button.label = Zotero.getString('sync.conflict.chooseThisVersion');
|
||||||
if (this.showButton) {
|
if (this.showButton) {
|
||||||
button.onclick = () => this.parent.choosePane(this);
|
button.onclick = this.handleClick.bind(this);
|
||||||
button.style.visibility = 'visible';
|
button.style.visibility = 'visible';
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -403,6 +415,17 @@
|
||||||
objbox.setAttribute("flex", "1");
|
objbox.setAttribute("flex", "1");
|
||||||
objbox.mode = this.type == 'file' ? 'filemerge' : 'merge';
|
objbox.mode = this.type == 'file' ? 'filemerge' : 'merge';
|
||||||
|
|
||||||
|
// Keyboard accessibility
|
||||||
|
objbox.preventFocus = true;
|
||||||
|
if (!this.isMergePane) {
|
||||||
|
this.groupbox.setAttribute('tabindex', 0);
|
||||||
|
this.groupbox.addEventListener('keypress', (event) => {
|
||||||
|
if (event.key == " ") {
|
||||||
|
this.handleClick();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// Store JSON
|
// Store JSON
|
||||||
this._data = val;
|
this._data = val;
|
||||||
|
|
||||||
|
@ -427,8 +450,8 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
click() {
|
handleClick() {
|
||||||
this.box.click();
|
this.parent.choosePane(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
_class(className) {
|
_class(className) {
|
||||||
|
|
|
@ -93,7 +93,9 @@ merge-group:not([mergetype="note"]) #right-pane:active attachment-box #title {
|
||||||
}
|
}
|
||||||
|
|
||||||
#left-pane:hover:not([selected=true]) h2,
|
#left-pane:hover:not([selected=true]) h2,
|
||||||
#right-pane:hover:not([selected=true]) h2 {
|
#left-pane groupbox:focus:not([selected=true]) h2,
|
||||||
|
#right-pane:hover:not([selected=true]) h2,
|
||||||
|
#right-pane groupbox:focus:not([selected=true]) h2 {
|
||||||
text-decoration: underline;
|
text-decoration: underline;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue