Skip to content

Commit b043b28

Browse files
committed
feat: custom renderer api
1 parent be39867 commit b043b28

File tree

15 files changed

+269
-20
lines changed

15 files changed

+269
-20
lines changed

packages/svelte/src/compiler/index.js

+1
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ export { default as preprocess } from './preprocess/index.js';
2020
* @returns {CompileResult}
2121
*/
2222
export function compile(source, options) {
23+
options.customRenderer = true;
2324
source = remove_bom(source);
2425
state.reset_warning_filter(options.warningFilter);
2526
const validated = validate_component_options(options, '');

packages/svelte/src/compiler/phases/2-analyze/visitors/Attribute.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -67,7 +67,7 @@ export function Attribute(node, context) {
6767
const expression = get_attribute_expression(node);
6868
const delegated_event = get_delegated_event(node.name.slice(2), expression, context);
6969

70-
if (delegated_event !== null) {
70+
if (delegated_event !== null && !context.state.options.customRenderer) {
7171
if (delegated_event.hoisted) {
7272
delegated_event.function.metadata.hoisted = true;
7373
}

packages/svelte/src/compiler/phases/3-transform/client/transform-client.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -167,7 +167,7 @@ export function client_component(analysis, options) {
167167
in_constructor: false,
168168
instance_level_snippets: [],
169169
module_level_snippets: [],
170-
is_functional_template_mode: options.templatingMode === 'functional',
170+
is_functional_template_mode: options.customRenderer || options.templatingMode === 'functional',
171171

172172
// these are set inside the `Fragment` visitor, and cannot be used until then
173173
init: /** @type {any} */ (null),

packages/svelte/src/compiler/phases/3-transform/client/visitors/shared/events.js

+5-2
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,8 @@ export function visit_event_attribute(node, context) {
6868
context.state.node,
6969
handler,
7070
capture,
71-
is_passive_event(event_name) ? true : undefined
71+
is_passive_event(event_name) ? true : undefined,
72+
context.state.options.customRenderer
7273
)
7374
);
7475

@@ -90,13 +91,15 @@ export function visit_event_attribute(node, context) {
9091
* @param {Expression} handler
9192
* @param {boolean} capture
9293
* @param {boolean | undefined} passive
94+
* @param {boolean | undefined} custom_renderer
9395
*/
94-
export function build_event(event_name, node, handler, capture, passive) {
96+
export function build_event(event_name, node, handler, capture, passive, custom_renderer) {
9597
return b.call(
9698
'$.event',
9799
b.literal(event_name),
98100
node,
99101
handler,
102+
custom_renderer && b.true,
100103
capture && b.true,
101104
passive === undefined ? undefined : b.literal(passive)
102105
);

packages/svelte/src/compiler/types/index.d.ts

+4
Original file line numberDiff line numberDiff line change
@@ -119,6 +119,10 @@ export interface CompileOptions extends ModuleCompileOptions {
119119
* @default 'string'
120120
*/
121121
templatingMode?: 'string' | 'functional';
122+
/**
123+
* If `true` the output will be adapted to accept a custom renderer.
124+
*/
125+
customRenderer?: boolean;
122126
/**
123127
* Set to `true` to force the compiler into runes mode, even if there are no indications of runes usage.
124128
* Set to `false` to force the compiler into ignoring runes, even if there are indications of runes usage.

packages/svelte/src/compiler/validate-options.js

+2
Original file line numberDiff line numberDiff line change
@@ -112,6 +112,8 @@ export const validate_component_options =
112112

113113
templatingMode: list(['string', 'functional']),
114114

115+
customRenderer: boolean(false),
116+
115117
preserveWhitespace: boolean(false),
116118

117119
runes: boolean(undefined),

packages/svelte/src/index.d.ts

+4
Original file line numberDiff line numberDiff line change
@@ -338,6 +338,10 @@ export type MountOptions<Props extends Record<string, any> = Record<string, any>
338338
* @default true
339339
*/
340340
intro?: boolean;
341+
/**
342+
* The custom renderer to use to mount the component.
343+
*/
344+
customRenderer?: any;
341345
} & ({} extends Props
342346
? {
343347
/**

packages/svelte/src/internal/client/dom/elements/events.js

+13-3
Original file line numberDiff line numberDiff line change
@@ -51,12 +51,21 @@ export function replay_events(dom) {
5151
* @param {EventTarget} dom
5252
* @param {EventListener} [handler]
5353
* @param {AddEventListenerOptions} [options]
54+
* @param {boolean} [custom_renderer]
5455
*/
55-
export function create_event(event_name, dom, handler, options = {}) {
56+
export function create_event(event_name, dom, handler, options = {}, custom_renderer = false) {
5657
/**
5758
* @this {EventTarget}
5859
*/
5960
function target_handler(/** @type {Event} */ event) {
61+
// if we have a custom renderer we just want to call the function
62+
// without a reactive context because we don't know if event propagation
63+
// is even a thing in the target renderer
64+
if (custom_renderer) {
65+
return without_reactive_context(() => {
66+
return handler?.call(this, event);
67+
});
68+
}
6069
if (!options.capture) {
6170
// Only call in the bubble phase, else delegated events would be called before the capturing events
6271
handle_event_propagation.call(dom, event);
@@ -109,13 +118,14 @@ export function on(element, type, handler, options = {}) {
109118
* @param {string} event_name
110119
* @param {Element} dom
111120
* @param {EventListener} [handler]
121+
* @param {boolean} [custom_renderer]
112122
* @param {boolean} [capture]
113123
* @param {boolean} [passive]
114124
* @returns {void}
115125
*/
116-
export function event(event_name, dom, handler, capture, passive) {
126+
export function event(event_name, dom, handler, custom_renderer, capture, passive) {
117127
var options = { capture, passive };
118-
var target_handler = create_event(event_name, dom, handler, options);
128+
var target_handler = create_event(event_name, dom, handler, options, custom_renderer);
119129

120130
// @ts-ignore
121131
if (dom === document.body || dom === window || dom === document) {

packages/svelte/src/internal/client/dom/operations.js

+53-9
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,8 @@ var first_child_getter;
1919
/** @type {() => Node | null} */
2020
var next_sibling_getter;
2121

22+
var renderer;
23+
2224
/**
2325
* Initialize these lazily to avoid issues when using the runtime in a server context
2426
* where these globals are not available while avoiding a separate server entry point
@@ -28,20 +30,23 @@ export function init_operations() {
2830
return;
2931
}
3032

31-
$window = window;
32-
$document = document;
33-
is_firefox = /Firefox/.test(navigator.userAgent);
33+
$window = typeof window === 'undefined' ? /** @type {Window} */ ({}) : window;
34+
$document = typeof document === 'undefined' ? /** @type {Document} */ ({}) : document;
35+
is_firefox = typeof navigator === 'undefined' ? true : /Firefox/.test(navigator.userAgent);
3436

35-
var element_prototype = Element.prototype;
36-
var node_prototype = Node.prototype;
37-
var text_prototype = Text.prototype;
37+
var element_prototype;
38+
if (window.Element) element_prototype = Element.prototype;
39+
var node_prototype;
40+
if (window.Node) node_prototype = Node.prototype;
41+
var text_prototype;
42+
if (window.Text) text_prototype = Text.prototype;
3843

3944
// @ts-ignore
40-
first_child_getter = get_descriptor(node_prototype, 'firstChild').get;
45+
first_child_getter = get_descriptor(node_prototype, 'firstChild')?.get;
4146
// @ts-ignore
42-
next_sibling_getter = get_descriptor(node_prototype, 'nextSibling').get;
47+
next_sibling_getter = get_descriptor(node_prototype, 'nextSibling')?.get;
4348

44-
if (is_extensible(element_prototype)) {
49+
if (element_prototype && is_extensible(element_prototype)) {
4550
// the following assignments improve perf of lookups on DOM nodes
4651
// @ts-expect-error
4752
element_prototype.__click = undefined;
@@ -73,6 +78,9 @@ export function init_operations() {
7378
* @returns {Text}
7479
*/
7580
export function create_text(value = '') {
81+
if (renderer) {
82+
return renderer.document.createTextNode(value);
83+
}
7684
return document.createTextNode(value);
7785
}
7886

@@ -83,6 +91,9 @@ export function create_text(value = '') {
8391
*/
8492
/*@__NO_SIDE_EFFECTS__*/
8593
export function get_first_child(node) {
94+
if (renderer_first_child) {
95+
return renderer_first_child.call(node);
96+
}
8697
return first_child_getter.call(node);
8798
}
8899

@@ -93,6 +104,9 @@ export function get_first_child(node) {
93104
*/
94105
/*@__NO_SIDE_EFFECTS__*/
95106
export function get_next_sibling(node) {
107+
if (renderer_next_sibling) {
108+
return renderer_next_sibling.call(node);
109+
}
96110
return next_sibling_getter.call(node);
97111
}
98112

@@ -213,6 +227,9 @@ export function clear_text_content(node) {
213227
* @returns
214228
*/
215229
export function create_element(tag, namespace, is) {
230+
if (renderer) {
231+
return renderer.document.createElement(tag);
232+
}
216233
let options = is ? { is } : undefined;
217234
if (namespace) {
218235
return document.createElementNS(namespace, tag, options);
@@ -221,6 +238,9 @@ export function create_element(tag, namespace, is) {
221238
}
222239

223240
export function create_fragment() {
241+
if (renderer) {
242+
return renderer.document.createDocumentFragment();
243+
}
224244
return document.createDocumentFragment();
225245
}
226246

@@ -229,6 +249,9 @@ export function create_fragment() {
229249
* @returns
230250
*/
231251
export function create_comment(data = '') {
252+
if (renderer) {
253+
return renderer.document.createComment(data);
254+
}
232255
return document.createComment(data);
233256
}
234257

@@ -245,3 +268,24 @@ export function set_attribute(element, key, value = '') {
245268
}
246269
return element.setAttribute(key, value);
247270
}
271+
272+
var renderer_next_sibling;
273+
var renderer_first_child;
274+
275+
export function push_renderer(custom_renderer) {
276+
var old_next_sibling = renderer_next_sibling;
277+
var old_first_child = renderer_first_child;
278+
var old_renderer = renderer;
279+
renderer_next_sibling = get_descriptor(custom_renderer.Node.prototype, 'nextSibling')?.get;
280+
renderer_first_child = get_descriptor(custom_renderer.Node.prototype, 'firstChild')?.get;
281+
renderer = custom_renderer;
282+
return () => {
283+
renderer_next_sibling = old_next_sibling;
284+
renderer_first_child = old_first_child;
285+
renderer = old_renderer;
286+
};
287+
}
288+
289+
export function get_renderer() {
290+
return renderer;
291+
}

packages/svelte/src/internal/client/reactivity/effects.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ import { set } from './sources.js';
3838
import * as e from '../errors.js';
3939
import { DEV } from 'esm-env';
4040
import { define_property } from '../../shared/utils.js';
41-
import { get_next_sibling } from '../dom/operations.js';
41+
import { get_next_sibling, get_renderer } from '../dom/operations.js';
4242
import { derived } from './deriveds.js';
4343
import { component_context, dev_current_component_function } from '../context.js';
4444

@@ -99,6 +99,8 @@ function create_effect(type, fn, sync, push = true) {
9999
nodes_end: null,
100100
f: type | DIRTY,
101101
first: null,
102+
// we only need to update the renderer for render effects
103+
renderer: (type & RENDER_EFFECT) !== 0 ? get_renderer() : undefined,
102104
fn,
103105
last: null,
104106
next: null,

packages/svelte/src/internal/client/reactivity/types.d.ts

+2
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,8 @@ export interface Effect extends Reaction {
6767
last: null | Effect;
6868
/** Parent effect */
6969
parent: Effect | null;
70+
/** The original renderer used when mounting the root component */
71+
renderer?: any;
7072
/** Dev only */
7173
component_function?: any;
7274
}

packages/svelte/src/internal/client/render.js

+18-2
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,8 @@ import {
66
create_text,
77
get_first_child,
88
get_next_sibling,
9-
init_operations
9+
init_operations,
10+
push_renderer
1011
} from './dom/operations.js';
1112
import { HYDRATION_END, HYDRATION_ERROR, HYDRATION_START } from '../../constants.js';
1213
import { active_effect } from './runtime.js';
@@ -86,17 +87,23 @@ export function mount(component, options) {
8687
* context?: Map<any, any>;
8788
* intro?: boolean;
8889
* recover?: boolean;
90+
* customRenderer?: any;
8991
* } : {
9092
* target: Document | Element | ShadowRoot;
9193
* props: Props;
9294
* events?: Record<string, (e: any) => any>;
9395
* context?: Map<any, any>;
9496
* intro?: boolean;
9597
* recover?: boolean;
98+
* customRenderer?: any;
9699
* }} options
97100
* @returns {Exports}
98101
*/
99102
export function hydrate(component, options) {
103+
let cleanup_renderer = undefined;
104+
if (options.customRenderer) {
105+
cleanup_renderer = push_renderer(options.customRenderer);
106+
}
100107
init_operations();
101108
options.intro = options.intro ?? false;
102109
const target = options.target;
@@ -153,6 +160,7 @@ export function hydrate(component, options) {
153160
set_hydrating(was_hydrating);
154161
set_hydrate_node(previous_hydrate_node);
155162
reset_head_anchor();
163+
cleanup_renderer?.();
156164
}
157165
}
158166

@@ -165,7 +173,14 @@ const document_listeners = new Map();
165173
* @param {MountOptions} options
166174
* @returns {Exports}
167175
*/
168-
function _mount(Component, { target, anchor, props = {}, events, context, intro = true }) {
176+
function _mount(
177+
Component,
178+
{ target, anchor, props = {}, events, context, intro = true, customRenderer }
179+
) {
180+
let cleanup_renderer = undefined;
181+
if (customRenderer) {
182+
cleanup_renderer = push_renderer(customRenderer);
183+
}
169184
init_operations();
170185

171186
var registered_events = new Set();
@@ -261,6 +276,7 @@ function _mount(Component, { target, anchor, props = {}, events, context, intro
261276
});
262277

263278
mounted_components.set(component, unmount);
279+
cleanup_renderer?.();
264280
return component;
265281
}
266282

packages/svelte/src/internal/client/runtime.js

+9-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ import {
3939
set_component_context,
4040
set_dev_current_component_function
4141
} from './context.js';
42-
import { is_firefox } from './dom/operations.js';
42+
import { is_firefox, push_renderer } from './dom/operations.js';
4343

4444
// Used for DEV time error handling
4545
/** @param {WeakSet<Error>} value */
@@ -421,6 +421,13 @@ export function update_reaction(reaction) {
421421

422422
reaction_sources = null;
423423
set_component_context(reaction.ctx);
424+
var cleanup_renderer = undefined;
425+
if ((reaction.f & DERIVED) === 0) {
426+
var effect = /** @type {Effect} */ (reaction);
427+
if (effect.renderer) {
428+
cleanup_renderer = push_renderer(effect.renderer);
429+
}
430+
}
424431
untracking = false;
425432
read_version++;
426433

@@ -498,6 +505,7 @@ export function update_reaction(reaction) {
498505
reaction_sources = previous_reaction_sources;
499506
set_component_context(previous_component_context);
500507
untracking = previous_untracking;
508+
cleanup_renderer?.();
501509

502510
reaction.f ^= EFFECT_IS_UPDATING;
503511
}

0 commit comments

Comments
 (0)