-
-
Notifications
You must be signed in to change notification settings - Fork 97
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
@preact/signals-react-transform
issues
#412
Comments
Not every component uses |
import { signal } from "@preact/signals-react";
const count = signal(0);
function CounterValue() {
// Whenever the `count` signal is updated, we'll
// re-render this component automatically for you
return <p>Value: {count.value}</p>;
} After the babel transform runs, it'll look something like: import { signal, useSignals } from "@preact/signals-react";
const count = signal(0);
function CounterValue() {
const effect = useSignals();
try {
// Whenever the `count` signal is updated, we'll
// re-render this component automatically for you
return <p>Value: {count.value}</p>;
} finally {
effect.endTracking();
}
} The This is not true statement, effect actually has no endTracking. |
Workaround for passing signal into jsx: import { Signal } from '@preact/signals-react';
import { useSignals } from '@preact/signals-react/runtime';
/** @noTrackSignals */
const SignalValue = ({ data }: { data: Signal }) => {
const effect = useSignals();
try {
return data.value;
} finally {
effect.f();
}
};
Object.defineProperty(Signal.prototype, 'type', {
configurable: true,
value: SignalValue,
}); |
What to to with this kind of code? import { ReadonlySignal, Signal } from "@preact-signals/unified-signals";
import { Fn, Objects } from "hotscript";
import { createTransformProps } from "react-fast-hoc";
import { Uncached } from "../$";
export interface WithSignalProp extends Fn {
return: this["arg1"] extends "children"
? this["arg0"]
: this["arg0"] extends (...args: any[]) => any
? this["arg0"]
: this["arg0"] extends Uncached<any> | ReadonlySignal<any>
? never | Uncached<this["arg0"]> | ReadonlySignal<this["arg0"]>
: this["arg0"] | Uncached<this["arg0"]> | ReadonlySignal<this["arg0"]>;
}
class WithSignalPropsHandler
implements ProxyHandler<Record<string | symbol, any>>
{
#valuesCache = new Map<string | symbol, unknown>();
get(target: Record<string | symbol, any>, p: string | symbol) {
const value = target[p];
if (!value) {
return value;
}
if (value instanceof Uncached || value instanceof Signal) {
return (
this.#valuesCache.get(p) ??
this.#valuesCache.set(p, value.value).get(p)!
);
}
return value;
}
}
/**
* Allows to pass props to third party components that are not aware of signals. This will subscribe to signals on demand.
*/
export const withSignalProps = createTransformProps<
[Objects.MapValues<WithSignalProp>]
>((props) => new Proxy(props, new WithSignalPropsHandler()), {
namePrefix: "Reactified.",
mimicToNewComponent: false,
}); I'm just wrapping props with proxy and reading signals on demand. How it will perform if I add Even if it will clone component in which i applied track signals, there is an issue with destructuring of props. For example: /** @trackSignals */
const A = ({a, b}) => <a>{a + b}</a> Transforms into /** @trackSignals */
const A = ({a, b}) => {
// losts reactivity, because `useSignals` used after destructuring
const effect = useSignals();
try {
return (<a>{a + b}</a>)
} finally {
effect.f()
}
} |
"All" mode transforms all components to be reactive to signals. You can still opt-out using `/** @notrackSignals */`. Related: #412
React trying to execute component recursively on third render, so it makes effect called while other effect works it("should not crash on signal change while rendering multiple times", async () => {
// this bug is not occurs in strict mode
function App() {
const s = useSignals();
try {
const sig = useSignal(0);
sig.value;
if (sig.peek() < 100) {
sig.value += 1;
}
return sig.value;
} finally {
s.f();
}
}
await render(<App />);
expect(scratch.innerHTML).to.equal("100");
}); |
react signals are not reactive for me anymore neither |
@Loque18 What is the lib version. Seems to be issue was written for 1.3.x version |
Correct setup:
Preact signals transform using @preact/signals-react, so it still patching react and we will have the same issues (#346 #411). So we should patch
@preact/signals-react
, because patching of react is inlined here. So just replacefunction(){Object.defineProperty(n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentDispatcher
tofunction(){return;Object.defineProperty(n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentDispatcher
App is launching but many places lost reactivity, for example:
Signals cannot be reactively inlined to jsx
The text was updated successfully, but these errors were encountered: