2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 12:08:49 -07:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2016-06-23 09:47:54 -07:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2018-07-30 22:34:15 -07:00
|
|
|
import {InjectionToken} from '../di/injection_token';
|
2016-06-08 16:38:52 -07:00
|
|
|
import {ViewEncapsulation} from '../metadata/view';
|
refactor(ivy): obviate the Bazel component of the ivy_switch (#26550)
Originally, the ivy_switch mechanism used Bazel genrules to conditionally
compile one TS file or another depending on whether ngc or ngtsc was the
selected compiler. This was done because we wanted to avoid importing
certain modules (and thus pulling them into the build) if Ivy was on or
off. This mechanism had a major drawback: ivy_switch became a bottleneck
in the import graph, as it both imports from many places in the codebase
and is imported by many modules in the codebase. This frequently resulted
in cyclic imports which caused issues both with TS and Closure compilation.
It turns out ngcc needs both code paths in the bundle to perform the switch
during its operation anyway, so import switching was later abandoned. This
means that there's no real reason why the ivy_switch mechanism needed to
operate at the Bazel level, and for the ivy_switch file to be a bottleneck.
This commit removes the Bazel-level ivy_switch mechanism, and introduces
an additional TypeScript transform in ngtsc (and the pass-through tsc
compiler used for testing JIT) to perform the same operation that ngcc
does, and flip the switch during ngtsc compilation. This allows the
ivy_switch file to be removed, and the individual switches to be located
directly next to their consumers in the codebase, greatly mitigating the
circular import issues and making the mechanism much easier to use.
As part of this commit, the tag for marking switched variables was changed
from __PRE_NGCC__ to __PRE_R3__, since it's no longer just ngcc which
flips these tags. Most variables were renamed from R3_* to SWITCH_* as well,
since they're referenced mostly in render2 code.
Test strategy: existing test coverage is more than sufficient - if this
didn't work correctly it would break the hello world and todo apps.
PR Close #26550
2018-10-17 15:44:44 -07:00
|
|
|
import {injectRenderer2 as render3InjectRenderer2} from '../render3/view_engine_compatibility';
|
|
|
|
import {noop} from '../util/noop';
|
2016-06-08 16:38:52 -07:00
|
|
|
|
2018-10-10 15:53:14 +02:00
|
|
|
|
2017-03-07 16:36:12 -08:00
|
|
|
export const Renderer2Interceptor = new InjectionToken<Renderer2[]>('Renderer2Interceptor');
|
2017-02-22 15:14:49 -08:00
|
|
|
|
2017-02-16 13:55:55 -08:00
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* Used by `RendererFactory2` to associate custom rendering data and styles
|
|
|
|
* with a rendering implementation.
|
2018-10-19 12:12:20 +01:00
|
|
|
* @publicApi
|
2017-02-16 13:55:55 -08:00
|
|
|
*/
|
2017-03-07 16:36:12 -08:00
|
|
|
export interface RendererType2 {
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* A unique identifying string for the new renderer, used when creating
|
|
|
|
* unique styles for encapsulation.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
id: string;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* The view encapsulation type, which determines how styles are applied to
|
|
|
|
* DOM elements. One of
|
|
|
|
* - `Emulated` (default): Emulate native scoping of styles.
|
|
|
|
* - `Native`: Use the native encapsulation mechanism of the renderer.
|
2018-07-18 10:26:59 -07:00
|
|
|
* - `ShadowDom`: Use modern [Shadow
|
2018-07-02 12:06:52 -07:00
|
|
|
* DOM](https://w3c.github.io/webcomponents/spec/shadow/) and
|
|
|
|
* create a ShadowRoot for component's host element.
|
|
|
|
* - `None`: Do not provide any template or style encapsulation.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
encapsulation: ViewEncapsulation;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Defines CSS styles to be stored on a renderer instance.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
styles: (string|any[])[];
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Defines arbitrary developer-defined data to be stored on a renderer instance.
|
|
|
|
* This is useful for renderers that delegate to other renderers.
|
|
|
|
*/
|
2017-02-24 12:10:19 -08:00
|
|
|
data: {[kind: string]: any};
|
2017-02-16 13:55:55 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* Creates and initializes a custom renderer that implements the `Renderer2` base class.
|
|
|
|
*
|
2018-10-19 12:12:20 +01:00
|
|
|
* @publicApi
|
2017-02-16 13:55:55 -08:00
|
|
|
*/
|
2017-03-07 16:36:12 -08:00
|
|
|
export abstract class RendererFactory2 {
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
2018-07-18 10:26:59 -07:00
|
|
|
* Creates and initializes a custom renderer for a host DOM element.
|
2018-07-02 12:06:52 -07:00
|
|
|
* @param hostElement The element to render.
|
|
|
|
* @param type The base class to implement.
|
|
|
|
* @returns The new custom renderer instance.
|
|
|
|
*/
|
2017-03-29 09:34:45 -07:00
|
|
|
abstract createRenderer(hostElement: any, type: RendererType2|null): Renderer2;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* A callback invoked when rendering has begun.
|
|
|
|
*/
|
2017-05-03 13:17:46 -07:00
|
|
|
abstract begin?(): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* A callback invoked when rendering has completed.
|
|
|
|
*/
|
2017-05-03 13:17:46 -07:00
|
|
|
abstract end?(): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Use with animations test-only mode. Notifies the test when rendering has completed.
|
|
|
|
* @returns The asynchronous result of the developer-defined function.
|
|
|
|
*/
|
2017-05-12 17:32:51 -04:00
|
|
|
abstract whenRenderingDone?(): Promise<any>;
|
2017-02-16 13:55:55 -08:00
|
|
|
}
|
|
|
|
|
2017-03-13 09:45:04 -07:00
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* Flags for renderer-specific style modifiers.
|
2018-10-19 12:12:20 +01:00
|
|
|
* @publicApi
|
2017-03-13 09:45:04 -07:00
|
|
|
*/
|
|
|
|
export enum RendererStyleFlags2 {
|
2020-01-15 16:52:54 -08:00
|
|
|
// TODO(misko): This needs to be refactored into a separate file so that it can be imported from
|
|
|
|
// `node_manipulation.ts` Currently doing the import cause resolution order to change and fails
|
|
|
|
// the tests. The work around is to have hard coded value in `node_manipulation.ts` for now.
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Marks a style as important.
|
|
|
|
*/
|
2017-03-13 09:45:04 -07:00
|
|
|
Important = 1 << 0,
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Marks a style as using dash case naming (this-is-dash-case).
|
|
|
|
*/
|
2017-03-13 09:45:04 -07:00
|
|
|
DashCase = 1 << 1
|
|
|
|
}
|
|
|
|
|
2017-02-14 21:03:18 -08:00
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* Extend this base class to implement custom rendering. By default, Angular
|
|
|
|
* renders a template into DOM. You can use custom rendering to intercept
|
|
|
|
* rendering calls, or to render to something other than DOM.
|
|
|
|
*
|
|
|
|
* Create your custom renderer using `RendererFactory2`.
|
|
|
|
*
|
2018-07-18 10:26:59 -07:00
|
|
|
* Use a custom renderer to bypass Angular's templating and
|
|
|
|
* make custom UI changes that can't be expressed declaratively.
|
2018-07-02 12:06:52 -07:00
|
|
|
* For example if you need to set a property or an attribute whose name is
|
2018-07-18 10:26:59 -07:00
|
|
|
* not statically known, use the `setProperty()` or
|
|
|
|
* `setAttribute()` method.
|
2018-07-02 12:06:52 -07:00
|
|
|
*
|
2018-10-19 12:12:20 +01:00
|
|
|
* @publicApi
|
2017-02-14 21:03:18 -08:00
|
|
|
*/
|
2017-03-07 16:36:12 -08:00
|
|
|
export abstract class Renderer2 {
|
2017-02-24 12:10:19 -08:00
|
|
|
/**
|
2018-07-18 10:26:59 -07:00
|
|
|
* Use to store arbitrary developer-defined data on a renderer instance,
|
2018-07-02 12:06:52 -07:00
|
|
|
* as an object containing key-value pairs.
|
2017-02-24 12:10:19 -08:00
|
|
|
* This is useful for renderers that delegate to other renderers.
|
|
|
|
*/
|
2017-09-28 13:36:56 -07:00
|
|
|
abstract get data(): {[key: string]: any};
|
2017-02-24 12:10:19 -08:00
|
|
|
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to destroy the renderer or the host element.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
abstract destroy(): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to create an instance of the host element.
|
|
|
|
* @param name An identifying name for the new element, unique within the namespace.
|
|
|
|
* @param namespace The namespace for the new element.
|
|
|
|
* @returns The new element.
|
|
|
|
*/
|
2017-03-29 09:34:45 -07:00
|
|
|
abstract createElement(name: string, namespace?: string|null): any;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to add a comment to the DOM of the host element.
|
|
|
|
* @param value The comment text.
|
|
|
|
* @returns The modified element.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
abstract createComment(value: string): any;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to add text to the DOM of the host element.
|
|
|
|
* @param value The text string.
|
|
|
|
* @returns The modified element.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
abstract createText(value: string): any;
|
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* If null or undefined, the view engine won't call it.
|
2017-02-16 13:55:55 -08:00
|
|
|
* This is used as a performance optimization for production mode.
|
|
|
|
*/
|
2018-06-18 16:38:33 -07:00
|
|
|
// TODO(issue/24571): remove '!'.
|
2020-04-13 16:40:21 -07:00
|
|
|
destroyNode!: ((node: any) => void)|null;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Appends a child to a given parent node in the host element DOM.
|
|
|
|
* @param parent The parent node.
|
|
|
|
* @param newChild The new child node.
|
|
|
|
*/
|
2017-02-14 21:03:18 -08:00
|
|
|
abstract appendChild(parent: any, newChild: any): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to insert a child node at a given position in a parent node
|
|
|
|
* in the host element DOM.
|
|
|
|
* @param parent The parent node.
|
|
|
|
* @param newChild The new child nodes.
|
2019-12-23 12:03:52 -05:00
|
|
|
* @param refChild The existing child node before which `newChild` is inserted.
|
2020-09-25 15:01:56 -07:00
|
|
|
* @param isMove Optional argument which signifies if the current `insertBefore` is a result of a
|
|
|
|
* move. Animation uses this information to trigger move animations. In the past the Animation
|
|
|
|
* would always assume that any `insertBefore` is a move. This is not strictly true because
|
|
|
|
* with runtime i18n it is possible to invoke `insertBefore` as a result of i18n and it should
|
|
|
|
* not trigger an animation move.
|
2018-07-02 12:06:52 -07:00
|
|
|
*/
|
2020-09-25 15:01:56 -07:00
|
|
|
abstract insertBefore(parent: any, newChild: any, refChild: any, isMove?: boolean): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to remove a child node from the host element's DOM.
|
|
|
|
* @param parent The parent node.
|
|
|
|
* @param oldChild The child node to remove.
|
2019-01-14 15:36:08 -08:00
|
|
|
* @param isHostElement Optionally signal to the renderer whether this element is a host element
|
|
|
|
* or not
|
2018-07-02 12:06:52 -07:00
|
|
|
*/
|
2019-01-14 15:36:08 -08:00
|
|
|
abstract removeChild(parent: any, oldChild: any, isHostElement?: boolean): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
2018-07-18 10:26:59 -07:00
|
|
|
* Implement this callback to prepare an element to be bootstrapped
|
|
|
|
* as a root element, and return the element instance.
|
2018-07-02 12:06:52 -07:00
|
|
|
* @param selectorOrNode The DOM element.
|
2018-07-15 18:53:18 -07:00
|
|
|
* @param preserveContent Whether the contents of the root element
|
|
|
|
* should be preserved, or cleared upon bootstrap (default behavior).
|
|
|
|
* Use with `ViewEncapsulation.ShadowDom` to allow simple native
|
|
|
|
* content projection via `<slot>` elements.
|
2018-07-02 12:06:52 -07:00
|
|
|
* @returns The root element.
|
|
|
|
*/
|
2018-07-11 15:57:27 -07:00
|
|
|
abstract selectRootElement(selectorOrNode: string|any, preserveContent?: boolean): any;
|
2017-02-14 21:03:18 -08:00
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* Implement this callback to get the parent of a given node
|
|
|
|
* in the host element's DOM.
|
|
|
|
* @param node The child node to query.
|
|
|
|
* @returns The parent node, or null if there is no parent.
|
|
|
|
* For WebWorkers, always returns true.
|
|
|
|
* This is because the check is synchronous,
|
|
|
|
* and the caller can't rely on checking for null.
|
2017-02-14 21:03:18 -08:00
|
|
|
*/
|
|
|
|
abstract parentNode(node: any): any;
|
|
|
|
/**
|
2018-07-02 12:06:52 -07:00
|
|
|
* Implement this callback to get the next sibling node of a given node
|
|
|
|
* in the host element's DOM.
|
|
|
|
* @returns The sibling node, or null if there is no sibling.
|
|
|
|
* For WebWorkers, always returns a value.
|
|
|
|
* This is because the check is synchronous,
|
|
|
|
* and the caller can't rely on checking for null.
|
2017-02-14 21:03:18 -08:00
|
|
|
*/
|
|
|
|
abstract nextSibling(node: any): any;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to set an attribute value for an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param name The attribute name.
|
|
|
|
* @param value The new value.
|
|
|
|
* @param namespace The namespace.
|
|
|
|
*/
|
2017-03-29 09:34:45 -07:00
|
|
|
abstract setAttribute(el: any, name: string, value: string, namespace?: string|null): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to remove an attribute from an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param name The attribute name.
|
|
|
|
* @param namespace The namespace.
|
|
|
|
*/
|
2017-03-29 09:34:45 -07:00
|
|
|
abstract removeAttribute(el: any, name: string, namespace?: string|null): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
/**
|
|
|
|
* Implement this callback to add a class to an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param name The class name.
|
|
|
|
*/
|
2017-02-14 21:03:18 -08:00
|
|
|
abstract addClass(el: any, name: string): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to remove a class from an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param name The class name.
|
|
|
|
*/
|
2017-02-14 21:03:18 -08:00
|
|
|
abstract removeClass(el: any, name: string): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to set a CSS style for an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param style The name of the style.
|
|
|
|
* @param value The new value.
|
|
|
|
* @param flags Flags for style variations. No flags are set by default.
|
|
|
|
*/
|
2017-03-13 09:45:04 -07:00
|
|
|
abstract setStyle(el: any, style: string, value: any, flags?: RendererStyleFlags2): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to remove the value from a CSS style for an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param style The name of the style.
|
|
|
|
* @param flags Flags for style variations to remove, if set. ???
|
|
|
|
*/
|
2017-03-13 09:45:04 -07:00
|
|
|
abstract removeStyle(el: any, style: string, flags?: RendererStyleFlags2): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to set the value of a property of an element in the DOM.
|
|
|
|
* @param el The element.
|
|
|
|
* @param name The property name.
|
|
|
|
* @param value The new value.
|
|
|
|
*/
|
2017-02-14 21:03:18 -08:00
|
|
|
abstract setProperty(el: any, name: string, value: any): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to set the value of a node in the host element.
|
|
|
|
* @param node The node.
|
|
|
|
* @param value The new value.
|
|
|
|
*/
|
2017-02-16 13:55:55 -08:00
|
|
|
abstract setValue(node: any, value: string): void;
|
2018-07-02 12:06:52 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implement this callback to start an event listener.
|
|
|
|
* @param target The context in which to listen for events. Can be
|
|
|
|
* the entire window or document, the body of the document, or a specific
|
|
|
|
* DOM element.
|
|
|
|
* @param eventName The event to listen for.
|
|
|
|
* @param callback A handler function to invoke when the event occurs.
|
2018-07-18 10:26:59 -07:00
|
|
|
* @returns An "unlisten" function for disposing of this handler.
|
2018-07-02 12:06:52 -07:00
|
|
|
*/
|
2017-02-14 21:03:18 -08:00
|
|
|
abstract listen(
|
|
|
|
target: 'window'|'document'|'body'|any, eventName: string,
|
2017-03-07 16:36:12 -08:00
|
|
|
callback: (event: any) => boolean | void): () => void;
|
2018-10-10 15:53:14 +02:00
|
|
|
|
2019-01-14 12:44:06 -08:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
* @nocollapse
|
|
|
|
*/
|
refactor(ivy): obviate the Bazel component of the ivy_switch (#26550)
Originally, the ivy_switch mechanism used Bazel genrules to conditionally
compile one TS file or another depending on whether ngc or ngtsc was the
selected compiler. This was done because we wanted to avoid importing
certain modules (and thus pulling them into the build) if Ivy was on or
off. This mechanism had a major drawback: ivy_switch became a bottleneck
in the import graph, as it both imports from many places in the codebase
and is imported by many modules in the codebase. This frequently resulted
in cyclic imports which caused issues both with TS and Closure compilation.
It turns out ngcc needs both code paths in the bundle to perform the switch
during its operation anyway, so import switching was later abandoned. This
means that there's no real reason why the ivy_switch mechanism needed to
operate at the Bazel level, and for the ivy_switch file to be a bottleneck.
This commit removes the Bazel-level ivy_switch mechanism, and introduces
an additional TypeScript transform in ngtsc (and the pass-through tsc
compiler used for testing JIT) to perform the same operation that ngcc
does, and flip the switch during ngtsc compilation. This allows the
ivy_switch file to be removed, and the individual switches to be located
directly next to their consumers in the codebase, greatly mitigating the
circular import issues and making the mechanism much easier to use.
As part of this commit, the tag for marking switched variables was changed
from __PRE_NGCC__ to __PRE_R3__, since it's no longer just ngcc which
flips these tags. Most variables were renamed from R3_* to SWITCH_* as well,
since they're referenced mostly in render2 code.
Test strategy: existing test coverage is more than sufficient - if this
didn't work correctly it would break the hello world and todo apps.
PR Close #26550
2018-10-17 15:44:44 -07:00
|
|
|
static __NG_ELEMENT_ID__: () => Renderer2 = () => SWITCH_RENDERER2_FACTORY();
|
2017-02-14 21:03:18 -08:00
|
|
|
}
|
refactor(ivy): obviate the Bazel component of the ivy_switch (#26550)
Originally, the ivy_switch mechanism used Bazel genrules to conditionally
compile one TS file or another depending on whether ngc or ngtsc was the
selected compiler. This was done because we wanted to avoid importing
certain modules (and thus pulling them into the build) if Ivy was on or
off. This mechanism had a major drawback: ivy_switch became a bottleneck
in the import graph, as it both imports from many places in the codebase
and is imported by many modules in the codebase. This frequently resulted
in cyclic imports which caused issues both with TS and Closure compilation.
It turns out ngcc needs both code paths in the bundle to perform the switch
during its operation anyway, so import switching was later abandoned. This
means that there's no real reason why the ivy_switch mechanism needed to
operate at the Bazel level, and for the ivy_switch file to be a bottleneck.
This commit removes the Bazel-level ivy_switch mechanism, and introduces
an additional TypeScript transform in ngtsc (and the pass-through tsc
compiler used for testing JIT) to perform the same operation that ngcc
does, and flip the switch during ngtsc compilation. This allows the
ivy_switch file to be removed, and the individual switches to be located
directly next to their consumers in the codebase, greatly mitigating the
circular import issues and making the mechanism much easier to use.
As part of this commit, the tag for marking switched variables was changed
from __PRE_NGCC__ to __PRE_R3__, since it's no longer just ngcc which
flips these tags. Most variables were renamed from R3_* to SWITCH_* as well,
since they're referenced mostly in render2 code.
Test strategy: existing test coverage is more than sufficient - if this
didn't work correctly it would break the hello world and todo apps.
PR Close #26550
2018-10-17 15:44:44 -07:00
|
|
|
|
|
|
|
|
|
|
|
export const SWITCH_RENDERER2_FACTORY__POST_R3__ = render3InjectRenderer2;
|
|
|
|
const SWITCH_RENDERER2_FACTORY__PRE_R3__ = noop;
|
|
|
|
const SWITCH_RENDERER2_FACTORY: typeof render3InjectRenderer2 = SWITCH_RENDERER2_FACTORY__PRE_R3__;
|