530 lines
13 KiB
JavaScript
530 lines
13 KiB
JavaScript
"use strict";
|
|
|
|
Object.defineProperty(exports, "__esModule", {
|
|
value: true
|
|
});
|
|
exports.default = applyDecs;
|
|
|
|
function createMetadataMethodsForProperty(metadataMap, kind, property, decoratorFinishedRef) {
|
|
return {
|
|
getMetadata: function (key) {
|
|
assertNotFinished(decoratorFinishedRef, "getMetadata");
|
|
assertMetadataKey(key);
|
|
var metadataForKey = metadataMap[key];
|
|
if (metadataForKey === void 0) return void 0;
|
|
|
|
if (kind === 1) {
|
|
var pub = metadataForKey.public;
|
|
|
|
if (pub !== void 0) {
|
|
return pub[property];
|
|
}
|
|
} else if (kind === 2) {
|
|
var priv = metadataForKey.private;
|
|
|
|
if (priv !== void 0) {
|
|
return priv.get(property);
|
|
}
|
|
} else if (Object.hasOwnProperty.call(metadataForKey, "constructor")) {
|
|
return metadataForKey.constructor;
|
|
}
|
|
},
|
|
setMetadata: function (key, value) {
|
|
assertNotFinished(decoratorFinishedRef, "setMetadata");
|
|
assertMetadataKey(key);
|
|
var metadataForKey = metadataMap[key];
|
|
|
|
if (metadataForKey === void 0) {
|
|
metadataForKey = metadataMap[key] = {};
|
|
}
|
|
|
|
if (kind === 1) {
|
|
var pub = metadataForKey.public;
|
|
|
|
if (pub === void 0) {
|
|
pub = metadataForKey.public = {};
|
|
}
|
|
|
|
pub[property] = value;
|
|
} else if (kind === 2) {
|
|
var priv = metadataForKey.priv;
|
|
|
|
if (priv === void 0) {
|
|
priv = metadataForKey.private = new Map();
|
|
}
|
|
|
|
priv.set(property, value);
|
|
} else {
|
|
metadataForKey.constructor = value;
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
function convertMetadataMapToFinal(obj, metadataMap) {
|
|
var parentMetadataMap = obj[Symbol.metadata || Symbol.for("Symbol.metadata")];
|
|
var metadataKeys = Object.getOwnPropertySymbols(metadataMap);
|
|
if (metadataKeys.length === 0) return;
|
|
|
|
for (var i = 0; i < metadataKeys.length; i++) {
|
|
var key = metadataKeys[i];
|
|
var metaForKey = metadataMap[key];
|
|
var parentMetaForKey = parentMetadataMap ? parentMetadataMap[key] : null;
|
|
var pub = metaForKey.public;
|
|
var parentPub = parentMetaForKey ? parentMetaForKey.public : null;
|
|
|
|
if (pub && parentPub) {
|
|
Object.setPrototypeOf(pub, parentPub);
|
|
}
|
|
|
|
var priv = metaForKey.private;
|
|
|
|
if (priv) {
|
|
var privArr = Array.from(priv.values());
|
|
var parentPriv = parentMetaForKey ? parentMetaForKey.private : null;
|
|
|
|
if (parentPriv) {
|
|
privArr = privArr.concat(parentPriv);
|
|
}
|
|
|
|
metaForKey.private = privArr;
|
|
}
|
|
|
|
if (parentMetaForKey) {
|
|
Object.setPrototypeOf(metaForKey, parentMetaForKey);
|
|
}
|
|
}
|
|
|
|
if (parentMetadataMap) {
|
|
Object.setPrototypeOf(metadataMap, parentMetadataMap);
|
|
}
|
|
|
|
obj[Symbol.metadata || Symbol.for("Symbol.metadata")] = metadataMap;
|
|
}
|
|
|
|
function createAddInitializerMethod(initializers, decoratorFinishedRef) {
|
|
return function addInitializer(initializer) {
|
|
assertNotFinished(decoratorFinishedRef, "addInitializer");
|
|
assertCallable(initializer, "An initializer");
|
|
initializers.push(initializer);
|
|
};
|
|
}
|
|
|
|
function memberDec(dec, name, desc, metadataMap, initializers, kind, isStatic, isPrivate, value) {
|
|
var kindStr;
|
|
|
|
switch (kind) {
|
|
case 1:
|
|
kindStr = "accessor";
|
|
break;
|
|
|
|
case 2:
|
|
kindStr = "method";
|
|
break;
|
|
|
|
case 3:
|
|
kindStr = "getter";
|
|
break;
|
|
|
|
case 4:
|
|
kindStr = "setter";
|
|
break;
|
|
|
|
default:
|
|
kindStr = "field";
|
|
}
|
|
|
|
var ctx = {
|
|
kind: kindStr,
|
|
name: isPrivate ? "#" + name : name,
|
|
isStatic: isStatic,
|
|
isPrivate: isPrivate
|
|
};
|
|
var decoratorFinishedRef = {
|
|
v: false
|
|
};
|
|
|
|
if (kind !== 0) {
|
|
ctx.addInitializer = createAddInitializerMethod(initializers, decoratorFinishedRef);
|
|
}
|
|
|
|
var metadataKind, metadataName;
|
|
|
|
if (isPrivate) {
|
|
metadataKind = 2;
|
|
metadataName = Symbol(name);
|
|
var access = {};
|
|
|
|
if (kind === 0) {
|
|
access.get = desc.get;
|
|
access.set = desc.set;
|
|
} else if (kind === 2) {
|
|
access.get = function () {
|
|
return desc.value;
|
|
};
|
|
} else {
|
|
if (kind === 1 || kind === 3) {
|
|
access.get = function () {
|
|
return desc.get.call(this);
|
|
};
|
|
}
|
|
|
|
if (kind === 1 || kind === 4) {
|
|
access.set = function (v) {
|
|
desc.set.call(this, v);
|
|
};
|
|
}
|
|
}
|
|
|
|
ctx.access = access;
|
|
} else {
|
|
metadataKind = 1;
|
|
metadataName = name;
|
|
}
|
|
|
|
try {
|
|
return dec(value, Object.assign(ctx, createMetadataMethodsForProperty(metadataMap, metadataKind, metadataName, decoratorFinishedRef)));
|
|
} finally {
|
|
decoratorFinishedRef.v = true;
|
|
}
|
|
}
|
|
|
|
function assertNotFinished(decoratorFinishedRef, fnName) {
|
|
if (decoratorFinishedRef.v) {
|
|
throw new Error("attempted to call " + fnName + " after decoration was finished");
|
|
}
|
|
}
|
|
|
|
function assertMetadataKey(key) {
|
|
if (typeof key !== "symbol") {
|
|
throw new TypeError("Metadata keys must be symbols, received: " + key);
|
|
}
|
|
}
|
|
|
|
function assertCallable(fn, hint) {
|
|
if (typeof fn !== "function") {
|
|
throw new TypeError(hint + " must be a function");
|
|
}
|
|
}
|
|
|
|
function assertValidReturnValue(kind, value) {
|
|
var type = typeof value;
|
|
|
|
if (kind === 1) {
|
|
if (type !== "object" || value === null) {
|
|
throw new TypeError("accessor decorators must return an object with get, set, or init properties or void 0");
|
|
}
|
|
|
|
if (value.get !== undefined) {
|
|
assertCallable(value.get, "accessor.get");
|
|
}
|
|
|
|
if (value.set !== undefined) {
|
|
assertCallable(value.set, "accessor.set");
|
|
}
|
|
|
|
if (value.init !== undefined) {
|
|
assertCallable(value.init, "accessor.init");
|
|
}
|
|
|
|
if (value.initializer !== undefined) {
|
|
assertCallable(value.initializer, "accessor.initializer");
|
|
}
|
|
} else if (type !== "function") {
|
|
var hint;
|
|
|
|
if (kind === 0) {
|
|
hint = "field";
|
|
} else if (kind === 10) {
|
|
hint = "class";
|
|
} else {
|
|
hint = "method";
|
|
}
|
|
|
|
throw new TypeError(hint + " decorators must return a function or void 0");
|
|
}
|
|
}
|
|
|
|
function getInit(desc) {
|
|
var initializer;
|
|
|
|
if ((initializer = desc.init) == null && (initializer = desc.initializer) && typeof console !== "undefined") {
|
|
console.warn(".initializer has been renamed to .init as of March 2022");
|
|
}
|
|
|
|
return initializer;
|
|
}
|
|
|
|
function applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, metadataMap, initializers) {
|
|
var decs = decInfo[0];
|
|
var desc, initializer, value;
|
|
|
|
if (isPrivate) {
|
|
if (kind === 0 || kind === 1) {
|
|
desc = {
|
|
get: decInfo[3],
|
|
set: decInfo[4]
|
|
};
|
|
} else if (kind === 3) {
|
|
desc = {
|
|
get: decInfo[3]
|
|
};
|
|
} else if (kind === 4) {
|
|
desc = {
|
|
set: decInfo[3]
|
|
};
|
|
} else {
|
|
desc = {
|
|
value: decInfo[3]
|
|
};
|
|
}
|
|
} else if (kind !== 0) {
|
|
desc = Object.getOwnPropertyDescriptor(base, name);
|
|
}
|
|
|
|
if (kind === 1) {
|
|
value = {
|
|
get: desc.get,
|
|
set: desc.set
|
|
};
|
|
} else if (kind === 2) {
|
|
value = desc.value;
|
|
} else if (kind === 3) {
|
|
value = desc.get;
|
|
} else if (kind === 4) {
|
|
value = desc.set;
|
|
}
|
|
|
|
var newValue, get, set;
|
|
|
|
if (typeof decs === "function") {
|
|
newValue = memberDec(decs, name, desc, metadataMap, initializers, kind, isStatic, isPrivate, value);
|
|
|
|
if (newValue !== void 0) {
|
|
assertValidReturnValue(kind, newValue);
|
|
|
|
if (kind === 0) {
|
|
initializer = newValue;
|
|
} else if (kind === 1) {
|
|
initializer = getInit(newValue);
|
|
get = newValue.get || value.get;
|
|
set = newValue.set || value.set;
|
|
value = {
|
|
get: get,
|
|
set: set
|
|
};
|
|
} else {
|
|
value = newValue;
|
|
}
|
|
}
|
|
} else {
|
|
for (var i = decs.length - 1; i >= 0; i--) {
|
|
var dec = decs[i];
|
|
newValue = memberDec(dec, name, desc, metadataMap, initializers, kind, isStatic, isPrivate, value);
|
|
|
|
if (newValue !== void 0) {
|
|
assertValidReturnValue(kind, newValue);
|
|
var newInit;
|
|
|
|
if (kind === 0) {
|
|
newInit = newValue;
|
|
} else if (kind === 1) {
|
|
newInit = getInit(newValue);
|
|
get = newValue.get || value.get;
|
|
set = newValue.set || value.set;
|
|
value = {
|
|
get: get,
|
|
set: set
|
|
};
|
|
} else {
|
|
value = newValue;
|
|
}
|
|
|
|
if (newInit !== void 0) {
|
|
if (initializer === void 0) {
|
|
initializer = newInit;
|
|
} else if (typeof initializer === "function") {
|
|
initializer = [initializer, newInit];
|
|
} else {
|
|
initializer.push(newInit);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (kind === 0 || kind === 1) {
|
|
if (initializer === void 0) {
|
|
initializer = function (instance, init) {
|
|
return init;
|
|
};
|
|
} else if (typeof initializer !== "function") {
|
|
var ownInitializers = initializer;
|
|
|
|
initializer = function (instance, init) {
|
|
var value = init;
|
|
|
|
for (var i = 0; i < ownInitializers.length; i++) {
|
|
value = ownInitializers[i].call(instance, value);
|
|
}
|
|
|
|
return value;
|
|
};
|
|
} else {
|
|
var originalInitializer = initializer;
|
|
|
|
initializer = function (instance, init) {
|
|
return originalInitializer.call(instance, init);
|
|
};
|
|
}
|
|
|
|
ret.push(initializer);
|
|
}
|
|
|
|
if (kind !== 0) {
|
|
if (kind === 1) {
|
|
desc.get = value.get;
|
|
desc.set = value.set;
|
|
} else if (kind === 2) {
|
|
desc.value = value;
|
|
} else if (kind === 3) {
|
|
desc.get = value;
|
|
} else if (kind === 4) {
|
|
desc.set = value;
|
|
}
|
|
|
|
if (isPrivate) {
|
|
if (kind === 1) {
|
|
ret.push(function (instance, args) {
|
|
return value.get.call(instance, args);
|
|
});
|
|
ret.push(function (instance, args) {
|
|
return value.set.call(instance, args);
|
|
});
|
|
} else if (kind === 2) {
|
|
ret.push(value);
|
|
} else {
|
|
ret.push(function (instance, args) {
|
|
return value.call(instance, args);
|
|
});
|
|
}
|
|
} else {
|
|
Object.defineProperty(base, name, desc);
|
|
}
|
|
}
|
|
}
|
|
|
|
function applyMemberDecs(ret, Class, protoMetadataMap, staticMetadataMap, decInfos) {
|
|
var protoInitializers;
|
|
var staticInitializers;
|
|
var existingProtoNonFields = new Map();
|
|
var existingStaticNonFields = new Map();
|
|
|
|
for (var i = 0; i < decInfos.length; i++) {
|
|
var decInfo = decInfos[i];
|
|
if (!Array.isArray(decInfo)) continue;
|
|
var kind = decInfo[1];
|
|
var name = decInfo[2];
|
|
var isPrivate = decInfo.length > 3;
|
|
var isStatic = kind >= 5;
|
|
var base;
|
|
var metadataMap;
|
|
var initializers;
|
|
|
|
if (isStatic) {
|
|
base = Class;
|
|
metadataMap = staticMetadataMap;
|
|
kind = kind - 5;
|
|
|
|
if (kind !== 0) {
|
|
staticInitializers = staticInitializers || [];
|
|
initializers = staticInitializers;
|
|
}
|
|
} else {
|
|
base = Class.prototype;
|
|
metadataMap = protoMetadataMap;
|
|
|
|
if (kind !== 0) {
|
|
protoInitializers = protoInitializers || [];
|
|
initializers = protoInitializers;
|
|
}
|
|
}
|
|
|
|
if (kind !== 0 && !isPrivate) {
|
|
var existingNonFields = isStatic ? existingStaticNonFields : existingProtoNonFields;
|
|
var existingKind = existingNonFields.get(name) || 0;
|
|
|
|
if (existingKind === true || existingKind === 3 && kind !== 4 || existingKind === 4 && kind !== 3) {
|
|
throw new Error("Attempted to decorate a public method/accessor that has the same name as a previously decorated public method/accessor. This is not currently supported by the decorators plugin. Property name was: " + name);
|
|
} else if (!existingKind && kind > 2) {
|
|
existingNonFields.set(name, kind);
|
|
} else {
|
|
existingNonFields.set(name, true);
|
|
}
|
|
}
|
|
|
|
applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, metadataMap, initializers);
|
|
}
|
|
|
|
pushInitializers(ret, protoInitializers);
|
|
pushInitializers(ret, staticInitializers);
|
|
}
|
|
|
|
function pushInitializers(ret, initializers) {
|
|
if (initializers) {
|
|
ret.push(function (instance) {
|
|
for (var i = 0; i < initializers.length; i++) {
|
|
initializers[i].call(instance);
|
|
}
|
|
|
|
return instance;
|
|
});
|
|
}
|
|
}
|
|
|
|
function applyClassDecs(ret, targetClass, metadataMap, classDecs) {
|
|
if (classDecs.length > 0) {
|
|
var initializers = [];
|
|
var newClass = targetClass;
|
|
var name = targetClass.name;
|
|
|
|
for (var i = classDecs.length - 1; i >= 0; i--) {
|
|
var decoratorFinishedRef = {
|
|
v: false
|
|
};
|
|
|
|
try {
|
|
var ctx = Object.assign({
|
|
kind: "class",
|
|
name: name,
|
|
addInitializer: createAddInitializerMethod(initializers, decoratorFinishedRef)
|
|
}, createMetadataMethodsForProperty(metadataMap, 0, name, decoratorFinishedRef));
|
|
var nextNewClass = classDecs[i](newClass, ctx);
|
|
} finally {
|
|
decoratorFinishedRef.v = true;
|
|
}
|
|
|
|
if (nextNewClass !== undefined) {
|
|
assertValidReturnValue(10, nextNewClass);
|
|
newClass = nextNewClass;
|
|
}
|
|
}
|
|
|
|
ret.push(newClass, function () {
|
|
for (var i = 0; i < initializers.length; i++) {
|
|
initializers[i].call(newClass);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
function applyDecs(targetClass, memberDecs, classDecs) {
|
|
var ret = [];
|
|
var staticMetadataMap = {};
|
|
var protoMetadataMap = {};
|
|
applyMemberDecs(ret, targetClass, protoMetadataMap, staticMetadataMap, memberDecs);
|
|
convertMetadataMapToFinal(targetClass.prototype, protoMetadataMap);
|
|
applyClassDecs(ret, targetClass, staticMetadataMap, classDecs);
|
|
convertMetadataMapToFinal(targetClass, staticMetadataMap);
|
|
return ret;
|
|
} |