2012-04-30 11:23:03 +00:00
|
|
|
/*\
|
2012-05-03 20:47:16 +00:00
|
|
|
title: $:/core/modules/utils.js
|
2012-04-30 11:23:03 +00:00
|
|
|
type: application/javascript
|
|
|
|
module-type: utils
|
|
|
|
|
|
|
|
Various static utility functions.
|
|
|
|
|
|
|
|
This file is a bit of a dumping ground; the expectation is that most of these functions will be refactored.
|
|
|
|
|
|
|
|
\*/
|
|
|
|
(function(){
|
|
|
|
|
2012-05-04 17:49:04 +00:00
|
|
|
/*jslint node: true, browser: true */
|
|
|
|
/*global $tw: false */
|
2012-04-30 11:23:03 +00:00
|
|
|
"use strict";
|
|
|
|
|
2012-05-08 14:11:29 +00:00
|
|
|
// Check if an object has a property
|
|
|
|
exports.hop = function(object,property) {
|
|
|
|
return Object.prototype.hasOwnProperty.call(object,property);
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Push entries onto an array, removing them first if they already exist in the array
|
|
|
|
array: array to modify
|
|
|
|
value: a single value to push or an array of values to push
|
|
|
|
*/
|
|
|
|
exports.pushTop = function(array,value) {
|
|
|
|
var t,p;
|
|
|
|
if($tw.utils.isArray(value)) {
|
|
|
|
// Remove any array entries that are duplicated in the new values
|
|
|
|
for(t=0; t<value.length; t++) {
|
|
|
|
p = array.indexOf(value[t]);
|
|
|
|
if(p !== -1) {
|
|
|
|
array.splice(p,1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Push the values on top of the main array
|
|
|
|
array.push.apply(array,value);
|
|
|
|
} else {
|
|
|
|
p = array.indexOf(value);
|
|
|
|
if(p !== -1) {
|
|
|
|
array.splice(p,1);
|
|
|
|
}
|
|
|
|
array.push(value);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Remove entries from an array
|
|
|
|
array: array to modify
|
|
|
|
value: a single value to remove, or an array of values to remove
|
|
|
|
*/
|
|
|
|
exports.removeArrayEntries = function(array,value) {
|
|
|
|
var t,p;
|
|
|
|
if($tw.utils.isArray(value)) {
|
|
|
|
for(t=0; t<value.length; t++) {
|
|
|
|
p = array.indexOf(value[t]);
|
|
|
|
if(p !== -1) {
|
|
|
|
array.splice(p,1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
p = array.indexOf(value);
|
|
|
|
if(p !== -1) {
|
|
|
|
array.splice(p,1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-05-03 20:47:16 +00:00
|
|
|
exports.deepCopy = function(object) {
|
|
|
|
var result,t;
|
|
|
|
if($tw.utils.isArray(object)) {
|
|
|
|
// Copy arrays
|
|
|
|
result = object.slice(0);
|
|
|
|
} else if(typeof object === "object") {
|
|
|
|
result = {};
|
|
|
|
for(t in object) {
|
|
|
|
if(object[t] !== undefined) {
|
|
|
|
result[t] = $tw.utils.deepCopy(object[t]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result = object;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.extendDeepCopy = function(object,extendedProperties) {
|
|
|
|
var result = $tw.utils.deepCopy(object),t;
|
2012-05-04 17:49:04 +00:00
|
|
|
for(t in extendedProperties) {
|
2012-05-05 10:21:15 +00:00
|
|
|
if(extendedProperties[t] !== undefined) {
|
|
|
|
result[t] = $tw.utils.deepCopy(extendedProperties[t]);
|
2012-05-03 20:47:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
2012-04-30 11:23:03 +00:00
|
|
|
exports.formatDateString = function (date,template) {
|
|
|
|
var t = template.replace(/0hh12/g,$tw.utils.pad($tw.utils.getHours12(date)));
|
|
|
|
t = t.replace(/hh12/g,$tw.utils.getHours12(date));
|
|
|
|
t = t.replace(/0hh/g,$tw.utils.pad(date.getHours()));
|
|
|
|
t = t.replace(/hh/g,date.getHours());
|
|
|
|
t = t.replace(/mmm/g,$tw.config.dateFormats.shortMonths[date.getMonth()]);
|
|
|
|
t = t.replace(/0mm/g,$tw.utils.pad(date.getMinutes()));
|
|
|
|
t = t.replace(/mm/g,date.getMinutes());
|
|
|
|
t = t.replace(/0ss/g,$tw.utils.pad(date.getSeconds()));
|
|
|
|
t = t.replace(/ss/g,date.getSeconds());
|
|
|
|
t = t.replace(/[ap]m/g,$tw.utils.getAmPm(date).toLowerCase());
|
|
|
|
t = t.replace(/[AP]M/g,$tw.utils.getAmPm(date).toUpperCase());
|
|
|
|
t = t.replace(/wYYYY/g,$tw.utils.getYearForWeekNo(date));
|
|
|
|
t = t.replace(/wYY/g,$tw.utils.pad($tw.utils.getYearForWeekNo(date)-2000));
|
|
|
|
t = t.replace(/YYYY/g,date.getFullYear());
|
|
|
|
t = t.replace(/YY/g,$tw.utils.pad(date.getFullYear()-2000));
|
|
|
|
t = t.replace(/MMM/g,$tw.config.dateFormats.months[date.getMonth()]);
|
|
|
|
t = t.replace(/0MM/g,$tw.utils.pad(date.getMonth()+1));
|
|
|
|
t = t.replace(/MM/g,date.getMonth()+1);
|
|
|
|
t = t.replace(/0WW/g,$tw.utils.pad($tw.utils.getWeek(date)));
|
|
|
|
t = t.replace(/WW/g,$tw.utils.getWeek(date));
|
|
|
|
t = t.replace(/DDD/g,$tw.config.dateFormats.days[date.getDay()]);
|
|
|
|
t = t.replace(/ddd/g,$tw.config.dateFormats.shortDays[date.getDay()]);
|
|
|
|
t = t.replace(/0DD/g,$tw.utils.pad(date.getDate()));
|
|
|
|
t = t.replace(/DDth/g,date.getDate()+$tw.utils.getDaySuffix(date));
|
|
|
|
t = t.replace(/DD/g,date.getDate());
|
|
|
|
var tz = date.getTimezoneOffset();
|
|
|
|
var atz = Math.abs(tz);
|
|
|
|
t = t.replace(/TZD/g,(tz < 0 ? '+' : '-') + $tw.utils.pad(Math.floor(atz / 60)) + ':' + $tw.utils.pad(atz % 60));
|
|
|
|
t = t.replace(/\\/g,"");
|
|
|
|
return t;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getAmPm = function(date) {
|
|
|
|
return date.getHours() >= 12 ? $tw.config.dateFormats.pm : $tw.config.dateFormats.am;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getDaySuffix = function(date) {
|
|
|
|
return $tw.config.dateFormats.daySuffixes[date.getDate()-1];
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getWeek = function(date) {
|
|
|
|
var dt = new Date(date.getTime());
|
|
|
|
var d = dt.getDay();
|
|
|
|
if(d === 0) d=7;// JavaScript Sun=0, ISO Sun=7
|
|
|
|
dt.setTime(dt.getTime()+(4-d)*86400000);// shift day to Thurs of same week to calculate weekNo
|
|
|
|
var n = Math.floor((dt.getTime()-new Date(dt.getFullYear(),0,1)+3600000)/86400000);
|
|
|
|
return Math.floor(n/7)+1;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getYearForWeekNo = function(date) {
|
|
|
|
var dt = new Date(date.getTime());
|
|
|
|
var d = dt.getDay();
|
|
|
|
if(d === 0) d=7;// JavaScript Sun=0, ISO Sun=7
|
|
|
|
dt.setTime(dt.getTime()+(4-d)*86400000);// shift day to Thurs of same week
|
|
|
|
return dt.getFullYear();
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getHours12 = function(date) {
|
|
|
|
var h = date.getHours();
|
|
|
|
return h > 12 ? h-12 : ( h > 0 ? h : 12 );
|
|
|
|
};
|
|
|
|
|
|
|
|
// Convert & to "&", < to "<", > to ">" and " to """
|
|
|
|
exports.htmlEncode = function(s)
|
|
|
|
{
|
|
|
|
if(s) {
|
|
|
|
return s.toString().replace(/&/mg,"&").replace(/</mg,"<").replace(/>/mg,">").replace(/\"/mg,""");
|
|
|
|
} else {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Converts all HTML entities to their character equivalents
|
|
|
|
exports.entityDecode = function(s) {
|
|
|
|
var e = s.substr(1,s.length-2); // Strip the & and the ;
|
|
|
|
if(e.charAt(0) === "#") {
|
|
|
|
if(e.charAt(1) === "x" || e.charAt(1) === "X") {
|
|
|
|
return String.fromCharCode(parseInt(e.substr(2),16));
|
|
|
|
} else {
|
|
|
|
return String.fromCharCode(parseInt(e.substr(1),10));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
var c = $tw.config.htmlEntities[e];
|
|
|
|
if(c) {
|
|
|
|
return String.fromCharCode(c);
|
|
|
|
} else {
|
|
|
|
return s; // Couldn't convert it as an entity, just return it raw
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.unescapeLineBreaks = function(s) {
|
|
|
|
return s.replace(/\\n/mg,"\n").replace(/\\b/mg," ").replace(/\\s/mg,"\\").replace(/\r/mg,"");
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns an escape sequence for given character. Uses \x for characters <=
|
|
|
|
* 0xFF to save space, \u for the rest.
|
|
|
|
*
|
|
|
|
* The code needs to be in sync with th code template in the compilation
|
|
|
|
* function for "action" nodes.
|
|
|
|
*/
|
|
|
|
// Copied from peg.js, thanks to David Majda
|
|
|
|
exports.escape = function(ch) {
|
|
|
|
var charCode = ch.charCodeAt(0);
|
|
|
|
if (charCode <= 0xFF) {
|
|
|
|
return '\\x' + $tw.utils.pad(charCode.toString(16).toUpperCase());
|
|
|
|
} else {
|
|
|
|
return '\\u' + $tw.utils.pad(charCode.toString(16).toUpperCase(),4);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Turns a string into a legal JavaScript string
|
|
|
|
// Copied from peg.js, thanks to David Majda
|
|
|
|
exports.stringify = function(s) {
|
|
|
|
/*
|
|
|
|
* ECMA-262, 5th ed., 7.8.4: All characters may appear literally in a string
|
|
|
|
* literal except for the closing quote character, backslash, carriage return,
|
|
|
|
* line separator, paragraph separator, and line feed. Any character may
|
|
|
|
* appear in the form of an escape sequence.
|
|
|
|
*
|
|
|
|
* For portability, we also escape escape all non-ASCII characters.
|
|
|
|
*/
|
|
|
|
return s
|
|
|
|
.replace(/\\/g, '\\\\') // backslash
|
|
|
|
.replace(/"/g, '\\"') // double quote character
|
|
|
|
.replace(/'/g, "\\'") // single quote character
|
|
|
|
.replace(/\r/g, '\\r') // carriage return
|
|
|
|
.replace(/\n/g, '\\n') // line feed
|
|
|
|
.replace(/[\x80-\uFFFF]/g, exports.escape); // non-ASCII characters
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.nextTick = function(fn) {
|
|
|
|
/*global window: false */
|
|
|
|
if(typeof window !== "undefined") {
|
|
|
|
// Apparently it would be faster to use postMessage - http://dbaron.org/log/20100309-faster-timeouts
|
|
|
|
window.setTimeout(fn,4);
|
|
|
|
} else {
|
|
|
|
process.nextTick(fn);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Determines whether element 'a' contains element 'b'
|
|
|
|
Code thanks to John Resig, http://ejohn.org/blog/comparing-document-position/
|
|
|
|
*/
|
|
|
|
exports.domContains = function(a,b) {
|
|
|
|
return a.contains ?
|
|
|
|
a != b && a.contains(b) :
|
|
|
|
!!(a.compareDocumentPosition(b) & 16);
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.hasClass = function(el,className) {
|
|
|
|
return el.className.split(" ").indexOf(className) !== -1;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.addClass = function(el,className) {
|
|
|
|
var c = el.className.split(" ");
|
|
|
|
if(c.indexOf(className) === -1) {
|
|
|
|
c.push(className);
|
|
|
|
}
|
|
|
|
el.className = c.join(" ");
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.removeClass = function(el,className) {
|
|
|
|
var c = el.className.split(" "),
|
|
|
|
p = c.indexOf(className);
|
|
|
|
if(p !== -1) {
|
|
|
|
c.splice(p,1);
|
|
|
|
el.className = c.join(" ");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.toggleClass = function(el,className,status) {
|
|
|
|
if(status === undefined) {
|
|
|
|
status = !exports.hasClass(el,className);
|
|
|
|
}
|
|
|
|
if(status) {
|
|
|
|
exports.addClass(el,className);
|
|
|
|
} else {
|
|
|
|
exports.removeClass(el,className);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.applyStyleSheet = function(id,css) {
|
|
|
|
var el = document.getElementById(id);
|
|
|
|
if(document.createStyleSheet) { // Older versions of IE
|
|
|
|
if(el) {
|
|
|
|
el.parentNode.removeChild(el);
|
|
|
|
}
|
|
|
|
document.getElementsByTagName("head")[0].insertAdjacentHTML("beforeEnd",
|
|
|
|
' <style id="' + id + '" type="text/css">' + css + '</style>'); // fails without
|
|
|
|
} else { // Modern browsers
|
|
|
|
if(el) {
|
|
|
|
el.replaceChild(document.createTextNode(css), el.firstChild);
|
|
|
|
} else {
|
|
|
|
el = document.createElement("style");
|
|
|
|
el.type = "text/css";
|
|
|
|
el.id = id;
|
|
|
|
el.appendChild(document.createTextNode(css));
|
|
|
|
document.getElementsByTagName("head")[0].appendChild(el);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-05-02 10:02:47 +00:00
|
|
|
/*
|
|
|
|
Parse a version number string of the form "1.2.3", "1.2.3.a4", or "1.2.3.b4" into:
|
|
|
|
{major: <number>, minor: <number>, revision: <number>, alpha: <number>, beta: <number>}
|
|
|
|
*/
|
|
|
|
exports.parseVersion = function(versionString) {
|
|
|
|
versionString = versionString || "0.0.0";
|
|
|
|
var components = versionString.split("."),
|
|
|
|
version = {
|
|
|
|
major: components[0] ? parseInt(components[0],10) : null,
|
|
|
|
minor: components[1] ? parseInt(components[1],10) : null,
|
|
|
|
revision: components[2] ? parseInt(components[2],10) : null
|
|
|
|
};
|
|
|
|
if(components[3]) {
|
|
|
|
if(components[3].charAt(0) === "a") {
|
|
|
|
version.alpha = parseInt(components[3].substr(1),10);
|
|
|
|
} else if(components[3].charAt(0) === "b") {
|
|
|
|
version.beta = parseInt(components[3].substr(1),10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return version;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getVersionString = function() {
|
|
|
|
return ($tw.version.major || "0") + "." +
|
|
|
|
($tw.version.minor || "0") + "." +
|
|
|
|
($tw.version.revision || "0") +
|
|
|
|
($tw.version.alpha ? ".a" + $tw.version.alpha : "") +
|
|
|
|
($tw.version.beta ? ".b" + $tw.version.beta : "");
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
Extract the version number from the meta tag or from the boot file
|
|
|
|
*/
|
|
|
|
|
|
|
|
if($tw.isBrowser) {
|
|
|
|
|
|
|
|
// Browser version
|
|
|
|
exports.extractVersionInfo = function() {
|
|
|
|
var metatags = document.getElementsByTagName("meta");
|
|
|
|
for(var t=0; t<metatags.length; t++) {
|
|
|
|
var m = metatags[t];
|
|
|
|
if(m.name === "tiddlywiki-version") {
|
|
|
|
return $tw.utils.parseVersion(m.content);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $tw.utils.parseVersion();
|
|
|
|
};
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
// Server version
|
|
|
|
exports.extractVersionInfo = function() {
|
|
|
|
var fs = require("fs");
|
|
|
|
return $tw.utils.parseVersion(fs.readFileSync($tw.boot.bootPath + "/version.txt").toString("utf8"));
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-04-30 11:23:03 +00:00
|
|
|
})();
|