realm-js/lib/util.js

165 lines
3.9 KiB
JavaScript
Raw Normal View History

2015-10-28 10:37:17 -07:00
/* Copyright 2015 Realm Inc - All Rights Reserved
* Proprietary and Confidential
*/
2015-10-07 17:08:19 -07:00
'use strict';
const constants = require('./constants');
const rpc = require('./rpc');
2015-10-07 17:08:19 -07:00
const {keys} = constants;
const mutationListeners = {};
module.exports = {
fireMutationListeners,
createList,
createMethods,
2015-10-19 17:28:20 -07:00
createMethod,
getterForProperty,
setterForProperty,
};
2015-10-07 17:08:19 -07:00
function addMutationListener(realmId, callback) {
let listeners = mutationListeners[realmId] || (mutationListeners[realmId] = new Set());
listeners.add(callback);
}
function removeMutationListener(realmId, callback) {
let listeners = mutationListeners[realmId];
if (listeners) {
listeners.delete(callback);
}
}
function fireMutationListeners(realmId) {
let listeners = mutationListeners[realmId];
if (listeners) {
listeners.forEach((cb) => cb());
}
}
function createList(prototype, realmId, info, mutable) {
let list = Object.create(prototype);
2015-10-14 16:05:49 -07:00
let size = 0;
2015-10-07 17:08:19 -07:00
Object.defineProperties(list, {
'length': {
get: getterForProperty('length'),
},
'-1': {
value: undefined,
},
});
2015-10-07 17:08:19 -07:00
let resize = function(length) {
if (length == null) {
length = list.length;
}
if (length == size) {
2015-10-07 17:08:19 -07:00
return;
}
let props = {};
if (length > size) {
for (let i = size; i < length; i++) {
props[i] = {
get: getterForProperty(i),
set: mutable ? setterForProperty(i) : undefined,
enumerable: true,
configurable: true,
};
}
}
else if (length < size) {
for (let i = size; i >= length; i--) {
delete list[i];
}
}
2015-10-07 17:08:19 -07:00
// Helpfully throw an exception on attempts to set to list[list.length].
props[length] = {
value: undefined,
configurable: true,
};
Object.defineProperties(list, props);
size = length;
};
2015-10-07 17:08:19 -07:00
list[keys.realm] = realmId;
list[keys.id] = info.id;
list[keys.type] = info.type;
resize(info.size);
addMutationListener(realmId, function listener() {
try {
resize();
} catch (e) {
// If the error indicates the list was deleted, then remove this listener.
if (e.message == 'Tableview is not attached') {
removeMutationListener(realmId, listener);
} else {
throw e;
}
}
});
return list;
2015-10-07 17:08:19 -07:00
}
function createMethods(prototype, type, methodNames, mutates) {
let props = {};
methodNames.forEach((name) => {
props[name] = {
value: createMethod(type, name, mutates),
2015-10-19 17:28:20 -07:00
};
});
2015-10-19 17:28:20 -07:00
Object.defineProperties(prototype, props);
}
function createMethod(type, name, mutates) {
2015-10-19 17:28:20 -07:00
return function() {
let realmId = this[keys.realm];
let id = this[keys.id];
if (!realmId || !id) {
throw new TypeError(name + ' method was not called a Realm object!');
}
if (this[keys.type] !== type) {
throw new TypeError(name + ' method was called on an object of the wrong type!');
}
2015-10-19 17:28:20 -07:00
let result = rpc.callMethod(realmId, id, name, Array.from(arguments));
if (mutates) {
fireMutationListeners(realmId);
2015-10-19 17:28:20 -07:00
}
2015-10-19 17:28:20 -07:00
return result;
};
}
function getterForProperty(name) {
return function() {
return rpc.getProperty(this[keys.realm], this[keys.id], name);
};
}
function setterForProperty(name) {
return function(value) {
let realmId = this[keys.realm];
rpc.setProperty(realmId, this[keys.id], name, value);
// If this isn't a primitive value, then it might create a new object in the Realm.
if (value && typeof value == 'object') {
fireMutationListeners(realmId);
}
};
}