Merge pull request #111 from samuelthomas2774/theme-scss
Add theme configuration to SCSS variables and add compile caching
This commit is contained in:
commit
c502ee8909
|
@ -42,7 +42,7 @@ export default class {
|
||||||
message: `Failed to load ${dir}`,
|
message: `Failed to load ${dir}`,
|
||||||
err
|
err
|
||||||
}));
|
}));
|
||||||
|
|
||||||
Logger.err(this.moduleName, err);
|
Logger.err(this.moduleName, err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -134,6 +134,7 @@ export default class {
|
||||||
});
|
});
|
||||||
return config;
|
return config;
|
||||||
});
|
});
|
||||||
|
userConfig.css = readUserConfig.css || null;
|
||||||
// userConfig.config = readUserConfig.config;
|
// userConfig.config = readUserConfig.config;
|
||||||
} catch (err) { /*We don't care if this fails it either means that user config doesn't exist or there's something wrong with it so we revert to default config*/
|
} catch (err) { /*We don't care if this fails it either means that user config doesn't exist or there's something wrong with it so we revert to default config*/
|
||||||
|
|
||||||
|
|
|
@ -36,48 +36,87 @@ class Theme {
|
||||||
get dirName() { return this.paths.dirName }
|
get dirName() { return this.paths.dirName }
|
||||||
get enabled() { return this.userConfig.enabled }
|
get enabled() { return this.userConfig.enabled }
|
||||||
get themeConfig() { return this.userConfig.config }
|
get themeConfig() { return this.userConfig.config }
|
||||||
get css() { return this.__themeInternals.css }
|
get css() { return this.userConfig.css }
|
||||||
get id() { return this.name.toLowerCase().replace(/\s+/g, '-') }
|
get id() { return this.name.toLowerCase().replace(/\s+/g, '-') }
|
||||||
|
|
||||||
async saveSettings(newSettings) {
|
async saveSettings(newSettings) {
|
||||||
if (newSettings) {
|
for (let category of newSettings) {
|
||||||
for (let category of newSettings) {
|
const oldCategory = this.themeConfig.find(c => c.category === category.category);
|
||||||
const oldCategory = this.themeConfig.find(c => c.category === category.category);
|
for (let setting of category.settings) {
|
||||||
for (let setting of category.settings) {
|
const oldSetting = oldCategory.settings.find(s => s.id === setting.id);
|
||||||
const oldSetting = oldCategory.settings.find(s => s.id === setting.id);
|
if (oldSetting.value === setting.value) continue;
|
||||||
if (oldSetting.value === setting.value) continue;
|
oldSetting.value = setting.value;
|
||||||
oldSetting.value = setting.value;
|
if (this.settingChanged) this.settingChanged(category.category, setting.id, setting.value);
|
||||||
if (this.settingChanged) this.settingChanged(category.category, setting.id, setting.value);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
// As the theme's configuration has changed it needs recompiling
|
||||||
await FileUtils.writeFile(`${this.themePath}/user.config.json`, JSON.stringify({ enabled: this.enabled, config: this.themeConfig }));
|
// When the compiled CSS has been save it will also save the configuration
|
||||||
} catch (err) {
|
await this.recompile();
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.settingsChanged) this.settingsChanged(this.themeConfig);
|
if (this.settingsChanged) this.settingsChanged(this.themeConfig);
|
||||||
|
|
||||||
return this.pluginConfig;
|
return this.pluginConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async saveConfiguration() {
|
||||||
|
try {
|
||||||
|
await FileUtils.writeFile(`${this.themePath}/user.config.json`, JSON.stringify({
|
||||||
|
enabled: this.enabled,
|
||||||
|
config: this.themeConfig,
|
||||||
|
css: this.css
|
||||||
|
}));
|
||||||
|
} catch (err) {
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
enable() {
|
enable() {
|
||||||
this.userConfig.enabled = true;
|
if (!this.enabled) {
|
||||||
this.saveSettings();
|
this.userConfig.enabled = true;
|
||||||
|
this.saveConfiguration();
|
||||||
|
}
|
||||||
DOM.injectTheme(this.css, this.id);
|
DOM.injectTheme(this.css, this.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
disable() {
|
disable() {
|
||||||
this.userConfig.enabled = false;
|
this.userConfig.enabled = false;
|
||||||
this.saveSettings();
|
this.saveConfiguration();
|
||||||
DOM.deleteTheme(this.id);
|
DOM.deleteTheme(this.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async compile() {
|
||||||
|
console.log('Compiling CSS');
|
||||||
|
|
||||||
|
let css = '';
|
||||||
|
if (this.info.type === 'sass') {
|
||||||
|
css = await ClientIPC.send('bd-compileSass', {
|
||||||
|
scss: ThemeManager.getConfigAsSCSS(this.themeConfig),
|
||||||
|
path: this.paths.mainPath
|
||||||
|
});
|
||||||
|
console.log(css);
|
||||||
|
} else {
|
||||||
|
css = await FileUtils.readFile(this.paths.mainPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
return css;
|
||||||
|
}
|
||||||
|
|
||||||
|
async recompile() {
|
||||||
|
const css = await this.compile();
|
||||||
|
this.userConfig.css = css;
|
||||||
|
|
||||||
|
await this.saveConfiguration();
|
||||||
|
|
||||||
|
if (this.enabled) {
|
||||||
|
DOM.deleteTheme(this.id);
|
||||||
|
DOM.injectTheme(this.css, this.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class extends ContentManager {
|
export default class ThemeManager extends ContentManager {
|
||||||
|
|
||||||
static get localThemes() {
|
static get localThemes() {
|
||||||
return this.localContent;
|
return this.localContent;
|
||||||
|
@ -94,14 +133,16 @@ export default class extends ContentManager {
|
||||||
static get loadContent() { return this.loadTheme }
|
static get loadContent() { return this.loadTheme }
|
||||||
static async loadTheme(paths, configs, info, main) {
|
static async loadTheme(paths, configs, info, main) {
|
||||||
try {
|
try {
|
||||||
let css = '';
|
const instance = new Theme({
|
||||||
if (info.type === 'sass') {
|
configs, info, main,
|
||||||
css = await ClientIPC.send('bd-compileSass', { path: paths.mainPath });
|
paths: {
|
||||||
} else {
|
contentPath: paths.contentPath,
|
||||||
css = await FileUtils.readFile(paths.mainPath);
|
dirName: paths.dirName,
|
||||||
}
|
mainPath: paths.mainPath
|
||||||
const instance = new Theme({ configs, info, main, paths: { contentPath: paths.contentPath, dirName: paths.dirName }, css });
|
}
|
||||||
if (instance.enabled) instance.enable();
|
});
|
||||||
|
if (!instance.css) instance.recompile();
|
||||||
|
else if (instance.enabled) instance.enable();
|
||||||
return instance;
|
return instance;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
throw err;
|
throw err;
|
||||||
|
@ -116,4 +157,28 @@ export default class extends ContentManager {
|
||||||
theme.disable();
|
theme.disable();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static getConfigAsSCSS(config) {
|
||||||
|
const variables = [];
|
||||||
|
|
||||||
|
for (let category of config) {
|
||||||
|
for (let setting of category.settings) {
|
||||||
|
let scss_name = null;
|
||||||
|
let scss_value = null;
|
||||||
|
let scss_line = null;
|
||||||
|
|
||||||
|
if (typeof setting.value == 'string')
|
||||||
|
scss_value = setting.scss_raw ? setting.value : '\'' + setting.value.replace(/\\/g, '\\\\').replace(/'/g, '\\\'') + '\'';
|
||||||
|
else if (typeof setting.value === 'boolean' || typeof setting.value === 'number')
|
||||||
|
scss_value = setting.value.toString();
|
||||||
|
|
||||||
|
scss_name = setting.id.replace(/[^a-zA-Z0-9-]/g, '-').replace(/--/g, '-');
|
||||||
|
|
||||||
|
scss_line = `$${scss_name}: ${scss_value};`;
|
||||||
|
variables.push(scss_line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return variables.join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,13 +5,21 @@
|
||||||
"version": 1.0,
|
"version": 1.0,
|
||||||
"description": "Example Theme 1 Description",
|
"description": "Example Theme 1 Description",
|
||||||
"icon": "",
|
"icon": "",
|
||||||
"type": "sass"
|
"type": "sass"
|
||||||
},
|
},
|
||||||
"main": "index.scss",
|
"main": "index.scss",
|
||||||
"defaultConfig": [
|
"defaultConfig": [
|
||||||
{
|
{
|
||||||
"category": "default",
|
"category": "default",
|
||||||
"settings": [
|
"settings": [
|
||||||
|
{
|
||||||
|
"id": "divBg",
|
||||||
|
"type": "text",
|
||||||
|
"value": "#00ff00",
|
||||||
|
"text": "Primary colour",
|
||||||
|
"hint": "A colour setting type would be nice here",
|
||||||
|
"scss_raw": true
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"id": "default-0",
|
"id": "default-0",
|
||||||
"type": "text",
|
"type": "text",
|
||||||
|
|
Loading…
Reference in New Issue