|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Sanity-check the conversion and remove this comment.
|
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS002: Fix invalid constructor
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* DS206: Consider reworking classes to avoid initClass
|
|
|
|
* DS207: Consider shorter variations of null checks
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
|
|
|
*/
|
|
|
|
const Cls = (app.views.ListFocus = class ListFocus extends app.View {
|
|
|
|
static initClass() {
|
|
|
|
this.activeClass = 'focus';
|
|
|
|
|
|
|
|
this.events =
|
|
|
|
{click: 'onClick'};
|
|
|
|
|
|
|
|
this.shortcuts = {
|
|
|
|
up: 'onUp',
|
|
|
|
down: 'onDown',
|
|
|
|
left: 'onLeft',
|
|
|
|
enter: 'onEnter',
|
|
|
|
superEnter: 'onSuperEnter',
|
|
|
|
escape: 'blur'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor(el) {
|
|
|
|
this.blur = this.blur.bind(this);
|
|
|
|
this.onDown = this.onDown.bind(this);
|
|
|
|
this.onUp = this.onUp.bind(this);
|
|
|
|
this.onLeft = this.onLeft.bind(this);
|
|
|
|
this.onEnter = this.onEnter.bind(this);
|
|
|
|
this.onSuperEnter = this.onSuperEnter.bind(this);
|
|
|
|
this.onClick = this.onClick.bind(this);
|
|
|
|
this.el = el;
|
|
|
|
super(...arguments);
|
|
|
|
this.focusOnNextFrame = $.framify(this.focus, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
focus(el, options) {
|
|
|
|
if (options == null) { options = {}; }
|
|
|
|
if (el && !el.classList.contains(this.constructor.activeClass)) {
|
|
|
|
this.blur();
|
|
|
|
el.classList.add(this.constructor.activeClass);
|
|
|
|
if (options.silent !== true) { $.trigger(el, 'focus'); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
blur() {
|
|
|
|
let cursor;
|
|
|
|
if (cursor = this.getCursor()) {
|
|
|
|
cursor.classList.remove(this.constructor.activeClass);
|
|
|
|
$.trigger(cursor, 'blur');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getCursor() {
|
|
|
|
return this.findByClass(this.constructor.activeClass) || this.findByClass(app.views.ListSelect.activeClass);
|
|
|
|
}
|
|
|
|
|
|
|
|
findNext(cursor) {
|
|
|
|
let next;
|
|
|
|
if (next = cursor.nextSibling) {
|
|
|
|
if (next.tagName === 'A') {
|
|
|
|
return next;
|
|
|
|
} else if (next.tagName === 'SPAN') { // pagination link
|
|
|
|
$.click(next);
|
|
|
|
return this.findNext(cursor);
|
|
|
|
} else if (next.tagName === 'DIV') { // sub-list
|
|
|
|
if (cursor.className.indexOf(' open') >= 0) {
|
|
|
|
return this.findFirst(next) || this.findNext(next);
|
|
|
|
} else {
|
|
|
|
return this.findNext(next);
|
|
|
|
}
|
|
|
|
} else if (next.tagName === 'H6') { // title
|
|
|
|
return this.findNext(next);
|
|
|
|
}
|
|
|
|
} else if (cursor.parentNode !== this.el) {
|
|
|
|
return this.findNext(cursor.parentNode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
findFirst(cursor) {
|
|
|
|
let first;
|
|
|
|
if (!(first = cursor.firstChild)) { return; }
|
|
|
|
|
|
|
|
if (first.tagName === 'A') {
|
|
|
|
return first;
|
|
|
|
} else if (first.tagName === 'SPAN') { // pagination link
|
|
|
|
$.click(first);
|
|
|
|
return this.findFirst(cursor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
findPrev(cursor) {
|
|
|
|
let prev;
|
|
|
|
if (prev = cursor.previousSibling) {
|
|
|
|
if (prev.tagName === 'A') {
|
|
|
|
return prev;
|
|
|
|
} else if (prev.tagName === 'SPAN') { // pagination link
|
|
|
|
$.click(prev);
|
|
|
|
return this.findPrev(cursor);
|
|
|
|
} else if (prev.tagName === 'DIV') { // sub-list
|
|
|
|
if (prev.previousSibling.className.indexOf('open') >= 0) {
|
|
|
|
return this.findLast(prev) || this.findPrev(prev);
|
|
|
|
} else {
|
|
|
|
return this.findPrev(prev);
|
|
|
|
}
|
|
|
|
} else if (prev.tagName === 'H6') { // title
|
|
|
|
return this.findPrev(prev);
|
|
|
|
}
|
|
|
|
} else if (cursor.parentNode !== this.el) {
|
|
|
|
return this.findPrev(cursor.parentNode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
findLast(cursor) {
|
|
|
|
let last;
|
|
|
|
if (!(last = cursor.lastChild)) { return; }
|
|
|
|
|
|
|
|
if (last.tagName === 'A') {
|
|
|
|
return last;
|
|
|
|
} else if ((last.tagName === 'SPAN') || (last.tagName === 'H6')) { // pagination link or title
|
|
|
|
return this.findPrev(last);
|
|
|
|
} else if (last.tagName === 'DIV') { // sub-list
|
|
|
|
return this.findLast(last);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onDown() {
|
|
|
|
let cursor;
|
|
|
|
if ((cursor = this.getCursor())) {
|
|
|
|
this.focusOnNextFrame(this.findNext(cursor));
|
|
|
|
} else {
|
|
|
|
this.focusOnNextFrame(this.findByTag('a'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onUp() {
|
|
|
|
let cursor;
|
|
|
|
if ((cursor = this.getCursor())) {
|
|
|
|
this.focusOnNextFrame(this.findPrev(cursor));
|
|
|
|
} else {
|
|
|
|
this.focusOnNextFrame(this.findLastByTag('a'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onLeft() {
|
|
|
|
const cursor = this.getCursor();
|
|
|
|
if (cursor && !cursor.classList.contains(app.views.ListFold.activeClass) && (cursor.parentNode !== this.el)) {
|
|
|
|
const prev = cursor.parentNode.previousSibling;
|
|
|
|
if (prev && prev.classList.contains(app.views.ListFold.targetClass)) { this.focusOnNextFrame(cursor.parentNode.previousSibling); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onEnter() {
|
|
|
|
let cursor;
|
|
|
|
if (cursor = this.getCursor()) {
|
|
|
|
$.click(cursor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onSuperEnter() {
|
|
|
|
let cursor;
|
|
|
|
if (cursor = this.getCursor()) {
|
|
|
|
$.popup(cursor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onClick(event) {
|
|
|
|
if ((event.which !== 1) || event.metaKey || event.ctrlKey) { return; }
|
|
|
|
const target = $.eventTarget(event);
|
|
|
|
if (target.tagName === 'A') {
|
|
|
|
this.focus(target, {silent: true});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
Cls.initClass();
|