283 lines
6.1 KiB
JavaScript
283 lines
6.1 KiB
JavaScript
/**
|
|
* node.js - base abstract node for blessed
|
|
* Copyright (c) 2013-2015, Christopher Jeffrey and contributors (MIT License).
|
|
* https://github.com/chjj/blessed
|
|
*/
|
|
|
|
/**
|
|
* Modules
|
|
*/
|
|
|
|
var EventEmitter = require('../events').EventEmitter;
|
|
|
|
/**
|
|
* Node
|
|
*/
|
|
|
|
function Node(options) {
|
|
var self = this;
|
|
var Screen = require('./screen');
|
|
|
|
if (!(this instanceof Node)) {
|
|
return new Node(options);
|
|
}
|
|
|
|
EventEmitter.call(this);
|
|
|
|
options = options || {};
|
|
this.options = options;
|
|
|
|
this.screen = this.screen || options.screen;
|
|
|
|
if (!this.screen) {
|
|
if (this.type === 'screen') {
|
|
this.screen = this;
|
|
} else if (Screen.total === 1) {
|
|
this.screen = Screen.global;
|
|
} else if (options.parent) {
|
|
this.screen = options.parent;
|
|
while (this.screen && this.screen.type !== 'screen') {
|
|
this.screen = this.screen.parent;
|
|
}
|
|
} else if (Screen.total) {
|
|
// This _should_ work in most cases as long as the element is appended
|
|
// synchronously after the screen's creation. Throw error if not.
|
|
this.screen = Screen.instances[Screen.instances.length - 1];
|
|
process.nextTick(function() {
|
|
if (!self.parent) {
|
|
throw new Error('Element (' + self.type + ')'
|
|
+ ' was not appended synchronously after the'
|
|
+ ' screen\'s creation. Please set a `parent`'
|
|
+ ' or `screen` option in the element\'s constructor'
|
|
+ ' if you are going to use multiple screens and'
|
|
+ ' append the element later.');
|
|
}
|
|
});
|
|
} else {
|
|
throw new Error('No active screen.');
|
|
}
|
|
}
|
|
|
|
this.parent = options.parent || null;
|
|
this.children = [];
|
|
this.$ = this._ = this.data = {};
|
|
this.uid = Node.uid++;
|
|
this.index = this.index != null ? this.index : -1;
|
|
|
|
if (this.type !== 'screen') {
|
|
this.detached = true;
|
|
}
|
|
|
|
if (this.parent) {
|
|
this.parent.append(this);
|
|
}
|
|
|
|
(options.children || []).forEach(this.append.bind(this));
|
|
}
|
|
|
|
Node.uid = 0;
|
|
|
|
Node.prototype.__proto__ = EventEmitter.prototype;
|
|
|
|
Node.prototype.type = 'node';
|
|
|
|
Node.prototype.insert = function(element, i) {
|
|
var self = this;
|
|
|
|
if (element.screen && element.screen !== this.screen) {
|
|
throw new Error('Cannot switch a node\'s screen.');
|
|
}
|
|
|
|
element.detach();
|
|
element.parent = this;
|
|
element.screen = this.screen;
|
|
|
|
if (i === 0) {
|
|
this.children.unshift(element);
|
|
} else if (i === this.children.length) {
|
|
this.children.push(element);
|
|
} else {
|
|
this.children.splice(i, 0, element);
|
|
}
|
|
|
|
element.emit('reparent', this);
|
|
this.emit('adopt', element);
|
|
|
|
(function emit(el) {
|
|
var n = el.detached !== self.detached;
|
|
el.detached = self.detached;
|
|
if (n) el.emit('attach');
|
|
el.children.forEach(emit);
|
|
})(element);
|
|
|
|
if (!this.screen.focused) {
|
|
this.screen.focused = element;
|
|
}
|
|
};
|
|
|
|
Node.prototype.prepend = function(element) {
|
|
this.insert(element, 0);
|
|
};
|
|
|
|
Node.prototype.append = function(element) {
|
|
this.insert(element, this.children.length);
|
|
};
|
|
|
|
Node.prototype.insertBefore = function(element, other) {
|
|
var i = this.children.indexOf(other);
|
|
if (~i) this.insert(element, i);
|
|
};
|
|
|
|
Node.prototype.insertAfter = function(element, other) {
|
|
var i = this.children.indexOf(other);
|
|
if (~i) this.insert(element, i + 1);
|
|
};
|
|
|
|
Node.prototype.remove = function(element) {
|
|
if (element.parent !== this) return;
|
|
|
|
var i = this.children.indexOf(element);
|
|
if (!~i) return;
|
|
|
|
element.clearPos();
|
|
|
|
element.parent = null;
|
|
|
|
this.children.splice(i, 1);
|
|
|
|
i = this.screen.clickable.indexOf(element);
|
|
if (~i) this.screen.clickable.splice(i, 1);
|
|
i = this.screen.keyable.indexOf(element);
|
|
if (~i) this.screen.keyable.splice(i, 1);
|
|
|
|
element.emit('reparent', null);
|
|
this.emit('remove', element);
|
|
|
|
(function emit(el) {
|
|
var n = el.detached !== true;
|
|
el.detached = true;
|
|
if (n) el.emit('detach');
|
|
el.children.forEach(emit);
|
|
})(element);
|
|
|
|
if (this.screen.focused === element) {
|
|
this.screen.rewindFocus();
|
|
}
|
|
};
|
|
|
|
Node.prototype.detach = function() {
|
|
if (this.parent) this.parent.remove(this);
|
|
};
|
|
|
|
Node.prototype.free = function() {
|
|
return;
|
|
};
|
|
|
|
Node.prototype.destroy = function() {
|
|
this.detach();
|
|
this.forDescendants(function(el) {
|
|
el.free();
|
|
el.destroyed = true;
|
|
el.emit('destroy');
|
|
}, this);
|
|
};
|
|
|
|
Node.prototype.forDescendants = function(iter, s) {
|
|
if (s) iter(this);
|
|
this.children.forEach(function emit(el) {
|
|
iter(el);
|
|
el.children.forEach(emit);
|
|
});
|
|
};
|
|
|
|
Node.prototype.forAncestors = function(iter, s) {
|
|
var el = this;
|
|
if (s) iter(this);
|
|
while (el = el.parent) {
|
|
iter(el);
|
|
}
|
|
};
|
|
|
|
Node.prototype.collectDescendants = function(s) {
|
|
var out = [];
|
|
this.forDescendants(function(el) {
|
|
out.push(el);
|
|
}, s);
|
|
return out;
|
|
};
|
|
|
|
Node.prototype.collectAncestors = function(s) {
|
|
var out = [];
|
|
this.forAncestors(function(el) {
|
|
out.push(el);
|
|
}, s);
|
|
return out;
|
|
};
|
|
|
|
Node.prototype.emitDescendants = function() {
|
|
var args = Array.prototype.slice(arguments)
|
|
, iter;
|
|
|
|
if (typeof args[args.length - 1] === 'function') {
|
|
iter = args.pop();
|
|
}
|
|
|
|
return this.forDescendants(function(el) {
|
|
if (iter) iter(el);
|
|
el.emit.apply(el, args);
|
|
}, true);
|
|
};
|
|
|
|
Node.prototype.emitAncestors = function() {
|
|
var args = Array.prototype.slice(arguments)
|
|
, iter;
|
|
|
|
if (typeof args[args.length - 1] === 'function') {
|
|
iter = args.pop();
|
|
}
|
|
|
|
return this.forAncestors(function(el) {
|
|
if (iter) iter(el);
|
|
el.emit.apply(el, args);
|
|
}, true);
|
|
};
|
|
|
|
Node.prototype.hasDescendant = function(target) {
|
|
return (function find(el) {
|
|
for (var i = 0; i < el.children.length; i++) {
|
|
if (el.children[i] === target) {
|
|
return true;
|
|
}
|
|
if (find(el.children[i]) === true) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
})(this);
|
|
};
|
|
|
|
Node.prototype.hasAncestor = function(target) {
|
|
var el = this;
|
|
while (el = el.parent) {
|
|
if (el === target) return true;
|
|
}
|
|
return false;
|
|
};
|
|
|
|
Node.prototype.get = function(name, value) {
|
|
if (this.data.hasOwnProperty(name)) {
|
|
return this.data[name];
|
|
}
|
|
return value;
|
|
};
|
|
|
|
Node.prototype.set = function(name, value) {
|
|
return this.data[name] = value;
|
|
};
|
|
|
|
/**
|
|
* Expose
|
|
*/
|
|
|
|
module.exports = Node;
|