UNPKG

matrix-react-sdk

Version:
158 lines (123 loc) 18.5 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _SettingsStore = _interopRequireDefault(require("../SettingsStore")); var _dispatcher = _interopRequireDefault(require("../../dispatcher/dispatcher")); var _actions = require("../../dispatcher/actions"); var _ThemeController = _interopRequireDefault(require("../controllers/ThemeController")); var _theme = require("../../theme"); var _SettingLevel = require("../SettingLevel"); /* Copyright 2019 Michael Telatynski <7t3chguy@gmail.com> Copyright 2019, 2020 The Matrix.org Foundation C.I.C. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ class ThemeWatcher { constructor() { (0, _defineProperty2.default)(this, "themeWatchRef", void 0); (0, _defineProperty2.default)(this, "systemThemeWatchRef", void 0); (0, _defineProperty2.default)(this, "dispatcherRef", void 0); (0, _defineProperty2.default)(this, "preferDark", void 0); (0, _defineProperty2.default)(this, "preferLight", void 0); (0, _defineProperty2.default)(this, "currentTheme", void 0); (0, _defineProperty2.default)(this, "onChange", () => { this.recheck(); }); (0, _defineProperty2.default)(this, "onAction", (payload /*: ActionPayload*/ ) => { if (payload.action === _actions.Action.RecheckTheme) { // XXX forceTheme this.recheck(payload.forceTheme); } }); this.themeWatchRef = null; this.systemThemeWatchRef = null; this.dispatcherRef = null; // we have both here as each may either match or not match, so by having both // we can get the tristate of dark/light/unsupported this.preferDark = global.matchMedia("(prefers-color-scheme: dark)"); this.preferLight = global.matchMedia("(prefers-color-scheme: light)"); this.currentTheme = this.getEffectiveTheme(); } start() { this.themeWatchRef = _SettingsStore.default.watchSetting("theme", null, this.onChange); this.systemThemeWatchRef = _SettingsStore.default.watchSetting("use_system_theme", null, this.onChange); if (this.preferDark.addEventListener) { this.preferDark.addEventListener('change', this.onChange); this.preferLight.addEventListener('change', this.onChange); } this.dispatcherRef = _dispatcher.default.register(this.onAction); } stop() { if (this.preferDark.addEventListener) { this.preferDark.removeEventListener('change', this.onChange); this.preferLight.removeEventListener('change', this.onChange); } _SettingsStore.default.unwatchSetting(this.systemThemeWatchRef); _SettingsStore.default.unwatchSetting(this.themeWatchRef); _dispatcher.default.unregister(this.dispatcherRef); } // XXX: forceTheme param added here as local echo appears to be unreliable // https://github.com/vector-im/element-web/issues/11443 recheck(forceTheme /*: string*/ ) { const oldTheme = this.currentTheme; this.currentTheme = forceTheme === undefined ? this.getEffectiveTheme() : forceTheme; if (oldTheme !== this.currentTheme) { (0, _theme.setTheme)(this.currentTheme); } } getEffectiveTheme() /*: string*/ { // Dev note: Much of this logic is replicated in the AppearanceUserSettingsTab // XXX: checking the isLight flag here makes checking it in the ThemeController // itself completely redundant since we just override the result here and we're // now effectively just using the ThemeController as a place to store the static // variable. The system theme setting probably ought to have an equivalent // controller that honours the same flag, although probablt better would be to // have the theme logic in one place rather than split between however many // different places. if (_ThemeController.default.isLogin) return 'light'; // If the user has specifically enabled the system matching option (excluding default), // then use that over anything else. We pick the lowest possible level for the setting // to ensure the ordering otherwise works. const systemThemeExplicit = _SettingsStore.default.getValueAt(_SettingLevel.SettingLevel.DEVICE, "use_system_theme", null, false, true); if (systemThemeExplicit) { console.log("returning explicit system theme"); if (this.preferDark.matches) return 'dark'; if (this.preferLight.matches) return 'light'; } // If the user has specifically enabled the theme (without the system matching option being // enabled specifically and excluding the default), use that theme. We pick the lowest possible // level for the setting to ensure the ordering otherwise works. const themeExplicit = _SettingsStore.default.getValueAt(_SettingLevel.SettingLevel.DEVICE, "theme", null, false, true); if (themeExplicit) { console.log("returning explicit theme: " + themeExplicit); return themeExplicit; } // If the user hasn't really made a preference in either direction, assume the defaults of the // settings and use those. if (_SettingsStore.default.getValue('use_system_theme')) { if (this.preferDark.matches) return 'dark'; if (this.preferLight.matches) return 'light'; } console.log("returning theme value"); return _SettingsStore.default.getValue('theme'); } isSystemThemeSupported() { return this.preferDark.matches || this.preferLight.matches; } } exports.default = ThemeWatcher; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/settings/watchers/ThemeWatcher.ts"],"names":["ThemeWatcher","constructor","recheck","payload","action","Action","RecheckTheme","forceTheme","themeWatchRef","systemThemeWatchRef","dispatcherRef","preferDark","global","matchMedia","preferLight","currentTheme","getEffectiveTheme","start","SettingsStore","watchSetting","onChange","addEventListener","dis","register","onAction","stop","removeEventListener","unwatchSetting","unregister","oldTheme","undefined","ThemeController","isLogin","systemThemeExplicit","getValueAt","SettingLevel","DEVICE","console","log","matches","themeExplicit","getValue","isSystemThemeSupported"],"mappings":";;;;;;;;;;;AAiBA;;AACA;;AACA;;AACA;;AACA;;AAEA;;AAvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAUe,MAAMA,YAAN,CAAmB;AAU9BC,EAAAA,WAAW,GAAG;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,oDAiCK,MAAM;AACrB,WAAKC,OAAL;AACH,KAnCa;AAAA,oDAqCK,CAACC;AAAD;AAAA,SAA4B;AAC3C,UAAIA,OAAO,CAACC,MAAR,KAAmBC,gBAAOC,YAA9B,EAA4C;AACxC;AACA,aAAKJ,OAAL,CAAaC,OAAO,CAACI,UAArB;AACH;AACJ,KA1Ca;AACV,SAAKC,aAAL,GAAqB,IAArB;AACA,SAAKC,mBAAL,GAA2B,IAA3B;AACA,SAAKC,aAAL,GAAqB,IAArB,CAHU,CAKV;AACA;;AACA,SAAKC,UAAL,GAAwBC,MAAN,CAAcC,UAAd,CAAyB,8BAAzB,CAAlB;AACA,SAAKC,WAAL,GAAyBF,MAAN,CAAcC,UAAd,CAAyB,+BAAzB,CAAnB;AAEA,SAAKE,YAAL,GAAoB,KAAKC,iBAAL,EAApB;AACH;;AAEMC,EAAAA,KAAP,GAAe;AACX,SAAKT,aAAL,GAAqBU,uBAAcC,YAAd,CAA2B,OAA3B,EAAoC,IAApC,EAA0C,KAAKC,QAA/C,CAArB;AACA,SAAKX,mBAAL,GAA2BS,uBAAcC,YAAd,CAA2B,kBAA3B,EAA+C,IAA/C,EAAqD,KAAKC,QAA1D,CAA3B;;AACA,QAAI,KAAKT,UAAL,CAAgBU,gBAApB,EAAsC;AAClC,WAAKV,UAAL,CAAgBU,gBAAhB,CAAiC,QAAjC,EAA2C,KAAKD,QAAhD;AACA,WAAKN,WAAL,CAAiBO,gBAAjB,CAAkC,QAAlC,EAA4C,KAAKD,QAAjD;AACH;;AACD,SAAKV,aAAL,GAAqBY,oBAAIC,QAAJ,CAAa,KAAKC,QAAlB,CAArB;AACH;;AAEMC,EAAAA,IAAP,GAAc;AACV,QAAI,KAAKd,UAAL,CAAgBU,gBAApB,EAAsC;AAClC,WAAKV,UAAL,CAAgBe,mBAAhB,CAAoC,QAApC,EAA8C,KAAKN,QAAnD;AACA,WAAKN,WAAL,CAAiBY,mBAAjB,CAAqC,QAArC,EAA+C,KAAKN,QAApD;AACH;;AACDF,2BAAcS,cAAd,CAA6B,KAAKlB,mBAAlC;;AACAS,2BAAcS,cAAd,CAA6B,KAAKnB,aAAlC;;AACAc,wBAAIM,UAAJ,CAAe,KAAKlB,aAApB;AACH;;AAaD;AACA;AACOR,EAAAA,OAAP,CAAeK;AAAf;AAAA,IAAoC;AAChC,UAAMsB,QAAQ,GAAG,KAAKd,YAAtB;AACA,SAAKA,YAAL,GAAoBR,UAAU,KAAKuB,SAAf,GAA2B,KAAKd,iBAAL,EAA3B,GAAsDT,UAA1E;;AACA,QAAIsB,QAAQ,KAAK,KAAKd,YAAtB,EAAoC;AAChC,2BAAS,KAAKA,YAAd;AACH;AACJ;;AAEMC,EAAAA,iBAAP;AAAA;AAAmC;AAC/B;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAIe,yBAAgBC,OAApB,EAA6B,OAAO,OAAP,CAVE,CAY/B;AACA;AACA;;AACA,UAAMC,mBAAmB,GAAGf,uBAAcgB,UAAd,CACxBC,2BAAaC,MADW,EACH,kBADG,EACiB,IADjB,EACuB,KADvB,EAC8B,IAD9B,CAA5B;;AAEA,QAAIH,mBAAJ,EAAyB;AACrBI,MAAAA,OAAO,CAACC,GAAR,CAAY,iCAAZ;AACA,UAAI,KAAK3B,UAAL,CAAgB4B,OAApB,EAA6B,OAAO,MAAP;AAC7B,UAAI,KAAKzB,WAAL,CAAiByB,OAArB,EAA8B,OAAO,OAAP;AACjC,KArB8B,CAuB/B;AACA;AACA;;;AACA,UAAMC,aAAa,GAAGtB,uBAAcgB,UAAd,CAClBC,2BAAaC,MADK,EACG,OADH,EACY,IADZ,EACkB,KADlB,EACyB,IADzB,CAAtB;;AAEA,QAAII,aAAJ,EAAmB;AACfH,MAAAA,OAAO,CAACC,GAAR,CAAY,+BAA+BE,aAA3C;AACA,aAAOA,aAAP;AACH,KA/B8B,CAiC/B;AACA;;;AACA,QAAItB,uBAAcuB,QAAd,CAAuB,kBAAvB,CAAJ,EAAgD;AAC5C,UAAI,KAAK9B,UAAL,CAAgB4B,OAApB,EAA6B,OAAO,MAAP;AAC7B,UAAI,KAAKzB,WAAL,CAAiByB,OAArB,EAA8B,OAAO,OAAP;AACjC;;AACDF,IAAAA,OAAO,CAACC,GAAR,CAAY,uBAAZ;AACA,WAAOpB,uBAAcuB,QAAd,CAAuB,OAAvB,CAAP;AACH;;AAEMC,EAAAA,sBAAP,GAAgC;AAC5B,WAAO,KAAK/B,UAAL,CAAgB4B,OAAhB,IAA2B,KAAKzB,WAAL,CAAiByB,OAAnD;AACH;;AA7G6B","sourcesContent":["/*\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2019, 2020 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport SettingsStore from '../SettingsStore';\nimport dis from '../../dispatcher/dispatcher';\nimport { Action } from '../../dispatcher/actions';\nimport ThemeController from \"../controllers/ThemeController\";\nimport { setTheme } from \"../../theme\";\nimport { ActionPayload } from '../../dispatcher/payloads';\nimport { SettingLevel } from \"../SettingLevel\";\n\nexport default class ThemeWatcher {\n    private themeWatchRef: string;\n    private systemThemeWatchRef: string;\n    private dispatcherRef: string;\n\n    private preferDark: MediaQueryList;\n    private preferLight: MediaQueryList;\n\n    private currentTheme: string;\n\n    constructor() {\n        this.themeWatchRef = null;\n        this.systemThemeWatchRef = null;\n        this.dispatcherRef = null;\n\n        // we have both here as each may either match or not match, so by having both\n        // we can get the tristate of dark/light/unsupported\n        this.preferDark = (<any>global).matchMedia(\"(prefers-color-scheme: dark)\");\n        this.preferLight = (<any>global).matchMedia(\"(prefers-color-scheme: light)\");\n\n        this.currentTheme = this.getEffectiveTheme();\n    }\n\n    public start() {\n        this.themeWatchRef = SettingsStore.watchSetting(\"theme\", null, this.onChange);\n        this.systemThemeWatchRef = SettingsStore.watchSetting(\"use_system_theme\", null, this.onChange);\n        if (this.preferDark.addEventListener) {\n            this.preferDark.addEventListener('change', this.onChange);\n            this.preferLight.addEventListener('change', this.onChange);\n        }\n        this.dispatcherRef = dis.register(this.onAction);\n    }\n\n    public stop() {\n        if (this.preferDark.addEventListener) {\n            this.preferDark.removeEventListener('change', this.onChange);\n            this.preferLight.removeEventListener('change', this.onChange);\n        }\n        SettingsStore.unwatchSetting(this.systemThemeWatchRef);\n        SettingsStore.unwatchSetting(this.themeWatchRef);\n        dis.unregister(this.dispatcherRef);\n    }\n\n    private onChange = () => {\n        this.recheck();\n    };\n\n    private onAction = (payload: ActionPayload) => {\n        if (payload.action === Action.RecheckTheme) {\n            // XXX forceTheme\n            this.recheck(payload.forceTheme);\n        }\n    };\n\n    // XXX: forceTheme param added here as local echo appears to be unreliable\n    // https://github.com/vector-im/element-web/issues/11443\n    public recheck(forceTheme?: string) {\n        const oldTheme = this.currentTheme;\n        this.currentTheme = forceTheme === undefined ? this.getEffectiveTheme() : forceTheme;\n        if (oldTheme !== this.currentTheme) {\n            setTheme(this.currentTheme);\n        }\n    }\n\n    public getEffectiveTheme(): string {\n        // Dev note: Much of this logic is replicated in the AppearanceUserSettingsTab\n\n        // XXX: checking the isLight flag here makes checking it in the ThemeController\n        // itself completely redundant since we just override the result here and we're\n        // now effectively just using the ThemeController as a place to store the static\n        // variable. The system theme setting probably ought to have an equivalent\n        // controller that honours the same flag, although probablt better would be to\n        // have the theme logic in one place rather than split between however many\n        // different places.\n        if (ThemeController.isLogin) return 'light';\n\n        // If the user has specifically enabled the system matching option (excluding default),\n        // then use that over anything else. We pick the lowest possible level for the setting\n        // to ensure the ordering otherwise works.\n        const systemThemeExplicit = SettingsStore.getValueAt(\n            SettingLevel.DEVICE, \"use_system_theme\", null, false, true);\n        if (systemThemeExplicit) {\n            console.log(\"returning explicit system theme\");\n            if (this.preferDark.matches) return 'dark';\n            if (this.preferLight.matches) return 'light';\n        }\n\n        // If the user has specifically enabled the theme (without the system matching option being\n        // enabled specifically and excluding the default), use that theme. We pick the lowest possible\n        // level for the setting to ensure the ordering otherwise works.\n        const themeExplicit = SettingsStore.getValueAt(\n            SettingLevel.DEVICE, \"theme\", null, false, true);\n        if (themeExplicit) {\n            console.log(\"returning explicit theme: \" + themeExplicit);\n            return themeExplicit;\n        }\n\n        // If the user hasn't really made a preference in either direction, assume the defaults of the\n        // settings and use those.\n        if (SettingsStore.getValue('use_system_theme')) {\n            if (this.preferDark.matches) return 'dark';\n            if (this.preferLight.matches) return 'light';\n        }\n        console.log(\"returning theme value\");\n        return SettingsStore.getValue('theme');\n    }\n\n    public isSystemThemeSupported() {\n        return this.preferDark.matches || this.preferLight.matches;\n    }\n}\n"]}