diff --git a/client/dist/js/browserWarning.js b/client/dist/js/browserWarning.js deleted file mode 100644 index e10f535c4..000000000 --- a/client/dist/js/browserWarning.js +++ /dev/null @@ -1 +0,0 @@ -webpackJsonp([1],{"./client/src/lib/browserWarning.js":function(e,r,n){"use strict";var s=n("./node_modules/detect-browser/browser.js"),a=function(e){return e&&e.__esModule?e:{default:e}}(s);if("ie"===a.default.name&&parseInt(a.default.version,10)<=10){var t=document.getElementById("browser-warning-wrapper");t&&(t.className=t.className+" browser-warning-wrapper--incompatible")}}},["./client/src/lib/browserWarning.js"]); \ No newline at end of file diff --git a/client/dist/styles/browser-warning.css b/client/dist/styles/browser-warning.css deleted file mode 100644 index 41848f7f1..000000000 --- a/client/dist/styles/browser-warning.css +++ /dev/null @@ -1 +0,0 @@ -.browser-warning-wrapper{display:none}.browser-warning-wrapper--incompatible{display:block}.browser-warning{position:fixed;color:#fff;background:#005a93;height:100%;width:100%;z-index:100000;top:0;left:0;overflow:auto}.browser-warning__content{text-align:center;padding:30px}.browser-warning__content a{color:#fff;text-decoration:underline}.browser-warning__content .ss-logo{text-indent:-9999em;display:inline-block;width:80px;height:80px;background-image:url();background-image:none,url();background-repeat:no-repeat;background-position:50% 50%}@media (min-width:768px){.browser-warning__content{margin:100px 0}}.browser-warning__list{padding:0;margin-top:70px;list-style:none}.browser-warning__list>li{display:inline-block;min-width:130px;margin:0 15px;padding-top:75px;-webkit-background-size:60px 60px;background-size:60px 60px;background-repeat:no-repeat;background-position:50% 0;margin-bottom:30px}.browser-warning__list>li[data-browser=ie]{background-image:url();background-image:none,url()}.browser-warning__list>li[data-browser=firefox]{background-image:url();background-image:none,url()}.browser-warning__list>li[data-browser=chrome]{background-image:url();background-image:none,url()} \ No newline at end of file diff --git a/client/src/boot/BootRoutes.js b/client/src/boot/BootRoutes.js index 6c32d8f89..f0a0d8267 100644 --- a/client/src/boot/BootRoutes.js +++ b/client/src/boot/BootRoutes.js @@ -5,14 +5,14 @@ import $ from 'jquery'; import React from 'react'; import { Provider } from 'react-redux'; -import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import { BrowserRouter as Router, Prompt } from 'react-router-dom'; -import { renderRoutes } from 'react-router-config'; +import renderReactRoutes from 'lib/renderReactRoutes'; import Config from 'lib/Config'; import pageRouter from 'lib/Router'; import reactRouteRegister from 'lib/ReactRouteRegister'; import App from 'containers/App/App'; -import { ApolloProvider } from 'react-apollo'; +import { ApolloProvider } from '@apollo/client'; // TODO If this doesn't work, checkout https://www.apollographql.com/docs/react/migrating/apollo-client-3-migration/#react-apollo import i18n from 'i18n'; import { isDirty } from 'redux-form'; import getFormState from 'lib/getFormState'; @@ -110,7 +110,8 @@ class BootRoutes { component: App, }); - ReactDOM.render( + const reactRoot = createRoot(document.getElementsByClassName('cms-content')[0]); + reactRoot.render( - {renderRoutes([reactRouteRegister.getRootRoute()])} + {renderReactRoutes([reactRouteRegister.getRootRoute()])} - , - document.getElementsByClassName('cms-content')[0] + ); } @@ -185,10 +185,8 @@ class BootRoutes { // with an event handler is rendered, which means the page router will intercept // events that should be caught by react component event handlers. // Note that this empty link is rendered into an element that doesn't exist in the DOM. - ReactDOM.render( - {}} />, - document.createElement('div') - ); + const root = createRoot(document.createElement('div')); + root.render( {}} />); // Start the page router pageRouter.start(); diff --git a/client/src/boot/apollo/buildCache.js b/client/src/boot/apollo/buildCache.js index 10547c2d2..704be2298 100644 --- a/client/src/boot/apollo/buildCache.js +++ b/client/src/boot/apollo/buildCache.js @@ -1,16 +1,20 @@ -import { InMemoryCache, IntrospectionFragmentMatcher } from 'apollo-cache-inmemory'; +import { InMemoryCache } from '@apollo/client/cache'; import dataIdFromObject from './dataIdFromObject'; -const buildCache = (introspectionQueryResultData) => ( - new InMemoryCache({ - fragmentMatcher: introspectionQueryResultData - ? new IntrospectionFragmentMatcher({ - introspectionQueryResultData, - }) - : null, +const buildCache = (introspectionQueryResultData) => { + const possibleTypes = {}; + if (introspectionQueryResultData) { + introspectionQueryResultData.__schema.types.forEach(supertype => { + if (supertype.possibleTypes) { + possibleTypes[supertype.name] = supertype.possibleTypes.map(subtype => subtype.name); + } + }); + } + return new InMemoryCache({ + possibleTypes, dataIdFromObject, addTypename: true, - }) -); + }); +}; export default buildCache; diff --git a/client/src/boot/apollo/buildClient.js b/client/src/boot/apollo/buildClient.js index 9df3e109a..3970f5d97 100644 --- a/client/src/boot/apollo/buildClient.js +++ b/client/src/boot/apollo/buildClient.js @@ -1,11 +1,9 @@ -/* global window */ -import ApolloClient from 'apollo-client'; -import { withClientState } from 'apollo-link-state'; -import { from } from 'apollo-link'; +// import { withClientState } from '@apollo/client/link/state'; // probably doesn't work +import { ApolloClient, from } from '@apollo/client'; +import Config from 'lib/Config'; import getGraphqlFragments from './getGraphqlFragments'; import buildNetworkComponents from './buildNetworkComponents'; import buildCache from './buildCache'; -import Config from 'lib/Config'; const buildClient = async (baseUrl) => { const graphQLConfig = Config.getSection('SilverStripe\\Admin\\LeftAndMain').graphql; @@ -19,11 +17,7 @@ const buildClient = async (baseUrl) => { } const cache = buildCache(fragmentData); const components = buildNetworkComponents(baseUrl); - const stateLink = withClientState({ - cache, - resolvers: {} - }); - const link = from([stateLink, ...components]); + const link = from(components); return new ApolloClient({ cache, link }); }; diff --git a/client/src/boot/apollo/buildNetworkComponents.js b/client/src/boot/apollo/buildNetworkComponents.js index 9a35f5dc1..f7b1d8a6e 100644 --- a/client/src/boot/apollo/buildNetworkComponents.js +++ b/client/src/boot/apollo/buildNetworkComponents.js @@ -1,6 +1,5 @@ -import { HttpLink } from 'apollo-link-http'; -import { onError } from 'apollo-link-error'; -import { ApolloLink } from 'apollo-link'; +import { ApolloLink, HttpLink } from '@apollo/client'; +import { onError } from '@apollo/client/link/error'; import Config from 'lib/Config'; const buildNetworkComponents = (baseUrl) => { diff --git a/client/src/bundles/bundle.js b/client/src/bundles/bundle.js index 5ae798bc0..631e82f9d 100644 --- a/client/src/bundles/bundle.js +++ b/client/src/bundles/bundle.js @@ -3,118 +3,116 @@ import/no-extraneous-dependencies, import/no-unresolved */ + // Legacy translation handler -require('i18n.js'); +import 'i18n.js'; // Expose fields (see webpack config for matching "externals" config) -require('expose-loader?SilverStripeComponent!lib/SilverStripeComponent'); -require('expose-loader?Backend!lib/Backend'); -require('expose-loader?schemaFieldValues!lib/schemaFieldValues'); -require('expose-loader?FormAlert!components/FormAlert/FormAlert'); -require('expose-loader?Injector!lib/Injector'); -require('expose-loader?reduxFieldReducer!lib/reduxFieldReducer'); -require('expose-loader?getFormState!lib/getFormState'); -require('expose-loader?PopoverField!components/PopoverField/PopoverField'); -require('expose-loader?FieldHolder!components/FieldHolder/FieldHolder'); -require('expose-loader?Form!components/Form/Form'); -require('expose-loader?FormConstants!components/Form/FormConstants'); -require('expose-loader?FormAction!components/FormAction/FormAction'); -require('expose-loader?SchemaActions!state/schema/SchemaActions'); -require('expose-loader?ToastsActions!state/toasts/ToastsActions'); -require('expose-loader?FileStatusIcon!components/FileStatusIcon/FileStatusIcon'); -require('expose-loader?FormBuilder!components/FormBuilder/FormBuilder'); -require('expose-loader?FormBuilderLoader!containers/FormBuilderLoader/FormBuilderLoader'); -require('expose-loader?FormBuilderModal!components/FormBuilderModal/FormBuilderModal'); -require('expose-loader?FileSchemaModalHandler!containers/InsertLinkModal/fileSchemaModalHandler'); -require('expose-loader?InsertLinkModal!containers/InsertLinkModal/InsertLinkModal'); -require('expose-loader?RecordsActions!state/records/RecordsActions'); -require('expose-loader?GridField!components/GridField/GridField'); -require('expose-loader?GridFieldCell!components/GridField/GridFieldCell'); -require('expose-loader?GridFieldHeader!components/GridField/GridFieldHeader'); -require('expose-loader?GridFieldHeaderCell!components/GridField/GridFieldHeaderCell'); -require('expose-loader?GridFieldRow!components/GridField/GridFieldRow'); -require('expose-loader?GridFieldTable!components/GridField/GridFieldTable'); -require('expose-loader?Accordion!components/Accordion/Accordion'); -require('expose-loader?AccordionBlock!components/Accordion/AccordionBlock'); -require('expose-loader?Button!components/Button/Button'); -require('expose-loader?BackButton!components/Button/BackButton'); -require('expose-loader?HiddenField!components/HiddenField/HiddenField'); -require('expose-loader?ListGroup!components/ListGroup/ListGroup'); -require('expose-loader?ListGroupItem!components/ListGroup/ListGroupItem'); -require('expose-loader?Loading!components/Loading/Loading'); -require('expose-loader?TextField!components/TextField/TextField'); -require('expose-loader?LiteralField!components/LiteralField/LiteralField'); -require('expose-loader?Toolbar!components/Toolbar/Toolbar'); -require('expose-loader?FileStatusIcon!components/FileStatusIcon/FileStatusIcon'); -require('expose-loader?Breadcrumb!components/Breadcrumb/Breadcrumb'); -require('expose-loader?ResizeAware!components/ResizeAware/ResizeAware'); -require('expose-loader?TabsActions!state/tabs/TabsActions'); -require('expose-loader?Tag!components/Tag/Tag'); -require('expose-loader?TagList!components/Tag/TagList'); -require('expose-loader?CompactTagList!components/Tag/CompactTagList'); -require('expose-loader?Tip!components/Tip/Tip'); -require('expose-loader?Search!components/Search/Search'); -require('expose-loader?SearchToggle!components/Search/SearchToggle'); -require('expose-loader?TreeDropdownFieldNode!components/TreeDropdownField/TreeDropdownFieldNode'); -require('expose-loader?TreeDropdownFieldMenu!components/TreeDropdownField/TreeDropdownFieldMenu'); -require('expose-loader?TreeDropdownField!components/TreeDropdownField/TreeDropdownField'); -require('expose-loader?BreadcrumbsActions!state/breadcrumbs/BreadcrumbsActions'); -require('expose-loader?RecordsActionTypes!state/records/RecordsActionTypes'); -require('expose-loader?UnsavedFormsActions!state/unsavedForms/UnsavedFormsActions'); -require('expose-loader?Badge!components/Badge/Badge'); -require('expose-loader?Button!components/Button/Button'); -require('expose-loader?BackButton!components/Button/BackButton'); -require('expose-loader?VersionedBadge!components/VersionedBadge/VersionedBadge'); -require('expose-loader?CheckboxSetField!components/CheckboxSetField/CheckboxSetField'); -require('expose-loader?Preview!components/Preview/Preview'); -require('expose-loader?ViewModeStates!state/viewMode/ViewModeStates'); -require('expose-loader?ViewModeActions!state/viewMode/ViewModeActions'); -require('expose-loader?ViewModeToggle!components/ViewModeToggle/ViewModeToggle'); -require('expose-loader?Focusedzone!components/Focusedzone/Focusedzone'); -require('expose-loader?Config!lib/Config'); -require('expose-loader?DataFormat!lib/DataFormat'); -require('expose-loader?ReactRouteRegister!lib/ReactRouteRegister'); -require('expose-loader?Router!lib/Router'); -require('expose-loader?TinyMCEActionRegistrar!lib/TinyMCEActionRegistrar'); -require('expose-loader?ShortcodeSerialiser!lib/ShortcodeSerialiser'); -require('expose-loader?formatWrittenNumber!lib/formatWrittenNumber'); -require('expose-loader?withDragDropContext!lib/withDragDropContext'); +import 'expose-loader?exposes=SilverStripeComponent!lib/SilverStripeComponent'; +import 'expose-loader?exposes=Backend!lib/Backend'; +import 'expose-loader?exposes=schemaFieldValues!lib/schemaFieldValues'; +import 'expose-loader?exposes=FormAlert!components/FormAlert/FormAlert'; +import 'expose-loader?exposes=Injector!lib/Injector'; +import 'expose-loader?exposes=reduxFieldReducer!lib/reduxFieldReducer'; +import 'expose-loader?exposes=getFormState!lib/getFormState'; +import 'expose-loader?exposes=PopoverField!components/PopoverField/PopoverField'; +import 'expose-loader?exposes=FieldHolder!components/FieldHolder/FieldHolder'; +import 'expose-loader?exposes=Form!components/Form/Form'; +import 'expose-loader?exposes=FormConstants!components/Form/FormConstants'; +import 'expose-loader?exposes=FormAction!components/FormAction/FormAction'; +import 'expose-loader?exposes=SchemaActions!state/schema/SchemaActions'; +import 'expose-loader?exposes=ToastsActions!state/toasts/ToastsActions'; +import 'expose-loader?exposes=FileStatusIcon!components/FileStatusIcon/FileStatusIcon'; +import 'expose-loader?exposes=FormBuilder!components/FormBuilder/FormBuilder'; +import 'expose-loader?exposes=FormBuilderLoader!containers/FormBuilderLoader/FormBuilderLoader'; +import 'expose-loader?exposes=FormBuilderModal!components/FormBuilderModal/FormBuilderModal'; +import 'expose-loader?exposes=FileSchemaModalHandler!containers/InsertLinkModal/fileSchemaModalHandler'; +import 'expose-loader?exposes=InsertLinkModal!containers/InsertLinkModal/InsertLinkModal'; +import 'expose-loader?exposes=RecordsActions!state/records/RecordsActions'; +import 'expose-loader?exposes=GridField!components/GridField/GridField'; +import 'expose-loader?exposes=GridFieldCell!components/GridField/GridFieldCell'; +import 'expose-loader?exposes=GridFieldHeader!components/GridField/GridFieldHeader'; +import 'expose-loader?exposes=GridFieldHeaderCell!components/GridField/GridFieldHeaderCell'; +import 'expose-loader?exposes=GridFieldRow!components/GridField/GridFieldRow'; +import 'expose-loader?exposes=GridFieldTable!components/GridField/GridFieldTable'; +import 'expose-loader?exposes=Accordion!components/Accordion/Accordion'; +import 'expose-loader?exposes=AccordionBlock!components/Accordion/AccordionBlock'; +import 'expose-loader?exposes=Button!components/Button/Button'; +import 'expose-loader?exposes=BackButton!components/Button/BackButton'; +import 'expose-loader?exposes=HiddenField!components/HiddenField/HiddenField'; +import 'expose-loader?exposes=ListGroup!components/ListGroup/ListGroup'; +import 'expose-loader?exposes=ListGroupItem!components/ListGroup/ListGroupItem'; +import 'expose-loader?exposes=Loading!components/Loading/Loading'; +import 'expose-loader?exposes=TextField!components/TextField/TextField'; +import 'expose-loader?exposes=LiteralField!components/LiteralField/LiteralField'; +import 'expose-loader?exposes=Toolbar!components/Toolbar/Toolbar'; +import 'expose-loader?exposes=Breadcrumb!components/Breadcrumb/Breadcrumb'; +import 'expose-loader?exposes=ResizeAware!components/ResizeAware/ResizeAware'; +import 'expose-loader?exposes=TabsActions!state/tabs/TabsActions'; +import 'expose-loader?exposes=Tag!components/Tag/Tag'; +import 'expose-loader?exposes=TagList!components/Tag/TagList'; +import 'expose-loader?exposes=CompactTagList!components/Tag/CompactTagList'; +import 'expose-loader?exposes=Tip!components/Tip/Tip'; +import 'expose-loader?exposes=Search!components/Search/Search'; +import 'expose-loader?exposes=SearchToggle!components/Search/SearchToggle'; +import 'expose-loader?exposes=TreeDropdownFieldNode!components/TreeDropdownField/TreeDropdownFieldNode'; +import 'expose-loader?exposes=TreeDropdownFieldMenu!components/TreeDropdownField/TreeDropdownFieldMenu'; +import 'expose-loader?exposes=TreeDropdownField!components/TreeDropdownField/TreeDropdownField'; +import 'expose-loader?exposes=BreadcrumbsActions!state/breadcrumbs/BreadcrumbsActions'; +import 'expose-loader?exposes=RecordsActionTypes!state/records/RecordsActionTypes'; +import 'expose-loader?exposes=UnsavedFormsActions!state/unsavedForms/UnsavedFormsActions'; +import 'expose-loader?exposes=Badge!components/Badge/Badge'; +import 'expose-loader?exposes=VersionedBadge!components/VersionedBadge/VersionedBadge'; +import 'expose-loader?exposes=CheckboxSetField!components/CheckboxSetField/CheckboxSetField'; +import 'expose-loader?exposes=Preview!components/Preview/Preview'; +import 'expose-loader?exposes=ViewModeStates!state/viewMode/ViewModeStates'; +import 'expose-loader?exposes=ViewModeActions!state/viewMode/ViewModeActions'; +import 'expose-loader?exposes=ViewModeToggle!components/ViewModeToggle/ViewModeToggle'; +import 'expose-loader?exposes=Focusedzone!components/Focusedzone/Focusedzone'; +import 'expose-loader?exposes=Config!lib/Config'; +import 'expose-loader?exposes=DataFormat!lib/DataFormat'; +import 'expose-loader?exposes=ReactRouteRegister!lib/ReactRouteRegister'; +import 'expose-loader?exposes=Router!lib/Router'; +import 'expose-loader?exposes=TinyMCEActionRegistrar!lib/TinyMCEActionRegistrar'; +import 'expose-loader?exposes=ShortcodeSerialiser!lib/ShortcodeSerialiser'; +import 'expose-loader?exposes=formatWrittenNumber!lib/formatWrittenNumber'; +import 'expose-loader?exposes=withDragDropContext!lib/withDragDropContext'; +import 'expose-loader?exposes=withRouter!lib/withRouter'; // Legacy CMS -require('../legacy/jquery.changetracker'); -require('../legacy/sspath'); -require('../legacy/ssui.core'); -require('../legacy/LeftAndMain'); -require('../legacy/LeftAndMain.ActionTabSet'); -require('../legacy/LeftAndMain.Panel'); -require('../legacy/LeftAndMain.Tree'); -require('../legacy/LeftAndMain.Content'); -require('../legacy/LeftAndMain.EditForm'); -require('../legacy/LeftAndMain.Menu'); -require('../legacy/LeftAndMain.MobileMenuToggle'); -require('../legacy/LeftAndMain.Preview'); -require('../legacy/LeftAndMain.BatchActions'); -require('../legacy/LeftAndMain.FieldHelp'); -require('../legacy/LeftAndMain.FieldDescriptionToggle'); -require('../legacy/LeftAndMain.TreeDropdownField'); -require('../legacy/AddToCampaignForm'); -require('../legacy/SecurityAdmin'); -require('../legacy/ModelAdmin'); -require('../legacy/ToastsContainer'); +import '../legacy/jquery.changetracker'; +import '../legacy/sspath'; +import '../legacy/ssui.core'; +import '../legacy/LeftAndMain'; +import '../legacy/LeftAndMain.ActionTabSet'; +import '../legacy/LeftAndMain.Panel'; +import '../legacy/LeftAndMain.Tree'; +import '../legacy/LeftAndMain.Content'; +import '../legacy/LeftAndMain.EditForm'; +import '../legacy/LeftAndMain.Menu'; +import '../legacy/LeftAndMain.MobileMenuToggle'; +import '../legacy/LeftAndMain.Preview'; +import '../legacy/LeftAndMain.BatchActions'; +import '../legacy/LeftAndMain.FieldHelp'; +import '../legacy/LeftAndMain.FieldDescriptionToggle'; +import '../legacy/LeftAndMain.TreeDropdownField'; +import '../legacy/AddToCampaignForm'; +import '../legacy/SecurityAdmin'; +import '../legacy/ModelAdmin'; +import '../legacy/ToastsContainer'; // Legacy form fields // Fields used by core legacy UIs, or available to users // To do: determine better way of using webpack to pull in optional javascript -require('../legacy/ConfirmedPasswordField'); -require('../legacy/SelectionGroup'); -require('../legacy/DateField'); -require('../legacy/ToggleCompositeField'); -require('../legacy/TreeDropdownField/TreeDropdownFieldEntwine'); -require('../legacy/UsedOnTable/UsedOnTableEntwine'); -require('../legacy/DateField'); -require('../legacy/DatetimeField'); -require('../legacy/HtmlEditorField'); -require('../legacy/TabSet'); -require('../legacy/GridField'); +import '../legacy/ConfirmedPasswordField'; +import '../legacy/SelectionGroup'; +import '../legacy/DateField'; +import '../legacy/ToggleCompositeField'; +import '../legacy/TreeDropdownField/TreeDropdownFieldEntwine'; +import '../legacy/UsedOnTable/UsedOnTableEntwine'; +import '../legacy/DatetimeField'; +import '../legacy/HtmlEditorField'; +import '../legacy/TabSet'; +import '../legacy/GridField'; -require('boot'); +import 'boot'; diff --git a/client/src/bundles/vendor.js b/client/src/bundles/vendor.js index 23a17a44d..fd1283faf 100644 --- a/client/src/bundles/vendor.js +++ b/client/src/bundles/vendor.js @@ -1,60 +1,56 @@ /* eslint-disable import/no-webpack-loader-syntax, - import/no-extraneous-dependencies, import/no-unresolved, import/extensions */ -// TODO Enable require(*.css) includes once https://github.com/webpack/extract-text-webpack-plugin/issues/179 +// TODO Enable import *.css) includes once https://github.com/webpack/extract-text-webpack-plugin/issues/179 // is resolved. Included in bundle.scss for now. -require('babel-polyfill'); -require('json-js'); +import 'core-js/stable'; // jQuery plugins require that the jQuery object is exposed as a global // webpack.ProvidePlugin is used to ensure that jQuery and $ are provided to all includes -require('script-loader!../../../thirdparty/jquery/jquery.js'); -require('expose-loader?jQuery!jquery'); +import '../../../thirdparty/jquery/jquery.js'; +// import 'expose-loader?exposes=$,jQuery!../../../thirdparty/jquery/jquery.js'; // Expose the libraries as globals for other modules to access // Note that these are order-dependent - earlier items should not depend on later ones -require('expose-loader?PropTypes!prop-types'); -require('expose-loader?classnames!classnames'); -require('expose-loader?DeepFreezeStrict!deep-freeze-strict'); -require('expose-loader?React!react'); -require('expose-loader?ReactDom!react-dom'); -require('expose-loader?ReactRouterDom!react-router-dom'); -require('expose-loader?Reactstrap!reactstrap'); -require('expose-loader?IsomorphicFetch!isomorphic-fetch'); -require('expose-loader?Redux!redux'); -require('expose-loader?ReactRedux!react-redux'); -require('expose-loader?ReduxThunk!redux-thunk'); -require('expose-loader?ReduxForm!redux-form'); -require('expose-loader?ReactSelect!react-select'); -require('expose-loader?ReactDND!react-dnd'); -require('expose-loader?ReactDNDHtml5Backend!react-dnd-html5-backend'); -require('expose-loader?Page!page.js'); -require('expose-loader?validator!validator'); -require('expose-loader?ApolloClient!apollo-client'); -require('expose-loader?ReactApollo!react-apollo'); -require('expose-loader?GraphQLTag!graphql-tag'); -require('expose-loader?GraphQLFragments!graphql-fragments'); -require('expose-loader?NodeUrl!url'); -require('expose-loader?qs!qs'); -require('expose-loader?modernizr!modernizr'); -require('expose-loader?history!history'); -require('expose-loader?moment!moment'); -require('expose-loader?merge!merge'); +import 'expose-loader?exposes=PropTypes!prop-types'; +import 'expose-loader?exposes=classnames!classnames'; +import 'expose-loader?exposes=DeepFreezeStrict!deep-freeze-strict'; +import 'expose-loader?exposes=React!react'; +import 'expose-loader?exposes=ReactDom!react-dom'; +import 'expose-loader?exposes=ReactRouterDom!react-router-dom'; +import 'expose-loader?exposes=Reactstrap!reactstrap'; +import 'expose-loader?exposes=IsomorphicFetch!isomorphic-fetch'; +import 'expose-loader?exposes=Redux!redux'; +import 'expose-loader?exposes=ReactRedux!react-redux'; +import 'expose-loader?exposes=ReduxThunk!redux-thunk'; +import 'expose-loader?exposes=ReduxForm!redux-form'; +import 'expose-loader?exposes=qs!qs'; +import 'expose-loader?exposes=ReactSelect!react-select'; +import 'expose-loader?exposes=ReactDND!react-dnd'; +import 'expose-loader?exposes=ReactDNDHtml5Backend!react-dnd-html5-backend'; +import 'expose-loader?exposes=Page!page.js'; +import 'expose-loader?exposes=validator!validator'; +// import 'expose-loader?exposes=ApolloClient!@apollo/client'; +import 'expose-loader?exposes=GraphQLTag!graphql-tag'; +import 'expose-loader?exposes=GraphQLFragments!graphql-fragments'; +import 'expose-loader?exposes=NodeUrl!url'; +import 'expose-loader?exposes=modernizr!modernizr'; +import 'expose-loader?exposes=moment!moment'; +import 'expose-loader?exposes=merge!merge'; -require('../../../thirdparty/jquery-ondemand/jquery.ondemand.js'); -require('../../../thirdparty/jquery-ui/jquery-ui.js'); -// require('../../../thirdparty/jquery-ui-themes/smoothness/jquery-ui.css'); -require('../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'); -require('../../../thirdparty/jquery-cookie/jquery.cookie.js'); -require('../../../thirdparty/jquery-query/jquery.query.js'); -require('../../../thirdparty/jquery-form/jquery.form.js'); -require('jquery-sizes/lib/jquery.sizes.js'); -require('../../../thirdparty/jstree/jquery.jstree.js'); -// require('../../../thirdparty/stree/themes/apple/style.css'); -require('../../../thirdparty/jquery-hoverIntent/jquery.hoverIntent.js'); +import '../../../thirdparty/jquery-ondemand/jquery.ondemand.js'; +import '../../../thirdparty/jquery-ui/jquery-ui.js'; +// import '../../../thirdparty/jquery-ui-themes/smoothness/jquery-ui.css'; +import '../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'; +import '../../../thirdparty/jquery-cookie/jquery.cookie.js'; +import '../../../thirdparty/jquery-query/jquery.query.js'; +import '../../../thirdparty/jquery-form/jquery.form.js'; +import 'jquery-sizes/lib/jquery.sizes.js'; +import '../../../thirdparty/jstree/jquery.jstree.js'; +// import '../../../thirdparty/stree/themes/apple/style.css'; +import '../../../thirdparty/jquery-hoverIntent/jquery.hoverIntent.js'; -require('chosen-js'); +import 'chosen-js'; diff --git a/client/src/components/GridFieldActions/GridFieldActions.js b/client/src/components/GridFieldActions/GridFieldActions.js index 2a0a88202..826c9c031 100644 --- a/client/src/components/GridFieldActions/GridFieldActions.js +++ b/client/src/components/GridFieldActions/GridFieldActions.js @@ -37,7 +37,7 @@ class GridFieldActions extends PureComponent { return groupsList; }, []); - const dropdownMenuProps = { right: true }; + const dropdownMenuProps = { end: true }; const dropdownToggleClassNames = [ 'action-menu__toggle', 'btn', diff --git a/client/src/components/Toasts/Toasts.js b/client/src/components/Toasts/Toasts.js index 9dcc9a0b9..13997bc5d 100644 --- a/client/src/components/Toasts/Toasts.js +++ b/client/src/components/Toasts/Toasts.js @@ -1,6 +1,6 @@ import React, { useState } from 'react'; import PropTypes from 'prop-types'; -import Toast, { toastPropType } from './Toast'; +import Toast from './Toast'; /** * Time in milliseconds to wait before we action the pause/resume calls. @@ -44,7 +44,7 @@ const Toasts = ({ toasts, onDismiss, onPause, onResume }) => { }; Toasts.propTypes = { - toasts: PropTypes.arrayOf(PropTypes.shape(toastPropType)).isRequired, + toasts: PropTypes.arrayOf(PropTypes.shape(Toast.propTypes)).isRequired, onDismiss: PropTypes.func.isRequired, onPause: PropTypes.func.isRequired, onResume: PropTypes.func.isRequired, diff --git a/client/src/components/TreeDropdownField/TreeDropdownField.js b/client/src/components/TreeDropdownField/TreeDropdownField.js index 1ebcb8d50..227445ef6 100644 --- a/client/src/components/TreeDropdownField/TreeDropdownField.js +++ b/client/src/components/TreeDropdownField/TreeDropdownField.js @@ -701,30 +701,30 @@ class TreeDropdownField extends Component { // Hack! Temporary fix until we can do a proper upgrade of react-select to >= 2.0. // eslint-disable-next-line func-names -Select.prototype.componentWillReceiveProps = function (nextProps) { - function handleRequired(value, multi) { - if (!value) { - return true; - } - return multi ? value.length === 0 : Object.keys(value).length === 0; - } - - const valueArray = this.getValueArray(nextProps.value, nextProps); - - if (nextProps.required) { - this.setState({ - required: handleRequired(valueArray[0], nextProps.multi) - }); - } else if (this.props.required) { - // Used to be required but it's not any more - this.setState({ required: false }); - } - // Array comparison in react-select is broken. - const [current, next] = [this.props.value, nextProps.value].map(JSON.stringify); - if (this.state.inputValue && current !== next && nextProps.onSelectResetsInput) { - this.setState({ inputValue: this.handleInputValueChange('') }); - } -}; +// Select.prototype.componentWillReceiveProps = function (nextProps) { +// function handleRequired(value, multi) { +// if (!value) { +// return true; +// } +// return multi ? value.length === 0 : Object.keys(value).length === 0; +// } + +// const valueArray = this.getValueArray(nextProps.value, nextProps); + +// if (nextProps.required) { +// this.setState({ +// required: handleRequired(valueArray[0], nextProps.multi) +// }); +// } else if (this.props.required) { +// // Used to be required but it's not any more +// this.setState({ required: false }); +// } +// // Array comparison in react-select is broken. +// const [current, next] = [this.props.value, nextProps.value].map(JSON.stringify); +// if (this.state.inputValue && current !== next && nextProps.onSelectResetsInput) { +// this.setState({ inputValue: this.handleInputValueChange('') }); +// } +// }; TreeDropdownField.propTypes = { extraClass: PropTypes.string, diff --git a/client/src/components/TreeDropdownField/TreeDropdownField.scss b/client/src/components/TreeDropdownField/TreeDropdownField.scss index 18118f692..b2eb003cd 100644 --- a/client/src/components/TreeDropdownField/TreeDropdownField.scss +++ b/client/src/components/TreeDropdownField/TreeDropdownField.scss @@ -1,5 +1,3 @@ -@import "~react-select/dist/react-select.css"; - .treedropdownfield__option+.treedropdownfield__option { border-top: 1px solid $border-color-light; } diff --git a/client/src/components/TreeDropdownField/treeUtils.js b/client/src/components/TreeDropdownField/treeUtils.js index 92e111a62..b2d57d0dd 100644 --- a/client/src/components/TreeDropdownField/treeUtils.js +++ b/client/src/components/TreeDropdownField/treeUtils.js @@ -35,7 +35,7 @@ export const findTreeByPath = (tree, path) => { * @param {*} id - id property of node to find path for * @return {Object} - The tree if found, or null if not found. */ -const findTreeByID = (tree, id) => { +export const findTreeByID = (tree, id) => { // No valid tree if (!id || !tree || !tree.children || Object.keys(tree).length === 0) { return null; diff --git a/client/src/containers/App/App.js b/client/src/containers/App/App.js index a9db4f66d..b42f89a54 100644 --- a/client/src/containers/App/App.js +++ b/client/src/containers/App/App.js @@ -1,13 +1,13 @@ import React from 'react'; import { provideInjector } from 'lib/Injector'; -import { renderRoutes } from 'react-router-config'; +import renderReactRoutes from 'lib/renderReactRoutes'; /** * Empty container for the moment, will eventually contain the CMS menu` * and apply to document.body, rather than just one specific DOM element. */ const App = ({ route }) => ( - {renderRoutes(route.routes())} + {renderReactRoutes(route.routes())} ); export default provideInjector(App); diff --git a/client/src/legacy/AddToCampaignForm.js b/client/src/legacy/AddToCampaignForm.js index 56be14b48..190b20048 100644 --- a/client/src/legacy/AddToCampaignForm.js +++ b/client/src/legacy/AddToCampaignForm.js @@ -3,6 +3,7 @@ import i18n from 'i18n'; import jQuery from 'jquery'; import React from 'react'; import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import { loadComponent } from 'lib/Injector'; const FormBuilderModal = loadComponent('FormBuilderModal'); @@ -44,6 +45,7 @@ jQuery.entwine('ss', ($) => { * The "add to campaign" dialog is used in a similar fashion in AssetAdmin. */ $('#add-to-campaign__dialog-wrapper').entwine({ + ReactRoot: null, onunmatch() { // solves errors given by ReactDOM "no matched root found" error. @@ -69,7 +71,11 @@ jQuery.entwine('ss', ($) => { const modalSchemaUrl = `${sectionConfig.form.AddToCampaignForm.schemaUrl}/${id}`; const title = i18n._t('Admin.ADD_TO_CAMPAIGN', 'Add to campaign'); - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( { responseClassBad="modal__response modal__response--error" responseClassGood="modal__response modal__response--good" identifier="Admin.AddToCampaign" - />, - this[0] + /> ); + this.setReactRoot(root); }, _clearModal() { - ReactDOM.unmountComponentAtNode(this[0]); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); + } // this.empty(); }, diff --git a/client/src/legacy/ConfirmedPasswordField.js b/client/src/legacy/ConfirmedPasswordField.js index bc4a5ccab..464e4d265 100644 --- a/client/src/legacy/ConfirmedPasswordField.js +++ b/client/src/legacy/ConfirmedPasswordField.js @@ -1,7 +1,7 @@ import $ from 'jquery'; // TODO Enable once https://github.com/webpack/extract-text-webpack-plugin/issues/179 is resolved. Included in bundle.scss for now. -// require('../styles/legacy/ConfirmedPasswordField.scss'); +// import '../styles/legacy/ConfirmedPasswordField.scss'; $(document).on('click', '.confirmedpassword .showOnClick a', function () { var $container = $('.showOnClickContainer', $(this).parent()); diff --git a/client/src/legacy/DatetimeField.js b/client/src/legacy/DatetimeField.js index b1feb2c02..92dee8a47 100644 --- a/client/src/legacy/DatetimeField.js +++ b/client/src/legacy/DatetimeField.js @@ -6,7 +6,7 @@ import i18n from 'i18n'; import moment from 'moment'; import modernizr from 'modernizr'; -require('../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'); +import '../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'; jQuery.entwine('ss', ($) => { $('input[type=datetime-local]').entwine({ diff --git a/client/src/legacy/GridField.js b/client/src/legacy/GridField.js index adef98cc0..2705a5773 100644 --- a/client/src/legacy/GridField.js +++ b/client/src/legacy/GridField.js @@ -2,15 +2,16 @@ import $ from 'jquery'; import i18n from 'i18n'; import React from 'react'; import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import Search from 'components/Search/Search.js'; import { schemaMerge } from 'lib/schemaFieldValues'; import { loadComponent } from 'lib/Injector'; -require('../../../thirdparty/jquery-ui/jquery-ui.js'); -require('../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'); +import '../../../thirdparty/jquery-ui/jquery-ui.js'; +import '../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'; // TODO Enable once https://github.com/webpack/extract-text-webpack-plugin/issues/179 is resolved. Included in bundle.scss for now. -// require('../styles/legacy/GridField.scss'); +// import '../styles/legacy/GridField.scss'; $.entwine('ss', function($) { $('.grid-field').entwine({ @@ -218,9 +219,9 @@ $.entwine('ss', function($) { * Function buildURLString splits string to "key => value" array * and replaces values for existing keys with the new value * and returns string with unique keys only - * - * @param {string} url - * @returns string + * + * @param {string} url + * @returns string */ buildURLString: function(url) { const link = [window.location.origin, url].join('/'); @@ -249,6 +250,7 @@ $.entwine('ss', function($) { Timer: null, Component: null, Actions: null, + ReactRoot: null, onmatch() { this._super(); @@ -275,10 +277,10 @@ $.entwine('ss', function($) { onunmatch() { this._super(); - // solves errors given by ReactDOM "no matched root found" error. - const container = this[0]; - if (container) { - ReactDOM.unmountComponentAtNode(container); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); } const actions = this.getActions(); @@ -296,10 +298,14 @@ $.entwine('ss', function($) { const GridFieldActions = this.getComponent(); // TODO: rework entwine so that react has control of holder - ReactDOM.render( - , - this[0] + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( + ); + this.setReactRoot(root); }, }) @@ -631,6 +637,7 @@ $.entwine('ss', function($) { $('.js-injector-boot .grid-field .grid-field__search-holder').entwine({ Component: null, + ReactRoot: null, onmatch() { this._super(); @@ -651,10 +658,10 @@ $.entwine('ss', function($) { onunmatch() { this._super(); - // solves errors given by ReactDOM "no matched root found" error. - const container = this[0]; - if (container) { - ReactDOM.unmountComponentAtNode(container); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); } }, @@ -722,7 +729,11 @@ $.entwine('ss', function($) { const handleSearch = (data) => this.search(data); const idName = String(props.gridfield).replace(/\-/g, '.'); - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( , - this[0] + /> ); + this.setReactRoot(root); }, }); @@ -798,7 +809,7 @@ $.entwine('ss', function($) { const gridField = $(this).getGridField(); const successCallback = function() { gridField.keepStateInHistory(); - }; + }; gridField.reload({ data: ajaxData }, successCallback); diff --git a/client/src/legacy/LeftAndMain.MobileMenuToggle.js b/client/src/legacy/LeftAndMain.MobileMenuToggle.js index 4cd402ccb..815d32392 100644 --- a/client/src/legacy/LeftAndMain.MobileMenuToggle.js +++ b/client/src/legacy/LeftAndMain.MobileMenuToggle.js @@ -1,7 +1,7 @@ import $ from 'jquery'; import MobileMenuToggleContainer from 'components/MobileMenuToggle/MobileMenuToggleContainer'; import { closeMobileMenu } from 'state/mobileMenu/MobileMenuActions'; -import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; $.entwine('ss', function($){ @@ -10,10 +10,8 @@ $.entwine('ss', function($){ onmatch: function() { const menuToggleWrapper = $('.cms-mobile-menu-toggle-wrapper'); if (menuToggleWrapper.length > 0) { - ReactDOM.render( - , - menuToggleWrapper[0] - ); + const root = createRoot(menuToggleWrapper[0]); + root.render(); } const store = window.ss.store; diff --git a/client/src/legacy/LeftAndMain.js b/client/src/legacy/LeftAndMain.js index 7a608bf79..3d5f2d20c 100644 --- a/client/src/legacy/LeftAndMain.js +++ b/client/src/legacy/LeftAndMain.js @@ -4,13 +4,14 @@ import $ from 'jquery'; import React from 'react'; import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import IframeDialog from 'components/IframeDialog/IframeDialog'; import Search from 'components/Search/Search'; import Loading from 'components/Loading/Loading'; import { schemaMerge } from 'lib/schemaFieldValues'; import { loadComponent } from 'lib/Injector'; -require('../legacy/ssui.core.js'); +import '../legacy/ssui.core.js'; $.noConflict(); @@ -1045,7 +1046,11 @@ $.entwine('ss', function($) { BackURL: window.location.href, }); - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( , - this[0] + /> ); }, @@ -1081,20 +1085,24 @@ $.entwine('ss', function($) { * like the breadcrumbs showing unnecessary loading status. */ $('form.loading,.cms-content.loading,.cms-content-fields.loading,.cms-content-view.loading,.ss-gridfield-item.loading').entwine({ + ReactRoot: null, onmatch: function() { this._super(); const container = $(''); this.append(container); - ReactDOM.render( - , - container[0] - ); + const root = createRoot(container[0]); + root.render(); + this.setReactRoot(root); }, onunmatch: function() { this._super(); const container = this.find('.cms-loading-container'); if (container && container.length) { - ReactDOM.unmountComponentAtNode(container[0]); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); + } container.remove(); } } @@ -1447,6 +1455,7 @@ $.entwine('ss', function($) { $('.js-injector-boot .search-holder').entwine({ Component: null, + ReactRoot: null, onmatch() { this._super(); @@ -1466,10 +1475,10 @@ $.entwine('ss', function($) { onunmatch() { this._super(); - // solves errors given by ReactDOM "no matched root found" error. - const container = this[0]; - if (container) { - ReactDOM.unmountComponentAtNode(container); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); } }, @@ -1523,7 +1532,11 @@ $.entwine('ss', function($) { const handleSearch = (data) => this.search(data); const narrowView = this.closest('.cms-content-tools').attr('id') === 'cms-content-tools-CMSMain'; - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( , - this[0] + /> ); + this.setReactRoot(root); }, }); }); diff --git a/client/src/legacy/ModelAdmin.js b/client/src/legacy/ModelAdmin.js index 170a6707d..3aef908f5 100644 --- a/client/src/legacy/ModelAdmin.js +++ b/client/src/legacy/ModelAdmin.js @@ -3,7 +3,7 @@ */ import $ from 'jquery'; -require('./LeftAndMain.js'); +import './LeftAndMain.js'; $.entwine('ss', function($){ $('.cms-content-tools #Form_SearchForm').entwine({ diff --git a/client/src/legacy/PermissionCheckboxSetField.js b/client/src/legacy/PermissionCheckboxSetField.js index a81bc99b5..c28dea047 100644 --- a/client/src/legacy/PermissionCheckboxSetField.js +++ b/client/src/legacy/PermissionCheckboxSetField.js @@ -1,7 +1,7 @@ import $ from 'jquery'; // TODO Enable once https://github.com/webpack/extract-text-webpack-plugin/issues/179 is resolved. Included in bundle.scss for now. -// require('../styles/legacy/CheckboxSetField.scss'); +// import '../styles/legacy/CheckboxSetField.scss'; $.entwine('ss', function($){ /** diff --git a/client/src/legacy/SecurityAdmin.js b/client/src/legacy/SecurityAdmin.js index a382e8f63..504918d39 100644 --- a/client/src/legacy/SecurityAdmin.js +++ b/client/src/legacy/SecurityAdmin.js @@ -3,8 +3,8 @@ */ import $ from 'jquery'; -require('./LeftAndMain.js'); -require('./PermissionCheckboxSetField.js'); +import './LeftAndMain.js'; +import './PermissionCheckboxSetField.js'; var refreshAfterImport = function(e) { // Check for a message , an indication that the form has been submitted. diff --git a/client/src/legacy/SelectionGroup.js b/client/src/legacy/SelectionGroup.js index bfb31f692..f8af79976 100644 --- a/client/src/legacy/SelectionGroup.js +++ b/client/src/legacy/SelectionGroup.js @@ -1,7 +1,7 @@ import $ from 'jquery'; // TODO Enable once https://github.com/webpack/extract-text-webpack-plugin/issues/179 is resolved. Included in bundle.scss for now. -// require('../styles/legacy/SelectionGroup.scss'); +// import '../styles/legacy/SelectionGroup.scss'; $(document).ready(function() { diff --git a/client/src/legacy/TabSet.js b/client/src/legacy/TabSet.js index 965b0300d..d6390d1ee 100644 --- a/client/src/legacy/TabSet.js +++ b/client/src/legacy/TabSet.js @@ -1,11 +1,11 @@ import $ from 'jquery'; -require('../../../thirdparty/jquery-ui/jquery-ui.js'); -require('../../../thirdparty/jquery-cookie/jquery.cookie.js'); -require('../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'); +import '../../../thirdparty/jquery-ui/jquery-ui.js'; +import '../../../thirdparty/jquery-cookie/jquery.cookie.js'; +import '../../../thirdparty/jquery-entwine/dist/jquery.entwine-dist.js'; // TODO Enable once https://github.com/webpack/extract-text-webpack-plugin/issues/179 is resolved. Included in bundle.scss for now. -// require('../../../thirdparty/jquery-ui-themes/smoothness/jquery-ui.css'); +// import '../../../thirdparty/jquery-ui-themes/smoothness/jquery-ui.css'; $.entwine('ss', function($){ $('.ss-tabset, .cms-tabset').entwine({ diff --git a/client/src/legacy/TinyMCE_sslink-email.js b/client/src/legacy/TinyMCE_sslink-email.js index 62e182ff2..7ae58d049 100644 --- a/client/src/legacy/TinyMCE_sslink-email.js +++ b/client/src/legacy/TinyMCE_sslink-email.js @@ -2,7 +2,7 @@ import i18n from 'i18n'; import TinyMCEActionRegistrar from 'lib/TinyMCEActionRegistrar'; import React from 'react'; -import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import jQuery from 'jquery'; import { createInsertLinkModal } from 'containers/InsertLinkModal/InsertLinkModal'; import { loadComponent } from 'lib/Injector'; @@ -66,7 +66,11 @@ jQuery.entwine('ss', ($) => { const requireLinkText = tagName !== 'A' && selectionContent.trim() === ''; // create/update the react component - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( { fileAttributes={attrs} identifier="Admin.InsertLinkEmailModal" requireLinkText={requireLinkText} - />, - this[0] + /> ); }, diff --git a/client/src/legacy/TinyMCE_sslink-external.js b/client/src/legacy/TinyMCE_sslink-external.js index 4ae80c7a5..fae4caa9e 100644 --- a/client/src/legacy/TinyMCE_sslink-external.js +++ b/client/src/legacy/TinyMCE_sslink-external.js @@ -2,7 +2,7 @@ import i18n from 'i18n'; import TinyMCEActionRegistrar from 'lib/TinyMCEActionRegistrar'; import React from 'react'; -import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import jQuery from 'jquery'; import { createInsertLinkModal } from 'containers/InsertLinkModal/InsertLinkModal'; import { loadComponent } from 'lib/Injector'; @@ -61,7 +61,11 @@ jQuery.entwine('ss', ($) => { const requireLinkText = tagName !== 'A' && selectionContent.trim() === ''; // create/update the react component - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( { fileAttributes={attrs} identifier="Admin.InsertLinkExternalModal" requireLinkText={requireLinkText} - />, - this[0] + /> ); }, diff --git a/client/src/legacy/ToastsContainer.js b/client/src/legacy/ToastsContainer.js index 71d83a99a..84147040c 100644 --- a/client/src/legacy/ToastsContainer.js +++ b/client/src/legacy/ToastsContainer.js @@ -1,7 +1,7 @@ /* global window */ import jQuery from 'jquery'; import React from 'react'; -import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import Injector, { loadComponent } from 'lib/Injector'; import { display } from 'state/toasts/ToastsActions'; @@ -15,7 +15,8 @@ jQuery.entwine('ss', ($) => { onmatch() { const container = $(''); this.append(container); - ReactDOM.render(, container[0]); + const root = createRoot(container[0]); + root.render(); }, }); }); diff --git a/client/src/legacy/ToggleCompositeField.js b/client/src/legacy/ToggleCompositeField.js index 8a59ee04f..062a90a62 100644 --- a/client/src/legacy/ToggleCompositeField.js +++ b/client/src/legacy/ToggleCompositeField.js @@ -1,10 +1,10 @@ import $ from 'jquery'; // entwine also required, but can't be included more than once without error -require('../../../thirdparty/jquery-ui/jquery-ui.js'); +import '../../../thirdparty/jquery-ui/jquery-ui.js'; // TODO Enable once https://github.com/webpack/extract-text-webpack-plugin/issues/179 is resolved. Included in bundle.scss for now. -// require('../../../thirdparty/jquery-ui-themes/smoothness/jquery-ui.css'); +// import '../../../thirdparty/jquery-ui-themes/smoothness/jquery-ui.css'; $.entwine('ss', function($){ $('.ss-toggle').entwine({ diff --git a/client/src/legacy/TreeDropdownField/TreeDropdownFieldEntwine.js b/client/src/legacy/TreeDropdownField/TreeDropdownFieldEntwine.js index f380f34fd..46208cf7e 100644 --- a/client/src/legacy/TreeDropdownField/TreeDropdownFieldEntwine.js +++ b/client/src/legacy/TreeDropdownField/TreeDropdownFieldEntwine.js @@ -2,6 +2,7 @@ import jQuery from 'jquery'; import React from 'react'; import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import { schemaMerge } from 'lib/schemaFieldValues'; import { MULTI_EMPTY_VALUE } from 'components/TreeDropdownField/TreeDropdownField'; import { loadComponent } from 'lib/Injector'; @@ -11,6 +12,7 @@ jQuery.entwine('ss', ($) => { Value: null, Timer: null, Component: null, + ReactRoot: null, onmatch() { this._super(); @@ -42,10 +44,10 @@ jQuery.entwine('ss', ($) => { onunmatch() { this._super(); - // solves errors given by ReactDOM "no matched root found" error. - const container = this[0]; - if (container) { - ReactDOM.unmountComponentAtNode(container); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); } }, @@ -67,15 +69,19 @@ jQuery.entwine('ss', ($) => { const TreeDropdownField = this.getComponent(); // TODO: rework entwine so that react has control of holder - ReactDOM.render( + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render( , - this[0] + /> ); + this.setReactRoot(root); }, getAttributes() { diff --git a/client/src/legacy/UsedOnTable/UsedOnTableEntwine.js b/client/src/legacy/UsedOnTable/UsedOnTableEntwine.js index 82ef7a75c..0d6ea9070 100644 --- a/client/src/legacy/UsedOnTable/UsedOnTableEntwine.js +++ b/client/src/legacy/UsedOnTable/UsedOnTableEntwine.js @@ -2,6 +2,7 @@ import jQuery from 'jquery'; import React from 'react'; import ReactDOM from 'react-dom'; +import { createRoot } from 'react-dom/client'; import { schemaMerge } from 'lib/schemaFieldValues'; import { loadComponent } from 'lib/Injector'; @@ -9,6 +10,7 @@ jQuery.entwine('ss', ($) => { $('.js-injector-boot .used-on__polyfill-holder').entwine({ Timer: null, Component: null, + ReactRoot: null, onmatch() { this._super(); @@ -26,10 +28,10 @@ jQuery.entwine('ss', ($) => { onunmatch() { this._super(); - // solves errors given by ReactDOM "no matched root found" error. - const container = this[0]; - if (container) { - ReactDOM.unmountComponentAtNode(container); + const root = this.getReactRoot(); + if (root) { + root.unmount(); + this.setReactRoot(null); } }, @@ -39,10 +41,12 @@ jQuery.entwine('ss', ($) => { const UsedOnTable = this.getComponent(); // TODO: rework entwine so that react has control of holder - ReactDOM.render( - , - this[0] - ); + let root = this.getReactRoot(); + if (!root) { + root = createRoot(this[0]); + } + root.render(); + this.setReactRoot(root); }, getAttributes() { diff --git a/client/src/legacy/ssui.core.js b/client/src/legacy/ssui.core.js index 81d055062..3488cda08 100644 --- a/client/src/legacy/ssui.core.js +++ b/client/src/legacy/ssui.core.js @@ -1,6 +1,6 @@ import $ from 'jquery'; -require('../../../thirdparty/jquery-ui/jquery-ui.js'); +import '../../../thirdparty/jquery-ui/jquery-ui.js'; /** * Extends jQueryUI dialog with iframe abilities (and related resizing logic), diff --git a/client/src/lib/dependency-injection/ApolloGraphqlProxy.js b/client/src/lib/dependency-injection/ApolloGraphqlProxy.js index c69d4e6de..1701fce2f 100644 --- a/client/src/lib/dependency-injection/ApolloGraphqlProxy.js +++ b/client/src/lib/dependency-injection/ApolloGraphqlProxy.js @@ -1,4 +1,4 @@ -import { graphql } from 'react-apollo'; +import { graphql } from '@apollo/client/react/hoc'; // TODO If this doesn't work, checkout https://www.apollographql.com/docs/react/migrating/apollo-client-3-migration/#react-apollo class ApolloGraphqlProxy { /** diff --git a/client/src/lib/dependency-injection/loadComponent.js b/client/src/lib/dependency-injection/loadComponent.js index aa517aa7d..c7c624f84 100644 --- a/client/src/lib/dependency-injection/loadComponent.js +++ b/client/src/lib/dependency-injection/loadComponent.js @@ -1,11 +1,11 @@ /* global window */ import React, { Component } from 'react'; import { Provider } from 'react-redux'; -import { ApolloProvider } from 'react-apollo'; +import { ApolloProvider } from '@apollo/client'; // TODO If this doesn't work, checkout https://www.apollographql.com/docs/react/migrating/apollo-client-3-migration/#react-apollo +import NotFoundComponent from 'components/NotFoundComponent/NotFoundComponent'; import provideInjector from './provideInjector'; import withInjector from './withInjector'; import Injector from './Container'; -import NotFoundComponent from 'components/NotFoundComponent/NotFoundComponent'; import contextType from './injectorContext'; /** diff --git a/client/src/lib/dependency-injection/tests/loadComponent-test.js b/client/src/lib/dependency-injection/tests/loadComponent-test.js index 3f5526cca..21bb3136a 100644 --- a/client/src/lib/dependency-injection/tests/loadComponent-test.js +++ b/client/src/lib/dependency-injection/tests/loadComponent-test.js @@ -46,7 +46,7 @@ jest.mock('../provideInjector', () => function mockInjector(Injected) { }); jest.mock('../Container', () => ({ ready: (callback) => { callback(); } })); -jest.mock('react-apollo', () => ({ +jest.mock('@apollo/client', () => ({ // TODO If this doesn't work, checkout https://www.apollographql.com/docs/react/migrating/apollo-client-3-migration/#react-apollo ApolloProvider: ({ children }) => {children}, })); diff --git a/client/src/lib/renderReactRoutes.js b/client/src/lib/renderReactRoutes.js new file mode 100644 index 000000000..aec75c5ac --- /dev/null +++ b/client/src/lib/renderReactRoutes.js @@ -0,0 +1,19 @@ +import React from 'react'; +import { Routes, Route } from 'react-router'; + +const renderReactRoutes = (routes) => (routes ? ( + + {routes.map((route) => ( + ( + // eslint-disable-next-line react/jsx-props-no-spreading + + )} + /> + ))} + +) : null); + +export default renderReactRoutes; diff --git a/client/src/styles/browser-warning.scss b/client/src/styles/browser-warning.scss deleted file mode 100644 index 34c15500f..000000000 --- a/client/src/styles/browser-warning.scss +++ /dev/null @@ -1,10 +0,0 @@ -// For IE version 9 and below. These browsers doesn't handle large -// resource files so need to break browser detection and warning code into -// its own file - -// Variables -@import "variables"; -@import "~bootstrap/scss/mixins"; - -// Components -@import "../components/BrowserWarning/BrowserWarning"; diff --git a/client/src/styles/legacy/_actionTabs.scss b/client/src/styles/legacy/_actionTabs.scss index 2ffc0f3df..d27faac73 100644 --- a/client/src/styles/legacy/_actionTabs.scss +++ b/client/src/styles/legacy/_actionTabs.scss @@ -206,13 +206,12 @@ $border: 1px solid darken(#D9D9D9, 15%); border: none; } &.loading { - background: transparent url("../images/network-save.gif") no-repeat ($spacer / 2) center; + background: transparent url('../../images/network-save.gif') no-repeat ($spacer / 2) center; .ui-button-text { padding-left: $spacer /* icon */ + ($spacer/4); } } } - } } } diff --git a/client/src/styles/legacy/_forms.scss b/client/src/styles/legacy/_forms.scss index 0e74fd6c3..3e6fb9c75 100644 --- a/client/src/styles/legacy/_forms.scss +++ b/client/src/styles/legacy/_forms.scss @@ -307,7 +307,7 @@ input.loading, button.loading, input.ui-state-default.loading, .ui-widget-content input.ui-state-default.loading, .ui-widget-header input.ui-state-default.loading { .ui-icon { - background: transparent url("../images/network-save.gif") no-repeat 0 0; + background: transparent url('../../images/network-save.gif') no-repeat 0 0; } color: $body-color; @@ -317,7 +317,7 @@ input.ui-state-default.loading, .ui-widget-content input.ui-state-default.loadin input.loading, button.loading { &.ss-ui-action-constructive .ui-icon { - background: transparent url("../images/network-save-constructive.gif") no-repeat 0 0; + background: transparent url('../../images/network-save-constructive.gif') no-repeat 0 0; } } @@ -473,7 +473,7 @@ button.loading { } .ss-ui-loading-icon { - background: url("../images/network-save.gif") no-repeat; + background: url('../../images/network-save.gif') no-repeat; display: block; width: 16px; height: 16px; diff --git a/client/src/styles/legacy/_retina.scss b/client/src/styles/legacy/_retina.scss index 08c385467..ab81076eb 100644 --- a/client/src/styles/legacy/_retina.scss +++ b/client/src/styles/legacy/_retina.scss @@ -17,7 +17,7 @@ .tree-holder, .cms-tree { &.jstree-apple { ins { - background-image: url(../images/sitetree_ss_default_icons@2x.png); + background-image: url('../../images/sitetree_ss_default_icons@2x.png'); background-size: 108px 72px; } } diff --git a/client/src/styles/legacy/_sprity.scss b/client/src/styles/legacy/_sprity.scss index 04b53953d..31289bf64 100644 --- a/client/src/styles/legacy/_sprity.scss +++ b/client/src/styles/legacy/_sprity.scss @@ -37,5 +37,5 @@ $sprites-32x32-2x-dialog-close-over: -0px -506px 60px 60px; } @mixin icon-sprites($dimension) { - background-image: url('../images/sprites/sprite-sprites-#{$dimension}.png'); + background-image: url('../../images/sprites/sprite-sprites-#{$dimension}.png'); } diff --git a/client/src/styles/legacy/_style.scss b/client/src/styles/legacy/_style.scss index c07064d2f..842c1881d 100644 --- a/client/src/styles/legacy/_style.scss +++ b/client/src/styles/legacy/_style.scss @@ -1405,7 +1405,7 @@ form.member-profile-form { text-indent: -9999em; display: inline-block; width: 20px; - background: url(../images/question.png) no-repeat 0px 0px; + background: url('../../images/question.png') no-repeat 0px 0px; } } @@ -1597,7 +1597,7 @@ form.member-profile-form { font-size: $font-size-root +1; padding: 0; border: 0; - background: transparent url(../images/textures/cms_content_header.png) repeat; + background: transparent url('../../images/textures/cms_content_header.png') repeat; box-shadow: $shadow-level-4 0 0 ($spacer / 4) inset; .ui-dialog-title { @@ -1610,7 +1610,7 @@ form.member-profile-form { overflow: auto; // TODO Replace with proper $.layout grid &.loading { - background-image: url("../images/spinner.gif"); + background-image: url('../../images/spinner.gif'); background-position: 50% 50%; background-repeat: no-repeat; } @@ -1652,7 +1652,7 @@ form.member-profile-form { } &.loading { - background-image: url("../images/spinner.gif"); + background-image: url('../../images/spinner.gif'); background-position: 50% 50%; background-repeat: no-repeat; } diff --git a/client/src/styles/legacy/_tree.scss b/client/src/styles/legacy/_tree.scss index 8c40a58d7..6cf5a76fe 100644 --- a/client/src/styles/legacy/_tree.scss +++ b/client/src/styles/legacy/_tree.scss @@ -489,7 +489,7 @@ } ins { background-color: transparent; - background-image: url(../images/sitetree_ss_default_icons.png); + background-image: url('../../images/sitetree_ss_default_icons.png'); &.font-icon-drag-handle { background: none; @@ -743,7 +743,7 @@ a .jstree-pageicon { &.jstree-loading { li#record-0 > .jstree-icon { - background: url(../images/throbber.gif) top left no-repeat; + background: url('../../images/throbber.gif') top left no-repeat; } } @@ -754,7 +754,7 @@ a .jstree-pageicon { background-image: none !important; } .jstree-pageicon { - background: url(../images/throbber.gif) top left no-repeat; + background: url('../../images/throbber.gif') top left no-repeat; } } } diff --git a/client/src/styles/legacy/_uitheme.scss b/client/src/styles/legacy/_uitheme.scss index 783d633cb..616ad5dfa 100644 --- a/client/src/styles/legacy/_uitheme.scss +++ b/client/src/styles/legacy/_uitheme.scss @@ -77,7 +77,7 @@ /** sorry about the !important but the specificity of other selectors mandates it over writing out very specific selectors **/ &-loading { - background-image: url(../images/throbber.gif) !important; + background-image: url('../../images/throbber.gif') !important; background-position: 97% center !important; background-repeat: no-repeat !important; background-size: auto !important; diff --git a/code/LeftAndMain.php b/code/LeftAndMain.php index bc72fa42f..564369d20 100644 --- a/code/LeftAndMain.php +++ b/code/LeftAndMain.php @@ -684,7 +684,7 @@ protected function init() Requirements::javascript('silverstripe/admin: thirdparty/bootstrap/js/dist/collapse.js'); Requirements::javascript('silverstripe/admin: thirdparty/bootstrap/js/dist/tooltip.js'); Requirements::customScript( - "window.jQuery('body').tooltip({ selector: '[data-toggle=tooltip]' });", + "window.jQuery('body').tooltip({ selector: '[data-bs-toggle=tooltip]' });", 'bootstrap.tooltip-boot' ); diff --git a/templates/SilverStripe/Admin/Includes/BackLink_Button.ss b/templates/SilverStripe/Admin/Includes/BackLink_Button.ss index 2253fda11..2c94243da 100644 --- a/templates/SilverStripe/Admin/Includes/BackLink_Button.ss +++ b/templates/SilverStripe/Admin/Includes/BackLink_Button.ss @@ -1,5 +1,5 @@ <% if $Backlink %> - <%t SilverStripe\Admin\LeftAndMain.NavigateUp "Navigate up a folder" %> + <%t SilverStripe\Admin\LeftAndMain.NavigateUp "Navigate up a folder" %> <% end_if %> diff --git a/templates/SilverStripe/Admin/Includes/BrowserWarning.ss b/templates/SilverStripe/Admin/Includes/BrowserWarning.ss deleted file mode 100644 index 0ced12b62..000000000 --- a/templates/SilverStripe/Admin/Includes/BrowserWarning.ss +++ /dev/null @@ -1,34 +0,0 @@ -<% require css('silverstripe/admin: client/dist/styles/browser-warning.css') %> - - - - - SilverStripe - <%t SilverStripe\Admin\LeftAndMain.BROWSER_WARNING_HEADING 'Unsupported browser' %> - - - <%t SilverStripe\Admin\LeftAndMain.BROWSER_WARNING_TEXT 'You're using a web browser that's not supported by SilverStripe.Please upgrade to one of the following options.' %> - - - Internet Explorer(11+) - - Mozilla Firefox - - Google Chrome - - - - - -<% require javascript('silverstripe/admin: client/dist/js/browserWarning.js') %> - - diff --git a/templates/SilverStripe/Admin/Includes/LeftAndMain_EditForm.ss b/templates/SilverStripe/Admin/Includes/LeftAndMain_EditForm.ss index 86d4bd6c0..fb3584f05 100644 --- a/templates/SilverStripe/Admin/Includes/LeftAndMain_EditForm.ss +++ b/templates/SilverStripe/Admin/Includes/LeftAndMain_EditForm.ss @@ -57,9 +57,9 @@ <% if $hasExtraClass('cms-previewable') %> <% if $Actions.last.id == 'Form_ItemEditForm_RightGroup' %> - <% include SilverStripe\\Admin\\LeftAndMain_ViewModeSelector SelectID="preview-mode-dropdown-in-content", ExtraClass="ml-0" %> + <% include SilverStripe\\Admin\\LeftAndMain_ViewModeSelector SelectID="preview-mode-dropdown-in-content", ExtraClass="ms-0" %> <% else %> - <% include SilverStripe\\Admin\\LeftAndMain_ViewModeSelector SelectID="preview-mode-dropdown-in-content", ExtraClass="ml-auto" %> + <% include SilverStripe\\Admin\\LeftAndMain_ViewModeSelector SelectID="preview-mode-dropdown-in-content", ExtraClass="ms-auto" %> <% end_if %> <% end_if %> diff --git a/templates/SilverStripe/Admin/Includes/LeftAndMain_MenuToggle.ss b/templates/SilverStripe/Admin/Includes/LeftAndMain_MenuToggle.ss index 0daff7dc2..30e037067 100644 --- a/templates/SilverStripe/Admin/Includes/LeftAndMain_MenuToggle.ss +++ b/templates/SilverStripe/Admin/Includes/LeftAndMain_MenuToggle.ss @@ -4,7 +4,7 @@ $ApplicationName - + <% if $CMSVersionNumber %> - $CMSVersion)">$CMSVersionNumber <% end_if %> @@ -27,5 +27,5 @@ <%t SilverStripe\Admin\LeftAndMain.MenuToggleAuto "Auto" %> -" aria-label="<%t SilverStripe\Admin\LeftAndMain.ExpandPanel "Expand panel" %>">» -" aria-label="<%t SilverStripe\Admin\LeftAndMain.CollapsePanel "Collapse panel" %>">« +" aria-label="<%t SilverStripe\Admin\LeftAndMain.ExpandPanel "Expand panel" %>">» +" aria-label="<%t SilverStripe\Admin\LeftAndMain.CollapsePanel "Collapse panel" %>">« diff --git a/templates/SilverStripe/Admin/LeftAndMain.ss b/templates/SilverStripe/Admin/LeftAndMain.ss index 90e4c8fb7..1bd4a59a1 100644 --- a/templates/SilverStripe/Admin/LeftAndMain.ss +++ b/templates/SilverStripe/Admin/LeftAndMain.ss @@ -16,7 +16,5 @@ $Content $PreviewPanel - - <% include SilverStripe\\Admin\\BrowserWarning %>
SilverStripe
- <%t SilverStripe\Admin\LeftAndMain.BROWSER_WARNING_TEXT 'You're using a web browser that's not supported by SilverStripe.Please upgrade to one of the following options.' %> -