package.src.array.js Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of mikado Show documentation
Show all versions of mikado Show documentation
Web's fastest template library to build user interfaces.
The newest version!
// COMPILER BLOCK -->
import {
DEBUG,
PROFILER,
SUPPORT_CACHE,
SUPPORT_KEYED,
SUPPORT_POOLS,
SUPPORT_CALLBACKS,
SUPPORT_ASYNC,
SUPPORT_REACTIVE,
SUPPORT_EVENTS,
SUPPORT_DOM_HELPERS,
MIKADO_DOM,
MIKADO_LIVE_POOL,
MIKADO_CLASS,
MIKADO_TPL_KEY,
MIKADO_TPL_PATH,
MIKADO_NODE_CACHE,
MIKADO_PROXY
} from "./config.js";
import { tick } from "./profiler.js";
// <-- COMPILER BLOCK
import Mikado, { apply_proxy } from "./mikado.js";
const proto = Array.prototype;
/** @type {Proxy} */
const proxy = window["Proxy"];
/** @type {boolean} */
let skip = false;
/**
* @param {Mikado=} mikado
*/
function debug(mikado){
if(!mikado){
throw new Error("The observable array was not assigned to a Mikado instance. You need to pass in the observable array when initiating a Mikado instance.");
}
}
/**
* @param {Array|Observer=} array
* @returns {Observer|Proxy}
* @constructor
*/
export default function Observer(array){
if(array instanceof Observer){
return array;
}
if(!(this instanceof Observer)){
return new Observer(array);
}
PROFILER && tick("observer.create");
this.mikado = null;
const length = array ? array.length : 0;
if(proxy){
if(length){
for(let i = 0; i < length; i++){
this[i] = array[i];
}
}
/** @export */
this.length = length;
this.proto = {
splice: proto.splice.bind(this),
pop: proto.pop.bind(this),
shift: proto.shift.bind(this),
unshift: proto.unshift.bind(this),
push: proto.push.bind(this),
};
return new Proxy(this, handler);
}
else{
this.proto = array || [];
for(let i = 0; i <= length; i++){
this.define(i);
}
this.define("length");
}
}
/**
* @param {!Mikado} mikado
* @const
*/
Observer.prototype.mount = function(mikado){
PROFILER && tick("observer.mount");
if(this.mikado !== mikado){
this.mikado && mikado.mount(this.mikado.root);
this.mikado = mikado;
}
return this;
}
/**
* @param {string|number} key
* @private
* @const
*/
Observer.prototype.define = function(key){
PROFILER && tick("observer.define");
Object.defineProperty(this, /** @type {string} */ (key), {
get: /** @this {Observer} */ function(){
return this.proto[key];
},
set: /** @this {Observer} */ function(val){
if(typeof key === "number"){
if(key === this.length){
// create pointer +1 of array length to trigger array growth
this.define(key + 1);
}
handler.set(this, key, val);
}
}
});
return this;
}
const handler = /** @type {!ProxyHandler} */ ({
// actually we do not need a trap for the getter
// get: function(target, prop){
// return target[prop];
// },
/**
* @param {Observer} target
* @param {string|number} prop
* @param {Object|number} value
* @return {boolean}
*/
set: function(target, prop, value){
PROFILER && tick("observer.write");
let index_by_number;
if(typeof prop === "number"){
index_by_number = true;
}
else{
const index = parseInt(prop, 10);
if(prop === ("" + index)){
index_by_number = true;
}
}
const mikado = target.mikado;
if(!skip){
skip = true;
if(mikado){
const target_length = target.length;
if(index_by_number){
prop = /** @type {number} */ (prop);
DEBUG && debug(mikado);
const mikado_length = mikado.length;
if(target_length !== mikado_length){
target.length = mikado_length;
}
let node;
if(prop >= mikado_length){
mikado.add(value);
target.length++;
}
else if(prop < mikado_length){
node = mikado.dom[prop];
if(mikado.recycle || (SUPPORT_KEYED && mikado.key && node[MIKADO_TPL_KEY] === value[mikado.key])){
mikado.update(node, value, null, prop);
}
else{
mikado.replace(node, value, null, prop);
}
}
}
else{
if(prop === "length"){
value = /** @type {number} */ (value);
if(value < target_length){
mikado.remove(value, target_length - value);
}
}
}
}
skip = false;
}
if(mikado){
// when recycle is disabled, the existing proxy needs to by updated
const recycle = mikado.recycle || ((SUPPORT_KEYED && mikado.key));
if(index_by_number && mikado.proxy && (!recycle || !value[MIKADO_PROXY])){
prop = /** @type {number} */ (prop);
value = /** @type {Object} */ (value);
value = apply_proxy(mikado, mikado.dom[prop], value);
}
}
(proxy ? target : target.proto)[prop] = value;
// accept changes:
return true;
}
});
if(PROFILER){
handler.get = function(target, prop){
PROFILER && tick("observer.read");
return target[prop];
}
}
/**
* @param {Array} array
*/
Observer.prototype.set = function(array){
PROFILER && tick("observer.set");
const mikado = this.mikado;
if(mikado.recycle || (SUPPORT_KEYED && mikado.key)){
skip = true;
mikado.render(array);
skip = false;
if(this.length > array.length){
this.splice(array.length);
}
}
// else if(mikado.recycle){
//
// const length = array.length;
//
// for(let i = 0; i < length; i++){
//
// this[i] = array[i];
// }
//
// if(this.length > length){
//
// this.splice(length);
// }
// }
else{
this.splice();
this.concat(array);
}
return this;
};
/**
* @param {number=} start
* @param {number=} count
* @param {*=} insert
* @returns {Array<*>}
*/
Observer.prototype.splice = function(start, count, insert){
PROFILER && tick("observer.splice");
DEBUG && debug(this.mikado);
skip = true;
start || (start = 0);
if(typeof count === "undefined"){
count = this.length - start;
if(count < 0) count = 0;
}
count && this.mikado.remove(/** @type {number} */ (start), count);
const tmp = insert ? this.proto.splice(start, count, insert) : this.proto.splice(start, count);
insert && this.mikado.add(insert, start/*, this.mikado.view*/);
//this.length += (insert ? 1 : 0) - count;
skip = false;
return tmp;
};
/**
* @param {Object} data
*/
Observer.prototype.push = function(data){
PROFILER && tick("observer.push");
DEBUG && debug(this.mikado);
skip = true;
this.mikado.add(data);
this[this.length] = data;
if(proxy) this.length++;
skip = false;
};
/**
* @param {Object} data
*/
Observer.prototype.unshift = function(data){
PROFILER && tick("observer.unshift");
DEBUG && debug(this.mikado);
skip = true;
this.mikado.add(data, 0);
this.proto.unshift(data);
skip = false;
};
/**
* @return {Object}
*/
Observer.prototype.pop = function(){
PROFILER && tick("observer.pop");
DEBUG && debug(this.mikado);
skip = true;
this.mikado.remove(this.length - 1);
const tmp = this.proto.pop();
skip = false;
return tmp;
};
/**
* @return {Object}
*/
Observer.prototype.shift = function(){
PROFILER && tick("observer.shift");
DEBUG && debug(this.mikado);
skip = true;
this.mikado.remove(0);
const tmp = this.proto.shift();
skip = false;
return tmp;
};
/**
* @param {Array
© 2015 - 2025 Weber Informatics LLC | Privacy Policy