mirror of
https://github.com/TryGhost/Ghost.git
synced 2024-12-22 10:21:36 +03:00
22e13acd65
- All var declarations are now const or let as per ES6 - All comma-separated lists / chained declarations are now one declaration per line - This is for clarity/readability but also made running the var-to-const/let switch smoother - ESLint rules updated to match How this was done: - npm install -g jscodeshift - git clone https://github.com/cpojer/js-codemod.git - git clone git@github.com:TryGhost/Ghost.git shallow-ghost - cd shallow-ghost - jscodeshift -t ../js-codemod/transforms/unchain-variables.js . -v=2 - jscodeshift -t ../js-codemod/transforms/no-vars.js . -v=2 - yarn - yarn test - yarn lint / fix various lint errors (almost all indent) by opening files and saving in vscode - grunt test-regression - sorted!
172 lines
5.0 KiB
JavaScript
172 lines
5.0 KiB
JavaScript
// # Has Helper
|
|
// Usage: `{{#has tag="video, music"}}`, `{{#has author="sam, pat"}}`
|
|
// `{{#has author="count:1"}}`, `{{#has tag="count:>1"}}`
|
|
//
|
|
// Checks if a post has a particular property
|
|
|
|
const {logging, i18n} = require('../services/proxy');
|
|
const _ = require('lodash');
|
|
const validAttrs = ['tag', 'author', 'slug','visibility', 'id', 'number', 'index', 'any', 'all'];
|
|
|
|
function handleCount(ctxAttr, data) {
|
|
if (!data || !_.isFinite(data.length)) {
|
|
return false;
|
|
}
|
|
let count;
|
|
|
|
if (ctxAttr.match(/count:\d+/)) {
|
|
count = Number(ctxAttr.match(/count:(\d+)/)[1]);
|
|
return count === data.length;
|
|
} else if (ctxAttr.match(/count:>\d/)) {
|
|
count = Number(ctxAttr.match(/count:>(\d+)/)[1]);
|
|
return count < data.length;
|
|
} else if (ctxAttr.match(/count:<\d/)) {
|
|
count = Number(ctxAttr.match(/count:<(\d+)/)[1]);
|
|
return count > data.length;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function evaluateTagList(expr, tags) {
|
|
return expr.split(',').map(function (v) {
|
|
return v.trim();
|
|
}).reduce(function (p, c) {
|
|
return p || (_.findIndex(tags, function (item) {
|
|
// Escape regex special characters
|
|
item = item.replace(/[-/\\^$*+?.()|[\]{}]/g, '\\$&');
|
|
item = new RegExp('^' + item + '$', 'i');
|
|
return item.test(c);
|
|
}) !== -1);
|
|
}, false);
|
|
}
|
|
|
|
function handleTag(data, attrs) {
|
|
if (!attrs.tag) {
|
|
return false;
|
|
}
|
|
|
|
if (attrs.tag.match(/count:/)) {
|
|
return handleCount(attrs.tag, data.tags);
|
|
}
|
|
|
|
return evaluateTagList(attrs.tag, _.map(data.tags, 'name')) || false;
|
|
}
|
|
|
|
function evaluateAuthorList(expr, authors) {
|
|
const authorList = expr.split(',').map(function (v) {
|
|
return v.trim().toLocaleLowerCase();
|
|
});
|
|
|
|
return _.filter(authors, (author) => {
|
|
return _.includes(authorList, author.name.toLocaleLowerCase());
|
|
}).length;
|
|
}
|
|
|
|
function handleAuthor(data, attrs) {
|
|
if (!attrs.author) {
|
|
return false;
|
|
}
|
|
|
|
if (attrs.author.match(/count:/)) {
|
|
return handleCount(attrs.author, data.authors);
|
|
}
|
|
|
|
return evaluateAuthorList(attrs.author, data.authors) || false;
|
|
}
|
|
|
|
function evaluateIntegerMatch(expr, integer) {
|
|
const nthMatch = expr.match(/^nth:(\d+)/);
|
|
if (nthMatch) {
|
|
return integer % parseInt(nthMatch[1], 10) === 0;
|
|
}
|
|
|
|
return expr.split(',').reduce(function (bool, _integer) {
|
|
return bool || parseInt(_integer, 10) === integer;
|
|
}, false);
|
|
}
|
|
|
|
function evaluateStringMatch(expr, str, ci) {
|
|
if (ci) {
|
|
return expr && str && expr.toLocaleLowerCase() === str.toLocaleLowerCase();
|
|
}
|
|
|
|
return expr === str;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param {String} type - either some or every - the lodash function to use
|
|
* @param {String} expr - the attribute value passed into {{#has}}
|
|
* @param {Object} obj - "this" context from the helper
|
|
* @param {Object} data - global params
|
|
*/
|
|
function evaluateList(type, expr, obj, data) {
|
|
return expr.split(',').map(function (prop) {
|
|
return prop.trim().toLocaleLowerCase();
|
|
})[type](function (prop) {
|
|
if (prop.match(/^@/)) {
|
|
return _.has(data, prop.replace(/@/, '')) && !_.isEmpty(_.get(data, prop.replace(/@/, '')));
|
|
} else {
|
|
return _.has(obj, prop) && !_.isEmpty(_.get(obj, prop));
|
|
}
|
|
});
|
|
}
|
|
|
|
module.exports = function has(options) {
|
|
options = options || {};
|
|
options.hash = options.hash || {};
|
|
options.data = options.data || {};
|
|
|
|
const self = this;
|
|
const attrs = _.pick(options.hash, validAttrs);
|
|
const data = _.pick(options.data, ['site', 'config', 'labs']);
|
|
|
|
const checks = {
|
|
tag: function () {
|
|
return handleTag(self, attrs);
|
|
},
|
|
author: function () {
|
|
return handleAuthor(self, attrs);
|
|
},
|
|
number: function () {
|
|
return attrs.number && evaluateIntegerMatch(attrs.number, options.data.number) || false;
|
|
},
|
|
index: function () {
|
|
return attrs.index && evaluateIntegerMatch(attrs.index, options.data.index) || false;
|
|
},
|
|
visibility: function () {
|
|
return attrs.visibility && evaluateStringMatch(attrs.visibility, self.visibility, true) || false;
|
|
},
|
|
slug: function () {
|
|
return attrs.slug && evaluateStringMatch(attrs.slug, self.slug, true) || false;
|
|
},
|
|
id: function () {
|
|
return attrs.id && evaluateStringMatch(attrs.id, self.id, true) || false;
|
|
},
|
|
any: function () {
|
|
return attrs.any && evaluateList('some', attrs.any, self, data) || false;
|
|
},
|
|
all: function () {
|
|
return attrs.all && evaluateList('every', attrs.all, self, data) || false;
|
|
}
|
|
};
|
|
|
|
let result;
|
|
|
|
if (_.isEmpty(attrs)) {
|
|
logging.warn(i18n.t('warnings.helpers.has.invalidAttribute'));
|
|
return;
|
|
}
|
|
|
|
result = _.some(attrs, function (value, attr) {
|
|
return checks[attr]();
|
|
});
|
|
|
|
if (result) {
|
|
return options.fn(this);
|
|
}
|
|
|
|
return options.inverse(this);
|
|
};
|