@@ -2,10 +2,11 @@ import * as React from 'react';
2
2
import * as ReactDOM from 'react-dom' ;
3
3
4
4
// Internally, the portalNode must be for either HTML or SVG elements
5
- const ELEMENT_TYPE_HTML = 'html' ;
5
+ const ELEMENT_TYPE_HTML_BLOCK = 'div' ;
6
+ const ELEMENT_TYPE_HTML_INLINE = 'span' ;
6
7
const ELEMENT_TYPE_SVG = 'svg' ;
7
8
8
- type ANY_ELEMENT_TYPE = typeof ELEMENT_TYPE_HTML | typeof ELEMENT_TYPE_SVG ;
9
+ type ANY_ELEMENT_TYPE = typeof ELEMENT_TYPE_HTML_BLOCK | typeof ELEMENT_TYPE_HTML_INLINE | typeof ELEMENT_TYPE_SVG ;
9
10
10
11
type Options = {
11
12
attributes : { [ key : string ] : string } ;
@@ -32,29 +33,36 @@ interface PortalNodeBase<C extends Component<any>> {
32
33
// latest placeholder we replaced. This avoids some race conditions.
33
34
unmount ( expectedPlaceholder ?: Node ) : void ;
34
35
}
35
- export interface HtmlPortalNode < C extends Component < any > = Component < any > > extends PortalNodeBase < C > {
36
+ export interface HtmlBlockPortalNode < C extends Component < any > = Component < any > > extends PortalNodeBase < C > {
36
37
element : HTMLElement ;
37
- elementType : typeof ELEMENT_TYPE_HTML ;
38
+ elementType : typeof ELEMENT_TYPE_HTML_BLOCK ;
39
+ }
40
+ export interface HtmlInlinePortalNode < C extends Component < any > = Component < any > > extends PortalNodeBase < C > {
41
+ element : HTMLElement ;
42
+ elementType : typeof ELEMENT_TYPE_HTML_INLINE ;
38
43
}
39
44
export interface SvgPortalNode < C extends Component < any > = Component < any > > extends PortalNodeBase < C > {
40
45
element : SVGElement ;
41
46
elementType : typeof ELEMENT_TYPE_SVG ;
42
47
}
43
- type AnyPortalNode < C extends Component < any > = Component < any > > = HtmlPortalNode < C > | SvgPortalNode < C > ;
48
+ type AnyPortalNode < C extends Component < any > = Component < any > > = HtmlBlockPortalNode < C > | HtmlInlinePortalNode < C > | SvgPortalNode < C > ;
44
49
45
50
46
51
const validateElementType = ( domElement : Element , elementType : ANY_ELEMENT_TYPE ) => {
47
52
const ownerDocument = ( domElement . ownerDocument ?? document ) as any ;
48
53
// Cast document to `any` because Typescript doesn't know about the legacy `Document.parentWindow` field, and also
49
54
// doesn't believe `Window.HTMLElement`/`Window.SVGElement` can be used in instanceof tests.
50
55
const ownerWindow = ownerDocument . defaultView ?? ownerDocument . parentWindow ?? window ; // `parentWindow` for IE8 and earlier
51
- if ( elementType === ELEMENT_TYPE_HTML ) {
52
- return domElement instanceof ownerWindow . HTMLElement ;
53
- }
54
- if ( elementType === ELEMENT_TYPE_SVG ) {
55
- return domElement instanceof ownerWindow . SVGElement ;
56
+
57
+ switch ( elementType ) {
58
+ case ELEMENT_TYPE_HTML_BLOCK :
59
+ case ELEMENT_TYPE_HTML_INLINE :
60
+ return domElement instanceof ownerWindow . HTMLElement ;
61
+ case ELEMENT_TYPE_SVG :
62
+ return domElement instanceof ownerWindow . SVGElement ;
63
+ default :
64
+ throw new Error ( `Unrecognized element type "${ elementType } " for validateElementType.` ) ;
56
65
}
57
- throw new Error ( `Unrecognized element type "${ elementType } " for validateElementType.` ) ;
58
66
} ;
59
67
60
68
// This is the internal implementation: the public entry points set elementType to an appropriate value
@@ -68,12 +76,17 @@ const createPortalNode = <C extends Component<any>>(
68
76
let lastPlaceholder : Node | undefined ;
69
77
70
78
let element ;
71
- if ( elementType === ELEMENT_TYPE_HTML ) {
72
- element = document . createElement ( 'div' ) ;
73
- } else if ( elementType === ELEMENT_TYPE_SVG ) {
74
- element = document . createElementNS ( SVG_NAMESPACE , 'g' ) ;
75
- } else {
76
- throw new Error ( `Invalid element type "${ elementType } " for createPortalNode: must be "html" or "svg".` ) ;
79
+
80
+ switch ( elementType ) {
81
+ case ELEMENT_TYPE_HTML_BLOCK :
82
+ case ELEMENT_TYPE_HTML_INLINE :
83
+ element = document . createElement ( elementType ) ;
84
+ break ;
85
+ case ELEMENT_TYPE_SVG :
86
+ element = document . createElementNS ( SVG_NAMESPACE , 'g' ) ;
87
+ break ;
88
+ default :
89
+ throw new Error ( `Invalid element type "${ elementType } " for createPortalNode: must be "div", "span" or "svg".` ) ;
77
90
}
78
91
79
92
if ( options && typeof options === "object" ) {
@@ -186,7 +199,7 @@ type OutPortalProps<C extends Component<any>> = {
186
199
187
200
class OutPortal < C extends Component < any > > extends React . PureComponent < OutPortalProps < C > > {
188
201
189
- private placeholderNode = React . createRef < HTMLDivElement > ( ) ;
202
+ private placeholderNode = React . createRef < HTMLElement > ( ) ;
190
203
private currentPortalNode ?: AnyPortalNode < C > ;
191
204
192
205
constructor ( props : OutPortalProps < C > ) {
@@ -236,18 +249,23 @@ class OutPortal<C extends Component<any>> extends React.PureComponent<OutPortalP
236
249
render ( ) {
237
250
// Render a placeholder to the DOM, so we can get a reference into
238
251
// our location in the DOM, and swap it out for the portaled node.
239
- // A <div> placeholder works fine even for SVG.
240
- return < div ref = { this . placeholderNode } /> ;
252
+ // A <span> placeholder:
253
+ // - prevents invalid HTML (e.g. inside <p>)
254
+ // - works fine even for SVG.
255
+ return < span ref = { this . placeholderNode } /> ;
241
256
}
242
257
}
243
258
244
- const createHtmlPortalNode = createPortalNode . bind ( null , ELEMENT_TYPE_HTML ) as
245
- < C extends Component < any > = Component < any > > ( options ?: Options ) => HtmlPortalNode < C > ;
259
+ const createHtmlPortalNode = createPortalNode . bind ( null , ELEMENT_TYPE_HTML_BLOCK ) as
260
+ < C extends Component < any > = Component < any > > ( options ?: Options ) => HtmlBlockPortalNode < C > ;
261
+ const createHtmlInlinePortalNode = createPortalNode . bind ( null , ELEMENT_TYPE_HTML_INLINE ) as
262
+ < C extends Component < any > = Component < any > > ( options ?: Options ) => HtmlInlinePortalNode < C > ;
246
263
const createSvgPortalNode = createPortalNode . bind ( null , ELEMENT_TYPE_SVG ) as
247
264
< C extends Component < any > = Component < any > > ( options ?: Options ) => SvgPortalNode < C > ;
248
265
249
266
export {
250
267
createHtmlPortalNode ,
268
+ createHtmlInlinePortalNode ,
251
269
createSvgPortalNode ,
252
270
InPortal ,
253
271
OutPortal ,
0 commit comments