All files / src/compiler/phases/3-transform/client/visitors global.js

98.83% Statements 169/171
95.23% Branches 60/63
100% Functions 5/5
98.8% Lines 165/167

Press n or j to go to the next uncovered block, b, p or k for the previous block.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 1682x 2x 2x 2x 2x 2x 2x 2x 2x 2x 18160x 14012x 25x 25x 13987x 13987x 13987x 13987x 14012x 14012x 14012x 2609x 14012x 21x 21x 21x 21x 21x 21x 6x 21x 6x 6x 21x 13981x 13981x 13981x 2x 2x 3024x 3024x 24x 24x 22x 22x 3024x 76x 76x 74x 74x 74x 13x 13x 74x 76x 2989x 2x 2x 1268x 2x 2x 184x 184x 184x 184x 138x 138x 138x 138x 138x 138x 138x 138x 138x 138x 138x 138x 27x 138x 122x 122x 122x 122x 122x 122x 122x 11x 11x 122x 111x 111x 111x 122x 122x 7x 7x 122x 122x 122x 16x 16x 183x 46x 46x 46x 1x 46x 1x 1x 1x 1x 1x 1x     1x 1x 46x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 4x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 45x 27x 27x 45x 15x 18x 3x 3x 3x 1x 3x 2x 2x 2x 2x 2x 2x 2x 3x 3x 45x 184x 2x  
/** @import { Expression, Node, Pattern, Statement } from 'estree' */
/** @import { Visitors } from '../types' */
import is_reference from 'is-reference';
import { serialize_get_binding, serialize_set_binding } from '../utils.js';
import * as b from '../../../../utils/builders.js';
import { is_ignored } from '../../../../state.js';
 
/** @type {Visitors} */
export const global_visitors = {
	Identifier(node, { path, state }) {
		if (is_reference(node, /** @type {Node} */ (path.at(-1)))) {
			if (node.name === '$$props') {
				return b.id('$$sanitized_props');
			}
 
			// Optimize prop access: If it's a member read access, we can use the $$props object directly
			const binding = state.scope.get(node.name);
			if (
				state.analysis.runes && // can't do this in legacy mode because the proxy does more than just read/write
				binding !== null &&
				node !== binding.node &&
				binding.kind === 'rest_prop'
			) {
				const parent = path.at(-1);
				const grand_parent = path.at(-2);
				if (
					parent?.type === 'MemberExpression' &&
					!parent.computed &&
					grand_parent?.type !== 'AssignmentExpression' &&
					grand_parent?.type !== 'UpdateExpression'
				) {
					return b.id('$$props');
				}
			}
 
			return serialize_get_binding(node, state);
		}
	},
	MemberExpression(node, { state, next }) {
		// rewrite `this.#foo` as `this.#foo.v` inside a constructor
		if (node.property.type === 'PrivateIdentifier') {
			const field = state.private_state.get(node.property.name);
			if (field) {
				return state.in_constructor ? b.member(node, b.id('v')) : b.call('$.get', node);
			}
		} else if (node.object.type === 'ThisExpression') {
			// rewrite `this.foo` as `this.#foo.v` inside a constructor
			if (node.property.type === 'Identifier' && !node.computed) {
				const field = state.public_state.get(node.property.name);
 
				if (field && state.in_constructor) {
					return b.member(b.member(b.this, field.id), b.id('v'));
				}
			}
		}
		next();
	},
	AssignmentExpression(node, context) {
		return serialize_set_binding(node, context, context.next);
	},
	UpdateExpression(node, context) {
		const { state, next, visit } = context;
		const argument = node.argument;
 
		if (argument.type === 'Identifier') {
			const binding = state.scope.get(argument.name);
			const is_store = binding?.kind === 'store_sub';
			const name = is_store ? argument.name.slice(1) : argument.name;
 
			// use runtime functions for smaller output
			if (
				binding?.kind === 'state' ||
				binding?.kind === 'frozen_state' ||
				binding?.kind === 'each' ||
				binding?.kind === 'legacy_reactive' ||
				binding?.kind === 'prop' ||
				binding?.kind === 'bindable_prop' ||
				is_store
			) {
				/** @type {Expression[]} */
				const args = [];
 
				let fn = '$.update';
				if (node.prefix) fn += '_pre';
 
				if (is_store) {
					fn += '_store';
					args.push(serialize_get_binding(b.id(name), state), b.call('$' + name));
				} else {
					if (binding.kind === 'prop' || binding.kind === 'bindable_prop') fn += '_prop';
					args.push(b.id(name));
				}
 
				if (node.operator === '--') {
					args.push(b.literal(-1));
				}
 
				return b.call(fn, ...args);
			}
 
			return next();
		} else if (
			argument.type === 'MemberExpression' &&
			argument.object.type === 'ThisExpression' &&
			argument.property.type === 'PrivateIdentifier' &&
			context.state.private_state.has(argument.property.name)
		) {
			let fn = '$.update';
			if (node.prefix) fn += '_pre';
 
			/** @type {Expression[]} */
			const args = [argument];
			if (node.operator === '--') {
				args.push(b.literal(-1));
			}
 
			return b.call(fn, ...args);
		} else {
			const ignore_invalid_mutation =
				is_ignored(node, 'ownership_invalid_mutation') && context.state.options.dev;
 
			/**
			 *
			 * @param {any} serialized
			 * @returns
			 */
			function maybe_wrap_skip_ownership(serialized) {
				if (!ignore_invalid_mutation) return serialized;
				return b.call('$.skip_ownership_validation', b.thunk(serialized));
			}
 
			// turn it into an IIFEE assignment expression: i++ -> (() => { const $$value = i; i+=1; return $$value; })
			const assignment = b.assignment(
				node.operator === '++' ? '+=' : '-=',
				/** @type {Pattern} */ (argument),
				b.literal(1)
			);
			const serialized_assignment = serialize_set_binding(
				assignment,
				context,
				() => assignment,
				node.prefix
			);
			const value = /** @type {Expression} */ (visit(argument));
			if (serialized_assignment === assignment) {
				// No change to output -> nothing to transform -> we can keep the original update expression
				return maybe_wrap_skip_ownership(next());
			} else if (context.state.analysis.runes) {
				return maybe_wrap_skip_ownership(serialized_assignment);
			} else {
				/** @type {Statement[]} */
				let statements;
				if (node.prefix) {
					statements = [b.stmt(serialized_assignment), b.return(value)];
				} else {
					const tmp_id = state.scope.generate('$$value');
					statements = [
						b.const(tmp_id, value),
						b.stmt(serialized_assignment),
						b.return(b.id(tmp_id))
					];
				}
				return maybe_wrap_skip_ownership(b.call(b.thunk(b.block(statements))));
			}
		}
	}
};