reactToWebComponent(ReactComponent, React, ReactDOM, options)
takes the following:
-
ReactComponent
- A React component that you want to convert to a Web Component. -
options
- An set of parameters.-
options.shadow
- Use shadow DOM rather than light DOM. -
options.props
- Array of camelCasedProps to watch as String values or { [camelCasedProps]: "string" | "number" | "boolean" | "function" | "json" }- When specifying Array or Object as the type, the string passed into the attribute must pass
JSON.parse()
requirements. - When specifying Boolean as the type, "true", "1", "yes", "TRUE", and "t" are mapped to
true
. All strings NOT begining with t, T, 1, y, or Y will befalse
. - When specifying Function as the type, the string passed into the attribute must be the name of a function on
window
(orglobal
). Thethis
context of the function will be the instance of the WebComponent / HTMLElement when called. - If PropTypes are defined on the React component, the
options.props
will be ignored and the PropTypes will be used instead. However, we strongly recommend usingoptions.props
instead of PropTypes as it is usually not a good idea to use PropTypes in production. - If
options.props
is an array of string (prop names), the type of those props will beString
.
- When specifying Array or Object as the type, the string passed into the attribute must pass
A new class inheriting from
HTMLElement
is returned. This class is of type CustomElementConstructor can be directly passed tocustomElements.define
as follows: -
customElements.define("web-greeting", reactToWebComponent(Greeting))
Or the class can be defined and used later:
const WebGreeting = reactToWebComponent(Greeting)
customElements.define("web-greeting", WebGreeting)
var myGreeting = new WebGreeting()
document.body.appendChild(myGreeting)
Or the class can be extended:
class WebGreeting extends reactToWebComponent(Greeting) {
disconnectedCallback() {
super.disconnectedCallback()
// special stuff
}
}
customElements.define("web-greeting", WebGreeting)
Components can also be implemented using shadow DOM with either open
or closed
mode.
const WebGreeting = reactToWebComponent(Greeting, {
shadow: "open",
})
customElements.define("web-greeting", WebGreeting)
var myGreeting = new WebGreeting()
document.body.appendChild(myGreeting)
var shadowContent = myGreeting.shadowRoot.children[0]
If propTypes are defined on the underlying React component, dashed-attributes on the webcomponent are converted into the corresponding camelCase React props and the string attribute value is passed in.
function Greeting({ camelCaseName }) {
return <h1>Hello, {camelCaseName}</h1>
}
Greeting.propTypes = {
camelCaseName: PropTypes.string.isRequired,
}
customElements.define(
"my-dashed-style-greeting",
reactToWebComponent(Greeting, {}),
)
document.body.innerHTML =
'<my-dashed-style-greeting camel-case-name="Christopher"></my-dashed-style-greeting>'
console.log(document.body.firstElementChild.innerHTML) // "<h1>Hello, Christopher</h1>"
If options.props
is specified, R2WC will use those props instead of the keys from propTypes. If it's an array, all corresponding kebob-cased attr values will be passed as strings to the underlying React component.
function Greeting({ camelCaseName }) {
return <h1>Hello, {camelCaseName}</h1>
}
customElements.define(
"my-dashed-style-greeting",
reactToWebComponent(Greeting, {
props: { camelCaseName: "string" },
}),
)
document.body.innerHTML =
'<my-dashed-style-greeting camel-case-name="Jane"></my-dashed-style-greeting>'
console.log(document.body.firstElementChild.innerHTML) // "<h1>Hello, Jane</h1>"
If options.props
is an object, the keys are the camelCased React props and the values are any one of the following built in javascript types.
This is the recommended way of passing props to r2wc.
"string" | "number" | "boolean" | "function" | "json"
"json" can be an array or object. The string passed into the attribute must pass JSON.parse()
requirements.
function AttrPropTypeCasting(props) {
console.log(props) // Note
return <h1>Hello, {props.stringProp}</h1>
}
customElements.define(
"attr-prop-type-casting",
reactToWebComponent(AttrPropTypeCasting, {
props: {
stringProp: "string",
numProp: "number",
floatProp: "number",
trueProp: "boolean",
falseProp: "boolean",
arrayProp: "json",
objProp: "json",
},
}),
)
document.body.innerHTML = `
<attr-prop-type-casting
string-prop="iloveyou"
num-prop="360"
float-prop="0.5"
true-prop="true"
false-prop="false"
array-prop='[true, 100.25, "👽", { "aliens": "welcome" }]'
obj-prop='{ "very": "object", "such": "wow!" }'
></attr-prop-type-casting>
`
/*
console.log(props) in the functions produces this:
{
stringProp: "iloveyou",
numProp: 360,
floatProp: 0.5,
trueProp: true,
falseProp: false,
arrayProp: [true, 100.25, "👽", { aliens: "welcome" }],
objProp: { very: "object", such: "wow!" },
}
*/
When Function
is specified as the type, attribute values on the web component will be converted into function references when passed into the underlying React component. The string value of the attribute must be a valid reference to a function on window
(or on global
).
function ThemeSelect({ handleClick }) {
return (
<div>
<button onClick={() => handleClick("V")}>V</button>
<button onClick={() => handleClick("Johnny")}>Johnny</button>
<button onClick={() => handleClick("Jane")}>Jane</button>
</div>
)
}
const WebThemeSelect = reactToWebComponent(ThemeSelect, {
props: {
handleClick: "function",
},
})
customElements.define("theme-select", WebThemeSelect)
window.globalFn = function (selected) {
// "this" is the instance of the WebComponent / HTMLElement
const thisIsEl = this === document.querySelector("theme-select")
console.log(thisIsEl, selected)
}
document.body.innerHTML =
"<theme-select handle-click='globalFn'></theme-select>"
setTimeout(
() => document.querySelector("theme-select button:last-child").click(),
0,
)
// ^ calls globalFn, logs: true, "Jane"