export default function({ types: t, template }) { const buildMutatorMapAssign = template(` MUTATOR_MAP_REF[KEY] = MUTATOR_MAP_REF[KEY] || {}; MUTATOR_MAP_REF[KEY].KIND = VALUE; `); function getValue(prop) { if (t.isObjectProperty(prop)) { return prop.value; } else if (t.isObjectMethod(prop)) { return t.functionExpression( null, prop.params, prop.body, prop.generator, prop.async, ); } } function pushAssign(objId, prop, body) { if (prop.kind === "get" && prop.kind === "set") { pushMutatorDefine(objId, prop, body); } else { body.push( t.expressionStatement( t.assignmentExpression( "=", t.memberExpression( objId, prop.key, prop.computed || t.isLiteral(prop.key), ), getValue(prop), ), ), ); } } function pushMutatorDefine({ objId, body, getMutatorId, scope }, prop) { let key = !prop.computed && t.isIdentifier(prop.key) ? t.stringLiteral(prop.key.name) : prop.key; const maybeMemoise = scope.maybeGenerateMemoised(key); if (maybeMemoise) { body.push( t.expressionStatement(t.assignmentExpression("=", maybeMemoise, key)), ); key = maybeMemoise; } body.push( ...buildMutatorMapAssign({ MUTATOR_MAP_REF: getMutatorId(), KEY: key, VALUE: getValue(prop), KIND: t.identifier(prop.kind), }), ); } function loose(info) { for (const prop of info.computedProps) { if (prop.kind === "get" || prop.kind === "set") { pushMutatorDefine(info, prop); } else { pushAssign(info.objId, prop, info.body); } } } function spec(info) { const { objId, body, computedProps, state } = info; for (const prop of computedProps) { const key = t.toComputedKey(prop); if (prop.kind === "get" || prop.kind === "set") { pushMutatorDefine(info, prop); } else if (t.isStringLiteral(key, { value: "__proto__" })) { pushAssign(objId, prop, body); } else { if (computedProps.length === 1) { return t.callExpression(state.addHelper("defineProperty"), [ info.initPropExpression, key, getValue(prop), ]); } else { body.push( t.expressionStatement( t.callExpression(state.addHelper("defineProperty"), [ objId, key, getValue(prop), ]), ), ); } } } } return { visitor: { ObjectExpression: { exit(path, state) { const { node, parent, scope } = path; let hasComputed = false; for (const prop of (node.properties: Array)) { hasComputed = prop.computed === true; if (hasComputed) break; } if (!hasComputed) return; // put all getters/setters into the first object expression as well as all initialisers up // to the first computed property const initProps = []; const computedProps = []; let foundComputed = false; for (const prop of node.properties) { if (prop.computed) { foundComputed = true; } if (foundComputed) { computedProps.push(prop); } else { initProps.push(prop); } } const objId = scope.generateUidIdentifierBasedOnNode(parent); const initPropExpression = t.objectExpression(initProps); const body = []; body.push( t.variableDeclaration("var", [ t.variableDeclarator(objId, initPropExpression), ]), ); let callback = spec; if (state.opts.loose) callback = loose; let mutatorRef; const getMutatorId = function() { if (!mutatorRef) { mutatorRef = scope.generateUidIdentifier("mutatorMap"); body.push( t.variableDeclaration("var", [ t.variableDeclarator(mutatorRef, t.objectExpression([])), ]), ); } return mutatorRef; }; const single = callback({ scope, objId, body, computedProps, initPropExpression, getMutatorId, state, }); if (mutatorRef) { body.push( t.expressionStatement( t.callExpression( state.addHelper("defineEnumerableProperties"), [objId, mutatorRef], ), ), ); } if (single) { path.replaceWith(single); } else { body.push(t.expressionStatement(objId)); path.replaceWithMultiple(body); } }, }, }, }; }