// ### Navigation Helper // `{{navigation}}` // Outputs navigation menu of static urls var proxy = require('./proxy'), string = require('../../server/lib/security/string'), _ = require('lodash'), SafeString = proxy.SafeString, createFrame = proxy.hbs.handlebars.createFrame, i18n = proxy.i18n, errors = proxy.errors, templates = proxy.templates; module.exports = function navigation(options) { options = options || {}; options.hash = options.hash || {}; options.data = options.data || {}; var navigationData = options.data.site.navigation, currentUrl = options.data.root.relativeUrl, self = this, output; if (!_.isObject(navigationData) || _.isFunction(navigationData)) { throw new errors.IncorrectUsageError({ message: i18n.t('warnings.helpers.navigation.invalidData') }); } if (navigationData.filter(function (e) { return (_.isUndefined(e.label) || _.isUndefined(e.url)); }).length > 0) { throw new errors.IncorrectUsageError({ message: i18n.t('warnings.helpers.navigation.valuesMustBeDefined') }); } // check for non-null string values if (navigationData.filter(function (e) { return ((!_.isNull(e.label) && !_.isString(e.label)) || (!_.isNull(e.url) && !_.isString(e.url))); }).length > 0) { throw new errors.IncorrectUsageError({ message: i18n.t('warnings.helpers.navigation.valuesMustBeString') }); } function _slugify(label) { return string.safe(label); } // strips trailing slashes and compares urls function _isCurrentUrl(href, currentUrl) { if (!currentUrl) { return false; } var strippedHref = href.replace(/\/+$/, ''), strippedCurrentUrl = currentUrl.replace(/\/+$/, ''); return strippedHref === strippedCurrentUrl; } // {{navigation}} should no-op if no data passed in if (navigationData.length === 0) { return new SafeString(''); } output = navigationData.map(function (e) { var out = {}; out.current = _isCurrentUrl(e.url, currentUrl); out.label = e.label; out.slug = _slugify(e.label); out.url = e.url; out.secure = self.secure; return out; }); // CASE: The navigation helper should have access to the navigation items at the top level. this.navigation = output; // CASE: The navigation helper will forward attributes passed to it. _.merge(this, options.hash); const data = createFrame(options.data); return templates.execute('navigation', this, {data}); };