UNPKG

matrix-react-sdk

Version:
90 lines (86 loc) 12.9 kB
"use strict"; 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":[]}