Skip to content

Commit e3cc21b

Browse files
committed
Make a couple of markups for the new attributes option
Move the explanation onto the docs for the method itself, and tweak indentation and types within the code slightly.
1 parent f1ef4fd commit e3cc21b

File tree

2 files changed

+18
-15
lines changed

2 files changed

+18
-15
lines changed

README.md

+13-10
Original file line numberDiff line numberDiff line change
@@ -111,13 +111,23 @@ Normally in `ComponentA`/`ComponentB` examples like the above, switching from `C
111111

112112
How does it work under the hood?
113113

114-
### `portals.createHtmlPortalNode`
114+
### `portals.createHtmlPortalNode([options])`
115115

116116
This creates a detached DOM node, with a little extra functionality attached to allow transmitting props later on.
117117

118-
This node will contain your portal contents later, within a `<div>`, and will eventually be attached in the target location. Its plain DOM node is available at `.element`, so you can mutate that to set any required props (e.g. `className`) with the standard DOM APIs.
118+
This node will contain your portal contents later, within a `<div>`, and will eventually be attached in the target location.
119119

120-
### `portals.createSvgPortalNode`
120+
An optional options object parameter can be passed to configure the node. The only supported option is `attributes`: this can be used to set the HTML attributes (style, class, etc.) of the intermediary, like so:
121+
122+
```javascript
123+
const portalNode = portals.createHtmlPortalNode({
124+
attributes: { id: "div-1", style: "background-color: #aaf; width: 100px;" }
125+
});
126+
```
127+
128+
The div's DOM node is also available at `.element`, so you can mutate that directly with the standard DOM APIs if preferred.
129+
130+
### `portals.createSvgPortalNode([options])`
121131

122132
This creates a detached SVG DOM node. It works identically to the node from `createHtmlPortalNode`, except it will work with SVG elements. Content is placed within a `<g>` instead of a `<div>`.
123133

@@ -144,13 +154,6 @@ It extra props to the InPortal (using the extra functionality we've attached to
144154
* Reverse portals tie rebuilding of the contents of the portal to InPortal (where it's defined), rather than the parent of the OutPortal (where it appears in the tree). That's great (that's the whole point really), but the contents of the InPortal will still be rebuilt anytime the InPortal itself is, e.g. if the InPortal's parent is rebuilt.
145155
* Be aware that if you call `create*PortalNode` in the render method of the parent of an InPortal, you'll get a different node to render into each time, and this will cause unnecessary rerenders, one every time the parent updates. It's generally better to create the node once and persist it, either using the useMemo hook, or in the initial construction of your class component.
146156
* By default, the types for nodes, InPortals and OutPortals allow any props and any components. Pass a component type to them to be more explicit and enforce prop types, e.g. `createHtmlPortalNode<MyComponent>`, `<portals.InPortal<MyComponent>>`, `<portals.OutPortal<MyComponent>>`.
147-
* You can pass `attributes` option to the `create*PortalNode` methods that would allow you to configure any attributes (style, class, etc.) to the intermediary div (or svg):
148-
149-
```
150-
const portalNode = portals.createHtmlPortalNode({
151-
attributes: { id: "div-1", style: "background-color: #aaf; width: 100px;" }
152-
});
153-
```
154157

155158
## Contributing
156159

src/index.tsx

+5-5
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ type ANY_ELEMENT_TYPE = typeof ELEMENT_TYPE_HTML | typeof ELEMENT_TYPE_SVG;
99

1010
type Options = {
1111
attributes: { [key: string]: string };
12-
};
12+
};
1313

1414
// ReactDOM can handle several different namespaces, but they're not exported publicly
1515
// https://github.com/facebook/react/blob/b87aabdfe1b7461e7331abb3601d9e6bb27544bc/packages/react-dom/src/shared/DOMNamespaces.js#L8-L10
@@ -73,8 +73,8 @@ const createPortalNode = <C extends Component<any>>(
7373
}
7474

7575
if (options && typeof options === "object") {
76-
for (const [key, value] of Object.entries(options?.attributes)) {
77-
element.setAttribute(key, value);
76+
for (const [key, value] of Object.entries(options.attributes)) {
77+
element.setAttribute(key, value);
7878
}
7979
}
8080

@@ -236,9 +236,9 @@ class OutPortal<C extends Component<any>> extends React.PureComponent<OutPortalP
236236
}
237237

238238
const createHtmlPortalNode = createPortalNode.bind(null, ELEMENT_TYPE_HTML) as
239-
<C extends Component<any> = Component<any>>() => HtmlPortalNode<C>;
239+
<C extends Component<any> = Component<any>>(options?: Options) => HtmlPortalNode<C>;
240240
const createSvgPortalNode = createPortalNode.bind(null, ELEMENT_TYPE_SVG) as
241-
<C extends Component<any> = Component<any>>() => SvgPortalNode<C>;
241+
<C extends Component<any> = Component<any>>(options?: Options) => SvgPortalNode<C>;
242242

243243
export {
244244
createHtmlPortalNode,

0 commit comments

Comments
 (0)