mirror of https://github.com/sveltejs/svelte
parent
cffc51d007
commit
f7b194fb11
@ -1,200 +1,100 @@
|
|||||||
import { is_promise } from '../../../common.js';
|
import { is_promise } from '../../../common.js';
|
||||||
import { hydrate_block_anchor } from '../../hydration.js';
|
import { hydrate_block_anchor } from '../../hydration.js';
|
||||||
import { remove } from '../../reconciler.js';
|
import { set_current_effect } from '../../runtime.js';
|
||||||
import {
|
import { pause_effect, render_effect, resume_effect } from '../../reactivity/computations.js';
|
||||||
current_block,
|
import { BRANCH_EFFECT } from '../../constants.js';
|
||||||
destroy_signal,
|
|
||||||
execute_effect,
|
|
||||||
flushSync,
|
|
||||||
push_destroy_fn
|
|
||||||
} from '../../runtime.js';
|
|
||||||
import { render_effect } from '../../reactivity/computations.js';
|
|
||||||
import { trigger_transitions } from '../../transitions.js';
|
|
||||||
import { AWAIT_BLOCK, UNINITIALIZED } from '../../constants.js';
|
|
||||||
|
|
||||||
/** @returns {import('../../types.js').AwaitBlock} */
|
|
||||||
export function create_await_block() {
|
|
||||||
return {
|
|
||||||
// dom
|
|
||||||
d: null,
|
|
||||||
// effect
|
|
||||||
e: null,
|
|
||||||
// parent
|
|
||||||
p: /** @type {import('../../types.js').Block} */ (current_block),
|
|
||||||
// pending
|
|
||||||
n: true,
|
|
||||||
// transition
|
|
||||||
r: null,
|
|
||||||
// type
|
|
||||||
t: AWAIT_BLOCK
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @template V
|
* @template V
|
||||||
* @param {Comment} anchor_node
|
* @param {Comment} anchor_node
|
||||||
* @param {(() => Promise<V>)} input
|
* @param {(() => Promise<V>)} get_input
|
||||||
* @param {null | ((anchor: Node) => void)} pending_fn
|
* @param {null | ((anchor: Node) => void)} pending_fn
|
||||||
* @param {null | ((anchor: Node, value: V) => void)} then_fn
|
* @param {null | ((anchor: Node, value: V) => void)} then_fn
|
||||||
* @param {null | ((anchor: Node, error: unknown) => void)} catch_fn
|
* @param {null | ((anchor: Node, error: unknown) => void)} catch_fn
|
||||||
* @returns {void}
|
* @returns {void}
|
||||||
*/
|
*/
|
||||||
export function await_block(anchor_node, input, pending_fn, then_fn, catch_fn) {
|
export function await_block(anchor_node, get_input, pending_fn, then_fn, catch_fn) {
|
||||||
const block = create_await_block();
|
|
||||||
|
|
||||||
/** @type {null | import('../../types.js').Render} */
|
|
||||||
let current_render = null;
|
|
||||||
hydrate_block_anchor(anchor_node);
|
hydrate_block_anchor(anchor_node);
|
||||||
|
|
||||||
/** @type {{}} */
|
/** @type {any} */
|
||||||
let latest_token;
|
let input;
|
||||||
|
|
||||||
/** @type {typeof UNINITIALIZED | V} */
|
/** @type {import('../../types.js').EffectSignal | null} */
|
||||||
let resolved_value = UNINITIALIZED;
|
let pending_effect;
|
||||||
|
|
||||||
/** @type {unknown} */
|
/** @type {import('../../types.js').EffectSignal | null} */
|
||||||
let error = UNINITIALIZED;
|
let then_effect;
|
||||||
let pending = false;
|
|
||||||
block.r =
|
/** @type {import('../../types.js').EffectSignal | null} */
|
||||||
/**
|
let catch_effect;
|
||||||
* @param {import('../../types.js').Transition} transition
|
|
||||||
* @returns {void}
|
const branch = render_effect(() => {
|
||||||
*/
|
if (input === (input = get_input())) return;
|
||||||
(transition) => {
|
|
||||||
const render = /** @type {import('../../types.js').Render} */ (current_render);
|
if (is_promise(input)) {
|
||||||
const transitions = render.s;
|
const promise = /** @type {Promise<any>} */ (input);
|
||||||
transitions.add(transition);
|
|
||||||
transition.f(() => {
|
if (pending_effect) {
|
||||||
transitions.delete(transition);
|
resume_effect(pending_effect);
|
||||||
if (transitions.size === 0) {
|
} else if (pending_fn) {
|
||||||
// If the current render has changed since, then we can remove the old render
|
pending_effect = render_effect(() => pending_fn(anchor_node), {}, true);
|
||||||
// effect as it's stale.
|
|
||||||
if (current_render !== render && render.e !== null) {
|
|
||||||
if (render.d !== null) {
|
|
||||||
remove(render.d);
|
|
||||||
render.d = null;
|
|
||||||
}
|
|
||||||
destroy_signal(render.e);
|
|
||||||
render.e = null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (then_effect) {
|
||||||
|
pause_effect(then_effect, () => {
|
||||||
|
then_effect = null;
|
||||||
});
|
});
|
||||||
};
|
|
||||||
const create_render_effect = () => {
|
|
||||||
/** @type {import('../../types.js').Render} */
|
|
||||||
const render = {
|
|
||||||
d: null,
|
|
||||||
e: null,
|
|
||||||
s: new Set(),
|
|
||||||
p: current_render
|
|
||||||
};
|
|
||||||
const effect = render_effect(
|
|
||||||
() => {
|
|
||||||
if (error === UNINITIALIZED) {
|
|
||||||
if (resolved_value === UNINITIALIZED) {
|
|
||||||
// pending = true
|
|
||||||
block.n = true;
|
|
||||||
if (pending_fn !== null) {
|
|
||||||
pending_fn(anchor_node);
|
|
||||||
}
|
|
||||||
} else if (then_fn !== null) {
|
|
||||||
// pending = false
|
|
||||||
block.n = false;
|
|
||||||
then_fn(anchor_node, resolved_value);
|
|
||||||
}
|
|
||||||
} else if (catch_fn !== null) {
|
|
||||||
// pending = false
|
|
||||||
block.n = false;
|
|
||||||
catch_fn(anchor_node, error);
|
|
||||||
}
|
|
||||||
render.d = block.d;
|
|
||||||
block.d = null;
|
|
||||||
},
|
|
||||||
block,
|
|
||||||
true,
|
|
||||||
true
|
|
||||||
);
|
|
||||||
render.e = effect;
|
|
||||||
current_render = render;
|
|
||||||
};
|
|
||||||
const render = () => {
|
|
||||||
const render = current_render;
|
|
||||||
if (render === null) {
|
|
||||||
create_render_effect();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const transitions = render.s;
|
|
||||||
if (transitions.size === 0) {
|
|
||||||
if (render.d !== null) {
|
|
||||||
remove(render.d);
|
|
||||||
render.d = null;
|
|
||||||
}
|
|
||||||
if (render.e) {
|
|
||||||
execute_effect(render.e);
|
|
||||||
} else {
|
|
||||||
create_render_effect();
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
create_render_effect();
|
if (catch_effect) {
|
||||||
trigger_transitions(transitions, 'out');
|
pause_effect(catch_effect, () => {
|
||||||
|
catch_effect = null;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
};
|
|
||||||
const await_effect = render_effect(
|
promise
|
||||||
() => {
|
.then((value) => {
|
||||||
const token = {};
|
if (promise !== input) return;
|
||||||
latest_token = token;
|
|
||||||
const promise = input();
|
if (pending_effect) {
|
||||||
if (is_promise(promise)) {
|
pause_effect(pending_effect, () => {
|
||||||
promise.then(
|
pending_effect = null;
|
||||||
/** @param {V} v */
|
});
|
||||||
(v) => {
|
|
||||||
if (latest_token === token) {
|
|
||||||
// Ensure UI is in sync before resolving value.
|
|
||||||
flushSync();
|
|
||||||
resolved_value = v;
|
|
||||||
pending = false;
|
|
||||||
render();
|
|
||||||
}
|
}
|
||||||
},
|
|
||||||
/** @param {unknown} _error */
|
if (then_fn) {
|
||||||
(_error) => {
|
if (then_effect) {
|
||||||
error = _error;
|
resume_effect(then_effect);
|
||||||
pending = false;
|
} else if (pending_fn) {
|
||||||
render();
|
set_current_effect(branch);
|
||||||
|
then_effect = render_effect(() => then_fn(anchor_node, value), {}, true);
|
||||||
|
set_current_effect(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
if (promise !== input) return;
|
||||||
|
|
||||||
|
if (pending_effect) {
|
||||||
|
pause_effect(pending_effect, () => {
|
||||||
|
pending_effect = null;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
);
|
|
||||||
if (resolved_value !== UNINITIALIZED || error !== UNINITIALIZED) {
|
if (catch_fn) {
|
||||||
error = UNINITIALIZED;
|
if (catch_effect) {
|
||||||
resolved_value = UNINITIALIZED;
|
resume_effect(catch_effect);
|
||||||
|
} else if (pending_fn) {
|
||||||
|
set_current_effect(branch);
|
||||||
|
catch_effect = render_effect(() => catch_fn(anchor_node, error), {}, true);
|
||||||
|
set_current_effect(null);
|
||||||
}
|
}
|
||||||
if (!pending) {
|
|
||||||
pending = true;
|
|
||||||
render();
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
error = UNINITIALIZED;
|
// TODO handle non-promises
|
||||||
resolved_value = promise;
|
|
||||||
pending = false;
|
|
||||||
render();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
block,
|
|
||||||
false
|
|
||||||
);
|
|
||||||
push_destroy_fn(await_effect, () => {
|
|
||||||
let render = current_render;
|
|
||||||
latest_token = {};
|
|
||||||
while (render !== null) {
|
|
||||||
const dom = render.d;
|
|
||||||
if (dom !== null) {
|
|
||||||
remove(dom);
|
|
||||||
}
|
|
||||||
const effect = render.e;
|
|
||||||
if (effect !== null) {
|
|
||||||
destroy_signal(effect);
|
|
||||||
}
|
|
||||||
render = render.p;
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
block.e = await_effect;
|
|
||||||
|
branch.f |= BRANCH_EFFECT; // TODO create a primitive for this
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in new issue