|
import type { Stores, StoresValues } from "svelte/store"; |
|
import { derived } from "svelte/store"; |
|
import { safeOnDestroy } from "./lifecycle"; |
|
import { noop } from "./noop"; |
|
|
|
type EffectOptions = { |
|
|
|
|
|
|
|
|
|
skipFirstRun?: boolean; |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function effect<S extends Stores>( |
|
stores: S, |
|
fn: (values: StoresValues<S>) => (() => void) | void, |
|
opts: EffectOptions = {} |
|
): () => void { |
|
const { skipFirstRun } = opts; |
|
let isFirstRun = true; |
|
let cb: (() => void) | void = undefined; |
|
|
|
|
|
const destroy = derived(stores, stores => { |
|
cb?.(); |
|
if (isFirstRun && skipFirstRun) { |
|
isFirstRun = false; |
|
} else { |
|
cb = fn(stores); |
|
} |
|
}).subscribe(noop); |
|
|
|
const unsub = () => { |
|
destroy(); |
|
cb?.(); |
|
}; |
|
|
|
|
|
safeOnDestroy(unsub); |
|
return unsub; |
|
} |
|
|