nyc-bookstores/node_modules/cheerio/lib/api/manipulation.js

193 lines
4.2 KiB
JavaScript
Raw Normal View History

2013-05-27 20:45:59 +00:00
var _ = require('underscore'),
parse = require('../parse'),
$ = require('../static'),
updateDOM = parse.update,
evaluate = parse.evaluate,
encode = require('../utils').encode,
slice = Array.prototype.slice;
/*
Creates an array of cheerio objects,
parsing strings if necessary
*/
var makeCheerioArray = function(elems) {
return _.reduce(elems, function(dom, elem) {
return dom.concat(elem.cheerio ? elem.toArray() : evaluate(elem));
}, []);
};
var _insert = function(concatenator) {
return function() {
var elems = slice.call(arguments),
dom = makeCheerioArray(elems);
return this.each(function(i, el) {
if (_.isFunction(elems[0])) return el; // not yet supported
updateDOM(concatenator(dom, el.children || (el.children = [])), el);
});
};
};
var append = exports.append = _insert(function(dom, children) {
return children.concat(dom);
});
var prepend = exports.prepend = _insert(function(dom, children) {
return dom.concat(children);
});
var after = exports.after = function() {
var elems = slice.call(arguments),
dom = makeCheerioArray(elems);
this.each(function(i, el) {
var siblings = el.parent.children,
index = siblings.indexOf(el);
// If not found, move on
if (!~index) return;
// Add element after `this` element
siblings.splice.apply(siblings, [++index, 0].concat(dom));
// Update next, prev, and parent pointers
updateDOM(siblings, el.parent);
el.parent.children = siblings;
});
return this;
};
var before = exports.before = function() {
var elems = slice.call(arguments),
dom = makeCheerioArray(elems);
this.each(function(i, el) {
var siblings = el.parent.children,
index = siblings.indexOf(el);
// If not found, move on
if (!~index) return;
// Add element before `el` element
siblings.splice.apply(siblings, [index, 0].concat(dom));
// Update next, prev, and parent pointers
updateDOM(siblings, el.parent);
el.parent.children = siblings;
});
return this;
};
/*
remove([selector])
*/
var remove = exports.remove = function(selector) {
var elems = this;
// Filter if we have selector
if (selector)
elems = elems.filter(selector);
elems.each(function(i, el) {
var siblings = el.parent.children,
index = siblings.indexOf(el);
if (!~index) return;
siblings.splice(index, 1);
// Update next, prev, and parent pointers
updateDOM(siblings, el.parent);
el.parent.children = siblings;
});
return this;
};
var replaceWith = exports.replaceWith = function(content) {
content = content.cheerio ? content.toArray() : evaluate(content);
this.each(function(i, el) {
var siblings = el.parent.children,
index = siblings.indexOf(el);
if (!~index) return;
siblings.splice.apply(siblings, [index, 1].concat(content));
updateDOM(siblings, el.parent);
el.parent.children = siblings;
});
return this;
};
var empty = exports.empty = function() {
this.each(function(i, el) {
el.children = [];
});
return this;
};
/**
* Set/Get the HTML
*/
var html = exports.html = function(str) {
if (str === undefined) {
if (!this[0] || !this[0].children) return null;
return $.html(this[0].children);
}
str = str.cheerio ? str.toArray() : evaluate(str);
this.each(function(i, el) {
el.children = str;
updateDOM(el.children, el);
});
return this;
};
var toString = exports.toString = function() {
return $.html(this);
};
var text = exports.text = function(str) {
// If `str` blank or an object
if (!str || typeof str === 'object') {
return $.text(this);
} else if (_.isFunction(str)) {
// Function support
return this.each(function(i, el) {
return this.text(str.call(el, i, this.text()));
});
}
var elem = {
data: encode(str),
type: 'text',
parent: null,
prev: null,
next: null,
children: []
};
// Append text node to each selected elements
this.each(function(i, el) {
el.children = elem;
updateDOM(el.children, el);
});
return this;
};
var clone = exports.clone = function() {
// Turn it into HTML, then recreate it,
// Seems to be the easiest way to reconnect everything correctly
return this.constructor($.html(this));
};