diff --git a/BREAKING_CHANGES.md b/BREAKING_CHANGES.md deleted file mode 100644 index 37767148..00000000 --- a/BREAKING_CHANGES.md +++ /dev/null @@ -1,5 +0,0 @@ -Breaking changes to be documented for v3 - -- gitea API config has changed. -- hidden removed as widget parameter -- events \ No newline at end of file diff --git a/packages/core/src/components/entry-editor/editor-control-pane/EditorControl.tsx b/packages/core/src/components/entry-editor/editor-control-pane/EditorControl.tsx index 25ca159a..3150620e 100644 --- a/packages/core/src/components/entry-editor/editor-control-pane/EditorControl.tsx +++ b/packages/core/src/components/entry-editor/editor-control-pane/EditorControl.tsx @@ -8,10 +8,6 @@ import { changeDraftField as changeDraftFieldAction, changeDraftFieldValidation, } from '@staticcms/core/actions/entries'; -import { - openMediaLibrary as openMediaLibraryAction, - removeInsertedMedia as removeInsertedMediaAction, -} from '@staticcms/core/actions/mediaLibrary'; import { query as queryAction } from '@staticcms/core/actions/search'; import useDebouncedCallback from '@staticcms/core/lib/hooks/useDebouncedCallback'; import useMemoCompare from '@staticcms/core/lib/hooks/useMemoCompare'; @@ -52,11 +48,8 @@ const EditorControl = ({ disabled = false, parentDuplicate = false, locale, - mediaPaths, - openMediaLibrary, parentPath, query, - removeInsertedMedia, t, value, forList = false, @@ -197,10 +190,7 @@ const EditorControl = ({ duplicate, label: getFieldLabel(field, t), locale, - mediaPaths, onChange: handleDebouncedChangeDraftField, - openMediaLibrary, - removeInsertedMedia, path, query, t, @@ -230,10 +220,7 @@ const EditorControl = ({ duplicate, t, locale, - mediaPaths, handleDebouncedChangeDraftField, - openMediaLibrary, - removeInsertedMedia, path, query, finalValue, @@ -271,7 +258,6 @@ function mapStateToProps(state: RootState, ownProps: EditorControlOwnProps) { return { ...ownProps, - mediaPaths: state.mediaLibrary.controlMedia, config: state.config, entry, collection, @@ -282,8 +268,6 @@ function mapStateToProps(state: RootState, ownProps: EditorControlOwnProps) { const mapDispatchToProps = { changeDraftField: changeDraftFieldAction, - openMediaLibrary: openMediaLibraryAction, - removeInsertedMedia: removeInsertedMediaAction, query: queryAction, }; diff --git a/packages/core/src/interface.ts b/packages/core/src/interface.ts index b1c08fad..bbca6359 100644 --- a/packages/core/src/interface.ts +++ b/packages/core/src/interface.ts @@ -316,13 +316,7 @@ export interface WidgetControlProps; onChange: (value: T | null | undefined) => void; - // @deprecated Use useMediaInsert instead - openMediaLibrary: EditorControlProps['openMediaLibrary']; - // @deprecated Use useMediaInsert instead - removeInsertedMedia: EditorControlProps['removeInsertedMedia']; i18n: I18nSettings | undefined; hasErrors: boolean; errors: FieldError[]; diff --git a/packages/core/test/data/widgets.mock.ts b/packages/core/test/data/widgets.mock.ts index 5aea2090..b19c06fb 100644 --- a/packages/core/test/data/widgets.mock.ts +++ b/packages/core/test/data/widgets.mock.ts @@ -60,7 +60,6 @@ export const createMockWidgetControlProps = < entry, value, path, - mediaPaths: {}, fieldsErrors, errors, hasErrors, @@ -74,8 +73,6 @@ export const createMockWidgetControlProps = < controlled: false, theme: 'light', onChange: jest.fn(), - openMediaLibrary: jest.fn(), - removeInsertedMedia: jest.fn(), query: jest.fn(), t: jest.fn(), ...extra, diff --git a/packages/docs/content/docs/beta-features.mdx b/packages/docs/content/docs/beta-features.mdx index 3fc0f231..dc0d16e9 100644 --- a/packages/docs/content/docs/beta-features.mdx +++ b/packages/docs/content/docs/beta-features.mdx @@ -81,54 +81,105 @@ Available transformations are `upper`, `lower`, `date('')`, `default('de ## Registering to CMS Events -You can execute a function when a specific CMS event occurs. +You can execute a function when a specific CMS event occurs. Supported events are `mounted`, `login`, `logout`, `change`, `preSave` and `postSave`. -Example usage: +### Mounted + +The `mounted` event handler fires once the CMS is fully loaded. ```javascript CMS.registerEventListener({ - name: 'postSave', - handler: ({ author, entry }) => console.info(JSON.stringify({ author, data: entry.data })), + name: 'mounted', + handler: () => { + // your code here + }, }); ``` -Supported events are `mounted`, `login`, `logout`, `change`, `preSave` and `postSave`. +### Login -### PreSave +The `login` event handler fires when a user logs into the CMS. -The `preSave` event handler can be used to modify the entry data like so: +```javascript +CMS.registerEventListener({ + name: 'login', + handler: ({ author: { login, name } }) => { + // your code here + }, +}); +``` + +### Logout + +The `logout` event handler fires when a user logs out of the CMS. + +```javascript +CMS.registerEventListener({ + name: 'logout', + handler: () => { + // your code here + }, +}); +``` + +### Pre Save + +The `preSave` event handler fires before the changes have been saved to your git backend, and can be used to modify the entry data like so: ```javascript CMS.registerEventListener({ name: 'preSave', - handler: ({ entry }) => { + collection: 'posts', + handler: ({ data: { entry } }) => { return { - ...entry, - data: { - ...entry.data, - title: 'new title', - }, + ...entry.data, + title: 'new title', }; }, }); ``` +### Post Save + +The `postSave` event handler fires after the changes have been saved to your git backend. + +```javascript +CMS.registerEventListener({ + name: 'postSave', + collection: 'posts', + handler: ({ data: { entry } }) => { + // your code here + }, +}); +``` + ### Change The `change` event handler must provide a field name, and can be used to modify the entry data like so: ```javascript -CMS.registerEventListener( - { - name: 'change', - handler: ({ entry }, { field }) => { - return 'newFieldValue'; - }, +CMS.registerEventListener({ + name: 'change', + collection: 'posts', + field: 'path.to.my.field', + handler: ({ data, collection, field }) => { + const currentValue = data.path.to.my.field; + + return { + ...data, + path: { + ...data.path, + to: { + ...data.path.to, + my: { + ...data.path.to.my, + field: `new${currentValue}` + } + } + } + }; }, - { - field: 'path.to.my.field', - }, -); +}); ``` ## i18n Support @@ -146,3 +197,9 @@ See [Gitea Backend](/docs/gitea-backend) for more information. Netlify Large Media allows you to store your media files outside of your git backend. This is helpful if you are trying to store large media files. See [Netlify Large Media](/docs/netlify-large-media) for more information. + +## Live Previews + +For react based projects, such as [NextJS](https://nestjs.com/), you can now utilize live preivews. + +See [Live Previews](/docs/live-previews) for more information. diff --git a/packages/docs/content/docs/gitea-backend.mdx b/packages/docs/content/docs/gitea-backend.mdx index 21d42587..617b84b2 100644 --- a/packages/docs/content/docs/gitea-backend.mdx +++ b/packages/docs/content/docs/gitea-backend.mdx @@ -9,7 +9,7 @@ beta: true For repositories stored on Gitea, the `gitea` backend allows CMS users to log in directly with their Gitea account. Note that all users must have push access to your content repository for this to work. -Please note that only Gitea **1.20** and upwards is supported due to API limitations in previous versions. +Please note that only Gitea `v1.20` and upwards is supported due to API limitations in previous versions. ## Authentication diff --git a/packages/docs/content/docs/live-previews.mdx b/packages/docs/content/docs/live-previews.mdx new file mode 100644 index 00000000..6ba96676 --- /dev/null +++ b/packages/docs/content/docs/live-previews.mdx @@ -0,0 +1,28 @@ +--- +group: Customization +title: Live Previews +beta: true +weight: 51 +--- + +Live previews are an experimental feature introduced in `v3` to provide updates on a live page of your website while you update the content in your CMS. Currently this feature only works with react based projects, such as [NextJS](https://nestjs.com/). + +## Getting Started + +You can turn on live previews for either the whole CMS, individual collections or individual files in a collection. + +Simply update your `editor` config with a url template for the live editor to use to load your live pages. + + + +```yaml + +``` + +```javascript +editor: { + live_preview: '' +} +``` + + diff --git a/packages/docs/content/docs/migration-guide-v2.mdx b/packages/docs/content/docs/migration-guide-v2.mdx deleted file mode 100644 index 3d1af99b..00000000 --- a/packages/docs/content/docs/migration-guide-v2.mdx +++ /dev/null @@ -1,284 +0,0 @@ ---- -group: Migration -title: How to Upgrade to v2 -weight: 101 ---- - -Static CMS v2 introduces a brand new UI and an updated media library. - -In this guide, we will walk you through the steps for upgrading to Static CMS v2. - -Please [report any issues](https://github.com/StaticJsCMS/static-cms/issues/new) you encounter while upgrading to Static CMS v2. - -## Installing - -To install the latest version of Static CMS: - -```bash -npm install @staticcms/core@^2.0.0 -``` - -Or if you’re using yarn: - -```bash -yarn add @staticcms/core@^2.0.0 -``` - -If you are using a CDN to load Static CMS, simply change your URL: - -```html - -``` - -### Proxy Server - -If you are using `@staticcms/proxy-server` you will need to ensure you are using version `3.0.0` or above for all features in Static CMS `v2.0.0` to work. - -## Deprecated Items Removed - -All previously deprecated items have been removed as part of this release. - -- `getAsset` - Use `useMediaAsset` React hook instead -- `createReactClass` - Use [react functional components](https://react.dev/learn) instead -- `isFieldDuplicate` - Use `duplicate` variable instead -- `isFieldHidden` - Use `hidden` variable instead - -## Importing Static CMS Styles - -In `v2.0.0` the apps stylings are not longer bundled into the main javscript file. For both CDN and bundled setups, you will need to include the css file yourself. - -**CDN**: - -```html - -``` - -**Bundling**: - -```js -import '@staticcms/core/dist/main.css'; -``` - -### Custom Preview Styles - -Some basic preview styles are now provided in order to properly support dark mode and make the basic previews look a bit better. However, if you [provide your own preview styles](/docs/custom-previews#editor-preview-styles) these default styles will not be included in the preview. - -## Nested Collections - -While still in beta, [Nested Collections](/docs/collection-types#nested-collections) are now fully working and supported in `v2.0.0`. However there are some breaking config changes. The `meta` config has been dropped and its `path` property has been moved into the `nested` prop. You can also no longer specify the widget type for the path. - -**Old Config** - - -```yaml -collections: - - name: pages - label: Pages - label_singular: 'Page' - folder: content/pages - create: true - nested: - depth: 100 - summary: '{{title}}' - fields: - - label: Title - name: title - widget: string - - label: Body - name: body - widget: markdown - meta: { path: { widget: string, label: 'Path', index_file: 'index' } } -``` - -```js -{ - collections: [ - { - name: 'pages', - label: 'Pages', - label_singular: 'Page', - folder: 'content/pages', - create: true, - nested: { - depth: 100, - summary: '{{title}}', - }, - fields: [ - { - label: 'Title', - name: 'title', - widget: 'string', - }, - { - label: 'Body', - name: 'body', - widget: 'markdown', - }, - ], - meta: { - path: { - widget: 'string', - label: 'Path', - index_file: 'index', - }, - }, - }, - ]; -} -``` - - - -**New Config** - - -```yaml -collections: - - name: pages - label: Pages - label_singular: 'Page' - folder: content/pages - create: true - nested: - depth: 100 - summary: '{{title}}' - path: { label: 'Path', index_file: 'index' } - fields: - - label: Title - name: title - widget: string - - label: Body - name: body - widget: markdown -``` - -```js -{ - collections: [ - { - name: 'pages', - label: 'Pages', - label_singular: 'Page', - folder: 'content/pages', - create: true, - nested: { - depth: 100, - summary: '{{title}}', - path: { - label: 'Path', - index_file: 'index', - }, - }, - fields: [ - { - label: 'Title', - name: 'title', - widget: 'string', - }, - { - label: 'Body', - name: 'body', - widget: 'markdown', - }, - ], - }, - ]; -} -``` - - - -## External Media Libraries - -External media integrations for Cloudinary and Uploadcare have been removed as part of an ongoing to effect to narrow the focus of Static CMS. With [Decap](https://decapcms.org/) (previously Netlify CMS) being supported again its not as critical for Static CMS to support every possible option. So, in order to focus our efforts on other features, it has been decided to remove these external media integrations. - -[Netlify Large Media](/docs/netlify-large-media) is still available. - -This brings with it some breaking changes for the `media_library` config property. - -**Old Config (for Image or File Field)** - - -```yaml -name: thumbnail -label: Featured Image -widget: image -default: /uploads/chocolate-dogecoin.jpg -media_library: - choose_url: true - config: - max_file_size: 512000 -``` - -```js -{ - name: "thumbnail", - label: "Featured Image", - widget: "image", - default: "/uploads/chocolate-dogecoin.jpg", - media_library: { - choose_url: true, - config: { - max_file_size: 512000 - } - } -} -``` - - - -**New Config** - - -```yaml -name: thumbnail -label: Featured Image -widget: image -default: /uploads/chocolate-dogecoin.jpg -choose_url: true -media_library: - max_file_size: 512000 -``` - -```js -{ - name: "thumbnail", - label: "Featured Image", - widget: "image", - default: "/uploads/chocolate-dogecoin.jpg", - choose_url: true, - media_library: { - max_file_size: 512000 - } -} -``` - - - -Also the `clearMediaControl` and `removeMediaControl` widget control props have been removed as they were only used for the external media library integrations. - -## Other Breaking Changes - -- [Card previews](/docs/custom-previews#collection-card-preview) now are only used for the card view. The `viewStyle` property has been removed. [Field previews](/docs/custom-previews#field-preview) can be used to change the table view. -- Widget Control component property changes: - - `isDisabled` renamed to `disabled` - - `isDuplicate` renamed to `duplicate` - - `isHidden` renamed to `hidden` - - `mediaPaths` is now object of id mapped to an object containing the `path` and optional `alt` -- `useMediaInsert` hook now requires a collection to be passed in. Its callback function now receives an object containing the `path` and optional `alt` instead of a string. -- `prePublish` and `postPublish` events have been dropped. - -## Other Changes - -- `summary_fields` property added to [collection configuration](/docs/collection-overview) to allow customization of the table view. This works with the new [field preview](/docs/custom-previews). -- New widget control property: `forSingleList`. It specifies if the widget is within a singleton `list` widget (string array, number array, etc) - -## Deprecations - -In the Widget Control component property, the following properties have been deprecated. They will be removed in `v3.0.0`. - -| Param | Type | Description | -| ------------------- | -------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -| mediaPaths | object | Use [useMediaInsert](/docs/custom-widgets#interacting-with-the-media-library) instead. Key/value object of control IDs (passed to the media library) mapping to media paths | -| openMediaLibrary | function | Use [useMediaInsert](/docs/custom-widgets#interacting-with-the-media-library) instead. Opens the media library popup. See [Open Media Library](#open-media-library) | -| removeInsertedMedia | function | Use [useMediaInsert](/docs/custom-widgets#interacting-with-the-media-library) instead. Removes draft media for a give control ID | diff --git a/packages/docs/content/docs/migration-guide-v3.mdx b/packages/docs/content/docs/migration-guide-v3.mdx new file mode 100644 index 00000000..48fafbb8 --- /dev/null +++ b/packages/docs/content/docs/migration-guide-v3.mdx @@ -0,0 +1,90 @@ +--- +group: Migration +title: How to Upgrade to v3 +weight: 101 +--- + +Static CMS v3 introduces: +- Mobile support +- Depedent fields (see [Field Conditions](/docs/widgets#field-conditions) for more information) +- Live previews for react based projects (see [Live Previews](/docs/live-previews) for more information) + +In this guide, we will walk you through the steps for upgrading to Static CMS v3. + +Please [report any issues](https://github.com/StaticJsCMS/static-cms/issues/new) you encounter while upgrading to Static CMS v3. + +## Installing + +To install the latest version of Static CMS: + +```bash +npm install @staticcms/core@^3.0.0 +``` + +Or if you're using yarn: + +```bash +yarn add @staticcms/core@^3.0.0 +``` + +If you are using a CDN to load Static CMS, simply change your URLs: + +```html + +``` + +```html + +``` + +## Gitea Backend Update + +While still remaining in beta, the Gitea backend has been evolving. This update switches the authentication mechanism to PKCE auth and improves performance when dealing with multiple file commits. + +To use Gitea with Static CMS v3, you need to update your Gitea instance to at least `v1.20`. You will also need to update your config to match the setup for PKCE authentication. See [Gitea authentication](/docs/gitea-backend#authentication). + +## CMS Events + +CMS Events have undergone a significant refactor in this update, including adding a new `change` event. You may need to update your config as follows to continue to use them. The `preSave` and `postSave` events along with the new `change` event, now require a `collection` be provided during registration, with an optional `file` if you are targeting a [file collection](/docs/collection-types#file-collections). All events now can handle async handlers as well. + +**Old setup** + +```js +CMS.registerEventListener({ + name: 'preSave', + handler: ({ entry }) => { + return { + ...entry, + data: { + ...entry.data, + title: 'new title', + }, + }; + }, +}); +``` + +**New Setup** + +```js +CMS.registerEventListener({ + name: 'preSave', + collection: 'posts', + handler: ({ data: { entry } }) => { + return { + ...entry.data, + title: 'new title', + }; + }, +}); +``` + +See [CMS Events](/docs/beta-features#registering-to-cms-events) for more details. + +## Other Breaking Changes + +- The following Widget Control component properties have been removed: + - `hidden` + - `mediaPaths` - Use [useMediaInsert](/docs/custom-widgets#interacting-with-the-media-library) instead. + - `openMediaLibrary` - Use [useMediaInsert](/docs/custom-widgets#interacting-with-the-media-library) instead. + - `removeInsertedMedia` - Use [useMediaInsert](/docs/custom-widgets#interacting-with-the-media-library) instead. diff --git a/packages/docs/content/releases.json b/packages/docs/content/releases.json index 7e947211..5e6bcf34 100644 --- a/packages/docs/content/releases.json +++ b/packages/docs/content/releases.json @@ -1,5 +1,11 @@ { "releases": [ + { + "date": "2023-08-01T10:00:00.000Z", + "version": "v3.0.0", + "type": "major", + "description": "Mobile support and dependent fields" + }, { "date": "2023-06-14T10:00:00.000Z", "version": "v2.5.1",