todomvc
Version:
> Helping you select an MV\* framework
207 lines (180 loc) • 5.56 kB
JavaScript
/*!
* CanJS - 2.0.3
* http://canjs.us/
* Copyright (c) 2013 Bitovi
* Tue, 26 Nov 2013 18:21:22 GMT
* Licensed MIT
* Includes: CanJS default build
* Download from: http://canjs.us/
*/
define(["can/util/library"], function (can) {
// ##string.js
// _Miscellaneous string utility functions._
// Several of the methods in this plugin use code adapated from Prototype
// Prototype JavaScript framework, version 1.6.0.1.
// © 2005-2007 Sam Stephenson
var strUndHash = /_|-/,
strColons = /\=\=/,
strWords = /([A-Z]+)([A-Z][a-z])/g,
strLowUp = /([a-z\d])([A-Z])/g,
strDash = /([a-z\d])([A-Z])/g,
strReplacer = /\{([^\}]+)\}/g,
strQuote = /"/g,
strSingleQuote = /'/g,
strHyphenMatch = /-+(.)?/g,
strCamelMatch = /[a-z][A-Z]/g,
// Returns the `prop` property from `obj`.
// If `add` is true and `prop` doesn't exist in `obj`, create it as an
// empty object.
getNext = function (obj, prop, add) {
var result = obj[prop];
if (result === undefined && add === true) { result = obj[prop] = {} }
return result
},
// Returns `true` if the object can have properties (no `null`s).
isContainer = function (current) {
return (/^f|^o/).test(typeof current);
},
convertBadValues = function(content) {
// Convert bad values into empty strings
var isInvalid = content === null || content === undefined || (isNaN(content) && ("" + content === 'NaN'));
return ( "" + ( isInvalid ? '' : content ) )
};
can.extend(can, {
// Escapes strings for HTML.
esc: function (content) {
return convertBadValues(content)
.replace(/&/g, '&')
.replace(/</g, '<')
.replace(/>/g, '>')
.replace(strQuote, '"')
.replace(strSingleQuote, "'");
},
getObject: function (name, roots, add) {
// The parts of the name we are looking up
// `['App','Models','Recipe']`
var parts = name ? name.split('.') : [],
length = parts.length,
current,
r = 0,
i, container, rootsLength;
// Make sure roots is an `array`.
roots = can.isArray(roots) ? roots : [roots || window];
rootsLength = roots.length
if (!length) {
return roots[0];
}
// For each root, mark it as current.
for (r; r < rootsLength; r++) {
current = roots[r];
container = undefined;
// Walk current to the 2nd to last object or until there
// is not a container.
for (i = 0; i < length && isContainer(current); i++) {
container = current;
current = getNext(container, parts[i]);
}
// If we found property break cycle
if (container !== undefined && current !== undefined) {
break
}
}
// Remove property from found container
if (add === false && current !== undefined) {
delete container[parts[i - 1]]
}
// When adding property add it to the first root
if (add === true && current === undefined) {
current = roots[0]
for (i = 0; i < length && isContainer(current); i++) {
current = getNext(current, parts[i], true);
}
}
return current;
},
// Capitalizes a string.
/**
* @function can.capitalize
* @parent can.util
* @description Capitalize the first letter of a string.
* @signature `can.capitalize(str)`
* @param {String} str The string to capitalize.
* @return {String} The string with the first letter capitalized.
*
* can.capitalize('candy is fun!'); //-> Returns: 'Candy is fun!'
*
*/
capitalize: function (s, cache) {
// Used to make newId.
return s.charAt(0).toUpperCase() + s.slice(1);
},
/**
* @function can.camelize
* @parent can.util
* @description Takes a hyphenated string and converts it to a camelCase string..
* @signature `can.camelize(str)`
* @param {String} str The string to camelCase.
* @return {String} The camelCased string.
*
* can.camelize('array-count'); //-> 'arrayCount'
*
*/
camelize: function(str){
return convertBadValues(str).replace(strHyphenMatch, function(match, chr){
return chr ? chr.toUpperCase() : ''
})
},
/**
* @function can.hyphenate
* @parent can.util
* @description Hypenates a camelCase string, and makes it lower case.
* @signature `can.capitalize(str)`
* @param {String} str The camelCase string to hyphenate.
* @return {String} The hyphenated string.
*
* can.hyphenate('fooBarBaz'); //-> 'foo-bar-baz'
*
*/
hyphenate: function(str) {
return convertBadValues(str).replace(strCamelMatch, function(str, offset) {
return str.charAt(0) + '-' + str.charAt(1).toLowerCase();
});
},
// Underscores a string.
underscore: function (s) {
return s
.replace(strColons, '/')
.replace(strWords, '$1_$2')
.replace(strLowUp, '$1_$2')
.replace(strDash, '_')
.toLowerCase();
},
// Micro-templating.
/**
*/
sub: function (str, data, remove) {
var obs = [];
str = str || '';
obs.push(str.replace(strReplacer, function (whole, inside) {
// Convert inside to type.
var ob = can.getObject(inside, data, remove === true ? false : undefined);
if (ob === undefined || ob === null) {
obs = null;
return "";
}
// If a container, push into objs (which will return objects found).
if (isContainer(ob) && obs) {
obs.push(ob);
return "";
}
return "" + ob;
}));
return obs === null ? obs : (obs.length <= 1 ? obs[0] : obs);
},
// These regex's are used throughout the rest of can, so let's make
// them available.
replacer: strReplacer,
undHash: strUndHash
});
return can;
});