Vulcan/packages/nova-lib/lib/utils.js

435 lines
No EOL
12 KiB
JavaScript

/*
Utilities
*/
import marked from 'marked';
import urlObject from 'url';
import moment from 'moment';
import sanitizeHtml from 'sanitize-html';
import getSlug from 'speakingurl';
import { getSetting } from './settings.js';
/**
* @summary The global namespace for Telescope utils.
* @namespace Telescope.utils
*/
export const Utils = {};
/**
* @summary Convert a camelCase string to dash-separated string
* @param {String} str
*/
Utils.camelToDash = function (str) {
return str.replace(/\W+/g, '-').replace(/([a-z\d])([A-Z])/g, '$1-$2').toLowerCase();
};
/**
* @summary Convert a camelCase string to a space-separated capitalized string
* See http://stackoverflow.com/questions/4149276/javascript-camelcase-to-regular-form
* @param {String} str
*/
Utils.camelToSpaces = function (str) {
return str.replace(/([A-Z])/g, ' $1').replace(/^./, function(str){ return str.toUpperCase(); });
};
/**
* @summary Convert an underscore-separated string to dash-separated string
* @param {String} str
*/
Utils.underscoreToDash = function (str) {
return str.replace('_', '-');
};
/**
* @summary Convert a dash separated string to camelCase.
* @param {String} str
*/
Utils.dashToCamel = function (str) {
return str.replace(/(\-[a-z])/g, function($1){return $1.toUpperCase().replace('-','');});
};
/**
* @summary Convert a string to camelCase and remove spaces.
* @param {String} str
*/
Utils.camelCaseify = function(str) {
str = this.dashToCamel(str.replace(' ', '-'));
str = str.slice(0,1).toLowerCase() + str.slice(1);
return str;
};
/**
* @summary Trim a sentence to a specified amount of words and append an ellipsis.
* @param {String} s - Sentence to trim.
* @param {Number} numWords - Number of words to trim sentence to.
*/
Utils.trimWords = function(s, numWords) {
if (!s)
return s;
var expString = s.split(/\s+/,numWords);
if(expString.length >= numWords)
return expString.join(" ")+"…";
return s;
};
/**
* @summary Trim a block of HTML code to get a clean text excerpt
* @param {String} html - HTML to trim.
*/
Utils.trimHTML = function (html, numWords) {
var text = Utils.stripHTML(html);
return Utils.trimWords(text, numWords);
};
/**
* @summary Capitalize a string.
* @param {String} str
*/
Utils.capitalize = function(str) {
return str.charAt(0).toUpperCase() + str.slice(1);
};
Utils.t = function(message) {
var d = new Date();
console.log("### "+message+" rendered at "+d.getHours()+":"+d.getMinutes()+":"+d.getSeconds()); // eslint-disable-line
};
Utils.nl2br = function(str) {
var breakTag = '<br />';
return (str + '').replace(/([^>\r\n]?)(\r\n|\n\r|\r|\n)/g, '$1'+ breakTag +'$2');
};
Utils.scrollPageTo = function(selector) {
$('body').scrollTop($(selector).offset().top);
};
Utils.getDateRange = function(pageNumber) {
var now = moment(new Date());
var dayToDisplay=now.subtract(pageNumber-1, 'days');
var range={};
range.start = dayToDisplay.startOf('day').valueOf();
range.end = dayToDisplay.endOf('day').valueOf();
// console.log("after: ", dayToDisplay.startOf('day').format("dddd, MMMM Do YYYY, h:mm:ss a"));
// console.log("before: ", dayToDisplay.endOf('day').format("dddd, MMMM Do YYYY, h:mm:ss a"));
return range;
};
//////////////////////////
// URL Helper Functions //
//////////////////////////
/**
* @summary Returns the user defined site URL or Meteor.absoluteUrl
*/
Utils.getSiteUrl = function () {
return getSetting('siteUrl', Meteor.absoluteUrl());
};
/**
* @summary The global namespace for Telescope utils.
* @param {String} url - the URL to redirect
*/
Utils.getOutgoingUrl = function (url) {
return Utils.getSiteUrl() + "out?url=" + encodeURIComponent(url);
};
Utils.slugify = function (s) {
var slug = getSlug(s, {
truncate: 60
});
// can't have posts with an "edit" slug
if (slug === "edit") {
slug = "edit-1";
}
return slug;
};
Utils.getUnusedSlug = function (collection, slug) {
let suffix = "";
let index = 0;
// test if slug is already in use
while (!!collection.findOne({slug: slug+suffix})) {
index++;
suffix = "-"+index;
}
return slug+suffix;
};
Utils.getShortUrl = function(post) {
return post.shortUrl || post.url;
};
Utils.getDomain = function(url) {
try {
return urlObject.parse(url).hostname.replace('www.', '');
} catch (error) {
return null;
}
};
Utils.invitesEnabled = function() {
return getSetting("requireViewInvite") || getSetting("requirePostInvite");
};
// add http: if missing
Utils.addHttp = function (url) {
try {
if (url.substring(0, 5) !== "http:" && url.substring(0, 6) !== "https:") {
url = "http:"+url;
}
return url;
} catch (error) {
return null;
}
};
/////////////////////////////
// String Helper Functions //
/////////////////////////////
Utils.cleanUp = function(s) {
return this.stripHTML(s);
};
Utils.sanitize = function(s) {
// console.log('// before sanitization:')
// console.log(s)
if(Meteor.isServer){
s = sanitizeHtml(s, {
allowedTags: [
'h3', 'h4', 'h5', 'h6', 'blockquote', 'p', 'a', 'ul',
'ol', 'nl', 'li', 'b', 'i', 'strong', 'em', 'strike',
'code', 'hr', 'br', 'div', 'table', 'thead', 'caption',
'tbody', 'tr', 'th', 'td', 'pre', 'img'
]
});
// console.log('// after sanitization:')
// console.log(s)
}
return s;
};
Utils.stripHTML = function(s) {
return s.replace(/<(?:.|\n)*?>/gm, '');
};
Utils.stripMarkdown = function(s) {
var htmlBody = marked(s);
return Utils.stripHTML(htmlBody);
};
// http://stackoverflow.com/questions/2631001/javascript-test-for-existence-of-nested-object-key
Utils.checkNested = function(obj /*, level1, level2, ... levelN*/) {
var args = Array.prototype.slice.call(arguments);
obj = args.shift();
for (var i = 0; i < args.length; i++) {
if (!obj.hasOwnProperty(args[i])) {
return false;
}
obj = obj[args[i]];
}
return true;
};
Utils.log = function (s) {
if(getSetting('debug', false) || process.env.NODE_ENV === "development") {
console.log(s); // eslint-disable-line
}
};
// see http://stackoverflow.com/questions/8051975/access-object-child-properties-using-a-dot-notation-string
Utils.getNestedProperty = function (obj, desc) {
var arr = desc.split(".");
while(arr.length && (obj = obj[arr.shift()]));
return obj;
};
// see http://stackoverflow.com/a/14058408/649299
_.mixin({
compactObject : function(object) {
var clone = _.clone(object);
_.each(clone, function(value, key) {
if(!value && typeof value !== "boolean") {
delete clone[key];
}
});
return clone;
}
});
// adapted from http://stackoverflow.com/a/22072374/649299
Utils.unflatten = function( array, idProperty, parentIdProperty, parent, tree ){
tree = typeof tree !== "undefined" ? tree : [];
let children = [];
if (typeof parent === "undefined") {
// if there is no parent, we're at the root level
// so we return all root nodes (i.e. nodes with no parent)
children = _.filter( array, node => !node[parentIdProperty]);
} else {
// if there *is* a parent, we return all its child nodes
// (i.e. nodes whose parentId is equal to the parent's id.)
children = _.filter( array, node => node[parentIdProperty] === parent[idProperty]);
}
// if we found children, we keep on iterating
if (!!children.length) {
if (typeof parent === "undefined") {
// if we're at the root, then the tree consist of all root nodes
tree = children;
} else {
// else, we add the children to the parent as the "childrenResults" property
parent.childrenResults = children;
}
// we call the function on each child
children.forEach(child => {
Utils.unflatten(array, idProperty, parentIdProperty, child);
});
}
return tree;
}
Utils.getFieldLabel = (fieldName, collection) => {
const label = collection.simpleSchema()._schema[fieldName].label;
const nameWithSpaces = Utils.camelToSpaces(fieldName);
return label || nameWithSpaces;
}
Utils.getLogoUrl = () => {
const logoUrl = getSetting("logoUrl");
if (!!logoUrl) {
const prefix = Utils.getSiteUrl().slice(0,-1);
// the logo may be hosted on another website
return logoUrl.indexOf('://') > -1 ? logoUrl : prefix + logoUrl;
}
};
// note(apollo): get collection's name from __typename given by react-apollo
Utils.getCollectionNameFromTypename = (type) => {
if (type.indexOf('Post') > -1) {
return 'posts';
} else if (type.indexOf('Cat') > -1) {
return 'categories';
} else if (type.indexOf('User') > -1) {
return 'users';
} else if (type.indexOf('Comment') > -1) {
return 'comments';
}
};
Utils.findIndex = (array, predicate) => {
let index = -1;
let continueLoop = true;
array.forEach((item, currentIndex) => {
if (continueLoop && predicate(item)) {
index = currentIndex
continueLoop = false
}
});
return index;
}
// adapted from http://stackoverflow.com/a/22072374/649299
Utils.unflatten = function( array, idProperty, parentIdProperty, parent, tree ){
tree = typeof tree !== "undefined" ? tree : [];
let children = [];
if (typeof parent === "undefined") {
// if there is no parent, we're at the root level
// so we return all root nodes (i.e. nodes with no parent)
children = _.filter( array, node => !node[parentIdProperty]);
} else {
// if there *is* a parent, we return all its child nodes
// (i.e. nodes whose parentId is equal to the parent's id.)
children = _.filter( array, node => node[parentIdProperty] === parent[idProperty]);
}
// if we found children, we keep on iterating
if (!!children.length) {
if (typeof parent === "undefined") {
// if we're at the root, then the tree consist of all root nodes
tree = children;
} else {
// else, we add the children to the parent as the "childrenResults" property
parent.childrenResults = children;
}
// we call the function on each child
children.forEach(child => {
Utils.unflatten(array, idProperty, parentIdProperty, child);
});
}
return tree;
};
// remove the telescope object from a schema and duplicate it at the root
Utils.stripTelescopeNamespace = (schema) => {
// grab the users schema keys
const schemaKeys = Object.keys(schema);
// remove any field beginning by telescope: .telescope, .telescope.upvotedPosts.$, ...
const filteredSchemaKeys = schemaKeys.filter(key => key.slice(0,9) !== 'telescope');
// replace the previous schema by an object based on this filteredSchemaKeys
return filteredSchemaKeys.reduce((sch, key) => ({...sch, [key]: schema[key]}), {});
}
/**
* Convert an array of field names into a Mongo fields specifier
* @param {Array} fieldsArray
*/
Utils.arrayToFields = (fieldsArray) => {
return _.object(fieldsArray, _.map(fieldsArray, function () {return true}));
}
/**
* Get the display name of a React component
* @param {React Component} WrappedComponent
*/
Utils.getComponentDisplayName = (WrappedComponent) => {
return WrappedComponent.displayName || WrappedComponent.name || 'Component';
};
/**
* Take a collection and a list of documents, and convert all their date fields to date objects
* This is necessary because Apollo doesn't support custom scalars, and stores dates as strings
* @param {Object} collection
* @param {Array} list
*/
Utils.convertDates = (collection, listOrDocument) => {
// if undefined, just return
if (!listOrDocument || !listOrDocument.length) return listOrDocument;
const list = Array.isArray(listOrDocument) ? listOrDocument : [listOrDocument];
const schema = collection.simpleSchema()._schema;
const dateFields = _.filter(_.keys(schema), fieldName => schema[fieldName].type === Date);
const convertedList = list.map(result => {
dateFields.forEach(fieldName => {
if (result[fieldName] && typeof result[fieldName] === 'string') {
result[fieldName] = new Date(result[fieldName]);
}
});
return result;
});
return Array.isArray(listOrDocument) ? convertedList : convertedList[0];
}