mirror of https://github.com/sveltejs/svelte
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
103 lines
1.9 KiB
103 lines
1.9 KiB
6 years ago
|
import { run_all, noop } from './internal';
|
||
7 years ago
|
|
||
6 years ago
|
export function readable(start, value) {
|
||
|
const subscribers = [];
|
||
|
let stop;
|
||
7 years ago
|
|
||
6 years ago
|
function set(newValue) {
|
||
|
if (newValue === value) return;
|
||
|
value = newValue;
|
||
6 years ago
|
subscribers.forEach(s => s[1]());
|
||
|
subscribers.forEach(s => s[0](value));
|
||
6 years ago
|
}
|
||
7 years ago
|
|
||
6 years ago
|
return {
|
||
6 years ago
|
subscribe(run, invalidate = noop) {
|
||
6 years ago
|
if (subscribers.length === 0) {
|
||
|
stop = start(set);
|
||
|
}
|
||
7 years ago
|
|
||
6 years ago
|
const subscriber = [run, invalidate];
|
||
|
subscribers.push(subscriber);
|
||
|
run(value);
|
||
7 years ago
|
|
||
6 years ago
|
return function() {
|
||
6 years ago
|
const index = subscribers.indexOf(subscriber);
|
||
6 years ago
|
if (index !== -1) subscribers.splice(index, 1);
|
||
|
|
||
|
if (subscribers.length === 0) {
|
||
|
stop && stop();
|
||
|
stop = null;
|
||
7 years ago
|
}
|
||
6 years ago
|
};
|
||
7 years ago
|
}
|
||
6 years ago
|
};
|
||
|
}
|
||
7 years ago
|
|
||
6 years ago
|
export function writable(value) {
|
||
|
const subscribers = [];
|
||
7 years ago
|
|
||
6 years ago
|
function set(newValue) {
|
||
|
if (newValue === value) return;
|
||
|
value = newValue;
|
||
6 years ago
|
subscribers.forEach(s => s[1]());
|
||
|
subscribers.forEach(s => s[0](value));
|
||
6 years ago
|
}
|
||
7 years ago
|
|
||
6 years ago
|
function update(fn) {
|
||
|
set(fn(value));
|
||
|
}
|
||
7 years ago
|
|
||
6 years ago
|
function subscribe(run, invalidate = noop) {
|
||
|
const subscriber = [run, invalidate];
|
||
|
subscribers.push(subscriber);
|
||
|
run(value);
|
||
7 years ago
|
|
||
6 years ago
|
return () => {
|
||
6 years ago
|
const index = subscribers.indexOf(subscriber);
|
||
6 years ago
|
if (index !== -1) subscribers.splice(index, 1);
|
||
|
};
|
||
|
}
|
||
7 years ago
|
|
||
6 years ago
|
return { set, update, subscribe };
|
||
|
}
|
||
7 years ago
|
|
||
6 years ago
|
export function derive(stores, fn) {
|
||
|
const single = !Array.isArray(stores);
|
||
|
if (single) stores = [stores];
|
||
7 years ago
|
|
||
6 years ago
|
const auto = fn.length === 1;
|
||
6 years ago
|
let value = {};
|
||
6 years ago
|
|
||
|
return readable(set => {
|
||
|
let inited = false;
|
||
|
const values = [];
|
||
|
|
||
6 years ago
|
let pending = 0;
|
||
|
|
||
6 years ago
|
const sync = () => {
|
||
6 years ago
|
if (pending) return;
|
||
6 years ago
|
const result = fn(single ? values[0] : values, set);
|
||
6 years ago
|
if (auto && (value !== (value = result))) set(result);
|
||
7 years ago
|
}
|
||
|
|
||
6 years ago
|
const unsubscribers = stores.map((store, i) => store.subscribe(
|
||
|
value => {
|
||
|
values[i] = value;
|
||
|
pending &= ~(1 << i);
|
||
|
if (inited) sync();
|
||
|
},
|
||
|
() => {
|
||
|
pending |= (1 << i);
|
||
|
})
|
||
|
);
|
||
7 years ago
|
|
||
6 years ago
|
inited = true;
|
||
|
sync();
|
||
|
|
||
|
return function stop() {
|
||
|
run_all(unsubscribers);
|
||
|
};
|
||
|
});
|
||
|
}
|