matrix-react-sdk
Version:
SDK for matrix.org using React
90 lines (86 loc) • 12.9 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.LandmarkNavigation = exports.Landmark = void 0;
var _RoomContext = require("../contexts/RoomContext");
var _actions = require("../dispatcher/actions");
var _dispatcher = _interopRequireDefault(require("../dispatcher/dispatcher"));
/*
* Copyright 2024 New Vector Ltd.
* Copyright 2024 The Matrix.org Foundation C.I.C.
*
* SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
* Please see LICENSE files in the repository root for full details.
*/
let Landmark = exports.Landmark = /*#__PURE__*/function (Landmark) {
Landmark[Landmark["ACTIVE_SPACE_BUTTON"] = 0] = "ACTIVE_SPACE_BUTTON";
Landmark[Landmark["ROOM_SEARCH"] = 1] = "ROOM_SEARCH";
Landmark[Landmark["ROOM_LIST"] = 2] = "ROOM_LIST";
Landmark[Landmark["MESSAGE_COMPOSER_OR_HOME"] = 3] = "MESSAGE_COMPOSER_OR_HOME";
return Landmark;
}({});
const ORDERED_LANDMARKS = [Landmark.ACTIVE_SPACE_BUTTON, Landmark.ROOM_SEARCH, Landmark.ROOM_LIST, Landmark.MESSAGE_COMPOSER_OR_HOME];
/**
* The landmarks are cycled through in the following order:
* ACTIVE_SPACE_BUTTON <-> ROOM_SEARCH <-> ROOM_LIST <-> MESSAGE_COMPOSER/HOME <-> ACTIVE_SPACE_BUTTON
*/
class LandmarkNavigation {
/**
* Get the next/previous landmark that must be focused from a given landmark
* @param currentLandmark The current landmark
* @param backwards If true, the landmark before currentLandmark in ORDERED_LANDMARKS is returned
* @returns The next landmark to focus
*/
static getLandmark(currentLandmark, backwards = false) {
const currentIndex = ORDERED_LANDMARKS.findIndex(l => l === currentLandmark);
const offset = backwards ? -1 : 1;
const newLandmark = ORDERED_LANDMARKS.at((currentIndex + offset) % ORDERED_LANDMARKS.length);
return newLandmark;
}
/**
* Focus the next landmark from a given landmark.
* This method will skip over any missing landmarks.
* @param currentLandmark The current landmark
* @param backwards If true, search the next landmark to the left in ORDERED_LANDMARKS
*/
static findAndFocusNextLandmark(currentLandmark, backwards = false) {
let landmark = currentLandmark;
let element = null;
while (element === null) {
landmark = LandmarkNavigation.getLandmark(landmark, backwards);
element = landmarkToDomElementMap[landmark]();
}
element?.focus({
focusVisible: true
});
}
}
/**
* The functions return:
* - The DOM element of the landmark if it exists
* - undefined if the DOM element exists but focus is given through an action
* - null if the landmark does not exist
*/
exports.LandmarkNavigation = LandmarkNavigation;
const landmarkToDomElementMap = {
[Landmark.ACTIVE_SPACE_BUTTON]: () => document.querySelector(".mx_SpaceButton_active"),
[Landmark.ROOM_SEARCH]: () => document.querySelector(".mx_RoomSearch"),
[Landmark.ROOM_LIST]: () => document.querySelector(".mx_RoomTile_selected") || document.querySelector(".mx_RoomTile"),
[Landmark.MESSAGE_COMPOSER_OR_HOME]: () => {
const isComposerOpen = !!document.querySelector(".mx_MessageComposer");
if (isComposerOpen) {
const inThread = !!document.activeElement?.closest(".mx_ThreadView");
_dispatcher.default.dispatch({
action: _actions.Action.FocusSendMessageComposer,
context: inThread ? _RoomContext.TimelineRenderingType.Thread : _RoomContext.TimelineRenderingType.Room
}, true);
// Special case where the element does exist but we focus it through an action.
return undefined;
} else {
return document.querySelector(".mx_HomePage");
}
}
};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_RoomContext","require","_actions","_dispatcher","_interopRequireDefault","Landmark","exports","ORDERED_LANDMARKS","ACTIVE_SPACE_BUTTON","ROOM_SEARCH","ROOM_LIST","MESSAGE_COMPOSER_OR_HOME","LandmarkNavigation","getLandmark","currentLandmark","backwards","currentIndex","findIndex","l","offset","newLandmark","at","length","findAndFocusNextLandmark","landmark","element","landmarkToDomElementMap","focus","focusVisible","document","querySelector","isComposerOpen","inThread","activeElement","closest","defaultDispatcher","dispatch","action","Action","FocusSendMessageComposer","context","TimelineRenderingType","Thread","Room","undefined"],"sources":["../../src/accessibility/LandmarkNavigation.ts"],"sourcesContent":["/*\n * Copyright 2024 New Vector Ltd.\n * Copyright 2024 The Matrix.org Foundation C.I.C.\n *\n * SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\n * Please see LICENSE files in the repository root for full details.\n */\n\nimport { TimelineRenderingType } from \"../contexts/RoomContext\";\nimport { Action } from \"../dispatcher/actions\";\nimport defaultDispatcher from \"../dispatcher/dispatcher\";\n\nexport const enum Landmark {\n    // This is the space/home button in the left panel.\n    ACTIVE_SPACE_BUTTON,\n    // This is the room filter in the left panel.\n    ROOM_SEARCH,\n    // This is the currently opened room/first room in the room list in the left panel.\n    ROOM_LIST,\n    // This is the message composer within the room if available or it is the welcome screen shown when no room is selected\n    MESSAGE_COMPOSER_OR_HOME,\n}\n\nconst ORDERED_LANDMARKS = [\n    Landmark.ACTIVE_SPACE_BUTTON,\n    Landmark.ROOM_SEARCH,\n    Landmark.ROOM_LIST,\n    Landmark.MESSAGE_COMPOSER_OR_HOME,\n];\n\n/**\n * The landmarks are cycled through in the following order:\n * ACTIVE_SPACE_BUTTON <-> ROOM_SEARCH <-> ROOM_LIST <-> MESSAGE_COMPOSER/HOME <-> ACTIVE_SPACE_BUTTON\n */\nexport class LandmarkNavigation {\n    /**\n     * Get the next/previous landmark that must be focused from a given landmark\n     * @param currentLandmark The current landmark\n     * @param backwards If true, the landmark before currentLandmark in ORDERED_LANDMARKS is returned\n     * @returns The next landmark to focus\n     */\n    private static getLandmark(currentLandmark: Landmark, backwards = false): Landmark {\n        const currentIndex = ORDERED_LANDMARKS.findIndex((l) => l === currentLandmark);\n        const offset = backwards ? -1 : 1;\n        const newLandmark = ORDERED_LANDMARKS.at((currentIndex + offset) % ORDERED_LANDMARKS.length)!;\n        return newLandmark;\n    }\n\n    /**\n     * Focus the next landmark from a given landmark.\n     * This method will skip over any missing landmarks.\n     * @param currentLandmark The current landmark\n     * @param backwards If true, search the next landmark to the left in ORDERED_LANDMARKS\n     */\n    public static findAndFocusNextLandmark(currentLandmark: Landmark, backwards = false): void {\n        let landmark = currentLandmark;\n        let element: HTMLElement | null | undefined = null;\n        while (element === null) {\n            landmark = LandmarkNavigation.getLandmark(landmark, backwards);\n            element = landmarkToDomElementMap[landmark]();\n        }\n        element?.focus({ focusVisible: true });\n    }\n}\n\n/**\n * The functions return:\n * - The DOM element of the landmark if it exists\n * - undefined if the DOM element exists but focus is given through an action\n * - null if the landmark does not exist\n */\nconst landmarkToDomElementMap: Record<Landmark, () => HTMLElement | null | undefined> = {\n    [Landmark.ACTIVE_SPACE_BUTTON]: () => document.querySelector<HTMLElement>(\".mx_SpaceButton_active\"),\n\n    [Landmark.ROOM_SEARCH]: () => document.querySelector<HTMLElement>(\".mx_RoomSearch\"),\n    [Landmark.ROOM_LIST]: () =>\n        document.querySelector<HTMLElement>(\".mx_RoomTile_selected\") ||\n        document.querySelector<HTMLElement>(\".mx_RoomTile\"),\n\n    [Landmark.MESSAGE_COMPOSER_OR_HOME]: () => {\n        const isComposerOpen = !!document.querySelector(\".mx_MessageComposer\");\n        if (isComposerOpen) {\n            const inThread = !!document.activeElement?.closest(\".mx_ThreadView\");\n            defaultDispatcher.dispatch(\n                {\n                    action: Action.FocusSendMessageComposer,\n                    context: inThread ? TimelineRenderingType.Thread : TimelineRenderingType.Room,\n                },\n                true,\n            );\n            // Special case where the element does exist but we focus it through an action.\n            return undefined;\n        } else {\n            return document.querySelector<HTMLElement>(\".mx_HomePage\");\n        }\n    },\n};\n"],"mappings":";;;;;;;AAQA,IAAAA,YAAA,GAAAC,OAAA;AACA,IAAAC,QAAA,GAAAD,OAAA;AACA,IAAAE,WAAA,GAAAC,sBAAA,CAAAH,OAAA;AAVA;AACA;AACA;AACA;AACA;AACA;AACA;AANA,IAYkBI,QAAQ,GAAAC,OAAA,CAAAD,QAAA,0BAARA,QAAQ;EAARA,QAAQ,CAARA,QAAQ;EAARA,QAAQ,CAARA,QAAQ;EAARA,QAAQ,CAARA,QAAQ;EAARA,QAAQ,CAARA,QAAQ;EAAA,OAARA,QAAQ;AAAA;AAW1B,MAAME,iBAAiB,GAAG,CACtBF,QAAQ,CAACG,mBAAmB,EAC5BH,QAAQ,CAACI,WAAW,EACpBJ,QAAQ,CAACK,SAAS,EAClBL,QAAQ,CAACM,wBAAwB,CACpC;;AAED;AACA;AACA;AACA;AACO,MAAMC,kBAAkB,CAAC;EAC5B;AACJ;AACA;AACA;AACA;AACA;EACI,OAAeC,WAAWA,CAACC,eAAyB,EAAEC,SAAS,GAAG,KAAK,EAAY;IAC/E,MAAMC,YAAY,GAAGT,iBAAiB,CAACU,SAAS,CAAEC,CAAC,IAAKA,CAAC,KAAKJ,eAAe,CAAC;IAC9E,MAAMK,MAAM,GAAGJ,SAAS,GAAG,CAAC,CAAC,GAAG,CAAC;IACjC,MAAMK,WAAW,GAAGb,iBAAiB,CAACc,EAAE,CAAC,CAACL,YAAY,GAAGG,MAAM,IAAIZ,iBAAiB,CAACe,MAAM,CAAE;IAC7F,OAAOF,WAAW;EACtB;;EAEA;AACJ;AACA;AACA;AACA;AACA;EACI,OAAcG,wBAAwBA,CAACT,eAAyB,EAAEC,SAAS,GAAG,KAAK,EAAQ;IACvF,IAAIS,QAAQ,GAAGV,eAAe;IAC9B,IAAIW,OAAuC,GAAG,IAAI;IAClD,OAAOA,OAAO,KAAK,IAAI,EAAE;MACrBD,QAAQ,GAAGZ,kBAAkB,CAACC,WAAW,CAACW,QAAQ,EAAET,SAAS,CAAC;MAC9DU,OAAO,GAAGC,uBAAuB,CAACF,QAAQ,CAAC,CAAC,CAAC;IACjD;IACAC,OAAO,EAAEE,KAAK,CAAC;MAAEC,YAAY,EAAE;IAAK,CAAC,CAAC;EAC1C;AACJ;;AAEA;AACA;AACA;AACA;AACA;AACA;AALAtB,OAAA,CAAAM,kBAAA,GAAAA,kBAAA;AAMA,MAAMc,uBAA+E,GAAG;EACpF,CAACrB,QAAQ,CAACG,mBAAmB,GAAG,MAAMqB,QAAQ,CAACC,aAAa,CAAc,wBAAwB,CAAC;EAEnG,CAACzB,QAAQ,CAACI,WAAW,GAAG,MAAMoB,QAAQ,CAACC,aAAa,CAAc,gBAAgB,CAAC;EACnF,CAACzB,QAAQ,CAACK,SAAS,GAAG,MAClBmB,QAAQ,CAACC,aAAa,CAAc,uBAAuB,CAAC,IAC5DD,QAAQ,CAACC,aAAa,CAAc,cAAc,CAAC;EAEvD,CAACzB,QAAQ,CAACM,wBAAwB,GAAG,MAAM;IACvC,MAAMoB,cAAc,GAAG,CAAC,CAACF,QAAQ,CAACC,aAAa,CAAC,qBAAqB,CAAC;IACtE,IAAIC,cAAc,EAAE;MAChB,MAAMC,QAAQ,GAAG,CAAC,CAACH,QAAQ,CAACI,aAAa,EAAEC,OAAO,CAAC,gBAAgB,CAAC;MACpEC,mBAAiB,CAACC,QAAQ,CACtB;QACIC,MAAM,EAAEC,eAAM,CAACC,wBAAwB;QACvCC,OAAO,EAAER,QAAQ,GAAGS,kCAAqB,CAACC,MAAM,GAAGD,kCAAqB,CAACE;MAC7E,CAAC,EACD,IACJ,CAAC;MACD;MACA,OAAOC,SAAS;IACpB,CAAC,MAAM;MACH,OAAOf,QAAQ,CAACC,aAAa,CAAc,cAAc,CAAC;IAC9D;EACJ;AACJ,CAAC","ignoreList":[]}