mirror of
https://github.com/Lissy93/dashy.git
synced 2025-07-27 15:44:27 +02:00
🚚 Re-writes all theming functionality. Much better now :)
This commit is contained in:
parent
0523c1933e
commit
18f6e4d268
@ -8,7 +8,7 @@
|
|||||||
:value="$store.getters.theme"
|
:value="$store.getters.theme"
|
||||||
class="theme-dropdown"
|
class="theme-dropdown"
|
||||||
:tabindex="-2"
|
:tabindex="-2"
|
||||||
@input="themeChanged"
|
@input="themeChangedInUI"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<IconPalette
|
<IconPalette
|
||||||
@ -28,18 +28,13 @@
|
|||||||
<script>
|
<script>
|
||||||
|
|
||||||
import CustomThemeMaker from '@/components/Settings/CustomThemeMaker';
|
import CustomThemeMaker from '@/components/Settings/CustomThemeMaker';
|
||||||
import {
|
|
||||||
LoadExternalTheme,
|
|
||||||
ApplyLocalTheme,
|
|
||||||
ApplyCustomVariables,
|
|
||||||
} from '@/utils/ThemeHelper';
|
|
||||||
import Defaults, { localStorageKeys } from '@/utils/defaults';
|
|
||||||
import Keys from '@/utils/StoreMutations';
|
import Keys from '@/utils/StoreMutations';
|
||||||
import ErrorHandler from '@/utils/ErrorHandler';
|
|
||||||
import IconPalette from '@/assets/interface-icons/config-color-palette.svg';
|
import IconPalette from '@/assets/interface-icons/config-color-palette.svg';
|
||||||
|
import ThemingMixin from '@/mixins/ThemingMixin';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'ThemeSelector',
|
name: 'ThemeSelector',
|
||||||
|
mixins: [ThemingMixin],
|
||||||
props: {
|
props: {
|
||||||
hidePallete: Boolean,
|
hidePallete: Boolean,
|
||||||
},
|
},
|
||||||
@ -47,101 +42,16 @@ export default {
|
|||||||
CustomThemeMaker,
|
CustomThemeMaker,
|
||||||
IconPalette,
|
IconPalette,
|
||||||
},
|
},
|
||||||
watch: {
|
|
||||||
/* When theme in VueX store changes, then update theme */
|
|
||||||
themeFromStore(newTheme) {
|
|
||||||
this.selectedTheme = newTheme;
|
|
||||||
this.updateTheme(newTheme);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
selectedTheme: '',
|
|
||||||
themeConfiguratorOpen: false, // Control the opening of theme config popup
|
themeConfiguratorOpen: false, // Control the opening of theme config popup
|
||||||
themeHelper: new LoadExternalTheme(),
|
|
||||||
ApplyLocalTheme,
|
|
||||||
ApplyCustomVariables,
|
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {},
|
||||||
/* Get appConfig from store */
|
|
||||||
appConfig() {
|
|
||||||
return this.$store.getters.appConfig;
|
|
||||||
},
|
|
||||||
/* Get users theme from store */
|
|
||||||
themeFromStore() {
|
|
||||||
return this.$store.getters.theme;
|
|
||||||
},
|
|
||||||
/* Combines all theme names (builtin and user defined) together */
|
|
||||||
themeNames: function themeNames() {
|
|
||||||
const externalThemeNames = Object.keys(this.externalThemes);
|
|
||||||
const specialThemes = ['custom'];
|
|
||||||
return [...this.extraThemeNames, ...externalThemeNames,
|
|
||||||
...Defaults.builtInThemes, ...specialThemes];
|
|
||||||
},
|
|
||||||
extraThemeNames() {
|
|
||||||
const userThemes = this.appConfig.cssThemes || [];
|
|
||||||
if (typeof userThemes === 'string') return [userThemes];
|
|
||||||
return userThemes;
|
|
||||||
},
|
|
||||||
/* Returns an array of links to external CSS from the Config */
|
|
||||||
externalThemes() {
|
|
||||||
const availibleThemes = {};
|
|
||||||
if (this.appConfig && this.appConfig.externalStyleSheet) {
|
|
||||||
const externals = this.appConfig.externalStyleSheet;
|
|
||||||
if (Array.isArray(externals)) {
|
|
||||||
externals.forEach((ext, i) => {
|
|
||||||
availibleThemes[`External Stylesheet ${i + 1}`] = ext;
|
|
||||||
});
|
|
||||||
} else if (typeof externals === 'string') {
|
|
||||||
availibleThemes['External Stylesheet'] = this.appConfig.externalStyleSheet;
|
|
||||||
} else {
|
|
||||||
ErrorHandler('External stylesheets must be of type string or string[]');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// availibleThemes.Default = '#';
|
|
||||||
return availibleThemes;
|
|
||||||
},
|
|
||||||
},
|
|
||||||
mounted() {
|
mounted() {
|
||||||
const initialTheme = this.getInitialTheme();
|
this.initializeTheme();
|
||||||
this.selectedTheme = initialTheme;
|
|
||||||
// Quicker loading, if the theme is local we can apply it immidiatley
|
|
||||||
if (this.isThemeLocal(initialTheme)) {
|
|
||||||
this.updateTheme(initialTheme);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it's an external stylesheet, then wait for promise to resolve
|
|
||||||
if (this.externalThemes && Object.entries(this.externalThemes).length > 0) {
|
|
||||||
const added = Object.keys(this.externalThemes).map(
|
|
||||||
name => this.themeHelper.add(name, this.externalThemes[name]),
|
|
||||||
);
|
|
||||||
// Once, added, then apply users initial theme
|
|
||||||
Promise.all(added).then(() => {
|
|
||||||
this.updateTheme(initialTheme);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
/* Called when dropdown changed
|
|
||||||
* Updates store, which will in turn update theme through watcher
|
|
||||||
*/
|
|
||||||
themeChanged() {
|
|
||||||
const pageId = this.$store.state.currentConfigInfo?.pageId || null;
|
|
||||||
this.$store.commit(Keys.SET_THEME, { theme: this.selectedTheme, pageId });
|
|
||||||
this.updateTheme(this.selectedTheme);
|
|
||||||
},
|
|
||||||
/* Returns the initial theme */
|
|
||||||
getInitialTheme() {
|
|
||||||
const localTheme = localStorage[localStorageKeys.THEME];
|
|
||||||
if (localTheme && localTheme !== 'undefined') return localTheme;
|
|
||||||
return this.appConfig.theme || Defaults.theme;
|
|
||||||
},
|
|
||||||
/* Determines if a given theme is local / not a custom user stylesheet */
|
|
||||||
isThemeLocal(themeToCheck) {
|
|
||||||
const localThemes = [...Defaults.builtInThemes, ...this.extraThemeNames];
|
|
||||||
return localThemes.includes(themeToCheck);
|
|
||||||
},
|
|
||||||
/* Opens the theme color configurator popup */
|
/* Opens the theme color configurator popup */
|
||||||
openThemeConfigurator() {
|
openThemeConfigurator() {
|
||||||
this.$store.commit(Keys.SET_MODAL_OPEN, true);
|
this.$store.commit(Keys.SET_MODAL_OPEN, true);
|
||||||
@ -154,24 +64,6 @@ export default {
|
|||||||
this.themeConfiguratorOpen = false;
|
this.themeConfiguratorOpen = false;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
/* Updates theme. Checks if the new theme is local or external,
|
|
||||||
and calls appropirate updating function. Updates local storage */
|
|
||||||
updateTheme(newTheme) {
|
|
||||||
if (newTheme === 'Default') {
|
|
||||||
this.resetToDefault();
|
|
||||||
this.themeHelper.theme = 'Default';
|
|
||||||
} else if (this.isThemeLocal(newTheme)) {
|
|
||||||
this.ApplyLocalTheme(newTheme);
|
|
||||||
} else {
|
|
||||||
this.themeHelper.theme = newTheme;
|
|
||||||
}
|
|
||||||
this.ApplyCustomVariables(newTheme);
|
|
||||||
// localStorage.setItem(localStorageKeys.THEME, newTheme);
|
|
||||||
},
|
|
||||||
/* Removes any applied themes */
|
|
||||||
resetToDefault() {
|
|
||||||
document.getElementsByTagName('html')[0].removeAttribute('data-theme');
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
@ -6,7 +6,6 @@ import Defaults, { localStorageKeys, iconCdns } from '@/utils/defaults';
|
|||||||
import Keys from '@/utils/StoreMutations';
|
import Keys from '@/utils/StoreMutations';
|
||||||
import { searchTiles } from '@/utils/Search';
|
import { searchTiles } from '@/utils/Search';
|
||||||
import { checkItemVisibility } from '@/utils/CheckItemVisibility';
|
import { checkItemVisibility } from '@/utils/CheckItemVisibility';
|
||||||
import { GetTheme, ApplyLocalTheme, ApplyCustomVariables } from '@/utils/ThemeHelper';
|
|
||||||
|
|
||||||
const HomeMixin = {
|
const HomeMixin = {
|
||||||
props: {
|
props: {
|
||||||
@ -40,16 +39,18 @@ const HomeMixin = {
|
|||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
async $route() {
|
async $route() {
|
||||||
await this.getConfigForRoute();
|
this.loadUpConfig();
|
||||||
this.setTheme();
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
async created() {
|
async created() {
|
||||||
// console.log(this.$router.currentRoute.path);
|
this.loadUpConfig();
|
||||||
const subPage = this.determineConfigFile();
|
|
||||||
await this.$store.dispatch(Keys.INITIALIZE_CONFIG, subPage);
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
/* When page loaded / sub-page changed, initiate config fetch */
|
||||||
|
async loadUpConfig() {
|
||||||
|
const subPage = this.determineConfigFile();
|
||||||
|
await this.$store.dispatch(Keys.INITIALIZE_CONFIG, subPage);
|
||||||
|
},
|
||||||
/* Based on the current route, get which config to display, null will use default */
|
/* Based on the current route, get which config to display, null will use default */
|
||||||
determineConfigFile() {
|
determineConfigFile() {
|
||||||
const pagePath = this.$router.currentRoute.path;
|
const pagePath = this.$router.currentRoute.path;
|
||||||
@ -75,9 +76,9 @@ const HomeMixin = {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
setTheme() {
|
setTheme() {
|
||||||
const theme = this.getSubPageTheme() || GetTheme();
|
// const theme = this.getSubPageTheme() || GetTheme();
|
||||||
ApplyLocalTheme(theme);
|
// ApplyLocalTheme(theme);
|
||||||
ApplyCustomVariables(theme);
|
// ApplyCustomVariables(theme);
|
||||||
},
|
},
|
||||||
updateModalVisibility(modalState) {
|
updateModalVisibility(modalState) {
|
||||||
this.$store.commit('SET_MODAL_OPEN', modalState);
|
this.$store.commit('SET_MODAL_OPEN', modalState);
|
||||||
|
@ -1,24 +1,26 @@
|
|||||||
// import {
|
/**
|
||||||
// LoadExternalTheme,
|
* This mixin can be extended by any component or view which needs to manage themes
|
||||||
// ApplyLocalTheme,
|
* It handles fetching and applying themes from the store, updating themes,
|
||||||
// ApplyCustomVariables,
|
* applying custom CSS variables and loading external stylesheets.
|
||||||
// } from '@/utils/ThemeHelper';
|
* */
|
||||||
import { builtInThemes, localStorageKeys, mainCssVars } from '@/utils/defaults';
|
|
||||||
import Keys from '@/utils/StoreMutations';
|
import Keys from '@/utils/StoreMutations';
|
||||||
import ErrorHandler from '@/utils/ErrorHandler';
|
import ErrorHandler from '@/utils/ErrorHandler';
|
||||||
|
import { builtInThemes, localStorageKeys, mainCssVars } from '@/utils/defaults';
|
||||||
|
|
||||||
const ThemingMixin = {
|
const ThemingMixin = {
|
||||||
data: () => ({
|
data: () => ({
|
||||||
selectedTheme: '',
|
selectedTheme: '', // Used only to bind current them to theme dropdown
|
||||||
// themeHelper: new LoadExternalTheme(),
|
|
||||||
}),
|
}),
|
||||||
computed: {
|
computed: {
|
||||||
|
/* This is the theme from the central store. When it changes, the UI will update */
|
||||||
themeFromStore() {
|
themeFromStore() {
|
||||||
return this.$store.getters.theme;
|
return this.$store.getters.theme;
|
||||||
},
|
},
|
||||||
appConfig() {
|
appConfig() {
|
||||||
return this.$store.getters.appConfig;
|
return this.$store.getters.appConfig;
|
||||||
},
|
},
|
||||||
|
/* Any extra user-defined themes, to add to dropdown */
|
||||||
extraThemeNames() {
|
extraThemeNames() {
|
||||||
const userThemes = this.appConfig?.cssThemes || [];
|
const userThemes = this.appConfig?.cssThemes || [];
|
||||||
if (typeof userThemes === 'string') return [userThemes];
|
if (typeof userThemes === 'string') return [userThemes];
|
||||||
@ -26,22 +28,22 @@ const ThemingMixin = {
|
|||||||
},
|
},
|
||||||
/* If user specified external stylesheet(s), format and return */
|
/* If user specified external stylesheet(s), format and return */
|
||||||
externalThemes() {
|
externalThemes() {
|
||||||
const availibleThemes = {};
|
const availableThemes = {};
|
||||||
if (this.appConfig?.externalStyleSheet) {
|
if (this.appConfig?.externalStyleSheet) {
|
||||||
const externals = this.appConfig.externalStyleSheet;
|
const externals = this.appConfig.externalStyleSheet;
|
||||||
if (Array.isArray(externals)) {
|
if (Array.isArray(externals)) {
|
||||||
externals.forEach((ext, i) => {
|
externals.forEach((ext, i) => {
|
||||||
availibleThemes[`External Stylesheet ${i + 1}`] = ext;
|
availableThemes[`External Stylesheet ${i + 1}`] = ext;
|
||||||
});
|
});
|
||||||
} else if (typeof externals === 'string') {
|
} else if (typeof externals === 'string') {
|
||||||
availibleThemes['External Stylesheet'] = this.appConfig.externalStyleSheet;
|
availableThemes['External Stylesheet'] = this.appConfig.externalStyleSheet;
|
||||||
} else {
|
} else {
|
||||||
ErrorHandler('External stylesheets must be of type string or string[]');
|
ErrorHandler('External stylesheets must be of type string or string[]');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return availibleThemes;
|
return availableThemes;
|
||||||
},
|
},
|
||||||
/* Combines all theme names (builtin and user defined) together */
|
/* Combines all theme names for dropdown (built-in, user-defined and stylesheets) */
|
||||||
themeNames() {
|
themeNames() {
|
||||||
const externalThemeNames = Object.keys(this.externalThemes);
|
const externalThemeNames = Object.keys(this.externalThemes);
|
||||||
return [...this.extraThemeNames, ...externalThemeNames, ...builtInThemes];
|
return [...this.extraThemeNames, ...externalThemeNames, ...builtInThemes];
|
||||||
@ -50,6 +52,7 @@ const ThemingMixin = {
|
|||||||
watch: {
|
watch: {
|
||||||
/* When theme in VueX store changes, then update theme */
|
/* When theme in VueX store changes, then update theme */
|
||||||
themeFromStore(newTheme) {
|
themeFromStore(newTheme) {
|
||||||
|
this.resetToDefault();
|
||||||
this.selectedTheme = newTheme;
|
this.selectedTheme = newTheme;
|
||||||
this.updateTheme(newTheme);
|
this.updateTheme(newTheme);
|
||||||
},
|
},
|
||||||
@ -58,9 +61,9 @@ const ThemingMixin = {
|
|||||||
/* Called when user changes theme through the UI
|
/* Called when user changes theme through the UI
|
||||||
* Updates store, which will in turn update theme through watcher
|
* Updates store, which will in turn update theme through watcher
|
||||||
*/
|
*/
|
||||||
themeChanged() {
|
themeChangedInUI() {
|
||||||
this.$store.commit(Keys.SET_THEME, this.selectedTheme);
|
this.$store.commit(Keys.SET_THEME, this.selectedTheme); // Update store
|
||||||
this.updateTheme(this.selectedTheme);
|
this.updateTheme(this.selectedTheme); // Apply theme to UI
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
* Gets any custom styles the user has applied, wither from local storage, or from the config
|
* Gets any custom styles the user has applied, wither from local storage, or from the config
|
||||||
@ -87,26 +90,40 @@ const ThemingMixin = {
|
|||||||
if (htmlTag.hasAttribute('data-theme')) htmlTag.removeAttribute('data-theme');
|
if (htmlTag.hasAttribute('data-theme')) htmlTag.removeAttribute('data-theme');
|
||||||
htmlTag.setAttribute('data-theme', newTheme);
|
htmlTag.setAttribute('data-theme', newTheme);
|
||||||
},
|
},
|
||||||
|
/* If using an external stylesheet, load it in */
|
||||||
|
applyRemoteTheme(href) {
|
||||||
|
this.resetToDefault();
|
||||||
|
const element = document.createElement('link');
|
||||||
|
element.setAttribute('rel', 'stylesheet');
|
||||||
|
element.setAttribute('type', 'text/css');
|
||||||
|
element.setAttribute('id', 'user-defined-stylesheet');
|
||||||
|
element.setAttribute('href', href);
|
||||||
|
document.getElementsByTagName('head')[0].appendChild(element);
|
||||||
|
},
|
||||||
/* Determines if a given theme is local / not a custom user stylesheet */
|
/* Determines if a given theme is local / not a custom user stylesheet */
|
||||||
isThemeLocal(themeToCheck) {
|
isThemeLocal(themeToCheck) {
|
||||||
const localThemes = [...builtInThemes, ...this.extraThemeNames];
|
const localThemes = [...builtInThemes, ...this.extraThemeNames];
|
||||||
return localThemes.includes(themeToCheck);
|
return localThemes.includes(themeToCheck);
|
||||||
},
|
},
|
||||||
/* Updates theme. Checks if the new theme is local or external,
|
/* Updates theme. Checks if the new theme is local or external,
|
||||||
and calls appropirate updating function. Updates local storage */
|
and calls appropriate updating function. Updates local storage */
|
||||||
updateTheme(newTheme) {
|
updateTheme(newTheme) {
|
||||||
// this.themeHelper.theme = newTheme;
|
|
||||||
if (newTheme.toLowerCase() === 'default') {
|
if (newTheme.toLowerCase() === 'default') {
|
||||||
this.resetToDefault();
|
this.resetToDefault();
|
||||||
} else if (this.isThemeLocal(newTheme)) {
|
} else if (this.isThemeLocal(newTheme)) {
|
||||||
this.applyLocalTheme(newTheme);
|
this.applyLocalTheme(newTheme);
|
||||||
|
} else if (this.externalThemes[newTheme]) {
|
||||||
|
this.applyRemoteTheme(this.externalThemes[newTheme]);
|
||||||
}
|
}
|
||||||
this.applyCustomVariables(newTheme);
|
this.applyCustomVariables(newTheme);
|
||||||
},
|
},
|
||||||
/* Removes any applied themes */
|
/* Removes any applied themes, and deletes any externally loaded stylesheets */
|
||||||
resetToDefault() {
|
resetToDefault() {
|
||||||
|
const externalStyles = document.getElementById('user-defined-stylesheet');
|
||||||
|
if (externalStyles) document.getElementsByTagName('head')[0].removeChild(externalStyles);
|
||||||
document.getElementsByTagName('html')[0].removeAttribute('data-theme');
|
document.getElementsByTagName('html')[0].removeAttribute('data-theme');
|
||||||
},
|
},
|
||||||
|
/* Call within mounted hook within a page to apply the correct theme */
|
||||||
initializeTheme() {
|
initializeTheme() {
|
||||||
const initialTheme = this.themeFromStore;
|
const initialTheme = this.themeFromStore;
|
||||||
this.selectedTheme = initialTheme;
|
this.selectedTheme = initialTheme;
|
||||||
@ -115,13 +132,7 @@ const ThemingMixin = {
|
|||||||
if (this.isThemeLocal(initialTheme)) {
|
if (this.isThemeLocal(initialTheme)) {
|
||||||
this.updateTheme(initialTheme);
|
this.updateTheme(initialTheme);
|
||||||
} else if (hasExternal) {
|
} else if (hasExternal) {
|
||||||
const added = Object.keys(this.externalThemes).map(
|
this.applyRemoteTheme(this.externalThemes[initialTheme]);
|
||||||
name => this.themeHelper.add(name, this.externalThemes[name]),
|
|
||||||
);
|
|
||||||
// Once, added, then apply users initial theme
|
|
||||||
Promise.all(added).then(() => {
|
|
||||||
this.updateTheme(initialTheme);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -349,7 +349,6 @@ const store = new Vuex.Store({
|
|||||||
const configContent = yaml.load(response.data);
|
const configContent = yaml.load(response.data);
|
||||||
// Certain values must be inherited from root config
|
// Certain values must be inherited from root config
|
||||||
const theme = configContent?.appConfig?.theme || rootConfig?.appConfig?.theme;
|
const theme = configContent?.appConfig?.theme || rootConfig?.appConfig?.theme;
|
||||||
console.log(theme);
|
|
||||||
configContent.appConfig = rootConfig.appConfig;
|
configContent.appConfig = rootConfig.appConfig;
|
||||||
configContent.pages = rootConfig.pages;
|
configContent.pages = rootConfig.pages;
|
||||||
configContent.appConfig.theme = theme;
|
configContent.appConfig.theme = theme;
|
||||||
|
@ -4,7 +4,6 @@ import { languages } from '@/utils/languages';
|
|||||||
import {
|
import {
|
||||||
visibleComponents,
|
visibleComponents,
|
||||||
localStorageKeys,
|
localStorageKeys,
|
||||||
theme as defaultTheme,
|
|
||||||
language as defaultLanguage,
|
language as defaultLanguage,
|
||||||
} from '@/utils/defaults';
|
} from '@/utils/defaults';
|
||||||
import ErrorHandler from '@/utils/ErrorHandler';
|
import ErrorHandler from '@/utils/ErrorHandler';
|
||||||
@ -26,6 +25,13 @@ export const makePageSlug = (pageName, pageType) => {
|
|||||||
return `/${pageType}/${formattedName}`;
|
return `/${pageType}/${formattedName}`;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* Put fetch path for additional configs in correct format */
|
||||||
|
export const formatConfigPath = (configPath) => {
|
||||||
|
if (configPath.includes('http')) return configPath;
|
||||||
|
if (configPath.substring(0, 1) !== '/') return `/${configPath}`;
|
||||||
|
return configPath;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initiates the Accumulator class and generates a complete config object
|
* Initiates the Accumulator class and generates a complete config object
|
||||||
* Self-executing function, returns the full user config as a JSON object
|
* Self-executing function, returns the full user config as a JSON object
|
||||||
@ -67,27 +73,6 @@ export const componentVisibility = (appConfig) => {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the users saved theme, first looks for local storage theme,
|
|
||||||
* then looks at user's appConfig, and finally checks the defaults
|
|
||||||
* @returns {string} Name of theme to apply
|
|
||||||
*/
|
|
||||||
export const getTheme = () => {
|
|
||||||
const localTheme = localStorage[localStorageKeys.THEME];
|
|
||||||
const appConfigTheme = config.appConfig.theme;
|
|
||||||
return localTheme || appConfigTheme || defaultTheme;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets any custom styles the user has applied, wither from local storage, or from the config
|
|
||||||
* @returns {object} An array of objects, one for each theme, containing kvps for variables
|
|
||||||
*/
|
|
||||||
export const getCustomColors = () => {
|
|
||||||
const localColors = JSON.parse(localStorage[localStorageKeys.CUSTOM_COLORS] || '{}');
|
|
||||||
const configColors = config.appConfig.customColors || {};
|
|
||||||
return Object.assign(configColors, localColors);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a list of items which the user has assigned a hotkey to
|
* Returns a list of items which the user has assigned a hotkey to
|
||||||
* So that when the hotkey is pressed, the app/ service can be launched
|
* So that when the hotkey is pressed, the app/ service can be launched
|
||||||
|
@ -5,7 +5,7 @@ const KEY_NAMES = [
|
|||||||
'INITIALIZE_MULTI_PAGE_CONFIG',
|
'INITIALIZE_MULTI_PAGE_CONFIG',
|
||||||
'SET_CONFIG',
|
'SET_CONFIG',
|
||||||
'SET_ROOT_CONFIG',
|
'SET_ROOT_CONFIG',
|
||||||
'SET_REMOTE_CONFIG',
|
'SET_CONFIG_ID',
|
||||||
'SET_CURRENT_SUB_PAGE',
|
'SET_CURRENT_SUB_PAGE',
|
||||||
'SET_MODAL_OPEN',
|
'SET_MODAL_OPEN',
|
||||||
'SET_LANGUAGE',
|
'SET_LANGUAGE',
|
||||||
|
@ -1,72 +0,0 @@
|
|||||||
import ErrorHandler from '@/utils/ErrorHandler';
|
|
||||||
import { getTheme, getCustomColors } from '@/utils/ConfigHelpers';
|
|
||||||
import { mainCssVars } from '@/utils/defaults';
|
|
||||||
|
|
||||||
/* Returns users current theme */
|
|
||||||
export const GetTheme = () => getTheme();
|
|
||||||
|
|
||||||
/* Gets user custom color preferences for current theme, and applies to DOM */
|
|
||||||
export const ApplyCustomVariables = (theme) => {
|
|
||||||
mainCssVars.forEach((vName) => { document.documentElement.style.removeProperty(`--${vName}`); });
|
|
||||||
const themeColors = getCustomColors()[theme];
|
|
||||||
if (themeColors) {
|
|
||||||
Object.keys(themeColors).forEach((customVar) => {
|
|
||||||
document.documentElement.style.setProperty(`--${customVar}`, themeColors[customVar]);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Sets the theme, by updating data-theme attribute on the html tag */
|
|
||||||
export const ApplyLocalTheme = (newTheme) => {
|
|
||||||
const htmlTag = document.getElementsByTagName('html')[0];
|
|
||||||
if (htmlTag.hasAttribute('data-theme')) htmlTag.removeAttribute('data-theme');
|
|
||||||
htmlTag.setAttribute('data-theme', newTheme);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A function for pre-loading, and easy switching of external stylesheets
|
|
||||||
* External CSS is preloaded to avoid FOUC
|
|
||||||
*/
|
|
||||||
export const LoadExternalTheme = function th() {
|
|
||||||
/* Preload selected external theme */
|
|
||||||
const preloadTheme = (href) => {
|
|
||||||
const link = document.createElement('link');
|
|
||||||
link.rel = 'stylesheet';
|
|
||||||
link.type = 'text/css';
|
|
||||||
link.href = href;
|
|
||||||
document.head.appendChild(link);
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
link.onload = e => {
|
|
||||||
const { sheet } = e.target;
|
|
||||||
sheet.disabled = true;
|
|
||||||
resolve(sheet);
|
|
||||||
};
|
|
||||||
link.onerror = reject;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Check theme is selected, and it exists */
|
|
||||||
const checkTheme = (themes, name) => {
|
|
||||||
if ((!name) || (name !== 'custom' && !themes[name])) {
|
|
||||||
ErrorHandler(`Theme: '${name || '[not selected]'}' does not exist.`);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Disable all but selected theme */
|
|
||||||
const selectTheme = (themes, name) => {
|
|
||||||
if (checkTheme(themes, name)) {
|
|
||||||
const t = themes; // To avoid ESLint complaining about mutating a param
|
|
||||||
Object.keys(themes).forEach(n => { t[n].disabled = (n !== name); });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const themes = {};
|
|
||||||
|
|
||||||
return {
|
|
||||||
add(name, href) { return preloadTheme(href).then(s => { themes[name] = s; }); },
|
|
||||||
set theme(name) { selectTheme(themes, name); },
|
|
||||||
get theme() { return Object.keys(themes).find(n => !themes[n].disabled); },
|
|
||||||
};
|
|
||||||
};
|
|
@ -19,7 +19,6 @@ import WebContent from '@/components/Workspace/WebContent';
|
|||||||
import WidgetView from '@/components/Workspace/WidgetView';
|
import WidgetView from '@/components/Workspace/WidgetView';
|
||||||
import MultiTaskingWebComtent from '@/components/Workspace/MultiTaskingWebComtent';
|
import MultiTaskingWebComtent from '@/components/Workspace/MultiTaskingWebComtent';
|
||||||
import Defaults from '@/utils/defaults';
|
import Defaults from '@/utils/defaults';
|
||||||
import { GetTheme, ApplyLocalTheme, ApplyCustomVariables } from '@/utils/ThemeHelper';
|
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'Workspace',
|
name: 'Workspace',
|
||||||
@ -27,9 +26,6 @@ export default {
|
|||||||
data: () => ({
|
data: () => ({
|
||||||
url: '',
|
url: '',
|
||||||
widgets: null,
|
widgets: null,
|
||||||
GetTheme,
|
|
||||||
ApplyLocalTheme,
|
|
||||||
ApplyCustomVariables,
|
|
||||||
}),
|
}),
|
||||||
computed: {
|
computed: {
|
||||||
sections() {
|
sections() {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user