Skip to content

Commit

Permalink
Change delay to double microtask for prop updates (#609)
Browse files Browse the repository at this point in the history
* Change delay to double microtask for prop updates

* Add changest
  • Loading branch information
JoviDeCroock authored Dec 12, 2024
1 parent 18b2f29 commit 8e6e2de
Show file tree
Hide file tree
Showing 2 changed files with 74 additions and 21 deletions.
5 changes: 5 additions & 0 deletions .changeset/tidy-rivers-fly.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@preact/signals": patch
---

Change timing to a double microtask so we are behind the Preact render queue but can't delay as much as a user-input coming in.
90 changes: 69 additions & 21 deletions packages/preact/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -79,21 +79,47 @@ function SignalValue(this: AugmentedComponent, { data }: { data: Signal }) {
const currentSignal = useSignal(data);
currentSignal.value = data;

const s = useComputed(() => {
let data = currentSignal.value;
let s = data.value;
return s === 0 ? 0 : s === true ? "" : s || "";
});
const s = useMemo(() => {
let self = this;
// mark the parent component as having computeds so it gets optimized
let v = this.__v;
while ((v = v.__!)) {
if (v.__c) {
v.__c._updateFlags |= HAS_COMPUTEDS;
break;
}
}

const isText = useComputed(() => {
return !isValidElement(s.peek()) || this.base?.nodeType === 3;
});
const wrappedSignal = computed(function (this: Effect) {
let data = currentSignal.value;
let s = data.value;
return s === 0 ? 0 : s === true ? "" : s || "";
});

useSignalEffect(() => {
if (isText.value) {
(this.base as Text).data = data.peek();
}
});
const isText = computed(
() => isValidElement(wrappedSignal.value) || this.base?.nodeType !== 3
);

this._updater!._callback = () => {
if (isValidElement(s.peek()) || this.base?.nodeType !== 3) {
this._updateFlags |= HAS_PENDING_UPDATE;
this.setState({});
return;
}
(this.base as Text).data = s.peek();
};

effect(function (this: Effect) {
if (!oldNotify) oldNotify = this._notify;
this._notify = notifyDomUpdates;
const val = wrappedSignal.value;
if (isText.value && self.base) {
(self.base as Text).data = val;
}
});

return wrappedSignal;
}, []);

return s.value;
}
Expand Down Expand Up @@ -228,8 +254,8 @@ function createPropUpdater(
props = newProps;
},
_dispose: effect(function (this: Effect) {
if (!oldNotifyEffects) oldNotifyEffects = this._notify;
this._notify = notifyEffects;
if (!oldNotify) oldNotify = this._notify;
this._notify = notifyDomUpdates;
const value = changeSignal.value.value;
// If Preact just rendered this value, don't render it again:
if (props[prop] === value) return;
Expand Down Expand Up @@ -349,26 +375,48 @@ export function useComputed<T>(compute: () => T) {
return useMemo(() => computed<T>(() => $compute.current()), []);
}

let oldNotifyEffects: (this: Effect) => void,
effectsQueue: Array<Effect> = [];
let oldNotify: (this: Effect) => void,
effectsQueue: Array<Effect> = [],
domQueue: Array<Effect> = [];

const defer =
const deferEffects =
typeof requestAnimationFrame === "undefined"
? setTimeout
: requestAnimationFrame;

const deferDomUpdates = (cb: any) => {
queueMicrotask(() => {
queueMicrotask(cb);
});
};

function flushEffects() {
batch(() => {
let inst: Effect | undefined;
while ((inst = effectsQueue.shift())) {
oldNotifyEffects.call(inst);
oldNotify.call(inst);
}
});
}

function notifyEffects(this: Effect) {
if (effectsQueue.push(this) === 1) {
(options.requestAnimationFrame || defer)(flushEffects);
(options.requestAnimationFrame || deferEffects)(flushEffects);
}
}

function flushDomUpdates() {
batch(() => {
let inst: Effect | undefined;
while ((inst = domQueue.shift())) {
oldNotify.call(inst);
}
});
}

function notifyDomUpdates(this: Effect) {
if (domQueue.push(this) === 1) {
(options.requestAnimationFrame || deferDomUpdates)(flushDomUpdates);
}
}

Expand All @@ -378,7 +426,7 @@ export function useSignalEffect(cb: () => void | (() => void)) {

useEffect(() => {
return effect(function (this: Effect) {
if (!oldNotifyEffects) oldNotifyEffects = this._notify;
if (!oldNotify) oldNotify = this._notify;
this._notify = notifyEffects;
return callback.current();
});
Expand Down

0 comments on commit 8e6e2de

Please sign in to comment.