Skip to content

Commit a122eeb

Browse files
committed
add cmsEditorBase url param
To make this work I had to relax the origin checks in cms-editor. This seems safe to do, this editor is basically made to be embedded on any page in an iframe. I also added a configuration for the serve package. By default it redirects *.html to just *. And when it does this it loses the url parameters. This breaks the embedding of cms-editor.html used by the iframe-control. Testing with serve is useful to make sure the admin.html file is working properly without having to deploy it.
1 parent 78e7d66 commit a122eeb

8 files changed

+36
-22
lines changed

cms/src/iframe-control.tsx

+9-5
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,21 @@ import { defaultCurriculumBranch } from "./cms-constants";
99
import "./iframe-control.scss";
1010

1111
(window as any).DISABLE_FIREBASE_SYNC = true;
12+
13+
const cmsEditorBase = urlParams.cmsEditorBase ?? ".";
14+
// the URL is relative to the current url of the CMS
15+
// If the cmsEditorBase is an absolute url then the current url will be ignored
16+
const cmsEditorBaseURL = new URL(cmsEditorBase, window.location.href);
17+
const validOrigin = cmsEditorBaseURL.origin;
18+
1219
interface IState {
1320
initialValue?: string;
14-
validOrigin: string;
1521
}
1622
export class IframeControl extends React.Component<CmsWidgetControlProps, IState> {
1723
constructor(props: CmsWidgetControlProps) {
1824
super(props);
1925
this.state = {
2026
initialValue: this.getValue(),
21-
validOrigin: `${window.location.protocol}//${window.location.host}`
2227
};
2328
}
2429

@@ -36,7 +41,7 @@ export class IframeControl extends React.Component<CmsWidgetControlProps, IState
3641
};
3742

