diff --git a/latest b/latest new file mode 120000 index 0000000000..817ba4bfff --- /dev/null +++ b/latest @@ -0,0 +1 @@ +v13 \ No newline at end of file diff --git a/v13/01-canvas-components/index.html b/v13/01-canvas-components/index.html new file mode 100644 index 0000000000..61cebfafaa --- /dev/null +++ b/v13/01-canvas-components/index.html @@ -0,0 +1,2747 @@ + + + + + + + + + + + + + + + + + + + + + + + Components Overview - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Elyra Canvas Components Overview

+

The Elyra Canvas package delivers two decoupled React objects: Common Canvas and Common Properties.

+

+

Common Canvas

+

Common Canvas displays a flow of data operations as nodes and links (edges) which the user can create and edit to get the flow they want. These visual flows of data operations are typically translated into data processing steps performed by the application. Common Canvas provides UI functionality for the visual display of flows in the browser and leaves persistence and execution of data flows to the application.

+

The common-canvas user can perform operations such as:

+
    +
  • Create a new node by dragging a node template from a palette onto the canvas.
  • +
  • Delete a node by clicking a context menu option.
  • +
  • Create a link by dragging a line from one node to another.
  • +
  • Delete a link by clicking a context menu option.
  • +
  • Add a comment to the canvas and draw a link from it to one or more nodes.
  • +
  • Edit a comment.
  • +
  • Move nodes and comments around in the canvas to get the desired arrangement.
  • +
  • Create a new node by dragging a node from the OS desktop (or elsewhere on the browser page) onto the canvas. This takes a little bit of development work.
  • +
  • And much more! …
  • +
+

Common Canvas Components

+

Common Canvas has several constituent parts that can be visible to the user and can be customized by the application:

+

+
    +
  • Flow editor - the main area of the UI where the flow is displayed and edited.
  • +
  • Palette - a set of node templates that can be dragged to the canvas to create new nodes
  • +
  • Context menu - a menu of options for nodes, comments, etc.
  • +
  • Context toolbar - a menu of options for nodes, comments, etc presented as a small toolbar
  • +
  • Toolbar - a set of tools across the top of the UI.
  • +
  • Notification panel - a panel for displaying runtime and other messages to the user.
  • +
  • State Tag - a small pill-shaped component that appears over the canvas to indicate its state: locked or +read-only.
  • +
  • Tooltips - information tips displayed when the mouse cursor is over an object.
  • +
+

In addition, there are three optional panels where application specific output can be displayed such as +properties, log info or data previews. +

+
    +
  • Right side flyout - a panel, often used to display node properties
  • +
  • Top panel - a panel which can be used to display other app related information
  • +
  • Bottom panel - a panel which can be used to display other app related information
  • +
+

Common Properties

+

Common Properties allows the application to display a Carbon compliant properties panel or dialog with just a Javascript (JSON) object as input. Common Properties supports the most commonly used UI components and also allows custom components to be added into its visual output.

+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.01-flow-editor/index.html b/v13/01.01-flow-editor/index.html new file mode 100644 index 0000000000..2703221bac --- /dev/null +++ b/v13/01.01-flow-editor/index.html @@ -0,0 +1,2862 @@ + + + + + + + + + + + + + + + + + + + + + + + Editor Overview - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Flow Editor Overview

+

The Flow Editor displays the flow to the user and allows the user to interact with the flow using the mouse/trackpad and the keyboard and other input devices. Typically, the flow shows a set of nodes connected by links (edges) that represent some flow of data, or flow of control, which is in the problem domain of the application.

+

+

The editor displays the following object types which the user can interact with:

+ +

Pipeline Flow

+

The defintions of objects displayed within the flow editor are contained in a JavaScript object descibed by the pipeline flow schema. A pipeline flow object can be serialized to, or parsed from, JSON and consequently saved to, or read from, a file.

+

The storage and mangement of pipeline flow files is handled by the application. The pipeline flow can be read from the canvas controller:

+

    const pflow = canvasController.getPipelineFlow();
+
+and a previously saved pipeline flow can be provided to Common Canvas also using the canvas controller:

+
    canvasController.setPipelineFlow(pFlow);
+
+

A pipeline flow contains an array of pipelines, one of which is the ‘primary pipeline’. Other pipelines will be sub-flows for supernodes contained within the primary pipeline (or other pipelines).

+

Each pipeline defines sets of nodes, links and comments with all their associated attributes.

+

Create nodes

+

Nodes can be created by dragging from the palette and dropping onto the flow editor canvas:

+

+

Nodes can also be automatically added to the flow editor canvas by double clicking them in the palette. The node will be added at an appropriate place on the canvas and connected to the preceding node if one is available:

+

+

Object Selection

+

Objects (nodes, links and comments) on the flow editor canvas can be selected by single clicking on them. Subsequently, if the user Command/Ctrl + clicks on another object it will be:

+
    +
  • Added to the set of selected objects if it is not currently selected or
  • +
  • Removed from the set of selected objects if it is currently selected.
  • +
+

Clicking on the flow editor canvas background will deselect all canvas objects.

+

A selection region can be pulled out to select multiple objects at once by pressing the Shift key and dragging on the canvas background.

+

+

A set of connected nodes through a flow can be selected by, clicking on a node to select it and then, Shift + click another downstream node. If the nodes are joined together through a set of inter-connected nodes, all of those nodes will be selected.

+

+

All canvas objects can be selected by displaying the default context menu or context toolbar for the flow-editor background and then clicking the ‘Select All’ option. Note: If the application provides its own canvas menus then it would need to add the selectAll internal action to the menu to enable the ‘select all’ function.

+

Alternatively, the user can press Ctrl/Command + a keyborad shortcut to select all objects when the keyboard focus is on the canvas.

+

+ +

Links can be created by dragging from one node’s output port to another node:

+

+

Reposition Objects

+

The Flow Editor allows direct manipulation of selected objects by dragging them to a new position.

+

+

Flow editor canvas context menu or context toolbar

+

The flow editor can display a context menu or context toolbar (depending on which is enabled) that can show options for the whole flow. It is displayed by the context menu gesture which is either right-clicking the mouse, or the trackpad equivalent, while the mouse cursor is over the canvas.

+

Zoom Objects

+

The objects on the flow-editor canvas can be zoomed in and out either using the zoomIn or zoomOut toolbar buttons or by performing the zoom gesture using the mouse or trackpad. The canvas background can be panned by dragging the background.

+

+

A large flow can be centrally positioned, making all the nodes visible within the viewport, by clicking the zoomToFit button.

+

+

Command Stack

+

Command actions that update the objects within the flow editor are added to the command stack and can be undone and then redone. The default toolbar has undo and redo buttons and the default context menu/toolbar for the canvas background has undo and redo options. There are also keyboard shortcuts to undo and redo.

+

+

Clipboard

+

Users can cut and copy one or more objects from the canvas onto the clipboard and paste them into the same flow or a different flow. The default canvas toolbar provides buttons for these three actions and they can also be accessed through the default context menus/toolbars for the objects on the canvas.

+

There are also keyboard shortcuts to cut, copy and paste.

+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.01.01-nodes/index.html b/v13/01.01.01-nodes/index.html new file mode 100644 index 0000000000..8c14e55da5 --- /dev/null +++ b/v13/01.01.01-nodes/index.html @@ -0,0 +1,3036 @@ + + + + + + + + + + + + + + + + + + + + + + + Nodes - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Nodes

+

Introduction

+

Nodes displayed in the flow editor can represent data operations or other types of +operations for the type of flow the application displays. Nodes are joined together in a flow by data links (edges). Association links can also join two nodes together (in a non-directional relationship) and nodes can be connected to comments with a comment link.

+

Nodes can be customized into a large number of different layouts depending on the needs of the application, as shown below:

+

+ + + + + +

+

Pipeline Flow Node

+

Node objects are stored in the nodes array in a pipeline which itself is in the pipelines array of the pipeline flow. The node object in the pipeline holds some basic information used for node display such as:

+
    +
  • id - unique identifier
  • +
  • position - x/y position relative to the canvas origin
  • +
  • label
  • +
  • description – displayed in the node tooltip (if enabled)
  • +
  • image – image location on the server or a JSX object
  • +
  • ports info (inputs and outputs)
  • +
  • optional decoration information
  • +
+

Additional information on the node layout is configured in the node layout fields.

+

Node Elements

+

Nodes are made up of a number of display elements:

+

+

In addition, the node might have an ellipsis icon if context menus are enabled and one or more decorations. All of the node elements are optional, except the selection highlighting element.

+

Node Format

+

The enableNodeFormatType field of the canvas config object can be used to change the basic format of the node between “Horizontal” and “Vertical” format:

+

+

Each format type has its own set of node layout fields which can be overriden by the application for further customization.

+

Node image

+

The node image is specified in the image field of the node object in the pipeline flow and the palette. It can either be a URL to an image file on the server or a JSX object.

+

Here’s an example of a node using an image field with a URL that specifies an SVG object imported from a file on the server:

+

+

Here’s an example of a node using a JSX object which is an icon imported from the Carbon icons library.

+

+

Node positioning

+

Nodes are positioned at the x/y coordinate from the canvas origin of the node’s top-left corner. The x/y coordinates are stored in the node object from the pipeline flow.

+

Ports

+

A node can have zero or more input ports and zero or more output ports.

+
    +
  • Input ports are the objects to which connections are made from other nodes that indicate a flow of data or control into the node.
  • +
  • Output ports are the objects from which connections are made to other nodes that indicate a flow of data or control out of the node.
  • +
+

In many applications, nodes have just a single input and/or output port. There is no strict definition of why a node might have multiple input ports or multiple output ports. Often they are used to indicate a different kind of data flowing into or out of the node. For example, a relational-database Join node might have two input ports one for the left table data and one for the right table data in the join. Or a filter node might take data and split it into two parts based on some criteria and write out the different data to separate output nodes, like this:

+

+

There can be multiple connections made into an input port or out of an output port. Each port has a maximum and minimum cardinality values which indicate the limits on the number of connections for the port. Common Canvas ensures the maximum cardinality is not exceeded for port connections.

+

Ports can be shown visually on, or close to, the node as circles or images. This positioning is controlled by the node layout fields.

+

Node Selection

+

See the Object Selection section in the flow editor page for details.

+ +

A guide is an image or shape drawn at the location of an output port. The user can drag the ‘guide’ to another node to create a new link connection to that node. The appearance of the guide can be customized by altering the node layout fields and CSS for the node. If output ports are visible, it appears as if the user is dragging the port but the guide is, in fact, a different node element.

+

+

+

Resizable nodes

+

Nodes can be resized if the enableResizeableNodes canvas config field is set to true. The node can be resized if the user drags the border area in the desired direction.

+

+ +

Nodes can be inserted into a link and the flow re-wired appropritely, if enableInsertNodeDroppedOnLink canvas config field is set to true.

+

+ +

A Node can be highlighted when a guide icon is being dragged towards it and a connection is allowed, if enableHighlightNodeOnNewLinkDrag canvas config field is set to true.

+

+

Other behaviors

+

There are other node behaviors which can be switched on or off. See the nodes section on the canvas config object.

+

React Nodes

+

The body of nodes can be drawn using a React object in place of the SVG elements usually displayed by Common Canvas. The React object is specified by the application in the nodeExternalObj field in the node layout fields.

+

Here is an example of two nodes using the CardNode React object from the Carbon Charts (React) library.

+

+

Branch highlighting

+

Branch highlighting allows the user to highlight upstream nodes, downstream nodes or the entire branch of nodes (both upstream and downstream) by choosing one of the ‘Highlight’ options from the context menu for a particular node.

+

Clicking this:

+

+

Results in this:

+

+

Common Canvas add a class called ‘d3-branch-highlight’ to the group <g> object of the node or the link in the branch. The application can use that class to specify its own CSS styling for the branch highlighting.

+

By default, Common Canvas provides colors and styles for the highlighted branch nodes and links, but the application can override them if desired using CSS like this.

+
d3-node-group.d3-branch-highlight {
+    .d3-node-body-outline {
+        stroke: red;
+    }
+}
+
+

Supernodes

+

A supernode is a special kind of node that can reference another pipeline in the same pipeline flow object (internal) or, in some cases, in another pipeline flow (external). The referenced pipeline is known as a sub-flow or sub-pipeline.

+

These types of supernodes/sub-flows combinations have different uses:

+
    +
  • Internal sub-flows are useful for organizing flows by separating out closely related sets of nodes from the main flow.
  • +
  • External sub-flows are useful for function reuse. That is, if a set of nodes performs some function that is needed by more than one pipeline flow, they would be placed into their own pipeline flow and referenced by the others.
  • +
+

+

Common Canvas allows the user to create a supernode by selecting a set of nodes to be placed in the sub-flow:

+

+

And then to view it as expanded ‘in-place:

+

+

The user can also navigate to a ‘full-page’ view (well, really it is a full-viewport view):

+

+

The sub-flow can be edited by the user, after it has been created, by adding, removing or editing its nodes, comments and links.

+

Supernodes can also be:

+
    +
  • Deconstructed – that is, the supernode is remove from the flow and the nodes from the sub-flow are inserted back into the parent flow.
  • +
  • Converted from internal to external - this means the sub-flow is removed from its current pipeline flow and placed in a separate pipeline flow object that can be saved by the application.
  • +
  • Converted from an external to internal - this means the sub-flow is copied from the external pipeline flow and placed into the pipeline flow of the supernode.
  • +
+

See the External Pipeline Flows page for more details.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.01.02-links/index.html b/v13/01.01.02-links/index.html new file mode 100644 index 0000000000..1926a5ec34 --- /dev/null +++ b/v13/01.01.02-links/index.html @@ -0,0 +1,2924 @@ + + + + + + + + + + + + + + + + + + + + + + + Links - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+ +
+ + + +
+
+ + + + + + + + + + + + +

Links

+ +

Introduction

+

Common Canvas will display one link line on the canvas for each link defined in the pipeline flow. There are three categories of links supported by Common Canvas:

+
    +
  • Data links - a directional link between the ports on two nodes
  • +
  • Association links - a relationsjip link between two nodes with no implied direction
  • +
  • Comment links - a link from a comment to a node
  • +
+

The appearance and behavior of links can be customized by:

+ + +

Data links are designed to model a flow from a source node to a target node. Data links are specified in the pipeline flow to connect a port on the source node to a port on the target node. Data links are typically displayed with an arrow head to display the flow along the link from source to target.

+

There are 4 types of data link:

+
    +
  • Curve
  • +
  • Elbow
  • +
  • Parallax
  • +
  • Straight
  • +
+

These can be changed by specifying the enableLinkType configuration field.

+

Also, data links can be drawn either:

+
    +
  • between ports on the source and target nodes or
  • +
  • “freeform” which means the link is drawn from and to the source node ignoring port positions. Typically with “freeform” links the application customizes the canvas to NOT display ports.
  • +
+

This choice can be changed by specifying the enableLinkMethod configuration field.

+

This table shows the different combinations:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
TypePortsFreeform
Curve
Elbow
Parallax
Straight
+ +

The enableLinkMethod config field can be set to “Freeform”. With this option, links are drawn directly between nodes without any reference to ports.

+

Here is an example of a “Freeform” link with a “Straight link type:

+

+

The “Freeform” display is useful for flows where no particular direction of the flow needs to be enforced and also where the flow has nodes with multiple ports but the ports do not need to be shown in any particular order or context.

+

When more than one link enters or exits into/from one side of the node the freeform links will be drawn to maintain some reasonable separation like this:

+

+

+ +

When enableLinkMethod is set to “Ports”, Common Canvas draws the links to maintain the flow from output ports to input ports regardless of where the nodes are positioned relative to one another.

+

Here is an example of the “Ports” links method with a “Curve” link type:

+

+

Links between ports can be useful if the application wants to allow the user to create loops in the flow, like this:

+

+ +

Applications can configure Common Canvas to position ports by default in one of four different orientations:

+
    +
  • For “LeftRight” output ports will be on the right of the node and input ports will be on the left of the node
  • +
  • For “RightLeft” output ports will be on the left of the node and input ports will be on the right of the node
  • +
  • For “TopBottom” output ports will be on the bottom of the node and input ports will be on the top of the node
  • +
  • For “BottomTop” output ports will be on the top of the node and input ports will be on the bottom of the node
  • +
+

The application can further customize the position of the ports on the node if desired.

+

The position of the ports will affect the way links are drawn from/to the port.

+

This can result in this kind of display where links draw into/out of ports in multiple different directions:

+

+ +

If the enableSelfRefLinks canvas config field is set to true, the user can create links that loop back to the node they originated from. These are created by the user dragging from the output port on the node and dropping the new link onto a input port, or the body, of the same node.

+

Here’s an example of a self-referencing link using a “Freeform” link method with an “Elbow” link type.

+

+

Self-referencing links behave the same as other links and can be displayed in the same combinations of link types (Curve, Elbow, Parallax and Straight) and link methods (Freeform or Ports) as other links.

+ +

Note: Typically data links must be drawn between nodes however, if the config field enableLinkSelection is set to Detachable, the links are allowed to be drawn to and/or from arbitrary points on the canvas.

+

With this option, the start and end of existing links can be dragged away from the nodes they are connected to and dropped onto an arbitrary point on the canvas to create a ‘detached’ link. Additionally, the end of any detached link that originates from, or points to, a point on the canvas can be dragged to a node to form a connection. Also, nodes can be dragged to the available ends of those ‘detached’ links to form a connection.

+ +

Association links are designed to capture a relationship between two nodes where there is no implied direction. By default these are displayed as a single straight link line in a dashed style. There are no arrow heads by default for that type of link.

+

+ +

Comment links connect a comment to one or more nodes. They can be created by the user by: (a) pulling out the small handle/circle that appears below a comment and dropping it on a node. (2) by selecting nodes before the comment is selected and then creating the comment. This will automatically create a comment link from the selected nodes to the newly created comment.

+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.01.03-comments/index.html b/v13/01.01.03-comments/index.html new file mode 100644 index 0000000000..69939f2797 --- /dev/null +++ b/v13/01.01.03-comments/index.html @@ -0,0 +1,2880 @@ + + + + + + + + + + + + + + + + + + + + + + + Comments - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Comments

+

Introduction

+

Comments are used to display explanatory information to the user about the flow. Comments can be standalone:

+

+

or connected to one or more nodes with a link line:

+

+

Creating comments

+

Comments can be created by the user by clicking the ‘Create Comment’ button in the default toolbar or by clicking the ‘Create comment’ option in the default canvas background context menu/toolbar.

+

Note: If the application is providing its own toolbar it would need to have a button defined with the createAutoComment internal action and, in the context menu/toolbar for the canvas background there would need to be an option defined with the createComment internal action.

+

Editing comments

+

The comment can be changed to ‘edit mode’ by either double clicking it or clicking the ‘Edit Comment’ option in the default context menu/toolbar. If the application is providing its own context menu/toolbar it would need to provide an option with the setCommentEditingMode action defined.

+ + + + + + + + + + + + + +
Doing this:Yields this:
+

After text has been entered into the comment entry area the user can click on the canvas background to finish editing and display the comment.

+

Comment Selection

+

See the Object Selection section in the flow editor page for details.

+

Resizing comments

+

Comments can be sized by dragging the border.

+

+

Connecting a comment to a node

+

The user can create a link from a comment to a node in one of two way:

+
    +
  1. By dragging the small gray ‘guide’ circle at the bottom of the comment and dropping it onto the target node or
  2. +
  3. By selecting one of more nodes and then creating the comment. This will automatically add comment links from the comments to each selected node.
  4. +
+

By default, the connection is shown as a dashed link line.

+

+

Coloring comment backgrounds

+

Comment backgrounds can be colored by clicking the ‘Change Color’ option in the default context menu/toolbar for the comment. If the application provides its own context menu/toolbar then the colorSelectedObjects action must be defined.

+

+

Markdown

+

Optionally, the application can switch on markdown support within comments. This allows the user to enter markdown syntax when the comment is in edit mode, which is then formated approriately in display mode.

+ + + + + + + + + + + + + +
Entering this:Yields this:
+

HTML in Markdown

+

As well as markdown syntax, the user can also enter HTML into the markdown text. This is then run when the comment is presented in display mode. HTML support allows the user more extensive customization capabilities including the specification of CSS for the text.

+

For example, specifying this will color the word Red: +

Some <span style="color: red">Red</span> text
+
+Here’s an example comment with some sample HTML:

+

html-markdown

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.01.04-decorations/index.html b/v13/01.01.04-decorations/index.html new file mode 100644 index 0000000000..7974a8bf8c --- /dev/null +++ b/v13/01.01.04-decorations/index.html @@ -0,0 +1,2732 @@ + + + + + + + + + + + + + + + + + + + + + + + Decorations - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Decorations

+

Introduction

+

In Common Canvas, nodes and links can be enhanced with decorations. A decoration is either:

+
    +
  • +

    An image

    +

    +

    +
  • +
  • +

    A label

    +

    +

    +
  • +
  • +

    A shape (SVG path) or

    +

    +

    +
  • +
  • +

    A JSX object

    +

    +

    +
  • +
+

Decorations can be applied to a node or link in a number of ways and can be made interactable so when the user clicks on the decoration the decorationActionHandler is called.

+

Here are some node decorations displayed by IBM’s SPSS Modeler application:

+

+

Here are some link decorations displayed by IBM’s Data Stage application:

+

+

Note: The link labels and the small icons on the links are decorations as well as the circle at the source of each link.

+

Tips for Decorations

+

A decoration can have a tooltip:

+

+

Editable label decoration

+

Label decorations can be made to be editable.

+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.02-palette/index.html b/v13/01.02-palette/index.html new file mode 100644 index 0000000000..611b027cc5 --- /dev/null +++ b/v13/01.02-palette/index.html @@ -0,0 +1,2779 @@ + + + + + + + + + + + + + + + + + + + + + + + Palette - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Palette

+ +

Introduction

+

Common Canvas can display a palette which appears in the left flyout panel. The palette can display a set of palette nodes (sometimes called node templates) split into different categories.

+

+

The user can drag any of the palette nodes from the palette onto the canvas to create a new instance of a node on the canvas. The user can then join nodes together by creating link lines.

+

+

Alternatively, the user can double click node templates to add, and automatically join, them to the current flow. When a palette node is double clicked, Common Canvas will look for the node at the end of the current flow of nodes and will add the node to the canvas to the right of the end node in the flow. Common Canvas will also automatically join the nodes together by creating a new link line.

+

+

Populating the palette

+

The host application must tell Common Canvas what node templates and categories it wants the user to see in the palette. This is done by providing Common Canvas with a palette (JavaScript) object by calling the canvas-controller setPipelineFlowPalette(palette) method. The palette object is often read from a JSON document stored in the host application’s repository or it can be automatically generated.

+

The palette object should conform to the palette schema. The host application can ensure that palette conforms to the schema by switching on schema validation in the canvas configuration.

+

Palette configuration

+

There are a number of configuration options that control the palette that are specified in the canvas config object:

+ +

Palette operation

+

The palette contents can be manipulated calling these canvas-controller palette methods

+

The palette can be opened and closed using these canvas-controller operational methods

+ +

The user can enter a search string into the Search field at the top of the palette. The behavior is as follows:

+
    +
  1. When the search field is empty, the palette shows categories which can be expanded to show the node templates within the expanded category.
  2. +
  3. As the user enters characters into the search field, Common Canvas immediately searches through the node template labels, node template descriptions and category labels for the characters entered and finds a subset of the node templates that match the search string. Common Canvas then replaces the category-based view of the palette with a list of node templates that match the search criteria. In this view the node templates also include the node description under the node label.
  4. +
  5. The search is case insensitive.
  6. +
  7. The search text is highlighted wherever it appears in the labels, descriptions or category labels.
  8. +
  9. Common Canvas uses a ranking algorithm to order the display of node templates so those most relevant to the search text are positioned at the top of the list. The ranking algorithm puts more weight on those node templates where the search text appears in the label than if the text appears in the description or the category label.
  10. +
  11. If the user enters words into the search field (separated by spaces), Common Canvas searches for each word separately. Therefore, if say the user entered “data import” into the search field, Common Canvas would find a node with “Import Data” as the label or even a node with a description that said “This node does lots of things and also data can be imported.”
  12. +
  13. Nodes that have text which have hits on multiple words are ranked more highly than node templates whose text only contains one search word.
  14. +
  15. Common Canvas uses a debounce function so that, if the user types the search string very quickly, Common Canvas does not perform multiple searches for each key press but only runs a full search when rapid typing has ended.
  16. +
+

Recommendation

+

According to the schema, node template descriptions are not mandatory in the palette object. However, it is recommended you provide descriptions for each of your node templates. The reason for this is that, the search function searches through node template descriptions, as well as node template labels and category labels. This means you can write your node template descriptions to contain appropriate keywords that a user might search for when looking for a node.

+

For example, if there is a node template called ‘Import Data’ and that node could import comma-separated files you could add ‘comma-separated’ and ‘csv’ into your description for that node template. If the user entered comma in the search field the ‘Import Data’ node template would be shown in the search results even though comma does not appear in the node template label.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.03-context-menu/index.html b/v13/01.03-context-menu/index.html new file mode 100644 index 0000000000..e35de33271 --- /dev/null +++ b/v13/01.03-context-menu/index.html @@ -0,0 +1,2668 @@ + + + + + + + + + + + + + + + + + + + + + + + Context Menu - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Context menu

+

A context menu is a small window containing a list of options applicable to an object or set of selected objects.

+

A traditional context menu, like the one shown below, will be displayed when enableContextToolbar is set to false (or omitted) from the canvas config, and either:

+
    +
  • a right-click (or equivalent on a trackpad) is performed on a canvas object or the canvas background or
  • +
  • Control + click is done on an obejct or
  • +
  • the ellipsis icon (only shown for nodes) is clicked
  • +
+

image

+

Populating the context menus

+

By default, Common Canvas will display appropriate context toolmenus for all flow editor canvas objects. The application can override or replace the default menus by implementing the contextMenuHandler callback.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.04-context-toolbar/index.html b/v13/01.04-context-toolbar/index.html new file mode 100644 index 0000000000..a9b61df645 --- /dev/null +++ b/v13/01.04-context-toolbar/index.html @@ -0,0 +1,2671 @@ + + + + + + + + + + + + + + + + + + + + + + + Context Toolbar - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Context Toolbar

+

Context toolbars are an alternative to traditional context menus. It can be enabled by setting enableContextToolbar to truein the canvas config.

+

A context toolbar is a small toolbar that appears above nodes, links and comments as the mouse cursor is hovered over them. A context toolbar for the canvas background can also be displayed by right-clicking on the canvas background.

+

The context toolbar displays a set of icons that represent the most likely actions the user would want to perform on the object under the mouse cursor. If necessary, the toolbar can also show an overflow (vertical ellipsis) icon that, when clicked, reveals additional actions that can be performed on the object.

+

For a “vertical” style node the context toolbar looks like this:

+

image

+

For a “horizontal” style node it looks like this:

+

image

+

when the user clicks the overflow icon it looks like this:

+

image

+

Note: Since the mouse cursor can be hovered over a node, comment or link that is NOT currently selected, the actions shown in the context toolbar will apply to just that object, even if there is one or more currently selected objects.

+

If the mouse cursor is hovered over a selected object when there are other selected objects, the actions in the context toolbar will be applicable to all the selected objects. This is the same as how a traditional content menu shows actions that are applicable to the set of selected objects.

+

Populating the context toolbars

+

By default, Common Canvas will display appropriate context toolbars for all flow editor canvas objects. The application can override or replace the default toolbars by implementing the contextMenuHandler callback.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.05-toolbar/index.html b/v13/01.05-toolbar/index.html new file mode 100644 index 0000000000..734ebeb20c --- /dev/null +++ b/v13/01.05-toolbar/index.html @@ -0,0 +1,2884 @@ + + + + + + + + + + + + + + + + + + + + + + + Toolbar - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Canvas Toolbar

+

The canvas toolbar appears at the top of the <div> occupied by Common Canvas. By default, the toolbar will be displayed with appropriate buttons to perform the most popular actions. The toolbar can be customized by the application to show whatever action buttons are required. Buttons can be positioned to the left or right of the toolbar area.

+

+

The toolbar is customized by providing a toolbar config object as one of the <CommonCanvas> props.

+

The toolbar can be hidden by setting the enableToolbarLayout field in the canvas config to “None”.

+

Overflow Menu Behavior

+

If there is enough width all toolbar buttons will be displayed:

+

+

but when the width is too narrow to accommodate all the buttons, the toolbar will display an overflow button:

+

+

and when the overflow button is clicked, a menu appears showing the extra action buttons:

+

+

Dividers

+

Action buttons can be separated by dividers so that related buttons can be grouped together:

+

+

Tooltips

+

Action buttons will show a tooltip on hover, if a label is provided:

+

+

Enabled/Disabled actions

+

Action buttons in the toolbar can be shown as either enabled or disabled:

+

+

+

Selected state

+

Action buttons that reflect a state can be displayed as unselected or selected:

+

+

+

Text with icon

+

Action buttons can be shown with text alongside the icon (either before or after), if a label is provided:

+

+

+

Carbon button styling

+

Action buttons can be styled like Carbon buttons, if required:

+

+ +

Action buttons can be configured to display a menu of related actions:

+

+

Sub-panel

+

Action buttons can be configured to display an application-specific panel:

+

+

Dual purpose buttons

+

Action buttons can be configured to be ‘dual purpose’ where the left side of the button peforms the associated action:

+

+

and the right side opens a panel of application-specified settings, associated with the left side part of the button:

+

+

Imbedded JSX

+

JSX can be provided to add additional information into the toolbar. In this case, the Autosaved time is added amongst the regular toolbar action buttons.

+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.06-notification-panel/index.html b/v13/01.06-notification-panel/index.html new file mode 100644 index 0000000000..df576679ac --- /dev/null +++ b/v13/01.06-notification-panel/index.html @@ -0,0 +1,2611 @@ + + + + + + + + + + + + + + + + + + + + + + + Notification Panel - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Notification Panel

+

The Notification Panel allows the application to display error, warning, success and info messages to the user.

+

+

The canvas toolbar provides a button to display a sub-panel that contains the notifications. The button will be added to the far right side of the toolbar whenever a Notification Config is specified to the <CommonCanvas> React object.

+

The notifications button in the toolbar indicates the state and number of the messages in the panel.

+

A colored circle will be shown to indicate the highest severity message to be shown in the notification panel: info/blue, success/green, warning/yellow or error/red. Also, a number is displayed within the icon to indicate the number of current messages.

+

+ + +

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.07-state-tag/index.html b/v13/01.07-state-tag/index.html new file mode 100644 index 0000000000..9ec357bd27 --- /dev/null +++ b/v13/01.07-state-tag/index.html @@ -0,0 +1,2610 @@ + + + + + + + + + + + + + + + + + + + + + + + State Tag - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

State Tag

+

A State Tag is an optional pill-shaped object that can displays the state of the flow to the user. This can be either ‘Locked’ or ‘Read-only’. By default, no state tag is shown.

+

The application can request a state tag be shown by setting the enableStateTag canvas config field.

+

When the State Tag is display it will show a tooltip, with an appropriate message, if the mouse cursor is hovered over it. The application can override the default message by implementing the Tip Handler callback.

+

+
+

Note

+

It is the application’s reponsibility for making the flow read-only. See the Read Only or Locked Flows page for more details.

+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.08-tooltips/index.html b/v13/01.08-tooltips/index.html new file mode 100644 index 0000000000..749ffc4d48 --- /dev/null +++ b/v13/01.08-tooltips/index.html @@ -0,0 +1,2695 @@ + + + + + + + + + + + + + + + + + + + + + + + Tooltips - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Tooltips

+

Tooltips (sometimes referred to as just ‘tips’) can be shown for the following objects in Common Canvas:

+ +

The application can switch tooltips on and off as required for the different types of object and can also customize what text is displayed in the tooltip.

+

Tooltips for the toolbar:

+

+

Configuring tooltips

+

By default, Common Canvas will display tooltips where appropriate. Tooltips can be switched on or off, for evertything except the toolbar, by specifying the tipConfig field for the canvas config object.

+

Customizing tooltip text

+

The text for tooltips can be customized, for evertything except the toolbar, by implementing the Tip Handler callback.

+

The toolbar tooltips can be customized as part of the Toolbar Config.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.09-panels/index.html b/v13/01.09-panels/index.html new file mode 100644 index 0000000000..3150f7948c --- /dev/null +++ b/v13/01.09-panels/index.html @@ -0,0 +1,2679 @@ + + + + + + + + + + + + + + + + + + + + + + + Panels - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Panels

+

Common Canvas has three additional optional panels: top panel, bottom panel, right-flyout. These can be used by the application to display any additional content to the user such as logs or data previews or properties of selected objects. Traditionally, the right-flyout is used to display node properties.

+

The application can, optionally, use Common Properties to display the controls for the properties of a node.

+

+

Customizing content and display

+

The content and display of the panels is controlled by these six props of the <CommonCanvas> React object:

+
    +
  • showRightFlyout - a Boolean
  • +
  • +

    rightFlyoutContent - JSX object showing content

    +
  • +
  • +

    showBottomPanel - a Boolean

    +
  • +
  • +

    bottomPanelContent - JSX object showing content

    +
  • +
  • +

    showTopPanel - a Boolean

    +
  • +
  • topPanelContent - JSX object showing content
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/01.10-common-properties/index.html b/v13/01.10-common-properties/index.html new file mode 100644 index 0000000000..b3166ec638 --- /dev/null +++ b/v13/01.10-common-properties/index.html @@ -0,0 +1,2608 @@ + + + + + + + + + + + + + + + + + + + + + + + Common Properties - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Common Properties

+

Common Properties is a React component that allows the application to display a set of UI components (radio buttons, text entry areas, drop down lists, sliders, tables, etc) to present the properties of a node to the user. Common Properties supports the most commonly used UI components and also allows custom components to be added into its visual output.

+

The Common Properties React object has a ‘prop’ called the Parameter Definition which is a Javascript (JSON) object and some configuration objects as props.

+

Traditionally, the right-flyout is used to display properties but the application can choose to display them elsewhere. The UI elements displayed by Common Properties comply with the Carbon design language.

+

For more details and examples of the components supported by Common Properties, see the Common Properties Components page.

+

Here’s an example of the type of output a <CommonProperties> React object can produce. This is from the IBM SPSS Modeler application:

+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/02-set-up/index.html b/v13/02-set-up/index.html new file mode 100644 index 0000000000..d9d3cb3ce4 --- /dev/null +++ b/v13/02-set-up/index.html @@ -0,0 +1,2883 @@ + + + + + + + + + + + + + + + + + + + + + + + Installation - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+ +
+ + + +
+
+ + + + + + + + + + + + +

Installation

+

NPM Install

+

You’ll need to build your application with Elyra Canvas.

+
    +
  • Elyra Canvas requires react, react-dom, react-intl, and react-redux libraries to be installed. See peerDependencies in package.json for versions requirements.
  • +
+

Use the command: +

npm install @elyra/canvas --save-dev
+
+or add this to your package.json file:

+

  "@elyra/canvas": "x.x.x"
+
+where x.x.x is the latest build and then run: +
npm install
+

+

Localization

+

If you want to see text displayed by Elyra Canvas components in different languages you must wrapper <CommonCanvas> and <CommonProperties>in an <IntlProvider> object.

+

The sample code below shows how <IntlProvider> should be imported and initialized. Your code can set this.locale to indicate which language should override the default which, in this sample code, is set to English en. The default locale will be used if this.locale is set to a language which is not currently supported.

+

If you want to provide translations for your own application’s text you can import your own bundles and load them into the this.messages object along with the common-canvas and common-properties text. If you do this you will have to move <IntlProvider/> so that it wrappers your React objects as well as <CommonCanvas/> and/or <CommonProperties>.

+
import { IntlProvider } from "react-intl";
+
+import CommandActionsBundles from "@elyra/canvas/locales/command-actions/locales";
+import CommonCanvasBundles from "@elyra/canvas/locales/common-canvas/locales";
+import CommonPropsBundles from "@elyra/canvas/locales/common-properties/locales";
+import PaletteBundles from "@elyra/canvas/locales/palette/locales";
+import ToolbarBundles from "@elyra/canvas/locales/toolbar/locales";
+
+class App extends React.Component {
+
+constructor() {
+    this.locale = "en";
+    // Create messages object once (here in constructor) - do not create messages
+    // in the render method, otherwise unnecessary renders inside
+    // common-canvas/common-properties will be performed.
+    this.messages = this._getMessages(
+        this.locale,
+        [CommandActionsBundles, CommonCanvasBundles, CommonPropsBundles,
+         PaletteBundles, ToolbarBundles]
+    );
+}
+
+_getMessages(locale, bundles) {
+  const messages = {};
+  for (const bundle of bundles) {
+    Object.assign(messages, bundle[locale]);
+  }
+  return messages;
+}
+
+render() {
+  <IntlProvider locale={this.locale} defaultLocale="en" messages={this.messages}>
+    {Add your <CommonCanvas/> or <CommonProperties/> element here.}
+  </IntlProvider>
+}
+
+

Overriding Styles and Color Themes

+

When building your application you will need to load fonts and override styles:

+

CSS styling for quick start

+

If you just want to get up and running and don’t care about scss then import these regular CSS files:

+
    +
  • @elyra/canvas/dist/styles/common-canvas.min.css
      +
    • version 8.x and older @elyra/canvas/dist/common-canvas.min.css
    • +
    +
  • +
+

More information about carbon components can be found here https://carbondesignsystem.com/developing/frameworks/react#getting-started

+ +

If you want to use the full power of scss styling with variable overrides etc then include these imports in your main SCSS file: +

@use "@carbon/react"; // Bring in all the styles for Carbon in your root/global stylesheet
+@import "@elyra/canvas/src/index.scss";
+

+
    +
  • use autoprefixer when building
  • +
  • if using webpack under the sass-loader and make sure to include
  • +
+
options: { includePaths: ["node_modules"] }
+
+

Again, you can refer to the test harness harness.scss and common.scss files for sample code.

+

3rd party styling

+

If you are using Common Properties then also include the react-virtualized styles: + - react-virtualized/styles.css

+

Loading Fonts

+

To get correct and efficient display of fonts in Elyra Canvas, the build process for your application should copy the IBM Plex font files from /node_modules/@ibm/plexto a ./fonts folder and the following should be added to the .scss file for your application:

+
@use "@carbon/react" as * with (
+    $font-path: "/fonts"
+);
+
+$font-prefix: './fonts';
+@import 'node_modules/@ibm/plex/scss/ibm-plex.scss';
+
+

You can see an example of this in the common.scss file for the Elyra Canvas Test Harness. The Test Harness is the equivalent of a host application.

+

The Gruntfile that builds the Test Harness contains the following, that ensures the fonts are copied from /node_modules/@ibm/plex to the <carbon fonts folder>: +

copy: {
+    fonts: {
+        files: [{
+            expand: true,
+            flatten: false,
+            cwd: "./node_modules/@ibm/plex",
+            src: ["IBM-Plex*/**"],
+            dest: ".build/fonts"
+        }]
+    }
+}
+...
+var buildTasks = ["copy:fonts"];
+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03-common-canvas/index.html b/v13/03-common-canvas/index.html new file mode 100644 index 0000000000..fe616bce31 --- /dev/null +++ b/v13/03-common-canvas/index.html @@ -0,0 +1,3108 @@ + + + + + + + + + + + + + + + + + + + + + + + Getting Started - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Getting started with Common Canvas

+

Hello Canvas!

+

Common Canvas is a React component. The <CommonCanvas> component is displayed in a <div> provided by your application. Here’s some sample code to show the minimum code needed to get a working canvas.

+
import React from "react";
+import AllTypesCanvas from "../../test_resources/diagrams/allTypesCanvas.json";
+import ModelerPalette from "../../test_resources/palettes/modelerPalette.json";
+import { CommonCanvas, CanvasController } from "@elyra/canvas";
+
+class App extends React.Component {
+    constructor(props) {
+        super(props);
+
+        this.canvasController = new CanvasController();
+        this.canvasController.setPipelineFlow(AllTypesCanvas);
+        this.canvasController.setPipelineFlowPalette(ModelerPalette);
+    }
+
+    render() {
+        return (
+            <div id="harness-app-container">
+                <CommonCanvas
+                    canvasController={this.canvasController}
+                />
+            </div>
+        );
+    }
+}
+
+

This code will display this:

+

+

The “Tiny App” is available as part of the test harness function. Click here to see the app running. You can try: dragging a node, editing a comment (double click on it), drag a node from the palette, click a button on the toolbar, zoom in and out using the scroll gesture.

+

Some sample code to look at:

+
    +
  • +

    This is the source code for app-tiny.js.

    +
  • +
  • +

    This app app-small.js, is more sophisticated and shows many of the options available to a Common Canvas developer such as configurations and callback handlers.

    +
  • +
  • +

    You can also look at the App.js file in the test harness section of this repo to see examples of code that uses the common-canvas component.

    +
  • +
+

Canvas Controller

+

The only mandatory prop for the <CommonCanvas> component is a regular JavaScript class called the Canvas Controller.

+

The Canvas Controller handles calls from the host application and actions performed by the user. It then updates the internal object model which stores:

+
    +
  1. The data that describes the flow of nodes, links and comments (called a pipelineFlow);
  2. +
  3. The data that describes the definition of the palette which contains node templates that can dragged to add nodes to the canvas;
  4. +
  5. The set of currently selected objects.
  6. +
  7. Notification messages
  8. +
  9. Breadcrumbs that indicate which sub-flow is being viewed
  10. +
  11. Layout information
  12. +
  13. And more …
  14. +
+

The Canvas Controller provides an API which allows the application to:

+
    +
  1. Set a new Pipeline Flow
  2. +
  3. Get the current pipelineFlow (after the user has edited it)
  4. +
  5. Update and edit objects in the canvas (for example, add node, delete link etc.)
  6. +
  7. Set the node definition data (for display of nodes in the palette)
  8. +
  9. Operate other aspets of the UI like opening panels, zooming, etc, etc.
  10. +
+

Getting started

+

To use Common Canvas in your React application complete the following steps:

+

Step 1 : Setup

+

Complete the setup steps documented in the Initial Setup page.

+

Step 2 : Import Common Canvas

+

Import the Common Canvas and Canvas Controller from the Elyra Canvas library. Elyra Canvas produces both esm and cjs outputs. By default esm will be used when webpack is used to build the application.

+

    import { CommonCanvas, CanvasController } from "@elyra/canvas";
+
+Common Canvas Only

+

To import only Common Canvas functionality in cjs format use:

+
    import { CommonCanvas, CanvasController } from "@elyra/canvas/dist/lib/canvas";
+
+

Step 3 : Create an instance of the canvas controller

+

To control the canvas you’ll need an instance of the canvas controller. Create an instance like this (probably in the constructor of your object). +

    this.canvasController = new CanvasController();
+

+

Step 4 : Set the palette data

+

Next you’ll need to populate the palette data. This step is optional if you don’t want to use the palette.

+

The palette data will specify the nodes (split into categories) that will appear in the palette. This is done by calling the canvas controller with:

+
    this.canvasController.setPipelineFlowPalette(pipelineFlowPalette);
+
+

The pipelineFlowPalette object should conform to the JSON schema found here:
+https://github.com/elyra-ai/pipeline-schemas/tree/master/common-canvas/palette

+

Some examples of palette JSON files can be found here:
+https://github.com/elyra-ai/canvas/tree/master/canvas_modules/harness/test_resources/palettes

+
+

Images

+

If the palette file references any images using a path you need those image files at the appropriate location.

+
+

Step 5 : (Optional) Set the flow data

+

This is an optional step. If you want a previously saved flow to be shown in the flow editor, so the user can start to edit it, you will need to call the canvas controller with:

+
    this.canvasController.setPipelineFlow(pipelineFlow);
+
+

The pipelineFlow object should conform to the JSON schema found here:
+https://github.com/elyra-ai/pipeline-schemas/tree/master/common-pipeline/pipeline-flow

+

Some examples of pipeline flow JSON files can be found here:
+https://github.com/elyra-ai/canvas/tree/master/canvas_modules/harness/test_resources/diagrams

+
+

Images

+

If the palette file references any images using a path you need those image files at the appropriate location.

+
+

Step 6 : Display the canvas

+

Inside your render code, add the following:

+

    return (
+        <div>
+            <IntlProvider>
+                <CommonCanvas canvasController={this.canvasController} />
+            </IntlProvider>
+        </div>
+    );
+
+The <div> should have the dimensions you want for your canvas to display in your page. For the canvasController prop, pass the instance of canvas controller created earlier. This is the only mandatory property. After providing this, and running your code, you will have a fully functioning canvas including: a palette; default toolbar; context menus; direct manipulation (move and resize) etc. To customize these behaviors and presentation continue with the sections below.

+

See the Localization section of the Initial Setup page to see how <IntlProvider> can be configured.

+

Common Canvas customization

+

If you want to customize the behavior of Common Canvas you can specify any combination of the following optional settings: +

    return (
+        <div>
+            <CommonCanvas
+                canvasController={this.canvasController}
+
+                config={this.canvasConfig}
+                toolbarConfig={this.toolbarConfig}
+                notificationConfig={this.notificationConfig}
+                contextMenuConfig={this.contextMenuConfig}
+                keyboardConfig={this.keyboardConfig}
+
+                contextMenuHandler={this.contextMenuHandler}
+                beforeEditActionHandler={this.beforeEditActionHandler}
+                editActionHandler={this.editActionHandler}
+                clickActionHandler={this.clickActionHandler}
+                decorationActionHandler={this.decorationActionHandler}
+                layoutHandler={this.layoutHandler}
+                tipHandler={this.tipHandler}
+                idGeneratorHandler={this.idGeneratorHandler}
+                selectionChangeHandler={this.selectionChangeHandler}
+                actionLabelHandler={this.actionLabelHandler}
+
+                showRightFlyout={showRightFlyout}
+                rightFlyoutContent={rightFlyoutContent}
+
+                showBottomPanel={showBottomPanel}
+                bottomPanelContent={bottomPanelContent}
+
+                showTopPanel={showTopPanel}
+                topPanelContent={topPanelContent}
+            >
+            </CommonCanvas>
+        </div>
+    );
+

+

Config objects

+

Common Canvas has five optional configuration objects. They are documented here: Config Objects

+

Handlers

+

There are several optional handlers implemented as callback functions. They are documented here: Common Canvas Callbacks

+

Right-flyout panel parameters

+

The right flyout panel appears on the right of the canvas area. You can add whatever content you like to this panel. Typically, it is used to display properties of nodes on the canvas. There are two optional parameters to let you manage the right flyout panel These are:

+
    +
  • showRightFlyout: This can be true or false to indicate whether the flyout panel is shown or not. The default is false.
  • +
  • rightFlyoutContent: content to display in the right flyout which is a JSX object. Nothing is displayed by default.
  • +
+

Bottom panel parameters

+

The bottom panel appears below the canvas area and between the palette and the right flyout panel. You can add whatever content you like to this panel. There are two optional parameters to let you manage the bottom panel. These are:

+
    +
  • showBottomPanel: This can be true or false to indicate whether the bottom panel is shown or not. The default is false.
  • +
  • bottomPanelContent: content to display in the bottom panel which is a JSX object. Nothing is displayed by default.
  • +
+

Top panel parameters

+

The top panel appears below the toolbar and between the palette and the right flyout panel. You can add whatever content you like to this panel. There are two optional parameters to let you manage the top panel. These are:

+
    +
  • showTopPanel: This can be true or false to indicate whether the top panel is shown or not. The default is false.
  • +
  • topPanelContent: content to display in the top panel which is a JSX object. Nothing is displayed by default.
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.02-configuration/index.html b/v13/03.02-configuration/index.html new file mode 100644 index 0000000000..f37147020e --- /dev/null +++ b/v13/03.02-configuration/index.html @@ -0,0 +1,2744 @@ + + + + + + + + + + + + + + + + + + + + + + + Configuration Overview - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Configuration Overview

+

There are five configuration objects that can be passed as props to the <CommonCanvas> +React object:

+

Canvas Config

+

This is the main canvas configuration. It allows customization of many aspects of the flow editor, visual layout, palette and tooltips.

+

Toolbar Config

+

Allows customization of the toolbar including the addition of application specific buttons.

+

Notification Config

+

Allows customization of the notification panel which is displayed by clicking the notification icon on the toolbar.

+

Context Menu Config

+

Allows some minor customization of the options the context menus/toolbars display.

+

Keyboard Config

+

Allows customization of the shortcut keys supported by the flow editor.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.02.01-canvas-config/index.html b/v13/03.02.01-canvas-config/index.html new file mode 100644 index 0000000000..e03b975525 --- /dev/null +++ b/v13/03.02.01-canvas-config/index.html @@ -0,0 +1,4397 @@ + + + + + + + + + + + + + + + + + + + + + + + Canvas Config - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Canvas Config object

+

The canvas config object is optional. If it is not provided, or any of the properties within it are not provided, Common Canvas will use reasonable defaults. Here’s an example of a canvas config object: +

    const commonCanvasConfig = {
+       "enableNodeFormatType": "Vertical",
+       "enableLinkType": "Straight"
+       }
+    };
+
+and this is how it is specified to Common Canvas: +
    render() {
+        return (
+            <CommonCanvas
+                canvasController={this.canvasController}
+                config={commonCanvasConfig}
+            />
+        );
+    }
+

+

Nodes

+

enableNodeFormatType

+

This can be “Horizontal” or “Vertical”. “Horizontal” is the default. “Horizontal” will display a node with an image and the label to the right of the image. “Vertical” will display the node with the label underneath the image. See the node customization section for details on what this will do.

+

enableNodeLayout

+

This is a simple Javascript object, the properties of which override the default node layout properties. For more details see the node customization section.

+

enableResizableNodes

+

This is a boolean. The default is false. If set to true, the user can resize nodes by dragging the edges of the node to increase or decrease the width and/or height of the node. When hovering the mouse cursor over the edge of the node the user will see a sizing cursor to indicate the resize function is available. This option works best when the node has a background rectangle that shows the extent of the sizing area.

+ +

This is a boolean. The default is false. If set to true, the user is allowed to create a link that points to the node the link originated from: a loop-back link. This is created by the user by dragging from the output port on the node and dropping the new link onto the same node.

+

enableLinkDirection

+

This can be “LeftRight”, “RightLeft”, “TopBottom”, or “BottomTop”. “LeftRight” is the default. This alters the default location for input and output ports on the node, as follows:

+
    +
  • For “LeftRight” input ports will be on the left of the node and output ports will be on the right of the node
  • +
  • For “RightLeft” input ports will be on the right of the node and output ports will be on the left of the node
  • +
  • For “TopBottom” input ports will be on the top of the node and output ports will be on the bottom of the node
  • +
  • For “BottomTop” input ports will be on the bottom of the node and output ports will be on the top of the node
  • +
+

In addition to these positions, the port positions can be further customized using the nodeLayout options.

+ +

This is a boolean. The default is false. If set to true, the user can drag nodes from the palette or from the canvas and drop them onto existing links in the flow. This causes the dropped node to be inserted between the two nodes joined by the link, meaning new links are created that join the new node to the previously joined nodes and the old link is removed. When the user performs the drop Common Canvas will call the editActionHandler with one of two possible commands:

+
    +
  • “createNodeOnLink” - when a node is being dragged from the palette leading to node creation & insertion
  • +
  • “insertNodeIntoLink” - when an existing node is dragged from the canvas leading to insertion of the existing node into the link
  • +
+

enablePositionNodeOnRightFlyoutOpen

+

This can be a boolean or an object. The default is false. If set to true, when the right-side flyout is open the currently selected node (assuming there is one) will be automatically positioned in the center of the viewport (canvas area). Instead of true this field can also be set to a simple JavaScript object like this { x: 30, y: 40 } where x and y indicate the position where the node will be positioned as a percentage of the width and height of the viewport respectively.

+

enableHighlightNodeOnNewLinkDrag

+

This is a boolean. The default is false. If set to true Common Canvas will add the “data-new-link-over” attribute to the node’s group <g> element, when the end of a new link is dragged to be close to and over a target node. This allows applications to alter the appearance of the target node as a new link is dragged towards it.

+

enableHighlightUnavailableNodes

+

This is a boolean. The default is false. If set to true, when the user begins to drag a new link line, Common Canvas will add a new class called d3-node-unavailable to all nodes which cannot accept the link as input. The class will be applied to each node’s group <g> element in the DOM. This class can be used for styling the unavailable nodes as desired using CSS. The default styling will ‘gray out’ the node label, node outline rectangle (if there is one) and the node icon (provided it is an SVG image). These styles can be overridden in the applications CSS if different styling is needed. This behavior also applies if the end of a partially or fully detached link is dragged.

+

enableRaiseNodesToTopOnHover

+

This is a boolean. The default is true. If set to false the nodes will be left in their original place in the DOM. If set to true, when the user moves the mouse cursor over a node, that node will be moved in the DOM so that the node appears on top of all other nodes. This is only really noticeable if nodes, or parts of nodes, overlap other nodes. It can be useful if your nodes have protruding ports or decorations and your users sometimes position nodes very close to one another. Note: the ‘true’ setting can adversely affect the behavior of scroll areas in a node that are displayed using a React object (using the nodeExternalObject node layout option) because when the node is moved in the DOM the scroll area gets reset to its initial position. Set this to false if you are displaying nodes using React objects.

+

enableMoveNodesOnSupernodeResize

+

This is a boolean. The default is true. If true, nodes surrounding a supernode will be moved when the supernode is expanded or manually resized so that the supernode does not overlay them. When set to false, the nodes surrounding a supernode will stay in their current positions when the supernode is expanded or manually resized. This may result in the nodes being overlaid by the supernode.

+

enableDisplayFullLabelOnHover

+

This is a boolean. The default is false. If set to true, any abbreviated node label will be displayed in full when the pointer hovers over the label. If set to false, abbreviated node labels will remain the same when the pointer hovers over them.

+

enableSingleOutputPortDisplay

+

This is a boolean. The default is false. If set to true, only the last of the ports from the array of output ports will be displayed for each node. This config property is only applicable to applications with very specialized styling and handling of ports. If set to true with regular applications, it may result in a confusing display to the user. The single port is displayed at a position specified by outputPortRightPosX and outputPortRightPosY layout properties. For exmaple: +

config = {
+    enableSingleOutputPortDisplay: true,
+    enableNodeLayout: {
+      outputPortRightPosX: 0,
+      outputPortRightPosY: 20
+};
+

+ +

enableLinkType

+

This can be “Curve”, “Elbow”, “Parallax” or “Straight”. “Curve” is the default. This will set the link style used to connect nodes.

+

enableLinkMethod

+

This can be: “Ports” or “Freeform”. The default is “Ports”. When set to “Ports”, each link will be drawn from a specific output port on the source node to a specific input port on the target node. Note: With this option, links will be drawn to the port locations even if the port display has been switched off by the application. When set to “Freeform” links will be drawn from the source node to the taget node ignoring where ports have been positioned. Applications displaying freeform links usually choose not to display input and output ports.

+

enableStraightLinksAsFreeform

+

This is a boolean. The default is true. If set to true then, when enableLinkType is set to “Straight”, enableLinkMethod will always be treated as if it is set to “Freeform”. If it is set to false, Common Canvas will use whatever value is specified in enableLinkMethod when enableLinkType is set to “Straight”.

+

This config field was introduced to enforce the earlier Elyra Canvas behavior, where “Straight” links are automatically displayed as “Freeform”.

+
+

enableStraightLinksAsFreeform - deprecated

+

This field is deprecated and will be removed in the next major Elyra Canvas version so it is recommeneded that applications using “Straight” links explicitely set enableLinkMethod to “Freeform” now because that will be mandatory when migrating to the next verion.

+
+

enableLinkSelection

+

This can be: “None”, “LinkOnly”, “Handles” or “Detachable”. The default is “None”. These have the following affect on the canvas:

+
    +
  • “None” - no selection of links is possible however user can right click on a link to get a context menu.
  • +
  • “LinkOnly” - a link may be selected and added to the set of currently selected objects (nodes and/or comments).
  • +
  • “Handles” - This includes the “LinkOnly” function. In addition, when a link is selected a handle (either a circle or an image) is displayed at the start and end of the link. The link handle can be dragged to a new node/port position to rewire the flow.
  • +
  • +

    “Detachable” - This includes the “LinkOnly” and “Handles” function. In addition, this option enables detachable links for the canvas. This means a link can exist either:

    +
      +
    • between a source node and an arbitrary point on the canvas (semi-detached) OR
    • +
    • between an arbitrary point on the canvas and a target node (semi-detached) OR
    • +
    • between two arbitrary points on the canvas (detached)
    • +
    +

    Additionally, “Detachable” mode, allows:

    +
      +
    • semi-detached or fully-detached links to be stored in and retrieved from the pipeline flow document.
    • +
    • semi-detached or fully-detached links to be manipulated with link handles. The link handles can be used to drag the end of the link away from its connecting source or target nodes and onto the canvas. Or semi-detached or fully detached links can be reattached to nodes/ports.
    • +
    • a new detached link to be created by drawing out a new link from a node and dropping it onto the canvas.
    • +
    • palette and canvas nodes, when they are dragged, to be dropped onto the ends of detached links to automatically attach them to the node being dragged.
    • +
    +
  • +
+

enableLinkReplaceOnNewConnection

+

This is a boolean. The default is false. If set to true, the user can drag a new connection to a target node, and if the input port on the target node has a maximum cardinality of one AND there is currently a connection to that port, the existing connection will be removed and the new connection is created; essentially this gesture replaces the existing link with the new one. If set to false the new connection will not be completed and the existing link will remain in place.

+

When set to true and a link is replaced, Common Canvas will call the beforeEditActionHandler the editActionHandler callback functions, if either are provided by the host application, with a data object parameter with the editType field set to "linkNodesAndReplace".

+

enableAssocLinkCreation

+

This is a Boolean. The default is false. If set to true it changes the nature of links that are created between nodes as follows:

+
    +
  • The user is able to pull out a link from either port on the node and drag it to another node
  • +
  • When a link is completed an association link is created rather than the regular data flow link that is created when this field is set to true. Association links describe an association between pairs of nodes and do not indicate any kind of data flow between those nodes.
  • +
+

enableAssocLinkType

+

This can be “Straight” or “RightSideCurve”. The default it “Straight”. This field changes the way association links are drawn on the canvas.

+

enableLinksOverNodes

+

This is a boolean. The default value is false. If set to true links are placed above nodes in the canvas. Hover over nodes/links would still have links above nodes if enableLinksOverNodes is set to true. This is useful if the ports are positioned within the boundaries of the node and the link lines need to be displayed to those positions.

+

Comments

+

enableMarkdownInComments

+

This is a boolean. The default is false. When set to true the user may enter markdown syntax into comments on the canvas when in edit mode for the comment. When the editing ends, the comment is shown in presentation mode and the markdown syntax is converted to HTML which is displayed in the comment and is styled by CSS.

+

Toolbar

+

enableToolbarLayout

+

This can be: “Top” or “None”. The default is “Top”. “Top” displays a toolbar at the top of the canvas area. See the Toolbar Config docs for details on how to customize the toolbar. “None” stops the toolbar from appearing.

+

Tips

+

tipConfig

+

This is a simple JavaScript object that configures whether tips for palette items, nodes, ports, links, decorations or the state tag are enabled (value set to true) or disabled (value set to false). By default, all tips are enabled. The following would switch off tips for ports and links. +

       "tipConfig": {
+           "palette": true,
+           "nodes": true,
+           "ports": false,
+           "links": false,
+           "decorations": true,
+           "stateTag": true
+       }
+
+The tips displayed by the palette can be further refined. For example, this would prevent tips for palette categories from being displayed, but would still display tips for node templates in the categories: +
       "tipConfig": {
+           "palette": {
+              "categories": false,
+              "nodeTemplates": true
+           },
+           "nodes": true,
+           "ports": false,
+           "links": false,
+           "decorations": true,
+           "stateTag": true
+       }
+

+

Note: The default content of tips can be overwritten by implementing the Tip Handler callback.

+

Context Toolbar

+

enableContextToolbar

+

This is a boolean. The default is false. When set to true, Common Canvas will display a context toolbar instead of a context menu for performing actions on canvas objects. A context toolbar is a small toolbar that appears above nodes, links and comments as the mouse cursor is hovered over them. The toolbar shows icons for actions the user is most likely to want to perform on the object. An overflow icon is displayed which, when clicked, shows a menu of additional actions. A context toolbar for the canvas background can also be displayed by right-clicking on the background. Common Canvas will display default context toolbars for nodes, links comments and the canvas background however, the default actions can be customized by implementing the Context Menu Handler` callback.

+

Palette

+

enablePaletteLayout

+

This can be: “Modal” or “Flyout” or “None”. The default is “Flyout”. “Flyout” displays a panel on the left side of the canvas containing the palette icons and “Modal” shows the palette icons in a dialog window. “None” stops the palette from appearing.

+

enableAutoLinkOnlyFromSelNodes

+

This is a boolean. The default is false. When set to true the auto-add function (where double clicking a node in the palette automatically adds it to the canvas) will only link up nodes when a node is already selected on the canvas and then, only if the selected node can be linked to the node that was double clicked. If false, the auto-add function will make a best guess at which node the double-clicked node should be added to.

+

enablePaletteHeader

+

This is a JSX object that will displayed in an open, wide palette. It is positioned below the Seach bar and above the categories and nodes. The default is null, which means nothing will be displayed. This option can be used by the application to add application specific function into the palette, for example, a button could be added.

+

enableNarrowPalette

+

This is a boolean. true is the default. If true when the palette is closed the narrow palette will be shown. When false the palette completely closes.

+
+

paletteInitialState - deprecated

+

Deprecated – This option is deprecated and will be removed soon. Use CanvasController.openPalette() to display an opened palette at start-up. This openPalette() can be called immediately after creating the canvas controller.

+

paletteInitialState is a boolean. false is the default. If set to true the palette will be opened when Common Canvas first appears to its full (non-narrow) state.

+
+

Zoom (scale and pan)

+

enableSaveZoom

+

This can be: “None”, “LocalStorage” or “Pipelineflow”. The default is “None”.

+
    +
  • “None” - When the canvas is zoomed, the zoom (scale and x/y pan) are not saved anywhere so if the canvas is closed and reopened it reopens with the default zoom which is a scale of 1 and x/y pan values of 0.
  • +
  • “LocalStorage” - The zoom for the canvas is stored in the browser’s local storage and will be reapplied to the canvas each time that canvas is shown in that browser. This applies to sub-flows, when the user displays them full-screen, as well as the primary flow. Sub-flows and the primary flow each have their own zoom amounts stored in local storage. Note: Zoom amounts stored in local storage can be cleared from storage by calling the canvasController.clearSavedZoomValues() API method.
  • +
  • “Pipelineflow” - The zoom is serialized into the pipeline flow document and when a pipeline flow document is provided to Common Canvas through the API the zoom will be applied to the canvas display. Zoom amounts can be stored for both primary and sub pipelines. (See the pipelineFlow schema specification).
  • +
+

enablePanIntoViewOnOpen

+

This can be either true or false. The default is false. If set to true, the canvas will be panned so as much of the canvas area (the area containing the nodes and comments) is visible in the viewport as possible. This will only happen when enableSaveZoom === “None” or if there is no saved zoom available either in local storage (when enableSaveZoom === “LocalStorage”) or in the pipelineFlow (when enableSaveZoom === “Pipelineflow”).

+

enableZoomIntoSubFlows

+

This is a boolean. The default is false. When set to true, Common Canvas will override the maximum zoom extent value which, by default is used for the entire canvas, to allow the user to zoom in on in-place sub-flows further than they can do on containing flows. This means the user can zoom in on multi-nested sub-flows so they are easier to view. To see this effect, the user must position the mouse pointer over the sub-flow before performing the zoom gesture.

+

Canvas Content

+

enableFocusOnMount

+

This is a boolean. The default is true. When set to true, the keyboard focus will automatically be set on the flow editor when Common Canvas first appears. This means, keyboard shortcut operations on the flow editor, and it contents, will be executed without the user having to click on the flow editor to get focus to be moved there. If set to false, Common Canvas makes no change to the focus so the application can set the focus wherever it wants.

+

emptyCanvasContent

+

This is a JSX or HTML snippet that contains some text or any elements (such as an image) that you want to display when the canvas is empty, that is, when it doesn’t have any nodes or comments. The default behavior if this config parameter is not provided is that Common Canvas will display an image and message saying: “Your flow is empty!”.

+

dropZoneCanvasContent

+

This is a JSX or HTML snippet that contains some text or any elements (such as an image) that you want to display when a data object is dragged from the desktop over the canvas. The default behavior if this config parameter is not provided is that Common Canvas will display an image and a message saying: “Drop to add to canvas and project”. The content will not be displayed unless the enableDropZoneOnExternalDrag configuration parameter (see above) is set to true.

+

enableDropZoneOnExternalDrag

+

This is a boolean. The default is false. If set to true a graphic overlay will be displayed over the canvas when a data object icon is dragged from the desktop over the canvas. The default graphic overlay will be an image and a message saying: “Drop to add to canvas and project” unless the dropZoneCanvasContent configuration parameter is provided.

+

See the Dragging an object from the desktop section for details on how to handle the drop of an external object onto the canvas.

+

enableStateTag

+

This can be either “None”, “Locked” or “ReadOnly”. The default is “None”. When set to either “Locked” or “ReadOnly”, a ‘state tag’ object will be shown permanently over the top of the canvas. The state tag will be positioned in the center and towards the top of the canvas. The state tag consists of a black background rectangle with rounded corners overlaid with an icon and a text label. A tooltip is displayed when the mouse pointer is hovered over the state tag. The icon, label and default tooltip will be set appropriately based on the value (“Locked” or “ReadOnly”) for this setting. The host application can override the tooltip by implementing the Tip Handler callback.

+

Canvas Operation

+

enableKeyboardNavigation

+

This can be either true or false. The default is false. If set to true, the user can use the keyboard to move the keyboard focus around the Common Canvas interface and perform actions on the flow objects using shortcut keys.

+

enableInteractionType

+

This can be “Mouse”, “Carbon” or “Trackpad”. The default is “Mouse”. +“Trackpad” has been deprecated and will be removed in the future.

+
    +
  • +

    With this set to “Mouse” the following interaction is enabled:

    +
      +
    • Zoom canvas = Rotate mouse wheel. +(Can be simulated with a trackpad with two finger up and down scroll)
    • +
    • Pan canvas = Left mouse key down on canvas background + drag. +(Can be simulated with a trackpad with press down on trackpad and drag.)
    • +
    • Region select on canvas = Shift key + left mouse key down on canvas background + drag. +(Can be simulated with a trackpad with Shift + finger down on trackpad + drag across canvas background)
    • +
    +
  • +
  • +

    With this set to “Carbon” the following interaction is enabled:

    +
      +
    • Zoom canvas = Rotate mouse wheel. +(Can be simulated with a trackpad with two finger up and down scroll)
    • +
    • Pan canvas = Press and hold space bar then left mouse key down on canvas background + drag. +(Can be simulated with a trackpad by press and hold space bar then press down on trackpad and drag.)
    • +
    • Region select on canvas = Left mouse key down on canvas background + drag. +(Can be simulated with a trackpad with press down on trackpad and drag.)
    • +
    +
  • +
  • +

    This setting is now deprecated With this set to “Trackpad” the following interactions are enabled:

    +
      +
    • Zoom canvas = Two finger pinch or two finger spread gesture. +(Can be simulated with a mouse as follows: Ctrl + rotate mouse wheel.)
    • +
    • Pan canvas = Two finger horizontal or vertical scroll gesture. +(Can be simulated with a mouse as follows: Vertical pan is rotate mouse wheel; Horizontal pan is Shift + rotate mouse wheel)
    • +
    • Region select on canvas = Finger down + drag. +(Can be simulated with a mouse as follows: Left button down + drag on canvas background)
    • +
    +
  • +
+

enableParentClass

+

This is a string which is a class name. The default is empty string. If a class name is provided it is applied to the top-most DOM element for Common Canvas. This can be used to make you CSS override rules more specific which means they will be used in preference to any default styles. For example, if you specify “my-app-styles” for this field then CSS like this: +

    .my-app-styles .d3-node-body-outline {
+        fill: orange;
+    }
+
+will override the style from the common-canvas CSS +
    .d3-node-body-outline {
+        fill: white;
+    }
+

+

enableDragWithoutSelect

+

This can be either true or false. The default is false. If set to true, the user can drag and drop a single node or a single comment without that gesture removing selection on any other nodes or comments. If the node being dragged was selected prior to the drag gesture then it and any other objects that are currently selected will be moved. With this parameter set to false (or missing) a drag and drop gesture will select the node or comment being dragged and will deselect any currently selected objects.

+

enableEditingActions

+

This is a boolean. The default is true. If set to false, various editing actions on the canvas will be prevented, as follows:

+
    +
  1. +

    Nodes cannot be created on the canvas using any of the following:

    +

    (a) dragging and dropping a node template from the palette onto the canvas or

    +

    (b) dragging and dropping an object onto the canvas from outside the canvas area, such as a file being dragged from the computer desktop onto the canvas (see note below), or

    +

    (c) double clicking a node in the palette to create a node

    +

    Note: It is not possible for Common Canvas to prevent an object being dragged from the computer desktop to the canvas so it is recommended the drop zone (which provides visual feedback about the drop) should be switched off by setting the enableDropZoneOnExternalDrag config field to false.

    +
  2. +
  3. +

    Nodes and comments cannot be dragged and moved.

    +
  4. +
  5. The end points of Links, when enableLinkSelection is set to "Detachable" or "Handles", cannot be dragged and moved.
  6. +
  7. Links from nodes and comments cannot be created by dragging from the port object on the source node or comment to the target node.
  8. +
  9. Comments, node labels and text decorations cannot be edited, neither by clicking the edit icon (which does not appear) nor by double clicking the text.
  10. +
  11. +

    Context menu options that alter the canvas objects will be removed from the context menu before it is displayed by Common Canvas. The options that will be removed are:

    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Option TextAction Identifier
    New CommentcreateComment
    DisconnectdisconnectNode
    Edit->Cutcut
    Edit->Copycopy
    Edit-Pastepaste
    Undoundo
    Redoredo
    DeletedeleteSelectedObjects
    Create supernodecreateSuperNode
    Create external supernodecreateSuperNodeExternal
    Deconstruct supernodedeconstructSuperNode
    Collapse supernodecollapseSuperNodeInPlace
    Expand supernodeexpandSuperNodeInPlace
    Convert external to localconvertSuperNodeExternalToLocal
    Convert local to externalconvertSupernodeLocalToExternal
    DeletedeleteLink
    Save to palettesaveToPalette
    +

    If your application adds its own editing actions to the context menu your code must remove them if they are not needed in some situations (e.g. you are displaying a read-only canvas).

    +
  12. +
  13. +

    Any default toolbar actions (tools) that alter the canvas objects will be disabled regardless of their specified enablement status. These actions are:

    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Default TooltipAction
    Undoundo
    Redoredo
    Cutcut
    Copycopy
    Pastepaste
    DeletedeleteSelectedObjects
    New commentcreateAutoComment
    Arrange HorizontallyarrangeHorizontally
    Arrange VerticallyarrangeVertically
    +
  14. +
  15. +

    Keyboard shortcuts that alter the canvas will not work. These are:

    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    ShortcutAction
    deletedelete
    Ctrl/Cmd + xcut
    Ctrl/Cmd + ccopy
    Ctrl/Cmd + vpaste
    Ctrl/Cmd + zundo
    Ctrl/Cmd + Shift + zredo
    Ctrl/Cmd + yredo
    +
  16. +
  17. +

    The browser’s Edit menu options (cut, copy, paste) will not work with the canvas objects regardless of the setting for enableBrowserEditMenu.

    +
  18. +
+

enableImageDisplay

+

This can be set to: “SVGInline”, “LoadSVGToDefs” or “SVGAsImage”. The default is “SVGInline”. This field controls the display of SVG image files (that is, files with a .svg extension) on the canvas, such as those displayed for node icons or decoration images. This option can be useful to improve performance when images are repeated a large number of times on the canvas and particularly when the browser cache is disabled. Note: this does not affect the display behavior of non-SVG files which are always displayed inside an <image> tag. The behavior for SVG image files is as follows:

+
    +
  • “SVGInline” - This is the default. With this setting, the image file is read in – from the server or cache (if available) – and the SVG tags within the file displayed in-line in the DOM. This means elements within the SVG can be customized using CSS on a node-by-node basis.
  • +
  • “LoadSVGToDefs” - With this option, each unique SVG file is read from the server (or cache) and written into a <symbol> element within the <defs> element of the canvas SVG area. A <use> element is then written to the place in the DOM for each place where the image should be displayed. Using this option means the SVG file for each image is only read once which should improve performance if images are repeated a lot on the canvas. However, it does limit the customization possibilities for the images on a node-by-node basis. Customization colors can be passed into the images using CSS inheritance, or the currentColor keyword, provided there are no internal classes applied to the elements in the SVG.
  • +
  • “SVGAsImage” - This option causes the SVG file to be displayed within an <image> element in the DOM. The file loading is handled internally by the browser. Again, with this option customization capabilities on a node-by-node basis are limited. Customization colors can be passed into the images using CSS inheritance, or the currentColor keyword, provided there are no internal classes applied to the elements in the SVG.
  • +
+

enableInternalObjectModel

+

This is a boolean. The default is true. It is recommended you leave this set to true. If you set this to false your code will be responsible for handling the object model, which is NOT recommended. When set to false, changes are not automatically saved into the common-canvas object model and are not reflected in the canvas display. Consequently, the application is expected to listen to events and update the common-canvas object model using the canvas-controller API if it want visual changes to the canvas.

+

enableRightFlyoutUnderToolbar

+

This is a boolean. The default is false. If set to true the right flyout panel, when opened, will appear below the toolbar and will not cause the toolbar to compress. The default behavior is that the right flyout panel, when opened, will appear at the side of the toolbar and will compress the space available for the toolbar to be displayed. Warning: the notifications panel which is tied to the notifications icon in the toolbar will appear over the top of the right-side flyout with this option set to true.

+

enableRightFlyoutDragToResize

+

This is a boolean. The default is false. If set to true, the right flyout panel can be resized by dragging its left border. When hovering over the left border of the flyout, the cursor will change to indicate that resizing is possible. Users can drag the border to adjust the width of the flyout, allowing it to expand or collapse. This functionality offers more flexible layout options for the user. If set to false, the right-flyout panel, when open, will be displayed with a default width and cannot be resized by dragging its edge. The application can add its own sizing function if required.

+

enableLeftFlyoutUnderToolbar

+

This is a boolean. The default is false. If set to true the left flyout panel, when opened, will appear below the toolbar and will not cause the toolbar to compress. The default behavior is that the left flyout panel, when opened, will appear at the side of the toolbar and will compress the space available for the toolbar to be displayed.

+

enableExternalPipelineFlows

+

This is a boolean. The default is false. If true, the context menu will include a Create External Supernode option when a set of objects are selected from which a supernode can be created.

+

Waring: The host application must implement some of the common-canvas callbacks for external pipeline flow support to work correctly. See the section on External Pipeline Flow support for more details.

+

enableSnapToGridType

+

This can be: “None”, “During” or “After”. The default is “None”.

+
    +
  • “None” - there is no snap to grid and objects can be moved to any position on the canvas.
  • +
  • “During” - when nodes or comments are moved or sized, the objects snap to an imaginary grid while the objects are being dragged or sized. This gives a somewhat jerky effect as the move or size is happening but has the advantage of telling the user exactly where the object will be when they release the mouse button to end the action.
  • +
  • “After” - nodes or comments snap to a grid when the drag or size event ends. This gives a smooth dragging and sizing effect but the user does not see the final position until they release the mouse button at the end of the action. By default the canvas uses reasonable values for the grid increments.
  • +
+

enableSnapToGridX

+

This optional value overrides the default horizontal increment of the snap-to-grid grid. It can be either a numeric value which is a number of pixels or it can be a numeric value followed by a % sign (e.g. “25%”) which indicates the grid will be a percentage of the default node width. Its default is dependent on whatever is set for enableNodeFormatType. That is for “Horizontal” it will be “20%” and for “Vertical” it will be “25%”.

+

enableSnapToGridY

+

This optional value overrides the default vertical increment of the snap-to-grid grid. It can be either a numeric value which is a number of pixels or it can be a numeric value followed by a % sign (e.g. “25%”) which indicates the grid will be a percentage of the default node height. Its default is dependent on whatever is set for enableNodeFormatType. That is for “Horizontal” it will be “33.33%” and for “Vertical” it will be “20%”.

+

enableAutoLayoutVerticalSpacing

+

This is the spacing in pixels which is used to separate nodes vertically when either the vertical or horizontal auto layout action is used.

+

enableAutoLayoutHorizontalSpacing

+

This is the spacing in pixels which is used to separate nodes horizontally when either the vertical or horizontal auto layout action is used. For horizontal auto layout, Common Canvas may override this value if it decides that more space is needed to prevent connecting lines from doubling back on themselves.

+

enableBrowserEditMenu

+

This is a boolean. true is the default. If true, the Cut/Copy/Paste items in the Browser’s Edit menu, including keyboard input for those actions, can be used for performing those actions on objects (e.g. Nodes) in the canvas. When false, those items in the Browser’s edit menu, including keyboard input for those actions, will not work for objects in the canvas. This will not prevent those actions working in the canvas when, say, invoked with the toolbar or canvas context menus, but this property can be used if the keyboard input for those actions into the canvas is disabled for Common Canvas using the Keyboard Config object.

+

schemaValidation

+

This is a boolean. false is the default. It tells Common Canvas whether you want pipleineFlow and palette objects to be validated against the schema files when they are submitted to the canvas controller, using the setPipelineFlow(pFlow) or setPipelineFlowPalette(palette) methods. If any validation errors are found messages are displayed in the browser console. It is recommended this option be set to true during development and testing but switched off in production since schema validation can be somewhat slow for large objects.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.02.02-toolbar-config/index.html b/v13/03.02.02-toolbar-config/index.html new file mode 100644 index 0000000000..e2670006e1 --- /dev/null +++ b/v13/03.02.02-toolbar-config/index.html @@ -0,0 +1,3071 @@ + + + + + + + + + + + + + + + + + + + + + + + Toolbar Config - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+ +
+ + + +
+
+ + + + + + + + + + + + +

Toolbar Config object

+

The canvas toolbar config object is optional. If it is not provided Common Canvas will display a reasonable default toolbar.

+

+

If the toolbar config is provided, it configures which action buttons and dividers are shown in the toolbar. A toolbar will be displayed for Common Canvas if the canvas configuration field enableToolbarLayout is set to “Top” (which is the default).

+

The toolbar config object specifies actions for the left and right side of the toolbar. The application can also optionally tell the toolbar not to handle the enable/disable state for the standard toolbar buttons using overrideAutoEnableDisable.

+

When the user clicks one of the action buttons the action will be performed either by Common Canvas or by the application, if it is an applicaiton specific action. In both cases, Common Canvas will call the beforeEditActionHandler and editActionHandler callbacks.

+

The toolbar supports a number of keyboard shortcuts for accessibility

+

The leftBar and rightBar fields contain an array of action objects: one element for each toolbar item. Here is an example, toolbar configuration object:

+

    import { Bookmark } from "@carbon/react/icons";
+    ...
+    ...
+    const toolbarConfig = {
+        leftBar: [
+            { action: "undo", label: "Undo" },
+            { action: "redo", label: "Redo" },
+            { divider: true },
+            { action: "cut", label: "Cut" },
+            { action: "copy", label: "Copy" },
+            { action: "paste", label: "Paste" },
+            { divider: true },
+            { action: "createAutoComment", label: "Add Comment", enable: true },
+            { action: "deleteSelectedObjects", label: "Delete" },
+            { action: "arrangeHorizontally", label: "Arrange Horizontally", enable: true }
+            { divider: true },
+            { action: "myBookmrk", label: "Add bookmark", enable: true, iconEnabled: (<Bookmark size={32}/>) }
+        ],
+        rightBar: [
+            { action: "stop", label: "Stop Execution", enable: false },
+            { divider: true },
+            { action: "run", label: "Run Pipeline", enable: false }
+        ],
+        overrideAutoEnableDisable: false
+    };
+    ...
+    ...
+    return <CommonCanvas toolbarConfig={toolbarConfig} canvasController={canvasController}/>
+
+Where:

+
    +
  • +

    leftBar - an array of action items to specify what is displayed on the left side of the toolbar.

    +
  • +
  • +

    rightBar - an array of action items to specify what is displayed on the right side of the toolbar. If this is omitted, Common Canvas will automatically populate the right side of the toolbar with zoom-in, zoom-out, and zoom-to-fit actions. To suppress these right side actions, specify the rightBar field as an empty array or an array containing the actions required on the right.

    +
  • +
  • +

    overrideAutoEnableDisable - a boolean. The default is false. By default Common Canvas has an auto-enablement feature that controls the enablement of common tools in the toolbar based on user actions (e.g enable the Delete icon when items are selected). If overrideAutoEnableDisable set to true it will switch off the auto-enablement feature. This is useful if the host application wants to disable all the nodes under certain circumstances. If set to true, the enable property in the action items for each tool is used to decide whether to display the icon as enabled or disabled. If set to false or omitted, Common Canvas will handle the auto-enablement of common actions. (See the actionsection below for more details.)

    +
  • +
+

The toolbar will display the objects in the same order they are defined in the leftBar and rightBar arrays.

+

Toolbar action object definition

+

Here is an example of an action object which must contain a unique action field as a minimum. +

    {
+        action: "run",
+        label: "Run",
+        enable: true,
+        iconEnabled: "/image/myOwnEnabledIcon.svg",
+        iconDisabled: "/image/myOwnDisabledIcon.svg",
+        incLabelWithIcon: "before",
+        kind: "primary",
+        tooltip: "Run the flow",
+        isSelected: false
+    }
+

+
    +
  • +

    action - a unique identifier string which is the name of the action to be performed. This action name will set in the editType field of the data parameter passed into the beforeEditActionHandler and editActionHandler callback methods. The application can use these callbacks to detect what action the user clicked in the toolbar.

    +

    The toolbar supports two types of action:

    +
      +
    • +

      Internal actions - These are requested by the application by specifying one of the reserved internal actions names. These actions are performed internally by Common Canvas. A default icon will be provided by Common Canvas for the most popular actions. For example, if the deleteSelectedObjects action is specified a trash can icon will be displayed in the toolbar and any selected objects will be deleted from the flow editor when the action is clicked. The application can override the default icon by specifying the iconEnabled field.

      +

      Disablement of these built in actions is also handled by Common Canvas: “undo”, “redo”, “cut”, “copy”, “paste”, “deleteSelectedObjects”, “createAutoComment”, “arrangeHorizontally” “arrangeVertically”, “zoomIn”, “zoomOut”, “zoomToFit”.

      +

      For example, when no canvas objects are selected the deleteSelectedObjects action (trash can icon) will be automatically disabled.

      +

      The application can switch off this automatic enable/disable function by setting overrideAutoEnableDisable field in the toolbar config to true. When overrideAutoEnableDisable set to true, the enable field in each action object will set the enablement appearance of the item.

      +
    • +
    • +

      External actions - These are specified by the application and must be handled by the application in the beforeEditActionHandler and editActionHandler callbacks. An icon must be specified for these actions. The application can use icons imported from the Carbon Icon library. For example, in the sample code above myBookmrk has been specified as a external action with the Carbon Bookmark icon specified. Common Canvas will ignore this action when the button is clicked and pass it through in the data.editType field to the editActionHandler callback.

      +
    • +
    +
  • +
  • +

    label - a short string or a JSX object that describes the action. This is used as the tooltip text or is (optionally) displayed next to the icon if incLabelWithIcon is specified.

    +
  • +
  • +

    enable - A boolean. The action button is clickable when set to true. If false, the button will be disabled. If not set, it will default to disabled (false). If overrideAutoEnableDisable is set to false, or omitted, this field is ignored for the standard action items (like cut, copy, paste) because Common Canvas handles their enable/disable appearance. If overrideAutoEnableDisable is set to true, this field will be used for standard action items.

    +

    +

    +
  • +
  • +

    iconEnabled - specifies the icon to display when enable is true. Common Canvas will provide icons for some of the reserved internal actions so the application doesn’t need to specify iconEnabled or iconDisabled for them:

    +

    It can be either:

    +
      +
    • a string containing the path to a custom SVG file to display or
    • +
    • a JSX expression, for example (<Edit32 />) where Edit32 is an icon imported from carbon icons. It is recommended to only pass very simple JSX expressions.
    • +
    +
  • +
  • +

    iconDisabled - specifies the icon to display when enable is false. If iconDisabled is not specified iconEnabled will be used instead. It can be omitted for any of the standard actions (see iconEnabled above).

    +

    It can be either:

    +
      +
    • a string containing the path to a custom SVG file to display or
    • +
    • a JSX expression, for example (<Edit32 />) where Edit32 is an icon imported from carbon icons. It is recommended to only pass very simple JSX expressions.
    • +
    +
  • +
  • +

    incLabelWithIcon - can be set to “no”, “before” or “after”. The default is “no”. This field specifies whether the label should be displayed in the toolbar with the icon and if so, where it is displayed with respect to the icon.

    +

    +

    +
  • +
  • +

    kind - can be set to “default”, “primary”, “danger”, “secondary”, “tertiary” or “ghost”. The default it “default”. These give the action the same styling as the equivalent kind’s of buttons in the carbon library.

    +

    +
  • +
  • +

    tooltip - A string or JSX object. The tooltip that will be displayed for the action. If this is not provided the label will be displayed as the tooltip instead.

    +
  • +
  • +

    isSelected - A boolean. When set to true the toolbar button displays a selection highlight. This is displayed as either a blue bar along the bottom border of buttons in the toolbar

    +

    +

    +

    or a checkmark for items that appear in sub-menus.

    +

    +

    When applied This is useful for implementing a button that switches on and off a mode, as opposed to a regular button which does not have any state.

    +

    This property can also be used to indicate a current state between a number of mutually exclusive settings. In this case, one button would be added to the toolbar for each setting and then the isSelected property would be set to true for the setting that is currently active. Then, when the user clicks a different option in the set, the application code would set isSelected to true for that button and set it to false for the previously selected button. (This would give behavior like a radio button set.)

    +
  • +
  • +

    className - This a string that will be appended to the class field for the top-level <div> that surrounds the action button. This can be used to set application-specific CSS for the button.

    +
  • +
  • +

    textContent - This is an additional string that will be displayed on top of the action button. It can be used to specify adittional information to enahance the icon. For example, a count of messages could be displayed over a icon that, when clicked, shows a messages panel.

    +
  • +
+

Automatic toolbar items

+

Two toolbar items are automatically added to the toolbar:

+
    +
  • A palette action which is used for opening and closing the palette. This is added to the left side of the toolbar if the enablePaletteLayout field is set to either “Flyout” (the default) or “Modal” in the canvas configuration.
  • +
  • A notification panel action which is used to open and close the notifications panel. This will be added to the right side of the toolbar if a notification configuration object is specified to the <CommonCanvas> react object.
  • +
+

The position and a subset of fields for these items can be customized by providing an appropriate object in either the leftBar or rightBar arrays. The action field for the items should be either “togglePalette” or “toggleNotificationPanel”. Any of the following fields can be optionally provided to override the default values: label, iconEnabled, incLabelWithIcon and tooltip.

+

For example, if the application provides the following in, say, the leftBar array: +

   import { AddAlt, SubtractAlt, Notification } from "@carbon/react/icons";
+   ...
+
+   const toolbarConfig = {
+       leftBar: [
+           ...
+           { action: "togglePalette",
+             label: this.canvasController.isPaletteOpen() ? "Close Palette" : "Open Palette",
+              iconEnabled: this.canvasController.isPaletteOpen() ? (<SubtractAlt />) : (<AddAlt />),
+              incLabelWithIcon: "after" },
+           { divider: true },
+           { action: "toggleNotificationPanel", iconEnabled: (<Notification />) },
+           ...
+       ]
+   };
+

+

Will look like this:

+

+

Sub-area properties

+

The toolbar button can show a ‘sub-area’ below the button when the button is clicked. This can be either a sub-menu which is a list of text options or a sub-panel which is a small window that can show whatever the application wants such as settings or messages.

+
    +
  • +

    subMenu - Specify either this or the subPanel field. This is an array of action objects (the same as those specified for the toolbar) which will displayed as a menu of options that appears under the icon on the toolbar.

    +

    This code:

    +

    import { TextScale } from "@carbon/react/icons";
    +...
    +const subMenuTextSize = [
    +    { action: "title", label: "Title", enable: true },
    +    { action: "header", label: "Header", enable: true },
    +    { action: "subheader", label: "Subheader", enable: true },
    +    { action: "body", label: "Body", enable: true }
    +];
    +...
    +const toolbarConfig = {
    +  leftBar: [
    +    ...
    +    { action: "text-size-submenu", incLabelWithIcon: "after", iconEnabled: (<TextScale size={32} />),
    +          label: "Text Size", enable: true, subMenu: subMenuTextSize, closeSubAreaOnClick: true
    +    }
    +    ...
    +  ]
    +}
    +
    +will display this:

    +

    +

    when the user clicks the options in the menu the associated action will be executed by calling the beforeEditActionHandler and editActionHandler callbacks.

    +
  • +
  • +

    subPanel - Specify either this or the subMenu field. This is a React object that will render the contents of a panel that appears under the action icon.

    +

    This code:

    +

    import { Settings } from "@carbon/react/icons";
    +...
    +const toolbarConfig = {
    +  leftBar: [
    +    ...
    +      { action: "settingspanel", iconEnabled: (<Settings />), label: "Settings", enable: true,
    +        subPanel: AppSettingsPanel,
    +        subPanelData: { saveData: (settings) => window.alert("Settings: " + settings) }
    +      }
    +    ...
    +  ]
    +}
    +
    +where this is the AppSettingsPanel class

    +

    will display this:

    +

    +
  • +
  • +

    subPanelData - This is a JavaScript object that will be passed as one of the props into the React object specified in the subPanel field. The application can use this to pass into the subPanel class any data or callback functions needed for the panel to do its job.

    +
  • +
  • +

    purpose - can be set to “single” or “dual”. The default is “single”. If set to “dual” the button will have two parts. The icon will be displayed to the left and to the right will be an up/down chevron icon. If the user clicks the icon on the left the action specified in the action field will be executed. If the user clicks the chevron a panel will be displayed below the chevron with the contents that are specified in the object specified in the subPanel field.

    +

    This code:

    +
    const toolbarConfig = {
    +  leftBar: [
    +    ...
    +      { action: "undo", label: "Undo", enable: true, purpose: "dual",
    +          subPanel: MultiCommandPanel, subPanelData: {} }
    +    ...
    +  ]
    +}
    +
    +

    where this is the MultiCommandPanel class

    +

    will display this:

    +

    +
  • +
  • +

    closeSubAreaOnClick - This is a boolean. The default is false. If set to true the sub-area will be close when sany element in it is clicked. This behavior might be useful if the sub-area displays a menu of options where only a single click on one of them is required.

    +
  • +
+

Toolbar divider object definition

+

Dividers can be added to separate groups of actions from other actions. This is displayed as a thin gray line. The divider object has one attribute: +

    {
+       divider: true
+    }
+

+
    +
  • divider - To show a divider in the toolbar, add an object with divider attribute set to true.
  • +
+

Advanced: JSX actions

+

Regular toolbar buttons, explained above, are displayed as set of Carbon Buttons. If the application needs to display something not wrappered in a button, you can provide your own JSX to display in the toolbar. Be aware however that, because of the way the toolbar is designed, there are restrictions on what the toolbar can do to display your JSX. For example, it cannot display anything with a height greater than the toolbar height.

+

If you provide your own JSX object it is displayed in a simple div in the toolbar. Some attributes are applied to the div to allow the action to work correctly within the toolbar - these cannot be changed. You are responsible for styling your JSX object to get it to appear the way you want.

+

Also be aware that, if the width of the toolbar reduces (maybe by the user sizing the page) your action may get moved into the overflow menu. It is also your responsibility to style the button so it appears as you want in the overflow menu as well as in the toolbar.

+

The JSX can be provided in the jsx field. Here is an example. The only other fields that are recognized with the jsx field are action and tooltip, all other fields will be ignored.

+
    {
+       action: "custom-loading",
+       jsx: (<div style={{ padding: "0 11px" }}><InlineLoading
+                status="active" description="Loading..." /></div>),
+       tooltip: "Loading the thing you wanted."
+    }
+
+
    +
  • +

    action - a unique identifier and the name of the action to be performed.

    +
  • +
  • +

    jsx - A JSX object or a funciton returning a JSX object. This will be displayed as the action in the toolbar.

    +

    To make the JSX object accessible inside the toolbar this field should be provided as a function. The function will pass a tabIndex field which should be set into the tabIndex property of the display object being created (see expample below). Additionally, the display object should have the `className set to “toolbar-jsx-obj” (see example below).

    +
  • +
  • +

    tooltip - A string or JSX object. This will be displayed as the tooltip for the action in the toolbar. If tooltip is omitted no tooltip will be added to your action. If tooltip is specified the jsx will be inside a tooltip div which is in the toolbar div mentioned above.

    +
  • +
+

For example, the following code that includes some Carbon React components: +

toolbarConfig = {
+    leftBar: [
+        {
+            action: "custom-loading",
+            tooltip: "A custom loading!",
+            jsx: (tabIndex) => (
+                <div style={{ padding: "4px 11px" }}>
+                    <InlineLoading status="active" description="Loading..."
+                        className={"toolbar-jsx-obj"}
+                        tabIndex={tabIndex}
+                    />
+                </div>
+            )
+        },
+        { divider: true },
+        {
+            action: "custom-dropdown",
+            tooltip: () => (this.suppressTooltip ? null : "A drop down using the overflow menu!"),
+            jsx: (tabIndex) => (
+                <div className="toolbar-custom-button">
+                    <OverflowMenu
+                        id={"ovf1"}
+                        renderIcon={TextScale32}
+                        iconDescription={""}
+                        onOpen={() => (
+                            this.suppressTooltip = true)
+                        }
+                        onClose={() => {
+                            this.suppressTooltip = false;
+                            window.alert("Option selected");
+                        }}
+                        className={"toolbar-jsx-obj"}
+                        tabIndex={tabIndex}
+                    >
+                        <OverflowMenuItem itemText="Big" />
+                        <OverflowMenuItem itemText="Medium" />
+                        <OverflowMenuItem itemText="Little" />
+                  </OverflowMenu>
+                </div>
+            )
+        }
+    ]
+};
+

+

will display this:

+

+
+

Deprecated toolbar config

+

The old toolbar configuration is still supported for now (but is deprecated). This allows the config to be provided as an array that defines just the left side of the toolbar. The right side will always show the zoom actions (zoomIn, zoomOut, zoomToFit) and a notifications panel icon (if a notification configuration object is provided in the <CommonCanvas> React object). These right side actions will always show on the right-hand side of the toolbar and are handled internally by the canvas. The entries in the array follow the same definition as described above. Note: there is no need to provide a palette action in the array because a palette icon and following divider will automatically be added to the toolbar when a palette is specified for the canvas.

+

An example of the toolbar config array should look like this: +

const toolbarConfig = [
+   { action: "stop", label: "Stop Execution", enable: false },
+   { action: "run", label: "Run Pipeline", enable: false },
+   { action: "undo", label: "Undo", enable: true },
+   { action: "redo", label: "Redo", enable: true },
+   { action: "cut", label: "Cut", enable: false },
+   { action: "copy", label: "Copy", enable: false },
+   { action: "paste", label: "Paste", enable: false },
+   { action: "createAutoComment", label: "Add Comment", enable: true },
+   { action: "deleteSelectedObjects", label: "Delete", enable: true },
+   { action: "arrangeHorizontally", label: "Arrange Horizontally", enable: true }
+];
+

+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.02.03-notification-config/index.html b/v13/03.02.03-notification-config/index.html new file mode 100644 index 0000000000..f735092dad --- /dev/null +++ b/v13/03.02.03-notification-config/index.html @@ -0,0 +1,2653 @@ + + + + + + + + + + + + + + + + + + + + + + + Notification Config - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Notification Config object

+

The Notification Config object specifies the appearance of the notifications panel. If a Notification Config is passed to Common Canvas a notifications button will be shown on the far right of the canvas toolbar. When clicked, the button opens the Notificaiton Panel.

+

For information about the structure of notification messages, refer to the Notification Messages page. Messages can be added and removed from the notification panel by calling the notification methods in the canvas-controller..

+

The Notification Config object looks like this: +

    const notificationConfig = {
+       action: "notification",
+       label: "Notifications",
+       enable: true,
+       notificationHeader: "Notification Messages",
+       notificationSubtitle: "subtitle",
+       emptyMessage: "You don't have any notifications right now.",
+       clearAllMessage: "Clear all",
+       keepOpen: true,
+       clearAllCallback: () => { console.log("Clear All clicked"); }
+    };
+

+
    +
  • +

    action - “notification” enables the notifications button to appear in the far right of the toolbar.

    +
  • +
  • +

    label - the Tooltip label to display for the notifications button in the toolbar.

    +
  • +
  • +

    enable - Toolbar button will have hover effect and is clickable when set to true. If false, the button will be disabled and unclickable. If not set, it will default to disabled (enable: false)

    +
  • +
  • +

    notificationHeader - String to display in the notification panel header. If not set, it will default to “Notifications”.

    +
  • +
  • +

    notificationSubtitle - String to be displayed as a sub-title in the panel header. If not set, panel header will be sized to only contain the notificationHeader string.

    +
  • +
  • +

    emptyMessage - String to be displayed when there are no notification messages to display.

    +
  • +
  • +

    clearAllMessage - String to be displayed on a button displayed at the bottom of the panel. The button can be clicked to clear all the messages from the panel. If omitted the button, and the footer area of the panel it appears in, will not be displayed.

    +
  • +
  • +

    keepOpen - A boolean which indicates when the panel will close. The default is false. If set to false, the panel will close when the user clicks on the page somewhere outside the panel. If set to true the panel will remain open when the user clicks somewhere on the page outside of the panel. With the option the user must click the x icon in the top right of the panel, or click the notification toolbar icon, to close the panel.

    +
  • +
  • +

    clearAllCallback - An optional callback function that will be called every time the “clear all” button is clicked.

    +
  • +
  • +

    secondaryButtonLabel - Label for the optional secondary button to be displayed in the notification panel. Both secondaryButtonLabel and secondaryButtonCallback must be specified for the button to appear.

    +
  • +
  • secondaryButtonCallback - A callback function that will be called when the secondary button is clicked.
  • +
  • secondaryButtonDisabled - Specify whether the secondary button is disabled or not.
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.02.04-context-menu-config/index.html b/v13/03.02.04-context-menu-config/index.html new file mode 100644 index 0000000000..439fe99128 --- /dev/null +++ b/v13/03.02.04-context-menu-config/index.html @@ -0,0 +1,2630 @@ + + + + + + + + + + + + + + + + + + + + + + + Context Menu Config - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Context Menu Config object

+

The context menu config object configures whether certain actions are available in the default context menu.

+
    const contextMenuConfig = {
+       enableCreateSupernodeNonContiguous: false,
+       defaultMenuEntries: {
+          saveToPalette: true,
+          createSupernode: false,
+          displaySupernodeFullPage: true,
+          colorBackground: false
+       }
+    };
+
+
    +
  • +

    enableCreateSupernodeNonContiguous - Allows the creation of supernodes from non-contiguous nodes. When set to true, the “Create supernode” menu item will be added to the default context menu when the currently selected nodes are contiguous or non-contiguous. When set to false, the “Create supernode” menu item is only added to the default context menu when the selected nodes are contiguous. The default value is false.

    +
  • +
  • +

    defaultMenuEntries - Controls what entries are generated in the default context menu generated by Common Canvas. This has the following properties:

    +
      +
    • ‘saveToPalette’ - This is a boolean. The default is false. If set to true, Common Canvas will add a ‘Save to palette’ option to the default node context menu.
    • +
    • ‘createSupernode’ - This is a boolean. The default is true. If set to false, Common Canvas will not show the ‘Create Supernode’ option in the default context menu for nodes.
    • +
    • ‘displaySupernodeFullPage’ - This is a boolean. The default is true. If set to false, Common Canvas will not show the ‘Display full page’ option in the default context menu for supernodes. When true, the option will be displayed. Clicking that option navigates the user to the full page view of the supernode’s pipeline as if the user had clicked the expansion icon of the expanded in-place supernode view.
    • +
    • ‘colorBackground’ - This is a boolean. The default is true. If set to false, Common Canvas will not show the ‘Color background’ option in the default context menu for comments. When true, the option will be displayed.
    • +
    +
  • +
+

[Note: If any host app wants more control over the default context menu here, please open an issue.]

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.02.05-keyboard-config/index.html b/v13/03.02.05-keyboard-config/index.html new file mode 100644 index 0000000000..4074c0e1fc --- /dev/null +++ b/v13/03.02.05-keyboard-config/index.html @@ -0,0 +1,2619 @@ + + + + + + + + + + + + + + + + + + + + + + + Keyboard Config - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Keyboard Config object

+

The keyboard config object configures whether certain actions are available from the keyboard when focus is on the flow editor. +See the Keyboard Support section for what key combinations are supported.

+

    const keyboardConfig = {
+       actions: {
+          delete: false,
+          undo: false,
+          redo: false,
+          selectAll: false,
+          deselectAll: false,
+          cutToClipboard: false,
+          copyToClipboard: false,
+          pasteFromClipboard: false
+    };
+
+All actions are true by default so it is only necessary to specify those actions you don’t want as false.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03-callbacks/index.html b/v13/03.03-callbacks/index.html new file mode 100644 index 0000000000..e216cfdc2d --- /dev/null +++ b/v13/03.03-callbacks/index.html @@ -0,0 +1,2846 @@ + + + + + + + + + + + + + + + + + + + + + + + Callbacks Overview - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Callbacks Overview

+

You can optionally provide callback listeners. These will be called when the user interacts with the canvas and allows +your application to perform processing specific to your application’s needs based on user interactions. If you don’t implement any of the callbacks, Common Canvas will perform appropriate actions if necessary.

+

These listeners are as follows:

+

Context Menu Handler

+

Overrides or adds to the default context menu (or context toolbar) displayed for nodes, links, comments, etc.

+

Before Edit Action Handler

+

Called for each edit action on the canvas. It is called before the internal object model has been updated and the edit action has completed, so this can be used to cancel user actions if necessary.

+

Edit Action Handler

+

Called for each edit action on the canvas. It is called after the internal object model has been updated and the edit action has completed.

+

Layout Handler

+

Allows the application to override layout settings for nodes on a node-by-node basis.

+

Decoration Action Handler

+

Called whenever the user clicks on a decoration which has its hotspot field set to true.

+

Tip Handler

+

Allows the application to override the tips displayed for canvas objects such as nodes and links.

+

ID Generator Handler

+

Called whenever a new object is created and allows the application to specify its own IDs for canvas objects.

+

Selection Change Handler

+

Called whenever the set of selected objects changes.

+

Click Action Handler

+

Called whenever something is clicked, double-clicked or right-clicked on the canvas. This can be used, for example, +to open a properties window when a node is double clicked.

+

Action Label Handler

+

Allows the application to override labels displayed for command actions that are shown for undo and redo actions.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.01-context-menu-handler/index.html b/v13/03.03.01-context-menu-handler/index.html new file mode 100644 index 0000000000..078c687976 --- /dev/null +++ b/v13/03.03.01-context-menu-handler/index.html @@ -0,0 +1,3058 @@ + + + + + + + + + + + + + + + + + + + + + + + Context Menu Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Context Menu Handler

+

This callback is optional. You don’t need to implement anything for it. If implemented, it must return an array of actions that describe what options are displayed in the Context Menu or Context Toolbar.

+

If this callback is not provided Common Canvas will handle context menu/toolbars, and their actions, internally. You only need to implement this callback if you want to add or remove options to/from the context menu/toolbar or provide your own menus in place of the default ones.

+

When is it called

+

For Context Menu

+

This callback will be called if the enableContextToolbar canvas config option is set to false (which is the default) and when the user performs a context menu gesture, such as mouse ‘right click’ or clicking an ellipsis icon, on a:

+
    +
  • Node
  • +
  • Link
  • +
  • Comment
  • +
  • Port
  • +
  • The canvas background or
  • +
  • Combination of objects - if a number of objects are selected
      +
    • Note: With multiple selected objects, the convention is that a context menu/toolbar should display actions that are applicable to all the objects that are selected, rather than just the object for which the menu was requested.
    • +
    +
  • +
+

For Context Toolbar

+

This callback will be called if the enableContextToolbar canvas config option is set to true and when the mouse cursor is hovered over any of the objects mentioend above.

+

contextMenuHandler

+
    contextMenuHandler(source, defaultMenu)
+
+

‘source’ parameter

+

The source object passed in looks like this: +

    {
+      type: "node",
+      targetObject: {<object_info>},
+      selectedObjectIds: ["node_1", "node_2"],
+      mousePos: {x: "10", y:"20"}
+    }
+
+type - Indicates the type of object for which the context menu was selected. Can be “node”, “input_port”, “output_port”, “link”, “canvas” or “comment”

+

targetObject - The object for which the context menu was requested. Not provided when type is “canvas”.

+

selectedObjectIds - An array containing the IDs of all currently selected nodes and/or comments and/or links.

+

mousePos - An object containing the coords of the mouse when the context menu was requested

+

‘defaultMenu’ parameter

+

This is an array describing the default menu that Common Canvas would usually display. If necessary, you can modify this array with your own elements or remove elements and then return the modified array. Alternatively, you can ignore this parameter and just return your own array.

+

Return array for Context Menus

+

The callback must return an array, that describes the context menu to be displayed. If the callback returns a null, then no menu/toolbar will be displayed.

+

There is one element in the array for each entry in the context menu. An entry can be either a context menu item, which consists of a label and an action, or a divider. An action can be disabled by setting the ‘enable’ field to false.

+

Here’s an example of a contextMenuHandler:

+
    contextMenuHandler(source, defaultMenu) {
+        if (source.type === "node") {
+            return [
+                { action: "deleteSelectedObjects", label: "Delete" },
+                { divider: true},
+                { action: "myApp_Action1", label: "My Action" },
+                { action: "paste", label: "Paste from clipboard", enable: false }
+            ];
+        }
+        return defaultMenu;
+    }
+
+

The above array will produce a context menu like this:

+

+

Customizing the default context menu

+

If you want to simply add your action to the default context menu provided by Common Canvas you can take the defaultMenu parameter provided to the callback, and add your menu item to it. Alternatively, you can provide a complete new context menu of your own.

+

Here is a sample implementation of contextMenuHandler, which takes a source object (described above) and the defaultMenu as parameters, and adds a custom action to the default menu when the user ‘right clicks’ the canvas background.

+
    contextMenuHandler(source, defaultMenu) {
+        let customMenu = defaultMenu;
+        if (source.type === "canvas") {
+            customMenu = customMenu.concat({ action: "myApp_Action1", label: "My Action" });
+        }
+        return customMenu;
+    }
+
+

Return array for Context Toolbar

+

To display a context toolbar the same type of array is returned as described above for context menu. However, there are some extra fields for the action elements in the array. These are

+

isToolbarItem - This is a boolean. The default is false. If set to true the action will be added to the toolbar and if set to false the action will be displayed in the overflow menu.

+

icon - This is the icon to display for the action. If isToolbarItem is set to true you must provide an icon otherwise the action will show as an empty space in the toolbar. If an icon is specified and isToolbarItem is set to false, the icon will be displayed next to the action in the overflow menu. For many internal actions, Common Canvas will automatically display an appropriate Carbon icon. See the Internal Actions page for a list of actions that have associated icons.

+

Dividers can also be added to the context toolbar by specifying ‘toolbarItem: true’

+

  import { Add } from "@carbon/react/icons";
+
+  ...
+
+  contextMenuHandler(source, defaultMenu) {
+        if (source.type === "node") {
+            return [
+                { action: "deleteSelectedObjects", label: "Delete", toolbarItem: true },
+                { divider: true, toolbarItem: true },
+                { action: "myAction1", label: "My Action1", toolbarItem: true, icon: (<Add />) },
+                { action: "disconnectNode", label: "Disconnect", enable: false },
+                { action: "cut", label: "Cut" },
+                { action: "copy", label: "Copy" }
+            ];
+        }
+        return defaultMenu;
+    }
+
+This will produce a context toolbar like this:

+

+

And when the overflow icon is clicked, like this:

+

+

Warning

+

The contents of the context toolbar is dependent on which object the mouse cursor is currently hovering over (which may be different to any of the currently selected objects). You should make sure the actions you return in the array are applicable to the object the mouse cursor is hovering over or, if it is hovering over a selected object and other objects are also selected, to the set of selected objects.

+

To help decide whether the mouse cursor is hovering over a selected object or not, the application can call the canvas controller’s helper function: ‘isContextToolbarForNonSelectedObj(source)’. This will return true if the mouse cursor is over a non-selected object.

+

Actions

+

When the user clicks an action in the menu the action is executed either internally or externally.

+

Internal acitons

+

Internal actions are implemented inside Common Canvas, like “deleteSelectedObjects” in the example above. Common Canvas supports a large number of internal actions.

+

External actions

+

External actions are custom actions you want Common Canvas to display for your application like “myApp_Action1”, in the example above. +Tip: To avoid any future name clashes with internal actions that might be added it is recommended you should make sure you action names are unique. For example, by adding a prefix to your application specfic actions.

+

Handling actions

+

When the user clicks an option in the context menu (or context toolbar) it causes the Before Edit Action Handler and then the Edit Action Handler callbacks to be called.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.02-before-edit-action-handler/index.html b/v13/03.03.02-before-edit-action-handler/index.html new file mode 100644 index 0000000000..5d9cdce544 --- /dev/null +++ b/v13/03.03.02-before-edit-action-handler/index.html @@ -0,0 +1,2682 @@ + + + + + + + + + + + + + + + + + + + + + + + Before Edit Action Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Before Edit Action Handler

+

This callback is optional. It is called before user actions are completed, so it provides the opportunity for the application to alter or cancel an action before it is applied to the object model and reflected in the flow editor display.

+

beforeEditActionHandler

+

    beforeEditActionHandler(data, command)
+
+You don’t need to implement anything for it but if you do implement it you must return either a data object or null. This callback is called in all the same instances where the editActionHandler is called. The difference is that this callback is called before the internal object model is updated. This gives your application the opportunity to examine the action that is about to be performed and either: let it continue; modify it and let it continue; or cancel it.

+

This callback is provided with two parameters: data and command.

+
    +
  1. data parameter - this is the same as the data object described for editActionHandler (see above)
  2. +
  3. command parameter - typically this will be null but for undo operations (that is where data.editType === “undo”) this will be the command that is about to be undone. For redo operations (that is where data.editType === “redo”) this will be the command that is about to be redone.
  4. +
+

This callback must return either the data object that was passed in or null. beforeEditActionHandler will behave as follows based on what is returned:

+
    +
  • If the data object is returned unmodified: the action will be performed as normal.
  • +
  • If the data object is returned modified: the action will be performed based on the modified data object. This means your application can alter the behavior of the action that will be performed. For example, you could intercept a createNode command and change the label for the new node in the nodeTemplate to something different. Then when the node is created the new label will be used. It is recommended you be very very careful when updating this object since there is no error checking in Common Canvas to ensure you modified the object correctly.
  • +
  • If null is returned: the action will be cancelled and not performed on the internal object model nor will editActionHandler be called.
  • +
+

If you need to do any asynchronous activity in the beforeEditActionHandler callback you can:

+
    +
  • Return null from the callback - which will cancel the current action
  • +
  • Do your asynchronous activity. While this is happening, the user ought to be prevented from modifying the canvas so you should display some sort of progress indicator or modal dialog to inform the user that some activity is occurring.
  • +
  • Then call CanvasController.editActionHandler(data) passing in the data object as a parameter with the modified properties. This will then execute the action as before. Note: This means the beforeEditActionHandler will be called a second time so be sure you put in a check to make sure you don’t get into a loop.
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.03-edit-action-handler/index.html b/v13/03.03.03-edit-action-handler/index.html new file mode 100644 index 0000000000..35d0a26080 --- /dev/null +++ b/v13/03.03.03-edit-action-handler/index.html @@ -0,0 +1,2754 @@ + + + + + + + + + + + + + + + + + + + + + + + Edit Action Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Edit Action Handler

+

This callback is optional. You don’t need to implement anything for it and it doesn’t return anything. It is called whenever the user does the following gestures on the canvas:

+
    +
  • Clicks a tool/icon in the toolbar.
  • +
  • Clicks an option in the context menu or context toolbar
  • +
  • Presses a key combination on the keyboard to cause the canvas to change.
  • +
  • Performs some direct manipulation on the canvas such as:
      +
    • Create a node
    • +
    • Moves one or a set of nodes/comments
    • +
    • Edits a comment
    • +
    • Links two nodes together
    • +
    • etc
    • +
    +
  • +
+

These is will either perform one of the many internal actions supported by Common Canvas or the application’s own external actions, if they have been added to the context menu/toolbar or the main canvas toolbar.

+

editActionHandler

+
    editActionHandler(data, command)
+
+

This callback is called after the common-canvas internal object model has been updated. This callback is provided with two parameters: data and command.

+
    +
  1. data parameter - This is the data for the initial request create by whatever action the user performed. The data fields can vary depending on the action. Here is an example: +
        {
    +      editType: "createComment",
    +      editSource: "contextmenu",
    +      selectedObjects: [],
    +      selectedObjectIds: [],
    +      offsetX: 100,
    +      offsetY: 42
    +    }
    +
  2. +
+
    +
  • +

    editType - This is the action that originates from either the toolbar, context menu, keyboard action or direct manipulation on the canvas. If you specified your own action in the context menu or in the toolbar this field will be your action’s name.

    +
  • +
  • +

    editSource - This is the source of the action. It can be set to “toolbar”, “contextmenu”, “keyboard” or “canvas” (for an action caused by direct manipulation on the canvas).

    +
  • +
  • +

    selectedObjects - An array of the currently selected objects.

    +
  • +
  • +

    selectedObjectIds - An array of the IDs of currently selected objects. Included for backward compatibility.

    +
  • +
  • +

    Other fields - Numerous other fields which vary based on the action and the source of the action.

    +
  • +
+
    +
  1. +

    command parameter - This is a Javascript class which is the command object that was executed to run the action ‘requested’ by the user and added to the command stack. The command object may contain fields which are the result of executing the command. For example, when the user creates a new node on the canvas the ‘createNode’ action will be performed and the command parameter in editActionHandler will contain a field called newNode which is the node that was created on the canvas.

    +

    If the user performed an undo action this will be the command that has been undone. If the user performed a redo action this will be the command that was redone.

    +
  2. +
+

Handling external actions

+

If you specified your application’s own ‘external’ action you can do whatever is necessary in this callback.

+

The editType field of the first parameter, passed in to the callback, will be set to the action name.

+

Here’s a simple expmple:

+
    editActionHandler(data, command) {
+        if (data.editType === "myAction") {
+            // Execute my action code here.
+        }
+    }
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.04-layout-handler/index.html b/v13/03.03.04-layout-handler/index.html new file mode 100644 index 0000000000..26c5bbb799 --- /dev/null +++ b/v13/03.03.04-layout-handler/index.html @@ -0,0 +1,2667 @@ + + + + + + + + + + + + + + + + + + + + + + + Layout Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Layout Handler

+

This is an optional handler you don’t need to implement anything for it unless you want to. The layoutHandler callback, when provided, is called for each node on the canvas and allows the application to customize the node layout properties on a +node-by-node basis.

+

layoutHandler

+

    layoutHandler(data)
+
+ The callback should return a JavaScript object whose properties will override the default properties for node layout. The callback is provided with a parameter data which is the node object. Your code can look at the node properties and decide which properties it needs to override. This can be used to change the node shape, styling and position and size of node elements like the image, main label etc.

+

For more details see the Node Customization section for more details.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.05-decoration-action-handler/index.html b/v13/03.03.05-decoration-action-handler/index.html new file mode 100644 index 0000000000..1af418d5de --- /dev/null +++ b/v13/03.03.05-decoration-action-handler/index.html @@ -0,0 +1,2671 @@ + + + + + + + + + + + + + + + + + + + + + + + Decoration Action Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Decoration Action Handler

+

Decorations are small images that can be displayed on or near to your nodes and links. They can be for display only or actionable (so the user can click on them). See the canvas JSON schema for information on how to define decorations for your nodes.

+

This callback is called when the user clicks on an actionable decoration. You don’t need to implement anything for this callback unless you added actionable decorations to your nodes. It doesn’t return anything. It is called whenever the user clicks on a decoration that you added to a node in the canvas JSON.

+

decorationActionHandler

+
    decorationActionHandler(object, id, pipelineId)
+
+

It is provided with these parameters:

+
    +
  • object – the node or link with which the decoration is associated.
  • +
  • id – the ID of the decoration that you provided in the canvas JSON or through the canvas-controller API
  • +
  • pipelineId – the ID of the pipeline for the node or link.
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.06-tip-handler/index.html b/v13/03.03.06-tip-handler/index.html new file mode 100644 index 0000000000..af247f3e37 --- /dev/null +++ b/v13/03.03.06-tip-handler/index.html @@ -0,0 +1,2995 @@ + + + + + + + + + + + + + + + + + + + + + + + Tip Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Tip Handler

+

This optional callback can be implemented to override the tooltip content that is displayed by default for each canvas object. It is called before tips are shown for: palette categories, palette node templates, nodes, ports, links, decorations and the state tag.

+

tipHandler

+

    tipHandler(tipType, data)
+
+Note: The display of tooltips (or not) can be controlled using the tipConfig field of the canvas config object.

+

Common Canvas provides default implementations for all of the tips except for links and decorations, as follows:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ObjectDefault tip behavior
Palette categoryContains the category name and the category description.
Palette node templateContains the category name, the node type and node description.
NodeContains the name, description and status icon and optionally, if the name was modified from the original name, the original node type.
PortThe port label is shown
LinkNo tip is shown by default
DecorationNo tip is shown by default
State tagAn appropriate explanation for the state displayed by the tag
+

To override the content, you can return either a string or JSX object. If your code returns null for a particular type of tip, Common Canvas will display the default tip for that object. See App.js in the test harness code for an example tipHandler.

+

Common Canvas calls the tipHandler callback with two parameters:

+
    +
  • tipType - the type of the tip
  • +
  • data - an object that describes the canvas element for which the tip was requested
  • +
+

Here are some specific examples:

+

Palette categories:

+
    +
  • tipType: “tipTypePaletteCategory”
  • +
  • data: An object with category information, like this:
  • +
+
{
+    category: {
+        id: "1234",
+        label: "Import",
+        description: "Category for import nodes",
+        image: "/images/import.svg"
+    }
+}
+
+

Palette nodes templates:

+
    +
  • tipType: “tipTypePaletteItem”
  • +
  • data: An object with node template information: +
    {
    +    nodeTemplate: {
    +        label: "C50",
    +        description: "C50 Model",
    +        operator_id_ref: "com.ibm.commonicons.models.c50",
    +        type: "model_node",
    +        image: "/images/common_node_icons/models/model_c50.svg",
    +        input_ports: [{...}],
    +        output_ports: []
    +    }
    +}
    +
  • +
+

Nodes:

+
    +
  • tipType: “tipTypeNode”
  • +
  • data: An object with pipelineId and node information: +
    {
    +    pipelineId: "153651d6-9b88-423c-b01b-861f12d01489",
    +    node: {
    +        id: "idGWRVT47XDV",
    +        type: "execution_node",
    +        operator_id_ref: "type",
    +        output_ports: [...],
    +        input_ports: [...],
    +        label: "Define Types",
    +        description: "",
    +        image: "",
    +        x_pos: 445,
    +        y_pos: 219
    +    }
    +}
    +
  • +
+

Ports:

+
    +
  • tipType: “tipTypePort”
  • +
  • data: An object with pipelineId, node and port information: +
    {
    +    pipelineId: "153651d6-9b88-423c-b01b-861f12d01489",
    +    node: {
    +        id: "idGWRVT47XDV",
    +        type: "execution_node",
    +        operator_id_ref: "type",
    +        output_ports: [{...}],
    +        input_ports: [{...}],
    +        label: "Define Types",
    +        description: "",
    +        image: "",
    +        x_pos: 445,
    +        y_pos: 219
    +    },
    +    port: {
    +        id: "outPort",
    +        label: "Output Port"
    +    }
    +}
    +
  • +
+ +
    +
  • tipType: “tipTypeLink”
  • +
  • data: An object with pipelineId and link information. +
    {
    +    pipelineId: "153651d6-9b88-423c-b01b-861f12d01489",
    +    link: {
    +        id: "canvas_link_3",
    +        x1: 515,
    +        y1: 248,
    +        x2: 611,
    +        y2: 180,
    +        class_name: "canvas-data-link",
    +        type: "nodeLink",
    +        src: {
    +            id: "idGWRVT47XDV",
    +            type: "execution_node",
    +            operator_id_ref: "type",
    +            output_ports: [{...}],
    +            input_ports: [{...}],
    +            label: "Define Types",
    +            description: "",
    +            image: "",
    +            x_pos: 445,
    +            y_pos: 219
    +        },
    +        srcPortId: "outPort",
    +        trg: {
    +            id: "id8I6RH2V91XW",
    +            type: "binding",
    +            operator_id_ref: "c50",
    +            output_ports: [],
    +            input_ports: [{...}],
    +            label: "C5.0",
    +            description: "",
    +            image: "",
    +            x_pos: 611,
    +            y_pos: 151
    +        },
    +        trgPortId: "inPort"
    +    }
    +}
    +
  • +
+

Decorations

+
    +
  • tipType: “tipTypeDec”
  • +
  • data: An object with pipelineId and decoration information. +
    {
    +    pipelineId: "153651d6-9b88-423c-b01b-861f12d01489",
    +    decoration: {
    +        "id": "2016",
    +        "position": "topRight",
    +        "label": "LCFC",
    +        "tooltip": "Foxes never quit"
    +    }
    +}
    +
  • +
+

State tag

+
    +
  • tipType: “tipTypeStateTag”
  • +
  • data: An object with pipelineId and decoration information. +
    {
    +    stateTagText: "This flow is locked and cannot be edited.",
    +    stateTagType: "Locked"
    +}
    +
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.07-id-generator-handler/index.html b/v13/03.03.07-id-generator-handler/index.html new file mode 100644 index 0000000000..4096d2834e --- /dev/null +++ b/v13/03.03.07-id-generator-handler/index.html @@ -0,0 +1,2857 @@ + + + + + + + + + + + + + + + + + + + + + + + ID Generator Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

ID Generator Handler

+

This is an optional callback. It is called when new objects are created in the canvas and allows the application to provide its own method to generate a unique id for the object.

+

idGeneratorHandler

+

    idGeneratorHandler(action, data)
+
+ If no idGeneratorHandler is set or the method returns null, an appropriate UUID is generated by Common Canvas for each object. The callback is called with a two parameters: + * An enumerated string to describe the action being performed (create_node, create_comment, create_node_link, create_comment_link, clone_node, clone_comment, clone_node_link or clone_comment_link) and + * A JavaScript object with additional data to describe the object

+

The callback should return a string containing the application’s unique ID for the object. +Here are more details about the different action and their parameters.

+
    +
  • Create node:
      +
    • action: create_node
    • +
    • data:
    • +
    +
  • +
+
nodeType:
+{
+   "label":"C50",
+   "description":"C50 Model",
+   "operator_id_ref":"com.ibm.commonicons.models.c50",
+   "type":"model_node",
+   "image":"/images/common_node_icons/models/model_c50.svg",
+   "input_ports":[
+      {
+         "id":"inPort",
+         "label":"Input Port",
+         "cardinality":{
+            "min":0,
+            "max":1
+         }
+      }
+   ],
+   "output_ports":[
+
+   ]
+}
+
+
    +
  • +

    Create comment:

    +
      +
    • action: create_comment
    • +
    • data: n/a
    • +
    +
  • +
  • +

    Create node link:

    +
      +
    • action: create_node_link
    • +
    • data:
    • +
    +
  • +
+
sourceNode: {
+   "id":"6a547456-f1ea-48a6-9721-45d6ae70dd6b",
+   "label":"Aggregate",
+   "type":"execution_node",
+   "operator_id_ref":"com.ibm.commonicons.operations.aggregate",
+   "image":"/images/common_node_icons/operations/operation_aggregate.svg",
+   "class_name":"d3-node-body",
+   "input_ports":[...],
+   "output_ports":[...],
+   "x_pos":55,
+   "y_pos":97.5,
+   "inputPortsHeight":20,
+   "outputPortsHeight":20,
+   "height":75,
+   "width":70
+},
+targetNode: {
+   "id":"71c96629-be46-418b-be63-0a02ef2fe2e0",
+   "label":"Append",
+   "type":"execution_node",
+   "operator_id_ref":"com.ibm.commonicons.operations.append",
+   "image":"/images/common_node_icons/operations/operation_append.svg",
+   "class_name":"d3-node-body",
+   "input_ports":[...],
+   "output_ports":[...],
+   "x_pos":264,
+   "y_pos":83.5,
+   "inputPortsHeight":20,
+   "outputPortsHeight":20,
+   "height":75,
+   "width":70
+}
+
+
    +
  • Create comment link:
      +
    • action: create_comment_link
    • +
    • data:
    • +
    +
  • +
+
comment: {
+   "id":"8c81aac7-ebe5-4f96-9d63-eabc22b09635",
+   "class_name":"d3-comment-rect",
+   "content":"",
+   "height":42,
+   "width":175,
+   "x_pos":50,
+   "y_pos":50
+},
+targetNode: {
+   "id":"71c96629-be46-418b-be63-0a02ef2fe2e0",
+   "label":"Append",
+   "type":"execution_node",
+   "operator_id_ref":"com.ibm.commonicons.operations.append",
+   "image":"/images/common_node_icons/operations/operation_append.svg",
+   "class_name":"d3-node-body",
+   "input_ports":[...],
+   "output_ports":[...],
+   "x_pos":264,
+   "y_pos":83.5,
+   "inputPortsHeight":20,
+   "outputPortsHeight":20,
+   "height":75,
+   "width":70
+}
+
+
    +
  • Clone node: triggered when copying & pasting a node
      +
    • action: clone_node
    • +
    • data:
    • +
    +
  • +
+
node: {
+   "id":"56d30c83-3a08-4147-933e-b01d3c348ac1",
+   "label":"Append",
+   "type":"execution_node",
+   "operator_id_ref":"com.ibm.commonicons.operations.append",
+   "image":"/images/common_node_icons/operations/operation_append.svg",
+   "class_name":"d3-node-body",
+   "input_ports":[...],
+   "output_ports":[...],
+   "x_pos":265,
+   "y_pos":177.5,
+   "inputPortsHeight":20,
+   "outputPortsHeight":20,
+   "height":75,
+   "width":70
+}
+
+
    +
  • Clone comment: triggered when copying & pasting a comment
      +
    • action: clone_comment
    • +
    • data:
    • +
    +
  • +
+
comment: {
+   "id":"8c81aac7-ebe5-4f96-9d63-eabc22b09635",
+   "class_name":"d3-comment-rect",
+   "content":"",
+   "height":42,
+   "width":175,
+   "x_pos":50,
+   "y_pos":50
+}
+
+
    +
  • Clone node link: triggered when copying & pasting two connected nodes
      +
    • action: clone_node_link
    • +
    • data:
    • +
    +
  • +
+

link: {
+   "id":"12c4308e-f572-402a-8dd3-604d438539d4",
+   "class_name":"d3-data-link",
+   "srcNodeId":"2b1af6c2-f98b-4728-97b3-416d40224bce",
+   "trgNodeId":"b43fffe6-dc01-4d30-8b6d-abd977850a2e",
+   "type":"nodeLink"
+},
+sourceNodeId: "56d30c83-3a08-4147-933e-b01d3c348ac1",
+targetNodeId: "815271f0-f4da-4793-ab8f-c4c32d3dd7e0"
+
+Note that the link to be cloned has references to the original source and target nodes, while the sourceNodeId and targetNodeId are the new node ids for the copied nodes. The new nodes are not part of the model yet.

+
    +
  • clone comment link: triggered when copying&pasting a comment and a node that are connected
      +
    • action: clone_comment_link
    • +
    • data:
    • +
    +
  • +
+

link: {
+   "id":"12c4308e-f572-402a-8dd3-604d438539d4",
+   "class_name":"d3-comment-link",
+   "srcNodeId":"2b1af6c2-f98b-4728-97b3-416d40224bce",
+   "trgNodeId":"b43fffe6-dc01-4d30-8b6d-abd977850a2e",
+   "type":"commentLink"
+},
+sourceNodeId: "56d30c83-3a08-4147-933e-b01d3c348ac1",
+targetNodeId: "815271f0-f4da-4793-ab8f-c4c32d3dd7e0"
+
+Note that the link to be cloned has references to the original comment and target node, while the sourceNodeId and targetNodeId are the new ids for the copied comment and node. The new node and comment are not part of the model yet.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.08-selection-change-handler/index.html b/v13/03.03.08-selection-change-handler/index.html new file mode 100644 index 0000000000..5ff77f93ee --- /dev/null +++ b/v13/03.03.08-selection-change-handler/index.html @@ -0,0 +1,2764 @@ + + + + + + + + + + + + + + + + + + + + + + + Selection Change Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Selection Change Handler

+

This is an optional callback. It is triggered whenever the set of selected objects on the canvas changes +either through selection or deselection.

+

selectionChangeHandler

+

    selectionChangeHandler(data)
+
+The callback contains a JavaScript object with the following format: +
{
+    "selection": [
+      "id6PXRG57DGIV"
+    ],
+    "selectedNodes": [
+      {
+        "id": "id6PXRG57DGIV",
+        "type": "binding",
+        "operator_id_ref": "variablefile",
+        "output_ports": [...],
+        "input_ports": [],
+        "label": "DRUG1n",
+        "description": "",
+        "image": "",
+        "x_pos": 96,
+        "y_pos": 219,
+        "class_name": "canvas-node",
+        "decorations": [],
+        "parameters": [],
+        "messages": [],
+        "inputPortsHeight": 0,
+        "outputPortsHeight": 20,
+        "height": 75,
+        "width": 70
+      }
+    ],
+    "selectedComments": [],
+    "addedNodes": [
+      {
+        "id": "id6PXRG57DGIV",
+        "type": "binding",
+        "operator_id_ref": "variablefile",
+        "output_ports": [...],
+        "input_ports": [],
+        "label": "DRUG1n",
+        "description": "",
+        "image": "",
+        "x_pos": 96,
+        "y_pos": 219,
+        "class_name": "canvas-node",
+        "decorations": [],
+        "parameters": [],
+        "messages": [],
+        "inputPortsHeight": 0,
+        "outputPortsHeight": 20,
+        "height": 75,
+        "width": 70
+      }
+    ],
+    "addedComments": [],
+    "deselectedNodes": [
+      {
+        "id": "id2PZSCTRPRIJ",
+        "type": "execution_node",
+        "operator_id_ref": "derive",
+        "output_ports": [...],
+        "input_ports": [...],
+        "label": "Na_to_K",
+        "description": "",
+        "image": "",
+        "x_pos": 219.01116943359375,
+        "y_pos": 162.3754425048828,
+        "class_name": "canvas-node",
+        "decorations": [],
+        "parameters": [],
+        "messages": [...],
+        "inputPortsHeight": 20,
+        "outputPortsHeight": 20,
+        "height": 75,
+        "width": 70
+      }
+    ],
+    "deselectedComments": [
+      {
+        "id": "id42ESQA3VPXB",
+        "content": " comment 1",
+        "height": 34,
+        "width": 128,
+        "x_pos": 132,
+        "y_pos": 103,
+        "class_name": "canvas-comment-1"
+      }
+    ],
+    previousPipelineId: "123-456",
+    selectedPipelineId: "789-012"
+}
+

+
    +
  • selection: Array with ids of selected nodes and comments
  • +
  • selectedNodes: Array of selected node objects
  • +
  • selectedComments: Array of selected comment objects
  • +
  • addedNodes: Array with node objects that were added to the selection
  • +
  • addedComments: Array with comment objects that were added to the selection
  • +
  • deselectedNodes: Array with node objects that were removed from the selection
  • +
  • deselectedComments: Array with comment objects that were removed from the selection
  • +
  • previousPipelineId: The ID of the Pipeline for the selected objects prior to the selection action
  • +
  • selectedPipelineId: The ID of the Pipeline for the newly selected objects
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.09-click-action-handler/index.html b/v13/03.03.09-click-action-handler/index.html new file mode 100644 index 0000000000..9fd8450df5 --- /dev/null +++ b/v13/03.03.09-click-action-handler/index.html @@ -0,0 +1,2682 @@ + + + + + + + + + + + + + + + + + + + + + + + Click Action Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Click Action Handler

+

This callback is optional. You don’t need to implement anything for it and it doesn’t need to return anything to Common Canvas. It is called whenever the user clicks or double clicks on something on the canvas. You could use this callback to implement opening a properties dialog when the user double clicks a node.

+

clickActionHandler

+
    clickActionHandler(source)
+
+

The callback is called with a single source parameter which contain information about the object that was clicked.

+

Note: When handling selections, it is recommended the selectionChangeHandler be used in preference to this handler when possible. selectionChangeHandler will notify you of all selection changes regardless of how they occur, such as when the user presses Ctrl+A on the keyboard to select all objects.

+

At the moment only click/double-click/context-menu on nodes and the canvas background are returned. It is provided with one parameter that looks like this:

+

    {
+      clickType: "DOUBLE_CLICK"
+      id: "node_1",
+      objectType: "node",
+      selectedObjectIds: ["node_1", "node_2"]
+    }
+
+The fields can be:

+
    +
  • clickType - This can be either “SINGLE_CLICK”, “SINGLE_CLICK_CONTEXTMENU” or “DOUBLE_CLICK”
  • +
  • objectType - Can be either “node”, “comment”, “canvas” or “region”. “region” is specified when the user pulls out a selection rectangle around a set of objects that might include nodes and comments.
  • +
  • id - The ID of the node or comment clicked. Only provided when objectType is “node” or “comment”
  • +
  • selectedObjectIds - An array of the selected objects (after the click action was performed).
  • +
+

Note: “SINGLE_CLICK_CONTEXTMENU” indicates that the user performed a contextmenu gesture when doing the click such as pressing the right-side mouse button or a two finger tap on a trackpad.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.03.10-action-label-handler/index.html b/v13/03.03.10-action-label-handler/index.html new file mode 100644 index 0000000000..8c02fc8553 --- /dev/null +++ b/v13/03.03.10-action-label-handler/index.html @@ -0,0 +1,2665 @@ + + + + + + + + + + + + + + + + + + + + + + + Action Label Handler - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Action Label Handler

+

This is an optional handler you don’t need to implement anything for it unless you want to. This callback allows your code to override the default tooltip text for the Undo and Redo buttons.

+

actionLabelHandler

+
    actionLabelHandler(action)
+
+

The actionLabelHandler callback, when provided, is called for each action that is performed in Common Canvas. The action object parameter, passed in to the callback, contains details of the action being performed. This callback should return either a string or null. If a string is returned it will be shown in the tooltip for the Undo button in the toolbar preceded by “Undo:” and the string will also appear in the tooltip for the Redo button (when appropriate) preceded by “Redo:”. If null is returned, Common Canvas will display the default text for the Undo and Redo buttons.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.04-canvas-controller/index.html b/v13/03.04-canvas-controller/index.html new file mode 100644 index 0000000000..7d968c3eea --- /dev/null +++ b/v13/03.04-canvas-controller/index.html @@ -0,0 +1,3998 @@ + + + + + + + + + + + + + + + + + + + + + + + API - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Canvas Controller API

+

Your application code can programmatically perform many of the actions that the user can do in the Common Canvas using the Canvas Controller API. +Note: See this page for differences between the structure of objects in the API and the schema.

+

In most cases within the API, the pipelineId parameter is optional. If pipelineId is omitted, the method will default to the pipeline that is currently displayed in the main canvas viewport.

+

Warning 1: Do not alter the IDs of objects that currently exist on the canvas. Changing object IDs can cause internal problems, in particular with the command stack.

+

Warning 2: When using external pipeline flows, Pipeline IDs must be globally unique identifiers.

+

The API provides the following:

+

Pipeline Flow methods

+
// Loads the pipelineFlow document provided into Common Canvas and displays it.
+// The document must conform to the pipelineFlow schema as documented in the
+// elyra-ai pipeline-schemas repo. Documents conforming to older versions may be
+// provided but they will be upgraded to the most recent version.
+setPipelineFlow(flow)
+
+// Clears the pipleine flow and displays an empty canvas.
+clearPipelineFlow()
+
+// Returns the current pipelineFlow document in the latest version of the
+// pipelineFlow schema as documented in the elyra-ai pipeline-schemas repo.
+getPipelineFlow()
+
+// Returns the current pipelineFlow document ID.
+getPipelineFlowId()
+
+// Returns the ID of the primary pipeline from the pipelineFlow.
+getPrimaryPipelineId()
+
+// Returns the external pipeline flow for the url passed in. The external
+// flow must have been loaded through some Common Canvas action for this
+// method to be able to return anything.
+getExternalPipelineFlow(url)
+
+// Returns the internal format of all canvas data stored in memory by
+// Common Canvas. Nodes, comments and links are returned in the internal
+// format.
+getCanvasInfo()
+
+// Returns the IDs of the ancestor pipleline of the pipeline ID passed in.
+getAncestorPipelineIds(pipelineId)
+
+// Removes all styles from nodes, comments and links. See the setObjectsStyle
+// and setLinkStyle methods for details on setting styles.
+// temporary - is a boolean that indicates whether temporary or permanent
+// styles should be removed.
+removeAllStyles(temporary)
+
+// Specifies the new styles for objects that are not highlighted during
+// branch highlighting.
+// newStyle - is a style specification object.
+setSubdueStyle(newStyle)
+
+

Pipeline methods

+
// Returns the pipeline object for the pipeline Id passed in.
+getPipeline(pipelineId)
+
+// Returns the ID of the pipeline object which is currently on display
+// in the canvas. Typically, this is the primary pipeline but will be
+// different if the user has navigated into one or more supernodes; in
+// which case it will be the ID of the pipeline at the level in the
+// supernode hierarchy that is currently on display.
+getCurrentPipelineId()
+
+// Returns truty if the pipeline is external (that is it is part of an
+// external pipeline flow). Otherwise, return falsy to indicate the pipeline
+// is local.
+isPipelineExternal(pipelineId)
+
+// Returns the flow validation messages for the pipeline ID passed in.
+getFlowMessages(pipelineId)
+
+// Returns a boolean to indicate whether there are any messages of
+// includeMsgsType in the pipeline identified by the pipeline ID passed in.
+// includeMsgsType - can be either "error" or "warning"
+isFlowValid(includeMsgTypes, pipelineId)
+
+// Rearranges the nodes in the canvas in the direction specified for the
+// pipeline ID passed in.
+// layoutDirection - can be "horizontal" or "vertical"
+autoLayout(layoutDirection, pipelineId)
+
+

Palette methods

+
// Loads the palette data as described in the palette schema in
+// elyra-ai pipeline-schemas repo. Any version can be loaded and it will be
+// upgraded to the latest version.
+setPipelineFlowPalette(palette)
+
+// Clears the palette data from Common Canvas.
+clearPaletteData()
+
+// Sets the loading text of the category. If set to a non-empty string the
+// category will show an InlineLoading control in the palette category div
+// with this text as the label. If set to falsey the palette category
+// will display as normal.
+setCategoryLoadingText(categoryId, loadingText)
+
+// Sets the empty text of the category. If set to a non-empty string and the
+// category does not have any nodes, the palette will show a warning icon with
+// this text as a message under the category title when the category is opened.
+// This message will not be displayed if the field is set to falsey or if
+// nodetypes are added to the category.
+setCategoryEmptyText(categoryId, emptyText)
+
+// Adds a new node into the palette:
+// nodeTypeObj - must conform to the style of node used by the palette as
+// described in the palette schema. See objects in nodeTypes array in the
+// palette schema:
+//  https://github.com/elyra-ai/pipeline-schemas/blob/master/common-canvas/palette/palette-v3-schema.json
+// category - is the name of the palette category where the node will be
+// added. If the category doesn't exist it will be created.
+// categoryLabel - Is an optional param. If a new category is created it will
+// be displayed with this label.
+// categoryDescription - Is an optional param. If a new category is created
+// it will be displayed with this description.
+// categoryImage - Is an optional param. The image displayed for the category provided as a
+// reference to an image or the image itself.
+addNodeTypeToPalette(nodeTypeObj, categoryId, categoryLabel, categoryDescription, categoryImage)
+
+// Adds an array of new node into the palette:
+// nodeTypeObjs - an array of nodetypes that must conform to the style of
+// nodes used by the palette as described in the palette schema. See objects
+// in nodeTypes array in the palette schema:
+//  https://github.com/elyra-ai/pipeline-schemas/blob/master/common-canvas/palette/palette-v3-schema.json
+// category - is the name of the palette category where the node will be
+// added. If the category doesn't exist it will be created.
+// categoryLabel - is an optional param. If a new category is created it will
+// be displayed with this label.
+// categoryImage - the image displayed for the category provided as a
+// reference to an image or the image itself.
+// categoryDescription - Is an optional param. If a new category is created
+// it will be displayed with this description.
+// categoryImage - Is an optional param. The image displayed for the category provided as a
+// reference to an image or the image itself.
+addNodeTypesToPalette(nodeTypeObjs, categoryId, categoryLabel, categoryDescription, categoryImage)
+
+// Removes nodetypes from a palette category
+// selObjectIds - an array of object IDs to identify the nodetypes to be
+// removed
+// categoryId - the ID of the category from which the nodes will be removed
+removeNodesFromPalette(selObjectIds, categoryId)
+
+// Returns the palette data document which will conform to the latest version
+// of the palette schema.
+getPaletteData()
+
+// Returns the palette node identified by the operator ID passed in.
+getPaletteNode(operatorId)
+
+// Returns the palette node identified by the node ID passed in.
+getPaletteNodeById(nodeId)
+
+// Returns the category of the palette node identified by the operator passed in
+getCategoryForNode(nodeOpIdRef)
+
+// Converts a node template from the format use in the palette (that conforms
+// to the schema) to the internal node format.
+convertNodeTemplate(nodeTemplate)
+
+// Opens the palette category identified by the category ID passed in.
+openPaletteCategory(categoryId)
+
+// Closes the palette category idetified by the category ID passed in.
+closePaletteCategory(categoryId)
+
+// Opens all the palette categories.
+openAllPaletteCategories()
+
+// Closes all the palette categories.
+closeAllPaletteCategories()
+
+// Returns true or false to indicate whether a palette category is open or not.
+isPaletteCategoryOpen(categoryId)
+
+

Selections methods

+
// Sets the currently selected objects replacing any current selections.
+// newSelection - An array of object IDs for nodes and/or comments
+// pipelineId - Optional. The ID of the pipeline where the objects exist.
+// Selected objects can only be in one pipeline. If this parameter is omitted
+// it is assumed the selections will be for objects in the 'top-level' pipeline
+// being displayed.
+setSelections(newSelection, pipelineId)
+
+// Clears all the current selections from the canvas.
+clearSelections()
+
+// Selects all the objects on the canvas.
+selectAll()
+
+// Returns an array of the IDs of the currently selected objects.
+getSelectedObjectIds()
+
+// Returns the currently selected Nodes.
+getSelectedNodes()
+
+// Returns the currently selected Comments.
+getSelectedComments()
+
+// Returns the ID of the pipeline in which the currently selected objects
+// exist. Only one pipeline may contain selected objects.
+getSelectedPipelineId()
+
+// Deletes all currently selected objects.
+deleteSelectedObjects()
+
+// Returns true if the currently selected objects are all linked together.
+// This is used when deciding to creating a supernode.
+areSelectedNodesContiguous()
+
+

Notification messages methods

+

The notification panel is displayed by the user by clicking the notifications icon in the toolbar. Your application can display whatever messages it wants in the notification panel. See the Notification Messages paage for the structure of message objects. The contents of the notification panel can be managed using the methods below: +

// Overwrites the array of notification messages shown in the notification
+// panel.
+// newMessage - An array of messages (see getNotificationMessages)
+setNotificationMessages(newMessages)
+
+// Deletes all notification messages shown in the notification panel.
+clearNotificationMessages()
+
+// Removes the notification messages from the given array of ids
+deleteNotificationMessages(ids)
+
+// Returns the array of current notification messages. If the messageType is
+// provided only messages of that type will be returned. If messageType is
+// not provided, all messages will be returned. The format of a notification
+// message is an object with these fields:
+// {
+//   "id": string (Required),
+//   "type" : enum, oneOf ["info", "success", "warning", "error"] (Required),
+//   "callback": function, the callback function when a message is clicked (Required),
+//   "title": string (Optional),
+//   "content": string, html, JSX Object (Optional),
+//   "timestamp": string (Optional),
+//   "closeMessage": string (Optional)
+// }
+getNotificationMessages(messageType)
+
+// Returns the maximum notification message type present in the current set
+// of notification messages. For this: ("error" > "warning" > "success" > "info")
+getNotificationMessagesMaxType()
+

+

Node AND comment methods

+

In Common Canvas nodes and comments are collectively known as objects. The following methods may be used to manage either collections of comments or nodes or a mixture of both. +Note:

+
    +
  • See this sections if you are working with styles.
  • +
  • See this section if you are working with decorations.
  • +
  • See this section for differences between the structure of objects in the API and the schema. +
    // Moves the objects identified in the data object which must be in the
    +// pipeline identified by the pipeline ID.
    +// data - A javascript object like this:
    +// {
    +//   nodes: []       // An array of node and comment IDs
    +//   offsetX: number // Offset in pixels the objects will move in the X dir
    +//   offsetY: number // Offset in pixels the objects will move in the Y dir
    +// }
    +moveObjects(data, pipelineId)
    +
    +// Deletes the objects specified in objectIds array.
    +// objectIds - An array of node and comment IDs
    +deleteObjects(objectIds, pipelineId)
    +
    +// Removes the links to and from the objects specified in the objectIds array.
    +// objectIds - An array of node and comment IDs
    +disconnectObjects(objectIds, pipelineId)
    +
    +// Deletes the object specified by the id in the pipleine specified by
    +// pipeline ID.
    +// @Deprecated Use deleteNode or deleteComment as appropriate instead.
    +deleteObject(id, pipelineId)
    +
    +// Sets the style of the objects specified by pipelineObjectIds to be
    +// the newStyle which will be either temporary or permanent.
    +// pipelineObjectIds: This identified the objects to be styles. It is a
    +// javascript object like this:
    +//   {
    +//     <pipelineID_1>: [
    +//       <objectID_1_1>,
    +//       <objectID_1_2>
    +//     ],
    +//     <pipelineID_2>: [
    +//         <objectID_2_1>,
    +//         <objectID_2_2>
    +//     ]
    +//   }
    +// newStyles - This is a style specification objects.
    +// temporary - A boolean to indicate if the style is serialized when
    +//             getPipelineFlow() method is called or not.
    +setObjectsStyle(pipelineObjectIds, newStyle, temporary)
    +
    +// Sets the styles of multiple objects at once.
    +// pipelineObjStyles - Specified the objects and the styles each should be
    +// set to. It is a javascript array like this:
    +//   [
    +//     { pipelineId: <pipelineId>, objId: <objectId>, style: <style_spec>},
    +//     { pipelineId: <pipelineId>, objId: <objectId>, style: <style_spec>},
    +//     { pipelineId: <pipelineId>, objId: <objectId>, style: <style_spec>}
    +//   ]
    +// temporary - A boolean to indicate if the styles are serialized when
    +//             getPipelineFlow() method is called or not.
    +setObjectsMultiStyle(pipelineObjStyles, temporary)
    +
  • +
+

Node methods

+
// Retuns an array of nodes for the pipeline specified by the pipelineId.
+getNodes(pipelineId)
+
+// Returns a new node created from the data parameter in the pipeline
+// identified by the pipelineId.
+// The data parameter must contain:
+// nodeTemplate -  a node template from the palette. The nodeTemplate
+//                 can be retrieved from the palette using with Canvas
+//                 Controller methods: getPaletteNode or getPaletteNodeById.
+// offsetX - the x coordinate of the new node
+// offsetY - the y coordinate of the new node
+createNode(data, pipelineId)
+
+// Adds a new node into the pipeline specified by the pipelineId.
+addNode(node, pipelineId)
+
+// Creates a node using the data parameter provided in the pipeline specified
+// by pipelineId and adds the command to the command stack (so the user can
+// undo/redo the command). This will also cause the beforeEditActionHandler
+// and editActionHandler callbacks to be called.
+// The data parameter must contain:
+// nodeTemplate -  a node template from the palette. The nodeTemplate
+//                 can be retrieved from the palette using with Canvas
+//                 Controller methods: getPaletteNode or getPaletteNodeById.
+// offsetX - the x coordinate of the new node
+// offsetY - the y coordinate of the new node
+//
+// If pipelineId is omitted the node will be created in the current
+// "top-level" pipeline.
+createNodeCommand(data, pipelineId)
+
+// Deletes the node specified.
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+deleteNode(nodeId, pipelineId)
+
+// Sets the node properties
+// nodeId - The ID of the node
+// properties - An object containing properties to be overriden in the node
+// pipelineId - The ID of the pipeline
+setNodeProperties(nodeId, properties, pipelineId)
+
+// Sets the node parameters
+// nodeId - The ID of the node
+// parameters - An array of parameters
+// pipelineId - The ID of the pipeline
+setNodeParameters(nodeId, parameters, pipelineId)
+
+// Sets the node UI parameters
+// nodeId - The ID of the node
+// parameters - An array of UI parameters
+// pipelineId - The ID of the pipeline
+setNodeUiParameters(nodeId, uiParameters, pipelineId)
+
+// Sets the node messages
+// nodeId - The ID of the node
+// messages - An array of messages
+// pipelineId - The ID of the pipeline
+setNodeMessages(nodeId, messages, pipelineId)
+
+// Sets a single message on a node
+// nodeId - The ID of the node
+// message - A message
+// pipelineId - The ID of the pipeline
+setNodeMessage(nodeId, message, pipelineId)
+
+// Sets the lable for a node
+// nodeId - The ID of the node
+// ndeLabel - The label
+// pipelineId - The ID of the pipeline
+setNodeLabel(nodeId, newLabel, pipelineId)
+
+// Sets the class name to newClassName of the nodes identified by nodeIds
+// array in the pipleine specified by pipeline ID. The class name will be
+// applied to the node body path.
+setNodesClassName(nodeIds, newClassName, pipelineId)
+
+// Sets the decorations on a node. The decorations array passed in
+// will replace any decorations currently applied to the node.
+// nodeId - The ID of the node
+// newDecorations - An array of decoration objects.
+// pipelineId - The ID of the pipeline
+setNodeDecorations(nodeId, newDecorations, pipelineId)
+
+// Sets the input ports on a node. The inputs array of ports provided will
+// replace any input ports for a node.
+// nodeId - The ID of the node
+// inputs - An array of input port objects.
+// pipelineId - The ID of the pipeline
+setNodeInputPorts(nodeId, inputs, pipelineId)
+
+// Sets the output ports on a node. The outputs array of ports provided will
+// replace any output ports for a node.
+// nodeId - The ID of the node
+// outputs - An array of output port objects.
+// pipelineId - The ID of the pipeline
+setNodeOutputPorts(nodeId, outputs, pipelineId)
+
+// Sets the decorations of multiple nodes at once. The decorations array
+// passed in will replace any decorations currently applied to the nodes.
+// pipelineNodeDecorations - Specifies the nodes and their decorations.
+// It is a JavaScript array like this:
+//   [
+//     { pipelineId: <pipelineId>, nodeId: <nodeId>, decorations: <decoration_spec_array>},
+//     { pipelineId: <pipelineId>, nodeId: <nodeId>, decorations: <decoration_spec_array>},
+//     { pipelineId: <pipelineId>, nodeId: <nodeId>, decorations: <decoration_spec_array>}
+//   ]
+setNodesMultiDecorations(pipelineNodeDecorations)
+
+// Sets the input port label on a node
+// nodeId - The ID of the node
+// portId - The ID of the input port
+// newLabel - The label
+// pipelineId - The ID of the pipeline
+setInputPortLabel(nodeId, portId, newLabel, pipelineId)
+
+// Sets the output port label on a node
+// nodeId - The ID of the node
+// portId - The ID of the output port
+// newLabel - The label
+// pipelineId - The ID of the pipeline
+setOutputPortLabel(nodeId, portId, newLabel, pipelineId)
+
+// Gets a node
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+getNode(nodeId, pipelineId)
+
+// Gets the UI parameters for a node
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+getNodeUiParameters(nodeId, pipelineId)
+
+// Gets the supernodes for a pipeline.
+// pipelineId - The ID of the pipeline
+getSupernodes(pipelineId)
+
+// Returns supernode ID that has a subflow_ref to the given pipelineId.
+getSupernodeObjReferencing(pipelineId)
+
+// Gets the messages for a node
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+getNodeMessages(nodeId, pipelineId)
+
+// Gets the array of input ports for the node or null if the node ID is
+// not recognized.
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+getNodeInputPorts(nodeId, pipelineId)
+
+// Gets the array of output ports for the node or null if the node ID is
+// not recognized.
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+getNodeOutputPorts(nodeId, pipelineId)
+
+// Gets a message for a specific control for a node
+// nodeId - The ID of the node
+// controlName - The control name
+// pipelineId - The ID of the pipeline
+getNodeMessage(nodeId, controlName, pipelineId)
+
+// Gets an array of decorations for a node
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+getNodeDecorations(nodeId, pipelineId)
+
+// Gets the class name associated with the node specified by nodeId in the
+// pipeline specified by pipelineId.
+getNodeClassName(nodeId, pipelineId)
+
+// Gets the style spcification for a node.
+// nodeId - The ID of the node
+// temporary - A boolean to indicate if the style is serialized when
+//             getPipelineFlow() method is called or not.
+// pipelineId - The ID of the pipeline
+getNodeStyle(nodeId, temporary, pipelineId)
+
+// Returns an array of nodes that are for the branch(es) that the nodes,
+// identified by the node IDs passed in, are within.
+// nodeIds - An array of node Ids
+// pipelineId - The ID of the pipeline where the nodes exist
+getBranchNodes(nodeIds, pipelineId)
+
+// Returns an array of nodes that are upstream from the nodes
+// identified by the node IDs passed in.
+// nodeIds - An array of node Ids
+// pipelineId - The ID of the pipeline where the nodes exist
+getUpstreamNodes(nodeIds, pipelineId)
+
+// Returns an array of nodes that are downstream from the nodes
+// identified by the node IDs passed in.
+// nodeIds - An array of node Ids
+// pipelineId - The ID of the pipeline where the nodes exist
+getDownstreamNodes(nodeIds, pipelineId)
+
+// Returns a boolean to indicate whether the supernode is expanded in place.
+// nodeId - The ID of the node
+// pipelineId - The ID of the pipeline
+isSuperNodeExpandedInPlace(nodeId, pipelineId)
+
+// Sets the label, for the node identified, to edit mode, provided the node
+// label is editable. This allows the user to edite the label text.
+setNodeLabelEditingMode(nodeId, pipelineId)
+
+// Sets the decoration label, for the decoration in the node identified, to edit
+// mode, provided the node label is editable. This allows the user to edit the
+// label text.
+setNodeDecorationLabelEditingMode(decId, nodeId, pipelineId)
+
+

Comment methods

+
// Returns the comments from the pipeline.
+// pipelineId - The ID of the pipeline
+getComments(pipelineId)
+
+// Returns a comment from the pipeline.
+// comId - The ID of the comment
+// pipelineId - The ID of the pipeline
+getComment(comId, pipelineId)
+
+// Returns a position object which indicates the position of where a new
+// comment should be placed in a situation where the mouse position cannot be
+// used (e.g. the toolbar button was clicked).
+// pipelineId - The ID of the pipeline
+getNewCommentPosition(pipelineId)
+
+// Creates a comment for the pipeline.
+// source - Input data
+// pipelineId - The ID of the pipeline
+createComment(source, pipelineId)
+
+// Adds a comment to the pipeline.
+// data - the data describing the comment
+// pipelineId - The ID of the pipeline
+addComment(data, pipelineId)
+
+// Edits a comment with the data.
+// data - the comment
+// pipelineId - The ID of the pipeline
+editComment(data, pipelineId)
+
+// Sets the properties in the comment identified by the commentId. The
+// commentProperties is an object containing one or more properties that will
+// replace the corresponding properties in the comment. For example: if
+// commentProperties is { x_pos: 50, y_pos: 70 } the comment
+// will be set to that position.
+setCommentProperties(commentId, commentProperties, pipelineId)
+
+// Sets the class name to newClassName of the comments identified by commentIds
+// array in the pipleine specified by pipeline ID. The class name will be
+// applied to the comment body path.
+setCommentsClassName(commentIds, newClassName, pipelineId)
+
+// Deletes a comment
+// comId - The ID of the comment
+// pipelineId - The ID of the pipeline
+deleteComment(comId, pipelineId)
+
+// Gets the class name associated with the comment specified by commentId in the
+// pipeline specified by pipelineId.
+getCommentClassName(commentId, pipelineId)
+
+// Gets the style spcification for a comment
+// commentId - The ID of the node
+// temporary - A boolean to indicate if the style is serialized when
+//             getPipelineFlow() method is called or not.
+// pipelineId - The ID of the pipeline
+getCommentStyle(commentId, temporary, pipelineId)
+
+// Hides all comments on the canvas.
+hideComments()
+
+// Shows all comments on the canvas - if they were previously hiding.
+showComments()
+
+// Returns true if comments are currently hiding.
+isHidingComments()
+
+// Sets the comment identified, to edit mode so the user can
+// edit the comment.
+setCommentEditingMode(commentId, pipelineId)
+
+ +
// Gets a link
+// linkId - The ID of the link
+// pipelineId - The ID of the pipeline
+getLink(linkId, pipelineId)
+
+// Returns an array of link objects for the pipelineId passed in.
+// pipelineId - The ID of the pipeline
+getLinks(pipelineId)
+
+// Sets the properties in the link identified by the linkId. The
+// linkProperties is an object containing one or more properties that will
+// replace the corresponding properties in the link. For exam`ple: if
+// linkProperties is { trgNodeId: "123", trgNodePortId: "789" } the target
+// node ID will be set to "123" and the target port ID set to "789".
+setLinkProperties(linkId, linkProperties, pipelineId)
+
+// Sets the source properties in the data link identified by the linkId. The
+// srcNodeId and srcNodePortId will be set to the values provided. If
+// srcNodePortId is set to null the current srcNodePortId will be removed
+// from the link. Also, if the link has a srcPos property (because its
+// source end is detached) that will be removed.
+setNodeDataLinkSrcInfo(linkId, srcNodeId, srcNodePortId, pipelineId)
+
+// Sets the target properties in the data link identified by the linkId. The
+// trgNodeId and trgNodePortId will be set to the values provided. If
+// trgNodePortId is set to null the current trgNodePortId will be removed
+// from the link. Also, if the link has a trgPos property (because its
+// target end is detached) that will be removed.
+setNodeDataLinkTrgInfo(linkId, trgNodeId, trgNodePortId, pipelineId)
+
+// Gets a node to node data link
+// srcNodeId - The ID of the source node
+// srcNodePortId - The ID of the source node port
+// trgNodeId - The ID of the target node
+// trgNodePortId - The ID of the target node port
+// pipelineId - The ID of the pipeline
+getNodeDataLinkFromInfo(srcNodeId, srcNodePortId, trgNodeId, trgNodePortId, pipelineId)
+
+// Gets a comment to node link
+// id1 - The ID of the comment
+// id2 - The ID of the node
+// pipelineId - The ID of the pipeline
+getCommentLinkFromInfo(id1, id2, pipelineId)
+
+// Gets a node to node association link
+// id1 - The ID of one of the node
+// id2 - The ID of one of the node
+// pipelineId - The ID of the pipeline
+getNodeAssocLinkFromInfo(id1, id2, pipelineId)
+
+// Adds links to a pipeline
+// linkList - An array of links
+// pipelineId - The ID of the pipeline
+addLinks(linkList, pipelineId)
+
+// Deletes a link
+// link - the link object to be deleted
+// pipelineId - The ID of the pipeline
+deleteLink(link, pipelineId)
+
+// Creates node to node links
+// data - Data describing the links
+// pipelineId - The ID of the pipeline
+createNodeLinks(data, pipelineId)
+
+// Creates comment links
+// data - Data describing the links
+// pipelineId - The ID of the pipeline
+createCommentLinks(data, pipelineId)
+
+// Sets the class name to newClassName of the links identified by linkIds
+// array in the pipleine specified by pipeline ID. The class name will be
+// applied to the link line path.
+setLinksClassName(linkIds, newClassName, pipelineId)
+
+// Sets the style of the links specified by pipelineLinkIds to be
+// the newStyle which will be either temporary or permanent.
+// pipelineLinkIds - This identifies the objects to be styles. It is a
+// javascript object like this:
+//   {
+//     <pipelineID_1>: [
+//       <linkID_1_1>,
+//       <linkID_1_2>
+//     ],
+//     <pipelineID_2>: [
+//         <linkID_2_1>,
+//         <linkID_2_2>
+//     ]
+//   }
+// newStyle - This is a style specification objects.
+// temporary - A boolean to indicate if the style is serialized when
+//             getPipelineFlow() method is called or not.
+setLinksStyle(pipelineLinkIds, newStyle, temporary)
+
+// Sets the styles of multiple links at once.
+// pipelineObjStyles - Specified the links and the styles each should be
+// set to. It is a javascript array like this:
+//   [
+//     { pipelineId: <pipelineId>, objId: <linkId>, style: <style_spec>},
+//     { pipelineId: <pipelineId>, objId: <linkId>, style: <style_spec>},
+//     { pipelineId: <pipelineId>, objId: <linkId>, style: <style_spec>}
+//   ]
+// temporary - A boolean to indicate if the styles are serialized when
+//             getPipelineFlow() method is called or not.
+setLinksMultiStyle(pipelineObjStyles, temporary)
+
+// Gets the class name associated with the link specified by linkId in the
+// pipeline specified by pipelineId.
+getLinkClassName(linkId, pipelineId)
+
+// Returns the style specification for a link.
+// linkIds - An array of links
+// temporary - A boolean to indicate if the style is serialized when
+//             getPipelineFlow() method is called or not.
+// pipelineId - The ID of the pipeline
+getLinkStyle(linkId, temporary, pipelineId)
+
+// Sets the decorations on a link. The decorations array passed in
+// will replace any decorations currently applied to the link.
+// linkId - The ID of the link
+// newDecorations - An array of decoration objects.
+// pipelineId - The ID of the pipeline
+setLinkDecorations(linkId, newDecorations, pipelineId)
+
+// Sets the decorations of multiple links at once. The decorations array
+// passed in will replace any decorations currently applied to the links.
+// pipelineLinkDecorations - Specifies the links and their decorations.
+// It is a javascript array like this:
+//   [
+//     { pipelineId: <pipelineId>, linkId: <linkId>, decorations: <decoration_spec_array>},
+//     { pipelineId: <pipelineId>, linkId: <linkId>, decorations: <decoration_spec_array>},
+//     { pipelineId: <pipelineId>, linkId: <linkId>, decorations: <decoration_spec_array>}
+//   ]
+setLinksMultiDecorations(pipelineLinkDecorations)
+
+// Gets an array of decorations for a link
+// linkId - The ID of the link
+// pipelineId - The ID of the pipeline
+getLinkDecorations(linkId, pipelineId)
+
+// Sets the decoration label, for the decoration in the link identified, to edit
+// mode provided the link label is editable. This allows the user to edit the
+// label text.
+setLinkDecorationLabelEditingMode(decId, linkId, pipelineId)
+
+ +
// Returns the current array of breadcrumbs. There will one breadcrumb object
+// for each level of supernode that the user has navigated into. This array
+// can be used to display breadcrumbs to the user to show where they are
+// within the navigation hierarchy within Common Canvas.
+getBreadcrumbs()
+
+// Returns the last breadcrumb which represents the level with supernode
+// hierarchy that the user has currently navigated to.
+getCurrentBreadcrumb()
+
+

Branch Highlight methods

+
// Highlights the branch(s) (both upstream and downstream) from the node
+// IDs passed in and returns the highlighted object Ids.
+// nodeIds - An array of node Ids
+// pipelineId - The ID of the pipeline
+highlightBranch(nodeIds, pipelineId)
+
+// Highlights the upstream nodes from the node IDs passed in
+// and returns the returns the highlighted object Ids.
+// nodeIds - An array of node Ids
+// pipelineId - The ID of the pipeline
+highlightUpstream(nodeIds, pipelineId)
+
+// Highlights the downstream nodes from the node IDs passed in
+// and returns highlighted object Ids.
+// nodeIds - An array of node Ids
+// pipelineId - The ID of the pipeline
+highlightDownstream(nodeIds, pipelineId)
+
+

Operational methods

+

These are general purpose methods for operation of the common-canvas components:

+

Logging methods

+
// Returns a Boolean to indicate whether canvas logging is switched on or off.
+getLoggingState()
+
+// Sets canvas logging based on the Boolean passed in.
+setLoggingState(state)
+
+

Palette methods

+
// Opens the palette
+openPalette()
+
+// Closes the palette
+closePalette()
+
+// Returns true if the palette is currently open
+isPaletteOpen()
+
+

Context menu methods

+
// Opens the context menu
+openContextMenu(menuDef)
+
+// Closes the context menu
+closeContextMenu()
+
+

Notification Panel methods

+
// Opens the notification panel
+openNotificationPanel()
+
+// Closes the notification panel
+closeNotificationPanel()
+
+// Either opens or closes the notifictaion panel based on its current status
+toggleNotificationPanel()
+
+

Right Flyout methods

+
// Returns a boolean to indicate if the right flyout is open or not
+isRightFlyoutOpen()
+
+

Top panel methods

+
// Returns a boolean to indicate if the top pnel is open or not
+isTopPanelOpen()
+
+

Bottom panel methods

+
// Returns a boolean to indicate if the bottom panel is open or not
+isBottomPanelOpen()
+
+// Sets the height of the bottom panel in pixels. This can be called
+// immediately after the CanvasController has been created, if the bottom
+// panel should be displayed at a specific height when it first opens.
+setBottomPanelHeight(height)
+
+

Canvas/pipeline navigation methods

+
// Displays a pipeline (identified by the pipelineId passed in). This must be
+// one of the pipelines referenced by the current set of breadcrumbs. It
+// cannot be used to open a new pipeline outside the current set of breadcruumbs.
+displaySubPipeline(pipelineId)
+
+// Displays a pipeline for a supernode (identifid by the supernodeId
+// parameter) in a parent pipeline (identifid by the pipelineId parameter).
+// This parent pipeline should be the last of the current set of breadcumbs.
+// That is, the pipeline currently shown "full page" in the canvas.
+displaySubPipelineForSupernode(supernodeId, pipelineId)
+
+// Displays full-page the previous pipeline from the one currently being displayed
+displayPreviousPipeline()
+
+

Command Stack interaction methods

+
// Adds the command object to the command stack which will cause the
+// do() method of the command to be called.
+do(command)
+
+// Calls the undo() method of the next available command on the command
+// stack that can be undone, if one is available.
+undo()
+
+// Undoes a number of commands on the command stack as indicated by the
+// 'count' parameter. If 'count' is bigger than the number of undoable commands
+// on the stack, all undoable commands currently on the command stack
+// will be undone. Uses the editActionHandler method which will cause
+// the app's editActionHandler to be called.
+undoMulti(count)
+
+// Calls the redo() method of the next available command on the command
+// stack that can be redone, if one is available.
+redo()
+
+// Redoes a number of commands on the command stack as indicated by the
+// 'count' parameter. If 'count' is bigger than the number of redoable commands
+// on the stack, all redoable commands currently on the command stack
+// will be redone. Uses the editActionHandler method which will cause
+// the app's editActionHandler to be called.
+redoMulti(count)
+
+// Returns true if there is a command on the command stack
+// available to be undone.
+canUndo()
+
+// Returns true if there is a command on the command stack
+// available to be redone.
+canRedo()
+
+// Returns a string which is the label that descibes the next undoable
+// command.
+getUndoLabel()
+
+// Returns a string which is the label that descibes the next redoable
+// command.
+getRedoLabel()
+
+// Returns an array of all undoable commands currently on the command stack.
+getAllUndoCommands()
+
+// Returns an array of all redoable commands currently on the command stack.
+getAllRedoCommands()
+
+// Clears the command stack of all currently stored commands.
+clearCommandStack()
+
+

Zoom methods

+
// Centers the canvas contents and zooms in
+zoomIn()
+
+// Centers the canvas contents and zooms out
+zoomOut()
+
+// Zooms the canvas contents to fit within the viewport
+zoomToFit()
+
+// Changes the zoom amounts for the canvas. This method does not alter the
+// pipelineFlow document. zoomObject is an object with three fields:
+// x: Is the horizontal translate amount which is a number indicating the
+//    pixel amount to move. Negative left and positive right
+// y: Is the vertical translate amount which is a number indicating the
+//    pixel amount to move. Negative up and positive down.
+// k: is the scale amount which is a number greater than 0 where 1 is the
+//    default scale size.
+zoomTo(zoomObject)
+
+// Increments the translation of the canvas by the x and y increment
+// amounts. The optional animateTime parameter can be provided to animate the
+// movement of the canvas. It is a time for the animation in milliseconds.
+// If omitted the movement happens immediately.
+translateBy(x, y, animateTime)
+
+// Returns the current zoom object for the currently displayed canvas or null
+// if the canvas is not yet rendered for the first time.
+getZoom()
+
+// Returns a zoom object required to pan the objects (nodes and/or comments
+// and/or links) identified by the objectIds array to 'reveal' the objects
+// in the viewport. Returns null if no nodes, comments or links can be found
+// using the IDs passed in. Note: node, comment and link IDs must be unique.
+// The zoom object returned can be provided to the CanvasController.zoomTo()
+// method to perform the zoom/pan action.
+// If the xPos and yPos parameters are provided it will return a zoom object
+// to pan the center of the objects specified, to a location where, xPos
+// is the percentage of the viewport width and yPos is the percentage of the
+// viewport height. So if you want the center of the objects specified to be
+// in the center of the viewport set xPos to 50 and yPos to 50.
+// If the xPos and yPos parameters are undefined (omitted) and all the
+// objects are currently fully within the canvas viewport, this method will
+// return null. This can be used to detect whether the objects are fully
+// visible or not.
+// If the xPos and yPos parameters are undefined and the objects are outside
+// the viewport, a zoom object will be returned that can be used to zoom them
+// so they appear at the nearest side of the viewport to where they are
+// currently positioned.
+// The zoom object returned has three fields:
+// x: Is the horizontal translate amount which is a number indicating the
+//    pixel amount to move. Negative left and positive right
+// y: Is the vertical translate amount which is a number indicating the
+//    pixel amount to move. Negative up and positive down.
+// k: is the scale amount which is a number greater than 0 where 1 is the
+//    default scale size.
+// Parameters:
+// objectIds - An array of nodes and/or comment IDs.
+// xPos - Optional. Can be set to percentage offset of the viewport width.
+// yPos - Optional. Can be set to percentage offset of the viewport height.
+getZoomToReveal(objectIds, xPos, yPos)
+
+// Clears any saved zoom values stored in local storage. This means
+// newly opened flows will appear with the default zoom. This method
+// is only applicable when the `enableSaveZoom` config parameter is
+// set to "LocalStorage".
+clearSavedZoomValues()
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.04.01-decorations/index.html b/v13/03.04.01-decorations/index.html new file mode 100644 index 0000000000..2c6cc874d4 --- /dev/null +++ b/v13/03.04.01-decorations/index.html @@ -0,0 +1,2822 @@ + + + + + + + + + + + + + + + + + + + + + + + Decorations - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Decorations

+

Your application can add Decorations – additional icons, text labels, shapes or JSX objects – to nodes or links in the canvas to indicate special status or attributes of the node or link. Decorations can be static (for display only) or interactive (a hotspot) so the user can click them to initiate some action.

+

Adding Decorations

+

Decorations can be added to the nodes and/or links in four different ways:

+
    +
  1. +

    By using the following CanvasController methods: +

    setNodeDecorations(nodeId, newDecorations, pipelineId)
    +setNodesMultiDecorations(pipelineNodeDecorations)
    +setLinkDecorations(linkId, newDecorations, pipelineId)
    +setLinksMultiDecorations(pipelineLinkDecorations)
    +
    + and can be retrieved using these CanvasController methods: +
    getNodeDecorations(nodeId, pipelineId)
    +getLinkDecorations(linkId, pipelineId)
    +
    + See the CanvasController API documentation for more details.

    +
  2. +
  3. +

    Node decorations can be specified in the nodeLayout object in the canvas config. Decorations specified in this way are applied to all nodes on the canvas.

    +
  4. +
  5. +

    Decorations can be applied to, and retrieved from, nodes or links in the pipelineFlow in the app_data.ui_data section for the node or link. (Note: JSX objects are not supported in the pipelineFlow document).

    +
  6. +
  7. +

    Decorations can be returned in the layout information returned from the layoutHandler CommonCanvas callback method.

    +
  8. +
+

Specification

+

The decoration specification used by these methods and the pipelineFlow is a JavaScript object with these possible properties:

+

These properties are applicable to an image decoration: +

     {
+       id: <decoration_id>,
+       image: <image_url>,
+       position: <position>,
+       distance: <number>
+       x_pos: <number>,
+       y_pos: <number>,
+       width: <number>,
+       height: <number>,
+       hotspot: <boolean>,
+       class_name: <class_name>,
+       outline: <boolean>,
+       tooltip: <string>,
+       temporary: <boolean>
+     }
+
+These properties are applicable to a label decoration: +
     {
+       id: <decoration_id>,
+       label: <string>,
+       label_editable: <boolean>,
+       label_align: <enum>,
+       label_single_line: <boolean>,
+       label_max_characters: <number>,
+       label_allow_return_key: <boolean>,
+       position: <position>,
+       distance: <number>
+       x_pos: <number>,
+       y_pos: <number>,
+       width: <number>,
+       height: <number>,
+       hotspot: <boolean>,
+       class_name: <class_name>,
+       tooltip: <string>,
+       temporary: <boolean>
+     }
+
+These properties are applicable to a shape decoration: +
     {
+       id: <decoration_id>,
+       path: <string>,
+       position: <position>,
+       distance: <number>
+       x_pos: <number>,
+       y_pos: <number>,
+       width: <number>,
+       height: <number>,
+       hotspot: <boolean>,
+       class_name: <class_name>,
+       tooltip: <string>,
+       temporary: <boolean>
+     }
+
+These properties are applicable to a JSX decoration: +
     {
+       id: <decoration_id>,
+       jsx: <JSX Object>,
+       position: <position>,
+       distance: <number>
+       x_pos: <number>,
+       y_pos: <number>,
+       width: <number>,
+       height: <number>,
+       hotspot: <boolean>,
+       class_name: <class_name>,
+       tooltip: <string>,
+     }
+

+

where:

+

id

+

A unique ID for the decoration within the context of the node or link to which the decorator is attached.

+

image

+

A reference to an image to display for the decoration. If an image is specified the image is displayed within an outline rectangle unless outline is set to false. The image should be a reference to your image like: “/images/decorations/zoom-in_32.svg”. Do not set label or path or jsx when this field is set.

+

path

+

An SVG shape that is displayed using this string as it’s SVG path. eg. “M 0 0 L 10 10 -10 10 Z” could be specified to draw a triangle. Do not set image or label or jsx when this field is set.

+

jsx

+

A JSX object that is is displayed at the specified decoration location. Do not set image or path or label when this field is set. Note JSX decorations are not supported in the pipelineFlow document.

+

label

+

A text string that is displayed at the specified decoration location. Do not set image or path or jsx when this field is set.

+

label_editable

+

A boolean that defaults to false. When set to true, if the mouse pointer is hovered over the label an edit icon is displayed next to the label which, when clicked, opens the label for editing. The label can also for double clicked to go to edit mode.

+

When editing is completed (by clicking outside the text area) an editDecorationLabel action is executed which results in calls to first the beforeEditActionHandler and then the editActionHandler callbacks.

+

label_align

+

Can be either “center” or “left”. When set to center the label will be centered on the point defined by the position, distance, x_pos and y_pos properties.

+

label_single_line

+

A boolean that defaults to false. When set to true the label is displayed on a single line and is truncated at the width of the label (specified in the width property for the decoration) and does not word wrap. If it is truncated an ellipsis (…) is displayed at the end of the truncated text.

+

If this property is set to false, long label text is displayed over a number of lines with word wrapping being controlled by the width set for the decoration. If the text extends beyond a second line an ellipsis (..) is displayed at the end of the second line. This is controlled by the -webkit-line-clamp: 2; CSS property. You can override this if you want the ellipsis to be displayed on a different line.

+

Note: For both single and multi-line labels you may need to set the height property for the decoration to show the text fully.

+

label_max_characters

+

A number or null. Defaults to null. If set to a number the label will be restricted to that number of characters. If the label in the pipeline flow document is longer than the max number it will be displayed but when it is edited the user will not be able to do anything except delete characters until the label is shorter than the max number. If this property is set to null or omitted an unlimited number of characters may be entered by the user.

+

label_allow_return_key

+

A boolean that defaults to false. When false, if the user presses the return key nothing will happen. This means multi-line labels will only word-wrap at the width of the decorator. Preventing newline insertion is useful if the label text appears elsewhere in the UI which is not able to show text with newline characters. When set to "save", if the user presses the return key, the editing will be completed and the label saved – this is the equivalent of clicking on the canvas background to complete the edit. If set to true, a new line character will be inserted in the label when the user presses return.

+

position

+

This is the anchor point to which the decoration is attached. For a node, this can be one of these 9 enumerated values: +

   "topLeft", "topCenter", "topRight",
+   "middleLeft", "middleCenter", "middleRight",
+   "bottomLeft", "bottomCenter", "bottomRight".
+
+If omitted it will default to “topLeft”.

+

For a link, this can be one of these 3 enumerated values: +

   "source"
+   "middle"
+   "target"
+
+source will position the decoration at the start point of the line and target will position it at the end point of the line. If omitted it will default to middle.

+

distance

+

A number of pixels. This is only applicable when the decoration is for a link line and then, only with straight connecting lines. That is, for node to node connections when the config property enableLinkType is set to "straight". When specified, it will move the anchor point for the decoration to a new position along the connecting line relative to the initial position specified in the decoration’s position property. A positive number moves the decoration along the line from the starting position towards the target of the link and a negative number backwards towards the source of the link. For example, if a straight link decoration has position of source and a distance of 20 the decoration’s anchor point will be 20 pixels along the link line from the source (start) point of the line. After the distance value has been applied to the anchor point of the decoration, any x_pos and y_pos adjustment will be applied to fine tune the decoration’s final position.

+

x_pos

+

This is the number of pixels horizontally from the anchor point that the decoration is positioned. It can be positive or negative. If omitted it takes a default value from the node layout fields. x_pos is not applicable if you specify an SVG path using the path field, because the SVG path can be used to position the shape.

+

y_pos

+

This is the number of pixels vertically from the anchor point that the decoration is positioned. It can be positive or negative. If omitted it takes a default value from the node layout fields. y_pos is not applicable if you specify an SVG path using the path field, because the SVG path can be used to position the shape.

+

width

+

This is the width for the decorator in pixels. For an image decorator, it is the width of the rectangle surrounding the image. For a label decorator it is the width allowed for display of the label text. If omitted it takes a default value from the node layout fields for Node decorations and from the canvas layout properties for Link properties.

+

height

+

This is the height for the decorator in pixels. For an image decorator, it is the height of the rectangle surrounding the image. For a label decorator it is the height allowed for display of the label text. If omitted it takes a default value from the node layout fields for Node decorations and from the canvas layout properties for Link properties.

+

hotspot

+

A Boolean. It defaults to false. When set to true the decoration becomes clickable and when it is clicked the decorationCallbackHandler is called with the ID of the decoration passed as a parameter.

+

class_name

+

An optional class that will be applied to the decoration. You can add a style rule that references that class in your CSS to style the decoration and override the default styles.

+

outline

+

A Boolean. It defaults to true. When a decoration is specified with an image field the image is typically displayed with a outline rectangle around it and with an offset within the rectangle to improve presentation. If outline is set to false the outline rectangle is not displayed and the image is displayed without any offset from its specified x_pos and y_pos.

+

tooltip

+

A String. When specified, the string will be shown in a tooltip when the pointer is hovered over the decoration. No tip will be displayed if the tooltip property is omitted. Note: for decoration tooltips to be displayed on the canvas, the decorations property of the tipConfig object in the canvas config must be set to true (which is its default setting).

+

temporary

+

A Boolean. It defaults to false. When set to true the decoration object will not be saved in the pipelineFlow document returned by the CanvasController.getPipelineFlow() method. Typically, this should be set to true when adding decorations programmatically to the nodes and links on the canvas.

+

Notes:

+
    +
  1. Using x_pos and y_pos, decorations can be displayed outside the node boundary.
  2. +
  3. If no image or label or path is provided, the default decoration is a rectangle displayed with the class_name provided.
  4. +
  5. You can specify as many decorations as you want by providing extra entries in the decorations array.
  6. +
  7. Images and labels are positioned differently. For images, the position defined for the decoration is the top left corner of the image. For labels it is anchor point for the label which is the base line of the string in the vertical direction and is dependent on the text-anchor CSS property applied to the text. So if you apply the text-anchor: middle CSS property to the label in the style related to class_name the label will be centered on the point calculated for the position of the decoration.
  8. +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.04.02-api-object-structure/index.html b/v13/03.04.02-api-object-structure/index.html new file mode 100644 index 0000000000..08475b4fc2 --- /dev/null +++ b/v13/03.04.02-api-object-structure/index.html @@ -0,0 +1,2719 @@ + + + + + + + + + + + + + + + + + + + + + + + Object Structure - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

API differences with schema

+

Because historically Common Canvas has had to deal with different external flow definitions, there are some differences between the nodes, comments and links that the canvas-controller API handles internally and those specified in the pipelineFlow schema. As follows:

+

Object structure

+

For the API methods that involve nodes, comments and links, those objects are passed in and out by the API in their internal formats rather than the formats defined in the schema files.

+

The internal structure is a somewhat flattened version of that in the schema definition. That means, properties that are in <object>.app_data.ui_data are flattened out and appear as properties in the <object> itself. So for example a node that conforms to the schema might look like this: +

    {
+      id: "1234",
+      op: "select",
+      ...
+      app_data : {
+        ui_data: {
+          label: "Selection node",
+          image: "/images/select.svg",
+          description: "A node for selection"
+          ...
+        },
+        other_data: {
+          prop1: "Something interesting"
+        }
+      }
+    }
+
+whereas when it is passed through the API it looks like this: +
    {
+      id: "1234",
+      op: "select",
+      label: "Selection node",
+      image: "/images/select.svg",
+      description: "A node for selection"
+      ...
+      app_data : {
+        other_data: {
+          prop1: "Something interesting"
+        }
+      }
+    }
+
+Note that, any properties in app_data, other than ui_data, are preserved in the internal format. So in the example, app_data.other_data in the schema format is preserved in the internal format.

+ +

The other difference between the API and the schema formats is with links.

+

In the pipeline flow schema, links are typically defined as properties within another object, for example, a node to node link is defined within a links array inside the inputs field of the target node and contains references to the node id and port of the source node. Also, links from comments to nodes are stored as an array in the comment object.

+

However, in the API and internally in Common Canvas, links are treated as a top level object; that is, there is an array of links stored internally which can be manipulated using the API methods. Each link has a unique ID. Consequently, links can be retrieved from the API by their ID field and properties of the links can be updated again by identifying the links using their ID. If you do not specify an ID for links in your pipelineFlow document a unique global ID will be generated for each link when the pipeline flow is loaded.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.04.03-creating-new-canvas-nodes/index.html b/v13/03.04.03-creating-new-canvas-nodes/index.html new file mode 100644 index 0000000000..823de10b0d --- /dev/null +++ b/v13/03.04.03-creating-new-canvas-nodes/index.html @@ -0,0 +1,2768 @@ + + + + + + + + + + + + + + + + + + + + + + + Creating New Canvas Nodes - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Creating New Canvas Nodes

+ +

Introduction

+

The host application code can programmatically create nodes on the canvas in two ways:

+
    +
  • +

    By calling CanvasController API methods to create and add the node to the canvas.

    +
  • +
  • +

    Creating and adding a node to the Pipeline Flow document

    +
  • +
+

Creating and adding a node to the Canvas Controller API

+

The following code will programmatically add a node to the canvas. These commands will update the common-canvas object model directly and will not be added to the command stack so the user will not be able to undo / redo these actions. Also, the beforeEditActionHandler and editActionHandler callbacks will not be called for these actions.

+

First you can get a node template from the canvas by calling +

    const template = canvasController.getPaletteNode("sort");
+
+where the parameter is the operation (op field) for the palette node. Alternatively, you can retrieve a node template using this method: +
    const template = canvasController.getPaletteNodeById(nodeId)
+
+which returns the node template based on the node ID. This can be useful if you have supernodes in your palette because supernodes do not have an op field. After creating the node template your code can alter fields (for example, the label) within the template. If you do change any fields be careful because Common Canvas doesn’t do any error checking on your fields.

+

Next you create the node: +

    const newNode = canvasController.createNode({
+            nodeTemplate: template,
+            offsetX: 200,
+            offsetY: 400
+});
+
+This will work correctly for regular nodes, and also supernodes, that have been pulled from the palette.

+

Next you add the node object to the canvas. +

    canvasController.addNode(newNode);
+
+The node will appear at the offsetX, offseY position within the coordinate system for the canvas.

+

If command stack is needed

+

This method allows the host application to create a node, or supernode, from a palette template object by creating and executing a command which will be added to the command stack (so the user can undo / redo it) and will also cause the beforeEditActionHandler and editActionHandler callbacks to be called.

+

First your code retrieves a node template from the palette as described above and then calls this method: +

    const data = {
+        nodeTemplate: template,
+        offsetX: 200,
+        offsetY: 400
+    };
+
+    canvasController.createNodeCommand(data, pipelineId)
+

+

Note: If pipelineId is omitted the node will be created in the current “top-level” pipeline.

+

Creating and adding a node using Pipeline Flow document

+

This approach works by your code adding one or more JSON objects directly to the pipeline flow object, either before the pipeline flow document is loaded into Common Canvas using CanvasController.setPipelineFlow(pFlow), or afterwards by retrieving the pipeline flow object from Common Canvas using CanvasController.getPipelineFlow() and then updating the nodes array of whichever pipeline you want to modify. This would require your code to navigate to the pipeline object (that you want to update) in the pipelines array of the pipeline flow and then add the node object to the nodes array in the pipeline object.

+

After updating the pipeline flow object your code would need to reload it into Common Canvas using CanvasController.setPipelineFlow(pFlow).

+

To use this approach you would need a good understanding of the pipeline flow schema and pipeline flow UI schema.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.04.05-notification-messages/index.html b/v13/03.04.05-notification-messages/index.html new file mode 100644 index 0000000000..07ab800d58 --- /dev/null +++ b/v13/03.04.05-notification-messages/index.html @@ -0,0 +1,2692 @@ + + + + + + + + + + + + + + + + + + + + + + + Notification Messages - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Notification Messages

+ +

Notification Message structure.

+

Notification messages are displayed in the notification panel which the user can access by clicking the Notifications button in the toolbar. The application specifies an array of messages using the notification methods in the canvas controller. The appearance of the panel is customized in the Notification Config.

+

The application passes to Common Canvas an array of messages each described by a simple JavaScript object like this: +

 {
+    id: <string>,
+    type: <string>,
+    title: <string or JSX object>,
+    content: <string or JSX object>,
+    timestamp: <string>,
+    callback: <function>,
+    closeMessage: <string or JSX object>
+}
+
+where:

+
    +
  • +

    id (string, required): this is a unique ID assigned to the notification message. This is passed as a parameter in the callback (see below) and is used to reference messages when deleting them through the CanvasController API.

    +
  • +
  • +

    type (string, required): this must be one of four values: “info”, “success”, “warning”, or “error”. If type is null, empty string, or undefined, the message type will be “unspecified”.

    +
  • +
  • +

    title (string or JSX object): the title of the notification message.

    +
  • +
  • +

    content (string or JSX object): the body of the notification message.

    +
  • +
  • +

    callback (function): an optional callback function that will be called when the notification message is clicked. Callback function is called with one parameter, id

    +
  • +
  • +

    timestamp (string or JSX object): an optional timestamp that will be rendered in a separate section with different formatting, if passed in

    +
  • +
  • +

    closeMessage (string or React object): an optional message that, if passed in, will display as clickable. Clicking on this will delete this individual message. If none is passed in, no delete option will be shown.

    +
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.04.06-styling-objects/index.html b/v13/03.04.06-styling-objects/index.html new file mode 100644 index 0000000000..41b72df362 --- /dev/null +++ b/v13/03.04.06-styling-objects/index.html @@ -0,0 +1,2659 @@ + + + + + + + + + + + + + + + + + + + + + + + Styling Canvas Objects - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Styling Canvas Objects

+

There are multiple was to specify the styles for the objects Common Canvas displays:

+
    +
  • A class name can be provided for nodes either within the pipelineFlow document or
  • +
  • The common-canvas default styes can be overriden within your CSS or SCSS file
  • +
+

For more precise styling of objects the style specification object can be used to provide a number of styles to Common Canvas to set the inline styles on numerous elements of an object. For example, a node is made up of: a selection outline; a node body (rectangle); an image, a label etc.

+

Style specifications may be applied to common-canvas objects using the following Canvas Controller methods: +

setObjectsStyle()
+setObjectsMultiStyle()
+setLinksStyle()
+setLinksMultiStyle()
+setSubdueStyle()
+
+and may be retrieved for objects using: +
getNodeStyle()
+getCommentStyle()
+getLinkStyle()
+

+

The style spec for a node can set the styles for all these elements in one shot. Here is a template of a style spec for a node: +

{
+    body: { default: <css_snippet>, hover: <css_snippet> },
+    image: { default: <css_snippet> },
+    label: { default: <css_snippet> },
+    selection_outline: { default: <css_snippet> }
+}
+ ```
+And here is a real example:
+```js
+{
+    body: {
+        default: "fill: coral; stroke: red;",
+        hover: "fill: cornflowerblue; stroke: blue;"
+    }
+};
+

+

Here is a template for styling a comment:

+
{
+    body: { default: <css_snippet>, hover: <css_snippet> },
+    text: { default: <css_snippet> },
+    selection_outline: { default: <css_snippet> },
+}
+
+

And here is a template for styling a link: +

{
+    line: { default: <css_snippet> , hover: <css_snippet>  }
+}
+

+
    +
  • <css_snippet> - is a string containing any CSS code that can be added inline to an SVG object. That means, for example, you need to use fill and stroke for colors etc It is recommended NOT to change the sizes of text fonts.
  • +
+

The CSS will be applied to the element of the object specified, either as the default inline style or as the style when the pointer hovers over the object.

+

When a hover style is applied to a graphical element it is applied in addition to the default style so there is no need to repeat styles in the hover <css_snippet> because they will augment the default style.

+

Because styles are applied as in-line styles they will override any styles provided in your application’s CSS and specified to Common Canvas through the class_name field of canvas objects.

+

If the <css_snippet> is specified as null the current style will be removed from the specified element of the object.

+

Finally, styles can be applied to your nodes, comments and links as either temporary or permanent styles by specifying the temporary boolean in the API methods. A temporary style is just applied for the duration of the session and are not persisted. You should use temporary=true styles that represent transient attributes of an object that should not be persisted in the pipelineFlow document. Use temporary=false styles for styles that you want to persist in the pipelineFlow document.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.05-keyboard-support/index.html b/v13/03.05-keyboard-support/index.html new file mode 100644 index 0000000000..509e3e3cf5 --- /dev/null +++ b/v13/03.05-keyboard-support/index.html @@ -0,0 +1,3354 @@ + + + + + + + + + + + + + + + + + + + + + + + Keyboard Support - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Keyboard support

+

Common Canvas supports a number of keyboard interactions as described below. Some keyboard shortcuts are only available if the config field enableKeyboardNavigation is set to true as indictaed above each table.

+

When any of the shortcut keys are pressed, if the shortcut has an action (listed below), Common Canvas will follow the same procedure as if the action was initiated from a context menu or from the canvas toolbar or by direct manipulation on the canvas. That is, it will: call the beforeEditActionHandler and the editActionHandler callbacks, with the data.editType parameter set to the action name and the data.editSource parameter set to “keyboard”; it will then update the object model with the change and refresh the flow editor display.

+

Note: In the tables below:

+
    +
  • “Meta” means either the Command key (⌘) on the Mac or, on Windows, the Windows key (⊞) or Control key (Ctrl).
  • +
  • “Alt” means either the Option key (⌥) on the Mac or, on Windows, the Alternative key (Alt).
  • +
+

Flow Editor

+

When focus is in the flow editor, either on the background or on a flow editor object

+

The shortcuts in this table are always available. The application can disable these actions by providing the keyboard config object to the common-canvas React component.

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
Meta + aselectAllSelects alll objects
Meta + Shift + adeselectAllDeselects all objects
[delete key]deleteSelectedObjectsDelete currently selected objects
[backspace key]deleteSelectedObjectsDelete currently selected objects
Meta + xcutCut selected objects to the clipboard
Meta + ccopyCopy selected objects to the clipboard
Meta + vpastePaste objects from the clipboard. If the mouse cursor is over
the canvas, objects will be pasted at the cursor position or,
if not, at a default position
Meta + zundoUndo last command
Meta + Shift + zredoRedo last undone command
Meta + yredoRedo last undone command
+

The shortcuts in this table are only available when the canvas config field enableKeyboardNavigation is set to true.

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[tab key]-Moves keyboard focus to the next group of objects in the flow editor
Shift + [tab key]-Moves keyboard focus to the previous group of objects in the flow editor
Meta + Shift + [plus key]zoomInZoom in the flow editor
Meta + Shift + [minus key]zoomOutZoom out the flow editor
Meta + Shift + [zero key]zoomToFitZooms to fit the flow obejcts within the flow editor viewport
Meta + Shift + [up arrow key]-Pans the flow obejcts within the flow editor viewport upwards
Meta + Shift + [down arrow key]-Pans the flow obejcts within the flow editor viewport downwards
Meta + Shift + [left arrow key]-Pans the flow obejcts within the flow editor viewport to the left
Meta + Shift + [right arrow key]-Pans the flow obejcts within the flow editor viewport to the right
Meta + [slash key]-Displays a content menu or context toolbar (depending on which is enabled) for the focused object
+ +

The shortcuts in this table are only available when the canvas config field enableKeyboardNavigation is set to true.

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[right arrow key]-Moves focus to next object in the group
[left arrow key ]-Moves focus to previous object in the group
[return key]-Selects the focused object
Meta + [return key]-Selects the focused object and adds it to the current set of selected objects
Shift + [return key]-Selects a range of nodes through from from the currently selected object to the focused object
[up arrow key]-When the focused object is a link, moves focus to the previous sibling link
[down arrow key]-When the focused object is a link, moves focus to the next sibling link
Meta + [slash key]-Displays a content menu or context toolbar (depending on which is enabled) for the focused object
Meta + [up arrow key]moveObjectsMoves the focused object, with any other selected objects, upwards
Meta + [down arrow key]moveObjectsMoves the focused object, with any other selected objects, downwards
Meta + [left arrow key]moveObjectsMoves the focused object, with any other selected objects, to the left
Meta + [right arrow key]moveObjectsMoves the focused object, with any other selected objects, to the right
Shift + [up arrow key]resizeObjectsReduces the height of the focused comment or node (if enableResizableNodes is true)
Shift + [down arrow key]resizeObjectsIncreases the height of the focused comment or node (if enableResizableNodes is true)
Shift + [left arrow key]resizeObjectsReduces the width of the focused comment or node (if enableResizableNodes is true)
Shift + [right arrow key]resizeObjectsIncreases the width of the focused comment or node (if enableResizableNodes is true)
Meta + Shift + [right angle bracket key]-When the focused object is a node, creates a link to it from the currently selected node or comment
Alt + [up arrow key]-When the focused object is a comment and contains scrollable text, scrolls the text down
Alt + [down arrow key]-When the focused object is a comment and contains scrollable text, scrolls the text up
+

Text Entry

+

The shortcuts in this table are only available when the canvas config field enableKeyboardNavigation is set to true.

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[esc key]-Cancels the text entry and discards any changes
Shift + [return key]-Completes the text entry and saves the changes made
[return key]-When allowReturnKey is set to “save”, completes the text entry and saves the changes made. Otherwise, it enters a new line into the text
[tab key]-When focus is on the text entry area, moves focus to the text toolbar
[tab key]-When focus is on the text toolbar, moves focus to the text entry area
Markdown text
Meta + b-Insert ‘bold’ syntax around the selected text
Meta + i-Insert ‘italics’ syntax around the selected text
Meta + Shift + x-Insert ‘strikethrough’ syntax around the selected text
Meta + Shift + 7-Insert ‘numbered list’ syntax around the selected text
Meta + Shift + 8-Insert ‘bulleted list’ syntax around the selected text
Meta + e-Insert ‘code’ syntax around the selected text
Meta + k-Insert ‘link’ syntax around the selected text
Meta + Shift + [right angle bracket]-Insert ‘quote’ syntax around the selected text
Meta + [right angle bracket]-Increases number of hashes in front of the selected text
Meta + [left angle bracket]-Decreases number of hashes in front of the selected text
+

Toolbar

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[right arrow key]-When focus is on a button in the toolbar, move focus to the button to the right of current focus position. When focus in on a menu item, opens any available sub-menu or sub-panel
[left arrow key]-When focus is on a button in the toolbar, move focus to the button to the left of current focus position. When focus is on a sub-menu or sub-panel, closes the area and moves focs to the parent menu.
[down arrow key]-When focus is on a button in the toolbar, opens sub-area (either a sub-menu or sub-panel) below button, if one is available. When focus is on a sub-menu, moves focus to the next menu entry.
[up arrow key]-When focus is on a sub-menu, moves focus to the previous menu entry.
[space bar]-Activate the button
[return key]-Activate the button
[esc key]-Close any open associated sub-area (either a sub-menu or sub-panel)
+

Palette

+

When focus is on the Search area

+ + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[tab key]-Moves focus to first category.
Shift + [tab key]-Moves focus out of the palette.
+

When focus is on a category

+ + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[tab key]-Moves focus to the next category.
Shift + [tab key]-Moves focus to the previous category.
[down arrow key]-When the category is open, moves focus to first node in the category.
+

When focus is on node in a category

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
Tab-Moves focus to the next category.
Shift + [tab key]-Moves focus to the parent category.
[down arrow]-Moves the focus down to next node in the category.
[up arrow]-Moves focus up to previous node in the category.
[space bar]createNodeAttachLinksAdds the node to the canvas and links it to an available existing node on canvas. Same as double clicking the node
Shift + [space bar]createNodeAttachLinks (addLinks: false)Adds the node to the canvas and does not create any links. Same as dragging a node onto the canvas.
+

Context toolbar / menu

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Keyboard ShortcutActionDescription
[down arrow key]-Move focus to next menu item below current focus position
[up arrow key]-Move focus to next menu item above current focus position
[right arrow key]-Opens cascade sub-area, if there is one, for the currenty focused item
[space bar key]-Activate the current menu item
[esc key]-Close the sub-area
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.06.01-node-customization/index.html b/v13/03.06.01-node-customization/index.html new file mode 100644 index 0000000000..839566b488 --- /dev/null +++ b/v13/03.06.01-node-customization/index.html @@ -0,0 +1,3448 @@ + + + + + + + + + + + + + + + + + + + + + + + Node Customization - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Node Customization

+

Node appearance and behavior can be customized using:

+ +

Customizing Layout fields

+

Node layout fields

+

Node layout properties define how all the elements of a node are displayed such as: the position and size of the icon image; the position of the main label; even the shape of the node itself.

+

There are two possible sets of node layout properties provided by Common Canvas, these are controlled by the enableNodeFormatType canvas configuration property which can be set to either “Horizontal” or “Vertical”.

+

+

Default values for node layout properties

+

The possible node layout properties are shown below with the values they have when enableNodeFormatType = "Horizontal". You can see the values for both sets of properties by looking at the layout-dimensions.js program

+
+Default node layout fields for Horizontal node format +
// Default node sizes. These dimensions might be overridden for nodes that have
+// more ports than will fit in the default size if inputPortAutoPosition is
+// set to true and outputPortAutoPosition is set to true. (See below).
+defaultNodeWidth: 160,
+defaultNodeHeight: 40,
+
+// A space separated list of classes that will be added to the group <g>
+// DOM element for the node.
+className: "",
+
+// Displays the node outline shape underneath the image and label.
+nodeShapeDisplay: true,
+
+// Default node shape. Can be "rectangle" or "port-arcs". Used when nodeShapeDisplay is true.
+nodeShape: "port-arcs",
+
+// An SVG path or a function that returns an SVG path. The paths define the node
+// shape and its selection highlighting respectively. If set to null, the paths
+// will be set by default based on the nodeShape setting.
+// If these fields are set to functions they will be called in real-time as the node
+// is being sized (provided enableResizableNodes config field is set to true).
+bodyPath: null,
+selectionPath: null,
+
+// Displays the external object specified, as the body of the node
+nodeExternalObject: false,
+
+// Display image
+imageDisplay: true,
+
+// Image dimensions
+imageWidth: 26,
+imageHeight: 26,
+
+// Image position
+imagePosition: "topLeft",
+imagePosX: 6,
+imagePosY: 7,
+
+// Display label
+labelDisplay: true,
+
+// Label dimensions
+labelWidth: 112,
+labelHeight: 19,
+
+// Label position
+labelPosition: "topLeft",
+labelPosX: 36,
+labelPosY: 12,
+
+// Label appearance
+labelEditable: false,
+labelAlign: "left", // can be "left" or "center"
+labelSingleLine: true, // false allow multi-line labels
+labelOutline: false,
+labelMaxCharacters: null, // null allows unlimited characters
+labelAllowReturnKey: false, // true allows line feed to be inserted into label, "save" to make the return key save the label.
+
+// An array of decorations to be applied to the node. For details see:
+// https://elyra-ai.github.io/canvas/03.04.01-decorations/
+// These are added to the node at run time and will not be saved into
+// the pipeline flow.
+decorations: [],
+
+// Positions and dimensions for 9 enumerated default decorator positions.
+// decoratorWidth and decoratorHeight are the dimensions of the outline
+// rectangle and decoratorPadding is the padding for the image within the
+// outline rectangle.
+decoratorTopY: 2,
+decoratorMiddleY: -8,
+decoratorBottomY: -18,
+
+decoratorLeftX: 2,
+decoratorCenterX: -8,
+decoratorRightX: -30,
+
+// Width, height and padding for image decorators
+decoratorWidth: 16,
+decoratorHeight: 16,
+decoratorPadding: 2,
+
+// Width and height for label decorators
+decoratorLabelWidth: 80,
+decoratorLabelHeight: 30,
+
+// Display drop shadow under and round the nodes
+dropShadow: true,
+
+// The gap between a node and its selection highlight rectangle
+nodeHighlightGap: 1,
+
+// The size of the node sizing area that extends around the node, over
+// which the mouse pointer will change to the sizing arrows.
+nodeSizingArea: 10,
+
+// Error indicator dimensions
+errorPosition: "topLeft",
+errorXPos: 24,
+errorYPos: 5,
+errorWidth: 10.5,
+errorHeight: 10.5,
+
+// When sizing a supernode this decides the size of the corner area for
+// diagonal sizing.
+nodeCornerResizeArea: 10,
+
+// What point to draw the data links from and to when enableLinkType is set
+// to "Straight" and enableLinkMethod is set to "Freeform".
+// Possible values are "image_center" or "node_center".
+drawNodeLinkLineFromTo: "node_center",
+
+// What point to draw the comment to node link line to. Possible values
+// are "image_center" or "node_center".
+drawCommentLinkLineTo: "node_center",
+
+// This is the size of the horizontal line protruding from the
+// port on the source node when drawing an elbow or straight connection line.
+minInitialLine: 30,
+
+// For the elbow connection type with nodes with multiple output ports,
+// this is used to increment the minInitialLine so that connection lines
+// do not overlap each other when they turn up or down after the elbow.
+minInitialLineIncrement: 8,
+
+// This is the minimum size of the horizontal line entering the
+// target port on the target node when drawing an Elbow connection line.
+minFinalLine: 30,
+
+// Display input ports.
+inputPortDisplay: true,
+
+// An array of elements to control display of input ports. Each element
+// can have a number of different structures like this:
+// Either
+// { type: "circle" } // Can also be "circleWithArrow"
+// Or
+// { type: "image", src: "path/picture.svg", width: 10, height: 10 }
+// Or
+// { type: "jsx", src: (<FaceCool />), width: 16, height: 16 }
+//
+// The order of the elements corresponds to the order of ports in the
+// inputs array for the node. If there are more input ports than elements
+// in the array, the last element will be used for all remaining ports.
+inputPortDisplayObjects: [
+    { type: "circleWithArrow" }
+],
+
+// Indicates whether multiple input ports should be automatically
+// positioned (true) or positioned based on the contents of
+// inputPortPositions array (false).
+inputPortAutoPosition: true,
+
+// An array of input port positions. Each element is structured like
+// this: { x_pos: 5, y_pos: 10, pos: "topLeft" }. x_pos and y_pos are
+// offsets from the pos point on the node.
+// The order of the elements corresponds to the order of ports in the
+// inputs array for the node. If there are more input ports than elements
+// in the array, the last element will be used for all remaining ports.
+inputPortPositions: [
+    { x_pos: 0, y_pos: 20, pos: "topLeft" }
+],
+
+// An array of elements to control display of input port guide objects.
+// That is the object drawn at the end of a new link as it is being dragged.
+// Each element can have a number of different structures like this:
+// Either
+// { type: "circle" } // Can also be "circleWithArrow"
+// Or
+// { type: "image", src: "path/picture.svg", width: 10, height: 10 }
+// Or
+// { type: "jsx", src: (<FaceCool />), width: 16, height: 16 }
+//
+// The order of the elements corresponds to the order of ports in the
+// inputs array for the node. If there are more input ports than elements
+// in the array, the last element will be used for all remaining ports.
+inputPortGuideObjects: [
+    { type: "circle" }
+],
+
+// Display output ports.
+outputPortDisplay: true,
+
+// An array of elements to control display of output ports. Each element
+// can have a number of different structures like this:
+// Either
+// { type: "circle" } // Can also be "circleWithArrow"
+// Or
+// { type: "image", src: "path/picture.svg", width: 10, height: 10 }
+// Or
+// { type: "jsx", src: (<FaceCool />), width: 16, height: 16 }
+//
+// The order of the elements corresponds to the order of ports in the
+// outputs array for the node. If there are more output ports than elements
+// in the array, the last element will be used for all remaining ports.
+outputPortDisplayObjects: [
+    { type: "circle" }
+],
+
+// Indicates whether multiple output ports should be automatically
+// positioned (true) or positioned based on the contents of
+// outputPortPositions array (false).
+outputPortAutoPosition: true,
+
+// An array of output port positions. Each element is structured like
+// this: { x_pos: 5, y_pos: 10, pos: "topRight" }. x_pos and y_pos are
+// offsets from the pos point on the node.
+// The order of the elements corresponds to the order of ports in the
+// outputs array for the node. If there are more output ports than elements
+// in the array, the last element will be used for all remaining ports.
+outputPortPositions: [
+    { x_pos: 0, y_pos: 20, pos: "topRight" }
+],
+
+// An array of elements to control display of output port guide objects.
+// That is the object drawn at the end of a new link as it is being dragged.
+// Each element can have a number of different structures like this:
+// Either
+// { type: "circle" } // Can also be "circleWithArrow"
+// Or
+// { type: "image", src: "path/picture.svg", width: 10, height: 10 }
+// Or
+// { type: "jsx", src: (<FaceCool />), width: 16, height: 16 }
+//
+// The order of the elements corresponds to the order of ports in the
+// outputs array for the node. If there are more output ports than elements
+// in the array, the last element will be used for all remaining ports.
+outputPortGuideObjects: [
+    { type: "circle" }
+],
+
+// Automatically increases the node size to accommodate its ports so both
+// input and output ports can be shown within the dimensions of
+// the node.
+autoSizeNode: true,
+
+// Radius of the either the input or output ports when they are set to "circle"
+portRadius: 3,
+
+// Size of an offset above and below the set of port arcs.
+portArcOffset: 3,
+
+// Radius of an imaginary circle around the port. This controls the
+// spacing of ports and the size of port arcs when nodeShape is set to
+// port-arcs.
+portArcRadius: 6,
+
+// Spacing between the port arcs around the ports.
+portArcSpacing: 3,
+
+// Position of the context toolbar relative to the node. Some adjustment
+// will be made to account for the width of the toolbar.
+contextToolbarPosition: "topRight",
+
+// Display of vertical ellipsis to show context menu
+ellipsisDisplay: true,
+ellipsisPosition: "topLeft",
+ellipsisWidth: 10,
+ellipsisHeight: 22,
+ellipsisPosX: 145,
+ellipsisPosY: 9,
+ellipsisHoverAreaPadding: 2
+
+
+

Node Element positioning

+

Node elements are positioned on the node as an x/y offset from one of nine positions:

+

+

The default for most elements is topLeft. The position is useful when nodes are resizable because, as a node is resized, the element will remain tied to its position. So if, for example, an element is tied to topRight and the node is resized to be wider the element will move to remain at the same offset from the topRight position.

+

The PosX and PosY properties for each element is an offset from the associated anchor position where PosX is the number of pixels to the right of the anchor position and PosY is a number of pixels down from the anchor position. Negative values can be provided to specify an offset to the left and up from the anchor position.

+

For example, these settings: +

    {
+        imagePosition: "middleCenter",
+        imagePosX: -10,
+        imagePosY: -10,
+        imageWidth: 20,
+        imageHeight: 20
+    }
+};
+
+would position the image 10 pixels left and 10 pixels above the very center of the node. Since the image is 20 x 20 pixels this would position the center of the image at the center of the node. If you have enabled re-sizeable nodes, this would keep the image centrally positioned while the node is being resized by the user. Like this:

+

+

Overriding the node layout properties for all nodes

+

If you want to change the appearance of all nodes on your canvas you can specify the enableNodeLayout configuration parameter in the canvas configuration object. The properties from this object will replace any properties in the default set, which was chosen based on the settings of enableNodeFormatType. So you don’t need to provide all of the properties; just the ones you want to replace.

+

Let’s say you want your nodes to be displayed as ellipses. You could provide the following settings in enableNodeLayout in the canvas config: +

const canvasConfig = {
+    enableNodeLayout: {
+        bodyPath: "     M  0 30 Q  0  0 60  0 Q 120  0 120 30 Q 120 60 60 60 Q  0 60  0 30 Z",
+        selectionPath: "M -5 30 Q -5 -5 60 -5 Q 125 -5 125 30 Q 125 65 60 65 Q -5 65 -5 30 Z",
+        defaultNodeWidth: 120,
+        defaultNodeHeight: 60,
+        imageWidth: 30,
+        imageHeight: 30,
+        imagePosX: 20,
+        imagePosY: 10,
+        labelEditable: true,
+        labelPosX: 60,
+        labelPosY: 37,
+        labelWidth: 90,
+        labelHeight: 17, // Should match the font size specified in CSS + padding
+        ellipsisDisplay: true,
+        ellipsisPosX: 100,
+        ellipsisPosY: 20,
+        portPosY: 30
+    }
+};
+

+

Overriding the node layout fields for individual nodes or groups of nodes

+

If you want each node, or category of nodes, to have a different layout based on some criteria you can use the layoutHandler callback method. When you specify this callback method to Common Canvas, it will be called for each node on the canvas, during initialization and, occasionally, at other times.

+

The method should return a simple JavaScript object that contains any node layout properties you want to override from the defaults and the ones specified in the enableNodeLayout field in the canvas config.

+
+

Info

+

There are three levels of properties provided where each overrides the previous set:

+
    +
  1. First Common Canvas takes the full default set of node layout fields based on the value for enableNodeFormatType.
  2. +
  3. Next Common Canvas overrides these with the fields from the enableNodeLayout object in the canvas config, if any are provided.
  4. +
  5. Finally, Common Canvas overrides the combined set with any fields from the object returned from the layoutHandler method if one is specified, for the node in question.
  6. +
+
+

The callback is provided with a data parameter which is the node object from the pipelineFlow so your code can examine the node object and return node layout properties as appropriate.

+
+

Tip

+

The layoutHandler callback is called while the canvas is being displayed, therefore it must return very quickly each time it is called otherwise your canvas display speed will be slowed down.

+
+

Here is a simple example of a layoutHandler callback method which will override the width of the node based on the width of the main label for any node where the node’s op field is set to Sort: +

layoutHandler(data) {
+    let customNodeLayout = {};
+    if (data.op === "Sort") {
+        const labLen = data.label ? data.label.length : 0;
+        const width = (labLen * 9) + 30; // Allow 9 pixels for each character and a bit extra for padding
+        customNodeLayout = {
+            defaultNodeWidth: width // Override default width with calculated width
+        };
+    }
+    return customNodeLayout;
+

+ +

The four options for enableLinkDirection in the canvas config are “LeftRight”, “RightLeft”, “TopBottom” and “BottomTop”. These will control the default position of the input and output ports at the boundaries of the nodes as follows:

+
    +
  • For “LeftRight” input ports will be on the left of the node and output ports will be on the right of the node
  • +
  • For “RightLeft” input ports will be on the right of the node and output ports will be on the left of the node
  • +
  • For “TopBottom” input ports will be on the top of the node and output ports will be on the bottom of the node
  • +
  • For “BottomTop” input ports will be on the bottom of the node and output ports will be on the top of the node
  • +
+

The application can customize these port default positions using the enableNodeLayout object in the canvas config. This object has these fields for port placement customization:

+
    +
  • inputPortAutoPosition and inputPortPositions to customize input ports and
  • +
  • outputPortAutoPosition and outputPortPositions to customize output ports.
  • +
+

So for example, the following settings:

+
    const config = {
+        enableNodeLayout: {
+            inputPortAutoPosition: false,
+            inputPortPositions: [
+                { x_pos: 0, y_pos: 5, pos: "topLeft" },
+                { x_pos: 0, y_pos: -5, pos: "bottomLeft" }
+            ]
+        }
+    };
+
+

will position the input ports like this:

+

+

If nodes are set to be resizeable (enableResizableNodes set to true) and the nodes is resized it will look like this, because the second port is tied to the “bottomLeft” position:

+

+

When ports are positioned on the node, regardless of whether they are positioned by default or have customized positions, they are assigned one of four ‘directions’ that links will be drawn to/from. These directions are dependent on the diagonal quadrant of the node the port is positioned within.

+

As an extreme example, if these fields are used to customize a node with four input ports (black circles) and four output ports (blue arrows) like this:

+

+

Common Canvas will assign a direction to the input and output ports are follows:

+

+

If the config field enableLinkMethod is set to “Ports”, and enableLinkType is set to “Curve”, the links will be drawn as follows:

+

+

This means an application like the one shown below can be built, where the links exit from the left of the ‘Inp_1’ node and point to the ‘Stage Variables’ node below it. Meanwhile, other links exist the ‘Inp_1’ node on its right and point to the ‘Out_2’ node. This is achieved by positioning some (invisible) output nodes on the left of ‘Inp_1’ and some on the right of ‘Inp_1’. Common Canvas takes care of drawing the links in the appropriate direction based on the port positions.

+

+

Node Images

+

For most flows, nodes are defined with an image (icon) that conveys the purpose of the node. They are displayed for the nodes in the flow and the palette. The nodes image can be provided as a URL or JSX object. Images can also be customized using node layout fields.

+

Node image layout fields

+

The following fields can be set in the node layout to customize images within a node: +

    // Display image
+    imageDisplay: true,
+
+    // Image dimensions
+    imageWidth: 26,
+    imageHeight: 26,
+
+    // Image position
+    imagePosition: "topLeft",
+    imagePosX: 6,
+    imagePosY: 7,
+

+

Node image as a URL

+

Typically, the image field of a node object in the pipeline flow (or palette) is a URL that references an image file on the server. This can be any type of image that can be displayed in an <image> tag however, if the file is recognized as an SVG file by having the .svg extension, the contents of the file will be loaded as inline SVG in the DOM. This allows more image customization using CSS.

+

The loading and management of images can be controlled using the enableImageDisplay canvas config field.

+
+

Data URLs

+

Data URLs can be used to embed an image file within the pipeline flow JSON however, this is not recommended for production use since it leads to bloated pipeline flow files because of the amount of data stored for each image and the repetition of images across multiple nodes of the same type.

+
+
+

Alternative palette image

+

The node definition contains a palette image field which, if specified, will be displayed as the node’s image only on the palette.

+
+

Node image as JSX

+

Images can also be provided to Common Canvas as JSX objects. This means, for example, that the application can provide Carbon icons as node images in the palette and on the canvas. Since JSX objects cannot be stringified, the application must programmatically set the node image field to reference the appropriate JSX object.

+

Also, when data is retrieved from Common Canvas using either canvasController.getPipelineFlow() or canvasController.getPaletteData(), the returned node object will not contain any image information.

+

The node image can be set something like this (where “123” is the node Id):

+

    import { JoinInner } from “@carbon/react/icons:
+    ...
+    ...
+    canvasController.setNodeProperties("123", { image: (<JoinInner size={20} />) });
+
+Alternatively, JSX images can be injected directly into the pipeline flow object or palette object before it is provided to Common Canvas.

+
+

Size prop

+

Note, the size prop is used when the icon is displayed in the palette however for a node in the flow the icon will be sized based on the imageHeight and imageWidth fields in the nodeLayout object specified in the enableNodeLayout field of the canvas config object.

+
+

Also, when nodes are pasted into the canvas the application will need to ensure the image fields of those nodes are set appropriately.

+

A new sample application called ‘JSX Icons’ has been added to the test harness to show Carbon icons being used as node images.

+

+

Customizing node colors and styles with CSS

+

Node DOM Construction

+

Node elements are drawn as SVG elements in the SVG area provided by the Flow Editor and are grouped together using a group <g> element.

+

+

The DOM elements that make up a node can be customized using CSS styles. This is done by either overriding the common-canvas CSS directly or, if customization is needed on a node-by-node basis, by assigning a class name to the group <g> element that is the container for all the node elements. The class can be applied to the group object in a number of different ways:

+
    +
  1. By specifying it in the app_data.ui_data.class_name field of the node in the pipeline flow document that is provided to Common Canvas using CanvasController.setPipelineFlow(pFlow)
  2. +
  3. By specifying it using the following API methods:
      +
    • CanvasController.setNodeProperties(nodeId, properties, pipelineId)
    • +
    • CanvasController.setNodesClassName(nodeIds, newClassName, pipelineId)
    • +
    +
  4. +
  5. By specifying a class name in the className field of the node layout properties in the canvas config. Like this +
    const canvasConfig = {
    +    enableNodeLayout: {
    +        className: "my-node-class"
    +    }
    +};
    +
  6. +
  7. By specifying a class name in the className field of the node layout properties returned from the layoutHandler.
  8. +
+

You can see the svg-canvas-d3.scss file for full details about what elements in the node can be styled but here are a list of some basic parts of the node:

+

Classes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PurposeDOM tagClassesNotes
Groupgd3-node-groupClasses specified for the node in the class_name field of the node object will be added here.
⮕ Selection areapathd3-node-selection-highlight
⮕ Outline shapepathd3-node-body-outline
⮕ Imagesvgd3-node-image
⮕ LabelforeignObjectd3-foreign-object-node-labelWill contain a div that contains the label text
⮕ Input portgd3-node-port-input
⮕ Output portgd3-node-port-output
⮕ Decorationsgd3-node-decorations-groupWill contain decoration elements, for example, image, path etc
+

So for example if you want the node body (the rectangle) to be colored orange you would provide a class name to the group element using one of the techniques mentioned above and then put this in you CSS:

+

.my-node-class .d3-node-body-outline {
+   fill: orange;
+}
+
+Note: You can use the enableParentClass canvas config field to make you CSS rulesets specific so your styles are picked up in preference to the common-canvas default styles.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.06.02-comment-customization/index.html b/v13/03.06.02-comment-customization/index.html new file mode 100644 index 0000000000..cf1a883915 --- /dev/null +++ b/v13/03.06.02-comment-customization/index.html @@ -0,0 +1,2775 @@ + + + + + + + + + + + + + + + + + + + + + + + Comment Customization - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Comment Customization

+

Comments can be customized by the application in the following ways:

+
    +
  • Setting the enableMarkdownInComments field in the canvas configuration.
  • +
  • Customizing comment appearance with CSS (see below).
  • +
+

Customizing comment appearance with CSS

+

Comment DOM Construction

+

Comment elements are drawn as SVG elements in the SVG area provided by the Flow Editor and are grouped together using a group <g> element.

+

+

The DOM elements that make up a comment can be customized using CSS styles. This is done by either overriding the common-canvas CSS directly or, if customization is needed on a comment-by-comment basis, by assigning a class name to the group <g> element that is the container for all the comment elements. The class can be applied to the group object in a number of different ways:

+
    +
  1. By specifying it in the app_data.ui_data.class_name field of the comment in the pipeline flow document that is provided to Common Canvas using CanvasController.setPipelineFlow(pFlow)
  2. +
  3. By specifying it using the following API methods:
      +
    • CanvasController.setCommentProperties(commentId, properties, pipelineId)
    • +
    • CanvasController.setCommentsClassName(commentIds, newClassName, pipelineId)
    • +
    +
  4. +
+

You can see the svg-canvas-d3.scss file for full details about what elements in the comment can be styled but here are a list of some basic parts of the comment:

+

Classes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PurposeDOM tagClassesNotes
Groupgd3-comment-groupClasses specified for the comment in the class_name field of the node object will be added here.
⮕ Sizing areapathd3-comment-sizing
⮕ Selection areapathd3-comment-selection-highlight
⮕ Backgroundpathd3-comment-rect
⮕ TextforeignObjectd3-foreign-object-comment-textWill contain a div that contains the comment text
⮕ Decorationsgd3-comment-decorations-groupWill contain decoration elements, for example, image, path etc
+

Note: You can use the enableParentClass canvas config field to make you CSS rulesets specific so your styles are picked up in preference to the common-canvas default styles.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.06.03-link-customization/index.html b/v13/03.06.03-link-customization/index.html new file mode 100644 index 0000000000..82e91846b1 --- /dev/null +++ b/v13/03.06.03-link-customization/index.html @@ -0,0 +1,2905 @@ + + + + + + + + + + + + + + + + + + + + + + + Link Customization - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Link Customization

+

Common Canvas provides a number of ways the links can be displayed. Links can be customized by the application in the following ways:

+
    +
  • Setting the link fields in the canvas configuration.
  • +
  • Customizing comment appearance with CSS (see below).
  • +
+ +

The two major config fields to customize link display are:

+
    +
  • enableLinkType that can be set to “Curve”, “Elbow”, “Parallax” or “Straight”
  • +
  • enableLinkMethod that can be set to “Ports” or “Freeform”.
  • +
+ +

When enableLinkMethod is set to “Ports”, the positions of ports on nodes affect the direction that links are drawn to/fom the ports. See this overriding port positions section in the node customization page for more details.

+

Customizing comment appearance with CSS

+ +

Links are drawn on the canvas using SVG elements in the DOM. Each link has a top level group <g> element and inside it some SVG paths. The first displayed path is the selection area. This is invisible but provides a selection/hover area for mouse interactions on the link. The second is a path to represent the link itself which is drawn over the top of the selection area path:

+

+

The DOM elements that make up a link can be customized using CSS styles. This is done by either overriding the common-canvas CSS directly or, if customization is needed on a link-by-link basis, by assigning a class name to the group <g> element that is the container for all the link elements. The class can be applied to the group object in a number of different ways:

+
    +
  1. By specifying it in the app_data.ui_data.class_name field of the link in the pipeline flow document that is provided to Common Canvas using CanvasController.setPipelineFlow(pFlow)
  2. +
  3. By specifying it using the following API methods:
      +
    • CanvasController.setLinkProperties(linkId, properties, pipelineId)
    • +
    • CanvasController.setLinksClassName(linkIds, newClassName, pipelineId)
    • +
    +
  4. +
+

You can see the svg-canvas-d3.scss file for full details about what elements in the comment can be styled but here are a list of some basic parts of the comment:

+

Classes

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PurposeDOM tagClassesNotes
Groupgd3-link-groupClasses specified for the link in the class_name field of the link object will be added here.
⮕ Selection areapathd3-link-selection-area
⮕ Link linepathd3-link-line
⮕ Arrow headpathd3-link-line-arrow-headOnly when enableLinkType is set to “Straight”
⮕ Decorationsgd3-link-decorations-groupWill contain decoration elements, for example, image, path etc
+

Note: The main link group will also have one of these classes: d3-node-link (for data links), d3-object-link (for association links) or d3-comment-link (for comment links).

+

Note: You can use the enableParentClass canvas config field to make you CSS rulesets specific so your styles are picked up in preference to the common-canvas default styles.

+ +

If a data link is retrieved from the canvas controller API it will have the following important fields:

+
    +
  • id - the unique identifier for the link.
  • +
  • type - set to “nodeLink”.
  • +
  • srcNodeId - the ID of the node the link is connected from.
  • +
  • srcNodePortId - the ID of the output port on the source node the link is connected from. Note: If this is undefined it indicates the node is connected to the first output port of the source node.
  • +
  • trgNodeId - the ID of the node the link is connected to.
  • +
  • trgNodePortId - the ID of the input port on the target node the link is connected to. Note: If this is undefined it indicates the node is connected to the first input port of the target node.
  • +
  • decorations - an array of decorations specified for the link.
  • +
  • app_data - any application specific data that was previously specified for the link in the pipeline flow or through the canvas controller API.
  • +
+

Note: Typically data links must be drawn between nodes however, if the config field enableLinkSelectionType is set to Detachable, the links are allowed to be drawn to and/or from arbitrary points on the canvas. If a link is drawn either semi-detached (from one node) or fully-detached (from both nodes) the following fields will be in the link object:

+
    +
  • srcPos - this is an object with two fields x_pos and y_pos. These provide the coordinates of the point on the canvas that the link is drawn from. If this exist then srcNodeId and srnNodePortId are not specified in the link object.
  • +
  • trgPos - this is an object with two fields x_pos and y_pos. These provide the coordinates of the point on the canvas that the link is drawn to. If this exist then srcNodeId and srnNodePortId are not specified in the link object.
  • +
+ +

(Note : internally, association links do have a srcNodeId and trgNodeId but that is just to keep the field names consistent with the data links.) Association links do not reference ports.

+

If an association link is retrieved from the canvas controller API it will have the following important fields:

+
    +
  • id - the unique identifier for the link.
  • +
  • type - set to “associationLink”.
  • +
  • srcNodeId - the ID of one of the nodes in the association.
  • +
  • trgNodeId - the ID of the other node in the association.
  • +
  • decorations - an array of decorations specified for the link.
  • +
  • app_data - any application specific data that was previously specified for the link in the pipeline flow or through the canvas controller API.
  • +
+ +

If a comment link is retrieved from the canvas controller API it will have the following important fields:

+
    +
  • id - the unique identifier for the link
  • +
  • type - set to “commentLink”
  • +
  • srcNodeId - the ID of comment.
  • +
  • trgNodeId - the ID of the node the comment is connected to.
  • +
  • app_data - any application specific data that was previously specified for the link in the pipeline flow or through the canvas controller API.
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.07-external-objects/index.html b/v13/03.07-external-objects/index.html new file mode 100644 index 0000000000..5c24a9d92c --- /dev/null +++ b/v13/03.07-external-objects/index.html @@ -0,0 +1,2740 @@ + + + + + + + + + + + + + + + + + + + + + + + Node Creation from External Object - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+ +
+ + + +
+
+ + + + + + + + + + + + +

Node Creation from External Object

+ +

Dragging object from within the browser

+

Common Canvas supports the ability for an object from the your browser page to be dragged onto the canvas to initiate the creation of a node on the canvas. To do this you need to set the object you want to drag into the canvas to be draggable:

+

    <div
+      draggable="true"
+      onDragStart={this.onDragStart}
+      onDragOver={this.onDragOver}>
+      ....
+      ....
+    </div>
+
+and then specify the on drag behavior like this +
    onDragStart(ev) {
+       const evData = {
+          operation: "addToCanvas",
+          data: {
+             editType: "createExternalNode",
+             field1: "field_val_1",
+             field2: "field_val_2"
+             ....
+             ....
+          }
+       };
+       ev.dataTransfer.setData("text", JSON.stringify(evData));
+    }
+
+where

+
    +
  • operation - is always set to “addToCanvas”
  • +
  • data - is the object that will be passed to the editActionHandler callback
  • +
  • editType - this is the type of editing operation. You can set it to anything except any of the reserved settings which are: ‘createNode’, ‘createNodeOnLink’, ‘createAutoNode’ and ‘createFromExternalObject’.
  • +
  • fields - an optional number of fields can be provided that describe the object you are dragging onto the canvas. For example, if it is a data object the fields might describe the data source details.
  • +
+

After the object has been dropped on the canvas your editActionHandler() callback method will be called with a parameter data object that contains the fields you specified in data in the drag data along with three additional fields, called pipelineId, offsetX and offsetY, containing the x,y co-ordinates of where the drop occurred. In your editActionHandler() method you can use the CanvasController API to add a node to the pipeline flow and get Common Canvas to display it.

+

Dragging object from the desktop or another application

+

If an object from the desktop or another application is dropped on the canvas your editActionHandler(data) method will be called with the data parameter set to an object like this: +

    {
+       dataTransfer: <The event data from the drag operation>,
+       editType: "createFromExternalObject",
+       editSource: "canvas",
+       offsetX: 200,
+       offsetY: 100,
+       pipelineId: "1234-5678"
+    }
+
+Your code can examine the dataTransfer object to see what object was dragged onto the canvas and then take appropriate action.

+

If you want a new node to appear on the canvas as a result of the object being dropped your code will need to create that node at the point where the drop occurred, using the Canvas Controller API. Here is some sample code that will

+
    +
  • create a new node template based on the “variablefile” node in the palette data
  • +
  • set the label of the node to be created to the name of the file being dropped
  • +
  • create a new node on the canvas at the offsetX, offsetY position
  • +
  • the command will be added to the command-stack so the user can click undo to undo the addition of the node
  • +
+

data is the parameter passed into your editActionHandler method. +

    if (data.editType === "createFromExternalObject") {
+        const nodeTemplate = canvasController.getPaletteNode("variablefile");
+        if (nodeTemplate) {
+            const convertedTemplate = canvasController.convertNodeTemplate(nodeTemplate);
+            convertedTemplate.label = data.dataTransfer.files[0].name;
+            const action = {
+                editType: "createNode",
+                nodeTemplate: convertedTemplate,
+                pipelineId: data.pipelineId,
+                offsetX: data.offsetX,
+                offsetY: data.offsetY
+            };
+            canvasController.editActionHandler(action);
+    }
+

+

Tip: You can optionally instruct Common Canvas to display a graphic over the canvas as the external object is being dragged over it. To do this you need to specify the enableDropZoneOnExternalDrag configuration parameter.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.08-external-pipeline-flows/index.html b/v13/03.08-external-pipeline-flows/index.html new file mode 100644 index 0000000000..e8ffac2856 --- /dev/null +++ b/v13/03.08-external-pipeline-flows/index.html @@ -0,0 +1,2857 @@ + + + + + + + + + + + + + + + + + + + + + + + External Pipeline Flows - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

External Pipeline Flows

+

Introduction

+

The pipeline flow schema describes two ways of storing a sub-flow pipelines for a supernode: local or external. Local sub-flows are stored within the pipeline flow document whereas external pipelines are stored in a separate pipeline flow document (usually as the primary pipeline in that pipeline flow). The supernode that references an external pipeline contains a url property (in <node>.subflow_ref.url) and a pipeline ID property (in <node>.subflow_ref.pipeline_id_ref) to identify the external pipeline flow and the pipeline within it.

+

Common Canvas supports external pipelines but the host application UI code needs to manage the storage and retrieval of any external pipeline flows. To do this the app UI code needs to:

+
    +
  • Manage the storage of newly created external pipeline flows.
  • +
  • Respond to calls from Common Canvas to provide the external pipeline flow(s) when requested.
  • +
  • Respond to call from Common Canvas when a local pipeline is changed to external or an external is changed to local.
  • +
+

Common Canvas supports lazy loading of external pipeline flows so they will only be requested from your code when the user performs some gesture that requires the pipeline (from within the external pipeline flow) to be displayed.

+

Creating an external sub-flow

+

When the common-canvas config property enableExternalPipelineFlows is set to true (the default) and, when a set of objects are selected from which a super node can be created, the default common-canvas context menu will include a Create External Supernode option.

+

When the Create External Supernode option is clicked the createSuperNodeExternal action is executed.

+

If your application doesn’t use the default context menu you can define your own context menu (returned from contextMenuHandler) to contain an option which maps to the createSuperNodeExternal action. See the section on the contextMenuHandler for details on how to do this.

+

When the createSuperNodeExternal action is executed, the beforeEditActionHandler callback is called before the external sub-flow is created. The beforeEditActionHandler is called where the first parameter data has two properties externalUrl and externalPipelineFlowId which will both be set to empty string. Your code must set these to whatever values you want for the url and pipeline flow ID. The url will be assigned to the subflow_ref.url property of the supernode that is being created. The pipeline flow ID will be assigned to the newly created pipeline flow.

+

Your code must return the data parameter from the beforeEditActionHandler callback if you want the action to proceed and create the sub-flow. If you need to do any asynchronous activity at this point see the documentation on the beforeEditActionHanlder for details on how to do that.

+

When the sub-flow has been created, Common Canvas will call the editActionHandler callback with the createSuperNodeExternal action. In this callback you can, if you wish, retrieve the pipeline flow document that has been created internally in Common Canvas using CanvasController.getExternalPipelineFlow(url). Your code can then save it to your repository. Alternatively, you can wait until some later time, like perhaps during an auto-save, to retrieve and store the pipeline flow externally in your repository.

+

Loading an external sub-flow

+

When the main pipeline flow, displayed by Common Canvas, contains a super node that references an external sub-flow it will need to be loaded whenever the user performs a gesture that causes it to be displayed or processed in some way - for example displaying it ‘in-place’ or converting it from an external to a local supernode. An external sub-flow will also need to be loaded if the top-level pipeline being displayed has a supernode, that refers to an external pipeline, that is already expanded in-place in the saved pipeline flow JSON document being displayed. So, actions that can cause the external pipeline to be loaded are:

+
    +
  • loadPipelineFlow
  • +
  • expandSuperNodeInPlace
  • +
  • displaySubPipeline
  • +
  • convertSuperNodeExternalToLocal
  • +
  • deconstructSuperNode
  • +
+

When any of these actions are performed Common Canvas will call the beforeEditActionHandler callback with the data parameter as the first parameter. The data object will have the following properties:

+
    +
  • editType - The name of the action being performed.
  • +
  • externalPipelineFlowLoad - This is a boolean which indicates whether the pipeline flow needs to be provided by your code.
  • +
  • externalUrl - This is the string which identifies the external pipeline flow document.
  • +
  • externalPipelineId - This is the ID of the pipeline being loaded
  • +
  • externalPipelineFlow - If externalPipelineFlowLoad if true this will be undefined. Otherwise it will be contain the previously loaded external pipeline flow.
  • +
+

You need to implement the beforeEditActionHandler so that:

+
    +
  • when the actions above are being performed and externalPipelineFlowLoad is true, you retrieve the external pipeline flow from your repository
  • +
  • you then assign it to the externalPipelineFlow property of the data object
  • +
  • you then return the data object from the callback.
  • +
+

Your code must return the data parameter from the beforeEditActionHandler callback if you want the action to proceed and load the external pipeline flow. If you need to do any asynchronous activity at this point see the documentation on the beforeEditActionHanlder for details on how to do that.

+

Converting a local supernode to an external supernode

+

When the common-canvas config property enableExternalPipelineFlows is set to true, and a local supernode’s is right clicked, the default common-canvas context menu will include a Convert local to external option. This will execute the convertSuperNodeLocalToExternal action.

+

The convertSuperNodeLocalToExternal action is similar to the createSuperNodeExternal action in that a new external pipeline flow is being created. Consequently, you can follow the instructions in the Creating an external sub-flow section for providing the appropriate properties of the data object in the beforeEditActionHandler and editActionHandler callbacks.

+

Converting an external supernode to a local supernode

+

When the common-canvas config property enableExternalPipelineFlows is set to true, and an external supernode’s is right clicked, the default common-canvas context menu will include a Convert external to local option. This will execute the convertSuperNodeExternalToLocal action.

+

The convertSuperNodeExternalToLocal action is similar to the expandSuperNodeInPlace action in that an external pipeline flow may need to be retrieved from your repository. Consequently, you can follow the instructions in the Loading an external sub-flow section for providing the appropriate properties of the data object in the beforeEditActionHandler callback.

+

Deleting an external supernode/sub-flow

+

When the user deletes an external supernode/sub-flow the supernode will be removed from the canvas. You code does not need to do anything unless you want to also remove the external pipeline flow from your repository.

+

Clipboard support

+

When a supernode, that refers to an external pipeline, is cut/copied and pasted, the pasted supernode refers to the same external pipeline as the supernode that was cut or copied. This means that if an external supernode is copied to the clipboard and then pasted into the same canvas the result will be two supernodes that refer to the same external pipeline.  

+

The same situation can occur if a supernode, that refers to an external pipeline, is in the palette and that node is dragged multiple times from the palette onto the canvas.

+

Manipulating objects in external pipelines using the Canvas Controller API

+

Objects (nodes, links comments, etc.) in an external pipeline can be updated by the host application calling the CanvasController API. However, such changes are only effective within Common Canvas. It is the host application’s responsibility to make sure these changes are persisted in the external pipeline flow document (if that is the behavior that is required). This can be done by the host application calling CanvasController.getExternalPipelineFlow(url) and then saving the returned document to the appropriate repository.

+

External pipelines in the Elyra Canvas Test Harness

+

The Test Harness supports external pipeline flows but will only persist any saved flows for the current session. (It just stores them in memory). You can examine the beforeEditActionHandler and editActionHandler in App.js in the test harness to see how it handles the different actions for managing external pipeline flows.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.09-read-only-or-locked-flows/index.html b/v13/03.09-read-only-or-locked-flows/index.html new file mode 100644 index 0000000000..1d473573a8 --- /dev/null +++ b/v13/03.09-read-only-or-locked-flows/index.html @@ -0,0 +1,2826 @@ + + + + + + + + + + + + + + + + + + + + + + + Read Only or Locked Flows - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Read Only or Locked Flows

+

Some host applications need to display flows that cannot be edited by the user. This might be because the flow artifact is currently being edited by another user (locked) or because the user does not have authority to edit the flow (read-only) or some other reason. We’ll use the term ‘read-only’ below to refer to both locked and read-only canvases.

+

There are many aspects of common-canvas components that need to be considered for a read-only canvas. Since Common Canvas is highly customizable it is not possible for the common-canvas code to manage components of the canvas that have been customized by the host application. For example, if the host app code added tools to the toolbar, Common Canvas does not know whether those tools should be disabled when displaying a read-only canvas or not. So the host application code will need to manage that. Let’s look at each element of Common Canvas and see what needs to be done.

+

General Config

+

There is one main canvas configuration property that will change the common-canvas behavior to implement a read-only canvas. This is enableEditingActions which defaults to true and needs to be set to false for read-only canvases. The sections below will cover what effect this will have on the different components of Common Canvas and what you need to do for any customizations you have made.

+

Canvas

+

Setting enableEditingActions to false will prevent nodes and comments (and detachable links) from being moved relative to one another. It will also prevent new links from being created and prevent text (like comments or node labels) from being edited.

+

With enableEditingActions set to false, the canvas can still be panned (left/right and up/down) and also zoomed in and out. Nodes, comments and links can still be clicked (to select) and right clicked (to display a context menu) and double clicked (usually to show properties). If you implemented any behavior for these interactions, using the clickActionHandler, you’ll need to review what your code is doing and make sure it is appropriate when a read-only canvas is being displayed.

+

Common Canvas allows objects from outside the canvas to be dropped onto the canvas to create a new node. For example, a file can be dragged from the operating system desktop onto the canvas to create a data node. With enableEditingActions set to false, the drag/drop gesture (which Common Canvas cannot prevent) will prevent a new node from being created. It is recommended you switch the enableDropZoneOnExternalDrag config property to false when displaying a read-only canvas. This will prevent a ‘drop zone’ graphic from appearing over the canvas when the object is dragged over the top of the canvas.

+

Some applications need to show a tag (called a ‘state tag’) over the canvas to emphasize the ‘read-only’ or ‘locked’ state of the canvas. This can be displayed using the enableStateTag canvas config property. The tooltip of the ‘state tag’ can be provided by implementing the tipHandler callback function.

+

Since there are numerous possible styles for nodes and links when displaying a read-only canvas, it isn’t possible for Common Canvas to style the canvas objects appropriately for all different designs. You will need to override any styles for nodes and links to fit your design. To help with this, Common Canvas sets a class called config-edit-actions-false on the top-level div that contains the common-canvas components. You can use this to build specific selectors in your CSS/SCSS that will override the styles applied to nodes and links by default. For example, to override node icon and label colors you could specify the following in your SCSS file: +

    .editing-actions-false {
+        .d3-node-group {
+        & .d3-node-label,
+            & svg path {
+                color: $disabled-02;
+                stroke: $disabled-02;
+                fill: $disabled-03;
+            }
+        }
+    }
+

+

Context menu

+

Setting enableEditingActions to false will prevent any options, that would edit the canvas objects, from being displayed in the common-canvas default context menu. For example, Delete will not show up in the context menu. See the enableEditingActions documentation for a list of options that are disabled.

+

Your application code can add your own options to the context menu by implementing the contextMenuHandler callback function. If you have added your own options, you should review your code and ensure options that might change the canvas objects are not added to the context menu when you are displaying a read-only canvas.

+

Note: The common-canvas default menu, passed as the second parameter to the contextMenuHandler callback, will still contain editing options. This means, if your code relies on them for some reason, (for example for calculating the position of your added options) your application code will still work OK. The editing options will be removed after your code returns the array of items that describe the desired menu from the contextMenuHandler callback.

+

Toolbar

+

The toolbar can display buttons for standard common-canvas actions and also buttons for any application-specific actions added by your code. Setting enableEditingActions to false will cause buttons for any standard common-canvas actions, that edit the canvas objects, to be disabled regardless of the setting for the enable property for each action. See the enableEditingActions documentation for a list of actions that are disabled.

+

You should review any application-specific action buttons you have added to the toolbar and decide if they need to be disabled or removed from the toolbar when displaying a read-only canvas.

+

If you decide you want to hide the toolbar, this can be achieved using the enableToolbarLayout canvas config field.

+

Keyboard

+

Common Canvas supports a number of keyboard shortcuts. Setting enableEditingActions to false, disables any keyboard shortcuts that edit the canvas objects (such as Delete). See the enableEditingActions documentation for a list of keyboard shortcuts that are disabled.

+

Palette

+

Setting enableEditingActions to false:

+
    +
  • prevents node templates from being dragged from the palette and
  • +
  • disables the double-click action on node templates which automatically adds a node to the canvas.
  • +
+

You should decide if your application should display the palette when displaying a read-only canvas or not. The palette can be hidden by setting the canvas config property enablePaletteLayout to “None”.

+

State Tag

+

Common Canvas allows the application to display a state tag, which is a label displayed directly on top of the canvas to allow the application to indicate what state (read-only or locked) the flow is in. The state tag can be switched on using the enableStateTag canvas configuration field.

+

Test Harness: Sample Read-Only application

+

The test harness contains a sample application called “Read-Only” which shows how a read-only canvas can be built using the enableEditingActions config option. In the application there are three toolbar buttons which can be used to navigate between Editable state, Read-Only state and Locked state. You can review the .jsx and .scss files in the application code to see how the application is implemented and styled.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.10-command-stack/index.html b/v13/03.10-command-stack/index.html new file mode 100644 index 0000000000..c989a37b66 --- /dev/null +++ b/v13/03.10-command-stack/index.html @@ -0,0 +1,2765 @@ + + + + + + + + + + + + + + + + + + + + + + + Command Stack - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

The Command Stack

+

The Command Stack is built into Common Canvas and automatically provides functionality to support do/undo/redo of commands performed in the flow editor. In addition, if needed, applications can add their own commands to the command stack.

+

The Command Stack maintains an internal stack of commands with a cursor that moves up and down when commands are undone or redone. Commands are JavaScript classes that implement a simple interface.

+

The canvas controller automatically creates an instance of the command stack. Common Canvas provides command objects for each of the commands that are performed by the user, such as: create node, delete comment, link nodes together, etc. which it adds to the command stack.

+

To allow the user to activate the undo and redo actions, Common Canvas provides: +* undo/redo buttons on the default toolbar and +* undo/redo options in its default context menu for the flow editor background, +* keyboard shortcuts: ctrl+z (undo) and ctrl+shift+z (redo) when keyboard focus is on the canvas.

+

If the applicaiton specifies its own canvas toolbar or its own context menu/toolbar for the flow editor canvas and it must include defintions for the undo and redo internal actions.

+

API control of the command stack (optional)

+

You can implement your own undo and redo UI, if required, using the canvas controller API.

+

The canvas controller has a number of methods that allows the application to interact with the command stack if necessary.

+

Building a command (optional)

+

Each command that is added to the command stack is a JavaScript class that needs to implement these methods:

+
   constructor()
+   do()
+   undo()
+   redo()
+   getLabel()
+
+

constructor() - Initial setup

+

do() - Performs all actions necessary to execute the command

+

undo() - Performs all actions necessary to reverse the actions performed in do()

+

redo() - Performs all actions necessary to re-execute the command. For some commands this is the same as do() but for others it is different.

+

getLabel() - Returns a label that descibes the action.

+

Here is some sample code that shows how a ‘create node’ command might be written:

+

   export default class CreateNodeAction extends Action {
+
+        constructor(data, canvasController) {
+            super(data);
+            this.canvasController = canvasController;
+            this.newNode = createNode(data);
+         }
+
+         do() {
+            this.canvasController.addNode(this.newNode);
+         }
+
+         undo() {
+            this.canvasController.deleteNode(this.newNode.id);
+         }
+
+         redo() {
+            this.canvasController.addNode(this.newNode);
+         }
+
+         getLabel() {
+            return "Add 1 node"
+         }
+   }
+
+ Note that the command has to keep a reference to the new node to allow the node to be added back + to the canvas in redo() even though it was deleted in undo().

+

Here is an example showing how to create a command action and push it on the stack using the canvas-conttoller:

+
   const command = new CreateNodeAction(data, this.canvasController);
+   this.canvasController.do(command);
+
+

Exported common-canvas action classes

+

Some of the internal action classes have been exported from Common Canvas and can be extended with additional +functionality, if necessaey. The classes that are exported are:

+
    +
  • CreateAutoNodeAction
  • +
  • CreateNodeAction
  • +
  • CreateNodeLinkAction
  • +
  • DeleteObjectsAction
  • +
  • DisconnectObjectsAction
  • +
  • PasteAction
  • +
+

The constructors for these classes all take the same two parameters. The data object that descrbes the command +and a reference to the canvas controller.

+

Applications can extend these classes to augment their basic behavior with application specific behavior. It is the application’s responsibility to add the extended object to the command stack when the user performs the corresponding action.

+

Although there are no plans to alter the internal workings of these six command action classes, there is always the chance that a change in the future might alter a field name or two. If you extend these classes, it is therefore recommended that you have sufficient regression tests for your extensions that would highlight such a problem, should it occur.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.11-internal-actions/index.html b/v13/03.11-internal-actions/index.html new file mode 100644 index 0000000000..ac849ecff8 --- /dev/null +++ b/v13/03.11-internal-actions/index.html @@ -0,0 +1,3111 @@ + + + + + + + + + + + + + + + + + + + + + + + Internal Actions - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Internal common-canvas actions

+

Internal actions can be generated by the user either:

+
    +
  • Clicking a toolbar button or
  • +
  • Clicking an option in a context menu or context toobar or
  • +
  • Pressing a keyboard shortcut key(s) or
  • +
  • Doing some direct manipulation like dragging one or more objects
  • +
+

Some actions can be generated from different sources. For example, the deleteSelectedObjects action can be generated by the user:

+
    +
  • Clicking the trash can icon in the toolbar or
  • +
  • Pressing the Delete key or
  • +
  • Clicking the Delete option in a context menu.
  • +
+

In each case, the action generates calls to the beforeEditActionHandler callback and then the editActionHandler callback. For each callback, the editType field of the first parameter will be set to one of the actions listed below.

+

These are the intenal actions:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Action NameNotes
Nodes
createNodeCreate a new node on the canvas at a location identifed by the user
createNodeOnLinkCreates a new node and inserts it into a link the palette node was dropped on
createNodeAttachLinksCreates a new node and attaches it to the detached link the palette node was dropped on
createAutoNodeCreates a node at an appropriate position and, if possible, make a connection to an adjacent node
insertNodeIntoLinkInserts a node from the canvas into the link it was dropped on
attachNodeToLinksAttaches a node to one or more detached links it was dropped on
setNodeLabelSets the node label
setNodeLabelEditingModePuts a node label into eiting mode
disconnectNodeRemoves all links to the selected node
Supernodes
createSuperNodeCreates an internal supernode
createSuperNodeExternalCreates an external supernode
deconstructSuperNodeRemoves a supernode and puts its contents into the canvas in its place
expandSuperNodeInPlaceExpands a supernode to view the sub-flow in-place
collapseSuperNodeInPlaceCollapses an in-place supernode to a regular node size
convertSuperNodeExternalToLocalConverts an external supernode to an internal one
convertSuperNodeLocalToExternalConverts an internal supernode to an external one
displaySubPipelineDisplays a sub-flow in full-page mode so it fills the viewport
displayPreviousPipelineDisplays the parent flow from the one currently being viewed
loadPipelineFlowLoads a pipeline from an external source
Comments
createCommentCreates a new comment on the canvas at a location identifed by the user
createAutoCommentCreates a node at an appropriate position on the canvas
commentsToggleToggles the comments between hide and show
commentsHideHides all comments on the canvas
commentsShowShows all comments on the canvas
setCommentEditingModePuts a comment into eiting mode
Canvas Objects
moveObjectsMoves one or more selected objects to a new position on the canvas
resizeObjectsResizes a node or comment
setObjectsStyleSets the style property of one or more nodes/comments
colorSelectedObjectsColors the background of the selected objects
deleteSelectedObjectsDeletes the selected objects
selectAllSelects all canvas objects
Links
updateLinkUpdates the link info for detached links
editCommentChanges the comment text, position and dimension info
linkNodesCreates a link between two nodes
linkNodesAndReplaceCreates a link between two nodes and relaces any exiting link to the target node
linkCommentCreates a link from a comment to a node
createDetachedLinkCreate a link from a node to a point on the canvas
setLinksStyleSets the style property of one or more links
deleteLinkDeletes a link
Decorations
editDecorationLabelPuts a label decoration into edit mode
Arrange nodes
arrangeHorizontallyArranges the nodes across the page from left to right
arrangeVerticallyArranges the nodes down the page from top to bottom
Palette
paletteToggleToggles the palette between opened and closed
paletteOpenOpens the palette (left flyout)
paletteCloseCloses the palette (left flyout)
saveToPaletteSaves the selected object into the palette
Notifications
toggleNotificationPanelToggles the notification panel between opened and closed
openNotificationPanelOpens the notification panel
closeNotificationPanelCloses the notification panel
Clipboard
cutCuts the selected canvas objects onto the clipboard
copyCopies the selected canvas objects onto the clipboard
pastePates the contents of the clipboard onto the canvas
Command Stack
undoUndoes the previous command on the command stack
redoRedoes the next comment on the command stack
Highlighting
highlightBranchHighlights all upstream and downstream nodes from the one selected
highlightDownstreamHighlights all upstream and downstream nodes from the one selected
highlightUpstreamHighlights all upstream and downstream nodes from the one selected
unhighlightUnhighlights all previously highlighted nodes
Zoom
zoomInZooms the canvas in by one step
zoomOutZooms the canvas out by one step
zoomToFitZooms so all the canvas objects are visible in the viewport
setZoomSets the zoom based on the enableSaveZoom config setting
+

Action names with built in icons

+

If you use any of the following action names, Common Canvas will automatically display an appropriate Carbon icon for that action either if it appears as a button in the toolbar or if it appears in the overflow menu.

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ActionCarbon Icon
stopStopFilledAlt
runPlay
undoUndo
redoRedo
cutCut
copyCopy
pastePaste
clipboardResult
createCommentAddComment
createAutoCommentAddComment
setCommentEditingModeEdit
setNodeLabelEditingModeEdit
commentsShowChat
commentsHideChatOff
colorBackgroundColorPalette
deleteLinkTrashCan
deleteSelectedObjectsTrashCan
zoomInZoomIn
zoomOutZoomOut
zoomToFitCenterToFit
arrangeHorizontallyArrangeHorizont
arrangeVerticallyArrangeVertical
toggleNotificationPanelNotificationCounter
paletteOpenOpenPanelFilledLeft
paletteCloseOpenPanelFilledLeft
paletteToggleOpenPanelFilledLeft
expandSuperNodeInPlaceMaximize
collapseSuperNodeInPlaceMinimize
displaySubPipelineLaunch
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.30.01-context-menu-wrapper/index.html b/v13/03.30.01-context-menu-wrapper/index.html new file mode 100644 index 0000000000..428b798e7b --- /dev/null +++ b/v13/03.30.01-context-menu-wrapper/index.html @@ -0,0 +1,2718 @@ + + + + + + + + + + + + + + + + + + + + + + + Context Menu Wrapper - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Context Menu Wrapper

+

This function is deprectated and will be removed in future release

+

Carbon 11 has a context menu component if your applications needs a context menu.

+

The context menu in Common Canvas can be used in your application by importing the ContextMenuWrapper React component. Aside from providing a standard context menu to allow users to select different actions, this context menu also allows for configuration of submenus as a menu item, as well as a visual dividers.

+

Getting started with Context Menu

+

Step 1: Import Context Menu Wrapper

+

To use Context Menu Wrapper in your React application you need to import the ContextMenuWrapper React component from the common-canvas library. It’s recommended to use All Components if Common Canvas is also being imported, otherwise use ContextMenuWrapper only option.

+

All Components +

    import {ContextMenuWrapper} from "@elyra/canvas";
+

+

ContextMenuWrapper only +

    import ContextMenuWrapper from "@elyra/canvas/dist/lib/context-menu";
+

+

Step 2: Pass in the correct props

+
    +
  • contextMenuDef array (required): an array of menu item objects consisting of action and label. You can also pass in a divider item.
  • +
+
    const menuDef = [
+        { action: ACTION.BUILD, label: "Build" },
+    { action: ACTION.EXTEND, label: "Extend" },
+    { action: ACTION.CLEAR, label: "Clear" },
+    { divider: true },
+    { action: ACTION.SCORE, label: "Score" },
+    ];
+
+
    +
  • containingDivId string (required): the id of the element that the context menu will be absolutely positioned inside. typically, the page element is used.
  • +
  • +

    contextMenuPos object (required): the position of the context menu within the containing div. +

        const menuPos = { x: 500  , y: 300  };
    +

    +
  • +
  • +

    contextMenuActionHandler func (required): this handler is where context menu actions are defined. +

        contextMenuActionHandler(action) {
    +        switch (action) {
    +            case "BUILD":
    +            break;
    +        case "EXTEND":
    +            break;
    +        default:
    +        }
    +    }
    +

    +
  • +
  • closeContextMenu func (required): this handler will be called when the context menu is closed.
  • +
  • stopPropagation bool (optional): this is optional and only for very specific, uncommon use cases. When this flag is set, if a user clicks outside the context menu, the event will not bubble to parent elements, preventing parent event handlers from being called.
  • +
+

Example

+
    const contextMenuWrapper = (
+        <ContextMenuWrapper
+            contextMenuDef={menuDef}
+            containingDivId="main-page"
+            contextMenuPos={menuPos}
+            contextMenuActionHandler={this.contextMenuActionHandler}
+            closeContextMenu={this.props.closeContextMenu}
+        />
+    );
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/03.30.02-flow-validation/index.html b/v13/03.30.02-flow-validation/index.html new file mode 100644 index 0000000000..aeaa43036e --- /dev/null +++ b/v13/03.30.02-flow-validation/index.html @@ -0,0 +1,2710 @@ + + + + + + + + + + + + + + + + + + + + + + + Flow Validation - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Flow Validation API

+

This function is deprectated and will be removed in future release

+

The Flow Validation API allows application code to programmatically validate the nodes in the flow against it’s property values. The API can be invoked after setting the Canvas Controller pipeline flow, when opening a new flow, or associated with a canvas context menu item. To use the Flow Validation API import the FlowValidation object from Common Canvas: +

import { FlowValidation } from "@elyra/canvas";
+
+then call the API on the object, for example: +
FlowValidation.validateFlow( ... );
+

+

The Flow Validation object provides the following API:

+

validateFlow(canvasController, parameterDataCallback, setNodeMessagesCallback, includeMsgTypes)
+    canvasController - an instance of the canvas controller
+    parameterDataCallback – function to get the parameter data or form data for a node
+    setNodeMessagesCallback – function to set the validation messages for a node. (optional)
+    includeMsgTypes - array[strings] Return invalid only if messages are found of types contained
+                      in the array. If not specified then any message type causes invalid return. (optional)
+    return - boolean If flow is valid returns true, otherwise returns false.
+
+where the two callbacks are defined as follows: +
parameterDataCallback(nodeId)
+    nodeId – string node Id.
+    function must return this object:
+    {
+        type: “parameterDef” | “form”,
+        data: Json_object
+    }
+
+/* The setNodeMessagesCallback is optional and only useful if not using internal object model.
+setNodeMessagesCallback(nodeId, messages)
+    nodeId – string node Id.
+    an array of message objects generated from the validation of the node.
+

+

The validateFlow() API will traverse the current flow and for each node invoke the parameterDataCallback() to get with a form data JSON or a parameterDef JSON. It will validate the JSON for the node and store any messages in the node objects within the model. The setNodeMessagesCallback() function will be called with all the messages generated for the node. This is only useful if the application is not using the internal object model. +The format of the message object is described in Pipeline Flow UI schema

+

Here is an example of using the FlowValidation API to validate a flow on opening:

+
import { CanvasController, FlowValidation } from "@elyra/canvas";
+
+getNodeForm(nodeId) {
+    const parameterDef = getParameterDefJSON(nodeId);
+    return { type: "parameterDef", data: parameterDef };
+}
+
+setNodeMessages(nodeId, messages) {
+    // code to persist messages in a store in addition to the internal model.
+}
+
+openCanvas(canvasJson) {
+    const canvasController = new CanvasController();
+    canvasController.setPipelineFlow(canvasJson);
+    FlowValidation.validateFlow(canvasController, this.getNodeForm, this.setNodeMessages);
+}
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04-common-properties/index.html b/v13/04-common-properties/index.html new file mode 100644 index 0000000000..812b6ea81f --- /dev/null +++ b/v13/04-common-properties/index.html @@ -0,0 +1,2895 @@ + + + + + + + + + + + + + + + + + + + + + + + Getting started - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Getting started with Common Properties

+

Introduction

+

Common Properties is directed by a JavaScript object called the Parameter Definition. The ‘paramDef’ object describes the properties to be displayed as controls in the <CommonProperties> React object.

+

You can look at the harness/src/client/App.js file to see examples of code that uses the common-properties component.

+

Building a properties panel in a React application

+

To use Common Properties in your React application complete the following steps:

+

Step 1 : Setup

+

Complete the setup steps documented in the Initial Setup page.

+

Step 2 : Import Common Properties

+

Import the Common Properties React component from the Elyra Canvas library. Elyra Canvas produces both esm and cjs outputs. By default esm will be used when webpack is used to build the application.

+

    import { CommonProperties } from "@elyra/canvas";
+
+Properties Only

+

To import only Common Properties functionality in cjs format use:

+
    import { CommonProperties } from "@elyra/canvas/dist/lib/properties";
+
+

Step 2 : Create the propertiesInfo object

+

Next, you’ll need to populate propertiesInfo, which is a required prop, with:

+

this.propertiesInfo = {
+  parameterDef: this.parameterDef,          // Required - Parameter definitions/hints/conditions
+  appData: "{user-defined}",                // Optional - User data returned in applyPropertyChanges
+  additionalComponents: "{components}",     // Optional - Additional component(s) to display
+  messages: "[node_messages]",              // Optional - Node messages array
+  expressionInfo: this.expressionInfo,      // Optional - Information for expression builde
+  initialEditorSize: "{size}",              // Optional - This value will override the value of
+                                            // editor_size in uiHints. This can have a value of
+                                            // "small", "medium", "large", or null
+  id: "{id}"                                // Optional - Unique parameter definition ID
+}
+
+See the Common Properties Parameter Definition page for more details about ‘paramDef’.

+

The expressionInfo object must conform to the expressionInfo schema

+

The optional messages attribute can be used to set validation messages associated with a node. The format of the message objects is defined in Pipeline Flow UI schema

+

Step 3 : Display the Common Properties object

+

Finally, you’ll need to display the Common Properties object. Inside your render code, add the following:

+
return (
+  <IntlProvider>
+    <CommonProperties
+      ref={(instance) => {
+        this.CommonProperties = instance;
+      }}
+      propertiesInfo={this.propertiesInfo}                  // Required
+      callbacks={this.callbacks}                            // Required
+      propertiesConfig={this.propertiesConfig}              // Optional
+      customPanels={[CustomSliderPanel, CustomTogglePanel]} // Optional
+      customControls={[CustomSliderControl]}                // Optional
+      customConditionOps={[CustomConditionOps]}             // Optional
+      light                                                 // Optional
+    />
+  </IntlProvider>
+);
+
+

See the Localization section of the Initial Setup page to see how <IntlProvider> can be configured.

+

Props

+
    +
  • propertiesInfo object: See above
  • +
  • callbacks object - See the Callbacks page
  • +
  • propertiesConfig object - See the Properties Config page
  • +
  • customPanels array: An array of custom panels. See Custom Panels section of the Common Properties Custom Components page.
  • +
  • customControls array: An array of custom controls. See Custom Controls section of the Common Properties Custom Components page..
  • +
  • customConditionOps array: An array of custom condition operators. See Custom Condition Operators section of the Custom Components page.
  • +
  • light boolean: Carbon controls in Common Properties will use light mode. When the light option is disabled, the background color will be the same as the Carbon theme background. When the light option is enabled, the background color is set to $ui-01. Defaults to true
  • +
+

Reference methods

+

The <CommonProperties> React object provides one reference method that can be called on a ref of the common-properties instance. +

/*
+* @closeEditor (boolean) - determines if closePropertiesDialog is called or not
+*/
+applyPropertiesEditing(closeEditor)
+

+

Using CommonProperties in CommonCanvas right-flyout panel

+

Common Canvas has a right-flyout panel that can render a React object. It can be used to render Common Properties in the right-flyout as follows:

+

Create a <CommonProperties> object with containerType set to "Custom" and rightFlyout set to true. +

    const rightFlyoutContent = (
+        <CommonProperties
+            propertiesInfo={this.propertiesInfo}
+            propertiesConfig={{ containerType: "Custom", rightFlyout: true }}
+            callbacks={this.callbacks}
+        />
+    );
+

+

Pass the <CommonProperties> object into the rightFlyoutContent prop of Common Canvas. Also, set the showRightFlyout boolean to tell Common Canvas the rightFlyout should be displayed (true) or hidden (false). +

    <CommonCanvas
+        canvasController={canvasController}
+        rightFlyoutContent={rightFlyoutContent}
+        showRightFlyout={showRightFlyout}
+    />
+

+

If the CommonProperties component is nested inside single or multiple layers of <div> elements, special consideration is needed for proper layout behavior.

+

    const rightFlyoutContent = (
+      <div className="parent-div">
+        <CommonProperties
+          propertiesInfo={this.propertiesInfo}
+          propertiesConfig={{ containerType: "Custom", rightFlyout: true }}
+          callbacks={this.callbacks}
+        />
+      </div>
+    );
+
+display: flex should be added to parent-div to allow Common Properties content to occupy full width and height available in right flyout.

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.01-parameter-definition/index.html b/v13/04.01-parameter-definition/index.html new file mode 100644 index 0000000000..7d225dabae --- /dev/null +++ b/v13/04.01-parameter-definition/index.html @@ -0,0 +1,3059 @@ + + + + + + + + + + + + + + + + + + + + + + + Parameter Definition - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Parameter Definition

+

The parameter definition object provides input for controlling the common-properties dialog. It consists of information that is available in the operator object combined with UI hints, information on data sets, and resources.

+

The parameter definition has a defined schema and set of examples located here: +schema and examples

+

The parameter definition consists of the following sections:

+

Title Definition

+

The title of the properties editor. If editable is set to true (default), the title can be edited. If set to false, the title will be readonly.

+
"titleDefinition": {
+    "title": "Properties Title",
+    "editable": true
+  }
+
+

Current Parameters

+

A list of input parameters and initial values upon input. The list is a set of key/value pairs with the key being the field name and the value is the initial value.

+

Example +

"current_parameters": {
+    "targetField": [],
+    "inputFieldList": [],
+    "elasticNetParam": 0.0,
+    "fitIntercept": false,
+    "maxIter": 75,
+    "regParam": 0.1,
+    "standardization": true,
+    "threshold": 0.5,
+    "tol": 0.0000010
+  }
+

+

Parameter Definitions

+

The list of parameters definitions for this property dialog. The list contains the name of the parameter, the data type of the parameter, the role and the default value. The information provided is as needed by the backend engine, i.e. the parameter name should be the name of the parameter that is expected for the backend engine.

+

The list of parameter definitions has the following attributes:

+
    +
  • id (string) Required Parameter identifier as consumed by the backend engine.
  • +
  • default (any) The default value of the parameter.
  • +
  • enum (array[string]) A restricted list of string values that are valid for the field.
  • +
  • type (string) Parameter type as consumed by the backend engine.
  • +
  • role (string) Parameter role, which is an optional specialization of the type.
  • +
  • required (boolean) Indication whether parameter is required or optional.
  • +
+

Parameter types have one of a fixed set of basic types. These are:

+
    +
  • integer
  • +
  • double
  • +
  • string
  • +
  • date
  • +
  • time
  • +
  • timestamp
  • +
  • custom
  • +
+

These can be used as maps or arrays e.g.:

+
array[<value-type>]: a sequence or list of values
+
+ +

Parameter roles defined by the role attribute can be one of:

+
    +
  • expression: an expression assumed to be in the expression language for the run time
  • +
  • column: value represents one or more columns from the data model visible to this operator
  • +
  • new_column: value represents the name of a new column to be added to the data model and must therefore not match an existing column and conform to existing syntactic restrictions.
  • +
+

Example +

"parameters": [
+    {
+      "id":"targetField",
+      "type":"string",
+      "default":"",
+      "role":"column"
+    },
+    {
+      "id":"inputFieldList",
+      "type":"array[string]",
+      "default":[],
+      "role":"column"
+    },
+    {
+      "id": "impurity",
+      "enum": [
+        "gini",
+        "entropy"
+      ],
+      "default": "gini"
+    },
+    {
+      "id":"elasticNetParam",
+      "type":"double",
+      "default":0.0
+    }
+]
+

+

UI-only Parameters

+

A list of input parameters and initial values upon input. This set of parameters are separated from the backend parameters (current parameters). The idea is that these parameters are not passed into the backend engine by the common-properties consumer.

+

Example +

"current_ui_parameters": {
+    "databaseResource": true,
+    "tol": 0.0000010
+  }
+

+

UI-only parameters require information about the parameters same as the parameter definition information used for the backend parameters. The UI-only parameter definition information is stored in the UI-hints section in the sub-section named ui_parameters. A description of the UI hints specifications can be found on the UI Hints page.

+

UI-only properties are returned to the consuming application via a separate parameter on the applyPropertiesChanges callback. See the Callbacks page.

+

Complex Types

+

The complex types section is an array of

+

The complex types have the following attributes:

+
    +
  • id (string) Required Identifier of complex type, can be referenced in other places.
  • +
  • type (string) If object is specified, Common Properties will return the values as an array of objects consisting of key value pairs. This defaults to array.
  • +
  • key_definition (object) A parameter definition attribute on the key parameter field.
  • +
  • parameters (object) Required List of parameters fields. Each parameter can be defined as a parameter definition attribute or a complex type attribute.
  • +
+

Example of complex types +

"complex_types": [
+    {
+      "id": "SortEntry",
+      "type": "object",
+      "key_definition": {
+        "id": "field",
+        "type": "string",
+        "role": "column",
+        "default": ""
+      },
+      "parameters": [
+        {
+          "id": "sort_order",
+          "enum": [
+            "Ascending",
+            "Descending"
+          ],
+          "default": "Ascending"
+        }
+      ]
+    }
+  ]
+

+

A Note on Parameters and Complex Types:

+

Note that both parameter and complex type definitions are in the exact same format as defined in the operator schema. Therefore the contents of operator JSON files can be used for these two sections.

+

UI Hints

+

A set of specifications for controlling the layout and flow of Common Properties. A description of the UI hints specifications can be found on the UI Hints page.

+

Conditions

+

A set of specifications for controlling validation checking of parameters during the common-properties dialog. A description of the Conditions specifications can be found here on the Conditions page.

+

Data Set Metadata

+

The data set metadata is an array of datarecord-metadata objects as defined in the datarecord-metadata JSON schema. Each datarecord-metadata object contains and array of fields that provide column information on the input data set. schema and examples

+

The fields have the following attributes:

+
    +
  • name (string) Required Field name. Must be unique within the dataset.
  • +
  • type (string) Required Field type. Can be a primitive type (string, integer, double, date, time, timestamp), or a vector, map, or struct containing those types. Required.
  • +
  • nullable (boolean) Indicates whether or not one can place null values into the field. Default: False.
  • +
  • metadata (object) A set of additional metadata attributes.
  • +
+

The additional metadata attributes are as follows:

+
    +
  • description (string) A description of the field.
  • +
  • measure (string) The field measurement type. The value can be one of the following. range, discrete, flag, set, ordered-set, typeless, collection, geospatial, default
  • +
  • role (string) Field role for modeling. The value can be one of the following. +input, +target, +both, +none, +partition, +split, +frequency, +record-id
  • +
  • max_string_length (number) Maximum character length for string fields. Length is unlimited when not present.
  • +
  • values (array[string]) Array of unique categorical values for the column.
  • +
  • ranges (object) Minimum and maximum discovered values for scalar data.
  • +
+

Example +

"dataset_metadata": [
+  {
+    "name": "Schema-1",
+    "fields": [
+      {
+        "name": "Age",
+        "type": "integer",
+        "metadata": {
+          "description": "",
+          "measure": "range",
+          "role": "input"
+        }
+      },
+      {
+        "name": "Sex",
+        "type": "string",
+        "metadata": {
+          "description": "",
+          "measure": "discrete",
+          "role": "input"
+        }
+      },
+      {
+        "name": "BP",
+        "type": "string",
+        "metadata": {
+          "description": "",
+          "measure": "discrete",
+          "role": "input"
+        }
+      }
+    ]
+  },
+  {
+    "name": "Schema-2",
+    "fields": [
+      {
+        "name": "Birthdate",
+        "type": "date",
+        "metadata": {
+          "description": "Date of birth",
+          "measure": "range",
+          "role": "input"
+        }
+      }
+    ]
+  }
+]
+

+

Resources

+

This is a map of string resources.

+

Example: +

"resources":{
+  "org.apache.spark.ml.classification.DecisionTreeClassifier.label":"Random Forest Classifier",
+  "org.apache.spark.ml.classification.DecisionTreeClassifier.desc":"Fitted Random Forest Classification Model",
+  "inputFieldList.label":"Input columns",
+  "inputFieldList.desc":"Select one or more input columns",
+  "targetField.label":"Target column",
+  "targetField.desc":"Select a target column",
+  "max_depth_not_valid":"The max depth parameter must be greater than or equal to zero",
+  "max_iter_not_valid": "The max iterations parameter must be greater than or equal to zero",
+  "min_instances_per_node_not_valid": "The minimum instances per node value must be >= 1",
+  "subsampling_rate_not_valid": "The subsampling rate value must be > 0 and <= 1"
+}
+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.02-callbacks/index.html b/v13/04.02-callbacks/index.html new file mode 100644 index 0000000000..e3cf33bd4d --- /dev/null +++ b/v13/04.02-callbacks/index.html @@ -0,0 +1,3016 @@ + + + + + + + + + + + + + + + + + + + + + + + Callbacks - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Callbacks

+

Common Properties provides the following callbacks. These are specified to Common Properties using an object like this:

+
  const callbacks = {
+      applyPropertyChanges: this.applyPropertyChanges,
+      closePropertiesDialog: this.closePropertiesEditorDialog,
+      propertyListener: this.propertyListener,
+      controllerHandler: this.propertiesControllerHandler,
+      actionHandler: this.propertyActionHandler,
+      buttonHandler: this.buttonHandler,
+      buttonIconHandler: this.buttonIconHandler,
+      helpClickHandler: this.helpClickHandler,
+      titleChangeHandler: this.titleChangeHandler,
+      propertiesActionLabelHandler: this.propertiesActionLabelHandler,
+      tooltipLinkHandler: this.tooltipLinkHandler
+  };
+
+

applyPropertyChanges(propertySet, appData, additionalInfo, undoInfo, uiProperties)

+

Executes when the user clicks OK in the property dialog. This callback allows users to save the current property values.

+
    +
  • propertySet: The set of current property values
  • +
  • appData: (optional) application data that was passed in to propertiesInfo
  • +
  • additionalInfo: Object with additional information returned:
      +
    • messages: (optional) An array of messages associated with the nodes current property values.
    • +
    • title: The title of the properties editor
    • +
    +
  • +
  • undoInfo: Object with information needed to undo this apply:
      +
    • properties: Set of property values;
    • +
    • messages: (optional) An array of messages associated with the nodes property values.
    • +
    • uiProperties: (optional) Set of UI only properties values
    • +
    +
  • +
  • uiProperties: The set of UI only property values (optional)
  • +
+
applyPropertyChanges(propertySet, appData, additionalInfo, undoInfo, uiProperties) {
+  var data = {
+    propertySet: propertySet,
+    appData: appData,
+    additionalInfo: {
+        messages: messages,
+        title: title
+    }
+  };
+}
+
+

closePropertiesDialog(closeSource)

+

Executes when user clicks Save or Cancel in the property editor dialog. This callback is used to control the visibility of the property editor dialog. closeSource identifies where this call was initiated from. It will equal “apply” if the user clicked on “Save” when no changes were made, or “cancel” if the user clicked on “Cancel”

+
closePropertiesDialog() {
+  this.setState({ showPropertiesDialog: false, propertiesInfo: {} });
+}
+
+

propertyListener()

+

Executes when a user property values are updated.

+
propertyListener(data) {
+
+}
+
+

controllerHandler()

+

Executes when the property controller is created. Returns the property controller. See the Properties Controller page for APIs.

+
controllerHandler(propertyController) {
+
+}
+
+

actionHandler()

+

Called whenever an action is ran. id and data come from uihints and appData is passed in with propertiesInfo.

+
actionHandler(id, appData, data) {
+
+}
+
+

buttonHandler()

+

Called when the edit button is clicked on in a readonlyTable control, or if a custom table button is clicked. The callback provides the following data:

+
    +
  • data: an object that consists of
      +
    • type: of button the click was invoked from.
        +
      • edit is returned from the edit button click of a readonlyTable control.
      • +
      • custom_button is returned from the custom button click of a complex type control.
      • +
      +
    • +
    • propertyId: of the control that was clicked.
    • +
    • buttonId: of the button that was clicked from a custom table button.
    • +
    +
  • +
+
buttonHandler(data) {
+
+}
+
+

buttonIconHandler()

+

Called when there is a buttons uihints set in the complex_type_info section of the parameter definition. This buttonIconHandler expects a Carbon Icon jsx object as the return value from the callback. This is used to display the Carbon icon in the custom table button. The buttonIconHandler provides the following data:

+
    +
  • data: an object that consists of
      +
    • type: customButtonIcon
    • +
    • propertyId: of the control that was clicked.
    • +
    • buttonId: of the button that was clicked from a custom table button.
    • +
    • carbonIcon: The name of the Carbon icon specified in the uihints. The corresponding jsx object is expected to be returned in the callback.
    • +
    +
  • +
+
buttonIconHandler(data, callbackIcon) {
+  if (data.type === "customButtonIcon" && data.carbonIcon === "Edit32") {
+    callbackIcon(<Edit32 />);
+  }
+}
+
+

propertyIconHandler()

+

Called when a user wants to pass in a specific object to a dropdown menu. The propertyIconHandler expects a jsx object as the return value from the callback. This is used to display the jsx object in the dropdown menu. The propertyIconHandler provides the following data:

+

-data: an object that consists of + - propertyId: of the dropdown that was selected + - enumValue: of the dropdown that was selected

+
propertyIconHandler(data, callbackIcon) {
+    const { iconSwitch } = this.state;
+    if (iconSwitch === true && data.propertyId.name === "oneofselect" && data.enumValue === "red") {
+        callbackIcon(<Icon />);
+    }
+}
+
+

helpClickHandler()

+

Executes when user clicks the help icon in the property editor dialog. The callback provides the following data:

+
    +
  • nodeTypeId: in case of parameterDef, id property of uihints;
  • +
  • helpData: Optional helpData specified in paramDef (see below).
  • +
  • appData: Optional application data that was passed in to propertiesInfo +
    helpClickHandler(nodeTypeId, helpData, appData) {
    +
    +}
    +
    +To control whether a node shows the help icon in the right flyout, a help object with optional helpData needs to be provided in the paramDef:
  • +
  • paramDef: Provide help object in operator’s uihints. If help object exists, the icon will be shown. Optionally, provide a helpData object within the help object, which will be passed in the helpClickHandler callback. + https://github.com/elyra-ai/pipeline-schemas/blob/master/common-pipeline/operators/uihints-v3-schema.json#L64
  • +
+

If no help object is found, no help link will be shown.

+

titleChangeHandler()

+

Called on properties title change. This callback can be used to validate the new title and return warning or error message if the new title is invalid. This callback is optional.

+

In case of error or warning, titleChangeHandler should call callbackFunction with an object having type and message. If the new title is valid, no need to call the callbackFunction. +

titleChangeHandler(title, callbackFunction) {
+  // If Title is valid. No need to send anything in callbackFunction
+  if (title.length > 15) {
+    callbackFunction({
+      type: "error",
+      message: "Only 15 characters are allowed in title."
+    });
+  }
+}
+
+where:

+
    +
  • type (string, required): This must be one of two values: “warning” or “error”.
  • +
  • message(string, required): Error or warning message. There is no restriction on length of the message.
  • +
+

propertiesActionLabelHandler()

+

propertiesActionLabelHandler()
+
+This is an optional handler you don’t need to implement anything for it unless you want to. This callback allows your code to override the default tooltip text for the Undo and Redo buttons. +The propertiesActionLabelHandler callback, when provided, is called for the save properties action that is performed in Common Properties. This callback should return a string or null. If a string is returned it will be shown in the tooltip for the Undo button in the toolbar preceded by “Undo:” and the string will also appear in the tooltip for the Redo button (when appropriate) preceded by “Redo:”. If null is returned, Common Properties will display the default text Save {node_name} node properties for the Undo and Redo buttons.

+

tooltipLinkHandler()

+

Optional callback used for adding a link in properties tooltips. link object must be defined under description in uiHints parameter info. Common Properties internally pass the link object to tooltipLinkHandler callback. +This callback must return an object having url and label.

+
tooltipLinkHandler(link) {
+    return { url: "https://www.google.com/", label: "More info" };
+}
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.03-ui-hints/index.html b/v13/04.03-ui-hints/index.html new file mode 100644 index 0000000000..c7630a762f --- /dev/null +++ b/v13/04.03-ui-hints/index.html @@ -0,0 +1,3171 @@ + + + + + + + + + + + + + + + + + + + + + + + UI Hints - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+ +
+ + + +
+
+ + + + + + + + + + + + +

UI Hints

+

UI hints is a section of the property definition JSON. It contains specifications to assist in the presentation and flow of the property dialog. The specifications indicate which controls to use to display and gather input on the fields. +UI hints schema

+

The UI Hints section consists a set of simple and complex attributes.

+

The simple attributes are:

+
    +
  • id (string) Dialog id.
  • +
  • label (object) External name of dialog. +
  • +
  • description (object) Description of dialog. +
  • +
  • editor_size (string) The width of the properties editor panel. This can have a value of "small", "medium", "large" or "max". The default is "small".
      +
    • When "small" is specified the properties panel is displayed with a default width of 320px and with a resize button that allows the panel to be increased in size up to the "medium" size which is 480px.
    • +
    • When "medium" is specified the properties panel has a width of 480px and with a resize button that allows the panel to be increased in size up to the "large" size which is 640px.
    • +
    • When "large" is specified the properties panel has a width of 640px and with a resize button that allows the panel to be increased in size up to the "max" size which is 900px.
    • +
    • When "max" is specified the properties panel has a width of 900px and no resize button is displayed.
    • +
    +
  • +
  • pixel_width (object) This optional property gives finer control over the minimum and maximum sizes of the properties editor panel. If this is omitted the properties editor width is controlled by the editor_size property. pixel_width is an object with two properties min and max which are both numbers.
      +
    • If min is specified it overrides the default size of the shrunken panel and max is based on the editor_size value.
    • +
    • If max is specified it overrides the default size of the expanded panel and min is based on the editor_size value.
    • +
    • If editor_size is set to "large" only the max value will be used to specify the size of the panel and no resize button will be displayed.
    • +
    +
  • +
+

A warning will be displayed in the console if you specify an invalid value for either min or max such as making min greater than max.

+

Example of the simple attributes: +

"uihints": {
+  "id":"org.apache.spark.ml.ibm.transformers.Distinct",
+  "icon": "images/transformationspark.svg",
+  "label": {
+    "default": "Distinct"
+  },
+  "editor_size": "medium",
+  "pixel_width": {
+    "min": 400,
+    "max": 800
+  },
+  "description": {
+    "default": "Remove rows to leave only rows with distinct combinations of rows"
+}
+

+

The complex attributes of the UI hints section are:

+

Group-info

+

Group info attributes.

+
    +
  • id (string) Required Panel id
  • +
  • type (string) The group type to be displayed. See the Group/Panel Controls section of the Controls page.
  • +
  • depends_on_ref (string) Property name this group depends upon. Valid for panelSelector groups only.
  • +
  • label (object) Group label. +
  • +
  • description (object) Group description. Only used with textPanel and actionPanel. +
  • +
  • parameter_refs (array[string]) List of parameter to be displayed.
  • +
  • action_refs (array[string]) List of action to be displayed. Used with actionPanel only.
  • +
  • group_info (object) List of additional group information.
  • +
  • data (any) Returned in custom panel constructor without any changes.
  • +
  • insert_panels (boolean) Indicates whether panels, contained with a panelSelector, should be inserted between the radio buttons of a radio button set indicated by the depends_on_ref parameter.
  • +
  • nested_panel (boolean) Indicate whether panel should be nested. Nested panels are indented by 16px from the left and display left border. Default is false.
  • +
  • class_name (string) Optional classname for this group
  • +
  • open (boolean) Optional used to determine if a panel should be open or not by default. Used with twistyPanel only. Default is false.
  • +
+

Example group info section: +

"group_info": [
+      {
+        "id": "settings",
+        "label": {
+          "default": "Settings"
+        },
+        "type": "columnSelection",
+        "parameter_refs": [
+          "keys"
+        ]
+      }
+    ]
+

+

UI-only Parameters

+

UI-only parameters require information about the parameters same as the parameter definition information used for the backend parameters. The UI-only parameter definition information is stored in the UI-hints section in the sub-section named ui_parameters. The format of the information in the ui_parameters sub-section is documented in the Parameter Definition section of Parameter Definitions page.

+

Example +

"ui_parameters": [
+    {
+      "id":"databaseResource",
+      "type":"boolean",
+      "default":true
+    },
+    {
+      "id":"toi",
+      "type":"double",
+      "default":0.0
+    }
+]
+

+

The UI-only parameters need to be added to other UI Hints sections (for example Group Info and canbe refered to by the parameter_ref field just like backend parameters.

+

Parameter Info

+

The parameter info section contains the list of parameters to gather values on through the property dialogs and UI hints about each parameter. The UI hints provide information to facilitate the UI controls used to display the parameter in the property dialogs.

+

Parameter info attributes.

+
    +
  • parameter_ref (string) Required Parameter name.
  • +
  • label (object) External name for parameter. +
  • +
  • label_visible (boolean) Determines whether to display the label for a control.
  • +
  • description (object) Description of parameter with optional placement context.
      +
    • See Resource Definition
    • +
    • placement (string) Placement context for the text. Valid values are as_tooltip, on_panel.
    • +
    • link (object) Optional link in the description. tooltipLinkHandler callback must be defined whenever link object is added in uiHints..
        +
      • id (string) Required unique link id.
      • +
      • data (object) Data passed to the tooltipLinkHandler callback.
      • +
      +
    • +
    +
  • +
  • control (string) Which control to use. See Parameter Controls section of the Controls page.
  • +
  • increment (number) Determines the increment/decrement value for the spinner control only. The default value is 1.
  • +
  • orientation (string) Determines how the control is displayed. Valid values are vertical, horizontal.
  • +
  • width (number) Column width for tables. The widths provided for table columns are used to calculate relative widths for each table column. So for example a 3 column table with widths of 20, 30, and 50 would use 20%, 30%, and 50% of the overall table width, respectively.
  • +
  • char_limit (number) Limits the number of characters a user can enter into the control for string parameters only.
  • +
  • display_chars (number) This has been deprecated and is subject to removal. Limits the number of characters displayed for a text field in a column in a table. The text will have an ellipsis appended at this limit. Defaults to 64 characters.
  • +
  • separator (string) Determines where to put a separator relative to the current control. Valid values are after, before.
  • +
  • visible (boolean) Determines whether to display control in a table cell. Used in complex types only.
  • +
  • read_only (boolean) Determines whether the control should be immutable or can be edited.
  • +
  • place_holder_text (object) Text hint for the user displayed input controls. +
  • +
  • helper_text (object) Additional text to be displayed below the control often used to explain the correct data format. +
  • +
  • resource_key (string) Used as a key for enum value labels in the resources section of property definition.
  • +
  • edit_style (string) Editing style of elements in a table. Valid values are subpanel, inline, on_panel.
  • +
  • value_icons (array[string]) For enumerated types, this defines the set of icons for the valid values. The ordering must be consistent with the order in the parameter enum attribute.
  • +
  • filterable (boolean) Determines if this column values can be filtered so that only rows that match the filter in column values are shown in the table. Applies to complex parameters only.
  • +
  • sortable (boolean) Determines if this column values can be sorted into ascending/descending order in a table. Applies to complex parameters only.
  • +
  • number_generator (object) Describes a number generator button beside numeric control. The ‘label’ element is a standard resource item, and the ‘range’ element contains ‘min’ and ‘max’ attributes to constrain the range of generated numbers.
  • +
  • dm_default (string) Data record metadata field to be used for default values in table cell columns. Typically this is used with parameters in complex structures in which the key field is a column name. Valid values are type, description, measure, and modeling_role.
  • +
  • dm_image (string) This can be set to display an icon of the corresponding dm type in the role:column field of a table. Valid values are measure, type, none
  • +
  • summary (boolean) Determines if parameter should be shown in the summary when using a summaryPanel.
  • +
  • text_before (object) Text to be displayed before the control +
  • +
  • text_after (object) Text to be displayed after the control +
  • +
  • custom_control_id (string) Id that is used to determine which custom control to use when control=custom
  • +
  • data (any) Returned in custom control constructor without any changes.
  • +
  • rows (integer) Number of rows to show in a table before scrolling starts. If one table in a panel is set to -1, that table will use the remaining available vertical space, down to a minimum of 2 rows. Used in expression and code controls to determine the number of rows to show for those controls.
  • +
  • moveable_rows (boolean) Determines if rows can be moved up or down in a table or array of strings.
  • +
  • action_ref (string) An action to be displayed.
  • +
  • date_format (string) A format string such as YYYY-MM-DD which describes the display and entry format for a date field.
  • +
  • time_format (string) A format string such as HH:mm:ss which describes the display and entry format for a time field.
  • +
  • custom_value_allowed (boolean) Determines if a dropdown, outside of a table, can allow a custom value to be entered.
  • +
  • class_name (string) Optional classname for this parameter
  • +
  • resizable (boolean) Determines if this column can be resized in a table. When a column is resized, width of all the columns to the right of resized column is adjusted. Applies to structure parameters only. Default is false.
  • +
+

Example parameter info section: +

    "parameter_info": [
+      {
+        "parameter_ref": "keys",
+        "label": {
+          "resource_key": "sort.keys.label"
+        },
+        "description": {
+          "resource_key": "sort.keys.desc",
+          "placement": "on_panel"
+        },
+        "rows": -1
+      }
+    ]
+

+

Complex Type Info

+

The complex_type_info section defines complex data types. This section is needed if in the parameters section of the parameter definition, one of the parameters has a type that is not the base type (i.e. an array or map of base types). The type of control used for this definition depends on the group info type value.

+

Complex Type info attributes.

+
    +
  • complex_type_ref (string) Required Name of complex type, can be referenced in other places.
  • +
  • key_definition (string) A set of parameter info attributes about the key parameter.
  • +
  • label (object) External name of subpanel. +
  • +
  • parameters (object) Required List of parameters that are part of this complex parameter. This parameter can either have a set of parameter_info attributes or other nested complex_type_info attributes.
  • +
  • header (boolean) If true then the table has a header row with column names. Defaults to true.
  • +
  • add_remove_rows (boolean) If true then the table can have rows added and removed. Defaults to true.
  • +
  • include_all_fields (boolean) When true and add_remove_rows is false, ensures that all fields are included in the control at all times.
  • +
  • row_selection (string enum) How many rows in a table can be selected at a time.
      +
    • single: only one row at a time is able to be selected.
    • +
    • multiple: multiple rows at a time are able to be selected.
    • +
    • multiple-edit: select multiple rows and allow the editing of column values of all selected rows. Clicking on “Edit” button in table toolbar, opens a subpanel which shows editable columns. All columns which have edit_style: "inline" or undefined are shown in the subpabel. Changing a column value in the subpanel, changes the value in all selected rows.
    • +
    +
  • +
  • buttons (array) An array of objects that define custom buttons to be displayed in this complex structure, overriding any default buttons. Each button object contains the following properties:
      +
    • id (string) Required: Unique identifier used to identify the button in the callback function.
    • +
    • label (object): Button label to display. If an icon is specified as well, the icon will be shown to the right of the label. +
    • +
    • description (object): Tooltip text to display when the button is hovered. +
    • +
    • icon (string): URL to .svg image to display.
    • +
    • carbon_icon (string): Host provided name of Carbon icon to display. A callback function is required for the host application to return the jsx icon object imported from @carbon/icons-react library.
    • +
    • enabled (boolean): Button will be enabled if true, disabled if false.
    • +
    • divider (string enum): Display a divider before or after this button. Defaults to after
        +
      • before Display divider before this button
      • +
      • after Display divider after this button
      • +
      +
    • +
    +
  • +
+

Example complex_type_info section: +

 "complex_type_info": [
+    {
+        "complex_type_ref": "SortEntry",
+        "row_selection": "multiple",
+        "moveable_rows": true,
+        "add_remove_rows": false,
+        "include_all_fields": true,
+        "key_definition": {
+          "parameter_ref": "field",
+          "width": 28,
+          "label": {
+            "resource_key": "SortEntry.field.label"
+          }
+        },
+        "parameters": [
+          {
+            "parameter_ref": "sort_order",
+            "width": 16,
+            "resource_key": "SortEntry.sort_order",
+            "label": {
+              "resource_key": "SortEntry.sort_order.label"
+            },
+            "control": "toggletext",
+            "value_icons": [
+              "/images/up-triangle.svg",
+              "/images/down-triangle.svg"
+            ]
+          }
+        ]
+      },
+      {
+        "complex_type_ref": "FieldStorageEntry",
+        "key_definition": {
+          "parameter_ref": "field",
+          "label": {
+            "default": "",
+            "resource_key": "FieldStorageEntry.field"
+          },
+          "width": 26,
+          "sortable": true,
+          "filterable": true
+        },
+        "parameters": [
+          {
+            "parameter_ref": "override",
+            "label": {
+              "default": "",
+              "resource_key": "FieldStorageEntry.override"
+            },
+            "width": 16,
+            "edit_style": "inline",
+            "sortable": true
+          },
+          {
+            "parameter_ref": "storage",
+            "label": {
+              "default": "",
+              "resource_key": "FieldStorageEntry.storage"
+            },
+            "width": 26,
+            "edit_style": "inline",
+            "dm_default": "type"
+          }
+        ],
+        "buttons": [
+          {
+            "id": "icon_button_1",
+            "carbon_icon": "Edit32",
+            "label": {
+              "resource_key": "table.somekey.label"
+            },
+            "description": {
+              "default": "This renders a button that has a label and Carbon icon to the right of the label.
+            },
+            "enabled": true
+          }
+       ]
+    }
+ ]
+

+

Title-info

+

The title_info sections defines what will be displayed in the title section alongside the titleDefinition.

+

Title info attributes.

+
    +
  • action_refs (array[string]) List of action to be displayed in the title section. Each action_ref must have a corresponding action defined in the action_info section.
  • +
+

Example title info section: +

"title_info": {
+   "action_refs": [
+      "increment"
+   ]
+}
+

+

Action Info

+

The action_info section defines an action. Actions are used to callback to the application to perform an operation.

+

Action info attributes.

+
    +
  • id (string) Required Id of the action.
  • +
  • label (object) Required External name of action. +
  • +
  • control (string) Required The type of action. Currently button and image are supported.
  • +
  • class_name (string) Optional classname for this action
  • +
  • image (object) Properties associate with an image action.
      +
    • url (string) Location of the image to display.
    • +
    • placement (string) Placement of image relative to a property. Values are right or left.
    • +
    • size (object) Pixel size of the image.
        +
      • height (number) Image height in pixels.
      • +
      • width (number) Image width in pixels.
      • +
      +
    • +
    • tooltip_direction (string) Set tooltip direction for action image. Values are right, left, top, or bottom. Default is bottom.
    • +
    +
  • +
  • button (object) Properties associated with action button.
      +
    • kind (string) Button kind. Values are same as carbon button kind values. Default is tertiary.
    • +
    • size (string) Button size. Values are sm, md, lg, xl. Default is sm.
    • +
    +
  • +
  • data (any) Returned back in action callback.
  • +
+
{
+  "id": "increment",
+  "label": {
+    "default": "Increment"
+  },
+  "control": "button",
+  "data": {
+    "parameter_ref": "number"
+  }
+}
+
+

Resource Definition

+

Used for user facing text. Allows for default values if no translations are provided.

+
    +
  • default (string) Default value if resource_key not defined.
  • +
  • resource_key (string) Used as a key for enum value labels in the resources section of property definition.
  • +
+

Dynamic text expressions

+

Used to dynamically set text based on a parameter value change. If parameter id is used then then current value for that parameter will be passed into the function.

+
    +
  • percent(<number or parameter id>, <integer>) Return the percent of the 1st parameter. The optional 2nd parameter determines the number of decimal places.
  • +
  • sum(<number or parameter id>, <number or parameter id>, ...) Returns the sum of all parameters
  • +
+
{
+  "parameter_ref": "numberfield",
+  "label": {
+    "default": "Number"
+  },
+  "text_after": {
+    "default": "Sum: ${sum(numberfield, 2)} with (numberfield, 2, numberfield). Percent: ${percent(numberfield,2)}"
+  }
+}
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.04-controls/index.html b/v13/04.04-controls/index.html new file mode 100644 index 0000000000..4128dcbb7b --- /dev/null +++ b/v13/04.04-controls/index.html @@ -0,0 +1,2917 @@ + + + + + + + + + + + + + + + + + + + + + + + Controls - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Controls

+

Common Properties Element Types

+

The editor determines the most appropriate control and panel type for each parameter based on the context (parameter type, role, group type, etc.). Although an author can specify a desired control for a given parameter via its uihints, this should be used sparingly and carefully - specifying an inappropriate control for a parameter will lead to undefined behavior.

+

Documentation containing examples of the following group/panels and control types is available at:

+

https://elyra-canvas-test-harness.u20youmx4sm.us-south.codeengine.appdomain.cloud/#/properties

+

Group/Panel Controls

+

Controls are grouped and arranged on panels and sub-panels within the “group_info” section of operator uihints. Some panels appear as tab controls, others are for managing shared dataset metadata, while still others contain nested sub-panels.

+

Group Types

+
    +
  • controls A general panel type containing parameter controls.
  • +
  • tabs A tabbed control, each tab containing sub-panels and controls.
  • +
  • subTabs A horizontal sub-tabbed control, each tab containing sub-panels and controls. If displayed within a Tearsheet container, subtabs will be displayed vertically.
  • +
  • panels A panel type that contains sub-panels.
  • +
  • panelSelector A panel containing sub-panels that are shown or hidden based upon a controlling radio selection.
  • +
  • columnSelection A panel type containing field-selection controls that share a common set of fields.
  • +
  • customPanel A custom panel for displaying user defined control. See Custom Panels section of the Custom Components page for more details.
  • +
  • summaryPanel A panel used in the flyout editor that will provide a link to open a wide flyout that can contain panels and parameter.
  • +
  • actionPanel A panel used for containing action controls.
  • +
  • textPanel A panel used to display static label and/or description.
  • +
  • twistyPanel A panel used to display a panel title that expands to the panel content when clicked.
  • +
  • columnPanel A panel used to display children side by side.
  • +
  • tearsheetPanel A tearsheet panel. The panel can be opened/closed using the propertyController methods setActiveTearsheet(groupId) and clearActiveTearsheet()
  • +
+

Parameter Controls

+

The following controls are supported in the Common Properties editor. Control types are intended for use with particular parameter types:

+

Control Types

+
    +
  • readonly A read only text field. Used for fields users shouldn’t edit.
  • +
  • hidden A control that has no UI to display.
  • +
  • textfield A single line editable text field.
  • +
  • passwordfield A masked single line text field with tooltip. The tooltip text can be customized by setting [parameter_id].passwordHide.tooltip and [parameter_id].passwordShow.tooltip in resources section.
  • +
  • textarea A multi-line text area.
  • +
  • list A single column table for editing a list of values.
  • +
  • expression An expression editing field that provides language specific syntax highlighting and text auto complete. An expression builder addon is available with the expression control. You must provide the expressionInfo field for the propertiesInfo config. See Common Properties Documentation for more details. To maximize in a tearsheet add this attribute and define a tearsheetPanel in group_info. +
    "data": {
    +  "tearsheet_ref": "<tearsheet groupId>"
    +}
    +
  • +
  • code An code editing field that provides language specific syntax highlighting and text auto complete. To maximize in a tearsheet add this attribute and define a tearsheetPanel in group_info. +
    "data": {
    +  "tearsheet_ref": "<tearsheet groupId>"
    +}
    +
  • +
  • numberfield A numeric text field. Number fields can also optionally display a random number generator button beside the control. See the uihints schema for details.
  • +
  • datefield A date input control whose date format tokens follow date-fns. Defaults to yyyy-mm-dd
  • +
  • timefield A time input control whose time format tokens follow date-fns. Defautls to H:m:s
  • +
  • datepicker A date input control with calendar picker whose date format tokens follow Flatpickr. Defaults to Y-m-d. Helper text can be included by adding [parameter_id].helper in the resources section.
  • +
  • datepickerRange A date input control with calendar picker for a range of dates. This follow the same rules as the datepicker control. Start and end labels defaults to Start and End respectively. Start, end, and helper labels can be customized by adding the following in the resources section: +
    "resources": {
    +  [parameter_id].range.start.label: "Custom start label",
    +  [parameter_id].range.start.desc: "Custom start description that will appear as tooltip next to the label",
    +  [parameter_id].range.start.helper: "Custom start helper that will appear as text below the input",
    +  [parameter_id].range.end.label: "Custom end label",
    +  [parameter_id].range.end.desc: "Custom end description that will appear as tooltip next to the label",
    +  [parameter_id].range.end.helper: "Custom end helper that will appear as text below the input"`
    +}
    +
  • +
  • spinner A standard spinner control to increment/decrement the number value.
  • +
  • checkbox A standard checkbox control.
  • +
  • radioset A radio set where a parameter value is selected from a small range of options. See the Conditions page for special radio button disabling options.
  • +
  • checkboxset A checkbox set for list type parameters with enumerated options where the count is less than 5.
  • +
  • oneofselect A standard dropdown list control.
  • +
  • multiselect A standard dropdown list control that allows for multiple selection.
  • +
  • someofselect A multi-selection control for enumerated list parameters where the count is greater than 4.
  • +
  • selectcolumn A dropdown list control that selects from available column names. When dropdown list is empty, selectcolumn control will display default placeholder text "...". This placeholder text can be customized by setting [parameter_id].emptyList.placeholder in resources section. When custom empty list placeholder text is provided, Common Properties will disable the empty list control.
  • +
  • selectcolumns A multi-select control for column selections.
  • +
  • selectschema A dropdown control that contains the available schemas in dataset_metadata. The name of the schema will be displayed if provided. If name is not provided, the index (zero-based) of the schema will be used instead. When dropdown list is empty, selectschema control will display default placeholder text "...". This placeholder text can be customized by setting [parameter_id].emptyList.placeholder in resources section. When custom empty list placeholder text is provided, Common Properties will disable the empty list control.
  • +
  • toggle A standard toggle control with default On/Off states. This text can be customized by setting [parameter_id].toggle.on.label and [parameter_id].toggle.off.label in resources section.
  • +
  • toggletext A two-state control with optional icons that can exist on its own or within table cells.
  • +
  • structuretable Table control for editing lists or maps of complex types that have field names in the first column.
  • +
  • structurelisteditor For lists or maps of complex types that are not field-oriented parameters.
  • +
  • structureeditor Allows one to define a structure and use it directly on a panel. Each structure member is surfaced as an individual control. Supports a layout setting that allows one to position structureeditor controls in a grid (see below).
  • +
  • readonlyTable A read only table. Used for tables to display fields that users shouldn’t edit.
  • +
  • custom A custom control for displaying a user defined control. See Custom Controls section of the Common Properties Custom Components page for more details.
  • +
  • slider A standard slider which allows to enter a numeric value within the slider range and also allows to drag and adjust the slider track to a specific value within the range. The slider labels for minimum and maximum values can be customized by setting them as [parameter_id].min.label for minimum value label and [parameter_id].max.label for maximum value label in resources section.
  • +
+

A Note on Field Name Storage

+

When a given node can accept more than a single datarecord-metadata object as input, it becomes necessary to store the schema name (a.k.a. ‘link_name’) along with each field name that is stored in parameter sets. In those cases, instead of using strings to store field names, they are represented in parameter sets as compound objects containing both ‘link_ref’ and ‘field_name’ elements:

+
"current_parameters": {
+  "field": { "link_ref": "Schema-1", "field_name": "Age" },
+  ...
+
+ +

In order to indicate that a given node can potentially accept multiple input data links and would thus require compound field name storage, all parameter definitions within the node that contain "role": "column", whether located at the top level or within complex types, should declare their data types as “object” instead of “string”:

+
"parameters": [
+  {
+    "id": "fields",
+    "type": "array[object]",
+    "role": "column",
+    "required": true
+  },
+  ...
+
+ +

edit_style

+

When editing complex type values in tables one can either edit cell values inline or in a sub panel:

+
    +
  • subpanel A small sub-dialog is launched to edit cell values.
  • +
  • inline Controls appear inline within table cells for editing values.
  • +
  • on_panel Control appears below the table when the row is selected.
  • +
+

Miscellaneous

+
    +
  • moveable_rows boolean A value that appears in “complex_type_definition” sections. If set to true allows rows in the table to be moved up and down for reordering.
  • +
  • row_selection enum [“single”, “multiple”] Determines how many rows can be selected in a table at one time. Defaults to multiple.
  • +
  • sortable boolean Both sortable and filterable apply to table columns. When set within the “key_definition” or the “parameters” sections of a structure definition, those columns are sortable and/or can be filtered upon.
  • +
  • filterable boolean (see sortable above)
  • +
  • language enum [“CLEM”, “text/x-hive”] The language for the expression control syntax highlight and text auto complete feature. If not specified, the expression control does not have syntax highlighting or text auto complete.
  • +
  • layout A two-dimensional string array value that appears in “complex_type_definition” sections and allows one to layout structureeditor controls in a two dimensional grid.
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.05-conditions/index.html b/v13/04.05-conditions/index.html new file mode 100644 index 0000000000..841cf26174 --- /dev/null +++ b/v13/04.05-conditions/index.html @@ -0,0 +1,3148 @@ + + + + + + + + + + + + + + + + + + + + + + + Conditions - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Conditions

+

Conditions define a set of specifications for evaluating parameter values. The specifications support complex interdependency checking such as relationships between multiple parameters (i.e. Valid values for parameter one depend upon the value of parameter two). Here is the +Conditions Schema

+

Documentation containing examples of the following conditions for the different supported controls are available at: +https://elyra-canvas-test-harness.u20youmx4sm.us-south.codeengine.appdomain.cloud/#/conditions

+

A conditions file contains an array of conditions. Each condition takes one of the following forms:

+

Validation definition

+

A single validation. The fail_message is displayed upon validation failure.

+

The attributes for the validation definition are:

+
    +
  • id (string) A unique identifier for the validation. This is required if multiple validations have the same focus_parameter_ref value.
  • +
  • +

    fail_message (object) Required The message to display if the validation fails. Each fail_message consist of the following attributes

    +
      +
    • message (object) Required The message to display. +
    • +
    • focus_parameter_ref (string) Required The parameter control to get focus after displaying the error/warning. If the validation refers to a table cell, then control must have the column indicator. For example if the validation is for MyTable cell column 2 then MyTable[2].
    • +
    • type (string) Type of messages. Valid values are error, warning, info.
    • +
    +
  • +
  • +

    evaluate (object) Specification for how to evaluate the validity of the parameter. The evaluate attribute can be one of the following structures.

    +
      +
    • condition (object) This is a single condition that evaluates to true or false.
        +
      • op (string) Required A single operator for the properties of the condition. Valid values are: +isEmpty, +isNotEmpty, +greaterThan, +lessThan, +equals, +notEquals, +matches, +notMatches, +contains, +notContains, +colNotExists, +isDateTime, +dmTypeEquals, +dmTypeNotEquals, +dmMeasurementEquals, +dmMeasurementNotEquals, +dmRoleEquals, +dmRoleNotEquals +lengthEquals, +lengthGreaterThan, +lengthLessThan.
      • +
      +
    • +
    • parameter_ref (string) Required The primary parameter.
    • +
    • parameter_2_ref (string) Second parameter for multi-parameter validation.
    • +
    • value (string, boolean, number) Value against which to compare the primary parameter value.
    • +
    • values (array[string]) Values against which to compare the primary parameter value is in. Used only in filter conditions.
    • +
    +
  • +
  • or (object) This is a container of ‘or’ conditions. Evaluates to true if ANY sub-condition evaluates to true. Can nest any number of additional conditional types.
  • +
  • and (object) This is a container of ‘and’ conditions. Evaluates to true if ALL sub-condition evaluates to true. Can nest any number of additional conditional types.
  • +
+

Enabled definition

+

Enablement test. Disables controls if evaluate is false.

+

The attributes for the enabled definition are:

+
    +
  • parameter_refs (array[string]) Array of parameter names affected by this operation. If evaluate is false, then the controls associated with these parameters are disabled. Note that individual radio buttons can be disabled by using the radio button value name instead of the overall property name in the parameter_refs array.
  • +
  • action_refs (array[string]) Array of action names affected by this operation. If evaluate is false, then the action button or image associated with these action names are disabled.
  • +
  • evaluate (object) see the evaluate attribute in validation definition.
  • +
+

Visible definition

+

Visibility test. Hides controls if evaluate is false.

+

The attributes for the visible definition are:

+
    +
  • parameter_refs (array[string]) Array of parameter names affected by this operation. If evaluate is false, then hide the controls associated with these parameters.
  • +
  • action_refs (array[string]) Array of action names affected by this operation. If evaluate is false, then the action button or image associated with these action names are disabled.
  • +
  • evaluate (object) see the evaluate attribute in validation definition.
  • +
+

Filter definition

+

Filter test. The filter will determine which data record fields to include in a control.

+

The attributes for the filter definition are:

+
    +
  • parameter_ref (string) Parameter id affected by this operation. This must be a parameter that operates upon datarecord-metadata columns.
  • +
  • parameter_refs (string) Exclusive with parameter_ref and used with dmSharedFields. Parameter ids affected by this operation. They must be parameters that operate upon datarecord-metadata columns.
  • +
  • evaluate (object) see the evaluate attribute in validation definition.
  • +
+

Supported operations (op):

+
    +
  • dmType - filters type value from schema.
  • +
  • dmMeasurement - filters measurement value from schema.
  • +
  • dmModelingRole - filters modeling_role value from schema.
  • +
  • dmSharedFields - shares source fields with all field chooser property names found in the parameter_refs array.
  • +
+

Examples: +

{
+  "filter": {
+    "parameter_ref": "fields_filter_measurement",
+    "evaluate": {
+      "condition": {
+        "op": "dmMeasurement",
+        "value": "discrete"
+      }
+    }
+  }
+}
+
+
{
+  "filter": {
+    "parameter_ref": "fields_filter_type",
+    "evaluate": {
+      "condition": {
+        "op": "dmType",
+        "values": ["integer", "double"]
+      }
+    }
+  }
+}
+
+
{
+  "filter": {
+    "parameter_refs": [
+       "fields_filter_type",
+       "multi_field_chooser_table",
+       "field_chooser_in_another_panel"
+    ],
+    "evaluate": {
+      "condition": {
+        "op": "dmSharedFields"
+      }
+    }
+  }
+}
+

+

Enum Filter definition

+

Filters the available options for enumeration parameters. Reduces the available enumeration items if evaluate is true.

+

The attributes for the enum_filter definition are:

+
    +
  • target (object) Contains a target parameter_ref reference and a replacement values array.
  • +
  • evaluate (object) see the evaluate attribute in validation definition.
  • +
+

Example: +

{
+  "enum_filter": {
+    "target": {
+      "parameter_ref": "radioset_filtered",
+      "values": [
+        "red",
+        "yellow",
+        "green"
+      ]
+    },
+    "evaluate": {
+      "condition": {
+        "parameter_ref": "filter_radios",
+        "op": "equals",
+        "value": true
+    }
+  }
+}
+

+

Allow Change definition

+

Allow change validates that a change is allowed on a property. If it evaluates to true then the value for the property is changed. This is typically used to restrict values that are invalid in one property based on the value in another property. For example, if the property represents a storage type with a value of string, then a property that represents a measurement type should not be allowed to be set to continuous.

+

The attributes for the allow_change definition are:

+
    +
  • parameter_refs (array[string]) Array of parameter names affected by this operation.
  • +
  • evaluate (object) see the evaluate attribute in validation definition.
  • +
+

Example: +

{
+      "allow_change": {
+        "parameter_refs": [
+          "ST_mse_table[2]"
+        ],
+        "evaluate": {
+          "or": [
+            {
+              "condition": {
+                "parameter_ref": "ST_mse_table[2]",
+                "op": "notEquals",
+                "value": "Football"
+              }
+            },
+            {
+              "condition": {
+                "parameter_ref": "ST_mse_table[5]",
+                "op": "notEquals",
+                "value": "European"
+              }
+            }
+          ]
+        }
+      }
+    }
+

+

Default value definition

+

Sets the default value on the parameter_ref property if condition evaluates to true. If multiple conditions evaluate to true only the first condition is used. Default value condition is evaluated only once when loading properties. If user updates the value of parameter_ref, default value will be overwritten by the new value.

+

The attributes for the default_value definition are:

+
    +
  • parameter_ref (string) Parameter whose default value is to be set.
  • +
  • value (string, boolean, number, object, array) This will be the default value of parameter_ref if condition evaluates to true.
  • +
  • evaluate (object) see the evaluate attribute in validation definition.
  • +
+

Example: +

{
+  "default_value": {
+    "parameter_ref": "conditional_default",
+    "value": "Value defined in default_value condition. You will see this sentence when default value of mode equals Include.",
+    "evaluate": {
+      "condition": {
+        "parameter_ref": "mode",
+        "op": "equals",
+        "value": "Include"
+      }
+    }
+  }
+},
+{
+  "default_value": {
+    "parameter_ref": "conditional_default",
+    "value": ["This is a second condition for conditional_default. You should never see this value."],
+    "evaluate": {
+      "condition": {
+        "parameter_ref": "mode",
+        "op": "equals",
+        "value": "Include"
+      }
+    }
+  }
+}
+

+

A note on table cell conditions

+

Support for table cell conditions is achieved via the use of the array subscript operator, []. When evaluating table cells, one uses the table identifier with an array subscript indicating the zero-based table column being operated upon (which also corresponds to the sub-control index as defined in complex_types).

+

So for example if one has a StructureTable property named myTable, column conditions on that table are referred to using myTable[1], myTable[3], etc.

+

Example

+

Example of a condition section. +

  "conditions": [
+      {
+        "validation": {
+          "fail_message": {
+            "type": "error",
+            "focus_parameter_ref": "inputFieldList",
+            "message": {
+              "resource_key": "input_field_list_not_empty"
+            }
+          },
+          "evaluate": {
+            "condition": {
+              "parameter_ref": "inputFieldList",
+              "op": "isNotEmpty"
+            }
+          }
+        }
+      },
+      {
+    "visible": {
+      "parameter_refs": [
+         "oneofselectPets"
+      ],
+      "evaluate": {
+         "condition": {
+            "parameter_ref": "oneofselectAnimals",
+        "op": "notContains",
+        "value": "lion"
+          }
+      }
+    }
+      },
+     {
+    "visible": {
+      "action_refs": [
+         "action_button"
+      ],
+      "evaluate": {
+         "condition": {
+            "parameter_ref": "button_hide_checkbox",
+        "op": "equals",
+        "value": false
+          }
+      }
+    }
+      },
+      {
+    "enabled": {
+      "parameter_refs": [
+        "radiosetColor"
+      ],
+      "evaluate": {
+        "condition": {
+          "parameter_ref": "checkboxEnable",
+          "op": "checked"
+            }
+      }
+    }
+      },
+      {
+    "validation": {
+      "fail_message": {
+        "type": "error",
+        "focus_parameter_ref": "subsamplingRate",
+        "message": {
+          "resource_key": "subsampling_rate_not_valid"
+        }
+      },
+      "evaluate": {
+        "and": [
+          {
+            "condition": {
+              "parameter_ref": "subsamplingRate",
+              "op": "greaterThan",
+              "value": 0
+            }
+          },
+          {
+            "or": [
+              {
+                "condition": {
+                  "parameter_ref": "subsamplingRate",
+                  "op": "lessThan",
+                  "value": 1
+                }
+              },
+              {
+                "condition": {
+                  "parameter_ref": "subsamplingRate",
+                  "op": "equals",
+                  "value": 1
+                }
+              }
+            ]
+          }
+        ]
+      }
+    }
+      },
+      {
+    "enabled": {
+      "parameter_refs": [
+        "field_types[2]"
+      ],
+      "evaluate": {
+        "condition": {
+          "parameter_ref": "field_types[1]",
+          "op": "checked"
+        }
+      }
+    }
+      }
+    }
+  ]
+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.06-custom-components/index.html b/v13/04.06-custom-components/index.html new file mode 100644 index 0000000000..982b7fa6e0 --- /dev/null +++ b/v13/04.06-custom-components/index.html @@ -0,0 +1,3038 @@ + + + + + + + + + + + + + + + + + + + + + + + Custom Components - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Custom Components Overview

+

Custom components allows applications to use custom code to drive different parts of the common-properties user interface. For some panels and controls it might be necessary to listen to different types of redux state changes to cause the panel/control to rerender. +Here is an example of a textfield listening to three state changes: +

import { connect } from "react-redux";
+
+// ... application code
+
+render() {
+  const value = this.props.value; // value passed by redux as a property
+
+  // ... rest of component render code
+
+}
+
+TextfieldControl.propTypes = {
+  // ... application props
+  state: PropTypes.string, // pass in by redux
+  value: PropTypes.string, // pass in by redux
+  messageInfo: PropTypes.object // pass in by redux
+};
+
+const mapStateToProps = (state, ownProps) => ({
+  value: ownProps.controller.getPropertyValue(ownProps.propertyId),
+  state: ownProps.controller.getControlState(ownProps.propertyId),
+  messageInfo: ownProps.controller.getErrorMessage(ownProps.propertyId)
+});
+export default connect(mapStateToProps, null)(TextfieldControl);
+

+

Custom Panels

+

Custom panels allow applications to create their own panels and controls that can live in the same dialogs as common-property panels and controls.

+

Custom panel interface

+
// Returns the 'id' for the group defined in uihints
+static id()
+
+constructor(parameters, controller, data)
+
+// Returns the content users want to display
+renderPanel()
+
+
    +
  • parameters - String array of parameters set under the customPanel group in uihints
  • +
  • controller - See here for API information.
  • +
  • data - Optional parameter. Returns values stored in data attribute of a group customPanel.
  • +
  • renderPanel() - Called on all Redux store changes: +
  • +
+

Custom React components

+

Example +

renderPanel() {
+    const controlId = this.parameters[0];
+    return (
+        <CustomCtrlToggle
+            key={controlId}
+            propertyId={name: controlId}
+            controller={this.controller}
+        />
+    );
+}
+

+

Examples

+

https://github.com/elyra-ai/canvas/tree/master/canvas_modules/harness/src/client/components/custom-panels

+

Custom Controls

+

Custom controls allow applications to create their own controls that can live in the same dialogs as common-property panels and controls.

+

Custom control interface

+
// Returns the 'custom_control_id' for the parameter defined in uihints
+static id()
+
+constructor(propertyId, controller, data, tableInfo)
+
+// Returns the content users want to display
+renderControl()
+
+
    +
  • propertyId - See propertyId for definition.
  • +
  • controller - See here for API information.
  • +
  • data - Returns values stored in data attribute of a parameter in uihints.
  • +
  • tableInfo - Set when custom control is a cell in a table.
      +
    • table (boolean) Set to true when in a table cell
    • +
    • editStyle (string) Valid values are “summary” and “inline”. “summary” is set when the control will display either below the table (“on_panel”) or in a “subpanel”. This allows the custom control to display a summary value in the cell and something else for the custom control.
    • +
    +
  • +
  • renderControl() - Called on all Redux store changes: +
  • +
+

Custom React components

+

Example +

renderControl() {
+    return (
+        <CustomCtrlToggle
+            key={controlId}
+            propertyId={this.propertyId}
+            controller={this.controller}
+        />
+    );
+}
+

+

Examples

+

https://github.com/elyra-ai/canvas/tree/master/canvas_modules/harness/src/client/components/custom-controls

+

Custom Condition Operators

+

Custom condition operators allow users to create their own operators that can then be used for enablement, visibility, validation, and enum filtering. The condition operators should always return a boolean value.

+

Custom operator interface

+
/**
+* This is the key used to determine if the operator should be ran.  Maps to the `op` defined in the
+* `condition` in uihints
+* @return string
+*/
+function op()
+
+/**
+* @param see below
+* @return boolean
+*/
+function evaluate(paramInfo, param2Info, value, controller)
+
+
    +
  • paramInfo (object) - parameter_ref set in the condition in uihints
      +
    • control (object) - contains information about the control.
    • +
    • value (any) - current property value
    • +
    +
  • +
  • param2Info (object) - parameter_2_ref set in the condition in uihints. See paramInfo for object info
  • +
  • value - value set in the condition in uihints. If no value specific this will be undefined
  • +
  • controller - See here for API information.
  • +
+

Example +

function op() {
+  return "customMax";
+}
+
+function evaluate(paramInfo, param2Info, value, controller) {
+  const supportedControls = ["numberfield"];
+  if (supportedControls.indexOf(paramInfo.control.controlType) >= 0) {
+    return paramInfo.value < value;
+  }
+  return true;
+}
+
+module.exports.op = op;
+module.exports.evaluate = evaluate;
+
+
{
+    "evaluate": {
+        "condition": {
+            "parameter_ref": "custom_op_num",
+            "op": "customMax",
+            "value": 100
+        }
+    }
+}
+

+

Examples

+

https://github.com/elyra-ai/canvas/tree/master/canvas_modules/harness/src/client/custom/condition-ops +https://github.com/elyra-ai/canvas/tree/master/canvas_modules/common-canvas/src/common-properties/ui-conditions/condition-ops

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.07-properties-controller/index.html b/v13/04.07-properties-controller/index.html new file mode 100644 index 0000000000..533a0f089a --- /dev/null +++ b/v13/04.07-properties-controller/index.html @@ -0,0 +1,3284 @@ + + + + + + + + + + + + + + + + + + + + + + + Properties Controller - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + + + + + + +

Properties Controller

+ +

Common Properties Controller API

+

propertyId +

const propertyId = {
+  name: {parameter name defined in operator definition},
+  row: {row in table/array}, // optional when col not set
+  col: {col in table}, // optional
+  propertyId: {propertyId of the nested structure} // optional
+}
+

+

Property methods

+
/*
+ * options - optional object of config options where:
+ *   setDefaultValues (boolean): when set to true, set default values from parameter definition
+ */
+setPropertyValues(values, options)
+updatePropertyValue(propertyId, value)
+
+/*
+ * options - optional object of config options where:
+ *   filterHiddenDisabled (boolean): when set to true, filter out data values with a state of disabled or hidden
+ *   filterHiddenControls (boolean): when set to true, filter out data values having control type hidden
+ *   applyProperties (boolean): when set to true, will return data values in the format expected by the `applyPropertyChanges` callback. If unset or false, will return the internal format used by Common Properties.
+ */
+getPropertyValue(propertyId, options)
+
+/*
+ * options - optional object of config options where:
+ *   filterHiddenDisabled (boolean): when set to true, filter out data values with a state of disabled or hidden
+ *   filterHiddenControls (boolean): when set to true, filter out data values having control type hidden
+ *   applyProperties (boolean): when set to true, will return data values in the format expected by the `applyPropertyChanges` callback. If unset or false, will return the internal format used by Common Properties.
+ */
+getPropertyValues(options)
+
+

Message methods

+
/*
+ * Returns current list of error messages
+ * @filteredPipeline (boolean) optional
+ * @filterHiddenDisable (boolean) optional. If true, will not return error messages from controls that are hidden or disabled
+ * @filterDisplayError (boolean) optional. If true, will not return error messages that are not displayed in the editor
+ * when filteredPipeline=true returns enabled/visible control messages and only 1 per control.
+ */
+getAllErrorMessages()
+getErrorMessages(filteredPipeline, filterHiddenDisable, filterSuccess, filterDisplayError = true)
+getErrorMessage(propertyId, filterHiddenDisable = false, filterSuccess = false, filterDisplayError = true)
+getRequiredErrorMessages()
+setErrorMessages(messages)
+updateErrorMessage(propertyId, message)
+
+

State methods (disable/enabled & hidden/visible)

+
getControlState(propertyId)
+getControlStates()
+setControlStates(states)
+
+/*
+ * @propertyId - see above
+ * @state - valid values are "enabled", "disabled", "visible", "hidden"
+ */
+updateControlState(propertyId, state)
+
+

DatasetMetadata methods

+
getDatasetMetadata()
+
+/*
+ * @datasetMetadata - see [schema](https://github.com/elyra-ai/pipeline-schemas/blob/master/common-pipeline/datarecord-metadata/datarecord-metadata-v1-schema.json)
+ */
+setDatasetMetadata(datasetMetadata)
+
+

Row selection methods

+
/*
+ * Returns table row selection indices as an array of integers.
+ * @propertyId - see above
+ */
+getSelectedRows(propertyId)
+
+/*
+ * Updates table row selections for the given table control.
+ * @propertyId - see above
+ * @selection - A zero-based array of integer selection indices
+ */
+updateSelectedRows(propertyId, selection)
+
+/*
+ * Clears selected table rows for the given table.
+ * @propertyId - see above
+ * If the propertyId is omitted all table row selections are cleared
+ */
+clearSelectedRows(propertyId)
+
+/*
+ * Adds a row selection listener for a table or list.
+ * @propertyId - see above
+ * @listener - callback function for when a selection is made in the table or list
+ */
+addRowSelectionListener(propertyId, listener)
+
+/*
+ * Removes the row selection listener from a table or list.
+ * @propertyId - see above
+ */
+removeRowSelectionListener(propertyId)
+
+

Validation methods

+
/*
+ * Runs validation conditions on all controls
+ */
+validatePropertiesValues()
+
+/*
+ * Validates a specific propertyId
+ * @propertyId - see above
+ */
+validateInput(propertyId)
+
+

Control methods

+
/*
+ * Update the enum values for a given control.  Used when enum values aren't static
+ * @propertyId - see above
+ * @valuesObj (array) [{ value: <string, number, boolean> , label: "<string>" }]
+ */
+updateControlEnumValues(propertyId, valuesObj)
+
+

General methods

+
/*
+ * Returns the current size of the RHS flyout.
+ */
+getEditorSize()
+
+/*
+ * Sets default property values from parameter definition in the propertiesController.
+ * Note - These values won't be displayed on the UI. Host applications can call getPropertyValues() to retrieve the values.
+ * @paramDef - Follows the format of https://github.com/elyra-ai/pipeline-schemas/blob/master/common-canvas/parameter-defs/parameter-defs-v3-schema.json
+ */
+setParamDef(paramDef)
+
+/*
+ * Returns the id of top-level active tab or accordion
+ */
+getTopLevelActiveGroupId()
+
+/*
+ * Makes the passed in groupId active.  Only works for top-level groups
+ */
+setTopLevelActiveGroupId(groupId)
+
+

Disable move row buttons methods

+
/*
+ * Disable table row move buttons for all propertyIds in given array
+ * @param propertyIds Array of propertyIds
+ *
+ */
+setDisableRowMoveButtons(propertyIds)
+
+/*
+ * Returns array of propertyIds for which row move buttons will be disabled
+ *  @return Array of propertyIds
+ */
+getDisableRowMoveButtons()
+
+/*
+ * Check if row move buttons should be disabled for given propertyId
+ * @param propertyId  The unique property identifier
+ * @return boolean
+ */
+isDisableRowMoveButtons(propertyId)
+
+

Custom panel and control methods

+
/*
+ * Only used in custom panel to allow for custom property summary values to be displayed
+ * Displays the value set in propertiesReducer for that parameter
+ * @propertyId - see above
+ * @label (string)
+ * @inSummary (boolean)
+ */
+setControlInSummary(propertyId, label, inSummary)
+
+/*
+ * Sets the content to be displayed in the summaryPanel for a customPanel property.
+ * The summary panel will directly display the content.
+ * @propertyId - see above
+ * @content = { value: <object> , label: "<value>" }
+ */
+updateCustPropSumPanelValue(propertyId, content)
+
+/*
+ * Returns a standard control that can then be used in a customPanel.
+ * @propertyId - See above
+ * @paramDef - Follows the format of https://github.com/elyra-ai/pipeline-schemas/blob/master/common-canvas/parameter-defs/parameter-defs-v1-schema.json).  titleDefinition, current_parameters, conditions, dataset_metadata are ignored and are optional.
+ * @parameter - This is the parameter from the paramDef to create the control for.
+ */
+createControl(propertyId, paramDef, parameter)
+
+/*
+ * Returns the translated text for a control given a resource key.
+ * Users should be able to use the values from resources that has been uploaded as part of paramDef.
+ * @key - Resource key
+ * @value - Default value returned when no resource or key has been found.
+ */
+getResource(key, value)
+
+

maxLength for single-line and multi-line control methods

+
/*
+ * Returns the maximum characters allowed for multi-line string controls
+ * Default value is 1024
+ */
+getMaxLengthForMultiLineControls()
+
+/*
+ * Returns the maximum characters allowed for single-line string controls
+ * Default value is 128
+ */
+getMaxLengthForSingleLineControls()
+
+

Enabling/disabling addRemoveRows methods

+
/*
+ * Set the addRemoveRows attribute to 'enabled' for the given propertyId
+ * @param propertyId The unique property identifier
+ * @param enabled boolean value to enable or disable addRemoveRows
+ */
+setAddRemoveRows(propertyId, enabled)
+
+/*
+ * Returns the true if addRemoveRows is enabled for the given propertyID
+ * @param propertyId The unique property identifier
+ * @return boolean
+ */
+getAddRemoveRows(propertyId)
+
+

Enabling/disabling properties editor “save” button methods

+
/*
+ * Set the main "save" button to disabled(true) or enabled(false)
+ * @param saveDisable (boolean)
+ */
+setSaveButtonDisable(saveDisable)
+
+/*
+ * Returns the true if the main "save" button is disabled, false otherwise
+ * @return boolean
+ */
+getSaveButtonDisable()
+
+

Add static rows for table controls which will disable the re-ordering of the rows that are set as static for the given propertyId

+
/*
+ * Set static rows for the given propertyId
+ * @param propertyId The unique property identifier
+ * @param staticRowsArr Array of first n row indexes or last n row indexes
+ */
+updateStaticRows(propertyId, staticRowsArr)
+
+/*
+ * Returns the static rows set for the given propertyId
+ * @param propertyId The unique property identifier
+ */
+getStaticRows(propertyId)
+
+/*
+ * Removes the static rows set for the given propertyId
+ * @param propertyId The unique property identifier
+ */
+clearStaticRows(propertyId)
+
+

Enabling/disabling custom table buttons

+
/*
+ * Set the table button to 'enabled' for the given propertyId
+ * @param propertyId The unique property identifier
+ * @param buttonId The unique button identifier
+ * @param enabled boolean value to enable or disable the button
+ */
+setTableButtonEnabled(propertyId, buttonId, enabled)
+
+/*
+ * Returns the table button states for the given propertyID
+ * @param propertyId The unique property identifier
+ * @return object An object of buttonIds mapped to their enabled state
+ */
+getTableButtons(propertyId)
+
+/*
+ * Returns the true if the table button is enabled for the given propertyID and buttonId
+ * @param propertyId The unique property identifier
+ * @param buttonId The unique button identifier
+ * @return boolean
+ */
+getTableButtonEnabled(propertyId, buttonId)
+
+

Column visibility methods

+
/*
+ * Check if given column is visible in the table
+ * @param propertyId The unique property identifier
+ * @param columnIndex Column index in the table
+ */
+getColumnVisibility(propertyId, columnIndex)
+
+/*
+ * Set column visibility
+ * @param propertyId The unique property identifier
+ * @param columnIndex Column index in the table
+ * @param value Boolean value to set column visible/invisible
+ */
+toggleColumnVisibility(propertyId, columnIndex, value)
+
+

Enabling/disabling wide flyout “OK” button methods

+
/*
+ * Set the "OK" button in Wide Flyout to disabled(true) or enabled(false) for given summary panel
+ * @param panelId {name: panel.id}
+ * @param wideFlyoutPrimaryButtonDisable boolean
+ */
+setWideFlyoutPrimaryButtonDisabled(panelId, wideFlyoutPrimaryButtonDisable)
+
+/*
+ * @param panelId {name: panel.id}
+ */
+getWideFlyoutPrimaryButtonDisabled(panelId)
+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/04.08-properties-config/index.html b/v13/04.08-properties-config/index.html new file mode 100644 index 0000000000..fa84157432 --- /dev/null +++ b/v13/04.08-properties-config/index.html @@ -0,0 +1,2663 @@ + + + + + + + + + + + + + + + + + + + + + + + Properties Config - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Properties Config

+

The Properties Config is an object passed as an optional prop to the <CommonProperties< React object containing the following fields:

+
    +
  • +

    containerType string: type of container to display the properties, can be “Modal”, “Tearsheet”, or “Custom”. default: "Custom"

    +
  • +
  • +

    rightFlyout boolean: If set to true, groups will be displayed as an accordion. If false, groups are displayed as tabs. default: false

    +
  • +
  • +

    applyOnBlur boolean: calls applyPropertyChanges when focus leave Common Properties. default: false

    +
  • +
  • +

    disableSaveOnRequiredErrors boolean: Disable the properties editor “save” button if there are required errors

    +
  • +
  • +

    enableResize boolean: adds a button that allows the right-side fly-out editor to expand/collapse between small and medium sizes. default: true

    +
  • +
  • +

    conditionReturnValueHandling string: used to determine how hidden or disabled control values are returned in applyPropertyChanges callback. Current options are “value” or “null”. default: "value"

    +
  • +
  • +

    buttonLabels object:

    +
      +
    • primary string: Label to use for the primary button of the properties dialog
    • +
    • secondary string: Label to use for the secondary button of the properties dialog
    • +
    +
  • +
  • +

    heading boolean: show heading and heading icon in right-side fly-out panels. default: false

    +
  • +
  • +

    schemaValidation boolean: If set to true, schema validation will be enabled when a parameter definition has been set in CommonProperties. Any errors found will be reported on the browser dev console. It is recommended you run with schema validation switched on while in development mode.

    +
  • +
  • +

    applyPropertiesWithoutEdit boolean: When true, will always call applyPropertyChanges even if no changes were made. default: false

    +
  • +
  • +

    maxLengthForMultiLineControls number - maximum characters allowed for multi-line string controls like textarea. default: 1024

    +
  • +
  • +

    maxLengthForSingleLineControls number - maximum characters allowed for single-line string controls like textfield. default: 128

    +
  • +
  • +

    convertValueDataTypes boolean - Default false. If set to true, currentParameter values whose data type does not match what is defined in the parameter definitions will be converted to the specified data type.

    +
  • +
  • +

    trimSpaces boolean - Default true. If set to false, condition ops(isEmpty, isNotEmpty) and required fields are allowed to only contain spaces without triggering condition errors.

    +
  • +
  • +

    showRequiredIndicator boolean - Default true to show (required) indicator. If set to false, show (optional) indicator next to properties label.

    +
  • +
  • +

    showAlertsTab boolean - Default true to show “Alerts” tab whenever there are error or warning messages. If set to false, Alerts tab won’t be displayed.

    +
  • +
  • +

    returnValueFiltering array - Default []. When set this will filter out any values in the array in the parameters returned when applyPropertyChanges is call. Only primitive data types are currently supported.

    +
  • +
  • +

    categoryView string - View categories in right-flyout. Can be "accordions" or "tabs". default: "accordions".

    +
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/05-faq/index.html b/v13/05-faq/index.html new file mode 100644 index 0000000000..2b94742c19 --- /dev/null +++ b/v13/05-faq/index.html @@ -0,0 +1,2776 @@ + + + + + + + + + + + + + + + + + + + + + + + Frequently asked questions - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Frequently asked questions

+

History

+
+What is the relationship between Elyra Canvas and the Elyra AI Toolkit? +

Elyra Canvas is a repo inside the Elyra AI Toolkit. The Elyra repo in the toolkit is dependent on Elyra Canvas but Elyra Canvas is not dependent on it or any repo in the toolkit other than the Elyra Pipeline Schemas repo.

+
+

Common Canvas

+

Questions about Nodes

+
+With resizeable nodes (enableResizableNodes: true) why don’t the shapes defined by bodyPath and selectionPath change when resizing a node? +

The bodyPath and selectionPath ‘shapes’ do not automatically redraw when the user resizes the node. However, a function that returns a path can be provided for those fields instead of a string, so the application can return whatever shape it prefers as the resizing progresses. Obviously, these functions are called in real-time so they needs to return very quickly.

+
+
+Can React objects specified in nodeExternalObject display supernodes and, if so, how is the sub-flow rendered? +

Yes, ’nodeExternalObject` is supported for supernodes. The way it works is:

+
    +
  • If the supernode is collapsed (that is, it looks like a regular node) then the React node is used in the same way as for a regular node
  • +
  • If the supernode it expanded in-place, then the React object is still used in the same way and the <svg> area that displays the sub-flow is displayed by Common Canvas over the top of the node body.
  • +
  • If the user is viewing the supernode ‘full-page’ then the sub-flow is rendered in the usual way since the parent supernode is not visible at that point.
  • +
+

There is an example of this in the ‘React Nodes - Carbon Charts’ sample application in the test harness.

+

+
+
+When displaying React nodes using nodeExternalObject field, how can the application pass in its own props to the component? +

The application can add whatever it wants to pass in as a field in either the nodeData object or the Canvas Controller — making sure not to over write any of the existing fields that are there by default of course. So if you wanted to pass in your own data on a node-by-node basis you could set a field in the app_data field of each node. Something like:

+
    const nodeId = "123";
+
+    const mydtaa = {
+        field1: val1
+        field2: val2
+    }
+
+    canvasController.setNodeProperties(
+        nodeId,
+        { app_data: mydata }
+    );
+
+

Then in the React object just reference the fields like: +

    const f1 = this.props.nodeData.app_data.field1;
+

+
+

Questions about Comments

+
+With markdown enabled in comments, why isn’t the whitespace preserved when the user leaves edit mode? +

That’s the way markdown works. It removes white space in the entered text. For example, if you enter similar text into a comment in a GitHub issue and look at the preview you’ll see the whitespace is removed.

+

Common Canvas is using a third party library to convert what the user enters to the HTML that is displayed so the removal of whitespace is not under its control.

+

However, Comments support the ability to enter HTML directly into the markdown text. HTML can be used to preserve whitespace and do many other styling and customizations to the text.

+
+
+Are there any plans to support different fonts in comments? +

In Elyra Canvas v13.0.0, there is now a feature that allows the user to enter HTML in to the markdown text. This allows fonts and many other customizations of the text — although the user does need to know what they are doing with HTML.

+
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/06-contributing/index.html b/v13/06-contributing/index.html new file mode 100644 index 0000000000..a72a6f5abd --- /dev/null +++ b/v13/06-contributing/index.html @@ -0,0 +1,2607 @@ + + + + + + + + + + + + + + + + + + + + + + + Guidelines for Development - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Guidelines for Development

+

Some guidelines for contributing to Elyra Canvas are included in this section including:

+ + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/06.01-style-guidelines/index.html b/v13/06.01-style-guidelines/index.html new file mode 100644 index 0000000000..f7912b8343 --- /dev/null +++ b/v13/06.01-style-guidelines/index.html @@ -0,0 +1,2668 @@ + + + + + + + + + + + + + + + + + + + + + + + Style Guidelines for Development - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Style Guidelines for Development

+

Elyra Canvas styling guidelines

+
    +
  • Used the data-id attribute on inputs to be used for automated tests. Format for Common Properties should be properties-
  • +
  • className format format for Common Properties should be properties-
  • +
  • Limit the use of html(DOM) ids
  • +
  • Minimum inline styling. This allows for consumers to easily override styling.
  • +
  • scss/sass styling should be added to the component’s folder
  • +
  • No important! in styling
  • +
  • Use variables for all colors(preferably from carbon)
  • +
+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/06.02-testing-guidelines/index.html b/v13/06.02-testing-guidelines/index.html new file mode 100644 index 0000000000..6e9ee8d5a7 --- /dev/null +++ b/v13/06.02-testing-guidelines/index.html @@ -0,0 +1,2782 @@ + + + + + + + + + + + + + + + + + + + + + Testing Guidelines for Development - Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + + + + + + +

Testing Guidelines for Development

+

Unit Testing

+

The Canvas unit tests are automated with the primary purpose of providing rapid feedback to the developers. The test cases are run during every development build. Unit test cases are written using Jest Jest Tutorial. The test cases should be written and delivered at the time that a feature or enhancement is delivered.

+

Unit test cases should focus on good coverage of a function/service. We are current investigating code coverage analysis tools and will update this doc when it is implemented.

+

Unit test case coverage should focus on these areas:

+
    +
  • All APIs and all UI elements.
  • +
  • All component properties.
  • +
  • A variety of input data.
  • +
+

Here is a good blog on JavaScript Unit Testing

+

Functional Testing

+

The Canvas Functional Test cases will be automated and run during code delivery. The function test case will be automated and written using Cypress.

+

Functional Testing coverage includes the following types of tests.

+
    +
  • Core functionality
  • +
  • Inter-operate with other Canvas elements.
  • +
  • Need to test both forward and backwards compatibility
  • +
  • Negative / bounds
  • +
  • Globalization / Localization
      +
    • Handling of all strings using UTF-8
    • +
    • Verifying non-English unicode data is handled appropriately
    • +
    • Externalizing all strings that may be presented for the user (e.g., error messages, UI labels, etc.)
    • +
    +
  • +
  • Access control security (roles / permissions / tenant management)
  • +
  • Malicious and security (code scans such as AppScan, ethical hacking)
  • +
  • Accessibility for UI
  • +
+

Debugging Tests

+

Jest tests (unit)

+

https://facebook.github.io/jest/docs/troubleshooting.html

+

With node 8 or newer

+
    +
  • Add debugger; statement to your Jest test suite program where you want to stop and begin debugging.
  • +
  • +

    If you want to run just a single test within your test program (rather than all of them) temporarily change the it() method for the test to be it.only(). For example, change:

    +

    it("should add a node", () => { ... })

    +

    to be:

    +

    it.only("should add a node", () => { ... })

    +
  • +
  • +

    In the console enter: npm run debug or npm run debug <test suite name>

    +
  • +
  • Open Chrome debugging tools by pasting this into the Chrome address field: chrome://inspect/
  • +
  • You should see a ‘remote’ target for node_modules/.bin/jest. Click on the inspect link below it.
  • +
  • Click on sources and then click the play button (right pointing blue triangle icon).
  • +
  • The code should run to the point where your debugger statement was added.
  • +
+

Testing Troubleshooting

+

When testing your application with Jest, this error might show up: crypto.getRandomValues() not supported. See https://github.com/uuidjs/uuid#getrandomvalues-not-supported for details.

+

To fix, added this to your jest setup file: +

const cryptoJest = require("crypto");
+Object.defineProperty(global.self, "crypto", {
+    value: {
+        getRandomValues: (arr) => cryptoJest.randomBytes(arr.length)
+    }
+});
+

+ + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/404.html b/v13/404.html new file mode 100644 index 0000000000..7b212f6748 --- /dev/null +++ b/v13/404.html @@ -0,0 +1,2513 @@ + + + + + + + + + + + + + + + + + + + Elyra Canvas + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + + + + +
+
+ + + +
+
+
+ + + + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ +

404 - Not found

+ +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/v13/assets/FlowIcon.png b/v13/assets/FlowIcon.png new file mode 100644 index 0000000000..4bb30ae3de Binary files /dev/null and b/v13/assets/FlowIcon.png differ diff --git a/v13/assets/cc-app-tiny.png b/v13/assets/cc-app-tiny.png new file mode 100644 index 0000000000..35a3644a16 Binary files /dev/null and b/v13/assets/cc-app-tiny.png differ diff --git a/v13/assets/cc-comment-color.gif b/v13/assets/cc-comment-color.gif new file mode 100644 index 0000000000..e6a6161c4a Binary files /dev/null and b/v13/assets/cc-comment-color.gif differ diff --git a/v13/assets/cc-comment-elements-dom.png b/v13/assets/cc-comment-elements-dom.png new file mode 100644 index 0000000000..2dc5cf8d10 Binary files /dev/null and b/v13/assets/cc-comment-elements-dom.png differ diff --git a/v13/assets/cc-comment-link.gif b/v13/assets/cc-comment-link.gif new file mode 100644 index 0000000000..2aedaaf014 Binary files /dev/null and b/v13/assets/cc-comment-link.gif differ diff --git a/v13/assets/cc-comment-markdown-edit.png b/v13/assets/cc-comment-markdown-edit.png new file mode 100644 index 0000000000..1cde89be11 Binary files /dev/null and b/v13/assets/cc-comment-markdown-edit.png differ diff --git a/v13/assets/cc-comment-markdown-out.png b/v13/assets/cc-comment-markdown-out.png new file mode 100644 index 0000000000..7345eb0cb1 Binary files /dev/null and b/v13/assets/cc-comment-markdown-out.png differ diff --git a/v13/assets/cc-comment-reg-edit.png b/v13/assets/cc-comment-reg-edit.png new file mode 100644 index 0000000000..6ab308f8e1 Binary files /dev/null and b/v13/assets/cc-comment-reg-edit.png differ diff --git a/v13/assets/cc-comment-reg-type.png b/v13/assets/cc-comment-reg-type.png new file mode 100644 index 0000000000..53c531eb9f Binary files /dev/null and b/v13/assets/cc-comment-reg-type.png differ diff --git a/v13/assets/cc-comment-resize.gif b/v13/assets/cc-comment-resize.gif new file mode 100644 index 0000000000..5e3e5a7782 Binary files /dev/null and b/v13/assets/cc-comment-resize.gif differ diff --git a/v13/assets/cc-comment1.png b/v13/assets/cc-comment1.png new file mode 100644 index 0000000000..8a5a1f39d2 Binary files /dev/null and b/v13/assets/cc-comment1.png differ diff --git a/v13/assets/cc-comment2.png b/v13/assets/cc-comment2.png new file mode 100644 index 0000000000..807b4d0b7e Binary files /dev/null and b/v13/assets/cc-comment2.png differ diff --git a/v13/assets/cc-context-menu.png b/v13/assets/cc-context-menu.png new file mode 100644 index 0000000000..901e2270bb Binary files /dev/null and b/v13/assets/cc-context-menu.png differ diff --git a/v13/assets/cc-context-toolbar-open-overflow.png b/v13/assets/cc-context-toolbar-open-overflow.png new file mode 100644 index 0000000000..3b2df542cc Binary files /dev/null and b/v13/assets/cc-context-toolbar-open-overflow.png differ diff --git a/v13/assets/cc-context-toolbar.png b/v13/assets/cc-context-toolbar.png new file mode 100644 index 0000000000..8688b18833 Binary files /dev/null and b/v13/assets/cc-context-toolbar.png differ diff --git a/v13/assets/cc-decoration-example-link.png b/v13/assets/cc-decoration-example-link.png new file mode 100644 index 0000000000..5aa7f3b847 Binary files /dev/null and b/v13/assets/cc-decoration-example-link.png differ diff --git a/v13/assets/cc-decoration-example-node.png b/v13/assets/cc-decoration-example-node.png new file mode 100644 index 0000000000..f6bb55721b Binary files /dev/null and b/v13/assets/cc-decoration-example-node.png differ diff --git a/v13/assets/cc-decoration-label-editable.gif b/v13/assets/cc-decoration-label-editable.gif new file mode 100644 index 0000000000..d18c98d04a Binary files /dev/null and b/v13/assets/cc-decoration-label-editable.gif differ diff --git a/v13/assets/cc-decoration-link-image.png b/v13/assets/cc-decoration-link-image.png new file mode 100644 index 0000000000..136bd70c68 Binary files /dev/null and b/v13/assets/cc-decoration-link-image.png differ diff --git a/v13/assets/cc-decoration-link-jsx.png b/v13/assets/cc-decoration-link-jsx.png new file mode 100644 index 0000000000..505084481a Binary files /dev/null and b/v13/assets/cc-decoration-link-jsx.png differ diff --git a/v13/assets/cc-decoration-link-label.png b/v13/assets/cc-decoration-link-label.png new file mode 100644 index 0000000000..a4e497079a Binary files /dev/null and b/v13/assets/cc-decoration-link-label.png differ diff --git a/v13/assets/cc-decoration-link-shape.png b/v13/assets/cc-decoration-link-shape.png new file mode 100644 index 0000000000..4894032256 Binary files /dev/null and b/v13/assets/cc-decoration-link-shape.png differ diff --git a/v13/assets/cc-decoration-node-image.png b/v13/assets/cc-decoration-node-image.png new file mode 100644 index 0000000000..bf7d0c985b Binary files /dev/null and b/v13/assets/cc-decoration-node-image.png differ diff --git a/v13/assets/cc-decoration-node-jsx.png b/v13/assets/cc-decoration-node-jsx.png new file mode 100644 index 0000000000..fb9487adae Binary files /dev/null and b/v13/assets/cc-decoration-node-jsx.png differ diff --git a/v13/assets/cc-decoration-node-label.png b/v13/assets/cc-decoration-node-label.png new file mode 100644 index 0000000000..a84e1458af Binary files /dev/null and b/v13/assets/cc-decoration-node-label.png differ diff --git a/v13/assets/cc-decoration-node-shape.png b/v13/assets/cc-decoration-node-shape.png new file mode 100644 index 0000000000..a13c3adc0e Binary files /dev/null and b/v13/assets/cc-decoration-node-shape.png differ diff --git a/v13/assets/cc-decoration-tooltip.png b/v13/assets/cc-decoration-tooltip.png new file mode 100644 index 0000000000..d9b137f623 Binary files /dev/null and b/v13/assets/cc-decoration-tooltip.png differ diff --git a/v13/assets/cc-elements.png b/v13/assets/cc-elements.png new file mode 100644 index 0000000000..dd910a07a8 Binary files /dev/null and b/v13/assets/cc-elements.png differ diff --git a/v13/assets/cc-flow-editor-all-select.gif b/v13/assets/cc-flow-editor-all-select.gif new file mode 100644 index 0000000000..c7c6407636 Binary files /dev/null and b/v13/assets/cc-flow-editor-all-select.gif differ diff --git a/v13/assets/cc-flow-editor-clipboard.gif b/v13/assets/cc-flow-editor-clipboard.gif new file mode 100644 index 0000000000..3af4487c26 Binary files /dev/null and b/v13/assets/cc-flow-editor-clipboard.gif differ diff --git a/v13/assets/cc-flow-editor-command-stack.gif b/v13/assets/cc-flow-editor-command-stack.gif new file mode 100644 index 0000000000..48faebad6a Binary files /dev/null and b/v13/assets/cc-flow-editor-command-stack.gif differ diff --git a/v13/assets/cc-flow-editor-create-link.gif b/v13/assets/cc-flow-editor-create-link.gif new file mode 100644 index 0000000000..bfb75484b7 Binary files /dev/null and b/v13/assets/cc-flow-editor-create-link.gif differ diff --git a/v13/assets/cc-flow-editor-create-node.gif b/v13/assets/cc-flow-editor-create-node.gif new file mode 100644 index 0000000000..d4b6305028 Binary files /dev/null and b/v13/assets/cc-flow-editor-create-node.gif differ diff --git a/v13/assets/cc-flow-editor-linked-nodes-select.gif b/v13/assets/cc-flow-editor-linked-nodes-select.gif new file mode 100644 index 0000000000..ad680102dc Binary files /dev/null and b/v13/assets/cc-flow-editor-linked-nodes-select.gif differ diff --git a/v13/assets/cc-flow-editor-move.gif b/v13/assets/cc-flow-editor-move.gif new file mode 100644 index 0000000000..2505cd478b Binary files /dev/null and b/v13/assets/cc-flow-editor-move.gif differ diff --git a/v13/assets/cc-flow-editor-region-select.gif b/v13/assets/cc-flow-editor-region-select.gif new file mode 100644 index 0000000000..acd5b69cf5 Binary files /dev/null and b/v13/assets/cc-flow-editor-region-select.gif differ diff --git a/v13/assets/cc-flow-editor-zoom-to-fit.gif b/v13/assets/cc-flow-editor-zoom-to-fit.gif new file mode 100644 index 0000000000..9f7e84104e Binary files /dev/null and b/v13/assets/cc-flow-editor-zoom-to-fit.gif differ diff --git a/v13/assets/cc-flow-editor-zoom.gif b/v13/assets/cc-flow-editor-zoom.gif new file mode 100644 index 0000000000..78dc43e37e Binary files /dev/null and b/v13/assets/cc-flow-editor-zoom.gif differ diff --git a/v13/assets/cc-flow-editor.png b/v13/assets/cc-flow-editor.png new file mode 100644 index 0000000000..59588dabdc Binary files /dev/null and b/v13/assets/cc-flow-editor.png differ diff --git a/v13/assets/cc-link-assoc.png b/v13/assets/cc-link-assoc.png new file mode 100644 index 0000000000..cbb560e6ca Binary files /dev/null and b/v13/assets/cc-link-assoc.png differ diff --git a/v13/assets/cc-link-comment.png b/v13/assets/cc-link-comment.png new file mode 100644 index 0000000000..fbf4072e9e Binary files /dev/null and b/v13/assets/cc-link-comment.png differ diff --git a/v13/assets/cc-link-elements-dom.png b/v13/assets/cc-link-elements-dom.png new file mode 100644 index 0000000000..5d9eeefea3 Binary files /dev/null and b/v13/assets/cc-link-elements-dom.png differ diff --git a/v13/assets/cc-link-freeform-curve-separation.gif b/v13/assets/cc-link-freeform-curve-separation.gif new file mode 100644 index 0000000000..5f42b7f27f Binary files /dev/null and b/v13/assets/cc-link-freeform-curve-separation.gif differ diff --git a/v13/assets/cc-link-freeform-curve.png b/v13/assets/cc-link-freeform-curve.png new file mode 100644 index 0000000000..8c497222c8 Binary files /dev/null and b/v13/assets/cc-link-freeform-curve.png differ diff --git a/v13/assets/cc-link-freeform-elbow-self-ref.png b/v13/assets/cc-link-freeform-elbow-self-ref.png new file mode 100644 index 0000000000..c192eeb445 Binary files /dev/null and b/v13/assets/cc-link-freeform-elbow-self-ref.png differ diff --git a/v13/assets/cc-link-freeform-elbow.png b/v13/assets/cc-link-freeform-elbow.png new file mode 100644 index 0000000000..dc6f7c05e0 Binary files /dev/null and b/v13/assets/cc-link-freeform-elbow.png differ diff --git a/v13/assets/cc-link-freeform-parallax.png b/v13/assets/cc-link-freeform-parallax.png new file mode 100644 index 0000000000..db388887d3 Binary files /dev/null and b/v13/assets/cc-link-freeform-parallax.png differ diff --git a/v13/assets/cc-link-freeform-straight-separation.gif b/v13/assets/cc-link-freeform-straight-separation.gif new file mode 100644 index 0000000000..3f03ec83ed Binary files /dev/null and b/v13/assets/cc-link-freeform-straight-separation.gif differ diff --git a/v13/assets/cc-link-freeform-straight.gif b/v13/assets/cc-link-freeform-straight.gif new file mode 100644 index 0000000000..d093fba011 Binary files /dev/null and b/v13/assets/cc-link-freeform-straight.gif differ diff --git a/v13/assets/cc-link-freeform-straight.png b/v13/assets/cc-link-freeform-straight.png new file mode 100644 index 0000000000..c66992e6ae Binary files /dev/null and b/v13/assets/cc-link-freeform-straight.png differ diff --git a/v13/assets/cc-link-mapping-transformer.png b/v13/assets/cc-link-mapping-transformer.png new file mode 100644 index 0000000000..5aeca4253d Binary files /dev/null and b/v13/assets/cc-link-mapping-transformer.png differ diff --git a/v13/assets/cc-link-multi-ports-connected.png b/v13/assets/cc-link-multi-ports-connected.png new file mode 100644 index 0000000000..047a4d0037 Binary files /dev/null and b/v13/assets/cc-link-multi-ports-connected.png differ diff --git a/v13/assets/cc-link-multi-ports-quadrants.png b/v13/assets/cc-link-multi-ports-quadrants.png new file mode 100644 index 0000000000..85c5d1f7a1 Binary files /dev/null and b/v13/assets/cc-link-multi-ports-quadrants.png differ diff --git a/v13/assets/cc-link-multi-ports.png b/v13/assets/cc-link-multi-ports.png new file mode 100644 index 0000000000..2b5d5d9cae Binary files /dev/null and b/v13/assets/cc-link-multi-ports.png differ diff --git a/v13/assets/cc-link-ports-curve.png b/v13/assets/cc-link-ports-curve.png new file mode 100644 index 0000000000..01ab24c248 Binary files /dev/null and b/v13/assets/cc-link-ports-curve.png differ diff --git a/v13/assets/cc-link-ports-elbow-loop.png b/v13/assets/cc-link-ports-elbow-loop.png new file mode 100644 index 0000000000..771023c90f Binary files /dev/null and b/v13/assets/cc-link-ports-elbow-loop.png differ diff --git a/v13/assets/cc-link-ports-elbow.png b/v13/assets/cc-link-ports-elbow.png new file mode 100644 index 0000000000..3bca95f369 Binary files /dev/null and b/v13/assets/cc-link-ports-elbow.png differ diff --git a/v13/assets/cc-link-ports-parallax.png b/v13/assets/cc-link-ports-parallax.png new file mode 100644 index 0000000000..5e06ead24b Binary files /dev/null and b/v13/assets/cc-link-ports-parallax.png differ diff --git a/v13/assets/cc-link-ports-straight.png b/v13/assets/cc-link-ports-straight.png new file mode 100644 index 0000000000..5e69ab4058 Binary files /dev/null and b/v13/assets/cc-link-ports-straight.png differ diff --git a/v13/assets/cc-link-ports.gif b/v13/assets/cc-link-ports.gif new file mode 100644 index 0000000000..79110c5a74 Binary files /dev/null and b/v13/assets/cc-link-ports.gif differ diff --git a/v13/assets/cc-node-elements-dom.png b/v13/assets/cc-node-elements-dom.png new file mode 100644 index 0000000000..ec52d231d9 Binary files /dev/null and b/v13/assets/cc-node-elements-dom.png differ diff --git a/v13/assets/cc-node-elements-pos.png b/v13/assets/cc-node-elements-pos.png new file mode 100644 index 0000000000..5771198bd3 Binary files /dev/null and b/v13/assets/cc-node-elements-pos.png differ diff --git a/v13/assets/cc-node-elements.png b/v13/assets/cc-node-elements.png new file mode 100644 index 0000000000..de47a1bfe0 Binary files /dev/null and b/v13/assets/cc-node-elements.png differ diff --git a/v13/assets/cc-node-filter.png b/v13/assets/cc-node-filter.png new file mode 100644 index 0000000000..b19af9f96b Binary files /dev/null and b/v13/assets/cc-node-filter.png differ diff --git a/v13/assets/cc-node-format.png b/v13/assets/cc-node-format.png new file mode 100644 index 0000000000..96a52f89ce Binary files /dev/null and b/v13/assets/cc-node-format.png differ diff --git a/v13/assets/cc-node-guide1.gif b/v13/assets/cc-node-guide1.gif new file mode 100644 index 0000000000..f023758073 Binary files /dev/null and b/v13/assets/cc-node-guide1.gif differ diff --git a/v13/assets/cc-node-guide2.gif b/v13/assets/cc-node-guide2.gif new file mode 100644 index 0000000000..3caf998277 Binary files /dev/null and b/v13/assets/cc-node-guide2.gif differ diff --git a/v13/assets/cc-node-highlight-target.gif b/v13/assets/cc-node-highlight-target.gif new file mode 100644 index 0000000000..96c14fdfce Binary files /dev/null and b/v13/assets/cc-node-highlight-target.gif differ diff --git a/v13/assets/cc-node-image-jsx-app.png b/v13/assets/cc-node-image-jsx-app.png new file mode 100644 index 0000000000..bc6cedbdc5 Binary files /dev/null and b/v13/assets/cc-node-image-jsx-app.png differ diff --git a/v13/assets/cc-node-image-jsx.png b/v13/assets/cc-node-image-jsx.png new file mode 100644 index 0000000000..2763111343 Binary files /dev/null and b/v13/assets/cc-node-image-jsx.png differ diff --git a/v13/assets/cc-node-image-svg.png b/v13/assets/cc-node-image-svg.png new file mode 100644 index 0000000000..0fe23c8e81 Binary files /dev/null and b/v13/assets/cc-node-image-svg.png differ diff --git a/v13/assets/cc-node-insert-to-link.gif b/v13/assets/cc-node-insert-to-link.gif new file mode 100644 index 0000000000..9dd7741794 Binary files /dev/null and b/v13/assets/cc-node-insert-to-link.gif differ diff --git a/v13/assets/cc-node-port-custom-resize.png b/v13/assets/cc-node-port-custom-resize.png new file mode 100644 index 0000000000..12ac0030d7 Binary files /dev/null and b/v13/assets/cc-node-port-custom-resize.png differ diff --git a/v13/assets/cc-node-port-custom.png b/v13/assets/cc-node-port-custom.png new file mode 100644 index 0000000000..828d489002 Binary files /dev/null and b/v13/assets/cc-node-port-custom.png differ diff --git a/v13/assets/cc-node-react.png b/v13/assets/cc-node-react.png new file mode 100644 index 0000000000..3c2a2127af Binary files /dev/null and b/v13/assets/cc-node-react.png differ diff --git a/v13/assets/cc-node-resize.gif b/v13/assets/cc-node-resize.gif new file mode 100644 index 0000000000..214f392a3c Binary files /dev/null and b/v13/assets/cc-node-resize.gif differ diff --git a/v13/assets/cc-node-supernode-create.gif b/v13/assets/cc-node-supernode-create.gif new file mode 100644 index 0000000000..d6f7a0bc81 Binary files /dev/null and b/v13/assets/cc-node-supernode-create.gif differ diff --git a/v13/assets/cc-node-supernode-full-page.gif b/v13/assets/cc-node-supernode-full-page.gif new file mode 100644 index 0000000000..3d04a3e077 Binary files /dev/null and b/v13/assets/cc-node-supernode-full-page.gif differ diff --git a/v13/assets/cc-node-supernode-in-place.png b/v13/assets/cc-node-supernode-in-place.png new file mode 100644 index 0000000000..71b0aa28b9 Binary files /dev/null and b/v13/assets/cc-node-supernode-in-place.png differ diff --git a/v13/assets/cc-node1.png b/v13/assets/cc-node1.png new file mode 100644 index 0000000000..a120be2270 Binary files /dev/null and b/v13/assets/cc-node1.png differ diff --git a/v13/assets/cc-node2.png b/v13/assets/cc-node2.png new file mode 100644 index 0000000000..064802462f Binary files /dev/null and b/v13/assets/cc-node2.png differ diff --git a/v13/assets/cc-node3.png b/v13/assets/cc-node3.png new file mode 100644 index 0000000000..92c0b14f63 Binary files /dev/null and b/v13/assets/cc-node3.png differ diff --git a/v13/assets/cc-node4.png b/v13/assets/cc-node4.png new file mode 100644 index 0000000000..7f54de125b Binary files /dev/null and b/v13/assets/cc-node4.png differ diff --git a/v13/assets/cc-node5.png b/v13/assets/cc-node5.png new file mode 100644 index 0000000000..293ce4662e Binary files /dev/null and b/v13/assets/cc-node5.png differ diff --git a/v13/assets/cc-node6.png b/v13/assets/cc-node6.png new file mode 100644 index 0000000000..50fe7ca66b Binary files /dev/null and b/v13/assets/cc-node6.png differ diff --git a/v13/assets/cc-node7.png b/v13/assets/cc-node7.png new file mode 100644 index 0000000000..009450df06 Binary files /dev/null and b/v13/assets/cc-node7.png differ diff --git a/v13/assets/cc-notif-icon-error.png b/v13/assets/cc-notif-icon-error.png new file mode 100644 index 0000000000..b303e1eb81 Binary files /dev/null and b/v13/assets/cc-notif-icon-error.png differ diff --git a/v13/assets/cc-notif-icon-info.png b/v13/assets/cc-notif-icon-info.png new file mode 100644 index 0000000000..ba3f0ee55e Binary files /dev/null and b/v13/assets/cc-notif-icon-info.png differ diff --git a/v13/assets/cc-notif-icon-success.png b/v13/assets/cc-notif-icon-success.png new file mode 100644 index 0000000000..01f3476f5e Binary files /dev/null and b/v13/assets/cc-notif-icon-success.png differ diff --git a/v13/assets/cc-notif-icon-warning.png b/v13/assets/cc-notif-icon-warning.png new file mode 100644 index 0000000000..7370a8e6ca Binary files /dev/null and b/v13/assets/cc-notif-icon-warning.png differ diff --git a/v13/assets/cc-notification-panel.png b/v13/assets/cc-notification-panel.png new file mode 100644 index 0000000000..b29b33dd5e Binary files /dev/null and b/v13/assets/cc-notification-panel.png differ diff --git a/v13/assets/cc-palette-double-click.gif b/v13/assets/cc-palette-double-click.gif new file mode 100644 index 0000000000..ca470a2266 Binary files /dev/null and b/v13/assets/cc-palette-double-click.gif differ diff --git a/v13/assets/cc-palette.gif b/v13/assets/cc-palette.gif new file mode 100644 index 0000000000..deee458786 Binary files /dev/null and b/v13/assets/cc-palette.gif differ diff --git a/v13/assets/cc-panels.png b/v13/assets/cc-panels.png new file mode 100644 index 0000000000..8daaba4bed Binary files /dev/null and b/v13/assets/cc-panels.png differ diff --git a/v13/assets/cc-pipeline-flow-supernode-diagram.png b/v13/assets/cc-pipeline-flow-supernode-diagram.png new file mode 100644 index 0000000000..671a1ad7cc Binary files /dev/null and b/v13/assets/cc-pipeline-flow-supernode-diagram.png differ diff --git a/v13/assets/cc-state-tag.png b/v13/assets/cc-state-tag.png new file mode 100644 index 0000000000..9f4cb0310c Binary files /dev/null and b/v13/assets/cc-state-tag.png differ diff --git a/v13/assets/cc-toolbar-action-disabled.png b/v13/assets/cc-toolbar-action-disabled.png new file mode 100644 index 0000000000..17f92443db Binary files /dev/null and b/v13/assets/cc-toolbar-action-disabled.png differ diff --git a/v13/assets/cc-toolbar-action-dual-closed.png b/v13/assets/cc-toolbar-action-dual-closed.png new file mode 100644 index 0000000000..ee98dc8e0e Binary files /dev/null and b/v13/assets/cc-toolbar-action-dual-closed.png differ diff --git a/v13/assets/cc-toolbar-action-dual-open.png b/v13/assets/cc-toolbar-action-dual-open.png new file mode 100644 index 0000000000..c7ecdcd957 Binary files /dev/null and b/v13/assets/cc-toolbar-action-dual-open.png differ diff --git a/v13/assets/cc-toolbar-action-enabled.png b/v13/assets/cc-toolbar-action-enabled.png new file mode 100644 index 0000000000..3c9096f467 Binary files /dev/null and b/v13/assets/cc-toolbar-action-enabled.png differ diff --git a/v13/assets/cc-toolbar-action-jsx-buttons.png b/v13/assets/cc-toolbar-action-jsx-buttons.png new file mode 100644 index 0000000000..18a92ca5a6 Binary files /dev/null and b/v13/assets/cc-toolbar-action-jsx-buttons.png differ diff --git a/v13/assets/cc-toolbar-action-jsx.png b/v13/assets/cc-toolbar-action-jsx.png new file mode 100644 index 0000000000..36291ac5e1 Binary files /dev/null and b/v13/assets/cc-toolbar-action-jsx.png differ diff --git a/v13/assets/cc-toolbar-action-kinds.png b/v13/assets/cc-toolbar-action-kinds.png new file mode 100644 index 0000000000..eb8f2ec294 Binary files /dev/null and b/v13/assets/cc-toolbar-action-kinds.png differ diff --git a/v13/assets/cc-toolbar-action-selected-menu.png b/v13/assets/cc-toolbar-action-selected-menu.png new file mode 100644 index 0000000000..84545d926e Binary files /dev/null and b/v13/assets/cc-toolbar-action-selected-menu.png differ diff --git a/v13/assets/cc-toolbar-action-selected-none.png b/v13/assets/cc-toolbar-action-selected-none.png new file mode 100644 index 0000000000..ba6ffbf16d Binary files /dev/null and b/v13/assets/cc-toolbar-action-selected-none.png differ diff --git a/v13/assets/cc-toolbar-action-selected.png b/v13/assets/cc-toolbar-action-selected.png new file mode 100644 index 0000000000..76f8644f8b Binary files /dev/null and b/v13/assets/cc-toolbar-action-selected.png differ diff --git a/v13/assets/cc-toolbar-action-sub-menu-open.png b/v13/assets/cc-toolbar-action-sub-menu-open.png new file mode 100644 index 0000000000..99c11f468a Binary files /dev/null and b/v13/assets/cc-toolbar-action-sub-menu-open.png differ diff --git a/v13/assets/cc-toolbar-action-sub-panel-open.png b/v13/assets/cc-toolbar-action-sub-panel-open.png new file mode 100644 index 0000000000..b6c10d0637 Binary files /dev/null and b/v13/assets/cc-toolbar-action-sub-panel-open.png differ diff --git a/v13/assets/cc-toolbar-action-text-after.png b/v13/assets/cc-toolbar-action-text-after.png new file mode 100644 index 0000000000..df64c196ab Binary files /dev/null and b/v13/assets/cc-toolbar-action-text-after.png differ diff --git a/v13/assets/cc-toolbar-action-text-before.png b/v13/assets/cc-toolbar-action-text-before.png new file mode 100644 index 0000000000..cc0e4f73a0 Binary files /dev/null and b/v13/assets/cc-toolbar-action-text-before.png differ diff --git a/v13/assets/cc-toolbar-action-tooltip.png b/v13/assets/cc-toolbar-action-tooltip.png new file mode 100644 index 0000000000..40623f93c2 Binary files /dev/null and b/v13/assets/cc-toolbar-action-tooltip.png differ diff --git a/v13/assets/cc-toolbar-auto-customize.png b/v13/assets/cc-toolbar-auto-customize.png new file mode 100644 index 0000000000..018b1db253 Binary files /dev/null and b/v13/assets/cc-toolbar-auto-customize.png differ diff --git a/v13/assets/cc-toolbar-divider.png b/v13/assets/cc-toolbar-divider.png new file mode 100644 index 0000000000..24f264c35a Binary files /dev/null and b/v13/assets/cc-toolbar-divider.png differ diff --git a/v13/assets/cc-toolbar-overflow-closed.png b/v13/assets/cc-toolbar-overflow-closed.png new file mode 100644 index 0000000000..ded506fc11 Binary files /dev/null and b/v13/assets/cc-toolbar-overflow-closed.png differ diff --git a/v13/assets/cc-toolbar-overflow-none.png b/v13/assets/cc-toolbar-overflow-none.png new file mode 100644 index 0000000000..6d1d3b7c9e Binary files /dev/null and b/v13/assets/cc-toolbar-overflow-none.png differ diff --git a/v13/assets/cc-toolbar-overflow-open.png b/v13/assets/cc-toolbar-overflow-open.png new file mode 100644 index 0000000000..df247e3361 Binary files /dev/null and b/v13/assets/cc-toolbar-overflow-open.png differ diff --git a/v13/assets/cc-toolbar-tooltips.gif b/v13/assets/cc-toolbar-tooltips.gif new file mode 100644 index 0000000000..10c276abe5 Binary files /dev/null and b/v13/assets/cc-toolbar-tooltips.gif differ diff --git a/v13/assets/cc-toolbar.png b/v13/assets/cc-toolbar.png new file mode 100644 index 0000000000..64afd7ce32 Binary files /dev/null and b/v13/assets/cc-toolbar.png differ diff --git a/v13/assets/components.png b/v13/assets/components.png new file mode 100644 index 0000000000..267c998c72 Binary files /dev/null and b/v13/assets/components.png differ diff --git a/v13/assets/cp-example.png b/v13/assets/cp-example.png new file mode 100644 index 0000000000..08cd4c9376 Binary files /dev/null and b/v13/assets/cp-example.png differ diff --git a/v13/assets/ellipsis-icon.png b/v13/assets/ellipsis-icon.png new file mode 100644 index 0000000000..bcf8616f84 Binary files /dev/null and b/v13/assets/ellipsis-icon.png differ diff --git a/v13/assets/faq-react-supernodes.png b/v13/assets/faq-react-supernodes.png new file mode 100644 index 0000000000..a97d40e9b7 Binary files /dev/null and b/v13/assets/faq-react-supernodes.png differ diff --git a/v13/assets/flow.svg b/v13/assets/flow.svg new file mode 100644 index 0000000000..20d3883968 --- /dev/null +++ b/v13/assets/flow.svg @@ -0,0 +1 @@ +flow--data \ No newline at end of file diff --git a/v13/assets/images/favicon.png b/v13/assets/images/favicon.png new file mode 100644 index 0000000000..1cf13b9f9d Binary files /dev/null and b/v13/assets/images/favicon.png differ diff --git a/v13/assets/javascripts/bundle.88dd0f4e.min.js b/v13/assets/javascripts/bundle.88dd0f4e.min.js new file mode 100644 index 0000000000..fb8f31090f --- /dev/null +++ b/v13/assets/javascripts/bundle.88dd0f4e.min.js @@ -0,0 +1,16 @@ +"use strict";(()=>{var Wi=Object.create;var gr=Object.defineProperty;var Di=Object.getOwnPropertyDescriptor;var Vi=Object.getOwnPropertyNames,Vt=Object.getOwnPropertySymbols,Ni=Object.getPrototypeOf,yr=Object.prototype.hasOwnProperty,ao=Object.prototype.propertyIsEnumerable;var io=(e,t,r)=>t in e?gr(e,t,{enumerable:!0,configurable:!0,writable:!0,value:r}):e[t]=r,$=(e,t)=>{for(var r in t||(t={}))yr.call(t,r)&&io(e,r,t[r]);if(Vt)for(var r of Vt(t))ao.call(t,r)&&io(e,r,t[r]);return e};var so=(e,t)=>{var r={};for(var o in e)yr.call(e,o)&&t.indexOf(o)<0&&(r[o]=e[o]);if(e!=null&&Vt)for(var o of Vt(e))t.indexOf(o)<0&&ao.call(e,o)&&(r[o]=e[o]);return r};var xr=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports);var zi=(e,t,r,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let n of Vi(t))!yr.call(e,n)&&n!==r&&gr(e,n,{get:()=>t[n],enumerable:!(o=Di(t,n))||o.enumerable});return e};var Mt=(e,t,r)=>(r=e!=null?Wi(Ni(e)):{},zi(t||!e||!e.__esModule?gr(r,"default",{value:e,enumerable:!0}):r,e));var co=(e,t,r)=>new Promise((o,n)=>{var i=p=>{try{s(r.next(p))}catch(c){n(c)}},a=p=>{try{s(r.throw(p))}catch(c){n(c)}},s=p=>p.done?o(p.value):Promise.resolve(p.value).then(i,a);s((r=r.apply(e,t)).next())});var lo=xr((Er,po)=>{(function(e,t){typeof Er=="object"&&typeof po!="undefined"?t():typeof define=="function"&&define.amd?define(t):t()})(Er,function(){"use strict";function e(r){var o=!0,n=!1,i=null,a={text:!0,search:!0,url:!0,tel:!0,email:!0,password:!0,number:!0,date:!0,month:!0,week:!0,time:!0,datetime:!0,"datetime-local":!0};function s(k){return!!(k&&k!==document&&k.nodeName!=="HTML"&&k.nodeName!=="BODY"&&"classList"in k&&"contains"in k.classList)}function p(k){var ft=k.type,qe=k.tagName;return!!(qe==="INPUT"&&a[ft]&&!k.readOnly||qe==="TEXTAREA"&&!k.readOnly||k.isContentEditable)}function c(k){k.classList.contains("focus-visible")||(k.classList.add("focus-visible"),k.setAttribute("data-focus-visible-added",""))}function l(k){k.hasAttribute("data-focus-visible-added")&&(k.classList.remove("focus-visible"),k.removeAttribute("data-focus-visible-added"))}function f(k){k.metaKey||k.altKey||k.ctrlKey||(s(r.activeElement)&&c(r.activeElement),o=!0)}function u(k){o=!1}function d(k){s(k.target)&&(o||p(k.target))&&c(k.target)}function y(k){s(k.target)&&(k.target.classList.contains("focus-visible")||k.target.hasAttribute("data-focus-visible-added"))&&(n=!0,window.clearTimeout(i),i=window.setTimeout(function(){n=!1},100),l(k.target))}function L(k){document.visibilityState==="hidden"&&(n&&(o=!0),X())}function X(){document.addEventListener("mousemove",J),document.addEventListener("mousedown",J),document.addEventListener("mouseup",J),document.addEventListener("pointermove",J),document.addEventListener("pointerdown",J),document.addEventListener("pointerup",J),document.addEventListener("touchmove",J),document.addEventListener("touchstart",J),document.addEventListener("touchend",J)}function te(){document.removeEventListener("mousemove",J),document.removeEventListener("mousedown",J),document.removeEventListener("mouseup",J),document.removeEventListener("pointermove",J),document.removeEventListener("pointerdown",J),document.removeEventListener("pointerup",J),document.removeEventListener("touchmove",J),document.removeEventListener("touchstart",J),document.removeEventListener("touchend",J)}function J(k){k.target.nodeName&&k.target.nodeName.toLowerCase()==="html"||(o=!1,te())}document.addEventListener("keydown",f,!0),document.addEventListener("mousedown",u,!0),document.addEventListener("pointerdown",u,!0),document.addEventListener("touchstart",u,!0),document.addEventListener("visibilitychange",L,!0),X(),r.addEventListener("focus",d,!0),r.addEventListener("blur",y,!0),r.nodeType===Node.DOCUMENT_FRAGMENT_NODE&&r.host?r.host.setAttribute("data-js-focus-visible",""):r.nodeType===Node.DOCUMENT_NODE&&(document.documentElement.classList.add("js-focus-visible"),document.documentElement.setAttribute("data-js-focus-visible",""))}if(typeof window!="undefined"&&typeof document!="undefined"){window.applyFocusVisiblePolyfill=e;var t;try{t=new CustomEvent("focus-visible-polyfill-ready")}catch(r){t=document.createEvent("CustomEvent"),t.initCustomEvent("focus-visible-polyfill-ready",!1,!1,{})}window.dispatchEvent(t)}typeof document!="undefined"&&e(document)})});var qr=xr((hy,On)=>{"use strict";/*! + * escape-html + * Copyright(c) 2012-2013 TJ Holowaychuk + * Copyright(c) 2015 Andreas Lubbe + * Copyright(c) 2015 Tiancheng "Timothy" Gu + * MIT Licensed + */var $a=/["'&<>]/;On.exports=Pa;function Pa(e){var t=""+e,r=$a.exec(t);if(!r)return t;var o,n="",i=0,a=0;for(i=r.index;i{/*! + * clipboard.js v2.0.11 + * https://clipboardjs.com/ + * + * Licensed MIT © Zeno Rocha + */(function(t,r){typeof It=="object"&&typeof Yr=="object"?Yr.exports=r():typeof define=="function"&&define.amd?define([],r):typeof It=="object"?It.ClipboardJS=r():t.ClipboardJS=r()})(It,function(){return function(){var e={686:function(o,n,i){"use strict";i.d(n,{default:function(){return Ui}});var a=i(279),s=i.n(a),p=i(370),c=i.n(p),l=i(817),f=i.n(l);function u(V){try{return document.execCommand(V)}catch(A){return!1}}var d=function(A){var M=f()(A);return u("cut"),M},y=d;function L(V){var A=document.documentElement.getAttribute("dir")==="rtl",M=document.createElement("textarea");M.style.fontSize="12pt",M.style.border="0",M.style.padding="0",M.style.margin="0",M.style.position="absolute",M.style[A?"right":"left"]="-9999px";var F=window.pageYOffset||document.documentElement.scrollTop;return M.style.top="".concat(F,"px"),M.setAttribute("readonly",""),M.value=V,M}var X=function(A,M){var F=L(A);M.container.appendChild(F);var D=f()(F);return u("copy"),F.remove(),D},te=function(A){var M=arguments.length>1&&arguments[1]!==void 0?arguments[1]:{container:document.body},F="";return typeof A=="string"?F=X(A,M):A instanceof HTMLInputElement&&!["text","search","url","tel","password"].includes(A==null?void 0:A.type)?F=X(A.value,M):(F=f()(A),u("copy")),F},J=te;function k(V){"@babel/helpers - typeof";return typeof Symbol=="function"&&typeof Symbol.iterator=="symbol"?k=function(M){return typeof M}:k=function(M){return M&&typeof Symbol=="function"&&M.constructor===Symbol&&M!==Symbol.prototype?"symbol":typeof M},k(V)}var ft=function(){var A=arguments.length>0&&arguments[0]!==void 0?arguments[0]:{},M=A.action,F=M===void 0?"copy":M,D=A.container,Y=A.target,$e=A.text;if(F!=="copy"&&F!=="cut")throw new Error('Invalid "action" value, use either "copy" or "cut"');if(Y!==void 0)if(Y&&k(Y)==="object"&&Y.nodeType===1){if(F==="copy"&&Y.hasAttribute("disabled"))throw new Error('Invalid "target" attribute. Please use "readonly" instead of "disabled" attribute');if(F==="cut"&&(Y.hasAttribute("readonly")||Y.hasAttribute("disabled")))throw new Error(`Invalid "target" attribute. You can't cut text from elements with "readonly" or "disabled" attributes`)}else throw new Error('Invalid "target" value, use a valid Element');if($e)return J($e,{container:D});if(Y)return F==="cut"?y(Y):J(Y,{container:D})},qe=ft;function Fe(V){"@babel/helpers - typeof";return typeof Symbol=="function"&&typeof Symbol.iterator=="symbol"?Fe=function(M){return typeof M}:Fe=function(M){return M&&typeof Symbol=="function"&&M.constructor===Symbol&&M!==Symbol.prototype?"symbol":typeof M},Fe(V)}function ki(V,A){if(!(V instanceof A))throw new TypeError("Cannot call a class as a function")}function no(V,A){for(var M=0;M0&&arguments[0]!==void 0?arguments[0]:{};this.action=typeof D.action=="function"?D.action:this.defaultAction,this.target=typeof D.target=="function"?D.target:this.defaultTarget,this.text=typeof D.text=="function"?D.text:this.defaultText,this.container=Fe(D.container)==="object"?D.container:document.body}},{key:"listenClick",value:function(D){var Y=this;this.listener=c()(D,"click",function($e){return Y.onClick($e)})}},{key:"onClick",value:function(D){var Y=D.delegateTarget||D.currentTarget,$e=this.action(Y)||"copy",Dt=qe({action:$e,container:this.container,target:this.target(Y),text:this.text(Y)});this.emit(Dt?"success":"error",{action:$e,text:Dt,trigger:Y,clearSelection:function(){Y&&Y.focus(),window.getSelection().removeAllRanges()}})}},{key:"defaultAction",value:function(D){return vr("action",D)}},{key:"defaultTarget",value:function(D){var Y=vr("target",D);if(Y)return document.querySelector(Y)}},{key:"defaultText",value:function(D){return vr("text",D)}},{key:"destroy",value:function(){this.listener.destroy()}}],[{key:"copy",value:function(D){var Y=arguments.length>1&&arguments[1]!==void 0?arguments[1]:{container:document.body};return J(D,Y)}},{key:"cut",value:function(D){return y(D)}},{key:"isSupported",value:function(){var D=arguments.length>0&&arguments[0]!==void 0?arguments[0]:["copy","cut"],Y=typeof D=="string"?[D]:D,$e=!!document.queryCommandSupported;return Y.forEach(function(Dt){$e=$e&&!!document.queryCommandSupported(Dt)}),$e}}]),M}(s()),Ui=Fi},828:function(o){var n=9;if(typeof Element!="undefined"&&!Element.prototype.matches){var i=Element.prototype;i.matches=i.matchesSelector||i.mozMatchesSelector||i.msMatchesSelector||i.oMatchesSelector||i.webkitMatchesSelector}function a(s,p){for(;s&&s.nodeType!==n;){if(typeof s.matches=="function"&&s.matches(p))return s;s=s.parentNode}}o.exports=a},438:function(o,n,i){var a=i(828);function s(l,f,u,d,y){var L=c.apply(this,arguments);return l.addEventListener(u,L,y),{destroy:function(){l.removeEventListener(u,L,y)}}}function p(l,f,u,d,y){return typeof l.addEventListener=="function"?s.apply(null,arguments):typeof u=="function"?s.bind(null,document).apply(null,arguments):(typeof l=="string"&&(l=document.querySelectorAll(l)),Array.prototype.map.call(l,function(L){return s(L,f,u,d,y)}))}function c(l,f,u,d){return function(y){y.delegateTarget=a(y.target,f),y.delegateTarget&&d.call(l,y)}}o.exports=p},879:function(o,n){n.node=function(i){return i!==void 0&&i instanceof HTMLElement&&i.nodeType===1},n.nodeList=function(i){var a=Object.prototype.toString.call(i);return i!==void 0&&(a==="[object NodeList]"||a==="[object HTMLCollection]")&&"length"in i&&(i.length===0||n.node(i[0]))},n.string=function(i){return typeof i=="string"||i instanceof String},n.fn=function(i){var a=Object.prototype.toString.call(i);return a==="[object Function]"}},370:function(o,n,i){var a=i(879),s=i(438);function p(u,d,y){if(!u&&!d&&!y)throw new Error("Missing required arguments");if(!a.string(d))throw new TypeError("Second argument must be a String");if(!a.fn(y))throw new TypeError("Third argument must be a Function");if(a.node(u))return c(u,d,y);if(a.nodeList(u))return l(u,d,y);if(a.string(u))return f(u,d,y);throw new TypeError("First argument must be a String, HTMLElement, HTMLCollection, or NodeList")}function c(u,d,y){return u.addEventListener(d,y),{destroy:function(){u.removeEventListener(d,y)}}}function l(u,d,y){return Array.prototype.forEach.call(u,function(L){L.addEventListener(d,y)}),{destroy:function(){Array.prototype.forEach.call(u,function(L){L.removeEventListener(d,y)})}}}function f(u,d,y){return s(document.body,u,d,y)}o.exports=p},817:function(o){function n(i){var a;if(i.nodeName==="SELECT")i.focus(),a=i.value;else if(i.nodeName==="INPUT"||i.nodeName==="TEXTAREA"){var s=i.hasAttribute("readonly");s||i.setAttribute("readonly",""),i.select(),i.setSelectionRange(0,i.value.length),s||i.removeAttribute("readonly"),a=i.value}else{i.hasAttribute("contenteditable")&&i.focus();var p=window.getSelection(),c=document.createRange();c.selectNodeContents(i),p.removeAllRanges(),p.addRange(c),a=p.toString()}return a}o.exports=n},279:function(o){function n(){}n.prototype={on:function(i,a,s){var p=this.e||(this.e={});return(p[i]||(p[i]=[])).push({fn:a,ctx:s}),this},once:function(i,a,s){var p=this;function c(){p.off(i,c),a.apply(s,arguments)}return c._=a,this.on(i,c,s)},emit:function(i){var a=[].slice.call(arguments,1),s=((this.e||(this.e={}))[i]||[]).slice(),p=0,c=s.length;for(p;p0&&i[i.length-1])&&(c[0]===6||c[0]===2)){r=0;continue}if(c[0]===3&&(!i||c[1]>i[0]&&c[1]=e.length&&(e=void 0),{value:e&&e[o++],done:!e}}};throw new TypeError(t?"Object is not iterable.":"Symbol.iterator is not defined.")}function N(e,t){var r=typeof Symbol=="function"&&e[Symbol.iterator];if(!r)return e;var o=r.call(e),n,i=[],a;try{for(;(t===void 0||t-- >0)&&!(n=o.next()).done;)i.push(n.value)}catch(s){a={error:s}}finally{try{n&&!n.done&&(r=o.return)&&r.call(o)}finally{if(a)throw a.error}}return i}function q(e,t,r){if(r||arguments.length===2)for(var o=0,n=t.length,i;o1||p(d,L)})},y&&(n[d]=y(n[d])))}function p(d,y){try{c(o[d](y))}catch(L){u(i[0][3],L)}}function c(d){d.value instanceof nt?Promise.resolve(d.value.v).then(l,f):u(i[0][2],d)}function l(d){p("next",d)}function f(d){p("throw",d)}function u(d,y){d(y),i.shift(),i.length&&p(i[0][0],i[0][1])}}function uo(e){if(!Symbol.asyncIterator)throw new TypeError("Symbol.asyncIterator is not defined.");var t=e[Symbol.asyncIterator],r;return t?t.call(e):(e=typeof he=="function"?he(e):e[Symbol.iterator](),r={},o("next"),o("throw"),o("return"),r[Symbol.asyncIterator]=function(){return this},r);function o(i){r[i]=e[i]&&function(a){return new Promise(function(s,p){a=e[i](a),n(s,p,a.done,a.value)})}}function n(i,a,s,p){Promise.resolve(p).then(function(c){i({value:c,done:s})},a)}}function H(e){return typeof e=="function"}function ut(e){var t=function(o){Error.call(o),o.stack=new Error().stack},r=e(t);return r.prototype=Object.create(Error.prototype),r.prototype.constructor=r,r}var zt=ut(function(e){return function(r){e(this),this.message=r?r.length+` errors occurred during unsubscription: +`+r.map(function(o,n){return n+1+") "+o.toString()}).join(` + `):"",this.name="UnsubscriptionError",this.errors=r}});function Qe(e,t){if(e){var r=e.indexOf(t);0<=r&&e.splice(r,1)}}var Ue=function(){function e(t){this.initialTeardown=t,this.closed=!1,this._parentage=null,this._finalizers=null}return e.prototype.unsubscribe=function(){var t,r,o,n,i;if(!this.closed){this.closed=!0;var a=this._parentage;if(a)if(this._parentage=null,Array.isArray(a))try{for(var s=he(a),p=s.next();!p.done;p=s.next()){var c=p.value;c.remove(this)}}catch(L){t={error:L}}finally{try{p&&!p.done&&(r=s.return)&&r.call(s)}finally{if(t)throw t.error}}else a.remove(this);var l=this.initialTeardown;if(H(l))try{l()}catch(L){i=L instanceof zt?L.errors:[L]}var f=this._finalizers;if(f){this._finalizers=null;try{for(var u=he(f),d=u.next();!d.done;d=u.next()){var y=d.value;try{ho(y)}catch(L){i=i!=null?i:[],L instanceof zt?i=q(q([],N(i)),N(L.errors)):i.push(L)}}}catch(L){o={error:L}}finally{try{d&&!d.done&&(n=u.return)&&n.call(u)}finally{if(o)throw o.error}}}if(i)throw new zt(i)}},e.prototype.add=function(t){var r;if(t&&t!==this)if(this.closed)ho(t);else{if(t instanceof e){if(t.closed||t._hasParent(this))return;t._addParent(this)}(this._finalizers=(r=this._finalizers)!==null&&r!==void 0?r:[]).push(t)}},e.prototype._hasParent=function(t){var r=this._parentage;return r===t||Array.isArray(r)&&r.includes(t)},e.prototype._addParent=function(t){var r=this._parentage;this._parentage=Array.isArray(r)?(r.push(t),r):r?[r,t]:t},e.prototype._removeParent=function(t){var r=this._parentage;r===t?this._parentage=null:Array.isArray(r)&&Qe(r,t)},e.prototype.remove=function(t){var r=this._finalizers;r&&Qe(r,t),t instanceof e&&t._removeParent(this)},e.EMPTY=function(){var t=new e;return t.closed=!0,t}(),e}();var Tr=Ue.EMPTY;function qt(e){return e instanceof Ue||e&&"closed"in e&&H(e.remove)&&H(e.add)&&H(e.unsubscribe)}function ho(e){H(e)?e():e.unsubscribe()}var Pe={onUnhandledError:null,onStoppedNotification:null,Promise:void 0,useDeprecatedSynchronousErrorHandling:!1,useDeprecatedNextContext:!1};var dt={setTimeout:function(e,t){for(var r=[],o=2;o0},enumerable:!1,configurable:!0}),t.prototype._trySubscribe=function(r){return this._throwIfClosed(),e.prototype._trySubscribe.call(this,r)},t.prototype._subscribe=function(r){return this._throwIfClosed(),this._checkFinalizedStatuses(r),this._innerSubscribe(r)},t.prototype._innerSubscribe=function(r){var o=this,n=this,i=n.hasError,a=n.isStopped,s=n.observers;return i||a?Tr:(this.currentObservers=null,s.push(r),new Ue(function(){o.currentObservers=null,Qe(s,r)}))},t.prototype._checkFinalizedStatuses=function(r){var o=this,n=o.hasError,i=o.thrownError,a=o.isStopped;n?r.error(i):a&&r.complete()},t.prototype.asObservable=function(){var r=new j;return r.source=this,r},t.create=function(r,o){return new To(r,o)},t}(j);var To=function(e){oe(t,e);function t(r,o){var n=e.call(this)||this;return n.destination=r,n.source=o,n}return t.prototype.next=function(r){var o,n;(n=(o=this.destination)===null||o===void 0?void 0:o.next)===null||n===void 0||n.call(o,r)},t.prototype.error=function(r){var o,n;(n=(o=this.destination)===null||o===void 0?void 0:o.error)===null||n===void 0||n.call(o,r)},t.prototype.complete=function(){var r,o;(o=(r=this.destination)===null||r===void 0?void 0:r.complete)===null||o===void 0||o.call(r)},t.prototype._subscribe=function(r){var o,n;return(n=(o=this.source)===null||o===void 0?void 0:o.subscribe(r))!==null&&n!==void 0?n:Tr},t}(g);var _r=function(e){oe(t,e);function t(r){var o=e.call(this)||this;return o._value=r,o}return Object.defineProperty(t.prototype,"value",{get:function(){return this.getValue()},enumerable:!1,configurable:!0}),t.prototype._subscribe=function(r){var o=e.prototype._subscribe.call(this,r);return!o.closed&&r.next(this._value),o},t.prototype.getValue=function(){var r=this,o=r.hasError,n=r.thrownError,i=r._value;if(o)throw n;return this._throwIfClosed(),i},t.prototype.next=function(r){e.prototype.next.call(this,this._value=r)},t}(g);var At={now:function(){return(At.delegate||Date).now()},delegate:void 0};var Ct=function(e){oe(t,e);function t(r,o,n){r===void 0&&(r=1/0),o===void 0&&(o=1/0),n===void 0&&(n=At);var i=e.call(this)||this;return i._bufferSize=r,i._windowTime=o,i._timestampProvider=n,i._buffer=[],i._infiniteTimeWindow=!0,i._infiniteTimeWindow=o===1/0,i._bufferSize=Math.max(1,r),i._windowTime=Math.max(1,o),i}return t.prototype.next=function(r){var o=this,n=o.isStopped,i=o._buffer,a=o._infiniteTimeWindow,s=o._timestampProvider,p=o._windowTime;n||(i.push(r),!a&&i.push(s.now()+p)),this._trimBuffer(),e.prototype.next.call(this,r)},t.prototype._subscribe=function(r){this._throwIfClosed(),this._trimBuffer();for(var o=this._innerSubscribe(r),n=this,i=n._infiniteTimeWindow,a=n._buffer,s=a.slice(),p=0;p0?e.prototype.schedule.call(this,r,o):(this.delay=o,this.state=r,this.scheduler.flush(this),this)},t.prototype.execute=function(r,o){return o>0||this.closed?e.prototype.execute.call(this,r,o):this._execute(r,o)},t.prototype.requestAsyncId=function(r,o,n){return n===void 0&&(n=0),n!=null&&n>0||n==null&&this.delay>0?e.prototype.requestAsyncId.call(this,r,o,n):(r.flush(this),0)},t}(gt);var Lo=function(e){oe(t,e);function t(){return e!==null&&e.apply(this,arguments)||this}return t}(yt);var kr=new Lo(Oo);var Mo=function(e){oe(t,e);function t(r,o){var n=e.call(this,r,o)||this;return n.scheduler=r,n.work=o,n}return t.prototype.requestAsyncId=function(r,o,n){return n===void 0&&(n=0),n!==null&&n>0?e.prototype.requestAsyncId.call(this,r,o,n):(r.actions.push(this),r._scheduled||(r._scheduled=vt.requestAnimationFrame(function(){return r.flush(void 0)})))},t.prototype.recycleAsyncId=function(r,o,n){var i;if(n===void 0&&(n=0),n!=null?n>0:this.delay>0)return e.prototype.recycleAsyncId.call(this,r,o,n);var a=r.actions;o!=null&&((i=a[a.length-1])===null||i===void 0?void 0:i.id)!==o&&(vt.cancelAnimationFrame(o),r._scheduled=void 0)},t}(gt);var _o=function(e){oe(t,e);function t(){return e!==null&&e.apply(this,arguments)||this}return t.prototype.flush=function(r){this._active=!0;var o=this._scheduled;this._scheduled=void 0;var n=this.actions,i;r=r||n.shift();do if(i=r.execute(r.state,r.delay))break;while((r=n[0])&&r.id===o&&n.shift());if(this._active=!1,i){for(;(r=n[0])&&r.id===o&&n.shift();)r.unsubscribe();throw i}},t}(yt);var me=new _o(Mo);var S=new j(function(e){return e.complete()});function Yt(e){return e&&H(e.schedule)}function Hr(e){return e[e.length-1]}function Xe(e){return H(Hr(e))?e.pop():void 0}function ke(e){return Yt(Hr(e))?e.pop():void 0}function Bt(e,t){return typeof Hr(e)=="number"?e.pop():t}var xt=function(e){return e&&typeof e.length=="number"&&typeof e!="function"};function Gt(e){return H(e==null?void 0:e.then)}function Jt(e){return H(e[bt])}function Xt(e){return Symbol.asyncIterator&&H(e==null?void 0:e[Symbol.asyncIterator])}function Zt(e){return new TypeError("You provided "+(e!==null&&typeof e=="object"?"an invalid object":"'"+e+"'")+" where a stream was expected. You can provide an Observable, Promise, ReadableStream, Array, AsyncIterable, or Iterable.")}function Zi(){return typeof Symbol!="function"||!Symbol.iterator?"@@iterator":Symbol.iterator}var er=Zi();function tr(e){return H(e==null?void 0:e[er])}function rr(e){return fo(this,arguments,function(){var r,o,n,i;return Nt(this,function(a){switch(a.label){case 0:r=e.getReader(),a.label=1;case 1:a.trys.push([1,,9,10]),a.label=2;case 2:return[4,nt(r.read())];case 3:return o=a.sent(),n=o.value,i=o.done,i?[4,nt(void 0)]:[3,5];case 4:return[2,a.sent()];case 5:return[4,nt(n)];case 6:return[4,a.sent()];case 7:return a.sent(),[3,2];case 8:return[3,10];case 9:return r.releaseLock(),[7];case 10:return[2]}})})}function or(e){return H(e==null?void 0:e.getReader)}function U(e){if(e instanceof j)return e;if(e!=null){if(Jt(e))return ea(e);if(xt(e))return ta(e);if(Gt(e))return ra(e);if(Xt(e))return Ao(e);if(tr(e))return oa(e);if(or(e))return na(e)}throw Zt(e)}function ea(e){return new j(function(t){var r=e[bt]();if(H(r.subscribe))return r.subscribe(t);throw new TypeError("Provided object does not correctly implement Symbol.observable")})}function ta(e){return new j(function(t){for(var r=0;r=2;return function(o){return o.pipe(e?b(function(n,i){return e(n,i,o)}):le,Te(1),r?De(t):Qo(function(){return new ir}))}}function jr(e){return e<=0?function(){return S}:E(function(t,r){var o=[];t.subscribe(T(r,function(n){o.push(n),e=2,!0))}function pe(e){e===void 0&&(e={});var t=e.connector,r=t===void 0?function(){return new g}:t,o=e.resetOnError,n=o===void 0?!0:o,i=e.resetOnComplete,a=i===void 0?!0:i,s=e.resetOnRefCountZero,p=s===void 0?!0:s;return function(c){var l,f,u,d=0,y=!1,L=!1,X=function(){f==null||f.unsubscribe(),f=void 0},te=function(){X(),l=u=void 0,y=L=!1},J=function(){var k=l;te(),k==null||k.unsubscribe()};return E(function(k,ft){d++,!L&&!y&&X();var qe=u=u!=null?u:r();ft.add(function(){d--,d===0&&!L&&!y&&(f=Ur(J,p))}),qe.subscribe(ft),!l&&d>0&&(l=new at({next:function(Fe){return qe.next(Fe)},error:function(Fe){L=!0,X(),f=Ur(te,n,Fe),qe.error(Fe)},complete:function(){y=!0,X(),f=Ur(te,a),qe.complete()}}),U(k).subscribe(l))})(c)}}function Ur(e,t){for(var r=[],o=2;oe.next(document)),e}function P(e,t=document){return Array.from(t.querySelectorAll(e))}function R(e,t=document){let r=fe(e,t);if(typeof r=="undefined")throw new ReferenceError(`Missing element: expected "${e}" to be present`);return r}function fe(e,t=document){return t.querySelector(e)||void 0}function Ie(){var e,t,r,o;return(o=(r=(t=(e=document.activeElement)==null?void 0:e.shadowRoot)==null?void 0:t.activeElement)!=null?r:document.activeElement)!=null?o:void 0}var wa=O(h(document.body,"focusin"),h(document.body,"focusout")).pipe(_e(1),Q(void 0),m(()=>Ie()||document.body),G(1));function et(e){return wa.pipe(m(t=>e.contains(t)),K())}function $t(e,t){return C(()=>O(h(e,"mouseenter").pipe(m(()=>!0)),h(e,"mouseleave").pipe(m(()=>!1))).pipe(t?Ht(r=>Le(+!r*t)):le,Q(e.matches(":hover"))))}function Jo(e,t){if(typeof t=="string"||typeof t=="number")e.innerHTML+=t.toString();else if(t instanceof Node)e.appendChild(t);else if(Array.isArray(t))for(let r of t)Jo(e,r)}function x(e,t,...r){let o=document.createElement(e);if(t)for(let n of Object.keys(t))typeof t[n]!="undefined"&&(typeof t[n]!="boolean"?o.setAttribute(n,t[n]):o.setAttribute(n,""));for(let n of r)Jo(o,n);return o}function sr(e){if(e>999){let t=+((e-950)%1e3>99);return`${((e+1e-6)/1e3).toFixed(t)}k`}else return e.toString()}function Tt(e){let t=x("script",{src:e});return C(()=>(document.head.appendChild(t),O(h(t,"load"),h(t,"error").pipe(v(()=>$r(()=>new ReferenceError(`Invalid script: ${e}`))))).pipe(m(()=>{}),_(()=>document.head.removeChild(t)),Te(1))))}var Xo=new g,Ta=C(()=>typeof ResizeObserver=="undefined"?Tt("https://unpkg.com/resize-observer-polyfill"):I(void 0)).pipe(m(()=>new ResizeObserver(e=>e.forEach(t=>Xo.next(t)))),v(e=>O(Ye,I(e)).pipe(_(()=>e.disconnect()))),G(1));function ce(e){return{width:e.offsetWidth,height:e.offsetHeight}}function ge(e){let t=e;for(;t.clientWidth===0&&t.parentElement;)t=t.parentElement;return Ta.pipe(w(r=>r.observe(t)),v(r=>Xo.pipe(b(o=>o.target===t),_(()=>r.unobserve(t)))),m(()=>ce(e)),Q(ce(e)))}function St(e){return{width:e.scrollWidth,height:e.scrollHeight}}function cr(e){let t=e.parentElement;for(;t&&(e.scrollWidth<=t.scrollWidth&&e.scrollHeight<=t.scrollHeight);)t=(e=t).parentElement;return t?e:void 0}function Zo(e){let t=[],r=e.parentElement;for(;r;)(e.clientWidth>r.clientWidth||e.clientHeight>r.clientHeight)&&t.push(r),r=(e=r).parentElement;return t.length===0&&t.push(document.documentElement),t}function Ve(e){return{x:e.offsetLeft,y:e.offsetTop}}function en(e){let t=e.getBoundingClientRect();return{x:t.x+window.scrollX,y:t.y+window.scrollY}}function tn(e){return O(h(window,"load"),h(window,"resize")).pipe(Me(0,me),m(()=>Ve(e)),Q(Ve(e)))}function pr(e){return{x:e.scrollLeft,y:e.scrollTop}}function Ne(e){return O(h(e,"scroll"),h(window,"scroll"),h(window,"resize")).pipe(Me(0,me),m(()=>pr(e)),Q(pr(e)))}var rn=new g,Sa=C(()=>I(new IntersectionObserver(e=>{for(let t of e)rn.next(t)},{threshold:0}))).pipe(v(e=>O(Ye,I(e)).pipe(_(()=>e.disconnect()))),G(1));function tt(e){return Sa.pipe(w(t=>t.observe(e)),v(t=>rn.pipe(b(({target:r})=>r===e),_(()=>t.unobserve(e)),m(({isIntersecting:r})=>r))))}function on(e,t=16){return Ne(e).pipe(m(({y:r})=>{let o=ce(e),n=St(e);return r>=n.height-o.height-t}),K())}var lr={drawer:R("[data-md-toggle=drawer]"),search:R("[data-md-toggle=search]")};function nn(e){return lr[e].checked}function Je(e,t){lr[e].checked!==t&&lr[e].click()}function ze(e){let t=lr[e];return h(t,"change").pipe(m(()=>t.checked),Q(t.checked))}function Oa(e,t){switch(e.constructor){case HTMLInputElement:return e.type==="radio"?/^Arrow/.test(t):!0;case HTMLSelectElement:case HTMLTextAreaElement:return!0;default:return e.isContentEditable}}function La(){return O(h(window,"compositionstart").pipe(m(()=>!0)),h(window,"compositionend").pipe(m(()=>!1))).pipe(Q(!1))}function an(){let e=h(window,"keydown").pipe(b(t=>!(t.metaKey||t.ctrlKey)),m(t=>({mode:nn("search")?"search":"global",type:t.key,claim(){t.preventDefault(),t.stopPropagation()}})),b(({mode:t,type:r})=>{if(t==="global"){let o=Ie();if(typeof o!="undefined")return!Oa(o,r)}return!0}),pe());return La().pipe(v(t=>t?S:e))}function ye(){return new URL(location.href)}function lt(e,t=!1){if(B("navigation.instant")&&!t){let r=x("a",{href:e.href});document.body.appendChild(r),r.click(),r.remove()}else location.href=e.href}function sn(){return new g}function cn(){return location.hash.slice(1)}function pn(e){let t=x("a",{href:e});t.addEventListener("click",r=>r.stopPropagation()),t.click()}function Ma(e){return O(h(window,"hashchange"),e).pipe(m(cn),Q(cn()),b(t=>t.length>0),G(1))}function ln(e){return Ma(e).pipe(m(t=>fe(`[id="${t}"]`)),b(t=>typeof t!="undefined"))}function Pt(e){let t=matchMedia(e);return ar(r=>t.addListener(()=>r(t.matches))).pipe(Q(t.matches))}function mn(){let e=matchMedia("print");return O(h(window,"beforeprint").pipe(m(()=>!0)),h(window,"afterprint").pipe(m(()=>!1))).pipe(Q(e.matches))}function Nr(e,t){return e.pipe(v(r=>r?t():S))}function zr(e,t){return new j(r=>{let o=new XMLHttpRequest;return o.open("GET",`${e}`),o.responseType="blob",o.addEventListener("load",()=>{o.status>=200&&o.status<300?(r.next(o.response),r.complete()):r.error(new Error(o.statusText))}),o.addEventListener("error",()=>{r.error(new Error("Network error"))}),o.addEventListener("abort",()=>{r.complete()}),typeof(t==null?void 0:t.progress$)!="undefined"&&(o.addEventListener("progress",n=>{var i;if(n.lengthComputable)t.progress$.next(n.loaded/n.total*100);else{let a=(i=o.getResponseHeader("Content-Length"))!=null?i:0;t.progress$.next(n.loaded/+a*100)}}),t.progress$.next(5)),o.send(),()=>o.abort()})}function je(e,t){return zr(e,t).pipe(v(r=>r.text()),m(r=>JSON.parse(r)),G(1))}function fn(e,t){let r=new DOMParser;return zr(e,t).pipe(v(o=>o.text()),m(o=>r.parseFromString(o,"text/html")),G(1))}function un(e,t){let r=new DOMParser;return zr(e,t).pipe(v(o=>o.text()),m(o=>r.parseFromString(o,"text/xml")),G(1))}function dn(){return{x:Math.max(0,scrollX),y:Math.max(0,scrollY)}}function hn(){return O(h(window,"scroll",{passive:!0}),h(window,"resize",{passive:!0})).pipe(m(dn),Q(dn()))}function bn(){return{width:innerWidth,height:innerHeight}}function vn(){return h(window,"resize",{passive:!0}).pipe(m(bn),Q(bn()))}function gn(){return z([hn(),vn()]).pipe(m(([e,t])=>({offset:e,size:t})),G(1))}function mr(e,{viewport$:t,header$:r}){let o=t.pipe(ee("size")),n=z([o,r]).pipe(m(()=>Ve(e)));return z([r,t,n]).pipe(m(([{height:i},{offset:a,size:s},{x:p,y:c}])=>({offset:{x:a.x-p,y:a.y-c+i},size:s})))}function _a(e){return h(e,"message",t=>t.data)}function Aa(e){let t=new g;return t.subscribe(r=>e.postMessage(r)),t}function yn(e,t=new Worker(e)){let r=_a(t),o=Aa(t),n=new g;n.subscribe(o);let i=o.pipe(Z(),ie(!0));return n.pipe(Z(),Re(r.pipe(W(i))),pe())}var Ca=R("#__config"),Ot=JSON.parse(Ca.textContent);Ot.base=`${new URL(Ot.base,ye())}`;function xe(){return Ot}function B(e){return Ot.features.includes(e)}function Ee(e,t){return typeof t!="undefined"?Ot.translations[e].replace("#",t.toString()):Ot.translations[e]}function Se(e,t=document){return R(`[data-md-component=${e}]`,t)}function ae(e,t=document){return P(`[data-md-component=${e}]`,t)}function ka(e){let t=R(".md-typeset > :first-child",e);return h(t,"click",{once:!0}).pipe(m(()=>R(".md-typeset",e)),m(r=>({hash:__md_hash(r.innerHTML)})))}function xn(e){if(!B("announce.dismiss")||!e.childElementCount)return S;if(!e.hidden){let t=R(".md-typeset",e);__md_hash(t.innerHTML)===__md_get("__announce")&&(e.hidden=!0)}return C(()=>{let t=new g;return t.subscribe(({hash:r})=>{e.hidden=!0,__md_set("__announce",r)}),ka(e).pipe(w(r=>t.next(r)),_(()=>t.complete()),m(r=>$({ref:e},r)))})}function Ha(e,{target$:t}){return t.pipe(m(r=>({hidden:r!==e})))}function En(e,t){let r=new g;return r.subscribe(({hidden:o})=>{e.hidden=o}),Ha(e,t).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))}function Rt(e,t){return t==="inline"?x("div",{class:"md-tooltip md-tooltip--inline",id:e,role:"tooltip"},x("div",{class:"md-tooltip__inner md-typeset"})):x("div",{class:"md-tooltip",id:e,role:"tooltip"},x("div",{class:"md-tooltip__inner md-typeset"}))}function wn(...e){return x("div",{class:"md-tooltip2",role:"tooltip"},x("div",{class:"md-tooltip2__inner md-typeset"},e))}function Tn(e,t){if(t=t?`${t}_annotation_${e}`:void 0,t){let r=t?`#${t}`:void 0;return x("aside",{class:"md-annotation",tabIndex:0},Rt(t),x("a",{href:r,class:"md-annotation__index",tabIndex:-1},x("span",{"data-md-annotation-id":e})))}else return x("aside",{class:"md-annotation",tabIndex:0},Rt(t),x("span",{class:"md-annotation__index",tabIndex:-1},x("span",{"data-md-annotation-id":e})))}function Sn(e){return x("button",{class:"md-clipboard md-icon",title:Ee("clipboard.copy"),"data-clipboard-target":`#${e} > code`})}var Ln=Mt(qr());function Qr(e,t){let r=t&2,o=t&1,n=Object.keys(e.terms).filter(p=>!e.terms[p]).reduce((p,c)=>[...p,x("del",null,(0,Ln.default)(c))," "],[]).slice(0,-1),i=xe(),a=new URL(e.location,i.base);B("search.highlight")&&a.searchParams.set("h",Object.entries(e.terms).filter(([,p])=>p).reduce((p,[c])=>`${p} ${c}`.trim(),""));let{tags:s}=xe();return x("a",{href:`${a}`,class:"md-search-result__link",tabIndex:-1},x("article",{class:"md-search-result__article md-typeset","data-md-score":e.score.toFixed(2)},r>0&&x("div",{class:"md-search-result__icon md-icon"}),r>0&&x("h1",null,e.title),r<=0&&x("h2",null,e.title),o>0&&e.text.length>0&&e.text,e.tags&&x("nav",{class:"md-tags"},e.tags.map(p=>{let c=s?p in s?`md-tag-icon md-tag--${s[p]}`:"md-tag-icon":"";return x("span",{class:`md-tag ${c}`},p)})),o>0&&n.length>0&&x("p",{class:"md-search-result__terms"},Ee("search.result.term.missing"),": ",...n)))}function Mn(e){let t=e[0].score,r=[...e],o=xe(),n=r.findIndex(l=>!`${new URL(l.location,o.base)}`.includes("#")),[i]=r.splice(n,1),a=r.findIndex(l=>l.scoreQr(l,1)),...p.length?[x("details",{class:"md-search-result__more"},x("summary",{tabIndex:-1},x("div",null,p.length>0&&p.length===1?Ee("search.result.more.one"):Ee("search.result.more.other",p.length))),...p.map(l=>Qr(l,1)))]:[]];return x("li",{class:"md-search-result__item"},c)}function _n(e){return x("ul",{class:"md-source__facts"},Object.entries(e).map(([t,r])=>x("li",{class:`md-source__fact md-source__fact--${t}`},typeof r=="number"?sr(r):r)))}function Kr(e){let t=`tabbed-control tabbed-control--${e}`;return x("div",{class:t,hidden:!0},x("button",{class:"tabbed-button",tabIndex:-1,"aria-hidden":"true"}))}function An(e){return x("div",{class:"md-typeset__scrollwrap"},x("div",{class:"md-typeset__table"},e))}function Ra(e){var o;let t=xe(),r=new URL(`../${e.version}/`,t.base);return x("li",{class:"md-version__item"},x("a",{href:`${r}`,class:"md-version__link"},e.title,((o=t.version)==null?void 0:o.alias)&&e.aliases.length>0&&x("span",{class:"md-version__alias"},e.aliases[0])))}function Cn(e,t){var o;let r=xe();return e=e.filter(n=>{var i;return!((i=n.properties)!=null&&i.hidden)}),x("div",{class:"md-version"},x("button",{class:"md-version__current","aria-label":Ee("select.version")},t.title,((o=r.version)==null?void 0:o.alias)&&t.aliases.length>0&&x("span",{class:"md-version__alias"},t.aliases[0])),x("ul",{class:"md-version__list"},e.map(Ra)))}var Ia=0;function ja(e){let t=z([et(e),$t(e)]).pipe(m(([o,n])=>o||n),K()),r=C(()=>Zo(e)).pipe(ne(Ne),pt(1),He(t),m(()=>en(e)));return t.pipe(Ae(o=>o),v(()=>z([t,r])),m(([o,n])=>({active:o,offset:n})),pe())}function Fa(e,t){let{content$:r,viewport$:o}=t,n=`__tooltip2_${Ia++}`;return C(()=>{let i=new g,a=new _r(!1);i.pipe(Z(),ie(!1)).subscribe(a);let s=a.pipe(Ht(c=>Le(+!c*250,kr)),K(),v(c=>c?r:S),w(c=>c.id=n),pe());z([i.pipe(m(({active:c})=>c)),s.pipe(v(c=>$t(c,250)),Q(!1))]).pipe(m(c=>c.some(l=>l))).subscribe(a);let p=a.pipe(b(c=>c),re(s,o),m(([c,l,{size:f}])=>{let u=e.getBoundingClientRect(),d=u.width/2;if(l.role==="tooltip")return{x:d,y:8+u.height};if(u.y>=f.height/2){let{height:y}=ce(l);return{x:d,y:-16-y}}else return{x:d,y:16+u.height}}));return z([s,i,p]).subscribe(([c,{offset:l},f])=>{c.style.setProperty("--md-tooltip-host-x",`${l.x}px`),c.style.setProperty("--md-tooltip-host-y",`${l.y}px`),c.style.setProperty("--md-tooltip-x",`${f.x}px`),c.style.setProperty("--md-tooltip-y",`${f.y}px`),c.classList.toggle("md-tooltip2--top",f.y<0),c.classList.toggle("md-tooltip2--bottom",f.y>=0)}),a.pipe(b(c=>c),re(s,(c,l)=>l),b(c=>c.role==="tooltip")).subscribe(c=>{let l=ce(R(":scope > *",c));c.style.setProperty("--md-tooltip-width",`${l.width}px`),c.style.setProperty("--md-tooltip-tail","0px")}),a.pipe(K(),ve(me),re(s)).subscribe(([c,l])=>{l.classList.toggle("md-tooltip2--active",c)}),z([a.pipe(b(c=>c)),s]).subscribe(([c,l])=>{l.role==="dialog"?(e.setAttribute("aria-controls",n),e.setAttribute("aria-haspopup","dialog")):e.setAttribute("aria-describedby",n)}),a.pipe(b(c=>!c)).subscribe(()=>{e.removeAttribute("aria-controls"),e.removeAttribute("aria-describedby"),e.removeAttribute("aria-haspopup")}),ja(e).pipe(w(c=>i.next(c)),_(()=>i.complete()),m(c=>$({ref:e},c)))})}function mt(e,{viewport$:t},r=document.body){return Fa(e,{content$:new j(o=>{let n=e.title,i=wn(n);return o.next(i),e.removeAttribute("title"),r.append(i),()=>{i.remove(),e.setAttribute("title",n)}}),viewport$:t})}function Ua(e,t){let r=C(()=>z([tn(e),Ne(t)])).pipe(m(([{x:o,y:n},i])=>{let{width:a,height:s}=ce(e);return{x:o-i.x+a/2,y:n-i.y+s/2}}));return et(e).pipe(v(o=>r.pipe(m(n=>({active:o,offset:n})),Te(+!o||1/0))))}function kn(e,t,{target$:r}){let[o,n]=Array.from(e.children);return C(()=>{let i=new g,a=i.pipe(Z(),ie(!0));return i.subscribe({next({offset:s}){e.style.setProperty("--md-tooltip-x",`${s.x}px`),e.style.setProperty("--md-tooltip-y",`${s.y}px`)},complete(){e.style.removeProperty("--md-tooltip-x"),e.style.removeProperty("--md-tooltip-y")}}),tt(e).pipe(W(a)).subscribe(s=>{e.toggleAttribute("data-md-visible",s)}),O(i.pipe(b(({active:s})=>s)),i.pipe(_e(250),b(({active:s})=>!s))).subscribe({next({active:s}){s?e.prepend(o):o.remove()},complete(){e.prepend(o)}}),i.pipe(Me(16,me)).subscribe(({active:s})=>{o.classList.toggle("md-tooltip--active",s)}),i.pipe(pt(125,me),b(()=>!!e.offsetParent),m(()=>e.offsetParent.getBoundingClientRect()),m(({x:s})=>s)).subscribe({next(s){s?e.style.setProperty("--md-tooltip-0",`${-s}px`):e.style.removeProperty("--md-tooltip-0")},complete(){e.style.removeProperty("--md-tooltip-0")}}),h(n,"click").pipe(W(a),b(s=>!(s.metaKey||s.ctrlKey))).subscribe(s=>{s.stopPropagation(),s.preventDefault()}),h(n,"mousedown").pipe(W(a),re(i)).subscribe(([s,{active:p}])=>{var c;if(s.button!==0||s.metaKey||s.ctrlKey)s.preventDefault();else if(p){s.preventDefault();let l=e.parentElement.closest(".md-annotation");l instanceof HTMLElement?l.focus():(c=Ie())==null||c.blur()}}),r.pipe(W(a),b(s=>s===o),Ge(125)).subscribe(()=>e.focus()),Ua(e,t).pipe(w(s=>i.next(s)),_(()=>i.complete()),m(s=>$({ref:e},s)))})}function Wa(e){return e.tagName==="CODE"?P(".c, .c1, .cm",e):[e]}function Da(e){let t=[];for(let r of Wa(e)){let o=[],n=document.createNodeIterator(r,NodeFilter.SHOW_TEXT);for(let i=n.nextNode();i;i=n.nextNode())o.push(i);for(let i of o){let a;for(;a=/(\(\d+\))(!)?/.exec(i.textContent);){let[,s,p]=a;if(typeof p=="undefined"){let c=i.splitText(a.index);i=c.splitText(s.length),t.push(c)}else{i.textContent=s,t.push(i);break}}}}return t}function Hn(e,t){t.append(...Array.from(e.childNodes))}function fr(e,t,{target$:r,print$:o}){let n=t.closest("[id]"),i=n==null?void 0:n.id,a=new Map;for(let s of Da(t)){let[,p]=s.textContent.match(/\((\d+)\)/);fe(`:scope > li:nth-child(${p})`,e)&&(a.set(p,Tn(p,i)),s.replaceWith(a.get(p)))}return a.size===0?S:C(()=>{let s=new g,p=s.pipe(Z(),ie(!0)),c=[];for(let[l,f]of a)c.push([R(".md-typeset",f),R(`:scope > li:nth-child(${l})`,e)]);return o.pipe(W(p)).subscribe(l=>{e.hidden=!l,e.classList.toggle("md-annotation-list",l);for(let[f,u]of c)l?Hn(f,u):Hn(u,f)}),O(...[...a].map(([,l])=>kn(l,t,{target$:r}))).pipe(_(()=>s.complete()),pe())})}function $n(e){if(e.nextElementSibling){let t=e.nextElementSibling;if(t.tagName==="OL")return t;if(t.tagName==="P"&&!t.children.length)return $n(t)}}function Pn(e,t){return C(()=>{let r=$n(e);return typeof r!="undefined"?fr(r,e,t):S})}var Rn=Mt(Br());var Va=0;function In(e){if(e.nextElementSibling){let t=e.nextElementSibling;if(t.tagName==="OL")return t;if(t.tagName==="P"&&!t.children.length)return In(t)}}function Na(e){return ge(e).pipe(m(({width:t})=>({scrollable:St(e).width>t})),ee("scrollable"))}function jn(e,t){let{matches:r}=matchMedia("(hover)"),o=C(()=>{let n=new g,i=n.pipe(jr(1));n.subscribe(({scrollable:c})=>{c&&r?e.setAttribute("tabindex","0"):e.removeAttribute("tabindex")});let a=[];if(Rn.default.isSupported()&&(e.closest(".copy")||B("content.code.copy")&&!e.closest(".no-copy"))){let c=e.closest("pre");c.id=`__code_${Va++}`;let l=Sn(c.id);c.insertBefore(l,e),B("content.tooltips")&&a.push(mt(l,{viewport$}))}let s=e.closest(".highlight");if(s instanceof HTMLElement){let c=In(s);if(typeof c!="undefined"&&(s.classList.contains("annotate")||B("content.code.annotate"))){let l=fr(c,e,t);a.push(ge(s).pipe(W(i),m(({width:f,height:u})=>f&&u),K(),v(f=>f?l:S)))}}return P(":scope > span[id]",e).length&&e.classList.add("md-code__content"),Na(e).pipe(w(c=>n.next(c)),_(()=>n.complete()),m(c=>$({ref:e},c)),Re(...a))});return B("content.lazy")?tt(e).pipe(b(n=>n),Te(1),v(()=>o)):o}function za(e,{target$:t,print$:r}){let o=!0;return O(t.pipe(m(n=>n.closest("details:not([open])")),b(n=>e===n),m(()=>({action:"open",reveal:!0}))),r.pipe(b(n=>n||!o),w(()=>o=e.open),m(n=>({action:n?"open":"close"}))))}function Fn(e,t){return C(()=>{let r=new g;return r.subscribe(({action:o,reveal:n})=>{e.toggleAttribute("open",o==="open"),n&&e.scrollIntoView()}),za(e,t).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))})}var Un=".node circle,.node ellipse,.node path,.node polygon,.node rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}marker{fill:var(--md-mermaid-edge-color)!important}.edgeLabel .label rect{fill:#0000}.flowchartTitleText{fill:var(--md-mermaid-label-fg-color)}.label{color:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.label foreignObject{line-height:normal;overflow:visible}.label div .edgeLabel{color:var(--md-mermaid-label-fg-color)}.edgeLabel,.edgeLabel p,.label div .edgeLabel{background-color:var(--md-mermaid-label-bg-color)}.edgeLabel,.edgeLabel p{fill:var(--md-mermaid-label-bg-color);color:var(--md-mermaid-edge-color)}.edgePath .path,.flowchart-link{stroke:var(--md-mermaid-edge-color);stroke-width:.05rem}.edgePath .arrowheadPath{fill:var(--md-mermaid-edge-color);stroke:none}.cluster rect{fill:var(--md-default-fg-color--lightest);stroke:var(--md-default-fg-color--lighter)}.cluster span{color:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}g #flowchart-circleEnd,g #flowchart-circleStart,g #flowchart-crossEnd,g #flowchart-crossStart,g #flowchart-pointEnd,g #flowchart-pointStart{stroke:none}.classDiagramTitleText{fill:var(--md-mermaid-label-fg-color)}g.classGroup line,g.classGroup rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}g.classGroup text{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.classLabel .box{fill:var(--md-mermaid-label-bg-color);background-color:var(--md-mermaid-label-bg-color);opacity:1}.classLabel .label{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.node .divider{stroke:var(--md-mermaid-node-fg-color)}.relation{stroke:var(--md-mermaid-edge-color)}.cardinality{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.cardinality text{fill:inherit!important}defs #classDiagram-compositionEnd,defs #classDiagram-compositionStart,defs #classDiagram-dependencyEnd,defs #classDiagram-dependencyStart,defs #classDiagram-extensionEnd,defs #classDiagram-extensionStart{fill:var(--md-mermaid-edge-color)!important;stroke:var(--md-mermaid-edge-color)!important}defs #classDiagram-aggregationEnd,defs #classDiagram-aggregationStart{fill:var(--md-mermaid-label-bg-color)!important;stroke:var(--md-mermaid-edge-color)!important}.statediagramTitleText{fill:var(--md-mermaid-label-fg-color)}g.stateGroup rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}g.stateGroup .state-title{fill:var(--md-mermaid-label-fg-color)!important;font-family:var(--md-mermaid-font-family)}g.stateGroup .composit{fill:var(--md-mermaid-label-bg-color)}.nodeLabel,.nodeLabel p{color:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}a .nodeLabel{text-decoration:underline}.node circle.state-end,.node circle.state-start,.start-state{fill:var(--md-mermaid-edge-color);stroke:none}.end-state-inner,.end-state-outer{fill:var(--md-mermaid-edge-color)}.end-state-inner,.node circle.state-end{stroke:var(--md-mermaid-label-bg-color)}.transition{stroke:var(--md-mermaid-edge-color)}[id^=state-fork] rect,[id^=state-join] rect{fill:var(--md-mermaid-edge-color)!important;stroke:none!important}.statediagram-cluster.statediagram-cluster .inner{fill:var(--md-default-bg-color)}.statediagram-cluster rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}.statediagram-state rect.divider{fill:var(--md-default-fg-color--lightest);stroke:var(--md-default-fg-color--lighter)}defs #statediagram-barbEnd{stroke:var(--md-mermaid-edge-color)}.entityTitleText{fill:var(--md-mermaid-label-fg-color)}.attributeBoxEven,.attributeBoxOdd{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}.entityBox{fill:var(--md-mermaid-label-bg-color);stroke:var(--md-mermaid-node-fg-color)}.entityLabel{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.relationshipLabelBox{fill:var(--md-mermaid-label-bg-color);fill-opacity:1;background-color:var(--md-mermaid-label-bg-color);opacity:1}.relationshipLabel{fill:var(--md-mermaid-label-fg-color)}.relationshipLine{stroke:var(--md-mermaid-edge-color)}defs #ONE_OR_MORE_END *,defs #ONE_OR_MORE_START *,defs #ONLY_ONE_END *,defs #ONLY_ONE_START *,defs #ZERO_OR_MORE_END *,defs #ZERO_OR_MORE_START *,defs #ZERO_OR_ONE_END *,defs #ZERO_OR_ONE_START *{stroke:var(--md-mermaid-edge-color)!important}defs #ZERO_OR_MORE_END circle,defs #ZERO_OR_MORE_START circle{fill:var(--md-mermaid-label-bg-color)}text:not([class]):last-child{fill:var(--md-mermaid-label-fg-color)}.actor{fill:var(--md-mermaid-sequence-actor-bg-color);stroke:var(--md-mermaid-sequence-actor-border-color)}text.actor>tspan{fill:var(--md-mermaid-sequence-actor-fg-color);font-family:var(--md-mermaid-font-family)}line{stroke:var(--md-mermaid-sequence-actor-line-color)}.actor-man circle,.actor-man line{fill:var(--md-mermaid-sequence-actorman-bg-color);stroke:var(--md-mermaid-sequence-actorman-line-color)}.messageLine0,.messageLine1{stroke:var(--md-mermaid-sequence-message-line-color)}.note{fill:var(--md-mermaid-sequence-note-bg-color);stroke:var(--md-mermaid-sequence-note-border-color)}.loopText,.loopText>tspan,.messageText,.noteText>tspan{stroke:none;font-family:var(--md-mermaid-font-family)!important}.messageText{fill:var(--md-mermaid-sequence-message-fg-color)}.loopText,.loopText>tspan{fill:var(--md-mermaid-sequence-loop-fg-color)}.noteText>tspan{fill:var(--md-mermaid-sequence-note-fg-color)}#arrowhead path{fill:var(--md-mermaid-sequence-message-line-color);stroke:none}.loopLine{fill:var(--md-mermaid-sequence-loop-bg-color);stroke:var(--md-mermaid-sequence-loop-border-color)}.labelBox{fill:var(--md-mermaid-sequence-label-bg-color);stroke:none}.labelText,.labelText>span{fill:var(--md-mermaid-sequence-label-fg-color);font-family:var(--md-mermaid-font-family)}.sequenceNumber{fill:var(--md-mermaid-sequence-number-fg-color)}rect.rect{fill:var(--md-mermaid-sequence-box-bg-color);stroke:none}rect.rect+text.text{fill:var(--md-mermaid-sequence-box-fg-color)}defs #sequencenumber{fill:var(--md-mermaid-sequence-number-bg-color)!important}";var Gr,Qa=0;function Ka(){return typeof mermaid=="undefined"||mermaid instanceof Element?Tt("https://unpkg.com/mermaid@11/dist/mermaid.min.js"):I(void 0)}function Wn(e){return e.classList.remove("mermaid"),Gr||(Gr=Ka().pipe(w(()=>mermaid.initialize({startOnLoad:!1,themeCSS:Un,sequence:{actorFontSize:"16px",messageFontSize:"16px",noteFontSize:"16px"}})),m(()=>{}),G(1))),Gr.subscribe(()=>co(this,null,function*(){e.classList.add("mermaid");let t=`__mermaid_${Qa++}`,r=x("div",{class:"mermaid"}),o=e.textContent,{svg:n,fn:i}=yield mermaid.render(t,o),a=r.attachShadow({mode:"closed"});a.innerHTML=n,e.replaceWith(r),i==null||i(a)})),Gr.pipe(m(()=>({ref:e})))}var Dn=x("table");function Vn(e){return e.replaceWith(Dn),Dn.replaceWith(An(e)),I({ref:e})}function Ya(e){let t=e.find(r=>r.checked)||e[0];return O(...e.map(r=>h(r,"change").pipe(m(()=>R(`label[for="${r.id}"]`))))).pipe(Q(R(`label[for="${t.id}"]`)),m(r=>({active:r})))}function Nn(e,{viewport$:t,target$:r}){let o=R(".tabbed-labels",e),n=P(":scope > input",e),i=Kr("prev");e.append(i);let a=Kr("next");return e.append(a),C(()=>{let s=new g,p=s.pipe(Z(),ie(!0));z([s,ge(e),tt(e)]).pipe(W(p),Me(1,me)).subscribe({next([{active:c},l]){let f=Ve(c),{width:u}=ce(c);e.style.setProperty("--md-indicator-x",`${f.x}px`),e.style.setProperty("--md-indicator-width",`${u}px`);let d=pr(o);(f.xd.x+l.width)&&o.scrollTo({left:Math.max(0,f.x-16),behavior:"smooth"})},complete(){e.style.removeProperty("--md-indicator-x"),e.style.removeProperty("--md-indicator-width")}}),z([Ne(o),ge(o)]).pipe(W(p)).subscribe(([c,l])=>{let f=St(o);i.hidden=c.x<16,a.hidden=c.x>f.width-l.width-16}),O(h(i,"click").pipe(m(()=>-1)),h(a,"click").pipe(m(()=>1))).pipe(W(p)).subscribe(c=>{let{width:l}=ce(o);o.scrollBy({left:l*c,behavior:"smooth"})}),r.pipe(W(p),b(c=>n.includes(c))).subscribe(c=>c.click()),o.classList.add("tabbed-labels--linked");for(let c of n){let l=R(`label[for="${c.id}"]`);l.replaceChildren(x("a",{href:`#${l.htmlFor}`,tabIndex:-1},...Array.from(l.childNodes))),h(l.firstElementChild,"click").pipe(W(p),b(f=>!(f.metaKey||f.ctrlKey)),w(f=>{f.preventDefault(),f.stopPropagation()})).subscribe(()=>{history.replaceState({},"",`#${l.htmlFor}`),l.click()})}return B("content.tabs.link")&&s.pipe(Ce(1),re(t)).subscribe(([{active:c},{offset:l}])=>{let f=c.innerText.trim();if(c.hasAttribute("data-md-switching"))c.removeAttribute("data-md-switching");else{let u=e.offsetTop-l.y;for(let y of P("[data-tabs]"))for(let L of P(":scope > input",y)){let X=R(`label[for="${L.id}"]`);if(X!==c&&X.innerText.trim()===f){X.setAttribute("data-md-switching",""),L.click();break}}window.scrollTo({top:e.offsetTop-u});let d=__md_get("__tabs")||[];__md_set("__tabs",[...new Set([f,...d])])}}),s.pipe(W(p)).subscribe(()=>{for(let c of P("audio, video",e))c.pause()}),Ya(n).pipe(w(c=>s.next(c)),_(()=>s.complete()),m(c=>$({ref:e},c)))}).pipe(Ke(se))}function zn(e,{viewport$:t,target$:r,print$:o}){return O(...P(".annotate:not(.highlight)",e).map(n=>Pn(n,{target$:r,print$:o})),...P("pre:not(.mermaid) > code",e).map(n=>jn(n,{target$:r,print$:o})),...P("pre.mermaid",e).map(n=>Wn(n)),...P("table:not([class])",e).map(n=>Vn(n)),...P("details",e).map(n=>Fn(n,{target$:r,print$:o})),...P("[data-tabs]",e).map(n=>Nn(n,{viewport$:t,target$:r})),...P("[title]",e).filter(()=>B("content.tooltips")).map(n=>mt(n,{viewport$:t})))}function Ba(e,{alert$:t}){return t.pipe(v(r=>O(I(!0),I(!1).pipe(Ge(2e3))).pipe(m(o=>({message:r,active:o})))))}function qn(e,t){let r=R(".md-typeset",e);return C(()=>{let o=new g;return o.subscribe(({message:n,active:i})=>{e.classList.toggle("md-dialog--active",i),r.textContent=n}),Ba(e,t).pipe(w(n=>o.next(n)),_(()=>o.complete()),m(n=>$({ref:e},n)))})}var Ga=0;function Ja(e,t){document.body.append(e);let{width:r}=ce(e);e.style.setProperty("--md-tooltip-width",`${r}px`),e.remove();let o=cr(t),n=typeof o!="undefined"?Ne(o):I({x:0,y:0}),i=O(et(t),$t(t)).pipe(K());return z([i,n]).pipe(m(([a,s])=>{let{x:p,y:c}=Ve(t),l=ce(t),f=t.closest("table");return f&&t.parentElement&&(p+=f.offsetLeft+t.parentElement.offsetLeft,c+=f.offsetTop+t.parentElement.offsetTop),{active:a,offset:{x:p-s.x+l.width/2-r/2,y:c-s.y+l.height+8}}}))}function Qn(e){let t=e.title;if(!t.length)return S;let r=`__tooltip_${Ga++}`,o=Rt(r,"inline"),n=R(".md-typeset",o);return n.innerHTML=t,C(()=>{let i=new g;return i.subscribe({next({offset:a}){o.style.setProperty("--md-tooltip-x",`${a.x}px`),o.style.setProperty("--md-tooltip-y",`${a.y}px`)},complete(){o.style.removeProperty("--md-tooltip-x"),o.style.removeProperty("--md-tooltip-y")}}),O(i.pipe(b(({active:a})=>a)),i.pipe(_e(250),b(({active:a})=>!a))).subscribe({next({active:a}){a?(e.insertAdjacentElement("afterend",o),e.setAttribute("aria-describedby",r),e.removeAttribute("title")):(o.remove(),e.removeAttribute("aria-describedby"),e.setAttribute("title",t))},complete(){o.remove(),e.removeAttribute("aria-describedby"),e.setAttribute("title",t)}}),i.pipe(Me(16,me)).subscribe(({active:a})=>{o.classList.toggle("md-tooltip--active",a)}),i.pipe(pt(125,me),b(()=>!!e.offsetParent),m(()=>e.offsetParent.getBoundingClientRect()),m(({x:a})=>a)).subscribe({next(a){a?o.style.setProperty("--md-tooltip-0",`${-a}px`):o.style.removeProperty("--md-tooltip-0")},complete(){o.style.removeProperty("--md-tooltip-0")}}),Ja(o,e).pipe(w(a=>i.next(a)),_(()=>i.complete()),m(a=>$({ref:e},a)))}).pipe(Ke(se))}function Xa({viewport$:e}){if(!B("header.autohide"))return I(!1);let t=e.pipe(m(({offset:{y:n}})=>n),Be(2,1),m(([n,i])=>[nMath.abs(i-n.y)>100),m(([,[n]])=>n),K()),o=ze("search");return z([e,o]).pipe(m(([{offset:n},i])=>n.y>400&&!i),K(),v(n=>n?r:I(!1)),Q(!1))}function Kn(e,t){return C(()=>z([ge(e),Xa(t)])).pipe(m(([{height:r},o])=>({height:r,hidden:o})),K((r,o)=>r.height===o.height&&r.hidden===o.hidden),G(1))}function Yn(e,{header$:t,main$:r}){return C(()=>{let o=new g,n=o.pipe(Z(),ie(!0));o.pipe(ee("active"),He(t)).subscribe(([{active:a},{hidden:s}])=>{e.classList.toggle("md-header--shadow",a&&!s),e.hidden=s});let i=ue(P("[title]",e)).pipe(b(()=>B("content.tooltips")),ne(a=>Qn(a)));return r.subscribe(o),t.pipe(W(n),m(a=>$({ref:e},a)),Re(i.pipe(W(n))))})}function Za(e,{viewport$:t,header$:r}){return mr(e,{viewport$:t,header$:r}).pipe(m(({offset:{y:o}})=>{let{height:n}=ce(e);return{active:o>=n}}),ee("active"))}function Bn(e,t){return C(()=>{let r=new g;r.subscribe({next({active:n}){e.classList.toggle("md-header__title--active",n)},complete(){e.classList.remove("md-header__title--active")}});let o=fe(".md-content h1");return typeof o=="undefined"?S:Za(o,t).pipe(w(n=>r.next(n)),_(()=>r.complete()),m(n=>$({ref:e},n)))})}function Gn(e,{viewport$:t,header$:r}){let o=r.pipe(m(({height:i})=>i),K()),n=o.pipe(v(()=>ge(e).pipe(m(({height:i})=>({top:e.offsetTop,bottom:e.offsetTop+i})),ee("bottom"))));return z([o,n,t]).pipe(m(([i,{top:a,bottom:s},{offset:{y:p},size:{height:c}}])=>(c=Math.max(0,c-Math.max(0,a-p,i)-Math.max(0,c+p-s)),{offset:a-i,height:c,active:a-i<=p})),K((i,a)=>i.offset===a.offset&&i.height===a.height&&i.active===a.active))}function es(e){let t=__md_get("__palette")||{index:e.findIndex(o=>matchMedia(o.getAttribute("data-md-color-media")).matches)},r=Math.max(0,Math.min(t.index,e.length-1));return I(...e).pipe(ne(o=>h(o,"change").pipe(m(()=>o))),Q(e[r]),m(o=>({index:e.indexOf(o),color:{media:o.getAttribute("data-md-color-media"),scheme:o.getAttribute("data-md-color-scheme"),primary:o.getAttribute("data-md-color-primary"),accent:o.getAttribute("data-md-color-accent")}})),G(1))}function Jn(e){let t=P("input",e),r=x("meta",{name:"theme-color"});document.head.appendChild(r);let o=x("meta",{name:"color-scheme"});document.head.appendChild(o);let n=Pt("(prefers-color-scheme: light)");return C(()=>{let i=new g;return i.subscribe(a=>{if(document.body.setAttribute("data-md-color-switching",""),a.color.media==="(prefers-color-scheme)"){let s=matchMedia("(prefers-color-scheme: light)"),p=document.querySelector(s.matches?"[data-md-color-media='(prefers-color-scheme: light)']":"[data-md-color-media='(prefers-color-scheme: dark)']");a.color.scheme=p.getAttribute("data-md-color-scheme"),a.color.primary=p.getAttribute("data-md-color-primary"),a.color.accent=p.getAttribute("data-md-color-accent")}for(let[s,p]of Object.entries(a.color))document.body.setAttribute(`data-md-color-${s}`,p);for(let s=0;sa.key==="Enter"),re(i,(a,s)=>s)).subscribe(({index:a})=>{a=(a+1)%t.length,t[a].click(),t[a].focus()}),i.pipe(m(()=>{let a=Se("header"),s=window.getComputedStyle(a);return o.content=s.colorScheme,s.backgroundColor.match(/\d+/g).map(p=>(+p).toString(16).padStart(2,"0")).join("")})).subscribe(a=>r.content=`#${a}`),i.pipe(ve(se)).subscribe(()=>{document.body.removeAttribute("data-md-color-switching")}),es(t).pipe(W(n.pipe(Ce(1))),ct(),w(a=>i.next(a)),_(()=>i.complete()),m(a=>$({ref:e},a)))})}function Xn(e,{progress$:t}){return C(()=>{let r=new g;return r.subscribe(({value:o})=>{e.style.setProperty("--md-progress-value",`${o}`)}),t.pipe(w(o=>r.next({value:o})),_(()=>r.complete()),m(o=>({ref:e,value:o})))})}var Jr=Mt(Br());function ts(e){e.setAttribute("data-md-copying","");let t=e.closest("[data-copy]"),r=t?t.getAttribute("data-copy"):e.innerText;return e.removeAttribute("data-md-copying"),r.trimEnd()}function Zn({alert$:e}){Jr.default.isSupported()&&new j(t=>{new Jr.default("[data-clipboard-target], [data-clipboard-text]",{text:r=>r.getAttribute("data-clipboard-text")||ts(R(r.getAttribute("data-clipboard-target")))}).on("success",r=>t.next(r))}).pipe(w(t=>{t.trigger.focus()}),m(()=>Ee("clipboard.copied"))).subscribe(e)}function ei(e,t){return e.protocol=t.protocol,e.hostname=t.hostname,e}function rs(e,t){let r=new Map;for(let o of P("url",e)){let n=R("loc",o),i=[ei(new URL(n.textContent),t)];r.set(`${i[0]}`,i);for(let a of P("[rel=alternate]",o)){let s=a.getAttribute("href");s!=null&&i.push(ei(new URL(s),t))}}return r}function ur(e){return un(new URL("sitemap.xml",e)).pipe(m(t=>rs(t,new URL(e))),de(()=>I(new Map)))}function os(e,t){if(!(e.target instanceof Element))return S;let r=e.target.closest("a");if(r===null)return S;if(r.target||e.metaKey||e.ctrlKey)return S;let o=new URL(r.href);return o.search=o.hash="",t.has(`${o}`)?(e.preventDefault(),I(new URL(r.href))):S}function ti(e){let t=new Map;for(let r of P(":scope > *",e.head))t.set(r.outerHTML,r);return t}function ri(e){for(let t of P("[href], [src]",e))for(let r of["href","src"]){let o=t.getAttribute(r);if(o&&!/^(?:[a-z]+:)?\/\//i.test(o)){t[r]=t[r];break}}return I(e)}function ns(e){for(let o of["[data-md-component=announce]","[data-md-component=container]","[data-md-component=header-topic]","[data-md-component=outdated]","[data-md-component=logo]","[data-md-component=skip]",...B("navigation.tabs.sticky")?["[data-md-component=tabs]"]:[]]){let n=fe(o),i=fe(o,e);typeof n!="undefined"&&typeof i!="undefined"&&n.replaceWith(i)}let t=ti(document);for(let[o,n]of ti(e))t.has(o)?t.delete(o):document.head.appendChild(n);for(let o of t.values()){let n=o.getAttribute("name");n!=="theme-color"&&n!=="color-scheme"&&o.remove()}let r=Se("container");return We(P("script",r)).pipe(v(o=>{let n=e.createElement("script");if(o.src){for(let i of o.getAttributeNames())n.setAttribute(i,o.getAttribute(i));return o.replaceWith(n),new j(i=>{n.onload=()=>i.complete()})}else return n.textContent=o.textContent,o.replaceWith(n),S}),Z(),ie(document))}function oi({location$:e,viewport$:t,progress$:r}){let o=xe();if(location.protocol==="file:")return S;let n=ur(o.base);I(document).subscribe(ri);let i=h(document.body,"click").pipe(He(n),v(([p,c])=>os(p,c)),pe()),a=h(window,"popstate").pipe(m(ye),pe());i.pipe(re(t)).subscribe(([p,{offset:c}])=>{history.replaceState(c,""),history.pushState(null,"",p)}),O(i,a).subscribe(e);let s=e.pipe(ee("pathname"),v(p=>fn(p,{progress$:r}).pipe(de(()=>(lt(p,!0),S)))),v(ri),v(ns),pe());return O(s.pipe(re(e,(p,c)=>c)),s.pipe(v(()=>e),ee("pathname"),v(()=>e),ee("hash")),e.pipe(K((p,c)=>p.pathname===c.pathname&&p.hash===c.hash),v(()=>i),w(()=>history.back()))).subscribe(p=>{var c,l;history.state!==null||!p.hash?window.scrollTo(0,(l=(c=history.state)==null?void 0:c.y)!=null?l:0):(history.scrollRestoration="auto",pn(p.hash),history.scrollRestoration="manual")}),e.subscribe(()=>{history.scrollRestoration="manual"}),h(window,"beforeunload").subscribe(()=>{history.scrollRestoration="auto"}),t.pipe(ee("offset"),_e(100)).subscribe(({offset:p})=>{history.replaceState(p,"")}),s}var ni=Mt(qr());function ii(e){let t=e.separator.split("|").map(n=>n.replace(/(\(\?[!=<][^)]+\))/g,"").length===0?"\uFFFD":n).join("|"),r=new RegExp(t,"img"),o=(n,i,a)=>`${i}${a}`;return n=>{n=n.replace(/[\s*+\-:~^]+/g," ").trim();let i=new RegExp(`(^|${e.separator}|)(${n.replace(/[|\\{}()[\]^$+*?.-]/g,"\\$&").replace(r,"|")})`,"img");return a=>(0,ni.default)(a).replace(i,o).replace(/<\/mark>(\s+)]*>/img,"$1")}}function jt(e){return e.type===1}function dr(e){return e.type===3}function ai(e,t){let r=yn(e);return O(I(location.protocol!=="file:"),ze("search")).pipe(Ae(o=>o),v(()=>t)).subscribe(({config:o,docs:n})=>r.next({type:0,data:{config:o,docs:n,options:{suggest:B("search.suggest")}}})),r}function si(e){var l;let{selectedVersionSitemap:t,selectedVersionBaseURL:r,currentLocation:o,currentBaseURL:n}=e,i=(l=Xr(n))==null?void 0:l.pathname;if(i===void 0)return;let a=ss(o.pathname,i);if(a===void 0)return;let s=ps(t.keys());if(!t.has(s))return;let p=Xr(a,s);if(!p||!t.has(p.href))return;let c=Xr(a,r);if(c)return c.hash=o.hash,c.search=o.search,c}function Xr(e,t){try{return new URL(e,t)}catch(r){return}}function ss(e,t){if(e.startsWith(t))return e.slice(t.length)}function cs(e,t){let r=Math.min(e.length,t.length),o;for(o=0;oS)),o=r.pipe(m(n=>{let[,i]=t.base.match(/([^/]+)\/?$/);return n.find(({version:a,aliases:s})=>a===i||s.includes(i))||n[0]}));r.pipe(m(n=>new Map(n.map(i=>[`${new URL(`../${i.version}/`,t.base)}`,i]))),v(n=>h(document.body,"click").pipe(b(i=>!i.metaKey&&!i.ctrlKey),re(o),v(([i,a])=>{if(i.target instanceof Element){let s=i.target.closest("a");if(s&&!s.target&&n.has(s.href)){let p=s.href;return!i.target.closest(".md-version")&&n.get(p)===a?S:(i.preventDefault(),I(new URL(p)))}}return S}),v(i=>ur(i).pipe(m(a=>{var s;return(s=si({selectedVersionSitemap:a,selectedVersionBaseURL:i,currentLocation:ye(),currentBaseURL:t.base}))!=null?s:i})))))).subscribe(n=>lt(n,!0)),z([r,o]).subscribe(([n,i])=>{R(".md-header__topic").appendChild(Cn(n,i))}),e.pipe(v(()=>o)).subscribe(n=>{var a;let i=__md_get("__outdated",sessionStorage);if(i===null){i=!0;let s=((a=t.version)==null?void 0:a.default)||"latest";Array.isArray(s)||(s=[s]);e:for(let p of s)for(let c of n.aliases.concat(n.version))if(new RegExp(p,"i").test(c)){i=!1;break e}__md_set("__outdated",i,sessionStorage)}if(i)for(let s of ae("outdated"))s.hidden=!1})}function ls(e,{worker$:t}){let{searchParams:r}=ye();r.has("q")&&(Je("search",!0),e.value=r.get("q"),e.focus(),ze("search").pipe(Ae(i=>!i)).subscribe(()=>{let i=ye();i.searchParams.delete("q"),history.replaceState({},"",`${i}`)}));let o=et(e),n=O(t.pipe(Ae(jt)),h(e,"keyup"),o).pipe(m(()=>e.value),K());return z([n,o]).pipe(m(([i,a])=>({value:i,focus:a})),G(1))}function pi(e,{worker$:t}){let r=new g,o=r.pipe(Z(),ie(!0));z([t.pipe(Ae(jt)),r],(i,a)=>a).pipe(ee("value")).subscribe(({value:i})=>t.next({type:2,data:i})),r.pipe(ee("focus")).subscribe(({focus:i})=>{i&&Je("search",i)}),h(e.form,"reset").pipe(W(o)).subscribe(()=>e.focus());let n=R("header [for=__search]");return h(n,"click").subscribe(()=>e.focus()),ls(e,{worker$:t}).pipe(w(i=>r.next(i)),_(()=>r.complete()),m(i=>$({ref:e},i)),G(1))}function li(e,{worker$:t,query$:r}){let o=new g,n=on(e.parentElement).pipe(b(Boolean)),i=e.parentElement,a=R(":scope > :first-child",e),s=R(":scope > :last-child",e);ze("search").subscribe(l=>s.setAttribute("role",l?"list":"presentation")),o.pipe(re(r),Wr(t.pipe(Ae(jt)))).subscribe(([{items:l},{value:f}])=>{switch(l.length){case 0:a.textContent=f.length?Ee("search.result.none"):Ee("search.result.placeholder");break;case 1:a.textContent=Ee("search.result.one");break;default:let u=sr(l.length);a.textContent=Ee("search.result.other",u)}});let p=o.pipe(w(()=>s.innerHTML=""),v(({items:l})=>O(I(...l.slice(0,10)),I(...l.slice(10)).pipe(Be(4),Vr(n),v(([f])=>f)))),m(Mn),pe());return p.subscribe(l=>s.appendChild(l)),p.pipe(ne(l=>{let f=fe("details",l);return typeof f=="undefined"?S:h(f,"toggle").pipe(W(o),m(()=>f))})).subscribe(l=>{l.open===!1&&l.offsetTop<=i.scrollTop&&i.scrollTo({top:l.offsetTop})}),t.pipe(b(dr),m(({data:l})=>l)).pipe(w(l=>o.next(l)),_(()=>o.complete()),m(l=>$({ref:e},l)))}function ms(e,{query$:t}){return t.pipe(m(({value:r})=>{let o=ye();return o.hash="",r=r.replace(/\s+/g,"+").replace(/&/g,"%26").replace(/=/g,"%3D"),o.search=`q=${r}`,{url:o}}))}function mi(e,t){let r=new g,o=r.pipe(Z(),ie(!0));return r.subscribe(({url:n})=>{e.setAttribute("data-clipboard-text",e.href),e.href=`${n}`}),h(e,"click").pipe(W(o)).subscribe(n=>n.preventDefault()),ms(e,t).pipe(w(n=>r.next(n)),_(()=>r.complete()),m(n=>$({ref:e},n)))}function fi(e,{worker$:t,keyboard$:r}){let o=new g,n=Se("search-query"),i=O(h(n,"keydown"),h(n,"focus")).pipe(ve(se),m(()=>n.value),K());return o.pipe(He(i),m(([{suggest:s},p])=>{let c=p.split(/([\s-]+)/);if(s!=null&&s.length&&c[c.length-1]){let l=s[s.length-1];l.startsWith(c[c.length-1])&&(c[c.length-1]=l)}else c.length=0;return c})).subscribe(s=>e.innerHTML=s.join("").replace(/\s/g," ")),r.pipe(b(({mode:s})=>s==="search")).subscribe(s=>{switch(s.type){case"ArrowRight":e.innerText.length&&n.selectionStart===n.value.length&&(n.value=e.innerText);break}}),t.pipe(b(dr),m(({data:s})=>s)).pipe(w(s=>o.next(s)),_(()=>o.complete()),m(()=>({ref:e})))}function ui(e,{index$:t,keyboard$:r}){let o=xe();try{let n=ai(o.search,t),i=Se("search-query",e),a=Se("search-result",e);h(e,"click").pipe(b(({target:p})=>p instanceof Element&&!!p.closest("a"))).subscribe(()=>Je("search",!1)),r.pipe(b(({mode:p})=>p==="search")).subscribe(p=>{let c=Ie();switch(p.type){case"Enter":if(c===i){let l=new Map;for(let f of P(":first-child [href]",a)){let u=f.firstElementChild;l.set(f,parseFloat(u.getAttribute("data-md-score")))}if(l.size){let[[f]]=[...l].sort(([,u],[,d])=>d-u);f.click()}p.claim()}break;case"Escape":case"Tab":Je("search",!1),i.blur();break;case"ArrowUp":case"ArrowDown":if(typeof c=="undefined")i.focus();else{let l=[i,...P(":not(details) > [href], summary, details[open] [href]",a)],f=Math.max(0,(Math.max(0,l.indexOf(c))+l.length+(p.type==="ArrowUp"?-1:1))%l.length);l[f].focus()}p.claim();break;default:i!==Ie()&&i.focus()}}),r.pipe(b(({mode:p})=>p==="global")).subscribe(p=>{switch(p.type){case"f":case"s":case"/":i.focus(),i.select(),p.claim();break}});let s=pi(i,{worker$:n});return O(s,li(a,{worker$:n,query$:s})).pipe(Re(...ae("search-share",e).map(p=>mi(p,{query$:s})),...ae("search-suggest",e).map(p=>fi(p,{worker$:n,keyboard$:r}))))}catch(n){return e.hidden=!0,Ye}}function di(e,{index$:t,location$:r}){return z([t,r.pipe(Q(ye()),b(o=>!!o.searchParams.get("h")))]).pipe(m(([o,n])=>ii(o.config)(n.searchParams.get("h"))),m(o=>{var a;let n=new Map,i=document.createNodeIterator(e,NodeFilter.SHOW_TEXT);for(let s=i.nextNode();s;s=i.nextNode())if((a=s.parentElement)!=null&&a.offsetHeight){let p=s.textContent,c=o(p);c.length>p.length&&n.set(s,c)}for(let[s,p]of n){let{childNodes:c}=x("span",null,p);s.replaceWith(...Array.from(c))}return{ref:e,nodes:n}}))}function fs(e,{viewport$:t,main$:r}){let o=e.closest(".md-grid"),n=o.offsetTop-o.parentElement.offsetTop;return z([r,t]).pipe(m(([{offset:i,height:a},{offset:{y:s}}])=>(a=a+Math.min(n,Math.max(0,s-i))-n,{height:a,locked:s>=i+n})),K((i,a)=>i.height===a.height&&i.locked===a.locked))}function Zr(e,o){var n=o,{header$:t}=n,r=so(n,["header$"]);let i=R(".md-sidebar__scrollwrap",e),{y:a}=Ve(i);return C(()=>{let s=new g,p=s.pipe(Z(),ie(!0)),c=s.pipe(Me(0,me));return c.pipe(re(t)).subscribe({next([{height:l},{height:f}]){i.style.height=`${l-2*a}px`,e.style.top=`${f}px`},complete(){i.style.height="",e.style.top=""}}),c.pipe(Ae()).subscribe(()=>{for(let l of P(".md-nav__link--active[href]",e)){if(!l.clientHeight)continue;let f=l.closest(".md-sidebar__scrollwrap");if(typeof f!="undefined"){let u=l.offsetTop-f.offsetTop,{height:d}=ce(f);f.scrollTo({top:u-d/2})}}}),ue(P("label[tabindex]",e)).pipe(ne(l=>h(l,"click").pipe(ve(se),m(()=>l),W(p)))).subscribe(l=>{let f=R(`[id="${l.htmlFor}"]`);R(`[aria-labelledby="${l.id}"]`).setAttribute("aria-expanded",`${f.checked}`)}),fs(e,r).pipe(w(l=>s.next(l)),_(()=>s.complete()),m(l=>$({ref:e},l)))})}function hi(e,t){if(typeof t!="undefined"){let r=`https://api.github.com/repos/${e}/${t}`;return st(je(`${r}/releases/latest`).pipe(de(()=>S),m(o=>({version:o.tag_name})),De({})),je(r).pipe(de(()=>S),m(o=>({stars:o.stargazers_count,forks:o.forks_count})),De({}))).pipe(m(([o,n])=>$($({},o),n)))}else{let r=`https://api.github.com/users/${e}`;return je(r).pipe(m(o=>({repositories:o.public_repos})),De({}))}}function bi(e,t){let r=`https://${e}/api/v4/projects/${encodeURIComponent(t)}`;return st(je(`${r}/releases/permalink/latest`).pipe(de(()=>S),m(({tag_name:o})=>({version:o})),De({})),je(r).pipe(de(()=>S),m(({star_count:o,forks_count:n})=>({stars:o,forks:n})),De({}))).pipe(m(([o,n])=>$($({},o),n)))}function vi(e){let t=e.match(/^.+github\.com\/([^/]+)\/?([^/]+)?/i);if(t){let[,r,o]=t;return hi(r,o)}if(t=e.match(/^.+?([^/]*gitlab[^/]+)\/(.+?)\/?$/i),t){let[,r,o]=t;return bi(r,o)}return S}var us;function ds(e){return us||(us=C(()=>{let t=__md_get("__source",sessionStorage);if(t)return I(t);if(ae("consent").length){let o=__md_get("__consent");if(!(o&&o.github))return S}return vi(e.href).pipe(w(o=>__md_set("__source",o,sessionStorage)))}).pipe(de(()=>S),b(t=>Object.keys(t).length>0),m(t=>({facts:t})),G(1)))}function gi(e){let t=R(":scope > :last-child",e);return C(()=>{let r=new g;return r.subscribe(({facts:o})=>{t.appendChild(_n(o)),t.classList.add("md-source__repository--active")}),ds(e).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))})}function hs(e,{viewport$:t,header$:r}){return ge(document.body).pipe(v(()=>mr(e,{header$:r,viewport$:t})),m(({offset:{y:o}})=>({hidden:o>=10})),ee("hidden"))}function yi(e,t){return C(()=>{let r=new g;return r.subscribe({next({hidden:o}){e.hidden=o},complete(){e.hidden=!1}}),(B("navigation.tabs.sticky")?I({hidden:!1}):hs(e,t)).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))})}function bs(e,{viewport$:t,header$:r}){let o=new Map,n=P(".md-nav__link",e);for(let s of n){let p=decodeURIComponent(s.hash.substring(1)),c=fe(`[id="${p}"]`);typeof c!="undefined"&&o.set(s,c)}let i=r.pipe(ee("height"),m(({height:s})=>{let p=Se("main"),c=R(":scope > :first-child",p);return s+.8*(c.offsetTop-p.offsetTop)}),pe());return ge(document.body).pipe(ee("height"),v(s=>C(()=>{let p=[];return I([...o].reduce((c,[l,f])=>{for(;p.length&&o.get(p[p.length-1]).tagName>=f.tagName;)p.pop();let u=f.offsetTop;for(;!u&&f.parentElement;)f=f.parentElement,u=f.offsetTop;let d=f.offsetParent;for(;d;d=d.offsetParent)u+=d.offsetTop;return c.set([...p=[...p,l]].reverse(),u)},new Map))}).pipe(m(p=>new Map([...p].sort(([,c],[,l])=>c-l))),He(i),v(([p,c])=>t.pipe(Fr(([l,f],{offset:{y:u},size:d})=>{let y=u+d.height>=Math.floor(s.height);for(;f.length;){let[,L]=f[0];if(L-c=u&&!y)f=[l.pop(),...f];else break}return[l,f]},[[],[...p]]),K((l,f)=>l[0]===f[0]&&l[1]===f[1])))))).pipe(m(([s,p])=>({prev:s.map(([c])=>c),next:p.map(([c])=>c)})),Q({prev:[],next:[]}),Be(2,1),m(([s,p])=>s.prev.length{let i=new g,a=i.pipe(Z(),ie(!0));if(i.subscribe(({prev:s,next:p})=>{for(let[c]of p)c.classList.remove("md-nav__link--passed"),c.classList.remove("md-nav__link--active");for(let[c,[l]]of s.entries())l.classList.add("md-nav__link--passed"),l.classList.toggle("md-nav__link--active",c===s.length-1)}),B("toc.follow")){let s=O(t.pipe(_e(1),m(()=>{})),t.pipe(_e(250),m(()=>"smooth")));i.pipe(b(({prev:p})=>p.length>0),He(o.pipe(ve(se))),re(s)).subscribe(([[{prev:p}],c])=>{let[l]=p[p.length-1];if(l.offsetHeight){let f=cr(l);if(typeof f!="undefined"){let u=l.offsetTop-f.offsetTop,{height:d}=ce(f);f.scrollTo({top:u-d/2,behavior:c})}}})}return B("navigation.tracking")&&t.pipe(W(a),ee("offset"),_e(250),Ce(1),W(n.pipe(Ce(1))),ct({delay:250}),re(i)).subscribe(([,{prev:s}])=>{let p=ye(),c=s[s.length-1];if(c&&c.length){let[l]=c,{hash:f}=new URL(l.href);p.hash!==f&&(p.hash=f,history.replaceState({},"",`${p}`))}else p.hash="",history.replaceState({},"",`${p}`)}),bs(e,{viewport$:t,header$:r}).pipe(w(s=>i.next(s)),_(()=>i.complete()),m(s=>$({ref:e},s)))})}function vs(e,{viewport$:t,main$:r,target$:o}){let n=t.pipe(m(({offset:{y:a}})=>a),Be(2,1),m(([a,s])=>a>s&&s>0),K()),i=r.pipe(m(({active:a})=>a));return z([i,n]).pipe(m(([a,s])=>!(a&&s)),K(),W(o.pipe(Ce(1))),ie(!0),ct({delay:250}),m(a=>({hidden:a})))}function Ei(e,{viewport$:t,header$:r,main$:o,target$:n}){let i=new g,a=i.pipe(Z(),ie(!0));return i.subscribe({next({hidden:s}){e.hidden=s,s?(e.setAttribute("tabindex","-1"),e.blur()):e.removeAttribute("tabindex")},complete(){e.style.top="",e.hidden=!0,e.removeAttribute("tabindex")}}),r.pipe(W(a),ee("height")).subscribe(({height:s})=>{e.style.top=`${s+16}px`}),h(e,"click").subscribe(s=>{s.preventDefault(),window.scrollTo({top:0})}),vs(e,{viewport$:t,main$:o,target$:n}).pipe(w(s=>i.next(s)),_(()=>i.complete()),m(s=>$({ref:e},s)))}function wi({document$:e,viewport$:t}){e.pipe(v(()=>P(".md-ellipsis")),ne(r=>tt(r).pipe(W(e.pipe(Ce(1))),b(o=>o),m(()=>r),Te(1))),b(r=>r.offsetWidth{let o=r.innerText,n=r.closest("a")||r;return n.title=o,B("content.tooltips")?mt(n,{viewport$:t}).pipe(W(e.pipe(Ce(1))),_(()=>n.removeAttribute("title"))):S})).subscribe(),B("content.tooltips")&&e.pipe(v(()=>P(".md-status")),ne(r=>mt(r,{viewport$:t}))).subscribe()}function Ti({document$:e,tablet$:t}){e.pipe(v(()=>P(".md-toggle--indeterminate")),w(r=>{r.indeterminate=!0,r.checked=!1}),ne(r=>h(r,"change").pipe(Dr(()=>r.classList.contains("md-toggle--indeterminate")),m(()=>r))),re(t)).subscribe(([r,o])=>{r.classList.remove("md-toggle--indeterminate"),o&&(r.checked=!1)})}function gs(){return/(iPad|iPhone|iPod)/.test(navigator.userAgent)}function Si({document$:e}){e.pipe(v(()=>P("[data-md-scrollfix]")),w(t=>t.removeAttribute("data-md-scrollfix")),b(gs),ne(t=>h(t,"touchstart").pipe(m(()=>t)))).subscribe(t=>{let r=t.scrollTop;r===0?t.scrollTop=1:r+t.offsetHeight===t.scrollHeight&&(t.scrollTop=r-1)})}function Oi({viewport$:e,tablet$:t}){z([ze("search"),t]).pipe(m(([r,o])=>r&&!o),v(r=>I(r).pipe(Ge(r?400:100))),re(e)).subscribe(([r,{offset:{y:o}}])=>{if(r)document.body.setAttribute("data-md-scrolllock",""),document.body.style.top=`-${o}px`;else{let n=-1*parseInt(document.body.style.top,10);document.body.removeAttribute("data-md-scrolllock"),document.body.style.top="",n&&window.scrollTo(0,n)}})}Object.entries||(Object.entries=function(e){let t=[];for(let r of Object.keys(e))t.push([r,e[r]]);return t});Object.values||(Object.values=function(e){let t=[];for(let r of Object.keys(e))t.push(e[r]);return t});typeof Element!="undefined"&&(Element.prototype.scrollTo||(Element.prototype.scrollTo=function(e,t){typeof e=="object"?(this.scrollLeft=e.left,this.scrollTop=e.top):(this.scrollLeft=e,this.scrollTop=t)}),Element.prototype.replaceWith||(Element.prototype.replaceWith=function(...e){let t=this.parentNode;if(t){e.length===0&&t.removeChild(this);for(let r=e.length-1;r>=0;r--){let o=e[r];typeof o=="string"?o=document.createTextNode(o):o.parentNode&&o.parentNode.removeChild(o),r?t.insertBefore(this.previousSibling,o):t.replaceChild(o,this)}}}));function ys(){return location.protocol==="file:"?Tt(`${new URL("search/search_index.js",eo.base)}`).pipe(m(()=>__index),G(1)):je(new URL("search/search_index.json",eo.base))}document.documentElement.classList.remove("no-js");document.documentElement.classList.add("js");var ot=Go(),Ut=sn(),Lt=ln(Ut),to=an(),Oe=gn(),hr=Pt("(min-width: 960px)"),Mi=Pt("(min-width: 1220px)"),_i=mn(),eo=xe(),Ai=document.forms.namedItem("search")?ys():Ye,ro=new g;Zn({alert$:ro});var oo=new g;B("navigation.instant")&&oi({location$:Ut,viewport$:Oe,progress$:oo}).subscribe(ot);var Li;((Li=eo.version)==null?void 0:Li.provider)==="mike"&&ci({document$:ot});O(Ut,Lt).pipe(Ge(125)).subscribe(()=>{Je("drawer",!1),Je("search",!1)});to.pipe(b(({mode:e})=>e==="global")).subscribe(e=>{switch(e.type){case"p":case",":let t=fe("link[rel=prev]");typeof t!="undefined"&<(t);break;case"n":case".":let r=fe("link[rel=next]");typeof r!="undefined"&<(r);break;case"Enter":let o=Ie();o instanceof HTMLLabelElement&&o.click()}});wi({viewport$:Oe,document$:ot});Ti({document$:ot,tablet$:hr});Si({document$:ot});Oi({viewport$:Oe,tablet$:hr});var rt=Kn(Se("header"),{viewport$:Oe}),Ft=ot.pipe(m(()=>Se("main")),v(e=>Gn(e,{viewport$:Oe,header$:rt})),G(1)),xs=O(...ae("consent").map(e=>En(e,{target$:Lt})),...ae("dialog").map(e=>qn(e,{alert$:ro})),...ae("palette").map(e=>Jn(e)),...ae("progress").map(e=>Xn(e,{progress$:oo})),...ae("search").map(e=>ui(e,{index$:Ai,keyboard$:to})),...ae("source").map(e=>gi(e))),Es=C(()=>O(...ae("announce").map(e=>xn(e)),...ae("content").map(e=>zn(e,{viewport$:Oe,target$:Lt,print$:_i})),...ae("content").map(e=>B("search.highlight")?di(e,{index$:Ai,location$:Ut}):S),...ae("header").map(e=>Yn(e,{viewport$:Oe,header$:rt,main$:Ft})),...ae("header-title").map(e=>Bn(e,{viewport$:Oe,header$:rt})),...ae("sidebar").map(e=>e.getAttribute("data-md-type")==="navigation"?Nr(Mi,()=>Zr(e,{viewport$:Oe,header$:rt,main$:Ft})):Nr(hr,()=>Zr(e,{viewport$:Oe,header$:rt,main$:Ft}))),...ae("tabs").map(e=>yi(e,{viewport$:Oe,header$:rt})),...ae("toc").map(e=>xi(e,{viewport$:Oe,header$:rt,main$:Ft,target$:Lt})),...ae("top").map(e=>Ei(e,{viewport$:Oe,header$:rt,main$:Ft,target$:Lt})))),Ci=ot.pipe(v(()=>Es),Re(xs),G(1));Ci.subscribe();window.document$=ot;window.location$=Ut;window.target$=Lt;window.keyboard$=to;window.viewport$=Oe;window.tablet$=hr;window.screen$=Mi;window.print$=_i;window.alert$=ro;window.progress$=oo;window.component$=Ci;})(); +//# sourceMappingURL=bundle.88dd0f4e.min.js.map + diff --git a/v13/assets/javascripts/bundle.88dd0f4e.min.js.map b/v13/assets/javascripts/bundle.88dd0f4e.min.js.map new file mode 100644 index 0000000000..dab2a8754b --- /dev/null +++ b/v13/assets/javascripts/bundle.88dd0f4e.min.js.map @@ -0,0 +1,7 @@ +{ + "version": 3, + "sources": ["node_modules/focus-visible/dist/focus-visible.js", "node_modules/escape-html/index.js", "node_modules/clipboard/dist/clipboard.js", "src/templates/assets/javascripts/bundle.ts", "node_modules/tslib/tslib.es6.mjs", "node_modules/rxjs/src/internal/util/isFunction.ts", "node_modules/rxjs/src/internal/util/createErrorClass.ts", "node_modules/rxjs/src/internal/util/UnsubscriptionError.ts", "node_modules/rxjs/src/internal/util/arrRemove.ts", "node_modules/rxjs/src/internal/Subscription.ts", "node_modules/rxjs/src/internal/config.ts", "node_modules/rxjs/src/internal/scheduler/timeoutProvider.ts", "node_modules/rxjs/src/internal/util/reportUnhandledError.ts", "node_modules/rxjs/src/internal/util/noop.ts", "node_modules/rxjs/src/internal/NotificationFactories.ts", "node_modules/rxjs/src/internal/util/errorContext.ts", "node_modules/rxjs/src/internal/Subscriber.ts", "node_modules/rxjs/src/internal/symbol/observable.ts", "node_modules/rxjs/src/internal/util/identity.ts", "node_modules/rxjs/src/internal/util/pipe.ts", "node_modules/rxjs/src/internal/Observable.ts", "node_modules/rxjs/src/internal/util/lift.ts", "node_modules/rxjs/src/internal/operators/OperatorSubscriber.ts", "node_modules/rxjs/src/internal/scheduler/animationFrameProvider.ts", "node_modules/rxjs/src/internal/util/ObjectUnsubscribedError.ts", "node_modules/rxjs/src/internal/Subject.ts", "node_modules/rxjs/src/internal/BehaviorSubject.ts", "node_modules/rxjs/src/internal/scheduler/dateTimestampProvider.ts", "node_modules/rxjs/src/internal/ReplaySubject.ts", "node_modules/rxjs/src/internal/scheduler/Action.ts", "node_modules/rxjs/src/internal/scheduler/intervalProvider.ts", "node_modules/rxjs/src/internal/scheduler/AsyncAction.ts", "node_modules/rxjs/src/internal/Scheduler.ts", "node_modules/rxjs/src/internal/scheduler/AsyncScheduler.ts", "node_modules/rxjs/src/internal/scheduler/async.ts", "node_modules/rxjs/src/internal/scheduler/QueueAction.ts", "node_modules/rxjs/src/internal/scheduler/QueueScheduler.ts", "node_modules/rxjs/src/internal/scheduler/queue.ts", "node_modules/rxjs/src/internal/scheduler/AnimationFrameAction.ts", "node_modules/rxjs/src/internal/scheduler/AnimationFrameScheduler.ts", "node_modules/rxjs/src/internal/scheduler/animationFrame.ts", "node_modules/rxjs/src/internal/observable/empty.ts", "node_modules/rxjs/src/internal/util/isScheduler.ts", "node_modules/rxjs/src/internal/util/args.ts", "node_modules/rxjs/src/internal/util/isArrayLike.ts", "node_modules/rxjs/src/internal/util/isPromise.ts", "node_modules/rxjs/src/internal/util/isInteropObservable.ts", "node_modules/rxjs/src/internal/util/isAsyncIterable.ts", "node_modules/rxjs/src/internal/util/throwUnobservableError.ts", "node_modules/rxjs/src/internal/symbol/iterator.ts", "node_modules/rxjs/src/internal/util/isIterable.ts", "node_modules/rxjs/src/internal/util/isReadableStreamLike.ts", "node_modules/rxjs/src/internal/observable/innerFrom.ts", "node_modules/rxjs/src/internal/util/executeSchedule.ts", "node_modules/rxjs/src/internal/operators/observeOn.ts", "node_modules/rxjs/src/internal/operators/subscribeOn.ts", "node_modules/rxjs/src/internal/scheduled/scheduleObservable.ts", "node_modules/rxjs/src/internal/scheduled/schedulePromise.ts", "node_modules/rxjs/src/internal/scheduled/scheduleArray.ts", "node_modules/rxjs/src/internal/scheduled/scheduleIterable.ts", "node_modules/rxjs/src/internal/scheduled/scheduleAsyncIterable.ts", "node_modules/rxjs/src/internal/scheduled/scheduleReadableStreamLike.ts", "node_modules/rxjs/src/internal/scheduled/scheduled.ts", "node_modules/rxjs/src/internal/observable/from.ts", "node_modules/rxjs/src/internal/observable/of.ts", "node_modules/rxjs/src/internal/observable/throwError.ts", "node_modules/rxjs/src/internal/util/EmptyError.ts", "node_modules/rxjs/src/internal/util/isDate.ts", "node_modules/rxjs/src/internal/operators/map.ts", "node_modules/rxjs/src/internal/util/mapOneOrManyArgs.ts", "node_modules/rxjs/src/internal/util/argsArgArrayOrObject.ts", "node_modules/rxjs/src/internal/util/createObject.ts", "node_modules/rxjs/src/internal/observable/combineLatest.ts", "node_modules/rxjs/src/internal/operators/mergeInternals.ts", "node_modules/rxjs/src/internal/operators/mergeMap.ts", "node_modules/rxjs/src/internal/operators/mergeAll.ts", "node_modules/rxjs/src/internal/operators/concatAll.ts", "node_modules/rxjs/src/internal/observable/concat.ts", "node_modules/rxjs/src/internal/observable/defer.ts", "node_modules/rxjs/src/internal/observable/fromEvent.ts", "node_modules/rxjs/src/internal/observable/fromEventPattern.ts", "node_modules/rxjs/src/internal/observable/timer.ts", "node_modules/rxjs/src/internal/observable/merge.ts", "node_modules/rxjs/src/internal/observable/never.ts", "node_modules/rxjs/src/internal/util/argsOrArgArray.ts", "node_modules/rxjs/src/internal/operators/filter.ts", "node_modules/rxjs/src/internal/observable/zip.ts", "node_modules/rxjs/src/internal/operators/audit.ts", "node_modules/rxjs/src/internal/operators/auditTime.ts", "node_modules/rxjs/src/internal/operators/bufferCount.ts", "node_modules/rxjs/src/internal/operators/catchError.ts", "node_modules/rxjs/src/internal/operators/scanInternals.ts", "node_modules/rxjs/src/internal/operators/combineLatest.ts", "node_modules/rxjs/src/internal/operators/combineLatestWith.ts", "node_modules/rxjs/src/internal/operators/debounce.ts", "node_modules/rxjs/src/internal/operators/debounceTime.ts", "node_modules/rxjs/src/internal/operators/defaultIfEmpty.ts", "node_modules/rxjs/src/internal/operators/take.ts", "node_modules/rxjs/src/internal/operators/ignoreElements.ts", "node_modules/rxjs/src/internal/operators/mapTo.ts", "node_modules/rxjs/src/internal/operators/delayWhen.ts", "node_modules/rxjs/src/internal/operators/delay.ts", "node_modules/rxjs/src/internal/operators/distinctUntilChanged.ts", "node_modules/rxjs/src/internal/operators/distinctUntilKeyChanged.ts", "node_modules/rxjs/src/internal/operators/throwIfEmpty.ts", "node_modules/rxjs/src/internal/operators/endWith.ts", "node_modules/rxjs/src/internal/operators/finalize.ts", "node_modules/rxjs/src/internal/operators/first.ts", "node_modules/rxjs/src/internal/operators/takeLast.ts", "node_modules/rxjs/src/internal/operators/merge.ts", "node_modules/rxjs/src/internal/operators/mergeWith.ts", "node_modules/rxjs/src/internal/operators/repeat.ts", "node_modules/rxjs/src/internal/operators/scan.ts", "node_modules/rxjs/src/internal/operators/share.ts", "node_modules/rxjs/src/internal/operators/shareReplay.ts", "node_modules/rxjs/src/internal/operators/skip.ts", "node_modules/rxjs/src/internal/operators/skipUntil.ts", "node_modules/rxjs/src/internal/operators/startWith.ts", "node_modules/rxjs/src/internal/operators/switchMap.ts", "node_modules/rxjs/src/internal/operators/takeUntil.ts", "node_modules/rxjs/src/internal/operators/takeWhile.ts", "node_modules/rxjs/src/internal/operators/tap.ts", "node_modules/rxjs/src/internal/operators/throttle.ts", "node_modules/rxjs/src/internal/operators/throttleTime.ts", "node_modules/rxjs/src/internal/operators/withLatestFrom.ts", "node_modules/rxjs/src/internal/operators/zip.ts", "node_modules/rxjs/src/internal/operators/zipWith.ts", "src/templates/assets/javascripts/browser/document/index.ts", "src/templates/assets/javascripts/browser/element/_/index.ts", "src/templates/assets/javascripts/browser/element/focus/index.ts", "src/templates/assets/javascripts/browser/element/hover/index.ts", "src/templates/assets/javascripts/utilities/h/index.ts", "src/templates/assets/javascripts/utilities/round/index.ts", "src/templates/assets/javascripts/browser/script/index.ts", "src/templates/assets/javascripts/browser/element/size/_/index.ts", "src/templates/assets/javascripts/browser/element/size/content/index.ts", "src/templates/assets/javascripts/browser/element/offset/_/index.ts", "src/templates/assets/javascripts/browser/element/offset/content/index.ts", "src/templates/assets/javascripts/browser/element/visibility/index.ts", "src/templates/assets/javascripts/browser/toggle/index.ts", "src/templates/assets/javascripts/browser/keyboard/index.ts", "src/templates/assets/javascripts/browser/location/_/index.ts", "src/templates/assets/javascripts/browser/location/hash/index.ts", "src/templates/assets/javascripts/browser/media/index.ts", "src/templates/assets/javascripts/browser/request/index.ts", "src/templates/assets/javascripts/browser/viewport/offset/index.ts", "src/templates/assets/javascripts/browser/viewport/size/index.ts", "src/templates/assets/javascripts/browser/viewport/_/index.ts", "src/templates/assets/javascripts/browser/viewport/at/index.ts", "src/templates/assets/javascripts/browser/worker/index.ts", "src/templates/assets/javascripts/_/index.ts", "src/templates/assets/javascripts/components/_/index.ts", "src/templates/assets/javascripts/components/announce/index.ts", "src/templates/assets/javascripts/components/consent/index.ts", "src/templates/assets/javascripts/templates/tooltip/index.tsx", "src/templates/assets/javascripts/templates/annotation/index.tsx", "src/templates/assets/javascripts/templates/clipboard/index.tsx", "src/templates/assets/javascripts/templates/search/index.tsx", "src/templates/assets/javascripts/templates/source/index.tsx", "src/templates/assets/javascripts/templates/tabbed/index.tsx", "src/templates/assets/javascripts/templates/table/index.tsx", "src/templates/assets/javascripts/templates/version/index.tsx", "src/templates/assets/javascripts/components/tooltip2/index.ts", "src/templates/assets/javascripts/components/content/annotation/_/index.ts", "src/templates/assets/javascripts/components/content/annotation/list/index.ts", "src/templates/assets/javascripts/components/content/annotation/block/index.ts", "src/templates/assets/javascripts/components/content/code/_/index.ts", "src/templates/assets/javascripts/components/content/details/index.ts", "src/templates/assets/javascripts/components/content/mermaid/index.css", "src/templates/assets/javascripts/components/content/mermaid/index.ts", "src/templates/assets/javascripts/components/content/table/index.ts", "src/templates/assets/javascripts/components/content/tabs/index.ts", "src/templates/assets/javascripts/components/content/_/index.ts", "src/templates/assets/javascripts/components/dialog/index.ts", "src/templates/assets/javascripts/components/tooltip/index.ts", "src/templates/assets/javascripts/components/header/_/index.ts", "src/templates/assets/javascripts/components/header/title/index.ts", "src/templates/assets/javascripts/components/main/index.ts", "src/templates/assets/javascripts/components/palette/index.ts", "src/templates/assets/javascripts/components/progress/index.ts", "src/templates/assets/javascripts/integrations/clipboard/index.ts", "src/templates/assets/javascripts/integrations/sitemap/index.ts", "src/templates/assets/javascripts/integrations/instant/index.ts", "src/templates/assets/javascripts/integrations/search/highlighter/index.ts", "src/templates/assets/javascripts/integrations/search/worker/message/index.ts", "src/templates/assets/javascripts/integrations/search/worker/_/index.ts", "src/templates/assets/javascripts/integrations/version/findurl/index.ts", "src/templates/assets/javascripts/integrations/version/index.ts", "src/templates/assets/javascripts/components/search/query/index.ts", "src/templates/assets/javascripts/components/search/result/index.ts", "src/templates/assets/javascripts/components/search/share/index.ts", "src/templates/assets/javascripts/components/search/suggest/index.ts", "src/templates/assets/javascripts/components/search/_/index.ts", "src/templates/assets/javascripts/components/search/highlight/index.ts", "src/templates/assets/javascripts/components/sidebar/index.ts", "src/templates/assets/javascripts/components/source/facts/github/index.ts", "src/templates/assets/javascripts/components/source/facts/gitlab/index.ts", "src/templates/assets/javascripts/components/source/facts/_/index.ts", "src/templates/assets/javascripts/components/source/_/index.ts", "src/templates/assets/javascripts/components/tabs/index.ts", "src/templates/assets/javascripts/components/toc/index.ts", "src/templates/assets/javascripts/components/top/index.ts", "src/templates/assets/javascripts/patches/ellipsis/index.ts", "src/templates/assets/javascripts/patches/indeterminate/index.ts", "src/templates/assets/javascripts/patches/scrollfix/index.ts", "src/templates/assets/javascripts/patches/scrolllock/index.ts", "src/templates/assets/javascripts/polyfills/index.ts"], + "sourcesContent": ["(function (global, factory) {\n typeof exports === 'object' && typeof module !== 'undefined' ? factory() :\n typeof define === 'function' && define.amd ? define(factory) :\n (factory());\n}(this, (function () { 'use strict';\n\n /**\n * Applies the :focus-visible polyfill at the given scope.\n * A scope in this case is either the top-level Document or a Shadow Root.\n *\n * @param {(Document|ShadowRoot)} scope\n * @see https://github.com/WICG/focus-visible\n */\n function applyFocusVisiblePolyfill(scope) {\n var hadKeyboardEvent = true;\n var hadFocusVisibleRecently = false;\n var hadFocusVisibleRecentlyTimeout = null;\n\n var inputTypesAllowlist = {\n text: true,\n search: true,\n url: true,\n tel: true,\n email: true,\n password: true,\n number: true,\n date: true,\n month: true,\n week: true,\n time: true,\n datetime: true,\n 'datetime-local': true\n };\n\n /**\n * Helper function for legacy browsers and iframes which sometimes focus\n * elements like document, body, and non-interactive SVG.\n * @param {Element} el\n */\n function isValidFocusTarget(el) {\n if (\n el &&\n el !== document &&\n el.nodeName !== 'HTML' &&\n el.nodeName !== 'BODY' &&\n 'classList' in el &&\n 'contains' in el.classList\n ) {\n return true;\n }\n return false;\n }\n\n /**\n * Computes whether the given element should automatically trigger the\n * `focus-visible` class being added, i.e. whether it should always match\n * `:focus-visible` when focused.\n * @param {Element} el\n * @return {boolean}\n */\n function focusTriggersKeyboardModality(el) {\n var type = el.type;\n var tagName = el.tagName;\n\n if (tagName === 'INPUT' && inputTypesAllowlist[type] && !el.readOnly) {\n return true;\n }\n\n if (tagName === 'TEXTAREA' && !el.readOnly) {\n return true;\n }\n\n if (el.isContentEditable) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Add the `focus-visible` class to the given element if it was not added by\n * the author.\n * @param {Element} el\n */\n function addFocusVisibleClass(el) {\n if (el.classList.contains('focus-visible')) {\n return;\n }\n el.classList.add('focus-visible');\n el.setAttribute('data-focus-visible-added', '');\n }\n\n /**\n * Remove the `focus-visible` class from the given element if it was not\n * originally added by the author.\n * @param {Element} el\n */\n function removeFocusVisibleClass(el) {\n if (!el.hasAttribute('data-focus-visible-added')) {\n return;\n }\n el.classList.remove('focus-visible');\n el.removeAttribute('data-focus-visible-added');\n }\n\n /**\n * If the most recent user interaction was via the keyboard;\n * and the key press did not include a meta, alt/option, or control key;\n * then the modality is keyboard. Otherwise, the modality is not keyboard.\n * Apply `focus-visible` to any current active element and keep track\n * of our keyboard modality state with `hadKeyboardEvent`.\n * @param {KeyboardEvent} e\n */\n function onKeyDown(e) {\n if (e.metaKey || e.altKey || e.ctrlKey) {\n return;\n }\n\n if (isValidFocusTarget(scope.activeElement)) {\n addFocusVisibleClass(scope.activeElement);\n }\n\n hadKeyboardEvent = true;\n }\n\n /**\n * If at any point a user clicks with a pointing device, ensure that we change\n * the modality away from keyboard.\n * This avoids the situation where a user presses a key on an already focused\n * element, and then clicks on a different element, focusing it with a\n * pointing device, while we still think we're in keyboard modality.\n * @param {Event} e\n */\n function onPointerDown(e) {\n hadKeyboardEvent = false;\n }\n\n /**\n * On `focus`, add the `focus-visible` class to the target if:\n * - the target received focus as a result of keyboard navigation, or\n * - the event target is an element that will likely require interaction\n * via the keyboard (e.g. a text box)\n * @param {Event} e\n */\n function onFocus(e) {\n // Prevent IE from focusing the document or HTML element.\n if (!isValidFocusTarget(e.target)) {\n return;\n }\n\n if (hadKeyboardEvent || focusTriggersKeyboardModality(e.target)) {\n addFocusVisibleClass(e.target);\n }\n }\n\n /**\n * On `blur`, remove the `focus-visible` class from the target.\n * @param {Event} e\n */\n function onBlur(e) {\n if (!isValidFocusTarget(e.target)) {\n return;\n }\n\n if (\n e.target.classList.contains('focus-visible') ||\n e.target.hasAttribute('data-focus-visible-added')\n ) {\n // To detect a tab/window switch, we look for a blur event followed\n // rapidly by a visibility change.\n // If we don't see a visibility change within 100ms, it's probably a\n // regular focus change.\n hadFocusVisibleRecently = true;\n window.clearTimeout(hadFocusVisibleRecentlyTimeout);\n hadFocusVisibleRecentlyTimeout = window.setTimeout(function() {\n hadFocusVisibleRecently = false;\n }, 100);\n removeFocusVisibleClass(e.target);\n }\n }\n\n /**\n * If the user changes tabs, keep track of whether or not the previously\n * focused element had .focus-visible.\n * @param {Event} e\n */\n function onVisibilityChange(e) {\n if (document.visibilityState === 'hidden') {\n // If the tab becomes active again, the browser will handle calling focus\n // on the element (Safari actually calls it twice).\n // If this tab change caused a blur on an element with focus-visible,\n // re-apply the class when the user switches back to the tab.\n if (hadFocusVisibleRecently) {\n hadKeyboardEvent = true;\n }\n addInitialPointerMoveListeners();\n }\n }\n\n /**\n * Add a group of listeners to detect usage of any pointing devices.\n * These listeners will be added when the polyfill first loads, and anytime\n * the window is blurred, so that they are active when the window regains\n * focus.\n */\n function addInitialPointerMoveListeners() {\n document.addEventListener('mousemove', onInitialPointerMove);\n document.addEventListener('mousedown', onInitialPointerMove);\n document.addEventListener('mouseup', onInitialPointerMove);\n document.addEventListener('pointermove', onInitialPointerMove);\n document.addEventListener('pointerdown', onInitialPointerMove);\n document.addEventListener('pointerup', onInitialPointerMove);\n document.addEventListener('touchmove', onInitialPointerMove);\n document.addEventListener('touchstart', onInitialPointerMove);\n document.addEventListener('touchend', onInitialPointerMove);\n }\n\n function removeInitialPointerMoveListeners() {\n document.removeEventListener('mousemove', onInitialPointerMove);\n document.removeEventListener('mousedown', onInitialPointerMove);\n document.removeEventListener('mouseup', onInitialPointerMove);\n document.removeEventListener('pointermove', onInitialPointerMove);\n document.removeEventListener('pointerdown', onInitialPointerMove);\n document.removeEventListener('pointerup', onInitialPointerMove);\n document.removeEventListener('touchmove', onInitialPointerMove);\n document.removeEventListener('touchstart', onInitialPointerMove);\n document.removeEventListener('touchend', onInitialPointerMove);\n }\n\n /**\n * When the polfyill first loads, assume the user is in keyboard modality.\n * If any event is received from a pointing device (e.g. mouse, pointer,\n * touch), turn off keyboard modality.\n * This accounts for situations where focus enters the page from the URL bar.\n * @param {Event} e\n */\n function onInitialPointerMove(e) {\n // Work around a Safari quirk that fires a mousemove on whenever the\n // window blurs, even if you're tabbing out of the page. \u00AF\\_(\u30C4)_/\u00AF\n if (e.target.nodeName && e.target.nodeName.toLowerCase() === 'html') {\n return;\n }\n\n hadKeyboardEvent = false;\n removeInitialPointerMoveListeners();\n }\n\n // For some kinds of state, we are interested in changes at the global scope\n // only. For example, global pointer input, global key presses and global\n // visibility change should affect the state at every scope:\n document.addEventListener('keydown', onKeyDown, true);\n document.addEventListener('mousedown', onPointerDown, true);\n document.addEventListener('pointerdown', onPointerDown, true);\n document.addEventListener('touchstart', onPointerDown, true);\n document.addEventListener('visibilitychange', onVisibilityChange, true);\n\n addInitialPointerMoveListeners();\n\n // For focus and blur, we specifically care about state changes in the local\n // scope. This is because focus / blur events that originate from within a\n // shadow root are not re-dispatched from the host element if it was already\n // the active element in its own scope:\n scope.addEventListener('focus', onFocus, true);\n scope.addEventListener('blur', onBlur, true);\n\n // We detect that a node is a ShadowRoot by ensuring that it is a\n // DocumentFragment and also has a host property. This check covers native\n // implementation and polyfill implementation transparently. If we only cared\n // about the native implementation, we could just check if the scope was\n // an instance of a ShadowRoot.\n if (scope.nodeType === Node.DOCUMENT_FRAGMENT_NODE && scope.host) {\n // Since a ShadowRoot is a special kind of DocumentFragment, it does not\n // have a root element to add a class to. So, we add this attribute to the\n // host element instead:\n scope.host.setAttribute('data-js-focus-visible', '');\n } else if (scope.nodeType === Node.DOCUMENT_NODE) {\n document.documentElement.classList.add('js-focus-visible');\n document.documentElement.setAttribute('data-js-focus-visible', '');\n }\n }\n\n // It is important to wrap all references to global window and document in\n // these checks to support server-side rendering use cases\n // @see https://github.com/WICG/focus-visible/issues/199\n if (typeof window !== 'undefined' && typeof document !== 'undefined') {\n // Make the polyfill helper globally available. This can be used as a signal\n // to interested libraries that wish to coordinate with the polyfill for e.g.,\n // applying the polyfill to a shadow root:\n window.applyFocusVisiblePolyfill = applyFocusVisiblePolyfill;\n\n // Notify interested libraries of the polyfill's presence, in case the\n // polyfill was loaded lazily:\n var event;\n\n try {\n event = new CustomEvent('focus-visible-polyfill-ready');\n } catch (error) {\n // IE11 does not support using CustomEvent as a constructor directly:\n event = document.createEvent('CustomEvent');\n event.initCustomEvent('focus-visible-polyfill-ready', false, false, {});\n }\n\n window.dispatchEvent(event);\n }\n\n if (typeof document !== 'undefined') {\n // Apply the polyfill to the global document, so that no JavaScript\n // coordination is required to use the polyfill in the top-level document:\n applyFocusVisiblePolyfill(document);\n }\n\n})));\n", "/*!\n * escape-html\n * Copyright(c) 2012-2013 TJ Holowaychuk\n * Copyright(c) 2015 Andreas Lubbe\n * Copyright(c) 2015 Tiancheng \"Timothy\" Gu\n * MIT Licensed\n */\n\n'use strict';\n\n/**\n * Module variables.\n * @private\n */\n\nvar matchHtmlRegExp = /[\"'&<>]/;\n\n/**\n * Module exports.\n * @public\n */\n\nmodule.exports = escapeHtml;\n\n/**\n * Escape special characters in the given string of html.\n *\n * @param {string} string The string to escape for inserting into HTML\n * @return {string}\n * @public\n */\n\nfunction escapeHtml(string) {\n var str = '' + string;\n var match = matchHtmlRegExp.exec(str);\n\n if (!match) {\n return str;\n }\n\n var escape;\n var html = '';\n var index = 0;\n var lastIndex = 0;\n\n for (index = match.index; index < str.length; index++) {\n switch (str.charCodeAt(index)) {\n case 34: // \"\n escape = '"';\n break;\n case 38: // &\n escape = '&';\n break;\n case 39: // '\n escape = ''';\n break;\n case 60: // <\n escape = '<';\n break;\n case 62: // >\n escape = '>';\n break;\n default:\n continue;\n }\n\n if (lastIndex !== index) {\n html += str.substring(lastIndex, index);\n }\n\n lastIndex = index + 1;\n html += escape;\n }\n\n return lastIndex !== index\n ? html + str.substring(lastIndex, index)\n : html;\n}\n", "/*!\n * clipboard.js v2.0.11\n * https://clipboardjs.com/\n *\n * Licensed MIT \u00A9 Zeno Rocha\n */\n(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"ClipboardJS\"] = factory();\n\telse\n\t\troot[\"ClipboardJS\"] = factory();\n})(this, function() {\nreturn /******/ (function() { // webpackBootstrap\n/******/ \tvar __webpack_modules__ = ({\n\n/***/ 686:\n/***/ (function(__unused_webpack_module, __webpack_exports__, __webpack_require__) {\n\n\"use strict\";\n\n// EXPORTS\n__webpack_require__.d(__webpack_exports__, {\n \"default\": function() { return /* binding */ clipboard; }\n});\n\n// EXTERNAL MODULE: ./node_modules/tiny-emitter/index.js\nvar tiny_emitter = __webpack_require__(279);\nvar tiny_emitter_default = /*#__PURE__*/__webpack_require__.n(tiny_emitter);\n// EXTERNAL MODULE: ./node_modules/good-listener/src/listen.js\nvar listen = __webpack_require__(370);\nvar listen_default = /*#__PURE__*/__webpack_require__.n(listen);\n// EXTERNAL MODULE: ./node_modules/select/src/select.js\nvar src_select = __webpack_require__(817);\nvar select_default = /*#__PURE__*/__webpack_require__.n(src_select);\n;// CONCATENATED MODULE: ./src/common/command.js\n/**\n * Executes a given operation type.\n * @param {String} type\n * @return {Boolean}\n */\nfunction command(type) {\n try {\n return document.execCommand(type);\n } catch (err) {\n return false;\n }\n}\n;// CONCATENATED MODULE: ./src/actions/cut.js\n\n\n/**\n * Cut action wrapper.\n * @param {String|HTMLElement} target\n * @return {String}\n */\n\nvar ClipboardActionCut = function ClipboardActionCut(target) {\n var selectedText = select_default()(target);\n command('cut');\n return selectedText;\n};\n\n/* harmony default export */ var actions_cut = (ClipboardActionCut);\n;// CONCATENATED MODULE: ./src/common/create-fake-element.js\n/**\n * Creates a fake textarea element with a value.\n * @param {String} value\n * @return {HTMLElement}\n */\nfunction createFakeElement(value) {\n var isRTL = document.documentElement.getAttribute('dir') === 'rtl';\n var fakeElement = document.createElement('textarea'); // Prevent zooming on iOS\n\n fakeElement.style.fontSize = '12pt'; // Reset box model\n\n fakeElement.style.border = '0';\n fakeElement.style.padding = '0';\n fakeElement.style.margin = '0'; // Move element out of screen horizontally\n\n fakeElement.style.position = 'absolute';\n fakeElement.style[isRTL ? 'right' : 'left'] = '-9999px'; // Move element to the same position vertically\n\n var yPosition = window.pageYOffset || document.documentElement.scrollTop;\n fakeElement.style.top = \"\".concat(yPosition, \"px\");\n fakeElement.setAttribute('readonly', '');\n fakeElement.value = value;\n return fakeElement;\n}\n;// CONCATENATED MODULE: ./src/actions/copy.js\n\n\n\n/**\n * Create fake copy action wrapper using a fake element.\n * @param {String} target\n * @param {Object} options\n * @return {String}\n */\n\nvar fakeCopyAction = function fakeCopyAction(value, options) {\n var fakeElement = createFakeElement(value);\n options.container.appendChild(fakeElement);\n var selectedText = select_default()(fakeElement);\n command('copy');\n fakeElement.remove();\n return selectedText;\n};\n/**\n * Copy action wrapper.\n * @param {String|HTMLElement} target\n * @param {Object} options\n * @return {String}\n */\n\n\nvar ClipboardActionCopy = function ClipboardActionCopy(target) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {\n container: document.body\n };\n var selectedText = '';\n\n if (typeof target === 'string') {\n selectedText = fakeCopyAction(target, options);\n } else if (target instanceof HTMLInputElement && !['text', 'search', 'url', 'tel', 'password'].includes(target === null || target === void 0 ? void 0 : target.type)) {\n // If input type doesn't support `setSelectionRange`. Simulate it. https://developer.mozilla.org/en-US/docs/Web/API/HTMLInputElement/setSelectionRange\n selectedText = fakeCopyAction(target.value, options);\n } else {\n selectedText = select_default()(target);\n command('copy');\n }\n\n return selectedText;\n};\n\n/* harmony default export */ var actions_copy = (ClipboardActionCopy);\n;// CONCATENATED MODULE: ./src/actions/default.js\nfunction _typeof(obj) { \"@babel/helpers - typeof\"; if (typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; }; } return _typeof(obj); }\n\n\n\n/**\n * Inner function which performs selection from either `text` or `target`\n * properties and then executes copy or cut operations.\n * @param {Object} options\n */\n\nvar ClipboardActionDefault = function ClipboardActionDefault() {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n // Defines base properties passed from constructor.\n var _options$action = options.action,\n action = _options$action === void 0 ? 'copy' : _options$action,\n container = options.container,\n target = options.target,\n text = options.text; // Sets the `action` to be performed which can be either 'copy' or 'cut'.\n\n if (action !== 'copy' && action !== 'cut') {\n throw new Error('Invalid \"action\" value, use either \"copy\" or \"cut\"');\n } // Sets the `target` property using an element that will be have its content copied.\n\n\n if (target !== undefined) {\n if (target && _typeof(target) === 'object' && target.nodeType === 1) {\n if (action === 'copy' && target.hasAttribute('disabled')) {\n throw new Error('Invalid \"target\" attribute. Please use \"readonly\" instead of \"disabled\" attribute');\n }\n\n if (action === 'cut' && (target.hasAttribute('readonly') || target.hasAttribute('disabled'))) {\n throw new Error('Invalid \"target\" attribute. You can\\'t cut text from elements with \"readonly\" or \"disabled\" attributes');\n }\n } else {\n throw new Error('Invalid \"target\" value, use a valid Element');\n }\n } // Define selection strategy based on `text` property.\n\n\n if (text) {\n return actions_copy(text, {\n container: container\n });\n } // Defines which selection strategy based on `target` property.\n\n\n if (target) {\n return action === 'cut' ? actions_cut(target) : actions_copy(target, {\n container: container\n });\n }\n};\n\n/* harmony default export */ var actions_default = (ClipboardActionDefault);\n;// CONCATENATED MODULE: ./src/clipboard.js\nfunction clipboard_typeof(obj) { \"@babel/helpers - typeof\"; if (typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\") { clipboard_typeof = function _typeof(obj) { return typeof obj; }; } else { clipboard_typeof = function _typeof(obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; }; } return clipboard_typeof(obj); }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function\"); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, writable: true, configurable: true } }); if (superClass) _setPrototypeOf(subClass, superClass); }\n\nfunction _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); }\n\nfunction _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }\n\nfunction _possibleConstructorReturn(self, call) { if (call && (clipboard_typeof(call) === \"object\" || typeof call === \"function\")) { return call; } return _assertThisInitialized(self); }\n\nfunction _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return self; }\n\nfunction _isNativeReflectConstruct() { if (typeof Reflect === \"undefined\" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === \"function\") return true; try { Date.prototype.toString.call(Reflect.construct(Date, [], function () {})); return true; } catch (e) { return false; } }\n\nfunction _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); }\n\n\n\n\n\n\n/**\n * Helper function to retrieve attribute value.\n * @param {String} suffix\n * @param {Element} element\n */\n\nfunction getAttributeValue(suffix, element) {\n var attribute = \"data-clipboard-\".concat(suffix);\n\n if (!element.hasAttribute(attribute)) {\n return;\n }\n\n return element.getAttribute(attribute);\n}\n/**\n * Base class which takes one or more elements, adds event listeners to them,\n * and instantiates a new `ClipboardAction` on each click.\n */\n\n\nvar Clipboard = /*#__PURE__*/function (_Emitter) {\n _inherits(Clipboard, _Emitter);\n\n var _super = _createSuper(Clipboard);\n\n /**\n * @param {String|HTMLElement|HTMLCollection|NodeList} trigger\n * @param {Object} options\n */\n function Clipboard(trigger, options) {\n var _this;\n\n _classCallCheck(this, Clipboard);\n\n _this = _super.call(this);\n\n _this.resolveOptions(options);\n\n _this.listenClick(trigger);\n\n return _this;\n }\n /**\n * Defines if attributes would be resolved using internal setter functions\n * or custom functions that were passed in the constructor.\n * @param {Object} options\n */\n\n\n _createClass(Clipboard, [{\n key: \"resolveOptions\",\n value: function resolveOptions() {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n this.action = typeof options.action === 'function' ? options.action : this.defaultAction;\n this.target = typeof options.target === 'function' ? options.target : this.defaultTarget;\n this.text = typeof options.text === 'function' ? options.text : this.defaultText;\n this.container = clipboard_typeof(options.container) === 'object' ? options.container : document.body;\n }\n /**\n * Adds a click event listener to the passed trigger.\n * @param {String|HTMLElement|HTMLCollection|NodeList} trigger\n */\n\n }, {\n key: \"listenClick\",\n value: function listenClick(trigger) {\n var _this2 = this;\n\n this.listener = listen_default()(trigger, 'click', function (e) {\n return _this2.onClick(e);\n });\n }\n /**\n * Defines a new `ClipboardAction` on each click event.\n * @param {Event} e\n */\n\n }, {\n key: \"onClick\",\n value: function onClick(e) {\n var trigger = e.delegateTarget || e.currentTarget;\n var action = this.action(trigger) || 'copy';\n var text = actions_default({\n action: action,\n container: this.container,\n target: this.target(trigger),\n text: this.text(trigger)\n }); // Fires an event based on the copy operation result.\n\n this.emit(text ? 'success' : 'error', {\n action: action,\n text: text,\n trigger: trigger,\n clearSelection: function clearSelection() {\n if (trigger) {\n trigger.focus();\n }\n\n window.getSelection().removeAllRanges();\n }\n });\n }\n /**\n * Default `action` lookup function.\n * @param {Element} trigger\n */\n\n }, {\n key: \"defaultAction\",\n value: function defaultAction(trigger) {\n return getAttributeValue('action', trigger);\n }\n /**\n * Default `target` lookup function.\n * @param {Element} trigger\n */\n\n }, {\n key: \"defaultTarget\",\n value: function defaultTarget(trigger) {\n var selector = getAttributeValue('target', trigger);\n\n if (selector) {\n return document.querySelector(selector);\n }\n }\n /**\n * Allow fire programmatically a copy action\n * @param {String|HTMLElement} target\n * @param {Object} options\n * @returns Text copied.\n */\n\n }, {\n key: \"defaultText\",\n\n /**\n * Default `text` lookup function.\n * @param {Element} trigger\n */\n value: function defaultText(trigger) {\n return getAttributeValue('text', trigger);\n }\n /**\n * Destroy lifecycle.\n */\n\n }, {\n key: \"destroy\",\n value: function destroy() {\n this.listener.destroy();\n }\n }], [{\n key: \"copy\",\n value: function copy(target) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {\n container: document.body\n };\n return actions_copy(target, options);\n }\n /**\n * Allow fire programmatically a cut action\n * @param {String|HTMLElement} target\n * @returns Text cutted.\n */\n\n }, {\n key: \"cut\",\n value: function cut(target) {\n return actions_cut(target);\n }\n /**\n * Returns the support of the given action, or all actions if no action is\n * given.\n * @param {String} [action]\n */\n\n }, {\n key: \"isSupported\",\n value: function isSupported() {\n var action = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : ['copy', 'cut'];\n var actions = typeof action === 'string' ? [action] : action;\n var support = !!document.queryCommandSupported;\n actions.forEach(function (action) {\n support = support && !!document.queryCommandSupported(action);\n });\n return support;\n }\n }]);\n\n return Clipboard;\n}((tiny_emitter_default()));\n\n/* harmony default export */ var clipboard = (Clipboard);\n\n/***/ }),\n\n/***/ 828:\n/***/ (function(module) {\n\nvar DOCUMENT_NODE_TYPE = 9;\n\n/**\n * A polyfill for Element.matches()\n */\nif (typeof Element !== 'undefined' && !Element.prototype.matches) {\n var proto = Element.prototype;\n\n proto.matches = proto.matchesSelector ||\n proto.mozMatchesSelector ||\n proto.msMatchesSelector ||\n proto.oMatchesSelector ||\n proto.webkitMatchesSelector;\n}\n\n/**\n * Finds the closest parent that matches a selector.\n *\n * @param {Element} element\n * @param {String} selector\n * @return {Function}\n */\nfunction closest (element, selector) {\n while (element && element.nodeType !== DOCUMENT_NODE_TYPE) {\n if (typeof element.matches === 'function' &&\n element.matches(selector)) {\n return element;\n }\n element = element.parentNode;\n }\n}\n\nmodule.exports = closest;\n\n\n/***/ }),\n\n/***/ 438:\n/***/ (function(module, __unused_webpack_exports, __webpack_require__) {\n\nvar closest = __webpack_require__(828);\n\n/**\n * Delegates event to a selector.\n *\n * @param {Element} element\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @param {Boolean} useCapture\n * @return {Object}\n */\nfunction _delegate(element, selector, type, callback, useCapture) {\n var listenerFn = listener.apply(this, arguments);\n\n element.addEventListener(type, listenerFn, useCapture);\n\n return {\n destroy: function() {\n element.removeEventListener(type, listenerFn, useCapture);\n }\n }\n}\n\n/**\n * Delegates event to a selector.\n *\n * @param {Element|String|Array} [elements]\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @param {Boolean} useCapture\n * @return {Object}\n */\nfunction delegate(elements, selector, type, callback, useCapture) {\n // Handle the regular Element usage\n if (typeof elements.addEventListener === 'function') {\n return _delegate.apply(null, arguments);\n }\n\n // Handle Element-less usage, it defaults to global delegation\n if (typeof type === 'function') {\n // Use `document` as the first parameter, then apply arguments\n // This is a short way to .unshift `arguments` without running into deoptimizations\n return _delegate.bind(null, document).apply(null, arguments);\n }\n\n // Handle Selector-based usage\n if (typeof elements === 'string') {\n elements = document.querySelectorAll(elements);\n }\n\n // Handle Array-like based usage\n return Array.prototype.map.call(elements, function (element) {\n return _delegate(element, selector, type, callback, useCapture);\n });\n}\n\n/**\n * Finds closest match and invokes callback.\n *\n * @param {Element} element\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @return {Function}\n */\nfunction listener(element, selector, type, callback) {\n return function(e) {\n e.delegateTarget = closest(e.target, selector);\n\n if (e.delegateTarget) {\n callback.call(element, e);\n }\n }\n}\n\nmodule.exports = delegate;\n\n\n/***/ }),\n\n/***/ 879:\n/***/ (function(__unused_webpack_module, exports) {\n\n/**\n * Check if argument is a HTML element.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.node = function(value) {\n return value !== undefined\n && value instanceof HTMLElement\n && value.nodeType === 1;\n};\n\n/**\n * Check if argument is a list of HTML elements.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.nodeList = function(value) {\n var type = Object.prototype.toString.call(value);\n\n return value !== undefined\n && (type === '[object NodeList]' || type === '[object HTMLCollection]')\n && ('length' in value)\n && (value.length === 0 || exports.node(value[0]));\n};\n\n/**\n * Check if argument is a string.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.string = function(value) {\n return typeof value === 'string'\n || value instanceof String;\n};\n\n/**\n * Check if argument is a function.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.fn = function(value) {\n var type = Object.prototype.toString.call(value);\n\n return type === '[object Function]';\n};\n\n\n/***/ }),\n\n/***/ 370:\n/***/ (function(module, __unused_webpack_exports, __webpack_require__) {\n\nvar is = __webpack_require__(879);\nvar delegate = __webpack_require__(438);\n\n/**\n * Validates all params and calls the right\n * listener function based on its target type.\n *\n * @param {String|HTMLElement|HTMLCollection|NodeList} target\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listen(target, type, callback) {\n if (!target && !type && !callback) {\n throw new Error('Missing required arguments');\n }\n\n if (!is.string(type)) {\n throw new TypeError('Second argument must be a String');\n }\n\n if (!is.fn(callback)) {\n throw new TypeError('Third argument must be a Function');\n }\n\n if (is.node(target)) {\n return listenNode(target, type, callback);\n }\n else if (is.nodeList(target)) {\n return listenNodeList(target, type, callback);\n }\n else if (is.string(target)) {\n return listenSelector(target, type, callback);\n }\n else {\n throw new TypeError('First argument must be a String, HTMLElement, HTMLCollection, or NodeList');\n }\n}\n\n/**\n * Adds an event listener to a HTML element\n * and returns a remove listener function.\n *\n * @param {HTMLElement} node\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listenNode(node, type, callback) {\n node.addEventListener(type, callback);\n\n return {\n destroy: function() {\n node.removeEventListener(type, callback);\n }\n }\n}\n\n/**\n * Add an event listener to a list of HTML elements\n * and returns a remove listener function.\n *\n * @param {NodeList|HTMLCollection} nodeList\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listenNodeList(nodeList, type, callback) {\n Array.prototype.forEach.call(nodeList, function(node) {\n node.addEventListener(type, callback);\n });\n\n return {\n destroy: function() {\n Array.prototype.forEach.call(nodeList, function(node) {\n node.removeEventListener(type, callback);\n });\n }\n }\n}\n\n/**\n * Add an event listener to a selector\n * and returns a remove listener function.\n *\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listenSelector(selector, type, callback) {\n return delegate(document.body, selector, type, callback);\n}\n\nmodule.exports = listen;\n\n\n/***/ }),\n\n/***/ 817:\n/***/ (function(module) {\n\nfunction select(element) {\n var selectedText;\n\n if (element.nodeName === 'SELECT') {\n element.focus();\n\n selectedText = element.value;\n }\n else if (element.nodeName === 'INPUT' || element.nodeName === 'TEXTAREA') {\n var isReadOnly = element.hasAttribute('readonly');\n\n if (!isReadOnly) {\n element.setAttribute('readonly', '');\n }\n\n element.select();\n element.setSelectionRange(0, element.value.length);\n\n if (!isReadOnly) {\n element.removeAttribute('readonly');\n }\n\n selectedText = element.value;\n }\n else {\n if (element.hasAttribute('contenteditable')) {\n element.focus();\n }\n\n var selection = window.getSelection();\n var range = document.createRange();\n\n range.selectNodeContents(element);\n selection.removeAllRanges();\n selection.addRange(range);\n\n selectedText = selection.toString();\n }\n\n return selectedText;\n}\n\nmodule.exports = select;\n\n\n/***/ }),\n\n/***/ 279:\n/***/ (function(module) {\n\nfunction E () {\n // Keep this empty so it's easier to inherit from\n // (via https://github.com/lipsmack from https://github.com/scottcorgan/tiny-emitter/issues/3)\n}\n\nE.prototype = {\n on: function (name, callback, ctx) {\n var e = this.e || (this.e = {});\n\n (e[name] || (e[name] = [])).push({\n fn: callback,\n ctx: ctx\n });\n\n return this;\n },\n\n once: function (name, callback, ctx) {\n var self = this;\n function listener () {\n self.off(name, listener);\n callback.apply(ctx, arguments);\n };\n\n listener._ = callback\n return this.on(name, listener, ctx);\n },\n\n emit: function (name) {\n var data = [].slice.call(arguments, 1);\n var evtArr = ((this.e || (this.e = {}))[name] || []).slice();\n var i = 0;\n var len = evtArr.length;\n\n for (i; i < len; i++) {\n evtArr[i].fn.apply(evtArr[i].ctx, data);\n }\n\n return this;\n },\n\n off: function (name, callback) {\n var e = this.e || (this.e = {});\n var evts = e[name];\n var liveEvents = [];\n\n if (evts && callback) {\n for (var i = 0, len = evts.length; i < len; i++) {\n if (evts[i].fn !== callback && evts[i].fn._ !== callback)\n liveEvents.push(evts[i]);\n }\n }\n\n // Remove event from queue to prevent memory leak\n // Suggested by https://github.com/lazd\n // Ref: https://github.com/scottcorgan/tiny-emitter/commit/c6ebfaa9bc973b33d110a84a307742b7cf94c953#commitcomment-5024910\n\n (liveEvents.length)\n ? e[name] = liveEvents\n : delete e[name];\n\n return this;\n }\n};\n\nmodule.exports = E;\nmodule.exports.TinyEmitter = E;\n\n\n/***/ })\n\n/******/ \t});\n/************************************************************************/\n/******/ \t// The module cache\n/******/ \tvar __webpack_module_cache__ = {};\n/******/ \t\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(__webpack_module_cache__[moduleId]) {\n/******/ \t\t\treturn __webpack_module_cache__[moduleId].exports;\n/******/ \t\t}\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = __webpack_module_cache__[moduleId] = {\n/******/ \t\t\t// no module.id needed\n/******/ \t\t\t// no module.loaded needed\n/******/ \t\t\texports: {}\n/******/ \t\t};\n/******/ \t\n/******/ \t\t// Execute the module function\n/******/ \t\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n/******/ \t\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/ \t\n/************************************************************************/\n/******/ \t/* webpack/runtime/compat get default export */\n/******/ \t!function() {\n/******/ \t\t// getDefaultExport function for compatibility with non-harmony modules\n/******/ \t\t__webpack_require__.n = function(module) {\n/******/ \t\t\tvar getter = module && module.__esModule ?\n/******/ \t\t\t\tfunction() { return module['default']; } :\n/******/ \t\t\t\tfunction() { return module; };\n/******/ \t\t\t__webpack_require__.d(getter, { a: getter });\n/******/ \t\t\treturn getter;\n/******/ \t\t};\n/******/ \t}();\n/******/ \t\n/******/ \t/* webpack/runtime/define property getters */\n/******/ \t!function() {\n/******/ \t\t// define getter functions for harmony exports\n/******/ \t\t__webpack_require__.d = function(exports, definition) {\n/******/ \t\t\tfor(var key in definition) {\n/******/ \t\t\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n/******/ \t\t\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n/******/ \t\t\t\t}\n/******/ \t\t\t}\n/******/ \t\t};\n/******/ \t}();\n/******/ \t\n/******/ \t/* webpack/runtime/hasOwnProperty shorthand */\n/******/ \t!function() {\n/******/ \t\t__webpack_require__.o = function(obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }\n/******/ \t}();\n/******/ \t\n/************************************************************************/\n/******/ \t// module exports must be returned from runtime so entry inlining is disabled\n/******/ \t// startup\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(686);\n/******/ })()\n.default;\n});", "/*\n * Copyright (c) 2016-2024 Martin Donath \n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport \"focus-visible\"\n\nimport {\n EMPTY,\n NEVER,\n Observable,\n Subject,\n defer,\n delay,\n filter,\n map,\n merge,\n mergeWith,\n shareReplay,\n switchMap\n} from \"rxjs\"\n\nimport { configuration, feature } from \"./_\"\nimport {\n at,\n getActiveElement,\n getOptionalElement,\n requestJSON,\n setLocation,\n setToggle,\n watchDocument,\n watchKeyboard,\n watchLocation,\n watchLocationTarget,\n watchMedia,\n watchPrint,\n watchScript,\n watchViewport\n} from \"./browser\"\nimport {\n getComponentElement,\n getComponentElements,\n mountAnnounce,\n mountBackToTop,\n mountConsent,\n mountContent,\n mountDialog,\n mountHeader,\n mountHeaderTitle,\n mountPalette,\n mountProgress,\n mountSearch,\n mountSearchHiglight,\n mountSidebar,\n mountSource,\n mountTableOfContents,\n mountTabs,\n watchHeader,\n watchMain\n} from \"./components\"\nimport {\n SearchIndex,\n setupClipboardJS,\n setupInstantNavigation,\n setupVersionSelector\n} from \"./integrations\"\nimport {\n patchEllipsis,\n patchIndeterminate,\n patchScrollfix,\n patchScrolllock\n} from \"./patches\"\nimport \"./polyfills\"\n\n/* ----------------------------------------------------------------------------\n * Functions - @todo refactor\n * ------------------------------------------------------------------------- */\n\n/**\n * Fetch search index\n *\n * @returns Search index observable\n */\nfunction fetchSearchIndex(): Observable {\n if (location.protocol === \"file:\") {\n return watchScript(\n `${new URL(\"search/search_index.js\", config.base)}`\n )\n .pipe(\n // @ts-ignore - @todo fix typings\n map(() => __index),\n shareReplay(1)\n )\n } else {\n return requestJSON(\n new URL(\"search/search_index.json\", config.base)\n )\n }\n}\n\n/* ----------------------------------------------------------------------------\n * Application\n * ------------------------------------------------------------------------- */\n\n/* Yay, JavaScript is available */\ndocument.documentElement.classList.remove(\"no-js\")\ndocument.documentElement.classList.add(\"js\")\n\n/* Set up navigation observables and subjects */\nconst document$ = watchDocument()\nconst location$ = watchLocation()\nconst target$ = watchLocationTarget(location$)\nconst keyboard$ = watchKeyboard()\n\n/* Set up media observables */\nconst viewport$ = watchViewport()\nconst tablet$ = watchMedia(\"(min-width: 960px)\")\nconst screen$ = watchMedia(\"(min-width: 1220px)\")\nconst print$ = watchPrint()\n\n/* Retrieve search index, if search is enabled */\nconst config = configuration()\nconst index$ = document.forms.namedItem(\"search\")\n ? fetchSearchIndex()\n : NEVER\n\n/* Set up Clipboard.js integration */\nconst alert$ = new Subject()\nsetupClipboardJS({ alert$ })\n\n/* Set up progress indicator */\nconst progress$ = new Subject()\n\n/* Set up instant navigation, if enabled */\nif (feature(\"navigation.instant\"))\n setupInstantNavigation({ location$, viewport$, progress$ })\n .subscribe(document$)\n\n/* Set up version selector */\nif (config.version?.provider === \"mike\")\n setupVersionSelector({ document$ })\n\n/* Always close drawer and search on navigation */\nmerge(location$, target$)\n .pipe(\n delay(125)\n )\n .subscribe(() => {\n setToggle(\"drawer\", false)\n setToggle(\"search\", false)\n })\n\n/* Set up global keyboard handlers */\nkeyboard$\n .pipe(\n filter(({ mode }) => mode === \"global\")\n )\n .subscribe(key => {\n switch (key.type) {\n\n /* Go to previous page */\n case \"p\":\n case \",\":\n const prev = getOptionalElement(\"link[rel=prev]\")\n if (typeof prev !== \"undefined\")\n setLocation(prev)\n break\n\n /* Go to next page */\n case \"n\":\n case \".\":\n const next = getOptionalElement(\"link[rel=next]\")\n if (typeof next !== \"undefined\")\n setLocation(next)\n break\n\n /* Expand navigation, see https://bit.ly/3ZjG5io */\n case \"Enter\":\n const active = getActiveElement()\n if (active instanceof HTMLLabelElement)\n active.click()\n }\n })\n\n/* Set up patches */\npatchEllipsis({ viewport$, document$ })\npatchIndeterminate({ document$, tablet$ })\npatchScrollfix({ document$ })\npatchScrolllock({ viewport$, tablet$ })\n\n/* Set up header and main area observable */\nconst header$ = watchHeader(getComponentElement(\"header\"), { viewport$ })\nconst main$ = document$\n .pipe(\n map(() => getComponentElement(\"main\")),\n switchMap(el => watchMain(el, { viewport$, header$ })),\n shareReplay(1)\n )\n\n/* Set up control component observables */\nconst control$ = merge(\n\n /* Consent */\n ...getComponentElements(\"consent\")\n .map(el => mountConsent(el, { target$ })),\n\n /* Dialog */\n ...getComponentElements(\"dialog\")\n .map(el => mountDialog(el, { alert$ })),\n\n /* Color palette */\n ...getComponentElements(\"palette\")\n .map(el => mountPalette(el)),\n\n /* Progress bar */\n ...getComponentElements(\"progress\")\n .map(el => mountProgress(el, { progress$ })),\n\n /* Search */\n ...getComponentElements(\"search\")\n .map(el => mountSearch(el, { index$, keyboard$ })),\n\n /* Repository information */\n ...getComponentElements(\"source\")\n .map(el => mountSource(el))\n)\n\n/* Set up content component observables */\nconst content$ = defer(() => merge(\n\n /* Announcement bar */\n ...getComponentElements(\"announce\")\n .map(el => mountAnnounce(el)),\n\n /* Content */\n ...getComponentElements(\"content\")\n .map(el => mountContent(el, { viewport$, target$, print$ })),\n\n /* Search highlighting */\n ...getComponentElements(\"content\")\n .map(el => feature(\"search.highlight\")\n ? mountSearchHiglight(el, { index$, location$ })\n : EMPTY\n ),\n\n /* Header */\n ...getComponentElements(\"header\")\n .map(el => mountHeader(el, { viewport$, header$, main$ })),\n\n /* Header title */\n ...getComponentElements(\"header-title\")\n .map(el => mountHeaderTitle(el, { viewport$, header$ })),\n\n /* Sidebar */\n ...getComponentElements(\"sidebar\")\n .map(el => el.getAttribute(\"data-md-type\") === \"navigation\"\n ? at(screen$, () => mountSidebar(el, { viewport$, header$, main$ }))\n : at(tablet$, () => mountSidebar(el, { viewport$, header$, main$ }))\n ),\n\n /* Navigation tabs */\n ...getComponentElements(\"tabs\")\n .map(el => mountTabs(el, { viewport$, header$ })),\n\n /* Table of contents */\n ...getComponentElements(\"toc\")\n .map(el => mountTableOfContents(el, {\n viewport$, header$, main$, target$\n })),\n\n /* Back-to-top button */\n ...getComponentElements(\"top\")\n .map(el => mountBackToTop(el, { viewport$, header$, main$, target$ }))\n))\n\n/* Set up component observables */\nconst component$ = document$\n .pipe(\n switchMap(() => content$),\n mergeWith(control$),\n shareReplay(1)\n )\n\n/* Subscribe to all components */\ncomponent$.subscribe()\n\n/* ----------------------------------------------------------------------------\n * Exports\n * ------------------------------------------------------------------------- */\n\nwindow.document$ = document$ /* Document observable */\nwindow.location$ = location$ /* Location subject */\nwindow.target$ = target$ /* Location target observable */\nwindow.keyboard$ = keyboard$ /* Keyboard observable */\nwindow.viewport$ = viewport$ /* Viewport observable */\nwindow.tablet$ = tablet$ /* Media tablet observable */\nwindow.screen$ = screen$ /* Media screen observable */\nwindow.print$ = print$ /* Media print observable */\nwindow.alert$ = alert$ /* Alert subject */\nwindow.progress$ = progress$ /* Progress indicator subject */\nwindow.component$ = component$ /* Component observable */\n", "/******************************************************************************\nCopyright (c) Microsoft Corporation.\n\nPermission to use, copy, modify, and/or distribute this software for any\npurpose with or without fee is hereby granted.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\nPERFORMANCE OF THIS SOFTWARE.\n***************************************************************************** */\n/* global Reflect, Promise, SuppressedError, Symbol, Iterator */\n\nvar extendStatics = function(d, b) {\n extendStatics = Object.setPrototypeOf ||\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\n return extendStatics(d, b);\n};\n\nexport function __extends(d, b) {\n if (typeof b !== \"function\" && b !== null)\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\n extendStatics(d, b);\n function __() { this.constructor = d; }\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\n}\n\nexport var __assign = function() {\n __assign = Object.assign || function __assign(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\n }\n return t;\n }\n return __assign.apply(this, arguments);\n}\n\nexport function __rest(s, e) {\n var t = {};\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\n t[p] = s[p];\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\n t[p[i]] = s[p[i]];\n }\n return t;\n}\n\nexport function __decorate(decorators, target, key, desc) {\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\n return c > 3 && r && Object.defineProperty(target, key, r), r;\n}\n\nexport function __param(paramIndex, decorator) {\n return function (target, key) { decorator(target, key, paramIndex); }\n}\n\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\n var _, done = false;\n for (var i = decorators.length - 1; i >= 0; i--) {\n var context = {};\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\n if (kind === \"accessor\") {\n if (result === void 0) continue;\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\n if (_ = accept(result.get)) descriptor.get = _;\n if (_ = accept(result.set)) descriptor.set = _;\n if (_ = accept(result.init)) initializers.unshift(_);\n }\n else if (_ = accept(result)) {\n if (kind === \"field\") initializers.unshift(_);\n else descriptor[key] = _;\n }\n }\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\n done = true;\n};\n\nexport function __runInitializers(thisArg, initializers, value) {\n var useValue = arguments.length > 2;\n for (var i = 0; i < initializers.length; i++) {\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\n }\n return useValue ? value : void 0;\n};\n\nexport function __propKey(x) {\n return typeof x === \"symbol\" ? x : \"\".concat(x);\n};\n\nexport function __setFunctionName(f, name, prefix) {\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\n};\n\nexport function __metadata(metadataKey, metadataValue) {\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\n}\n\nexport function __awaiter(thisArg, _arguments, P, generator) {\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n}\n\nexport function __generator(thisArg, body) {\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === \"function\" ? Iterator : Object).prototype);\n return g.next = verb(0), g[\"throw\"] = verb(1), g[\"return\"] = verb(2), typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n function verb(n) { return function (v) { return step([n, v]); }; }\n function step(op) {\n if (f) throw new TypeError(\"Generator is already executing.\");\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n if (y = 0, t) op = [op[0] & 2, t.value];\n switch (op[0]) {\n case 0: case 1: t = op; break;\n case 4: _.label++; return { value: op[1], done: false };\n case 5: _.label++; y = op[1]; op = [0]; continue;\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\n default:\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n if (t[2]) _.ops.pop();\n _.trys.pop(); continue;\n }\n op = body.call(thisArg, _);\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n }\n}\n\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\n if (k2 === undefined) k2 = k;\n var desc = Object.getOwnPropertyDescriptor(m, k);\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\n desc = { enumerable: true, get: function() { return m[k]; } };\n }\n Object.defineProperty(o, k2, desc);\n}) : (function(o, m, k, k2) {\n if (k2 === undefined) k2 = k;\n o[k2] = m[k];\n});\n\nexport function __exportStar(m, o) {\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\n}\n\nexport function __values(o) {\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\n if (m) return m.call(o);\n if (o && typeof o.length === \"number\") return {\n next: function () {\n if (o && i >= o.length) o = void 0;\n return { value: o && o[i++], done: !o };\n }\n };\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\n}\n\nexport function __read(o, n) {\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\n if (!m) return o;\n var i = m.call(o), r, ar = [], e;\n try {\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\n }\n catch (error) { e = { error: error }; }\n finally {\n try {\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\n }\n finally { if (e) throw e.error; }\n }\n return ar;\n}\n\n/** @deprecated */\nexport function __spread() {\n for (var ar = [], i = 0; i < arguments.length; i++)\n ar = ar.concat(__read(arguments[i]));\n return ar;\n}\n\n/** @deprecated */\nexport function __spreadArrays() {\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\n r[k] = a[j];\n return r;\n}\n\nexport function __spreadArray(to, from, pack) {\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\n if (ar || !(i in from)) {\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\n ar[i] = from[i];\n }\n }\n return to.concat(ar || Array.prototype.slice.call(from));\n}\n\nexport function __await(v) {\n return this instanceof __await ? (this.v = v, this) : new __await(v);\n}\n\nexport function __asyncGenerator(thisArg, _arguments, generator) {\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\n return i = Object.create((typeof AsyncIterator === \"function\" ? AsyncIterator : Object).prototype), verb(\"next\"), verb(\"throw\"), verb(\"return\", awaitReturn), i[Symbol.asyncIterator] = function () { return this; }, i;\n function awaitReturn(f) { return function (v) { return Promise.resolve(v).then(f, reject); }; }\n function verb(n, f) { if (g[n]) { i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; if (f) i[n] = f(i[n]); } }\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\n function fulfill(value) { resume(\"next\", value); }\n function reject(value) { resume(\"throw\", value); }\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\n}\n\nexport function __asyncDelegator(o) {\n var i, p;\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\n}\n\nexport function __asyncValues(o) {\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n var m = o[Symbol.asyncIterator], i;\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\n}\n\nexport function __makeTemplateObject(cooked, raw) {\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\n return cooked;\n};\n\nvar __setModuleDefault = Object.create ? (function(o, v) {\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\n}) : function(o, v) {\n o[\"default\"] = v;\n};\n\nexport function __importStar(mod) {\n if (mod && mod.__esModule) return mod;\n var result = {};\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\n __setModuleDefault(result, mod);\n return result;\n}\n\nexport function __importDefault(mod) {\n return (mod && mod.__esModule) ? mod : { default: mod };\n}\n\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\n}\n\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\n}\n\nexport function __classPrivateFieldIn(state, receiver) {\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\n}\n\nexport function __addDisposableResource(env, value, async) {\n if (value !== null && value !== void 0) {\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\n var dispose, inner;\n if (async) {\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\n dispose = value[Symbol.asyncDispose];\n }\n if (dispose === void 0) {\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\n dispose = value[Symbol.dispose];\n if (async) inner = dispose;\n }\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\n if (inner) dispose = function() { try { inner.call(this); } catch (e) { return Promise.reject(e); } };\n env.stack.push({ value: value, dispose: dispose, async: async });\n }\n else if (async) {\n env.stack.push({ async: true });\n }\n return value;\n}\n\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\n var e = new Error(message);\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\n};\n\nexport function __disposeResources(env) {\n function fail(e) {\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\n env.hasError = true;\n }\n var r, s = 0;\n function next() {\n while (r = env.stack.pop()) {\n try {\n if (!r.async && s === 1) return s = 0, env.stack.push(r), Promise.resolve().then(next);\n if (r.dispose) {\n var result = r.dispose.call(r.value);\n if (r.async) return s |= 2, Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\n }\n else s |= 1;\n }\n catch (e) {\n fail(e);\n }\n }\n if (s === 1) return env.hasError ? Promise.reject(env.error) : Promise.resolve();\n if (env.hasError) throw env.error;\n }\n return next();\n}\n\nexport default {\n __extends,\n __assign,\n __rest,\n __decorate,\n __param,\n __metadata,\n __awaiter,\n __generator,\n __createBinding,\n __exportStar,\n __values,\n __read,\n __spread,\n __spreadArrays,\n __spreadArray,\n __await,\n __asyncGenerator,\n __asyncDelegator,\n __asyncValues,\n __makeTemplateObject,\n __importStar,\n __importDefault,\n __classPrivateFieldGet,\n __classPrivateFieldSet,\n __classPrivateFieldIn,\n __addDisposableResource,\n __disposeResources,\n};\n", "/**\n * Returns true if the object is a function.\n * @param value The value to check\n */\nexport function isFunction(value: any): value is (...args: any[]) => any {\n return typeof value === 'function';\n}\n", "/**\n * Used to create Error subclasses until the community moves away from ES5.\n *\n * This is because compiling from TypeScript down to ES5 has issues with subclassing Errors\n * as well as other built-in types: https://github.com/Microsoft/TypeScript/issues/12123\n *\n * @param createImpl A factory function to create the actual constructor implementation. The returned\n * function should be a named function that calls `_super` internally.\n */\nexport function createErrorClass(createImpl: (_super: any) => any): T {\n const _super = (instance: any) => {\n Error.call(instance);\n instance.stack = new Error().stack;\n };\n\n const ctorFunc = createImpl(_super);\n ctorFunc.prototype = Object.create(Error.prototype);\n ctorFunc.prototype.constructor = ctorFunc;\n return ctorFunc;\n}\n", "import { createErrorClass } from './createErrorClass';\n\nexport interface UnsubscriptionError extends Error {\n readonly errors: any[];\n}\n\nexport interface UnsubscriptionErrorCtor {\n /**\n * @deprecated Internal implementation detail. Do not construct error instances.\n * Cannot be tagged as internal: https://github.com/ReactiveX/rxjs/issues/6269\n */\n new (errors: any[]): UnsubscriptionError;\n}\n\n/**\n * An error thrown when one or more errors have occurred during the\n * `unsubscribe` of a {@link Subscription}.\n */\nexport const UnsubscriptionError: UnsubscriptionErrorCtor = createErrorClass(\n (_super) =>\n function UnsubscriptionErrorImpl(this: any, errors: (Error | string)[]) {\n _super(this);\n this.message = errors\n ? `${errors.length} errors occurred during unsubscription:\n${errors.map((err, i) => `${i + 1}) ${err.toString()}`).join('\\n ')}`\n : '';\n this.name = 'UnsubscriptionError';\n this.errors = errors;\n }\n);\n", "/**\n * Removes an item from an array, mutating it.\n * @param arr The array to remove the item from\n * @param item The item to remove\n */\nexport function arrRemove(arr: T[] | undefined | null, item: T) {\n if (arr) {\n const index = arr.indexOf(item);\n 0 <= index && arr.splice(index, 1);\n }\n}\n", "import { isFunction } from './util/isFunction';\nimport { UnsubscriptionError } from './util/UnsubscriptionError';\nimport { SubscriptionLike, TeardownLogic, Unsubscribable } from './types';\nimport { arrRemove } from './util/arrRemove';\n\n/**\n * Represents a disposable resource, such as the execution of an Observable. A\n * Subscription has one important method, `unsubscribe`, that takes no argument\n * and just disposes the resource held by the subscription.\n *\n * Additionally, subscriptions may be grouped together through the `add()`\n * method, which will attach a child Subscription to the current Subscription.\n * When a Subscription is unsubscribed, all its children (and its grandchildren)\n * will be unsubscribed as well.\n *\n * @class Subscription\n */\nexport class Subscription implements SubscriptionLike {\n /** @nocollapse */\n public static EMPTY = (() => {\n const empty = new Subscription();\n empty.closed = true;\n return empty;\n })();\n\n /**\n * A flag to indicate whether this Subscription has already been unsubscribed.\n */\n public closed = false;\n\n private _parentage: Subscription[] | Subscription | null = null;\n\n /**\n * The list of registered finalizers to execute upon unsubscription. Adding and removing from this\n * list occurs in the {@link #add} and {@link #remove} methods.\n */\n private _finalizers: Exclude[] | null = null;\n\n /**\n * @param initialTeardown A function executed first as part of the finalization\n * process that is kicked off when {@link #unsubscribe} is called.\n */\n constructor(private initialTeardown?: () => void) {}\n\n /**\n * Disposes the resources held by the subscription. May, for instance, cancel\n * an ongoing Observable execution or cancel any other type of work that\n * started when the Subscription was created.\n * @return {void}\n */\n unsubscribe(): void {\n let errors: any[] | undefined;\n\n if (!this.closed) {\n this.closed = true;\n\n // Remove this from it's parents.\n const { _parentage } = this;\n if (_parentage) {\n this._parentage = null;\n if (Array.isArray(_parentage)) {\n for (const parent of _parentage) {\n parent.remove(this);\n }\n } else {\n _parentage.remove(this);\n }\n }\n\n const { initialTeardown: initialFinalizer } = this;\n if (isFunction(initialFinalizer)) {\n try {\n initialFinalizer();\n } catch (e) {\n errors = e instanceof UnsubscriptionError ? e.errors : [e];\n }\n }\n\n const { _finalizers } = this;\n if (_finalizers) {\n this._finalizers = null;\n for (const finalizer of _finalizers) {\n try {\n execFinalizer(finalizer);\n } catch (err) {\n errors = errors ?? [];\n if (err instanceof UnsubscriptionError) {\n errors = [...errors, ...err.errors];\n } else {\n errors.push(err);\n }\n }\n }\n }\n\n if (errors) {\n throw new UnsubscriptionError(errors);\n }\n }\n }\n\n /**\n * Adds a finalizer to this subscription, so that finalization will be unsubscribed/called\n * when this subscription is unsubscribed. If this subscription is already {@link #closed},\n * because it has already been unsubscribed, then whatever finalizer is passed to it\n * will automatically be executed (unless the finalizer itself is also a closed subscription).\n *\n * Closed Subscriptions cannot be added as finalizers to any subscription. Adding a closed\n * subscription to a any subscription will result in no operation. (A noop).\n *\n * Adding a subscription to itself, or adding `null` or `undefined` will not perform any\n * operation at all. (A noop).\n *\n * `Subscription` instances that are added to this instance will automatically remove themselves\n * if they are unsubscribed. Functions and {@link Unsubscribable} objects that you wish to remove\n * will need to be removed manually with {@link #remove}\n *\n * @param teardown The finalization logic to add to this subscription.\n */\n add(teardown: TeardownLogic): void {\n // Only add the finalizer if it's not undefined\n // and don't add a subscription to itself.\n if (teardown && teardown !== this) {\n if (this.closed) {\n // If this subscription is already closed,\n // execute whatever finalizer is handed to it automatically.\n execFinalizer(teardown);\n } else {\n if (teardown instanceof Subscription) {\n // We don't add closed subscriptions, and we don't add the same subscription\n // twice. Subscription unsubscribe is idempotent.\n if (teardown.closed || teardown._hasParent(this)) {\n return;\n }\n teardown._addParent(this);\n }\n (this._finalizers = this._finalizers ?? []).push(teardown);\n }\n }\n }\n\n /**\n * Checks to see if a this subscription already has a particular parent.\n * This will signal that this subscription has already been added to the parent in question.\n * @param parent the parent to check for\n */\n private _hasParent(parent: Subscription) {\n const { _parentage } = this;\n return _parentage === parent || (Array.isArray(_parentage) && _parentage.includes(parent));\n }\n\n /**\n * Adds a parent to this subscription so it can be removed from the parent if it\n * unsubscribes on it's own.\n *\n * NOTE: THIS ASSUMES THAT {@link _hasParent} HAS ALREADY BEEN CHECKED.\n * @param parent The parent subscription to add\n */\n private _addParent(parent: Subscription) {\n const { _parentage } = this;\n this._parentage = Array.isArray(_parentage) ? (_parentage.push(parent), _parentage) : _parentage ? [_parentage, parent] : parent;\n }\n\n /**\n * Called on a child when it is removed via {@link #remove}.\n * @param parent The parent to remove\n */\n private _removeParent(parent: Subscription) {\n const { _parentage } = this;\n if (_parentage === parent) {\n this._parentage = null;\n } else if (Array.isArray(_parentage)) {\n arrRemove(_parentage, parent);\n }\n }\n\n /**\n * Removes a finalizer from this subscription that was previously added with the {@link #add} method.\n *\n * Note that `Subscription` instances, when unsubscribed, will automatically remove themselves\n * from every other `Subscription` they have been added to. This means that using the `remove` method\n * is not a common thing and should be used thoughtfully.\n *\n * If you add the same finalizer instance of a function or an unsubscribable object to a `Subscription` instance\n * more than once, you will need to call `remove` the same number of times to remove all instances.\n *\n * All finalizer instances are removed to free up memory upon unsubscription.\n *\n * @param teardown The finalizer to remove from this subscription\n */\n remove(teardown: Exclude): void {\n const { _finalizers } = this;\n _finalizers && arrRemove(_finalizers, teardown);\n\n if (teardown instanceof Subscription) {\n teardown._removeParent(this);\n }\n }\n}\n\nexport const EMPTY_SUBSCRIPTION = Subscription.EMPTY;\n\nexport function isSubscription(value: any): value is Subscription {\n return (\n value instanceof Subscription ||\n (value && 'closed' in value && isFunction(value.remove) && isFunction(value.add) && isFunction(value.unsubscribe))\n );\n}\n\nfunction execFinalizer(finalizer: Unsubscribable | (() => void)) {\n if (isFunction(finalizer)) {\n finalizer();\n } else {\n finalizer.unsubscribe();\n }\n}\n", "import { Subscriber } from './Subscriber';\nimport { ObservableNotification } from './types';\n\n/**\n * The {@link GlobalConfig} object for RxJS. It is used to configure things\n * like how to react on unhandled errors.\n */\nexport const config: GlobalConfig = {\n onUnhandledError: null,\n onStoppedNotification: null,\n Promise: undefined,\n useDeprecatedSynchronousErrorHandling: false,\n useDeprecatedNextContext: false,\n};\n\n/**\n * The global configuration object for RxJS, used to configure things\n * like how to react on unhandled errors. Accessible via {@link config}\n * object.\n */\nexport interface GlobalConfig {\n /**\n * A registration point for unhandled errors from RxJS. These are errors that\n * cannot were not handled by consuming code in the usual subscription path. For\n * example, if you have this configured, and you subscribe to an observable without\n * providing an error handler, errors from that subscription will end up here. This\n * will _always_ be called asynchronously on another job in the runtime. This is because\n * we do not want errors thrown in this user-configured handler to interfere with the\n * behavior of the library.\n */\n onUnhandledError: ((err: any) => void) | null;\n\n /**\n * A registration point for notifications that cannot be sent to subscribers because they\n * have completed, errored or have been explicitly unsubscribed. By default, next, complete\n * and error notifications sent to stopped subscribers are noops. However, sometimes callers\n * might want a different behavior. For example, with sources that attempt to report errors\n * to stopped subscribers, a caller can configure RxJS to throw an unhandled error instead.\n * This will _always_ be called asynchronously on another job in the runtime. This is because\n * we do not want errors thrown in this user-configured handler to interfere with the\n * behavior of the library.\n */\n onStoppedNotification: ((notification: ObservableNotification, subscriber: Subscriber) => void) | null;\n\n /**\n * The promise constructor used by default for {@link Observable#toPromise toPromise} and {@link Observable#forEach forEach}\n * methods.\n *\n * @deprecated As of version 8, RxJS will no longer support this sort of injection of a\n * Promise constructor. If you need a Promise implementation other than native promises,\n * please polyfill/patch Promise as you see appropriate. Will be removed in v8.\n */\n Promise?: PromiseConstructorLike;\n\n /**\n * If true, turns on synchronous error rethrowing, which is a deprecated behavior\n * in v6 and higher. This behavior enables bad patterns like wrapping a subscribe\n * call in a try/catch block. It also enables producer interference, a nasty bug\n * where a multicast can be broken for all observers by a downstream consumer with\n * an unhandled error. DO NOT USE THIS FLAG UNLESS IT'S NEEDED TO BUY TIME\n * FOR MIGRATION REASONS.\n *\n * @deprecated As of version 8, RxJS will no longer support synchronous throwing\n * of unhandled errors. All errors will be thrown on a separate call stack to prevent bad\n * behaviors described above. Will be removed in v8.\n */\n useDeprecatedSynchronousErrorHandling: boolean;\n\n /**\n * If true, enables an as-of-yet undocumented feature from v5: The ability to access\n * `unsubscribe()` via `this` context in `next` functions created in observers passed\n * to `subscribe`.\n *\n * This is being removed because the performance was severely problematic, and it could also cause\n * issues when types other than POJOs are passed to subscribe as subscribers, as they will likely have\n * their `this` context overwritten.\n *\n * @deprecated As of version 8, RxJS will no longer support altering the\n * context of next functions provided as part of an observer to Subscribe. Instead,\n * you will have access to a subscription or a signal or token that will allow you to do things like\n * unsubscribe and test closed status. Will be removed in v8.\n */\n useDeprecatedNextContext: boolean;\n}\n", "import type { TimerHandle } from './timerHandle';\ntype SetTimeoutFunction = (handler: () => void, timeout?: number, ...args: any[]) => TimerHandle;\ntype ClearTimeoutFunction = (handle: TimerHandle) => void;\n\ninterface TimeoutProvider {\n setTimeout: SetTimeoutFunction;\n clearTimeout: ClearTimeoutFunction;\n delegate:\n | {\n setTimeout: SetTimeoutFunction;\n clearTimeout: ClearTimeoutFunction;\n }\n | undefined;\n}\n\nexport const timeoutProvider: TimeoutProvider = {\n // When accessing the delegate, use the variable rather than `this` so that\n // the functions can be called without being bound to the provider.\n setTimeout(handler: () => void, timeout?: number, ...args) {\n const { delegate } = timeoutProvider;\n if (delegate?.setTimeout) {\n return delegate.setTimeout(handler, timeout, ...args);\n }\n return setTimeout(handler, timeout, ...args);\n },\n clearTimeout(handle) {\n const { delegate } = timeoutProvider;\n return (delegate?.clearTimeout || clearTimeout)(handle as any);\n },\n delegate: undefined,\n};\n", "import { config } from '../config';\nimport { timeoutProvider } from '../scheduler/timeoutProvider';\n\n/**\n * Handles an error on another job either with the user-configured {@link onUnhandledError},\n * or by throwing it on that new job so it can be picked up by `window.onerror`, `process.on('error')`, etc.\n *\n * This should be called whenever there is an error that is out-of-band with the subscription\n * or when an error hits a terminal boundary of the subscription and no error handler was provided.\n *\n * @param err the error to report\n */\nexport function reportUnhandledError(err: any) {\n timeoutProvider.setTimeout(() => {\n const { onUnhandledError } = config;\n if (onUnhandledError) {\n // Execute the user-configured error handler.\n onUnhandledError(err);\n } else {\n // Throw so it is picked up by the runtime's uncaught error mechanism.\n throw err;\n }\n });\n}\n", "/* tslint:disable:no-empty */\nexport function noop() { }\n", "import { CompleteNotification, NextNotification, ErrorNotification } from './types';\n\n/**\n * A completion object optimized for memory use and created to be the\n * same \"shape\" as other notifications in v8.\n * @internal\n */\nexport const COMPLETE_NOTIFICATION = (() => createNotification('C', undefined, undefined) as CompleteNotification)();\n\n/**\n * Internal use only. Creates an optimized error notification that is the same \"shape\"\n * as other notifications.\n * @internal\n */\nexport function errorNotification(error: any): ErrorNotification {\n return createNotification('E', undefined, error) as any;\n}\n\n/**\n * Internal use only. Creates an optimized next notification that is the same \"shape\"\n * as other notifications.\n * @internal\n */\nexport function nextNotification(value: T) {\n return createNotification('N', value, undefined) as NextNotification;\n}\n\n/**\n * Ensures that all notifications created internally have the same \"shape\" in v8.\n *\n * TODO: This is only exported to support a crazy legacy test in `groupBy`.\n * @internal\n */\nexport function createNotification(kind: 'N' | 'E' | 'C', value: any, error: any) {\n return {\n kind,\n value,\n error,\n };\n}\n", "import { config } from '../config';\n\nlet context: { errorThrown: boolean; error: any } | null = null;\n\n/**\n * Handles dealing with errors for super-gross mode. Creates a context, in which\n * any synchronously thrown errors will be passed to {@link captureError}. Which\n * will record the error such that it will be rethrown after the call back is complete.\n * TODO: Remove in v8\n * @param cb An immediately executed function.\n */\nexport function errorContext(cb: () => void) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n const isRoot = !context;\n if (isRoot) {\n context = { errorThrown: false, error: null };\n }\n cb();\n if (isRoot) {\n const { errorThrown, error } = context!;\n context = null;\n if (errorThrown) {\n throw error;\n }\n }\n } else {\n // This is the general non-deprecated path for everyone that\n // isn't crazy enough to use super-gross mode (useDeprecatedSynchronousErrorHandling)\n cb();\n }\n}\n\n/**\n * Captures errors only in super-gross mode.\n * @param err the error to capture\n */\nexport function captureError(err: any) {\n if (config.useDeprecatedSynchronousErrorHandling && context) {\n context.errorThrown = true;\n context.error = err;\n }\n}\n", "import { isFunction } from './util/isFunction';\nimport { Observer, ObservableNotification } from './types';\nimport { isSubscription, Subscription } from './Subscription';\nimport { config } from './config';\nimport { reportUnhandledError } from './util/reportUnhandledError';\nimport { noop } from './util/noop';\nimport { nextNotification, errorNotification, COMPLETE_NOTIFICATION } from './NotificationFactories';\nimport { timeoutProvider } from './scheduler/timeoutProvider';\nimport { captureError } from './util/errorContext';\n\n/**\n * Implements the {@link Observer} interface and extends the\n * {@link Subscription} class. While the {@link Observer} is the public API for\n * consuming the values of an {@link Observable}, all Observers get converted to\n * a Subscriber, in order to provide Subscription-like capabilities such as\n * `unsubscribe`. Subscriber is a common type in RxJS, and crucial for\n * implementing operators, but it is rarely used as a public API.\n *\n * @class Subscriber\n */\nexport class Subscriber extends Subscription implements Observer {\n /**\n * A static factory for a Subscriber, given a (potentially partial) definition\n * of an Observer.\n * @param next The `next` callback of an Observer.\n * @param error The `error` callback of an\n * Observer.\n * @param complete The `complete` callback of an\n * Observer.\n * @return A Subscriber wrapping the (partially defined)\n * Observer represented by the given arguments.\n * @nocollapse\n * @deprecated Do not use. Will be removed in v8. There is no replacement for this\n * method, and there is no reason to be creating instances of `Subscriber` directly.\n * If you have a specific use case, please file an issue.\n */\n static create(next?: (x?: T) => void, error?: (e?: any) => void, complete?: () => void): Subscriber {\n return new SafeSubscriber(next, error, complete);\n }\n\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n protected isStopped: boolean = false;\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n protected destination: Subscriber | Observer; // this `any` is the escape hatch to erase extra type param (e.g. R)\n\n /**\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n * There is no reason to directly create an instance of Subscriber. This type is exported for typings reasons.\n */\n constructor(destination?: Subscriber | Observer) {\n super();\n if (destination) {\n this.destination = destination;\n // Automatically chain subscriptions together here.\n // if destination is a Subscription, then it is a Subscriber.\n if (isSubscription(destination)) {\n destination.add(this);\n }\n } else {\n this.destination = EMPTY_OBSERVER;\n }\n }\n\n /**\n * The {@link Observer} callback to receive notifications of type `next` from\n * the Observable, with a value. The Observable may call this method 0 or more\n * times.\n * @param {T} [value] The `next` value.\n * @return {void}\n */\n next(value?: T): void {\n if (this.isStopped) {\n handleStoppedNotification(nextNotification(value), this);\n } else {\n this._next(value!);\n }\n }\n\n /**\n * The {@link Observer} callback to receive notifications of type `error` from\n * the Observable, with an attached `Error`. Notifies the Observer that\n * the Observable has experienced an error condition.\n * @param {any} [err] The `error` exception.\n * @return {void}\n */\n error(err?: any): void {\n if (this.isStopped) {\n handleStoppedNotification(errorNotification(err), this);\n } else {\n this.isStopped = true;\n this._error(err);\n }\n }\n\n /**\n * The {@link Observer} callback to receive a valueless notification of type\n * `complete` from the Observable. Notifies the Observer that the Observable\n * has finished sending push-based notifications.\n * @return {void}\n */\n complete(): void {\n if (this.isStopped) {\n handleStoppedNotification(COMPLETE_NOTIFICATION, this);\n } else {\n this.isStopped = true;\n this._complete();\n }\n }\n\n unsubscribe(): void {\n if (!this.closed) {\n this.isStopped = true;\n super.unsubscribe();\n this.destination = null!;\n }\n }\n\n protected _next(value: T): void {\n this.destination.next(value);\n }\n\n protected _error(err: any): void {\n try {\n this.destination.error(err);\n } finally {\n this.unsubscribe();\n }\n }\n\n protected _complete(): void {\n try {\n this.destination.complete();\n } finally {\n this.unsubscribe();\n }\n }\n}\n\n/**\n * This bind is captured here because we want to be able to have\n * compatibility with monoid libraries that tend to use a method named\n * `bind`. In particular, a library called Monio requires this.\n */\nconst _bind = Function.prototype.bind;\n\nfunction bind any>(fn: Fn, thisArg: any): Fn {\n return _bind.call(fn, thisArg);\n}\n\n/**\n * Internal optimization only, DO NOT EXPOSE.\n * @internal\n */\nclass ConsumerObserver implements Observer {\n constructor(private partialObserver: Partial>) {}\n\n next(value: T): void {\n const { partialObserver } = this;\n if (partialObserver.next) {\n try {\n partialObserver.next(value);\n } catch (error) {\n handleUnhandledError(error);\n }\n }\n }\n\n error(err: any): void {\n const { partialObserver } = this;\n if (partialObserver.error) {\n try {\n partialObserver.error(err);\n } catch (error) {\n handleUnhandledError(error);\n }\n } else {\n handleUnhandledError(err);\n }\n }\n\n complete(): void {\n const { partialObserver } = this;\n if (partialObserver.complete) {\n try {\n partialObserver.complete();\n } catch (error) {\n handleUnhandledError(error);\n }\n }\n }\n}\n\nexport class SafeSubscriber extends Subscriber {\n constructor(\n observerOrNext?: Partial> | ((value: T) => void) | null,\n error?: ((e?: any) => void) | null,\n complete?: (() => void) | null\n ) {\n super();\n\n let partialObserver: Partial>;\n if (isFunction(observerOrNext) || !observerOrNext) {\n // The first argument is a function, not an observer. The next\n // two arguments *could* be observers, or they could be empty.\n partialObserver = {\n next: (observerOrNext ?? undefined) as (((value: T) => void) | undefined),\n error: error ?? undefined,\n complete: complete ?? undefined,\n };\n } else {\n // The first argument is a partial observer.\n let context: any;\n if (this && config.useDeprecatedNextContext) {\n // This is a deprecated path that made `this.unsubscribe()` available in\n // next handler functions passed to subscribe. This only exists behind a flag\n // now, as it is *very* slow.\n context = Object.create(observerOrNext);\n context.unsubscribe = () => this.unsubscribe();\n partialObserver = {\n next: observerOrNext.next && bind(observerOrNext.next, context),\n error: observerOrNext.error && bind(observerOrNext.error, context),\n complete: observerOrNext.complete && bind(observerOrNext.complete, context),\n };\n } else {\n // The \"normal\" path. Just use the partial observer directly.\n partialObserver = observerOrNext;\n }\n }\n\n // Wrap the partial observer to ensure it's a full observer, and\n // make sure proper error handling is accounted for.\n this.destination = new ConsumerObserver(partialObserver);\n }\n}\n\nfunction handleUnhandledError(error: any) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n captureError(error);\n } else {\n // Ideal path, we report this as an unhandled error,\n // which is thrown on a new call stack.\n reportUnhandledError(error);\n }\n}\n\n/**\n * An error handler used when no error handler was supplied\n * to the SafeSubscriber -- meaning no error handler was supplied\n * do the `subscribe` call on our observable.\n * @param err The error to handle\n */\nfunction defaultErrorHandler(err: any) {\n throw err;\n}\n\n/**\n * A handler for notifications that cannot be sent to a stopped subscriber.\n * @param notification The notification being sent\n * @param subscriber The stopped subscriber\n */\nfunction handleStoppedNotification(notification: ObservableNotification, subscriber: Subscriber) {\n const { onStoppedNotification } = config;\n onStoppedNotification && timeoutProvider.setTimeout(() => onStoppedNotification(notification, subscriber));\n}\n\n/**\n * The observer used as a stub for subscriptions where the user did not\n * pass any arguments to `subscribe`. Comes with the default error handling\n * behavior.\n */\nexport const EMPTY_OBSERVER: Readonly> & { closed: true } = {\n closed: true,\n next: noop,\n error: defaultErrorHandler,\n complete: noop,\n};\n", "/**\n * Symbol.observable or a string \"@@observable\". Used for interop\n *\n * @deprecated We will no longer be exporting this symbol in upcoming versions of RxJS.\n * Instead polyfill and use Symbol.observable directly *or* use https://www.npmjs.com/package/symbol-observable\n */\nexport const observable: string | symbol = (() => (typeof Symbol === 'function' && Symbol.observable) || '@@observable')();\n", "/**\n * This function takes one parameter and just returns it. Simply put,\n * this is like `(x: T): T => x`.\n *\n * ## Examples\n *\n * This is useful in some cases when using things like `mergeMap`\n *\n * ```ts\n * import { interval, take, map, range, mergeMap, identity } from 'rxjs';\n *\n * const source$ = interval(1000).pipe(take(5));\n *\n * const result$ = source$.pipe(\n * map(i => range(i)),\n * mergeMap(identity) // same as mergeMap(x => x)\n * );\n *\n * result$.subscribe({\n * next: console.log\n * });\n * ```\n *\n * Or when you want to selectively apply an operator\n *\n * ```ts\n * import { interval, take, identity } from 'rxjs';\n *\n * const shouldLimit = () => Math.random() < 0.5;\n *\n * const source$ = interval(1000);\n *\n * const result$ = source$.pipe(shouldLimit() ? take(5) : identity);\n *\n * result$.subscribe({\n * next: console.log\n * });\n * ```\n *\n * @param x Any value that is returned by this function\n * @returns The value passed as the first parameter to this function\n */\nexport function identity(x: T): T {\n return x;\n}\n", "import { identity } from './identity';\nimport { UnaryFunction } from '../types';\n\nexport function pipe(): typeof identity;\nexport function pipe(fn1: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction,\n fn8: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction,\n fn8: UnaryFunction,\n fn9: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction,\n fn8: UnaryFunction,\n fn9: UnaryFunction,\n ...fns: UnaryFunction[]\n): UnaryFunction;\n\n/**\n * pipe() can be called on one or more functions, each of which can take one argument (\"UnaryFunction\")\n * and uses it to return a value.\n * It returns a function that takes one argument, passes it to the first UnaryFunction, and then\n * passes the result to the next one, passes that result to the next one, and so on. \n */\nexport function pipe(...fns: Array>): UnaryFunction {\n return pipeFromArray(fns);\n}\n\n/** @internal */\nexport function pipeFromArray(fns: Array>): UnaryFunction {\n if (fns.length === 0) {\n return identity as UnaryFunction;\n }\n\n if (fns.length === 1) {\n return fns[0];\n }\n\n return function piped(input: T): R {\n return fns.reduce((prev: any, fn: UnaryFunction) => fn(prev), input as any);\n };\n}\n", "import { Operator } from './Operator';\nimport { SafeSubscriber, Subscriber } from './Subscriber';\nimport { isSubscription, Subscription } from './Subscription';\nimport { TeardownLogic, OperatorFunction, Subscribable, Observer } from './types';\nimport { observable as Symbol_observable } from './symbol/observable';\nimport { pipeFromArray } from './util/pipe';\nimport { config } from './config';\nimport { isFunction } from './util/isFunction';\nimport { errorContext } from './util/errorContext';\n\n/**\n * A representation of any set of values over any amount of time. This is the most basic building block\n * of RxJS.\n *\n * @class Observable\n */\nexport class Observable implements Subscribable {\n /**\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n */\n source: Observable | undefined;\n\n /**\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n */\n operator: Operator | undefined;\n\n /**\n * @constructor\n * @param {Function} subscribe the function that is called when the Observable is\n * initially subscribed to. This function is given a Subscriber, to which new values\n * can be `next`ed, or an `error` method can be called to raise an error, or\n * `complete` can be called to notify of a successful completion.\n */\n constructor(subscribe?: (this: Observable, subscriber: Subscriber) => TeardownLogic) {\n if (subscribe) {\n this._subscribe = subscribe;\n }\n }\n\n // HACK: Since TypeScript inherits static properties too, we have to\n // fight against TypeScript here so Subject can have a different static create signature\n /**\n * Creates a new Observable by calling the Observable constructor\n * @owner Observable\n * @method create\n * @param {Function} subscribe? the subscriber function to be passed to the Observable constructor\n * @return {Observable} a new observable\n * @nocollapse\n * @deprecated Use `new Observable()` instead. Will be removed in v8.\n */\n static create: (...args: any[]) => any = (subscribe?: (subscriber: Subscriber) => TeardownLogic) => {\n return new Observable(subscribe);\n };\n\n /**\n * Creates a new Observable, with this Observable instance as the source, and the passed\n * operator defined as the new observable's operator.\n * @method lift\n * @param operator the operator defining the operation to take on the observable\n * @return a new observable with the Operator applied\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n * If you have implemented an operator using `lift`, it is recommended that you create an\n * operator by simply returning `new Observable()` directly. See \"Creating new operators from\n * scratch\" section here: https://rxjs.dev/guide/operators\n */\n lift(operator?: Operator): Observable {\n const observable = new Observable();\n observable.source = this;\n observable.operator = operator;\n return observable;\n }\n\n subscribe(observerOrNext?: Partial> | ((value: T) => void)): Subscription;\n /** @deprecated Instead of passing separate callback arguments, use an observer argument. Signatures taking separate callback arguments will be removed in v8. Details: https://rxjs.dev/deprecations/subscribe-arguments */\n subscribe(next?: ((value: T) => void) | null, error?: ((error: any) => void) | null, complete?: (() => void) | null): Subscription;\n /**\n * Invokes an execution of an Observable and registers Observer handlers for notifications it will emit.\n *\n * Use it when you have all these Observables, but still nothing is happening.\n *\n * `subscribe` is not a regular operator, but a method that calls Observable's internal `subscribe` function. It\n * might be for example a function that you passed to Observable's constructor, but most of the time it is\n * a library implementation, which defines what will be emitted by an Observable, and when it be will emitted. This means\n * that calling `subscribe` is actually the moment when Observable starts its work, not when it is created, as it is often\n * the thought.\n *\n * Apart from starting the execution of an Observable, this method allows you to listen for values\n * that an Observable emits, as well as for when it completes or errors. You can achieve this in two\n * of the following ways.\n *\n * The first way is creating an object that implements {@link Observer} interface. It should have methods\n * defined by that interface, but note that it should be just a regular JavaScript object, which you can create\n * yourself in any way you want (ES6 class, classic function constructor, object literal etc.). In particular, do\n * not attempt to use any RxJS implementation details to create Observers - you don't need them. Remember also\n * that your object does not have to implement all methods. If you find yourself creating a method that doesn't\n * do anything, you can simply omit it. Note however, if the `error` method is not provided and an error happens,\n * it will be thrown asynchronously. Errors thrown asynchronously cannot be caught using `try`/`catch`. Instead,\n * use the {@link onUnhandledError} configuration option or use a runtime handler (like `window.onerror` or\n * `process.on('error)`) to be notified of unhandled errors. Because of this, it's recommended that you provide\n * an `error` method to avoid missing thrown errors.\n *\n * The second way is to give up on Observer object altogether and simply provide callback functions in place of its methods.\n * This means you can provide three functions as arguments to `subscribe`, where the first function is equivalent\n * of a `next` method, the second of an `error` method and the third of a `complete` method. Just as in case of an Observer,\n * if you do not need to listen for something, you can omit a function by passing `undefined` or `null`,\n * since `subscribe` recognizes these functions by where they were placed in function call. When it comes\n * to the `error` function, as with an Observer, if not provided, errors emitted by an Observable will be thrown asynchronously.\n *\n * You can, however, subscribe with no parameters at all. This may be the case where you're not interested in terminal events\n * and you also handled emissions internally by using operators (e.g. using `tap`).\n *\n * Whichever style of calling `subscribe` you use, in both cases it returns a Subscription object.\n * This object allows you to call `unsubscribe` on it, which in turn will stop the work that an Observable does and will clean\n * up all resources that an Observable used. Note that cancelling a subscription will not call `complete` callback\n * provided to `subscribe` function, which is reserved for a regular completion signal that comes from an Observable.\n *\n * Remember that callbacks provided to `subscribe` are not guaranteed to be called asynchronously.\n * It is an Observable itself that decides when these functions will be called. For example {@link of}\n * by default emits all its values synchronously. Always check documentation for how given Observable\n * will behave when subscribed and if its default behavior can be modified with a `scheduler`.\n *\n * #### Examples\n *\n * Subscribe with an {@link guide/observer Observer}\n *\n * ```ts\n * import { of } from 'rxjs';\n *\n * const sumObserver = {\n * sum: 0,\n * next(value) {\n * console.log('Adding: ' + value);\n * this.sum = this.sum + value;\n * },\n * error() {\n * // We actually could just remove this method,\n * // since we do not really care about errors right now.\n * },\n * complete() {\n * console.log('Sum equals: ' + this.sum);\n * }\n * };\n *\n * of(1, 2, 3) // Synchronously emits 1, 2, 3 and then completes.\n * .subscribe(sumObserver);\n *\n * // Logs:\n * // 'Adding: 1'\n * // 'Adding: 2'\n * // 'Adding: 3'\n * // 'Sum equals: 6'\n * ```\n *\n * Subscribe with functions ({@link deprecations/subscribe-arguments deprecated})\n *\n * ```ts\n * import { of } from 'rxjs'\n *\n * let sum = 0;\n *\n * of(1, 2, 3).subscribe(\n * value => {\n * console.log('Adding: ' + value);\n * sum = sum + value;\n * },\n * undefined,\n * () => console.log('Sum equals: ' + sum)\n * );\n *\n * // Logs:\n * // 'Adding: 1'\n * // 'Adding: 2'\n * // 'Adding: 3'\n * // 'Sum equals: 6'\n * ```\n *\n * Cancel a subscription\n *\n * ```ts\n * import { interval } from 'rxjs';\n *\n * const subscription = interval(1000).subscribe({\n * next(num) {\n * console.log(num)\n * },\n * complete() {\n * // Will not be called, even when cancelling subscription.\n * console.log('completed!');\n * }\n * });\n *\n * setTimeout(() => {\n * subscription.unsubscribe();\n * console.log('unsubscribed!');\n * }, 2500);\n *\n * // Logs:\n * // 0 after 1s\n * // 1 after 2s\n * // 'unsubscribed!' after 2.5s\n * ```\n *\n * @param {Observer|Function} observerOrNext (optional) Either an observer with methods to be called,\n * or the first of three possible handlers, which is the handler for each value emitted from the subscribed\n * Observable.\n * @param {Function} error (optional) A handler for a terminal event resulting from an error. If no error handler is provided,\n * the error will be thrown asynchronously as unhandled.\n * @param {Function} complete (optional) A handler for a terminal event resulting from successful completion.\n * @return {Subscription} a subscription reference to the registered handlers\n * @method subscribe\n */\n subscribe(\n observerOrNext?: Partial> | ((value: T) => void) | null,\n error?: ((error: any) => void) | null,\n complete?: (() => void) | null\n ): Subscription {\n const subscriber = isSubscriber(observerOrNext) ? observerOrNext : new SafeSubscriber(observerOrNext, error, complete);\n\n errorContext(() => {\n const { operator, source } = this;\n subscriber.add(\n operator\n ? // We're dealing with a subscription in the\n // operator chain to one of our lifted operators.\n operator.call(subscriber, source)\n : source\n ? // If `source` has a value, but `operator` does not, something that\n // had intimate knowledge of our API, like our `Subject`, must have\n // set it. We're going to just call `_subscribe` directly.\n this._subscribe(subscriber)\n : // In all other cases, we're likely wrapping a user-provided initializer\n // function, so we need to catch errors and handle them appropriately.\n this._trySubscribe(subscriber)\n );\n });\n\n return subscriber;\n }\n\n /** @internal */\n protected _trySubscribe(sink: Subscriber): TeardownLogic {\n try {\n return this._subscribe(sink);\n } catch (err) {\n // We don't need to return anything in this case,\n // because it's just going to try to `add()` to a subscription\n // above.\n sink.error(err);\n }\n }\n\n /**\n * Used as a NON-CANCELLABLE means of subscribing to an observable, for use with\n * APIs that expect promises, like `async/await`. You cannot unsubscribe from this.\n *\n * **WARNING**: Only use this with observables you *know* will complete. If the source\n * observable does not complete, you will end up with a promise that is hung up, and\n * potentially all of the state of an async function hanging out in memory. To avoid\n * this situation, look into adding something like {@link timeout}, {@link take},\n * {@link takeWhile}, or {@link takeUntil} amongst others.\n *\n * #### Example\n *\n * ```ts\n * import { interval, take } from 'rxjs';\n *\n * const source$ = interval(1000).pipe(take(4));\n *\n * async function getTotal() {\n * let total = 0;\n *\n * await source$.forEach(value => {\n * total += value;\n * console.log('observable -> ' + value);\n * });\n *\n * return total;\n * }\n *\n * getTotal().then(\n * total => console.log('Total: ' + total)\n * );\n *\n * // Expected:\n * // 'observable -> 0'\n * // 'observable -> 1'\n * // 'observable -> 2'\n * // 'observable -> 3'\n * // 'Total: 6'\n * ```\n *\n * @param next a handler for each value emitted by the observable\n * @return a promise that either resolves on observable completion or\n * rejects with the handled error\n */\n forEach(next: (value: T) => void): Promise;\n\n /**\n * @param next a handler for each value emitted by the observable\n * @param promiseCtor a constructor function used to instantiate the Promise\n * @return a promise that either resolves on observable completion or\n * rejects with the handled error\n * @deprecated Passing a Promise constructor will no longer be available\n * in upcoming versions of RxJS. This is because it adds weight to the library, for very\n * little benefit. If you need this functionality, it is recommended that you either\n * polyfill Promise, or you create an adapter to convert the returned native promise\n * to whatever promise implementation you wanted. Will be removed in v8.\n */\n forEach(next: (value: T) => void, promiseCtor: PromiseConstructorLike): Promise;\n\n forEach(next: (value: T) => void, promiseCtor?: PromiseConstructorLike): Promise {\n promiseCtor = getPromiseCtor(promiseCtor);\n\n return new promiseCtor((resolve, reject) => {\n const subscriber = new SafeSubscriber({\n next: (value) => {\n try {\n next(value);\n } catch (err) {\n reject(err);\n subscriber.unsubscribe();\n }\n },\n error: reject,\n complete: resolve,\n });\n this.subscribe(subscriber);\n }) as Promise;\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): TeardownLogic {\n return this.source?.subscribe(subscriber);\n }\n\n /**\n * An interop point defined by the es7-observable spec https://github.com/zenparsing/es-observable\n * @method Symbol.observable\n * @return {Observable} this instance of the observable\n */\n [Symbol_observable]() {\n return this;\n }\n\n /* tslint:disable:max-line-length */\n pipe(): Observable;\n pipe(op1: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction,\n op8: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction,\n op8: OperatorFunction,\n op9: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction,\n op8: OperatorFunction,\n op9: OperatorFunction,\n ...operations: OperatorFunction[]\n ): Observable;\n /* tslint:enable:max-line-length */\n\n /**\n * Used to stitch together functional operators into a chain.\n * @method pipe\n * @return {Observable} the Observable result of all of the operators having\n * been called in the order they were passed in.\n *\n * ## Example\n *\n * ```ts\n * import { interval, filter, map, scan } from 'rxjs';\n *\n * interval(1000)\n * .pipe(\n * filter(x => x % 2 === 0),\n * map(x => x + x),\n * scan((acc, x) => acc + x)\n * )\n * .subscribe(x => console.log(x));\n * ```\n */\n pipe(...operations: OperatorFunction[]): Observable {\n return pipeFromArray(operations)(this);\n }\n\n /* tslint:disable:max-line-length */\n /** @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise */\n toPromise(): Promise;\n /** @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise */\n toPromise(PromiseCtor: typeof Promise): Promise;\n /** @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise */\n toPromise(PromiseCtor: PromiseConstructorLike): Promise;\n /* tslint:enable:max-line-length */\n\n /**\n * Subscribe to this Observable and get a Promise resolving on\n * `complete` with the last emission (if any).\n *\n * **WARNING**: Only use this with observables you *know* will complete. If the source\n * observable does not complete, you will end up with a promise that is hung up, and\n * potentially all of the state of an async function hanging out in memory. To avoid\n * this situation, look into adding something like {@link timeout}, {@link take},\n * {@link takeWhile}, or {@link takeUntil} amongst others.\n *\n * @method toPromise\n * @param [promiseCtor] a constructor function used to instantiate\n * the Promise\n * @return A Promise that resolves with the last value emit, or\n * rejects on an error. If there were no emissions, Promise\n * resolves with undefined.\n * @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise\n */\n toPromise(promiseCtor?: PromiseConstructorLike): Promise {\n promiseCtor = getPromiseCtor(promiseCtor);\n\n return new promiseCtor((resolve, reject) => {\n let value: T | undefined;\n this.subscribe(\n (x: T) => (value = x),\n (err: any) => reject(err),\n () => resolve(value)\n );\n }) as Promise;\n }\n}\n\n/**\n * Decides between a passed promise constructor from consuming code,\n * A default configured promise constructor, and the native promise\n * constructor and returns it. If nothing can be found, it will throw\n * an error.\n * @param promiseCtor The optional promise constructor to passed by consuming code\n */\nfunction getPromiseCtor(promiseCtor: PromiseConstructorLike | undefined) {\n return promiseCtor ?? config.Promise ?? Promise;\n}\n\nfunction isObserver(value: any): value is Observer {\n return value && isFunction(value.next) && isFunction(value.error) && isFunction(value.complete);\n}\n\nfunction isSubscriber(value: any): value is Subscriber {\n return (value && value instanceof Subscriber) || (isObserver(value) && isSubscription(value));\n}\n", "import { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { OperatorFunction } from '../types';\nimport { isFunction } from './isFunction';\n\n/**\n * Used to determine if an object is an Observable with a lift function.\n */\nexport function hasLift(source: any): source is { lift: InstanceType['lift'] } {\n return isFunction(source?.lift);\n}\n\n/**\n * Creates an `OperatorFunction`. Used to define operators throughout the library in a concise way.\n * @param init The logic to connect the liftedSource to the subscriber at the moment of subscription.\n */\nexport function operate(\n init: (liftedSource: Observable, subscriber: Subscriber) => (() => void) | void\n): OperatorFunction {\n return (source: Observable) => {\n if (hasLift(source)) {\n return source.lift(function (this: Subscriber, liftedSource: Observable) {\n try {\n return init(liftedSource, this);\n } catch (err) {\n this.error(err);\n }\n });\n }\n throw new TypeError('Unable to lift unknown Observable type');\n };\n}\n", "import { Subscriber } from '../Subscriber';\n\n/**\n * Creates an instance of an `OperatorSubscriber`.\n * @param destination The downstream subscriber.\n * @param onNext Handles next values, only called if this subscriber is not stopped or closed. Any\n * error that occurs in this function is caught and sent to the `error` method of this subscriber.\n * @param onError Handles errors from the subscription, any errors that occur in this handler are caught\n * and send to the `destination` error handler.\n * @param onComplete Handles completion notification from the subscription. Any errors that occur in\n * this handler are sent to the `destination` error handler.\n * @param onFinalize Additional teardown logic here. This will only be called on teardown if the\n * subscriber itself is not already closed. This is called after all other teardown logic is executed.\n */\nexport function createOperatorSubscriber(\n destination: Subscriber,\n onNext?: (value: T) => void,\n onComplete?: () => void,\n onError?: (err: any) => void,\n onFinalize?: () => void\n): Subscriber {\n return new OperatorSubscriber(destination, onNext, onComplete, onError, onFinalize);\n}\n\n/**\n * A generic helper for allowing operators to be created with a Subscriber and\n * use closures to capture necessary state from the operator function itself.\n */\nexport class OperatorSubscriber extends Subscriber {\n /**\n * Creates an instance of an `OperatorSubscriber`.\n * @param destination The downstream subscriber.\n * @param onNext Handles next values, only called if this subscriber is not stopped or closed. Any\n * error that occurs in this function is caught and sent to the `error` method of this subscriber.\n * @param onError Handles errors from the subscription, any errors that occur in this handler are caught\n * and send to the `destination` error handler.\n * @param onComplete Handles completion notification from the subscription. Any errors that occur in\n * this handler are sent to the `destination` error handler.\n * @param onFinalize Additional finalization logic here. This will only be called on finalization if the\n * subscriber itself is not already closed. This is called after all other finalization logic is executed.\n * @param shouldUnsubscribe An optional check to see if an unsubscribe call should truly unsubscribe.\n * NOTE: This currently **ONLY** exists to support the strange behavior of {@link groupBy}, where unsubscription\n * to the resulting observable does not actually disconnect from the source if there are active subscriptions\n * to any grouped observable. (DO NOT EXPOSE OR USE EXTERNALLY!!!)\n */\n constructor(\n destination: Subscriber,\n onNext?: (value: T) => void,\n onComplete?: () => void,\n onError?: (err: any) => void,\n private onFinalize?: () => void,\n private shouldUnsubscribe?: () => boolean\n ) {\n // It's important - for performance reasons - that all of this class's\n // members are initialized and that they are always initialized in the same\n // order. This will ensure that all OperatorSubscriber instances have the\n // same hidden class in V8. This, in turn, will help keep the number of\n // hidden classes involved in property accesses within the base class as\n // low as possible. If the number of hidden classes involved exceeds four,\n // the property accesses will become megamorphic and performance penalties\n // will be incurred - i.e. inline caches won't be used.\n //\n // The reasons for ensuring all instances have the same hidden class are\n // further discussed in this blog post from Benedikt Meurer:\n // https://benediktmeurer.de/2018/03/23/impact-of-polymorphism-on-component-based-frameworks-like-react/\n super(destination);\n this._next = onNext\n ? function (this: OperatorSubscriber, value: T) {\n try {\n onNext(value);\n } catch (err) {\n destination.error(err);\n }\n }\n : super._next;\n this._error = onError\n ? function (this: OperatorSubscriber, err: any) {\n try {\n onError(err);\n } catch (err) {\n // Send any errors that occur down stream.\n destination.error(err);\n } finally {\n // Ensure finalization.\n this.unsubscribe();\n }\n }\n : super._error;\n this._complete = onComplete\n ? function (this: OperatorSubscriber) {\n try {\n onComplete();\n } catch (err) {\n // Send any errors that occur down stream.\n destination.error(err);\n } finally {\n // Ensure finalization.\n this.unsubscribe();\n }\n }\n : super._complete;\n }\n\n unsubscribe() {\n if (!this.shouldUnsubscribe || this.shouldUnsubscribe()) {\n const { closed } = this;\n super.unsubscribe();\n // Execute additional teardown if we have any and we didn't already do so.\n !closed && this.onFinalize?.();\n }\n }\n}\n", "import { Subscription } from '../Subscription';\n\ninterface AnimationFrameProvider {\n schedule(callback: FrameRequestCallback): Subscription;\n requestAnimationFrame: typeof requestAnimationFrame;\n cancelAnimationFrame: typeof cancelAnimationFrame;\n delegate:\n | {\n requestAnimationFrame: typeof requestAnimationFrame;\n cancelAnimationFrame: typeof cancelAnimationFrame;\n }\n | undefined;\n}\n\nexport const animationFrameProvider: AnimationFrameProvider = {\n // When accessing the delegate, use the variable rather than `this` so that\n // the functions can be called without being bound to the provider.\n schedule(callback) {\n let request = requestAnimationFrame;\n let cancel: typeof cancelAnimationFrame | undefined = cancelAnimationFrame;\n const { delegate } = animationFrameProvider;\n if (delegate) {\n request = delegate.requestAnimationFrame;\n cancel = delegate.cancelAnimationFrame;\n }\n const handle = request((timestamp) => {\n // Clear the cancel function. The request has been fulfilled, so\n // attempting to cancel the request upon unsubscription would be\n // pointless.\n cancel = undefined;\n callback(timestamp);\n });\n return new Subscription(() => cancel?.(handle));\n },\n requestAnimationFrame(...args) {\n const { delegate } = animationFrameProvider;\n return (delegate?.requestAnimationFrame || requestAnimationFrame)(...args);\n },\n cancelAnimationFrame(...args) {\n const { delegate } = animationFrameProvider;\n return (delegate?.cancelAnimationFrame || cancelAnimationFrame)(...args);\n },\n delegate: undefined,\n};\n", "import { createErrorClass } from './createErrorClass';\n\nexport interface ObjectUnsubscribedError extends Error {}\n\nexport interface ObjectUnsubscribedErrorCtor {\n /**\n * @deprecated Internal implementation detail. Do not construct error instances.\n * Cannot be tagged as internal: https://github.com/ReactiveX/rxjs/issues/6269\n */\n new (): ObjectUnsubscribedError;\n}\n\n/**\n * An error thrown when an action is invalid because the object has been\n * unsubscribed.\n *\n * @see {@link Subject}\n * @see {@link BehaviorSubject}\n *\n * @class ObjectUnsubscribedError\n */\nexport const ObjectUnsubscribedError: ObjectUnsubscribedErrorCtor = createErrorClass(\n (_super) =>\n function ObjectUnsubscribedErrorImpl(this: any) {\n _super(this);\n this.name = 'ObjectUnsubscribedError';\n this.message = 'object unsubscribed';\n }\n);\n", "import { Operator } from './Operator';\nimport { Observable } from './Observable';\nimport { Subscriber } from './Subscriber';\nimport { Subscription, EMPTY_SUBSCRIPTION } from './Subscription';\nimport { Observer, SubscriptionLike, TeardownLogic } from './types';\nimport { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';\nimport { arrRemove } from './util/arrRemove';\nimport { errorContext } from './util/errorContext';\n\n/**\n * A Subject is a special type of Observable that allows values to be\n * multicasted to many Observers. Subjects are like EventEmitters.\n *\n * Every Subject is an Observable and an Observer. You can subscribe to a\n * Subject, and you can call next to feed values as well as error and complete.\n */\nexport class Subject extends Observable implements SubscriptionLike {\n closed = false;\n\n private currentObservers: Observer[] | null = null;\n\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n observers: Observer[] = [];\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n isStopped = false;\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n hasError = false;\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n thrownError: any = null;\n\n /**\n * Creates a \"subject\" by basically gluing an observer to an observable.\n *\n * @nocollapse\n * @deprecated Recommended you do not use. Will be removed at some point in the future. Plans for replacement still under discussion.\n */\n static create: (...args: any[]) => any = (destination: Observer, source: Observable): AnonymousSubject => {\n return new AnonymousSubject(destination, source);\n };\n\n constructor() {\n // NOTE: This must be here to obscure Observable's constructor.\n super();\n }\n\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n lift(operator: Operator): Observable {\n const subject = new AnonymousSubject(this, this);\n subject.operator = operator as any;\n return subject as any;\n }\n\n /** @internal */\n protected _throwIfClosed() {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n }\n\n next(value: T) {\n errorContext(() => {\n this._throwIfClosed();\n if (!this.isStopped) {\n if (!this.currentObservers) {\n this.currentObservers = Array.from(this.observers);\n }\n for (const observer of this.currentObservers) {\n observer.next(value);\n }\n }\n });\n }\n\n error(err: any) {\n errorContext(() => {\n this._throwIfClosed();\n if (!this.isStopped) {\n this.hasError = this.isStopped = true;\n this.thrownError = err;\n const { observers } = this;\n while (observers.length) {\n observers.shift()!.error(err);\n }\n }\n });\n }\n\n complete() {\n errorContext(() => {\n this._throwIfClosed();\n if (!this.isStopped) {\n this.isStopped = true;\n const { observers } = this;\n while (observers.length) {\n observers.shift()!.complete();\n }\n }\n });\n }\n\n unsubscribe() {\n this.isStopped = this.closed = true;\n this.observers = this.currentObservers = null!;\n }\n\n get observed() {\n return this.observers?.length > 0;\n }\n\n /** @internal */\n protected _trySubscribe(subscriber: Subscriber): TeardownLogic {\n this._throwIfClosed();\n return super._trySubscribe(subscriber);\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n this._throwIfClosed();\n this._checkFinalizedStatuses(subscriber);\n return this._innerSubscribe(subscriber);\n }\n\n /** @internal */\n protected _innerSubscribe(subscriber: Subscriber) {\n const { hasError, isStopped, observers } = this;\n if (hasError || isStopped) {\n return EMPTY_SUBSCRIPTION;\n }\n this.currentObservers = null;\n observers.push(subscriber);\n return new Subscription(() => {\n this.currentObservers = null;\n arrRemove(observers, subscriber);\n });\n }\n\n /** @internal */\n protected _checkFinalizedStatuses(subscriber: Subscriber) {\n const { hasError, thrownError, isStopped } = this;\n if (hasError) {\n subscriber.error(thrownError);\n } else if (isStopped) {\n subscriber.complete();\n }\n }\n\n /**\n * Creates a new Observable with this Subject as the source. You can do this\n * to create custom Observer-side logic of the Subject and conceal it from\n * code that uses the Observable.\n * @return {Observable} Observable that the Subject casts to\n */\n asObservable(): Observable {\n const observable: any = new Observable();\n observable.source = this;\n return observable;\n }\n}\n\n/**\n * @class AnonymousSubject\n */\nexport class AnonymousSubject extends Subject {\n constructor(\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n public destination?: Observer,\n source?: Observable\n ) {\n super();\n this.source = source;\n }\n\n next(value: T) {\n this.destination?.next?.(value);\n }\n\n error(err: any) {\n this.destination?.error?.(err);\n }\n\n complete() {\n this.destination?.complete?.();\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n return this.source?.subscribe(subscriber) ?? EMPTY_SUBSCRIPTION;\n }\n}\n", "import { Subject } from './Subject';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\n\n/**\n * A variant of Subject that requires an initial value and emits its current\n * value whenever it is subscribed to.\n *\n * @class BehaviorSubject\n */\nexport class BehaviorSubject extends Subject {\n constructor(private _value: T) {\n super();\n }\n\n get value(): T {\n return this.getValue();\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n const subscription = super._subscribe(subscriber);\n !subscription.closed && subscriber.next(this._value);\n return subscription;\n }\n\n getValue(): T {\n const { hasError, thrownError, _value } = this;\n if (hasError) {\n throw thrownError;\n }\n this._throwIfClosed();\n return _value;\n }\n\n next(value: T): void {\n super.next((this._value = value));\n }\n}\n", "import { TimestampProvider } from '../types';\n\ninterface DateTimestampProvider extends TimestampProvider {\n delegate: TimestampProvider | undefined;\n}\n\nexport const dateTimestampProvider: DateTimestampProvider = {\n now() {\n // Use the variable rather than `this` so that the function can be called\n // without being bound to the provider.\n return (dateTimestampProvider.delegate || Date).now();\n },\n delegate: undefined,\n};\n", "import { Subject } from './Subject';\nimport { TimestampProvider } from './types';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\nimport { dateTimestampProvider } from './scheduler/dateTimestampProvider';\n\n/**\n * A variant of {@link Subject} that \"replays\" old values to new subscribers by emitting them when they first subscribe.\n *\n * `ReplaySubject` has an internal buffer that will store a specified number of values that it has observed. Like `Subject`,\n * `ReplaySubject` \"observes\" values by having them passed to its `next` method. When it observes a value, it will store that\n * value for a time determined by the configuration of the `ReplaySubject`, as passed to its constructor.\n *\n * When a new subscriber subscribes to the `ReplaySubject` instance, it will synchronously emit all values in its buffer in\n * a First-In-First-Out (FIFO) manner. The `ReplaySubject` will also complete, if it has observed completion; and it will\n * error if it has observed an error.\n *\n * There are two main configuration items to be concerned with:\n *\n * 1. `bufferSize` - This will determine how many items are stored in the buffer, defaults to infinite.\n * 2. `windowTime` - The amount of time to hold a value in the buffer before removing it from the buffer.\n *\n * Both configurations may exist simultaneously. So if you would like to buffer a maximum of 3 values, as long as the values\n * are less than 2 seconds old, you could do so with a `new ReplaySubject(3, 2000)`.\n *\n * ### Differences with BehaviorSubject\n *\n * `BehaviorSubject` is similar to `new ReplaySubject(1)`, with a couple of exceptions:\n *\n * 1. `BehaviorSubject` comes \"primed\" with a single value upon construction.\n * 2. `ReplaySubject` will replay values, even after observing an error, where `BehaviorSubject` will not.\n *\n * @see {@link Subject}\n * @see {@link BehaviorSubject}\n * @see {@link shareReplay}\n */\nexport class ReplaySubject extends Subject {\n private _buffer: (T | number)[] = [];\n private _infiniteTimeWindow = true;\n\n /**\n * @param bufferSize The size of the buffer to replay on subscription\n * @param windowTime The amount of time the buffered items will stay buffered\n * @param timestampProvider An object with a `now()` method that provides the current timestamp. This is used to\n * calculate the amount of time something has been buffered.\n */\n constructor(\n private _bufferSize = Infinity,\n private _windowTime = Infinity,\n private _timestampProvider: TimestampProvider = dateTimestampProvider\n ) {\n super();\n this._infiniteTimeWindow = _windowTime === Infinity;\n this._bufferSize = Math.max(1, _bufferSize);\n this._windowTime = Math.max(1, _windowTime);\n }\n\n next(value: T): void {\n const { isStopped, _buffer, _infiniteTimeWindow, _timestampProvider, _windowTime } = this;\n if (!isStopped) {\n _buffer.push(value);\n !_infiniteTimeWindow && _buffer.push(_timestampProvider.now() + _windowTime);\n }\n this._trimBuffer();\n super.next(value);\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n this._throwIfClosed();\n this._trimBuffer();\n\n const subscription = this._innerSubscribe(subscriber);\n\n const { _infiniteTimeWindow, _buffer } = this;\n // We use a copy here, so reentrant code does not mutate our array while we're\n // emitting it to a new subscriber.\n const copy = _buffer.slice();\n for (let i = 0; i < copy.length && !subscriber.closed; i += _infiniteTimeWindow ? 1 : 2) {\n subscriber.next(copy[i] as T);\n }\n\n this._checkFinalizedStatuses(subscriber);\n\n return subscription;\n }\n\n private _trimBuffer() {\n const { _bufferSize, _timestampProvider, _buffer, _infiniteTimeWindow } = this;\n // If we don't have an infinite buffer size, and we're over the length,\n // use splice to truncate the old buffer values off. Note that we have to\n // double the size for instances where we're not using an infinite time window\n // because we're storing the values and the timestamps in the same array.\n const adjustedBufferSize = (_infiniteTimeWindow ? 1 : 2) * _bufferSize;\n _bufferSize < Infinity && adjustedBufferSize < _buffer.length && _buffer.splice(0, _buffer.length - adjustedBufferSize);\n\n // Now, if we're not in an infinite time window, remove all values where the time is\n // older than what is allowed.\n if (!_infiniteTimeWindow) {\n const now = _timestampProvider.now();\n let last = 0;\n // Search the array for the first timestamp that isn't expired and\n // truncate the buffer up to that point.\n for (let i = 1; i < _buffer.length && (_buffer[i] as number) <= now; i += 2) {\n last = i;\n }\n last && _buffer.splice(0, last + 1);\n }\n }\n}\n", "import { Scheduler } from '../Scheduler';\nimport { Subscription } from '../Subscription';\nimport { SchedulerAction } from '../types';\n\n/**\n * A unit of work to be executed in a `scheduler`. An action is typically\n * created from within a {@link SchedulerLike} and an RxJS user does not need to concern\n * themselves about creating and manipulating an Action.\n *\n * ```ts\n * class Action extends Subscription {\n * new (scheduler: Scheduler, work: (state?: T) => void);\n * schedule(state?: T, delay: number = 0): Subscription;\n * }\n * ```\n *\n * @class Action\n */\nexport class Action extends Subscription {\n constructor(scheduler: Scheduler, work: (this: SchedulerAction, state?: T) => void) {\n super();\n }\n /**\n * Schedules this action on its parent {@link SchedulerLike} for execution. May be passed\n * some context object, `state`. May happen at some point in the future,\n * according to the `delay` parameter, if specified.\n * @param {T} [state] Some contextual data that the `work` function uses when\n * called by the Scheduler.\n * @param {number} [delay] Time to wait before executing the work, where the\n * time unit is implicit and defined by the Scheduler.\n * @return {void}\n */\n public schedule(state?: T, delay: number = 0): Subscription {\n return this;\n }\n}\n", "import type { TimerHandle } from './timerHandle';\ntype SetIntervalFunction = (handler: () => void, timeout?: number, ...args: any[]) => TimerHandle;\ntype ClearIntervalFunction = (handle: TimerHandle) => void;\n\ninterface IntervalProvider {\n setInterval: SetIntervalFunction;\n clearInterval: ClearIntervalFunction;\n delegate:\n | {\n setInterval: SetIntervalFunction;\n clearInterval: ClearIntervalFunction;\n }\n | undefined;\n}\n\nexport const intervalProvider: IntervalProvider = {\n // When accessing the delegate, use the variable rather than `this` so that\n // the functions can be called without being bound to the provider.\n setInterval(handler: () => void, timeout?: number, ...args) {\n const { delegate } = intervalProvider;\n if (delegate?.setInterval) {\n return delegate.setInterval(handler, timeout, ...args);\n }\n return setInterval(handler, timeout, ...args);\n },\n clearInterval(handle) {\n const { delegate } = intervalProvider;\n return (delegate?.clearInterval || clearInterval)(handle as any);\n },\n delegate: undefined,\n};\n", "import { Action } from './Action';\nimport { SchedulerAction } from '../types';\nimport { Subscription } from '../Subscription';\nimport { AsyncScheduler } from './AsyncScheduler';\nimport { intervalProvider } from './intervalProvider';\nimport { arrRemove } from '../util/arrRemove';\nimport { TimerHandle } from './timerHandle';\n\nexport class AsyncAction extends Action {\n public id: TimerHandle | undefined;\n public state?: T;\n // @ts-ignore: Property has no initializer and is not definitely assigned\n public delay: number;\n protected pending: boolean = false;\n\n constructor(protected scheduler: AsyncScheduler, protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n public schedule(state?: T, delay: number = 0): Subscription {\n if (this.closed) {\n return this;\n }\n\n // Always replace the current state with the new state.\n this.state = state;\n\n const id = this.id;\n const scheduler = this.scheduler;\n\n //\n // Important implementation note:\n //\n // Actions only execute once by default, unless rescheduled from within the\n // scheduled callback. This allows us to implement single and repeat\n // actions via the same code path, without adding API surface area, as well\n // as mimic traditional recursion but across asynchronous boundaries.\n //\n // However, JS runtimes and timers distinguish between intervals achieved by\n // serial `setTimeout` calls vs. a single `setInterval` call. An interval of\n // serial `setTimeout` calls can be individually delayed, which delays\n // scheduling the next `setTimeout`, and so on. `setInterval` attempts to\n // guarantee the interval callback will be invoked more precisely to the\n // interval period, regardless of load.\n //\n // Therefore, we use `setInterval` to schedule single and repeat actions.\n // If the action reschedules itself with the same delay, the interval is not\n // canceled. If the action doesn't reschedule, or reschedules with a\n // different delay, the interval will be canceled after scheduled callback\n // execution.\n //\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, delay);\n }\n\n // Set the pending flag indicating that this action has been scheduled, or\n // has recursively rescheduled itself.\n this.pending = true;\n\n this.delay = delay;\n // If this action has already an async Id, don't request a new one.\n this.id = this.id ?? this.requestAsyncId(scheduler, this.id, delay);\n\n return this;\n }\n\n protected requestAsyncId(scheduler: AsyncScheduler, _id?: TimerHandle, delay: number = 0): TimerHandle {\n return intervalProvider.setInterval(scheduler.flush.bind(scheduler, this), delay);\n }\n\n protected recycleAsyncId(_scheduler: AsyncScheduler, id?: TimerHandle, delay: number | null = 0): TimerHandle | undefined {\n // If this action is rescheduled with the same delay time, don't clear the interval id.\n if (delay != null && this.delay === delay && this.pending === false) {\n return id;\n }\n // Otherwise, if the action's delay time is different from the current delay,\n // or the action has been rescheduled before it's executed, clear the interval id\n if (id != null) {\n intervalProvider.clearInterval(id);\n }\n\n return undefined;\n }\n\n /**\n * Immediately executes this action and the `work` it contains.\n * @return {any}\n */\n public execute(state: T, delay: number): any {\n if (this.closed) {\n return new Error('executing a cancelled action');\n }\n\n this.pending = false;\n const error = this._execute(state, delay);\n if (error) {\n return error;\n } else if (this.pending === false && this.id != null) {\n // Dequeue if the action didn't reschedule itself. Don't call\n // unsubscribe(), because the action could reschedule later.\n // For example:\n // ```\n // scheduler.schedule(function doWork(counter) {\n // /* ... I'm a busy worker bee ... */\n // var originalAction = this;\n // /* wait 100ms before rescheduling the action */\n // setTimeout(function () {\n // originalAction.schedule(counter + 1);\n // }, 100);\n // }, 1000);\n // ```\n this.id = this.recycleAsyncId(this.scheduler, this.id, null);\n }\n }\n\n protected _execute(state: T, _delay: number): any {\n let errored: boolean = false;\n let errorValue: any;\n try {\n this.work(state);\n } catch (e) {\n errored = true;\n // HACK: Since code elsewhere is relying on the \"truthiness\" of the\n // return here, we can't have it return \"\" or 0 or false.\n // TODO: Clean this up when we refactor schedulers mid-version-8 or so.\n errorValue = e ? e : new Error('Scheduled action threw falsy error');\n }\n if (errored) {\n this.unsubscribe();\n return errorValue;\n }\n }\n\n unsubscribe() {\n if (!this.closed) {\n const { id, scheduler } = this;\n const { actions } = scheduler;\n\n this.work = this.state = this.scheduler = null!;\n this.pending = false;\n\n arrRemove(actions, this);\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, null);\n }\n\n this.delay = null!;\n super.unsubscribe();\n }\n }\n}\n", "import { Action } from './scheduler/Action';\nimport { Subscription } from './Subscription';\nimport { SchedulerLike, SchedulerAction } from './types';\nimport { dateTimestampProvider } from './scheduler/dateTimestampProvider';\n\n/**\n * An execution context and a data structure to order tasks and schedule their\n * execution. Provides a notion of (potentially virtual) time, through the\n * `now()` getter method.\n *\n * Each unit of work in a Scheduler is called an `Action`.\n *\n * ```ts\n * class Scheduler {\n * now(): number;\n * schedule(work, delay?, state?): Subscription;\n * }\n * ```\n *\n * @class Scheduler\n * @deprecated Scheduler is an internal implementation detail of RxJS, and\n * should not be used directly. Rather, create your own class and implement\n * {@link SchedulerLike}. Will be made internal in v8.\n */\nexport class Scheduler implements SchedulerLike {\n public static now: () => number = dateTimestampProvider.now;\n\n constructor(private schedulerActionCtor: typeof Action, now: () => number = Scheduler.now) {\n this.now = now;\n }\n\n /**\n * A getter method that returns a number representing the current time\n * (at the time this function was called) according to the scheduler's own\n * internal clock.\n * @return {number} A number that represents the current time. May or may not\n * have a relation to wall-clock time. May or may not refer to a time unit\n * (e.g. milliseconds).\n */\n public now: () => number;\n\n /**\n * Schedules a function, `work`, for execution. May happen at some point in\n * the future, according to the `delay` parameter, if specified. May be passed\n * some context object, `state`, which will be passed to the `work` function.\n *\n * The given arguments will be processed an stored as an Action object in a\n * queue of actions.\n *\n * @param {function(state: ?T): ?Subscription} work A function representing a\n * task, or some unit of work to be executed by the Scheduler.\n * @param {number} [delay] Time to wait before executing the work, where the\n * time unit is implicit and defined by the Scheduler itself.\n * @param {T} [state] Some contextual data that the `work` function uses when\n * called by the Scheduler.\n * @return {Subscription} A subscription in order to be able to unsubscribe\n * the scheduled work.\n */\n public schedule(work: (this: SchedulerAction, state?: T) => void, delay: number = 0, state?: T): Subscription {\n return new this.schedulerActionCtor(this, work).schedule(state, delay);\n }\n}\n", "import { Scheduler } from '../Scheduler';\nimport { Action } from './Action';\nimport { AsyncAction } from './AsyncAction';\nimport { TimerHandle } from './timerHandle';\n\nexport class AsyncScheduler extends Scheduler {\n public actions: Array> = [];\n /**\n * A flag to indicate whether the Scheduler is currently executing a batch of\n * queued actions.\n * @type {boolean}\n * @internal\n */\n public _active: boolean = false;\n /**\n * An internal ID used to track the latest asynchronous task such as those\n * coming from `setTimeout`, `setInterval`, `requestAnimationFrame`, and\n * others.\n * @type {any}\n * @internal\n */\n public _scheduled: TimerHandle | undefined;\n\n constructor(SchedulerAction: typeof Action, now: () => number = Scheduler.now) {\n super(SchedulerAction, now);\n }\n\n public flush(action: AsyncAction): void {\n const { actions } = this;\n\n if (this._active) {\n actions.push(action);\n return;\n }\n\n let error: any;\n this._active = true;\n\n do {\n if ((error = action.execute(action.state, action.delay))) {\n break;\n }\n } while ((action = actions.shift()!)); // exhaust the scheduler queue\n\n this._active = false;\n\n if (error) {\n while ((action = actions.shift()!)) {\n action.unsubscribe();\n }\n throw error;\n }\n }\n}\n", "import { AsyncAction } from './AsyncAction';\nimport { AsyncScheduler } from './AsyncScheduler';\n\n/**\n *\n * Async Scheduler\n *\n * Schedule task as if you used setTimeout(task, duration)\n *\n * `async` scheduler schedules tasks asynchronously, by putting them on the JavaScript\n * event loop queue. It is best used to delay tasks in time or to schedule tasks repeating\n * in intervals.\n *\n * If you just want to \"defer\" task, that is to perform it right after currently\n * executing synchronous code ends (commonly achieved by `setTimeout(deferredTask, 0)`),\n * better choice will be the {@link asapScheduler} scheduler.\n *\n * ## Examples\n * Use async scheduler to delay task\n * ```ts\n * import { asyncScheduler } from 'rxjs';\n *\n * const task = () => console.log('it works!');\n *\n * asyncScheduler.schedule(task, 2000);\n *\n * // After 2 seconds logs:\n * // \"it works!\"\n * ```\n *\n * Use async scheduler to repeat task in intervals\n * ```ts\n * import { asyncScheduler } from 'rxjs';\n *\n * function task(state) {\n * console.log(state);\n * this.schedule(state + 1, 1000); // `this` references currently executing Action,\n * // which we reschedule with new state and delay\n * }\n *\n * asyncScheduler.schedule(task, 3000, 0);\n *\n * // Logs:\n * // 0 after 3s\n * // 1 after 4s\n * // 2 after 5s\n * // 3 after 6s\n * ```\n */\n\nexport const asyncScheduler = new AsyncScheduler(AsyncAction);\n\n/**\n * @deprecated Renamed to {@link asyncScheduler}. Will be removed in v8.\n */\nexport const async = asyncScheduler;\n", "import { AsyncAction } from './AsyncAction';\nimport { Subscription } from '../Subscription';\nimport { QueueScheduler } from './QueueScheduler';\nimport { SchedulerAction } from '../types';\nimport { TimerHandle } from './timerHandle';\n\nexport class QueueAction extends AsyncAction {\n constructor(protected scheduler: QueueScheduler, protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n public schedule(state?: T, delay: number = 0): Subscription {\n if (delay > 0) {\n return super.schedule(state, delay);\n }\n this.delay = delay;\n this.state = state;\n this.scheduler.flush(this);\n return this;\n }\n\n public execute(state: T, delay: number): any {\n return delay > 0 || this.closed ? super.execute(state, delay) : this._execute(state, delay);\n }\n\n protected requestAsyncId(scheduler: QueueScheduler, id?: TimerHandle, delay: number = 0): TimerHandle {\n // If delay exists and is greater than 0, or if the delay is null (the\n // action wasn't rescheduled) but was originally scheduled as an async\n // action, then recycle as an async action.\n\n if ((delay != null && delay > 0) || (delay == null && this.delay > 0)) {\n return super.requestAsyncId(scheduler, id, delay);\n }\n\n // Otherwise flush the scheduler starting with this action.\n scheduler.flush(this);\n\n // HACK: In the past, this was returning `void`. However, `void` isn't a valid\n // `TimerHandle`, and generally the return value here isn't really used. So the\n // compromise is to return `0` which is both \"falsy\" and a valid `TimerHandle`,\n // as opposed to refactoring every other instanceo of `requestAsyncId`.\n return 0;\n }\n}\n", "import { AsyncScheduler } from './AsyncScheduler';\n\nexport class QueueScheduler extends AsyncScheduler {\n}\n", "import { QueueAction } from './QueueAction';\nimport { QueueScheduler } from './QueueScheduler';\n\n/**\n *\n * Queue Scheduler\n *\n * Put every next task on a queue, instead of executing it immediately\n *\n * `queue` scheduler, when used with delay, behaves the same as {@link asyncScheduler} scheduler.\n *\n * When used without delay, it schedules given task synchronously - executes it right when\n * it is scheduled. However when called recursively, that is when inside the scheduled task,\n * another task is scheduled with queue scheduler, instead of executing immediately as well,\n * that task will be put on a queue and wait for current one to finish.\n *\n * This means that when you execute task with `queue` scheduler, you are sure it will end\n * before any other task scheduled with that scheduler will start.\n *\n * ## Examples\n * Schedule recursively first, then do something\n * ```ts\n * import { queueScheduler } from 'rxjs';\n *\n * queueScheduler.schedule(() => {\n * queueScheduler.schedule(() => console.log('second')); // will not happen now, but will be put on a queue\n *\n * console.log('first');\n * });\n *\n * // Logs:\n * // \"first\"\n * // \"second\"\n * ```\n *\n * Reschedule itself recursively\n * ```ts\n * import { queueScheduler } from 'rxjs';\n *\n * queueScheduler.schedule(function(state) {\n * if (state !== 0) {\n * console.log('before', state);\n * this.schedule(state - 1); // `this` references currently executing Action,\n * // which we reschedule with new state\n * console.log('after', state);\n * }\n * }, 0, 3);\n *\n * // In scheduler that runs recursively, you would expect:\n * // \"before\", 3\n * // \"before\", 2\n * // \"before\", 1\n * // \"after\", 1\n * // \"after\", 2\n * // \"after\", 3\n *\n * // But with queue it logs:\n * // \"before\", 3\n * // \"after\", 3\n * // \"before\", 2\n * // \"after\", 2\n * // \"before\", 1\n * // \"after\", 1\n * ```\n */\n\nexport const queueScheduler = new QueueScheduler(QueueAction);\n\n/**\n * @deprecated Renamed to {@link queueScheduler}. Will be removed in v8.\n */\nexport const queue = queueScheduler;\n", "import { AsyncAction } from './AsyncAction';\nimport { AnimationFrameScheduler } from './AnimationFrameScheduler';\nimport { SchedulerAction } from '../types';\nimport { animationFrameProvider } from './animationFrameProvider';\nimport { TimerHandle } from './timerHandle';\n\nexport class AnimationFrameAction extends AsyncAction {\n constructor(protected scheduler: AnimationFrameScheduler, protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n protected requestAsyncId(scheduler: AnimationFrameScheduler, id?: TimerHandle, delay: number = 0): TimerHandle {\n // If delay is greater than 0, request as an async action.\n if (delay !== null && delay > 0) {\n return super.requestAsyncId(scheduler, id, delay);\n }\n // Push the action to the end of the scheduler queue.\n scheduler.actions.push(this);\n // If an animation frame has already been requested, don't request another\n // one. If an animation frame hasn't been requested yet, request one. Return\n // the current animation frame request id.\n return scheduler._scheduled || (scheduler._scheduled = animationFrameProvider.requestAnimationFrame(() => scheduler.flush(undefined)));\n }\n\n protected recycleAsyncId(scheduler: AnimationFrameScheduler, id?: TimerHandle, delay: number = 0): TimerHandle | undefined {\n // If delay exists and is greater than 0, or if the delay is null (the\n // action wasn't rescheduled) but was originally scheduled as an async\n // action, then recycle as an async action.\n if (delay != null ? delay > 0 : this.delay > 0) {\n return super.recycleAsyncId(scheduler, id, delay);\n }\n // If the scheduler queue has no remaining actions with the same async id,\n // cancel the requested animation frame and set the scheduled flag to\n // undefined so the next AnimationFrameAction will request its own.\n const { actions } = scheduler;\n if (id != null && actions[actions.length - 1]?.id !== id) {\n animationFrameProvider.cancelAnimationFrame(id as number);\n scheduler._scheduled = undefined;\n }\n // Return undefined so the action knows to request a new async id if it's rescheduled.\n return undefined;\n }\n}\n", "import { AsyncAction } from './AsyncAction';\nimport { AsyncScheduler } from './AsyncScheduler';\n\nexport class AnimationFrameScheduler extends AsyncScheduler {\n public flush(action?: AsyncAction): void {\n this._active = true;\n // The async id that effects a call to flush is stored in _scheduled.\n // Before executing an action, it's necessary to check the action's async\n // id to determine whether it's supposed to be executed in the current\n // flush.\n // Previous implementations of this method used a count to determine this,\n // but that was unsound, as actions that are unsubscribed - i.e. cancelled -\n // are removed from the actions array and that can shift actions that are\n // scheduled to be executed in a subsequent flush into positions at which\n // they are executed within the current flush.\n const flushId = this._scheduled;\n this._scheduled = undefined;\n\n const { actions } = this;\n let error: any;\n action = action || actions.shift()!;\n\n do {\n if ((error = action.execute(action.state, action.delay))) {\n break;\n }\n } while ((action = actions[0]) && action.id === flushId && actions.shift());\n\n this._active = false;\n\n if (error) {\n while ((action = actions[0]) && action.id === flushId && actions.shift()) {\n action.unsubscribe();\n }\n throw error;\n }\n }\n}\n", "import { AnimationFrameAction } from './AnimationFrameAction';\nimport { AnimationFrameScheduler } from './AnimationFrameScheduler';\n\n/**\n *\n * Animation Frame Scheduler\n *\n * Perform task when `window.requestAnimationFrame` would fire\n *\n * When `animationFrame` scheduler is used with delay, it will fall back to {@link asyncScheduler} scheduler\n * behaviour.\n *\n * Without delay, `animationFrame` scheduler can be used to create smooth browser animations.\n * It makes sure scheduled task will happen just before next browser content repaint,\n * thus performing animations as efficiently as possible.\n *\n * ## Example\n * Schedule div height animation\n * ```ts\n * // html:
\n * import { animationFrameScheduler } from 'rxjs';\n *\n * const div = document.querySelector('div');\n *\n * animationFrameScheduler.schedule(function(height) {\n * div.style.height = height + \"px\";\n *\n * this.schedule(height + 1); // `this` references currently executing Action,\n * // which we reschedule with new state\n * }, 0, 0);\n *\n * // You will see a div element growing in height\n * ```\n */\n\nexport const animationFrameScheduler = new AnimationFrameScheduler(AnimationFrameAction);\n\n/**\n * @deprecated Renamed to {@link animationFrameScheduler}. Will be removed in v8.\n */\nexport const animationFrame = animationFrameScheduler;\n", "import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\n\n/**\n * A simple Observable that emits no items to the Observer and immediately\n * emits a complete notification.\n *\n * Just emits 'complete', and nothing else.\n *\n * ![](empty.png)\n *\n * A simple Observable that only emits the complete notification. It can be used\n * for composing with other Observables, such as in a {@link mergeMap}.\n *\n * ## Examples\n *\n * Log complete notification\n *\n * ```ts\n * import { EMPTY } from 'rxjs';\n *\n * EMPTY.subscribe({\n * next: () => console.log('Next'),\n * complete: () => console.log('Complete!')\n * });\n *\n * // Outputs\n * // Complete!\n * ```\n *\n * Emit the number 7, then complete\n *\n * ```ts\n * import { EMPTY, startWith } from 'rxjs';\n *\n * const result = EMPTY.pipe(startWith(7));\n * result.subscribe(x => console.log(x));\n *\n * // Outputs\n * // 7\n * ```\n *\n * Map and flatten only odd numbers to the sequence `'a'`, `'b'`, `'c'`\n *\n * ```ts\n * import { interval, mergeMap, of, EMPTY } from 'rxjs';\n *\n * const interval$ = interval(1000);\n * const result = interval$.pipe(\n * mergeMap(x => x % 2 === 1 ? of('a', 'b', 'c') : EMPTY),\n * );\n * result.subscribe(x => console.log(x));\n *\n * // Results in the following to the console:\n * // x is equal to the count on the interval, e.g. (0, 1, 2, 3, ...)\n * // x will occur every 1000ms\n * // if x % 2 is equal to 1, print a, b, c (each on its own)\n * // if x % 2 is not equal to 1, nothing will be output\n * ```\n *\n * @see {@link Observable}\n * @see {@link NEVER}\n * @see {@link of}\n * @see {@link throwError}\n */\nexport const EMPTY = new Observable((subscriber) => subscriber.complete());\n\n/**\n * @param scheduler A {@link SchedulerLike} to use for scheduling\n * the emission of the complete notification.\n * @deprecated Replaced with the {@link EMPTY} constant or {@link scheduled} (e.g. `scheduled([], scheduler)`). Will be removed in v8.\n */\nexport function empty(scheduler?: SchedulerLike) {\n return scheduler ? emptyScheduled(scheduler) : EMPTY;\n}\n\nfunction emptyScheduled(scheduler: SchedulerLike) {\n return new Observable((subscriber) => scheduler.schedule(() => subscriber.complete()));\n}\n", "import { SchedulerLike } from '../types';\nimport { isFunction } from './isFunction';\n\nexport function isScheduler(value: any): value is SchedulerLike {\n return value && isFunction(value.schedule);\n}\n", "import { SchedulerLike } from '../types';\nimport { isFunction } from './isFunction';\nimport { isScheduler } from './isScheduler';\n\nfunction last(arr: T[]): T | undefined {\n return arr[arr.length - 1];\n}\n\nexport function popResultSelector(args: any[]): ((...args: unknown[]) => unknown) | undefined {\n return isFunction(last(args)) ? args.pop() : undefined;\n}\n\nexport function popScheduler(args: any[]): SchedulerLike | undefined {\n return isScheduler(last(args)) ? args.pop() : undefined;\n}\n\nexport function popNumber(args: any[], defaultValue: number): number {\n return typeof last(args) === 'number' ? args.pop()! : defaultValue;\n}\n", "export const isArrayLike = ((x: any): x is ArrayLike => x && typeof x.length === 'number' && typeof x !== 'function');", "import { isFunction } from \"./isFunction\";\n\n/**\n * Tests to see if the object is \"thennable\".\n * @param value the object to test\n */\nexport function isPromise(value: any): value is PromiseLike {\n return isFunction(value?.then);\n}\n", "import { InteropObservable } from '../types';\nimport { observable as Symbol_observable } from '../symbol/observable';\nimport { isFunction } from './isFunction';\n\n/** Identifies an input as being Observable (but not necessary an Rx Observable) */\nexport function isInteropObservable(input: any): input is InteropObservable {\n return isFunction(input[Symbol_observable]);\n}\n", "import { isFunction } from './isFunction';\n\nexport function isAsyncIterable(obj: any): obj is AsyncIterable {\n return Symbol.asyncIterator && isFunction(obj?.[Symbol.asyncIterator]);\n}\n", "/**\n * Creates the TypeError to throw if an invalid object is passed to `from` or `scheduled`.\n * @param input The object that was passed.\n */\nexport function createInvalidObservableTypeError(input: any) {\n // TODO: We should create error codes that can be looked up, so this can be less verbose.\n return new TypeError(\n `You provided ${\n input !== null && typeof input === 'object' ? 'an invalid object' : `'${input}'`\n } where a stream was expected. You can provide an Observable, Promise, ReadableStream, Array, AsyncIterable, or Iterable.`\n );\n}\n", "export function getSymbolIterator(): symbol {\n if (typeof Symbol !== 'function' || !Symbol.iterator) {\n return '@@iterator' as any;\n }\n\n return Symbol.iterator;\n}\n\nexport const iterator = getSymbolIterator();\n", "import { iterator as Symbol_iterator } from '../symbol/iterator';\nimport { isFunction } from './isFunction';\n\n/** Identifies an input as being an Iterable */\nexport function isIterable(input: any): input is Iterable {\n return isFunction(input?.[Symbol_iterator]);\n}\n", "import { ReadableStreamLike } from '../types';\nimport { isFunction } from './isFunction';\n\nexport async function* readableStreamLikeToAsyncGenerator(readableStream: ReadableStreamLike): AsyncGenerator {\n const reader = readableStream.getReader();\n try {\n while (true) {\n const { value, done } = await reader.read();\n if (done) {\n return;\n }\n yield value!;\n }\n } finally {\n reader.releaseLock();\n }\n}\n\nexport function isReadableStreamLike(obj: any): obj is ReadableStreamLike {\n // We don't want to use instanceof checks because they would return\n // false for instances from another Realm, like an