|
|
|
@ -1,5 +1,6 @@
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
|
|
const _ = require('lodash');
|
|
|
|
|
const winston = require('winston');
|
|
|
|
|
const nconf = require('nconf');
|
|
|
|
|
const fs = require('fs');
|
|
|
|
@ -48,11 +49,19 @@ const buildImports = {
|
|
|
|
|
function boostrapImport(themeData) {
|
|
|
|
|
// see https://getbootstrap.com/docs/5.0/customize/sass/#variable-defaults
|
|
|
|
|
// for an explanation of this order and https://bootswatch.com/help/
|
|
|
|
|
const { bootswatchSkin, bsVariables } = themeData;
|
|
|
|
|
const { bootswatchSkin, bsVariables, isCustomSkin } = themeData;
|
|
|
|
|
function bsvariables() {
|
|
|
|
|
if (bootswatchSkin) {
|
|
|
|
|
if (isCustomSkin) {
|
|
|
|
|
return themeData._variables || '';
|
|
|
|
|
}
|
|
|
|
|
return `@import "bootswatch/dist/${bootswatchSkin}/variables";`;
|
|
|
|
|
}
|
|
|
|
|
return bsVariables;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [
|
|
|
|
|
bootswatchSkin ?
|
|
|
|
|
`@import "bootswatch/dist/${bootswatchSkin}/variables";` :
|
|
|
|
|
bsVariables,
|
|
|
|
|
bsvariables(),
|
|
|
|
|
'@import "bootstrap/scss/mixins/banner";',
|
|
|
|
|
'@include bsBanner("");',
|
|
|
|
|
// functions must be included first
|
|
|
|
@ -115,7 +124,7 @@ function boostrapImport(themeData) {
|
|
|
|
|
'@import "generics";',
|
|
|
|
|
'@import "client";', // core page styles
|
|
|
|
|
'@import "./theme";', // rest of the theme scss
|
|
|
|
|
bootswatchSkin ? `@import "bootswatch/dist/${bootswatchSkin}/bootswatch";` : '',
|
|
|
|
|
bootswatchSkin && !isCustomSkin ? `@import "bootswatch/dist/${bootswatchSkin}/bootswatch";` : '',
|
|
|
|
|
].join('\n');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -171,11 +180,14 @@ async function getBundleMetadata(target) {
|
|
|
|
|
|
|
|
|
|
// Skin support
|
|
|
|
|
let skin;
|
|
|
|
|
let isCustomSkin = false;
|
|
|
|
|
if (target.startsWith('client-')) {
|
|
|
|
|
skin = target.split('-')[1];
|
|
|
|
|
|
|
|
|
|
if (CSS.supportedSkins.includes(skin)) {
|
|
|
|
|
target = 'client';
|
|
|
|
|
skin = target.split('-').slice(1).join('-');
|
|
|
|
|
const isBootswatchSkin = CSS.supportedSkins.includes(skin);
|
|
|
|
|
isCustomSkin = !isBootswatchSkin && await CSS.isCustomSkin(skin);
|
|
|
|
|
target = 'client';
|
|
|
|
|
if (!isBootswatchSkin && !isCustomSkin) {
|
|
|
|
|
skin = ''; // invalid skin or deleted use default
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -191,6 +203,9 @@ async function getBundleMetadata(target) {
|
|
|
|
|
paths.unshift(`${baseThemePath}/node_modules`);
|
|
|
|
|
themeData.bsVariables = parseInt(themeData.useBSVariables, 10) === 1 ? (themeData.bsVariables || '') : '';
|
|
|
|
|
themeData.bootswatchSkin = skin;
|
|
|
|
|
themeData.isCustomSkin = isCustomSkin;
|
|
|
|
|
const customSkin = isCustomSkin ? await CSS.getCustomSkin(skin) : null;
|
|
|
|
|
themeData._variables = customSkin && customSkin._variables;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const [scssImports, cssImports, acpScssImports] = await Promise.all([
|
|
|
|
@ -210,6 +225,80 @@ async function getBundleMetadata(target) {
|
|
|
|
|
return { paths: paths, imports: imports };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CSS.getSkinSwitcherOptions = async function (uid) {
|
|
|
|
|
const user = require('../user');
|
|
|
|
|
const meta = require('./index');
|
|
|
|
|
const [userSettings, customSkins] = await Promise.all([
|
|
|
|
|
user.getSettings(uid),
|
|
|
|
|
CSS.getCustomSkins(),
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
const defaultSkin = _.capitalize(meta.config.bootswatchSkin) || '[[user:no-skin]]';
|
|
|
|
|
const defaultSkins = [
|
|
|
|
|
{ name: `[[user:default, ${defaultSkin}]]`, value: '', selected: userSettings.bootswatchSkin === '' },
|
|
|
|
|
{ name: '[[user:no-skin]]', value: 'noskin', selected: userSettings.bootswatchSkin === 'noskin' },
|
|
|
|
|
];
|
|
|
|
|
const lightSkins = [
|
|
|
|
|
'cerulean', 'cosmo', 'flatly', 'journal', 'litera',
|
|
|
|
|
'lumen', 'lux', 'materia', 'minty', 'morph', 'pulse', 'sandstone',
|
|
|
|
|
'simplex', 'sketchy', 'spacelab', 'united', 'yeti', 'zephyr',
|
|
|
|
|
];
|
|
|
|
|
const darkSkins = [
|
|
|
|
|
'cyborg', 'darkly', 'quartz', 'slate', 'solar', 'superhero', 'vapor',
|
|
|
|
|
];
|
|
|
|
|
function parseSkins(skins) {
|
|
|
|
|
skins = skins.map(skin => ({
|
|
|
|
|
name: _.capitalize(skin),
|
|
|
|
|
value: skin,
|
|
|
|
|
}));
|
|
|
|
|
skins.forEach((skin) => {
|
|
|
|
|
skin.selected = skin.value === userSettings.bootswatchSkin;
|
|
|
|
|
});
|
|
|
|
|
return skins;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
default: defaultSkins,
|
|
|
|
|
custom: customSkins.map(s => ({...s, selected: s.value === userSettings.bootswatchSkin })),
|
|
|
|
|
light: parseSkins(lightSkins),
|
|
|
|
|
dark: parseSkins(darkSkins),
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CSS.getCustomSkins = async function (opts = {}) {
|
|
|
|
|
const meta = require('./index');
|
|
|
|
|
const slugify = require('../slugify');
|
|
|
|
|
const { loadVariables } = opts;
|
|
|
|
|
const customSkins = await meta.settings.get('custom-skins');
|
|
|
|
|
const returnSkins = [];
|
|
|
|
|
if (customSkins && Array.isArray(customSkins['custom-skin-list'])) {
|
|
|
|
|
customSkins['custom-skin-list'].forEach((customSkin) => {
|
|
|
|
|
if (customSkin) {
|
|
|
|
|
returnSkins.push({
|
|
|
|
|
name: customSkin['custom-skin-name'],
|
|
|
|
|
value: slugify(customSkin['custom-skin-name']),
|
|
|
|
|
_variables: loadVariables ? customSkin._variables : undefined,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return returnSkins;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CSS.isSkinValid = async function (skin) {
|
|
|
|
|
return CSS.supportedSkins.includes(skin) || await CSS.isCustomSkin(skin);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CSS.isCustomSkin = async function (skin) {
|
|
|
|
|
const skins = await CSS.getCustomSkins();
|
|
|
|
|
return !!skins.find(s => s.value === skin);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CSS.getCustomSkin = async function (skin) {
|
|
|
|
|
const skins = await CSS.getCustomSkins({ loadVariables: true });
|
|
|
|
|
return skins.find(s => s.value === skin);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CSS.buildBundle = async function (target, fork) {
|
|
|
|
|
if (target === 'client') {
|
|
|
|
|
let files = await fs.promises.readdir(path.join(__dirname, '../../build/public'));
|
|
|
|
|