2023-09-13 16:49:22 +02:00
|
|
|
import type { Context } from './context.js';
|
2022-10-10 09:01:15 -04:00
|
|
|
import { incrementId } from './context.js';
|
2024-08-28 12:41:35 +02:00
|
|
|
import type {
|
|
|
|
ArrayObjectMapping,
|
|
|
|
AstroPreactAttrs,
|
|
|
|
PropNameToSignalMap,
|
|
|
|
SignalLike,
|
|
|
|
SignalToKeyOrIndexMap,
|
2024-08-28 10:42:23 +00:00
|
|
|
Signals,
|
2024-08-28 12:41:35 +02:00
|
|
|
} from './types.js';
|
2022-10-10 09:01:15 -04:00
|
|
|
|
|
|
|
function isSignal(x: any): x is SignalLike {
|
|
|
|
return x != null && typeof x === 'object' && typeof x.peek === 'function' && 'value' in x;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function restoreSignalsOnProps(ctx: Context, props: Record<string, any>) {
|
|
|
|
// Restore signal props that were mutated for serialization
|
|
|
|
let propMap: PropNameToSignalMap;
|
|
|
|
if (ctx.propsToSignals.has(props)) {
|
|
|
|
propMap = ctx.propsToSignals.get(props)!;
|
|
|
|
} else {
|
|
|
|
propMap = new Map();
|
|
|
|
ctx.propsToSignals.set(props, propMap);
|
|
|
|
}
|
|
|
|
for (const [key, signal] of propMap) {
|
|
|
|
props[key] = signal;
|
|
|
|
}
|
|
|
|
return propMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function serializeSignals(
|
|
|
|
ctx: Context,
|
|
|
|
props: Record<string, any>,
|
|
|
|
attrs: AstroPreactAttrs,
|
|
|
|
map: PropNameToSignalMap,
|
|
|
|
) {
|
|
|
|
// Check for signals
|
2024-08-28 12:41:35 +02:00
|
|
|
const signals: Signals = {};
|
2022-10-10 09:01:15 -04:00
|
|
|
for (const [key, value] of Object.entries(props)) {
|
2024-08-28 12:41:35 +02:00
|
|
|
const isPropArray = Array.isArray(value);
|
2024-09-06 13:26:51 +02:00
|
|
|
// `typeof null` is 'object' in JS, so we need to check for `null` specifically
|
2024-09-06 11:27:38 +00:00
|
|
|
const isPropObject =
|
|
|
|
!isSignal(value) && typeof props[key] === 'object' && props[key] !== null && !isPropArray;
|
2024-08-28 12:41:35 +02:00
|
|
|
|
|
|
|
if (isPropObject || isPropArray) {
|
|
|
|
const values = isPropObject ? Object.keys(props[key]) : value;
|
|
|
|
values.forEach((valueKey: number | string, valueIndex: number) => {
|
|
|
|
const signal = isPropObject ? props[key][valueKey] : valueKey;
|
|
|
|
if (isSignal(signal)) {
|
|
|
|
const keyOrIndex = isPropObject ? valueKey.toString() : valueIndex;
|
|
|
|
|
|
|
|
props[key] = isPropObject
|
|
|
|
? Object.assign({}, props[key], { [keyOrIndex]: signal.peek() })
|
|
|
|
: props[key].map((v: SignalLike, i: number) =>
|
|
|
|
i === valueIndex ? [signal.peek(), i] : v,
|
|
|
|
);
|
|
|
|
|
|
|
|
const currentMap = (map.get(key) || []) as SignalToKeyOrIndexMap;
|
|
|
|
map.set(key, [...currentMap, [signal, keyOrIndex]]);
|
|
|
|
|
|
|
|
const currentSignals = (signals[key] || []) as ArrayObjectMapping;
|
|
|
|
signals[key] = [...currentSignals, [getSignalId(ctx, signal), keyOrIndex]];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else if (isSignal(value)) {
|
2022-10-10 09:01:15 -04:00
|
|
|
// Set the value to the current signal value
|
|
|
|
// This mutates the props on purpose, so that it will be serialized correct.
|
|
|
|
props[key] = value.peek();
|
|
|
|
map.set(key, value);
|
|
|
|
|
2024-08-28 12:41:35 +02:00
|
|
|
signals[key] = getSignalId(ctx, value);
|
2022-10-10 09:01:15 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Object.keys(signals).length) {
|
|
|
|
attrs['data-preact-signals'] = JSON.stringify(signals);
|
|
|
|
}
|
|
|
|
}
|
2024-08-28 12:41:35 +02:00
|
|
|
|
|
|
|
function getSignalId(ctx: Context, item: SignalLike) {
|
|
|
|
let id = ctx.signals.get(item);
|
|
|
|
if (!id) {
|
|
|
|
id = incrementId(ctx);
|
|
|
|
ctx.signals.set(item, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|