3843
sendInitialValueToEditor() {
39-
const { initialValue, validOrigin } = this.state;
44+
const { initialValue } = this.state;
4045
const iframedEditor = document.getElementById("editor") as HTMLIFrameElement;
4146
if (iframedEditor.contentWindow) {
4247
iframedEditor.contentWindow.postMessage(
@@ -48,7 +53,6 @@ export class IframeControl extends React.Component<CmsWidgetControlProps, IState
4853
}
4954

5055
isValidMessageEvent = (event: MessageEvent) => {
51-
const { validOrigin } = this.state;
5256
return event.data.type === "updateContent" &&
5357
event.data.content &&
5458
event.origin === validOrigin;
@@ -62,7 +66,7 @@ export class IframeControl extends React.Component<CmsWidgetControlProps, IState
6266

6367
render() {
6468
const curriculumBranch = urlParams.curriculumBranch ?? defaultCurriculumBranch;
65-
const iframeBaseUrl = `./cms-editor.html?curriculumBranch=${curriculumBranch}`;
69+
const iframeBaseUrl = `${cmsEditorBase}/cms-editor.html?curriculumBranch=${curriculumBranch}`;
6670
const iframeUrl = urlParams.unit
6771
? `${iframeBaseUrl}&unit=${urlParams.unit}`
6872
: iframeBaseUrl;

docs/cms.md

+13-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
1-
CLUE includes a CMS which can be accessed at `/admin.html`. It uses [Decap CMS](https://decapcms.org/). It is configured to edit the content in the `clue-curriculum` repository. It can be configured with the following URL params:
1+
CLUE includes a CMS which can be accessed at `/admin.html` in a production build. It uses [Decap CMS](https://decapcms.org/). It is configured to edit the content in the `clue-curriculum` repository. It is best to always pass the CMS a `unit` url parameter. Otherwise it will download pretty much all of the files in `clue-curriculum`. With a `unit` parameter it will only download the files for that unit.
2+
3+
# URL Parameters
24
- **`curriculumBranch`** By default the CMS edits the `author` branch. You can change the branch by passing a different branch name to this parameter. The CMS will not create this branch for you. You'll need to create it yourself.
35
- **`unit`** By default the CMS displays all of the units at the same time. It is better to work with a single unit at a time by using the `unit` param. It should be passed the unit code. This limits what is displayed in the CMS and it also configures the media library to show the images from that unit.
6+
- **`cmsEditorBase`** By default the CMS will use an iframe pointed at `./cms-editor.html` to edit the CLUE documents. You can use this param to replace `.` with something else. This is useful for local testing (see below)
47
- **`localCMSBackend`** By default the CMS works with the `github` backend. This means even when doing local CLUE development the CMS will update the `clue-curriculum` repository directly. If you add the `localCMSBackend` parameter the CMS will attempt to work with a local git proxy running at localhost:8081. You can start this proxy with:
58

69
`cd ../clue-curriculum; npx netlify-cms-proxy-server`
@@ -44,9 +47,16 @@ This document editor is located in `/src/cms/`
4447
## Local development
4548
To work on the CMS locally you'll need to start both CLUE and the CMS:
4649
- start CLUE by running `npm run start` in the top level folder
47-
- start the CMS by running `npm run start` in the `/cms` folder
50+
- in a new terminal, open the `/cms` folder
51+
- make sure its dependencies are installed: `npm i`
52+
- start the CMS by running `npm run start`
53+
- open the CMS with `http://localhost:[cms_port]/?cmsEditorBase=http://localhost:[clue_port]&unit=[clue_unit_code]&curriculumBranch=[your own branch]`
54+
55+
Typically CLUE will be running on portal 8080 and the CMS will be running on 8081. In this case the url above would be `http://localhost:8081/?cmsEditorBase=http://localhost:8080&unit=[clue_unit_code]&curriculumBranch=[your own branch]`
56+
57+
With this approach you'll be editing the content in the `clue-curriculum` repository directly. By default this will use the `author` branch in the `clue-curriculum` repository. So you aren't making changes to the same branch as other people, you should make your own branch in that repository and pass it to the `curriculumBranch` parameter. You have to make this branch using your own git tools, the CMS cannot create branches itself.
4858

49-
TODO: Then you'll need to configure the CMS so it can find your the local `cms-editor.html` from the running CLUE. This can be done using a parameter to the CMS url. Without this parameter it will try to look in `./cms-editor.html` which will not work.
59+
If you want to have more local access to the curriculum you are editing, and you don't want to be updating the `clue-curriculum` repository, you can use the `localCMSBackend` parameter. See above for details on how to use this. Note the proxy needs to be on its own port. If it can't start on 8081 (because the cms is running there) then you'll need to configure it. See the "Configure the Decap CMS proxy server port number" section of this page: https://decapcms.org/docs/working-with-a-local-git-repository
5060

5161
## Remote build
5262
In the CI (github actions), the toplevel `npm run build` is used. This will build both CLUE and the CMS. And then it will copy the files from `/cms/dist` to `/dist/cms`. Additional it copyes the file `/cms/dist/admin.html` to `/dist/admin.html`. This admin.html file refers to its resources in the cms folder like `cms/admin.js` and `cms/admin.css`. This file is called `admin.html` because that was its original name, and now various authors have direct links to it. So we don't want to change that name.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@
8484
"migrate": "node ./migrations/migrate",
8585
"migrate:debug": "node --inspect-brk ./migrations/migrate",
8686
"postinstall": "patch-package",
87-
"serve": "npx serve -l 8080 dist",
87+
"serve": "npx serve -c ../serve.json -l 8080 dist",
8888
"serve:top-test": "npx http-server top-test",
8989
"start": "webpack-dev-server --hot",
9090
"start:secure": "webpack-dev-server --https --hot --cert ~/.localhost-ssl/localhost.pem --key ~/.localhost-ssl/localhost.key",

serve.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"cleanUrls": false
3+
}

src/cms/cms-editor.tsx

+2-3
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,9 @@ import { DocumentEditor } from "./document-editor";
55
import { DocumentModelType } from "../models/document/document";
66

77
let initialValue = undefined as DocumentModelType | undefined;
8-
const validOrigin = `${window.location.protocol}//${window.location.host}`;
98

109
(window as any).addEventListener("message", (event: MessageEvent) => {
11-
if (event.origin === validOrigin && event.data.initialValue) {
10+
if (event.data.initialValue) {
1211
initialValue = JSON.parse(event.data.initialValue);
1312
if (initialValue) {
1413
renderEditor();
@@ -20,7 +19,7 @@ const validOrigin = `${window.location.protocol}//${window.location.host}`;
2019

2120
const handleUpdateContent = (json: Record<string, any>) => {
2221
const stringifiedJson = JSON.stringify(json);
23-
window.parent.postMessage({ type: "updateContent", content: stringifiedJson }, validOrigin);
22+
window.parent.postMessage({ type: "updateContent", content: stringifiedJson }, "*");
2423
};
2524

2625
const renderEditor = () => {

src/cms/document-editor.tsx

+3
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,9 @@ export class DocumentEditor extends React.Component<IProps, IState> {
5050

5151
// Update the widget's value whenever a change is made to the document's content
5252
this.disposer = onSnapshot(document, snapshot => {
53+
// FIXME: If the document has tiles at the end of it which aren't setup by the unit json,
54+
// then exportAsJson creates an invalid JSON file. It leaves a trailing comma after the
55+
// last known tile.
5356
const json = document.content?.exportAsJson({ includeTileIds: true });
5457
if (json) {
5558
const parsedJson = JSON.parse(json);

src/utilities/url-params.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,8 @@ export interface QueryParams {
9090
functions?: string; // "emulator" or host:port url
9191
// do not use persistentUI in some cy tests that rely on demo
9292
noPersistentUI?: boolean;
93+
// mouse sensor can be enabled for cypress drag and drop tests for dnd-kit
94+
mouseSensor?: boolean;
9395

9496
//
9597
// CMS options (admin.html)
@@ -99,8 +101,9 @@ export interface QueryParams {
99101
curriculumBranch?: string;
100102
// work with a local checkout of the curriculum instead of github
101103
localCMSBackend?: boolean;
102-
// mouse sensor can be enabled for cypress drag and drop tests for dnd-kit
103-
mouseSensor?: boolean;
104+
// change the location of the cms-editor.html used by iframe widget to edit
105+
// CLUE documents.
106+
cmsEditorBase?: string;
104107

105108
//
106109
// Standalone document editor options (doc-editor.html)

webpack.config.js

-8
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@ module.exports = (env, argv) => {
3737
// on the devServer start up time
3838
entry: {
3939
index: './src/index.tsx',
40-
admin: './cms/src/admin.tsx',
4140
'cms-editor': './src/cms/cms-editor.tsx',
4241
'doc-editor': './src/doc-editor.tsx'
4342
},
@@ -237,13 +236,6 @@ module.exports = (env, argv) => {
237236
filename: 'index.html',
238237
publicPath: '.',
239238
}),
240-
new HtmlWebpackPlugin({
241-
...baseHtmlPluginConfig,
242-
chunks: ['admin'],
243-
filename: 'admin.html',
244-
publicPath: '.',
245-
template: 'cms/src/admin.html'
246-
}),
247239
new HtmlWebpackPlugin({
248240
...baseHtmlPluginConfig,
249241
chunks: ['cms-editor'],

0 commit comments

Comments
 (0)