2014-10-27 04:06:41 +00:00
|
|
|
/* jslint node: true */
|
|
|
|
'use strict';
|
|
|
|
|
2017-01-28 19:33:06 +00:00
|
|
|
const Config = require('./config.js').config;
|
|
|
|
const art = require('./art.js');
|
|
|
|
const ansi = require('./ansi_term.js');
|
|
|
|
const Log = require('./logger.js').log;
|
|
|
|
const configCache = require('./config_cache.js');
|
|
|
|
const getFullConfig = require('./config_util.js').getFullConfig;
|
|
|
|
const asset = require('./asset.js');
|
|
|
|
const ViewController = require('./view_controller.js').ViewController;
|
2017-02-08 03:20:10 +00:00
|
|
|
const Errors = require('./enig_error.js').Errors;
|
2017-01-28 19:33:06 +00:00
|
|
|
|
|
|
|
const fs = require('fs');
|
|
|
|
const paths = require('path');
|
|
|
|
const async = require('async');
|
|
|
|
const _ = require('lodash');
|
|
|
|
const assert = require('assert');
|
2015-05-13 05:04:22 +00:00
|
|
|
|
2014-10-27 04:06:41 +00:00
|
|
|
exports.getThemeArt = getThemeArt;
|
2015-12-24 18:56:04 +00:00
|
|
|
exports.getAvailableThemes = getAvailableThemes;
|
2014-10-30 04:23:44 +00:00
|
|
|
exports.getRandomTheme = getRandomTheme;
|
2016-01-15 05:44:33 +00:00
|
|
|
exports.setClientTheme = setClientTheme;
|
2014-10-29 11:30:20 +00:00
|
|
|
exports.initAvailableThemes = initAvailableThemes;
|
2014-11-05 06:50:42 +00:00
|
|
|
exports.displayThemeArt = displayThemeArt;
|
2015-07-25 22:10:12 +00:00
|
|
|
exports.displayThemedPause = displayThemedPause;
|
2017-02-08 03:20:10 +00:00
|
|
|
exports.displayThemedPrompt = displayThemedPrompt;
|
2015-07-25 22:10:12 +00:00
|
|
|
exports.displayThemedAsset = displayThemedAsset;
|
2014-10-27 04:06:41 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
function refreshThemeHelpers(theme) {
|
|
|
|
//
|
|
|
|
// Create some handy helpers
|
|
|
|
//
|
|
|
|
theme.helpers = {
|
|
|
|
getPasswordChar : function() {
|
|
|
|
var pwChar = Config.defaults.passwordChar;
|
|
|
|
if(_.has(theme, 'customization.defaults.general')) {
|
|
|
|
var themePasswordChar = theme.customization.defaults.general.passwordChar;
|
|
|
|
if(_.isString(themePasswordChar)) {
|
|
|
|
pwChar = themePasswordChar.substr(0, 1);
|
|
|
|
} else if(_.isNumber(themePasswordChar)) {
|
|
|
|
pwChar = String.fromCharCode(themePasswordChar);
|
2015-05-13 05:04:22 +00:00
|
|
|
}
|
2015-09-10 03:31:04 +00:00
|
|
|
}
|
|
|
|
return pwChar;
|
|
|
|
},
|
|
|
|
getDateFormat : function(style) {
|
|
|
|
style = style || 'short';
|
2015-05-13 05:04:22 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
var format = Config.defaults.dateFormat[style] || 'MM/DD/YYYY';
|
2015-07-23 03:35:35 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
if(_.has(theme, 'customization.defaults.dateFormat')) {
|
|
|
|
return theme.customization.defaults.dateFormat[style] || format;
|
|
|
|
}
|
|
|
|
return format;
|
|
|
|
},
|
|
|
|
getTimeFormat : function(style) {
|
|
|
|
style = style || 'short';
|
2015-07-24 04:23:44 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
var format = Config.defaults.timeFormat[style] || 'h:mm a';
|
2015-07-24 04:23:44 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
if(_.has(theme, 'customization.defaults.timeFormat')) {
|
|
|
|
return theme.customization.defaults.timeFormat[style] || format;
|
|
|
|
}
|
|
|
|
return format;
|
|
|
|
},
|
|
|
|
getDateTimeFormat : function(style) {
|
|
|
|
style = style || 'short';
|
2015-07-24 04:23:44 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
var format = Config.defaults.dateTimeFormat[style] || 'MM/DD/YYYY h:mm a';
|
2015-09-02 04:42:54 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
if(_.has(theme, 'customization.defaults.dateTimeFormat')) {
|
|
|
|
return theme.customization.defaults.dateTimeFormat[style] || format;
|
|
|
|
}
|
2015-09-02 04:42:54 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
return format;
|
|
|
|
}
|
2015-12-24 18:56:04 +00:00
|
|
|
};
|
2015-09-10 03:31:04 +00:00
|
|
|
}
|
2015-09-02 04:42:54 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
function loadTheme(themeID, cb) {
|
2015-04-17 04:29:53 +00:00
|
|
|
|
2015-09-10 03:31:04 +00:00
|
|
|
var path = paths.join(Config.paths.themes, themeID, 'theme.hjson');
|
|
|
|
|
2015-09-27 21:35:24 +00:00
|
|
|
configCache.getConfigWithOptions( { filePath : path, forceReCache : true }, function loaded(err, theme) {
|
2015-09-10 03:31:04 +00:00
|
|
|
if(err) {
|
|
|
|
cb(err);
|
|
|
|
} else {
|
2015-12-24 18:56:04 +00:00
|
|
|
if(!_.isObject(theme.info) ||
|
|
|
|
!_.isString(theme.info.name) ||
|
|
|
|
!_.isString(theme.info.author))
|
|
|
|
{
|
|
|
|
cb(new Error('Invalid or missing "info" section!'));
|
2015-09-10 03:31:04 +00:00
|
|
|
return;
|
2014-10-27 04:06:41 +00:00
|
|
|
}
|
2015-09-10 03:31:04 +00:00
|
|
|
|
|
|
|
refreshThemeHelpers(theme);
|
|
|
|
|
|
|
|
cb(null, theme, path);
|
2014-10-27 04:06:41 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-01-28 19:33:06 +00:00
|
|
|
const availableThemes = {};
|
2014-10-30 04:23:44 +00:00
|
|
|
|
2017-01-28 19:33:06 +00:00
|
|
|
const IMMUTABLE_MCI_PROPERTIES = [
|
|
|
|
'maxLength', 'argName', 'submit', 'validate'
|
2016-01-15 05:44:33 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
function getMergedTheme(menuConfig, promptConfig, theme) {
|
|
|
|
assert(_.isObject(menuConfig));
|
|
|
|
assert(_.isObject(theme));
|
|
|
|
|
|
|
|
// :TODO: merge in defaults (customization.defaults{} )
|
|
|
|
// :TODO: apply generic stuff, e.g. "VM" (vs "VM1")
|
|
|
|
|
|
|
|
//
|
|
|
|
// Create a *clone* of menuConfig (menu.hjson) then bring in
|
|
|
|
// promptConfig (prompt.hjson)
|
|
|
|
//
|
|
|
|
var mergedTheme = _.cloneDeep(menuConfig);
|
|
|
|
|
2017-01-28 19:33:06 +00:00
|
|
|
if(_.isObject(promptConfig.prompts)) {
|
|
|
|
mergedTheme.prompts = _.cloneDeep(promptConfig.prompts);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Add in data we won't be altering directly from the theme
|
|
|
|
//
|
|
|
|
mergedTheme.info = theme.info;
|
|
|
|
mergedTheme.helpers = theme.helpers;
|
|
|
|
|
|
|
|
//
|
|
|
|
// merge customizer to disallow immutable MCI properties
|
|
|
|
//
|
|
|
|
var mciCustomizer = function(objVal, srcVal, key) {
|
2016-01-15 05:44:33 +00:00
|
|
|
return IMMUTABLE_MCI_PROPERTIES.indexOf(key) > -1 ? objVal : srcVal;
|
|
|
|
};
|
2017-01-28 19:33:06 +00:00
|
|
|
|
|
|
|
function getFormKeys(fromObj) {
|
|
|
|
return _.remove(_.keys(fromObj), function pred(k) {
|
|
|
|
return !isNaN(k); // remove all non-numbers
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function mergeMciProperties(dest, src) {
|
|
|
|
Object.keys(src).forEach(function mciEntry(mci) {
|
|
|
|
_.mergeWith(dest[mci], src[mci], mciCustomizer);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function applyThemeMciBlock(dest, src, formKey) {
|
|
|
|
if(_.isObject(src.mci)) {
|
|
|
|
mergeMciProperties(dest, src.mci);
|
|
|
|
} else {
|
|
|
|
if(_.has(src, [ formKey, 'mci' ])) {
|
|
|
|
mergeMciProperties(dest, src[formKey].mci);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-01-15 05:44:33 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// menu.hjson can have a couple different structures:
|
|
|
|
// 1) Explicit declaration of expected MCI code(s) under 'form:<id>' before a 'mci' block
|
|
|
|
// (this allows multiple layout types defined by one menu for example)
|
|
|
|
//
|
|
|
|
// 2) Non-explicit declaration: 'mci' directly under 'form:<id>'
|
|
|
|
//
|
|
|
|
// theme.hjson has it's own mix:
|
|
|
|
// 1) Explicit: Form ID before 'mci' (generally used where there are > 1 forms)
|
|
|
|
//
|
|
|
|
// 2) Non-explicit: 'mci' directly under an entry
|
|
|
|
//
|
|
|
|
// Additionally, #1 or #2 may be under an explicit key of MCI code(s) to match up
|
|
|
|
// with menu.hjson in #1.
|
|
|
|
//
|
|
|
|
// * When theming an explicit menu.hjson entry (1), we will use a matching explicit
|
|
|
|
// entry with a matching MCI code(s) key in theme.hjson (e.g. menu="ETVM"/theme="ETVM"
|
|
|
|
// and fall back to generic if a match is not found.
|
|
|
|
//
|
|
|
|
// * If theme.hjson provides form ID's, use them. Otherwise, we'll apply directly assuming
|
|
|
|
// there is a generic 'mci' block.
|
|
|
|
//
|
2017-01-28 19:33:06 +00:00
|
|
|
function applyToForm(form, menuTheme, formKey) {
|
|
|
|
if(_.isObject(form.mci)) {
|
|
|
|
// non-explicit: no MCI code(s) key assumed since we found 'mci' directly under form ID
|
|
|
|
applyThemeMciBlock(form.mci, menuTheme, formKey);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
var menuMciCodeKeys = _.remove(_.keys(form), function pred(k) {
|
|
|
|
return k === k.toUpperCase(); // remove anything not uppercase
|
|
|
|
});
|
|
|
|
|
|
|
|
menuMciCodeKeys.forEach(function mciKeyEntry(mciKey) {
|
|
|
|
var applyFrom;
|
|
|
|
if(_.has(menuTheme, [ mciKey, 'mci' ])) {
|
|
|
|
applyFrom = menuTheme[mciKey];
|
|
|
|
} else {
|
|
|
|
applyFrom = menuTheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
applyThemeMciBlock(form[mciKey].mci, applyFrom);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2016-01-15 05:44:33 +00:00
|
|
|
|
2017-01-28 19:33:06 +00:00
|
|
|
[ 'menus', 'prompts' ].forEach(function areaEntry(sectionName) {
|
|
|
|
_.keys(mergedTheme[sectionName]).forEach(function menuEntry(menuName) {
|
|
|
|
var createdFormSection = false;
|
|
|
|
var mergedThemeMenu = mergedTheme[sectionName][menuName];
|
|
|
|
|
|
|
|
if(_.has(theme, [ 'customization', sectionName, menuName ])) {
|
|
|
|
var menuTheme = theme.customization[sectionName][menuName];
|
|
|
|
|
|
|
|
// config block is direct assign/overwrite
|
|
|
|
// :TODO: should probably be _.merge()
|
|
|
|
if(menuTheme.config) {
|
|
|
|
mergedThemeMenu.config = _.assign(mergedThemeMenu.config || {}, menuTheme.config);
|
|
|
|
}
|
|
|
|
|
|
|
|
if('menus' === sectionName) {
|
|
|
|
if(_.isObject(mergedThemeMenu.form)) {
|
|
|
|
getFormKeys(mergedThemeMenu.form).forEach(function formKeyEntry(formKey) {
|
|
|
|
applyToForm(mergedThemeMenu.form[formKey], menuTheme, formKey);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
if(_.isObject(menuTheme.mci)) {
|
|
|
|
//
|
|
|
|
// Not specified at menu level means we apply anything from the
|
|
|
|
// theme to form.0.mci{}
|
|
|
|
//
|
|
|
|
mergedThemeMenu.form = { 0 : { mci : { } } };
|
|
|
|
mergeMciProperties(mergedThemeMenu.form[0], menuTheme);
|
|
|
|
createdFormSection = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if('prompts' === sectionName) {
|
|
|
|
// no 'form' or form keys for prompts -- direct to mci
|
|
|
|
applyToForm(mergedThemeMenu, menuTheme);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Finished merging for this menu/prompt
|
|
|
|
//
|
|
|
|
// If the following conditions are true, set runtime.autoNext to true:
|
|
|
|
// * This is a menu
|
|
|
|
// * There is/was no explicit 'form' section
|
|
|
|
// * There is no 'prompt' specified
|
|
|
|
//
|
|
|
|
if('menus' === sectionName && !_.isString(mergedThemeMenu.prompt) &&
|
|
|
|
(createdFormSection || !_.isObject(mergedThemeMenu.form)))
|
|
|
|
{
|
|
|
|
mergedThemeMenu.runtime = _.merge(mergedThemeMenu.runtime || {}, { autoNext : true } );
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2016-01-15 05:44:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
return mergedTheme;
|
|
|
|
}
|
|
|
|
|
2014-10-29 11:30:20 +00:00
|
|
|
function initAvailableThemes(cb) {
|
2017-01-28 19:33:06 +00:00
|
|
|
var menuConfig;
|
|
|
|
var promptConfig;
|
2016-01-15 05:44:33 +00:00
|
|
|
|
2014-10-30 04:23:44 +00:00
|
|
|
async.waterfall(
|
|
|
|
[
|
2017-01-28 19:33:06 +00:00
|
|
|
function loadMenuConfig(callback) {
|
|
|
|
getFullConfig(Config.general.menuFile, function gotConfig(err, mc) {
|
|
|
|
menuConfig = mc;
|
|
|
|
callback(err);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function loadPromptConfig(callback) {
|
|
|
|
getFullConfig(Config.general.promptFile, function gotConfig(err, pc) {
|
|
|
|
promptConfig = pc;
|
|
|
|
callback(err);
|
|
|
|
});
|
|
|
|
},
|
2014-10-30 04:23:44 +00:00
|
|
|
function getDir(callback) {
|
2016-01-15 05:44:33 +00:00
|
|
|
fs.readdir(Config.paths.themes, function dirRead(err, files) {
|
2014-10-30 04:23:44 +00:00
|
|
|
callback(err, files);
|
|
|
|
});
|
|
|
|
},
|
2014-10-31 04:59:21 +00:00
|
|
|
function filterFiles(files, callback) {
|
2016-01-15 05:44:33 +00:00
|
|
|
var filtered = files.filter(function filter(file) {
|
2014-10-29 11:30:20 +00:00
|
|
|
return fs.statSync(paths.join(Config.paths.themes, file)).isDirectory();
|
2014-10-30 04:23:44 +00:00
|
|
|
});
|
|
|
|
callback(null, filtered);
|
|
|
|
},
|
|
|
|
function populateAvailable(filtered, callback) {
|
2016-01-15 05:44:33 +00:00
|
|
|
// :TODO: this is a bit broken with callback placement and configCache.on() handler
|
|
|
|
|
|
|
|
filtered.forEach(function themeEntry(themeId) {
|
2015-09-10 03:31:04 +00:00
|
|
|
loadTheme(themeId, function themeLoaded(err, theme, themePath) {
|
2014-10-30 04:23:44 +00:00
|
|
|
if(!err) {
|
2017-01-28 19:33:06 +00:00
|
|
|
availableThemes[themeId] = getMergedTheme(menuConfig, promptConfig, theme);
|
2015-09-10 03:31:04 +00:00
|
|
|
|
|
|
|
configCache.on('recached', function recached(path) {
|
|
|
|
if(themePath === path) {
|
|
|
|
loadTheme(themeId, function reloaded(err, reloadedTheme) {
|
|
|
|
Log.debug( { info : theme.info }, 'Theme recached' );
|
|
|
|
|
|
|
|
availableThemes[themeId] = reloadedTheme;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-05-13 05:04:22 +00:00
|
|
|
Log.debug( { info : theme.info }, 'Theme loaded');
|
2015-12-24 18:56:04 +00:00
|
|
|
} else {
|
|
|
|
Log.warn( { themeId : themeId, error : err.toString() }, 'Failed to load theme');
|
2014-10-30 04:23:44 +00:00
|
|
|
}
|
|
|
|
});
|
2015-05-13 05:04:22 +00:00
|
|
|
|
2014-10-30 04:23:44 +00:00
|
|
|
});
|
2014-10-31 04:59:21 +00:00
|
|
|
callback(null);
|
2014-10-30 04:23:44 +00:00
|
|
|
}
|
|
|
|
],
|
|
|
|
function onComplete(err) {
|
|
|
|
if(err) {
|
|
|
|
cb(err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-10-29 11:30:20 +00:00
|
|
|
cb(null, availableThemes.length);
|
2014-10-30 04:23:44 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-12-24 18:56:04 +00:00
|
|
|
function getAvailableThemes() {
|
|
|
|
return availableThemes;
|
|
|
|
}
|
|
|
|
|
2015-04-21 05:24:15 +00:00
|
|
|
function getRandomTheme() {
|
|
|
|
if(Object.getOwnPropertyNames(availableThemes).length > 0) {
|
|
|
|
var themeIds = Object.keys(availableThemes);
|
|
|
|
return themeIds[Math.floor(Math.random() * themeIds.length)];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-15 05:44:33 +00:00
|
|
|
function setClientTheme(client, themeId) {
|
2017-01-28 19:33:06 +00:00
|
|
|
var desc;
|
|
|
|
|
|
|
|
try {
|
|
|
|
client.currentTheme = getAvailableThemes()[themeId];
|
|
|
|
desc = 'Set client theme';
|
|
|
|
} catch(e) {
|
|
|
|
client.currentTheme = getAvailableThemes()[Config.defaults.theme];
|
|
|
|
desc = 'Failed setting theme by supplied ID; Using default';
|
|
|
|
}
|
|
|
|
|
|
|
|
client.log.debug( { themeId : themeId, info : client.currentTheme.info }, desc);
|
2016-01-15 05:44:33 +00:00
|
|
|
}
|
|
|
|
|
2015-09-28 04:05:40 +00:00
|
|
|
function getThemeArt(options, cb) {
|
|
|
|
//
|
|
|
|
// options - required:
|
2017-02-04 16:20:36 +00:00
|
|
|
// name
|
2015-09-28 04:05:40 +00:00
|
|
|
//
|
|
|
|
// options - optional
|
2017-02-04 16:20:36 +00:00
|
|
|
// client - needed for user's theme/etc.
|
|
|
|
// themeId
|
|
|
|
// asAnsi
|
|
|
|
// readSauce
|
|
|
|
// random
|
2015-09-28 04:05:40 +00:00
|
|
|
//
|
2017-02-04 16:20:36 +00:00
|
|
|
if(!options.themeId && _.has(options, 'client.user.properties.theme_id')) {
|
2015-09-28 04:05:40 +00:00
|
|
|
options.themeId = options.client.user.properties.theme_id;
|
|
|
|
} else {
|
|
|
|
options.themeId = Config.defaults.theme;
|
2014-10-27 04:06:41 +00:00
|
|
|
}
|
|
|
|
|
2015-06-09 04:41:57 +00:00
|
|
|
// :TODO: replace asAnsi stuff with something like retrieveAs = 'ansi' | 'pipe' | ...
|
|
|
|
// :TODO: Some of these options should only be set if not provided!
|
2015-09-28 04:05:40 +00:00
|
|
|
options.asAnsi = true; // always convert to ANSI
|
|
|
|
options.readSauce = true; // read SAUCE, if avail
|
|
|
|
options.random = _.isBoolean(options.random) ? options.random : true; // FILENAME<n>.EXT support
|
2014-10-27 04:06:41 +00:00
|
|
|
|
2015-09-28 04:05:40 +00:00
|
|
|
//
|
2017-05-20 00:41:13 +00:00
|
|
|
// We look for themed art in the following order:
|
|
|
|
// 1) Direct/relative path
|
|
|
|
// 2) Via theme supplied by |themeId|
|
|
|
|
// 3) Via default theme
|
|
|
|
// 4) General art directory
|
2015-09-28 04:05:40 +00:00
|
|
|
//
|
|
|
|
async.waterfall(
|
|
|
|
[
|
2016-07-04 18:58:41 +00:00
|
|
|
function fromPath(callback) {
|
|
|
|
//
|
|
|
|
// We allow relative (to enigma-bbs) or full paths
|
|
|
|
//
|
2017-05-20 00:41:13 +00:00
|
|
|
if('/' === options.name.charAt(0)) {
|
2016-07-04 18:58:41 +00:00
|
|
|
// just take the path as-is
|
|
|
|
options.basePath = paths.dirname(options.name);
|
|
|
|
} else if(options.name.indexOf('/') > -1) {
|
|
|
|
// make relative to base BBS dir
|
|
|
|
options.basePath = paths.join(__dirname, '../', paths.dirname(options.name));
|
|
|
|
} else {
|
|
|
|
return callback(null, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
art.getArt(options.name, options, (err, artInfo) => {
|
|
|
|
return callback(null, artInfo);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
function fromSuppliedTheme(artInfo, callback) {
|
|
|
|
if(artInfo) {
|
|
|
|
return callback(null, artInfo);
|
|
|
|
}
|
|
|
|
|
2015-09-28 04:05:40 +00:00
|
|
|
options.basePath = paths.join(Config.paths.themes, options.themeId);
|
2017-05-20 00:41:13 +00:00
|
|
|
art.getArt(options.name, options, (err, artInfo) => {
|
|
|
|
return callback(null, artInfo);
|
2015-09-28 04:05:40 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
function fromDefaultTheme(artInfo, callback) {
|
|
|
|
if(artInfo || Config.defaults.theme === options.themeId) {
|
2017-05-20 00:41:13 +00:00
|
|
|
return callback(null, artInfo);
|
2014-10-27 04:06:41 +00:00
|
|
|
}
|
2017-05-20 00:41:13 +00:00
|
|
|
|
|
|
|
options.basePath = paths.join(Config.paths.themes, Config.defaults.theme);
|
|
|
|
art.getArt(options.name, options, (err, artInfo) => {
|
|
|
|
return callback(null, artInfo);
|
|
|
|
});
|
2015-09-28 04:05:40 +00:00
|
|
|
},
|
|
|
|
function fromGeneralArtDir(artInfo, callback) {
|
|
|
|
if(artInfo) {
|
2017-05-20 00:41:13 +00:00
|
|
|
return callback(null, artInfo);
|
2015-09-28 04:05:40 +00:00
|
|
|
}
|
2017-05-20 00:41:13 +00:00
|
|
|
|
|
|
|
options.basePath = Config.paths.art;
|
|
|
|
art.getArt(options.name, options, (err, artInfo) => {
|
|
|
|
return callback(err, artInfo);
|
|
|
|
});
|
2015-09-28 04:05:40 +00:00
|
|
|
}
|
|
|
|
],
|
2015-10-06 21:22:49 +00:00
|
|
|
function complete(err, artInfo) {
|
|
|
|
if(err) {
|
2017-05-20 00:41:13 +00:00
|
|
|
const logger = _.get(options, 'client.log') || Log;
|
|
|
|
logger.debug( { reason : err.message }, 'Cannot find theme art');
|
2015-10-06 21:22:49 +00:00
|
|
|
}
|
2017-02-04 16:20:36 +00:00
|
|
|
return cb(err, artInfo);
|
2015-10-06 21:22:49 +00:00
|
|
|
}
|
2015-09-28 04:05:40 +00:00
|
|
|
);
|
2014-11-05 06:50:42 +00:00
|
|
|
}
|
|
|
|
|
2015-04-17 04:29:53 +00:00
|
|
|
function displayThemeArt(options, cb) {
|
|
|
|
assert(_.isObject(options));
|
|
|
|
assert(_.isObject(options.client));
|
|
|
|
assert(_.isString(options.name));
|
|
|
|
|
2015-09-28 04:05:40 +00:00
|
|
|
getThemeArt(options, function themeArt(err, artInfo) {
|
2014-11-05 06:50:42 +00:00
|
|
|
if(err) {
|
|
|
|
cb(err);
|
|
|
|
} else {
|
2015-09-27 21:35:24 +00:00
|
|
|
// :TODO: just use simple merge of options -> displayOptions
|
2016-09-01 04:06:28 +00:00
|
|
|
/*
|
2015-04-17 04:29:53 +00:00
|
|
|
var dispOptions = {
|
2015-07-25 22:10:12 +00:00
|
|
|
art : artInfo.data,
|
|
|
|
sauce : artInfo.sauce,
|
|
|
|
client : options.client,
|
|
|
|
font : options.font,
|
2015-09-27 21:35:24 +00:00
|
|
|
trailingLF : options.trailingLF,
|
2015-04-17 04:29:53 +00:00
|
|
|
};
|
|
|
|
|
2015-06-26 04:34:33 +00:00
|
|
|
art.display(dispOptions, function displayed(err, mciMap, extraInfo) {
|
|
|
|
cb(err, { mciMap : mciMap, artInfo : artInfo, extraInfo : extraInfo } );
|
2014-11-05 06:50:42 +00:00
|
|
|
});
|
2016-09-01 04:06:28 +00:00
|
|
|
*/
|
|
|
|
const displayOpts = {
|
|
|
|
sauce : artInfo.sauce,
|
|
|
|
font : options.font,
|
|
|
|
trailingLF : options.trailingLF,
|
|
|
|
};
|
|
|
|
|
|
|
|
art.display(options.client, artInfo.data, displayOpts, (err, mciMap, extraInfo) => {
|
|
|
|
return cb(err, { mciMap : mciMap, artInfo : artInfo, extraInfo : extraInfo } );
|
|
|
|
});
|
2014-11-05 06:50:42 +00:00
|
|
|
}
|
|
|
|
});
|
2015-07-06 01:05:55 +00:00
|
|
|
}
|
2015-07-24 04:23:44 +00:00
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
/*
|
|
|
|
function displayThemedPrompt(name, client, options, cb) {
|
|
|
|
|
|
|
|
async.waterfall(
|
|
|
|
[
|
|
|
|
function loadConfig(callback) {
|
|
|
|
configCache.getModConfig('prompt.hjson', (err, promptJson) => {
|
|
|
|
if(err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
2015-07-24 04:23:44 +00:00
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
if(_.has(promptJson, [ 'prompts', name ] )) {
|
|
|
|
return callback(Errors.DoesNotExist(`Prompt "${name}" does not exist`));
|
|
|
|
}
|
2015-07-25 00:33:59 +00:00
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
const promptConfig = promptJson.prompts[name];
|
|
|
|
if(!_.isObject(promptConfig)) {
|
|
|
|
return callback(Errors.Invalid(`Prompt "${name} is invalid`));
|
|
|
|
}
|
2015-07-24 04:23:44 +00:00
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
return callback(null, promptConfig);
|
2015-07-29 04:31:28 +00:00
|
|
|
});
|
2015-07-24 04:23:44 +00:00
|
|
|
},
|
2017-02-08 03:20:10 +00:00
|
|
|
function display(promptConfig, callback) {
|
|
|
|
if(options.clearScreen) {
|
|
|
|
client.term.rawWrite(ansi.clearScreen());
|
|
|
|
}
|
|
|
|
|
2015-11-01 20:32:52 +00:00
|
|
|
//
|
2017-02-08 03:20:10 +00:00
|
|
|
// If we did not clear the screen, don't let the font change
|
|
|
|
//
|
|
|
|
const dispOptions = Object.assign( {}, promptConfig.options );
|
|
|
|
if(!options.clearScreen) {
|
|
|
|
dispOptions.font = 'not_really_a_font!';
|
|
|
|
}
|
|
|
|
|
|
|
|
displayThemedAsset(
|
|
|
|
promptConfig.art,
|
|
|
|
client,
|
|
|
|
dispOptions,
|
|
|
|
(err, artData) => {
|
|
|
|
if(err) {
|
|
|
|
return callback(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return callback(null, promptConfig, artData.mciMap);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
},
|
|
|
|
function prepViews(promptConfig, mciMap, callback) {
|
|
|
|
vc = new ViewController( { client : client } );
|
|
|
|
|
|
|
|
const loadOpts = {
|
|
|
|
promptName : name,
|
|
|
|
mciMap : mciMap,
|
|
|
|
config : promptConfig,
|
|
|
|
};
|
|
|
|
|
|
|
|
vc.loadFromPromptConfig(loadOpts, err => {
|
|
|
|
callback(null);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
function displayThemedPrompt(name, client, options, cb) {
|
|
|
|
|
|
|
|
const useTempViewController = _.isUndefined(options.viewController);
|
|
|
|
|
|
|
|
async.waterfall(
|
|
|
|
[
|
|
|
|
function display(callback) {
|
|
|
|
const promptConfig = client.currentTheme.prompts[name];
|
|
|
|
if(!promptConfig) {
|
|
|
|
return callback(Errors.DoesNotExist(`Missing "${name}" prompt configuration!`));
|
|
|
|
}
|
|
|
|
|
|
|
|
if(options.clearScreen) {
|
2017-02-12 07:23:39 +00:00
|
|
|
client.term.rawWrite(ansi.resetScreen());
|
2017-02-08 03:20:10 +00:00
|
|
|
}
|
|
|
|
|
2015-11-01 20:32:52 +00:00
|
|
|
//
|
2017-02-08 03:20:10 +00:00
|
|
|
// If we did *not* clear the screen, don't let the font change
|
|
|
|
// as it will mess with the output of the existing art displayed in a terminal
|
|
|
|
//
|
|
|
|
const dispOptions = Object.assign( {}, promptConfig.options );
|
|
|
|
if(!options.clearScreen) {
|
|
|
|
dispOptions.font = 'not_really_a_font!'; // kludge :)
|
|
|
|
}
|
2015-11-01 20:32:52 +00:00
|
|
|
|
2015-07-25 22:10:12 +00:00
|
|
|
displayThemedAsset(
|
|
|
|
promptConfig.art,
|
2017-02-08 03:20:10 +00:00
|
|
|
client,
|
2015-11-01 20:32:52 +00:00
|
|
|
dispOptions,
|
2017-02-08 03:20:10 +00:00
|
|
|
(err, artInfo) => {
|
|
|
|
return callback(err, promptConfig, artInfo);
|
2015-07-25 22:10:12 +00:00
|
|
|
}
|
|
|
|
);
|
2015-07-25 00:33:59 +00:00
|
|
|
},
|
2017-02-08 03:20:10 +00:00
|
|
|
function discoverCursorPosition(promptConfig, artInfo, callback) {
|
|
|
|
if(!options.clearPrompt) {
|
|
|
|
// no need to query cursor - we're not gonna use it
|
|
|
|
return callback(null, promptConfig, artInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
client.once('cursor position report', pos => {
|
2015-07-25 22:10:12 +00:00
|
|
|
artInfo.startRow = pos[0] - artInfo.height;
|
2017-02-08 03:20:10 +00:00
|
|
|
return callback(null, promptConfig, artInfo);
|
2015-07-24 04:23:44 +00:00
|
|
|
});
|
2017-02-08 03:20:10 +00:00
|
|
|
|
|
|
|
client.term.rawWrite(ansi.queryPos());
|
2015-07-24 04:23:44 +00:00
|
|
|
},
|
2017-02-08 03:20:10 +00:00
|
|
|
function createMCIViews(promptConfig, artInfo, callback) {
|
|
|
|
const tempViewController = useTempViewController ? new ViewController( { client : client } ) : options.viewController;
|
|
|
|
|
|
|
|
const loadOpts = {
|
|
|
|
promptName : name,
|
|
|
|
mciMap : artInfo.mciMap,
|
|
|
|
config : promptConfig,
|
|
|
|
};
|
|
|
|
|
|
|
|
tempViewController.loadFromPromptConfig(loadOpts, () => {
|
|
|
|
return callback(null, artInfo, tempViewController);
|
2015-07-25 22:10:12 +00:00
|
|
|
});
|
|
|
|
},
|
2017-02-08 03:20:10 +00:00
|
|
|
function pauseForUserInput(artInfo, tempViewController, callback) {
|
|
|
|
if(!options.pause) {
|
|
|
|
return callback(null, artInfo, tempViewController);
|
|
|
|
}
|
|
|
|
|
|
|
|
client.waitForKeyPress( () => {
|
|
|
|
return callback(null, artInfo, tempViewController);
|
2015-07-24 04:23:44 +00:00
|
|
|
});
|
|
|
|
},
|
2017-02-08 03:20:10 +00:00
|
|
|
function clearPauseArt(artInfo, tempViewController, callback) {
|
2015-07-24 04:23:44 +00:00
|
|
|
if(options.clearPrompt) {
|
2016-01-15 05:44:33 +00:00
|
|
|
if(artInfo.startRow && artInfo.height) {
|
2017-02-08 03:20:10 +00:00
|
|
|
client.term.rawWrite(ansi.goto(artInfo.startRow, 1));
|
2016-02-10 05:30:59 +00:00
|
|
|
|
|
|
|
// Note: Does not work properly in NetRunner < 2.0b17:
|
2017-02-08 03:20:10 +00:00
|
|
|
client.term.rawWrite(ansi.deleteLine(artInfo.height));
|
2015-07-25 00:33:59 +00:00
|
|
|
} else {
|
2017-02-08 03:20:10 +00:00
|
|
|
client.term.rawWrite(ansi.eraseLine(1));
|
2015-07-25 00:33:59 +00:00
|
|
|
}
|
2015-07-24 04:23:44 +00:00
|
|
|
}
|
2017-02-08 03:20:10 +00:00
|
|
|
|
|
|
|
return callback(null, tempViewController);
|
2015-07-24 04:23:44 +00:00
|
|
|
}
|
|
|
|
],
|
2017-02-08 03:20:10 +00:00
|
|
|
(err, tempViewController) => {
|
2015-07-24 04:23:44 +00:00
|
|
|
if(err) {
|
2017-02-08 03:20:10 +00:00
|
|
|
client.log.warn( { error : err.message }, `Failed displaying "${name}" prompt` );
|
2015-07-24 04:23:44 +00:00
|
|
|
}
|
2015-07-25 22:10:12 +00:00
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
if(tempViewController && useTempViewController) {
|
|
|
|
tempViewController.detachClientEvents();
|
2015-07-25 22:10:12 +00:00
|
|
|
}
|
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
return cb(null);
|
2015-07-24 04:23:44 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2015-07-25 22:10:12 +00:00
|
|
|
|
2017-02-08 03:20:10 +00:00
|
|
|
//
|
|
|
|
// Pause prompts are a special prompt by the name 'pause'.
|
|
|
|
//
|
|
|
|
function displayThemedPause(client, options, cb) {
|
|
|
|
|
|
|
|
if(!cb && _.isFunction(options)) {
|
|
|
|
cb = options;
|
|
|
|
options = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!_.isBoolean(options.clearPrompt)) {
|
|
|
|
options.clearPrompt = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const promptOptions = Object.assign( {}, options, { pause : true } );
|
|
|
|
return displayThemedPrompt('pause', client, promptOptions, cb);
|
|
|
|
}
|
|
|
|
|
2015-07-25 22:10:12 +00:00
|
|
|
function displayThemedAsset(assetSpec, client, options, cb) {
|
|
|
|
assert(_.isObject(client));
|
|
|
|
|
|
|
|
// options are... optional
|
|
|
|
if(3 === arguments.length) {
|
|
|
|
cb = options;
|
|
|
|
options = {};
|
|
|
|
}
|
|
|
|
|
2016-07-04 18:58:41 +00:00
|
|
|
const artAsset = asset.getArtAsset(assetSpec);
|
2015-07-25 22:10:12 +00:00
|
|
|
if(!artAsset) {
|
2016-07-04 18:58:41 +00:00
|
|
|
return cb(new Error('Asset not found: ' + assetSpec));
|
2015-07-25 22:10:12 +00:00
|
|
|
}
|
|
|
|
|
2015-09-27 21:35:24 +00:00
|
|
|
// :TODO: just use simple merge of options -> displayOptions
|
2015-07-25 22:10:12 +00:00
|
|
|
var dispOpts = {
|
|
|
|
name : artAsset.asset,
|
|
|
|
client : client,
|
|
|
|
font : options.font,
|
2015-09-27 21:35:24 +00:00
|
|
|
trailingLF : options.trailingLF,
|
2015-07-25 22:10:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
switch(artAsset.type) {
|
2016-08-04 04:43:06 +00:00
|
|
|
case 'art' :
|
|
|
|
displayThemeArt(dispOpts, function displayed(err, artData) {
|
|
|
|
return cb(err, err ? null : { mciMap : artData.mciMap, height : artData.extraInfo.height } );
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'method' :
|
|
|
|
// :TODO: fetch & render via method
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'inline ' :
|
|
|
|
// :TODO: think about this more in relation to themes, etc. How can this come
|
|
|
|
// from a theme (with override from menu.json) ???
|
|
|
|
// look @ client.currentTheme.inlineArt[name] -> menu/prompt[name]
|
|
|
|
break;
|
|
|
|
|
|
|
|
default :
|
|
|
|
return cb(new Error('Unsupported art asset type: ' + artAsset.type));
|
2015-07-25 22:10:12 +00:00
|
|
|
}
|
|
|
|
}
|