feat: add support for React@18 in backward compatible way (#9435)

Any React version matching this semver is supported: >= 16.8 < 19

Refs #8126
Refs #8414
This commit is contained in:
Vladimír Gorej
2023-12-20 16:50:22 +01:00
committed by GitHub
parent 08fe66b8fd
commit 98b53090cb
11 changed files with 2153 additions and 24278 deletions

View File

@@ -30,4 +30,4 @@ export default class ServersContainer extends React.Component {
/>
</div> ) : null
}
}
}

View File

@@ -1,176 +1,149 @@
import React from "react"
/**
* @prettier
*/
import React, { useCallback, useEffect } from "react"
import { OrderedMap } from "immutable"
import PropTypes from "prop-types"
import ImPropTypes from "react-immutable-proptypes"
export default class Servers extends React.Component {
const Servers = ({
servers,
currentServer,
setSelectedServer,
setServerVariableValue,
getServerVariable,
getEffectiveServerValue,
}) => {
const currentServerDefinition =
servers.find((s) => s.get("url") === currentServer) || OrderedMap()
const currentServerVariableDefs =
currentServerDefinition.get("variables") || OrderedMap()
const shouldShowVariableUI = currentServerVariableDefs.size !== 0
static propTypes = {
servers: ImPropTypes.list.isRequired,
currentServer: PropTypes.string.isRequired,
setSelectedServer: PropTypes.func.isRequired,
setServerVariableValue: PropTypes.func.isRequired,
getServerVariable: PropTypes.func.isRequired,
getEffectiveServerValue: PropTypes.func.isRequired
}
useEffect(() => {
if (currentServer) return
componentDidMount() {
let { servers, currentServer } = this.props
// fire 'change' event to set default 'value' of select
setSelectedServer(servers.first()?.get("url"))
}, [])
if(currentServer) {
useEffect(() => {
// server has changed, we may need to set default values
const currentServerDefinition = servers.find(
(server) => server.get("url") === currentServer
)
if (!currentServerDefinition) {
setSelectedServer(servers.first().get("url"))
return
}
// fire 'change' event to set default 'value' of select
this.setServer(servers.first()?.get("url"))
}
UNSAFE_componentWillReceiveProps(nextProps) {
let {
servers,
setServerVariableValue,
getServerVariable
} = nextProps
if (this.props.currentServer !== nextProps.currentServer || this.props.servers !== nextProps.servers) {
// Server has changed, we may need to set default values
let currentServerDefinition = servers
.find(v => v.get("url") === nextProps.currentServer)
let prevServerDefinition = this.props.servers
.find(v => v.get("url") === this.props.currentServer) || OrderedMap()
if(!currentServerDefinition) {
return this.setServer(servers.first().get("url"))
}
let prevServerVariableDefs = prevServerDefinition.get("variables") || OrderedMap()
let prevServerVariableDefaultKey = prevServerVariableDefs.find(v => v.get("default")) || OrderedMap()
let prevServerVariableDefaultValue = prevServerVariableDefaultKey.get("default")
let currentServerVariableDefs = currentServerDefinition.get("variables") || OrderedMap()
let currentServerVariableDefaultKey = currentServerVariableDefs.find(v => v.get("default")) || OrderedMap()
let currentServerVariableDefaultValue = currentServerVariableDefaultKey.get("default")
currentServerVariableDefs.map((val, key) => {
let currentValue = getServerVariable(nextProps.currentServer, key)
// note: it is possible for both key/val to be the same across definitions,
// but we will try to detect a change in default values between definitions
// only set the default value if the user hasn't set one yet
// or if the definition appears to have changed
if (!currentValue || prevServerVariableDefaultValue !== currentServerVariableDefaultValue) {
setServerVariableValue({
server: nextProps.currentServer,
key,
val: val.get("default") || ""
})
}
const currentServerVariableDefs =
currentServerDefinition.get("variables") || OrderedMap()
currentServerVariableDefs.map((val, key) => {
setServerVariableValue({
server: currentServer,
key,
val: val.get("default") || "",
})
}
}
})
}, [currentServer, servers])
onServerChange =( e ) => {
this.setServer( e.target.value )
const handleServerChange = useCallback(
(e) => {
setSelectedServer(e.target.value)
},
[setSelectedServer]
)
// set default variable values
}
const handleServerVariableChange = useCallback(
(e) => {
const variableName = e.target.getAttribute("data-variable")
const newVariableValue = e.target.value
onServerVariableValueChange = ( e ) => {
let {
setServerVariableValue,
currentServer
} = this.props
let variableName = e.target.getAttribute("data-variable")
let newVariableValue = e.target.value
if(typeof setServerVariableValue === "function") {
setServerVariableValue({
server: currentServer,
key: variableName,
val: newVariableValue
val: newVariableValue,
})
}
}
},
[setServerVariableValue, currentServer]
)
setServer = ( value ) => {
let { setSelectedServer } = this.props
setSelectedServer(value)
}
render() {
let { servers,
currentServer,
getServerVariable,
getEffectiveServerValue
} = this.props
let currentServerDefinition = servers.find(s => s.get("url") === currentServer) || OrderedMap()
let currentServerVariableDefs = currentServerDefinition.get("variables") || OrderedMap()
let shouldShowVariableUI = currentServerVariableDefs.size !== 0
return (
<div className="servers">
<label htmlFor="servers">
<select onChange={ this.onServerChange } value={currentServer}>
{ servers.valueSeq().map(
( server ) =>
<option
value={ server.get("url") }
key={ server.get("url") }>
{ server.get("url") }
{ server.get("description") && ` - ${server.get("description")}` }
return (
<div className="servers">
<label htmlFor="servers">
<select onChange={handleServerChange} value={currentServer}>
{servers
.valueSeq()
.map((server) => (
<option value={server.get("url")} key={server.get("url")}>
{server.get("url")}
{server.get("description") && ` - ${server.get("description")}`}
</option>
).toArray()}
</select>
</label>
{ shouldShowVariableUI ?
<div>
<div className={"computed-url"}>
Computed URL:
<code>
{getEffectiveServerValue(currentServer)}
</code>
</div>
<h4>Server variables</h4>
<table>
<tbody>
{
currentServerVariableDefs.entrySeq().map(([name, val]) => {
return <tr key={name}>
<td>{name}</td>
<td>
{ val.get("enum") ?
<select data-variable={name} onChange={this.onServerVariableValueChange}>
{val.get("enum").map(enumValue => {
return <option
selected={enumValue === getServerVariable(currentServer, name)}
))
.toArray()}
</select>
</label>
{shouldShowVariableUI && (
<div>
<div className={"computed-url"}>
Computed URL:
<code>{getEffectiveServerValue(currentServer)}</code>
</div>
<h4>Server variables</h4>
<table>
<tbody>
{currentServerVariableDefs.entrySeq().map(([name, val]) => {
return (
<tr key={name}>
<td>{name}</td>
<td>
{val.get("enum") ? (
<select
data-variable={name}
onChange={handleServerVariableChange}
>
{val.get("enum").map((enumValue) => {
return (
<option
selected={
enumValue ===
getServerVariable(currentServer, name)
}
key={enumValue}
value={enumValue}>
value={enumValue}
>
{enumValue}
</option>
})}
</select> :
<input
type={"text"}
value={getServerVariable(currentServer, name) || ""}
onChange={this.onServerVariableValueChange}
data-variable={name}
></input>
}
</td>
</tr>
})
}
</tbody>
</table>
</div>: null
}
</div>
)
}
)
})}
</select>
) : (
<input
type={"text"}
value={getServerVariable(currentServer, name) || ""}
onChange={handleServerVariableChange}
data-variable={name}
></input>
)}
</td>
</tr>
)
})}
</tbody>
</table>
</div>
)}
</div>
)
}
Servers.propTypes = {
servers: ImPropTypes.list.isRequired,
currentServer: PropTypes.string.isRequired,
setSelectedServer: PropTypes.func.isRequired,
setServerVariableValue: PropTypes.func.isRequired,
getServerVariable: PropTypes.func.isRequired,
getEffectiveServerValue: PropTypes.func.isRequired,
}
export default Servers

View File

@@ -0,0 +1,25 @@
/**
* @prettier
*/
import { getComponent } from "core/plugins/view/root-injects"
import { render } from "./root-injects"
const ViewLegacyPlugin = ({ React, getSystem, getStore, getComponents }) => {
const rootInjects = {}
const reactMajorVersion = parseInt(React?.version, 10)
if (reactMajorVersion >= 16 && reactMajorVersion < 18) {
rootInjects.render = render(
getSystem,
getStore,
getComponent,
getComponents
)
}
return {
rootInjects,
}
}
export default ViewLegacyPlugin

View File

@@ -0,0 +1,12 @@
/**
* @prettier
*/
import React from "react"
import ReactDOM from "react-dom"
export const render =
(getSystem, getStore, getComponent, getComponents) => (domNode) => {
const App = getComponent(getSystem, getStore, getComponents)("App", "root")
ReactDOM.render(<App />, domNode)
}

View File

@@ -82,7 +82,10 @@ export const withMappedContainer = (getSystem, getStore, memGetComponent) => (co
export const render = (getSystem, getStore, getComponent, getComponents) => (domNode) => {
const App = getComponent(getSystem, getStore, getComponents)("App", "root")
ReactDOM.render(<App/>, domNode)
const { createRoot } = ReactDOM
const root = createRoot(domNode)
root.render(<App/>)
}
export const getComponent = (getSystem, getStore, getComponents) => (componentName, container, config = {}) => {