From b0e6e346a4578feb5a536731820668906598483a Mon Sep 17 00:00:00 2001 From: Gutenberg Repository Automation Date: Wed, 14 Dec 2022 17:21:30 +0000 Subject: [PATCH] Merge changes published in the Gutenberg plugin "release/14.8" branch --- .eslintrc.js | 310 +- .github/PULL_REQUEST_TEMPLATE.md | 3 + .../workflows/gradle-wrapper-validation.yml | 10 + .github/workflows/performance.yml | 2 +- .github/workflows/rnmobile-android-runner.yml | 40 +- bin/plugin/cli.js | 4 + bin/plugin/commands/performance.js | 30 +- changelog.txt | 873 +++ docs/contributors/code/release.md | 2 + docs/explanations/faq.md | 6 +- docs/getting-started/create-block/README.md | 2 + .../create-block/attributes.md | 2 +- .../create-block/block-anatomy.md | 4 +- docs/how-to-guides/block-tutorial/README.md | 4 +- .../applying-styles-with-stylesheets.md | 2 +- .../extending-the-query-loop-block.md | 2 + .../block-api/block-metadata.md | 2 +- .../block-api/block-registration.md | 6 +- .../block-api/block-supports.md | 31 +- docs/reference-guides/core-blocks.md | 17 +- .../data/data-core-block-editor.md | 14 +- .../data/data-core-edit-site.md | 82 +- .../theme-json-reference/theme-json-living.md | 7 + gutenberg.php | 4 +- lib/block-supports/duotone.php | 4 +- lib/block-supports/layout.php | 93 +- lib/blocks.php | 4 +- lib/client-assets.php | 49 +- .../wordpress-6.0/block-editor-settings.php | 35 - lib/compat/wordpress-6.0/block-gallery.php | 57 - .../wordpress-6.0/block-patterns-update.php | 217 - lib/compat/wordpress-6.0/block-patterns.php | 44 - .../wordpress-6.0/block-template-utils.php | 132 - lib/compat/wordpress-6.0/blocks.php | 194 - ...nberg-rest-edit-site-export-controller.php | 89 - ...-rest-pattern-directory-controller-6-0.php | 48 - ...st-block-pattern-categories-controller.php | 152 - .../wordpress-6.0/class-wp-theme-json-6-0.php | 680 --- .../class-wp-theme-json-resolver-6-0.php | 223 - lib/compat/wordpress-6.0/client-assets.php | 117 - lib/compat/wordpress-6.0/edit-form-blocks.php | 66 - lib/compat/wordpress-6.0/functions.php | 27 - .../get-global-styles-and-settings.php | 104 - lib/compat/wordpress-6.0/post-lock.php | 67 - .../wordpress-6.0/render-svg-filters.php | 38 - lib/compat/wordpress-6.0/rest-api.php | 65 - lib/compat/wordpress-6.0/site-editor.php | 95 - lib/compat/wordpress-6.0/theme-i18n.json | 72 - lib/compat/wordpress-6.0/theme.json | 245 - lib/compat/wordpress-6.1/block-patterns.php | 177 - .../wordpress-6.1/block-template-utils.php | 4 +- lib/compat/wordpress-6.1/blocks.php | 6 +- ...rg-rest-block-patterns-controller-6-1.php} | 99 +- ...ss-gutenberg-rest-templates-controller.php | 2 +- .../wordpress-6.1/class-wp-theme-json-6-1.php | 2 +- .../class-wp-theme-json-resolver-6-1.php | 2 +- .../get-global-styles-and-settings.php | 98 +- lib/compat/wordpress-6.1/rest-api.php | 9 - .../wordpress-6.1/template-parts-screen.php | 11 +- .../wordpress-6.2/block-editor-settings.php | 28 + lib/compat/wordpress-6.2/block-patterns.php | 277 +- ...erg-rest-block-patterns-controller-6-2.php | 107 + ...erg-rest-global-styles-controller-6-2.php} | 259 +- ...-rest-pattern-directory-controller-6-2.php | 127 +- .../wordpress-6.2/class-wp-theme-json-6-2.php | 172 + .../class-wp-theme-json-resolver-6-2.php | 195 + lib/compat/wordpress-6.2/default-filters.php | 10 +- lib/compat/wordpress-6.2/edit-form-blocks.php | 24 + .../get-global-styles-and-settings.php | 207 +- lib/compat/wordpress-6.2/rest-api.php | 18 + lib/compat/wordpress-6.2/script-loader.php | 79 + lib/compat/wordpress-6.2/site-editor.php | 24 + ...-rest-block-editor-settings-controller.php | 6 + ...class-wp-theme-json-resolver-gutenberg.php | 81 +- lib/experimental/class-wp-webfonts.php | 9 +- lib/experimental/editor-settings.php | 3 + lib/experimental/html/class-wp-html-span.php | 52 + .../html/class-wp-html-tag-processor.php | 470 +- lib/experimental/html/index.php | 1 + lib/experimental/kses.php | 66 + lib/experiments-page.php | 33 +- lib/load.php | 37 +- package-lock.json | 4864 ++++++++++++----- package.json | 26 +- packages/annotations/package.json | 3 - packages/babel-plugin-makepot/index.js | 7 +- packages/base-styles/CHANGELOG.md | 4 + packages/base-styles/_colors.scss | 2 + .../_default-custom-properties.scss | 2 + packages/base-styles/_mixins.scss | 35 +- packages/base-styles/_variables.scss | 6 +- packages/base-styles/_z-index.scss | 12 +- packages/block-directory/CHANGELOG.md | 4 + packages/block-directory/package.json | 4 +- .../index.js | 5 +- packages/block-editor/CHANGELOG.md | 8 + packages/block-editor/package.json | 6 +- .../block-editor/src/autocompleters/block.js | 8 +- .../block-editor/src/autocompleters/link.js | 2 + .../alignment-control/test/index.js | 5 +- .../block-alignment-control/test/index.js | 5 +- .../src/components/block-card/index.js | 48 +- .../src/components/block-card/style.scss | 4 + .../{style.scss => content.scss} | 8 +- .../components/block-draggable/content.scss | 20 + .../block-draggable/index.native.js | 94 +- .../src/components/block-draggable/style.scss | 21 - .../block-draggable/test/helpers.native.js | 16 +- .../block-draggable/test/index.native.js | 80 +- .../src/components/block-edit/edit.js | 7 +- .../src/components/block-edit/edit.native.js | 11 +- .../src/components/block-inspector/index.js | 177 +- .../src/components/block-inspector/style.scss | 10 +- .../{style.scss => content.scss} | 0 .../block-list/block-list-context.native.js | 13 +- .../src/components/block-list/block.js | 97 +- .../src/components/block-list/block.native.js | 101 +- .../block-list/{style.scss => content.scss} | 16 +- .../block-actions-menu.native.js | 30 +- .../test/__snapshots__/index.native.js.snap | 2 - .../components/block-pattern-setup/index.js | 3 +- .../components/block-patterns-list/index.js | 71 +- .../src/components/block-preview/README.md | 25 +- .../src/components/block-preview/auto.js | 8 +- .../src/components/block-preview/content.scss | 4 + .../src/components/block-preview/index.js | 19 +- .../src/components/block-preview/live.js | 19 - .../src/components/block-preview/style.scss | 7 - .../components/block-preview/test/index.js | 53 +- .../block-selection-clearer/test/index.js | 24 +- .../block-settings-dropdown.js | 52 +- .../components/block-switcher/test/index.js | 4 + .../src/components/block-toolbar/index.js | 17 +- .../src/components/block-toolbar/style.scss | 10 + .../components/block-tools/insertion-point.js | 50 +- .../src/components/block-tools/style.scss | 17 +- .../block-variation-picker/index.js | 5 +- .../test/index.js | 5 +- .../{style.scss => content.scss} | 0 .../src/components/height-control/index.js | 123 + .../height-control/stories/index.js | 21 + .../src/components/height-control/style.scss | 5 + .../src/components/iframe/index.js | 43 +- .../components/image-editor/use-save-image.js | 2 + .../components/image-editor/zoom-dropdown.js | 1 + packages/block-editor/src/components/index.js | 1 + .../inner-blocks/{style.scss => content.scss} | 0 .../src/components/inner-blocks/index.js | 29 +- .../use-inner-block-template-sync.js | 38 +- .../components/inserter-list-item/index.js | 12 +- .../components/inserter-list-item/style.scss | 26 + .../block-patterns-explorer/sidebar.js | 1 + .../components/inserter/block-patterns-tab.js | 99 +- .../components/inserter/block-types-tab.js | 5 +- .../inserter/hooks/use-debounced-input.js | 17 + .../src/components/inserter/index.js | 12 +- .../src/components/inserter/index.native.js | 2 +- .../components/inserter/media-tab/hooks.js | 88 + .../components/inserter/media-tab/index.js | 3 + .../inserter/media-tab/media-list.js | 93 + .../inserter/media-tab/media-panel.js | 83 + .../inserter/media-tab/media-tab.js | 135 + .../components/inserter/media-tab/utils.js | 37 + .../src/components/inserter/menu.js | 68 +- .../inserter/mobile-tab-navigation.js | 85 + .../src/components/inserter/quick-inserter.js | 1 + .../src/components/inserter/search-results.js | 5 +- .../src/components/inserter/stories/index.js | 2 +- .../inserter/stories/{ => utils}/fixtures.js | 0 .../src/components/inserter/style.scss | 187 +- .../src/components/inserter/tabs.js | 13 +- .../advanced-controls-panel.js | 37 + .../inspector-controls-tabs/index.js | 62 + .../inspector-controls-tabs/settings-tab.js | 18 + .../inspector-controls-tabs/styles-tab.js | 51 + .../use-inspector-controls-tabs.js | 89 + .../use-is-list-view-tab-disabled.js | 9 + .../inspector-controls-tabs/utils.js | 28 + .../components/inspector-controls/groups.js | 2 + .../line-height-control/test/index.js | 10 +- .../src/components/link-control/README.md | 2 +- .../src/components/link-control/index.js | 1 + .../components/link-control/search-input.js | 1 - .../src/components/link-control/test/index.js | 589 +- .../link-control/use-internal-input-value.js | 6 +- .../src/components/list-view/block.js | 3 + .../src/components/list-view/branch.js | 18 +- .../src/components/list-view/style.scss | 29 +- .../{style.scss => content.scss} | 0 .../media-replace-flow/test/index.js | 46 +- .../media-upload/test/index.native.js | 2 + .../components/off-canvas-editor/appender.js | 93 + .../off-canvas-editor/block-edit-button.js | 27 + .../src/components/off-canvas-editor/block.js | 110 +- .../components/off-canvas-editor/branch.js | 8 +- .../src/components/off-canvas-editor/index.js | 92 +- .../src/components/off-canvas-editor/leaf.js | 6 +- .../components/off-canvas-editor/link-ui.js | 166 + .../components/off-canvas-editor/style.scss | 405 +- .../off-canvas-editor/update-attributes.js | 99 + .../plain-text/{style.scss => content.scss} | 0 .../recursion-provider/test/index.js | 56 +- .../src/components/rich-text/content.scss | 42 + .../rich-text/format-toolbar/index.js | 10 +- .../src/components/rich-text/index.js | 4 +- .../src/components/rich-text/index.native.js | 2 - .../src/components/rich-text/style.scss | 43 - .../rich-text/use-insert-replacement-text.js | 31 + .../rich-text/use-undo-automatic-change.js | 8 +- .../src/components/rich-text/utils.js | 23 +- .../spacing-input-control.js | 9 + .../components/ungroup-button/index.native.js | 8 +- .../src/components/url-input/index.js | 130 +- .../test/__snapshots__/index.js.snap | 14 +- .../src/components/url-popover/test/index.js | 30 +- .../use-block-display-information/index.js | 19 +- .../src/components/use-setting/index.js | 14 +- .../src/components/use-setting/test/index.js | 99 + packages/block-editor/src/content.scss | 10 + .../block-editor/src/hooks/child-layout.js | 190 + .../block-editor/src/hooks/content-lock-ui.js | 2 +- packages/block-editor/src/hooks/dimensions.js | 52 +- packages/block-editor/src/hooks/layout.js | 60 + packages/block-editor/src/hooks/min-height.js | 21 +- packages/block-editor/src/store/reducer.js | 882 +-- packages/block-editor/src/store/selectors.js | 113 +- .../src/store/test/performance.js | 71 + .../block-editor/src/store/test/reducer.js | 1114 ++-- .../block-editor/src/store/test/selectors.js | 3126 ++++++----- packages/block-editor/src/style.scss | 11 +- packages/block-editor/src/utils/sorting.js | 54 + .../block-editor/src/utils/test/sorting.js | 49 + packages/block-library/CHANGELOG.md | 4 + packages/block-library/package.json | 6 +- .../block-library/src/audio/edit.native.js | 3 +- packages/block-library/src/audio/index.js | 1 + packages/block-library/src/avatar/edit.js | 1 + .../block-library/src/avatar/user-control.js | 1 + packages/block-library/src/block/block.json | 3 +- packages/block-library/src/block/edit.js | 17 +- .../block-library/src/block/edit.native.js | 36 +- packages/block-library/src/block/editor.scss | 17 + packages/block-library/src/block/index.php | 3 +- .../src/block/test/edit.native.js | 60 +- .../block-library/src/button/edit.native.js | 1 + .../src/buttons/test/edit.native.js | 91 +- packages/block-library/src/code/transforms.js | 4 +- packages/block-library/src/columns/edit.js | 1 + .../block-library/src/columns/edit.native.js | 44 +- .../src/comment-author-avatar/edit.js | 1 + .../block-library/src/cover/edit.native.js | 51 +- .../src/cover/edit/inspector-controls.js | 2 + .../src/cover/test/edit.native.js | 199 +- .../src/embed/embed-controls.native.js | 1 + .../src/embed/embed-preview.native.js | 3 +- .../src/embed/test/index.native.js | 468 +- packages/block-library/src/file/inspector.js | 1 + .../test/__snapshots__/edit.native.js.snap | 8 +- packages/block-library/src/gallery/edit.js | 20 +- .../src/gallery/gallery.native.js | 3 +- packages/block-library/src/gallery/shared.js | 18 +- .../block-library/src/gallery/transforms.js | 7 +- .../src/gallery/use-get-media.js | 2 +- .../src/gallery/use-get-media.native.js | 3 +- packages/block-library/src/gallery/v1/edit.js | 15 +- .../src/gallery/v1/gallery.native.js | 3 +- .../block-library/src/gallery/v1/shared.js | 9 +- packages/block-library/src/group/block.json | 4 +- packages/block-library/src/group/edit.js | 8 +- .../block-library/src/group/placeholder.js | 31 +- .../test/__snapshots__/edit.native.js.snap | 2 +- packages/block-library/src/heading/block.json | 3 +- .../block-library/src/heading/deprecated.js | 345 +- packages/block-library/src/heading/index.php | 52 + .../test/__snapshots__/index.native.js.snap | 7 + .../src/heading/test/index.native.js | 44 + .../block-library/src/home-link/index.php | 2 +- packages/block-library/src/html/block.json | 2 +- packages/block-library/src/image/edit.js | 9 +- .../block-library/src/image/edit.native.js | 3 +- packages/block-library/src/image/editor.scss | 11 - packages/block-library/src/image/image.js | 21 +- .../src/image/test/edit.native.js | 24 +- packages/block-library/src/index.js | 2 + .../block-library/src/latest-comments/edit.js | 1 + .../block-library/src/latest-posts/edit.js | 2 + packages/block-library/src/list-item/index.js | 2 + .../block-library/src/list-item/transforms.js | 17 + packages/block-library/src/list-item/utils.js | 11 +- .../src/list/test/edit.native.js | 98 +- packages/block-library/src/list/transforms.js | 11 - packages/block-library/src/list/utils.js | 8 +- packages/block-library/src/media-text/edit.js | 8 +- .../missing/test/edit-integration.native.js | 22 +- .../block-library/src/navigation-link/edit.js | 387 +- .../src/navigation-link/index.php | 32 + .../src/navigation-link/link-ui.js | 213 + .../src/navigation-link/test/edit.js | 105 +- .../src/navigation-link/transforms.js | 14 + .../src/navigation-link/update-attributes.js | 99 + .../src/navigation-submenu/edit.js | 234 +- .../src/navigation-submenu/index.php | 40 + .../src/navigation/edit/are-blocks-dirty.js | 51 + .../src/navigation/edit/index.js | 528 +- .../edit/menu-inspector-controls.js | 78 + .../edit/navigation-menu-selector.js | 21 +- .../navigation/edit/test/are-blocks-dirty.js | 121 + .../navigation/edit/unsaved-inner-blocks.js | 19 +- .../block-library/src/navigation/index.php | 67 +- .../src/navigation/view-modal.js | 12 +- .../src/page-list-item/block.json | 51 + .../block-library/src/page-list-item/edit.js | 94 + .../block-library/src/page-list-item/index.js | 24 + .../block-library/src/page-list-item/init.js | 6 + .../block-library/src/page-list/block.json | 7 +- .../block-library/src/page-list/constants.js | 8 + .../src/page-list/convert-to-links-modal.js | 95 +- .../page-list/convert-to-navigation-links.js | 60 + packages/block-library/src/page-list/edit.js | 297 +- .../block-library/src/page-list/index.php | 9 + .../page-list/test/convert-to-links-modal.js | 62 +- packages/block-library/src/paragraph/index.js | 6 - .../block-library/src/post-author/edit.js | 5 +- .../block-library/src/post-content/index.php | 3 +- .../src/post-featured-image/index.php | 9 +- .../src/post-featured-image/overlay.js | 1 + .../src/post-template/block.json | 8 + .../src/preformatted/transforms.js | 5 +- packages/block-library/src/query/block.json | 8 - .../block-library/src/query/deprecated.js | 343 +- .../query/edit/inspector-controls/index.js | 1 + .../src/query/edit/query-placeholder.js | 20 +- packages/block-library/src/rss/edit.js | 3 + .../block-library/src/search/edit.native.js | 58 +- packages/block-library/src/search/index.js | 6 +- .../src/shortcode/test/edit.native.js | 33 +- packages/block-library/src/site-logo/edit.js | 18 +- packages/block-library/src/site-logo/index.js | 1 + .../block-library/src/site-title/index.js | 1 + .../src/social-link/social-list.js | 9 +- .../src/social-link/test/index.native.js | 26 +- .../src/social-links/test/edit.native.js | 7 +- .../src/spacer/test/index.native.js | 75 +- .../src/table-of-contents/edit.js | 4 +- packages/block-library/src/table/block.json | 15 + packages/block-library/src/table/edit.js | 3 +- packages/block-library/src/table/editor.scss | 4 - packages/block-library/src/table/index.js | 1 + packages/block-library/src/table/save.js | 6 +- packages/block-library/src/table/state.js | 29 +- .../block-library/src/table/transforms.js | 3 +- packages/block-library/src/tag-cloud/edit.js | 5 +- .../src/template-part/edit/selection-modal.js | 7 +- .../src/template-part/edit/utils/hooks.js | 10 +- .../src/template-part/editor.scss | 22 +- .../block-library/src/template-part/index.php | 3 +- .../block-library/src/text-columns/edit.js | 1 + .../utils/transformation-categories.native.js | 1 + packages/block-library/src/video/edit.js | 2 +- .../block-library/src/video/edit.native.js | 3 +- packages/block-library/test/babel-plugin.js | 70 +- packages/blocks/CHANGELOG.md | 4 + packages/blocks/README.md | 20 +- packages/blocks/package.json | 3 +- packages/blocks/src/api/index.js | 1 + .../src/api/parser/get-block-attributes.js | 22 +- .../src/api/raw-handling/paste-handler.js | 1 + .../api/raw-handling/test/paste-handler.js | 131 + packages/blocks/src/api/registration.js | 2 +- packages/blocks/src/api/utils.js | 37 +- packages/blocks/src/api/validation/index.js | 6 +- packages/blocks/src/store/actions.js | 36 +- packages/blocks/src/store/reducer.js | 20 +- packages/blocks/src/store/selectors.js | 4 +- packages/blocks/src/store/test/selectors.js | 68 +- packages/components/CHANGELOG.md | 55 + packages/components/package.json | 7 +- .../src/alignment-matrix-control/README.md | 14 +- .../{cell.js => cell.tsx} | 8 +- .../{icon.js => icon.tsx} | 9 +- .../{index.js => index.tsx} | 35 +- .../stories/{index.js => index.tsx} | 26 +- ...> alignment-matrix-control-icon-styles.ts} | 14 +- ....js => alignment-matrix-control-styles.ts} | 16 +- .../test/{index.js => index.tsx} | 10 +- .../src/alignment-matrix-control/types.ts | 54 + .../{utils.js => utils.tsx} | 38 +- .../styles/angle-picker-control-styles.js | 4 + packages/components/src/autocomplete/index.js | 55 +- .../components/src/base-control/README.md | 33 +- packages/components/src/base-control/hooks.ts | 45 + .../components/src/base-control/index.tsx | 25 +- .../src/base-control/stories/index.tsx | 18 +- .../src/base-control/test/index.tsx | 52 + packages/components/src/base-control/types.ts | 10 +- packages/components/src/base-field/README.md | 43 +- .../src/base-field/{hook.js => hook.ts} | 20 +- .../src/base-field/{index.js => index.ts} | 0 .../src/base-field/{styles.js => styles.ts} | 4 +- .../{index.js.snap => index.tsx.snap} | 8 +- .../base-field/test/{index.js => index.tsx} | 15 +- packages/components/src/base-field/types.ts | 29 + .../border-box-control/README.md | 1 + .../src/border-box-control/test/index.js | 4 +- .../border-control/component.tsx | 1 + .../components/src/border-control/styles.ts | 11 +- .../src/box-control/linked-button.js | 19 +- .../components/src/box-control/test/index.js | 341 +- .../components/src/button/stories/index.js | 30 + packages/components/src/button/style.scss | 31 +- packages/components/src/button/test/index.js | 47 +- packages/components/src/card/test/index.tsx | 52 +- .../src/circular-option-picker/style.scss | 1 + .../components/src/color-palette/index.tsx | 10 +- .../src/color-palette/stories/index.tsx | 6 +- .../test/__snapshots__/index.tsx.snap | 8 +- .../src/color-palette/test/index.tsx | 120 +- .../components/src/color-palette/types.ts | 2 +- .../components/src/color-picker/README.md | 1 + .../src/color-picker/input-with-slider.tsx | 1 + .../components/src/color-picker/styles.ts | 4 - .../components/src/confirm-dialog/README.md | 5 +- .../src/custom-select-control/README.md | 14 + .../src/custom-select-control/index.js | 16 +- .../src/dashicon/{index.js => index.tsx} | 14 +- packages/components/src/dashicon/types.ts | 17 + .../src/date-time/date-time/index.tsx | 4 +- .../test/__snapshots__/index.test.js.snap | 12 +- .../components/src/disabled/test/index.tsx | 32 +- .../src/dropdown-menu/test/index.js | 30 +- .../components/src/dropdown/test/index.js | 88 +- .../src/duotone-picker/duotone-picker.js | 4 +- .../src/focal-point-picker/test/media.js | 47 +- .../src/form-token-field/style.scss | 2 +- .../components/src/gradient-picker/index.js | 13 +- packages/components/src/grid/test/grid.tsx | 62 +- .../higher-order/navigate-regions/style.scss | 24 +- .../with-fallback-styles/index.js | 6 +- .../higher-order/with-filters/test/index.js | 195 +- .../with-focus-return/test/index.js | 5 +- packages/components/src/index.js | 4 +- .../components/src/input-control/index.tsx | 70 +- .../src/input-control/input-base.tsx | 9 +- .../src/input-control/stories/index.tsx | 6 + .../src/input-control/test/index.js | 18 + .../components/src/input-control/types.ts | 2 + .../src/mobile/bottom-sheet/cell.native.js | 2 +- .../mobile/bottom-sheet/switch-cell.native.js | 4 +- .../global-styles-context/index.native.js | 20 +- .../keyboard-aware-flat-list/index.ios.js | 4 +- .../mobile/link-settings/test/edit.native.js | 192 +- packages/components/src/modal/index.tsx | 11 + .../components/src/modal/stories/index.tsx | 3 + packages/components/src/modal/test/index.tsx | 15 + .../src/navigation/stories/index.js | 12 +- .../stories/{ => utils}/controlled-state.js | 6 +- .../navigation/stories/{ => utils}/default.js | 6 +- .../navigation/stories/{ => utils}/group.js | 8 +- .../stories/{ => utils}/hide-if-empty.js | 6 +- .../stories/{ => utils}/more-examples.js | 8 +- .../navigation/stories/{ => utils}/search.js | 10 +- .../navigator-provider/component.tsx | 23 +- packages/components/src/notice/test/list.js | 13 +- .../stories/{index.js => index.tsx} | 31 +- ...rol-styles.js => number-control-styles.ts} | 8 +- .../src/number-control/test/index.js | 478 -- .../src/number-control/test/index.tsx | 600 ++ packages/components/src/palette-edit/index.js | 127 +- .../components/src/palette-edit/test/index.js | 26 +- packages/components/src/popover/index.tsx | 12 +- packages/components/src/popover/style.scss | 23 +- .../components/src/popover/test/index.tsx | 40 +- .../components/src/query-controls/index.js | 1 + .../src/range-control/test/index.tsx | 117 +- .../components/src/resizable-box/style.scss | 1 + packages/components/src/sandbox/test/index.js | 24 +- .../components/src/search-control/style.scss | 4 + packages/components/src/snackbar/README.md | 71 +- .../src/snackbar/{index.js => index.tsx} | 74 +- .../src/snackbar/{list.js => list.tsx} | 41 +- .../components/src/snackbar/stories/index.js | 89 - .../components/src/snackbar/stories/index.tsx | 96 + .../components/src/snackbar/stories/list.tsx | 98 + packages/components/src/snackbar/types.ts | 116 + packages/components/src/tab-panel/README.md | 9 + packages/components/src/tab-panel/index.tsx | 17 +- packages/components/src/tab-panel/style.scss | 80 +- .../components/src/tab-panel/test/index.tsx | 111 + packages/components/src/tab-panel/types.ts | 21 +- .../src/text-control/stories/index.tsx | 1 - packages/components/src/text-control/types.ts | 7 +- .../text/test/__snapshots__/index.tsx.snap | 1 + packages/components/src/text/test/index.tsx | 144 +- packages/components/src/theme/README.md | 34 +- .../components/src/theme/color-algorithms.ts | 138 + packages/components/src/theme/index.tsx | 39 +- .../components/src/theme/stories/index.tsx | 67 + packages/components/src/theme/styles.ts | 39 +- .../src/theme/test/color-algorithms.ts | 100 + packages/components/src/theme/test/index.tsx | 103 +- packages/components/src/theme/types.ts | 46 +- .../components/src/toggle-control/index.tsx | 6 +- .../src/toggle-group-control/test/index.tsx | 15 +- .../src/toolbar-group/test/index.js | 14 +- .../src/tools-panel/stories/index.js | 2 +- .../tools-panel-with-item-group-slot.js | 25 +- .../components/src/tools-panel/test/index.js | 34 +- packages/components/src/tooltip/README.md | 7 + packages/components/src/tooltip/style.scss | 4 +- packages/components/src/tooltip/test/index.js | 127 +- .../src/ui/context/context-system-provider.js | 5 +- .../context/test/context-system-provider.js | 9 +- .../src/ui/control-label/test/index.js | 8 +- .../components/src/unit-control/README.md | 10 +- .../components/src/unit-control/index.tsx | 3 + .../src/unit-control/test/index.tsx | 1 + packages/components/src/unit-control/types.ts | 4 + .../components/src/utils/colors-values.js | 2 +- .../components/src/utils/config-values.js | 2 - .../src/utils/hooks/use-controlled-value.ts | 4 +- .../components/src/utils/theme-variables.scss | 20 + packages/components/tsconfig.json | 6 +- packages/compose/CHANGELOG.md | 4 + packages/compose/package.json | 2 +- .../with-global-events/test/index.js | 3 +- .../hooks/use-constrained-tabbing/index.js | 10 +- .../src/hooks/use-disabled/test/index.js | 6 +- .../src/hooks/use-focus-outside/index.ts | 17 + packages/core-data/CHANGELOG.md | 4 + packages/core-data/package.json | 3 +- packages/core-data/src/actions.js | 18 +- packages/core-data/src/entities.js | 10 +- packages/core-data/src/fetch/fetch-media.js | 13 + packages/core-data/src/fetch/index.js | 1 + .../core-data/src/queried-data/reducer.js | 4 +- packages/core-data/src/reducer.js | 7 +- packages/core-data/src/selectors.ts | 13 +- .../src/utils/conservative-map-item.js | 4 +- packages/customize-widgets/CHANGELOG.md | 4 + packages/customize-widgets/package.json | 5 +- .../components/keyboard-shortcuts/index.js | 13 + .../use-sidebar-block-editor.js | 7 +- packages/data-controls/package.json | 3 - packages/data/CHANGELOG.md | 8 + packages/data/README.md | 8 +- packages/data/package.json | 2 +- .../data/src/components/use-select/index.js | 4 +- .../src/components/use-select/test/index.js | 5 +- packages/data/src/index.js | 10 +- .../src/plugins/persistence/test/index.js | 7 +- packages/data/src/redux-store/index.js | 12 + packages/data/src/registry.js | 85 +- packages/data/src/test/registry.js | 9 +- .../test/__snapshots__/build.js.snap | 32 +- .../e2e-test-utils-playwright/src/index.ts | 1 + .../src/request-utils/templates.ts | 13 +- .../src/site-editor/index.ts | 23 + .../src/site-editor/toggle-canvas-mode.js | 8 + .../e2e-test-utils-playwright/src/test.ts | 6 +- packages/e2e-test-utils/CHANGELOG.md | 4 + packages/e2e-test-utils/README.md | 59 +- packages/e2e-test-utils/src/index.js | 9 +- packages/e2e-test-utils/src/inserter.js | 21 +- packages/e2e-test-utils/src/site-editor.js | 138 +- packages/e2e-test-utils/src/templates.js | 13 +- packages/e2e-tests/CHANGELOG.md | 4 + .../config/setup-performance-test.js | 3 + packages/e2e-tests/package.json | 4 +- .../plugins/inner-blocks-templates/index.js | 29 + .../blocks/__snapshots__/heading.test.js.snap | 14 +- .../blocks/__snapshots__/quote.test.js.snap | 2 +- .../specs/editor/blocks/heading.test.js | 2 +- .../specs/editor/blocks/quote.test.js | 6 +- .../__snapshots__/block-grouping.test.js.snap | 8 +- .../inserting-blocks.test.js.snap | 4 +- ...ep-styles-on-block-transforms.test.js.snap | 6 +- .../__snapshots__/list-view.test.js.snap | 2 +- .../multi-block-selection.test.js.snap | 8 +- .../__snapshots__/rich-text.test.js.snap | 2 +- .../editor/various/block-switcher.test.js | 27 +- .../various/multi-block-selection.test.js | 2 +- .../editor/various/reusable-blocks.test.js | 4 +- .../blocks/post-comments-form.test.js | 15 +- .../specs/performance/site-editor.test.js | 2 + .../site-editor/global-styles-sidebar.test.js | 2 + .../site-editor/multi-entity-editing.test.js | 309 -- .../site-editor/multi-entity-saving.test.js | 5 + .../site-editor/settings-sidebar.test.js | 3 + .../site-editor/site-editor-export.test.js | 2 + .../src/components/layout/shortcuts.js | 13 + .../src/components/notices/index.js | 25 +- .../src/components/sidebar/style.scss | 82 +- packages/edit-post/CHANGELOG.md | 4 + packages/edit-post/package.json | 4 +- .../src/components/block-manager/index.js | 16 +- .../edit-post/src/components/header/index.js | 6 +- .../edit-post/src/components/layout/index.js | 4 +- .../preferences-modal/meta-boxes-section.js | 5 +- .../enable-custom-fields.js.snap | 60 +- .../test/__snapshots__/index.js.snap | 53 +- .../__snapshots__/meta-boxes-section.js.snap | 57 +- .../sidebar/post-schedule/style.scss | 3 +- .../components/sidebar/post-status/index.js | 2 +- .../components/sidebar/post-template/index.js | 11 +- .../sidebar/post-template/style.scss | 2 + .../components/sidebar/post-url/style.scss | 3 + .../sidebar/post-visibility/style.scss | 2 + .../sidebar/settings-header/style.scss | 81 +- .../sidebar/settings-sidebar/index.js | 10 +- .../visual-editor/test/index.native.js | 29 +- packages/edit-post/src/editor.native.js | 25 +- .../src/hooks/validate-multiple-use/index.js | 8 +- packages/edit-post/src/plugins/index.js | 2 + .../plugins/navigation-list-view-menu-item.js | 56 + packages/edit-site/CHANGELOG.md | 4 + packages/edit-site/package.json | 5 +- .../add-custom-template-modal.js | 1 + .../src/components/add-new-template/index.js | 9 +- .../add-new-template/new-template-part.js | 18 +- .../add-new-template/new-template.js | 21 +- .../components/add-new-template/style.scss | 4 - .../edit-site/src/components/app/index.js | 57 +- .../components/block-editor/editor-canvas.js | 69 + .../src/components/block-editor/index.js | 189 +- .../block-editor/resizable-editor.js | 73 +- .../src/components/block-editor/style.scss | 26 +- .../src/components/code-editor/style.scss | 2 +- .../edit-site/src/components/editor/index.js | 406 +- .../src/components/editor/style.scss | 22 - .../index.js} | 12 +- .../global-styles/block-preview-panel.js | 29 + .../components/global-styles/context-menu.js | 13 +- .../components/global-styles/custom-css.js | 73 + .../global-styles/dimensions-panel.js | 7 +- .../global-styles/global-styles-provider.js | 56 +- .../src/components/global-styles/hooks.js | 11 +- .../src/components/global-styles/palette.js | 2 +- .../global-styles/screen-block-list.js | 3 +- .../components/global-styles/screen-block.js | 10 +- .../components/global-styles/screen-border.js | 23 + .../components/global-styles/screen-colors.js | 3 +- .../components/global-styles/screen-css.js | 33 + .../components/global-styles/screen-layout.js | 3 - .../components/global-styles/screen-root.js | 31 +- .../global-styles/screen-style-variations.js | 7 +- .../src/components/global-styles/style.scss | 33 +- .../src/components/global-styles/ui.js | 51 +- .../global-styles/use-global-styles-output.js | 5 + .../src/components/global-styles/utils.js | 13 +- .../document-actions/index.js | 23 +- .../document-actions/style.scss | 9 +- .../src/components/header-edit-mode/index.js | 174 +- .../components/header-edit-mode/style.scss | 38 +- .../components/keyboard-shortcuts/index.js | 13 + .../edit-site/src/components/layout/index.js | 278 + .../src/components/layout/style.scss | 176 + .../edit-site/src/components/list/header.js | 6 +- .../edit-site/src/components/list/index.js | 43 +- .../edit-site/src/components/list/style.scss | 14 +- .../src/components/navigate-to-link/index.js | 10 +- .../components/navigation-sidebar/index.js | 46 - .../navigation-panel/constants.js | 94 - .../navigation-panel/index.js | 142 - .../navigation-panel/style.scss | 152 - .../navigation-panel/template-hierarchy.js | 81 - .../navigation-toggle/index.js | 114 - .../navigation-toggle/style.scss | 71 - .../navigation-toggle/test/index.js | 65 - .../edit-site/src/components/routes/index.js | 2 +- .../global-styles-sidebar.js | 49 +- .../src/components/sidebar-edit-mode/index.js | 6 +- .../navigation-menu-sidebar/style.scss | 24 +- .../settings-header/style.scss | 81 +- .../components/sidebar-edit-mode/style.scss | 13 - .../sidebar-navigation-item/index.js | 51 + .../sidebar-navigation-item/style.scss | 17 + .../sidebar-navigation-screen-main/index.js | 72 + .../index.js | 181 + .../style.scss | 9 + .../sidebar-navigation-screen/index.js | 55 + .../sidebar-navigation-screen/style.scss | 41 + .../edit-site/src/components/sidebar/index.js | 34 + .../src/components/sidebar/style.scss | 8 + .../src/components/site-icon/index.js | 56 + .../src/components/site-icon/style.scss | 10 + .../src/components/site-title/index.js | 39 + .../src/components/style-book/index.js | 193 + .../src/components/style-book/style.scss | 78 + .../use-init-edited-entity-from-url.js} | 6 +- .../use-sync-sidebar-path-with-url.js | 36 + .../components/template-details/style.scss | 4 + packages/edit-site/src/index.js | 56 +- packages/edit-site/src/store/actions.js | 141 +- packages/edit-site/src/store/reducer.js | 120 +- packages/edit-site/src/store/selectors.js | 162 +- packages/edit-site/src/store/test/actions.js | 18 +- packages/edit-site/src/store/test/reducer.js | 200 +- .../edit-site/src/store/test/selectors.js | 45 +- packages/edit-site/src/style.scss | 22 +- packages/edit-widgets/CHANGELOG.md | 4 + packages/edit-widgets/package.json | 4 +- .../components/keyboard-shortcuts/index.js | 13 + .../src/components/sidebar/style.scss | 85 +- packages/editor/CHANGELOG.md | 4 + packages/editor/package.json | 4 +- .../src/components/document-outline/check.js | 8 +- .../src/components/editor-notices/index.js | 19 +- .../src/components/editor-snackbars/index.js | 11 +- .../src/components/page-attributes/parent.js | 6 +- .../src/components/post-author/combobox.js | 1 + .../src/components/post-format/index.js | 8 +- .../maybe-category-panel.js | 21 +- .../maybe-post-format-panel.js | 4 +- .../post-taxonomies/flat-term-selector.js | 6 +- .../hierarchical-term-selector.js | 4 +- .../src/components/post-taxonomies/index.js | 10 +- .../components/post-taxonomies/test/index.js | 2 +- .../src/components/post-title/index.native.js | 5 +- .../editor/src/components/post-url/index.js | 8 +- .../src/components/provider/index.native.js | 1 + .../provider/use-block-editor-settings.js | 106 +- .../use-block-editor-settings.native.js | 4 +- packages/element/CHANGELOG.md | 4 + packages/element/package.json | 8 +- .../src/test/create-interpolate-element.js | 10 +- .../data-no-store-string-literals.js | 5 - packages/format-library/CHANGELOG.md | 4 + packages/format-library/package.json | 4 +- .../format-library/src/default-formats.js | 2 + .../src/text-color/test/index.native.js | 72 +- packages/format-library/src/unknown/index.js | 42 + packages/icons/src/index.js | 5 + packages/icons/src/library/border.js | 12 + packages/icons/src/library/drawer-left.js | 21 + packages/icons/src/library/drawer-right.js | 21 + packages/icons/src/library/seen.js | 12 + packages/icons/src/library/shadow.js | 12 + packages/interface/CHANGELOG.md | 4 + packages/interface/package.json | 4 +- .../components/interface-skeleton/index.js | 29 +- .../components/interface-skeleton/style.scss | 18 +- .../src/components/navigable-region/README.md | 12 +- .../src/components/navigable-region/index.js | 12 +- packages/is-shallow-equal/README.md | 6 +- packages/jest-preset-default/package.json | 4 +- packages/keyboard-shortcuts/CHANGELOG.md | 4 + packages/keyboard-shortcuts/package.json | 2 +- packages/list-reusable-blocks/CHANGELOG.md | 4 + packages/list-reusable-blocks/package.json | 4 +- packages/notices/package.json | 3 - packages/nux/CHANGELOG.md | 4 + packages/nux/package.json | 4 +- .../dot-tip/test/__snapshots__/index.js.snap | 5 +- .../nux/src/components/dot-tip/test/index.js | 14 +- packages/plugins/CHANGELOG.md | 4 + packages/plugins/package.json | 2 +- packages/preferences/CHANGELOG.md | 4 + packages/preferences/package.json | 4 +- .../react-native-aztec/android/build.gradle | 9 +- packages/react-native-aztec/package.json | 2 +- packages/react-native-aztec/src/AztecView.js | 1 + .../android/react-native-bridge/build.gradle | 34 +- .../WPAndroidGlue/WPAndroidGlueCode.java | 12 +- .../react-native-bridge/ios/Gutenberg.swift | 6 +- packages/react-native-bridge/package.json | 2 +- packages/react-native-editor/.bundle/config | 2 + packages/react-native-editor/.gitignore | 11 +- packages/react-native-editor/CHANGELOG.md | 10 + .../gutenberg-editor-drag-and-drop.test.js | 5 +- .../gutenberg-editor-paste.test.js | 4 +- ...utenberg-editor-unsupported-blocks.test.js | 6 +- .../__device-tests__/helpers/test-data.js | 2 +- .../android/app/build.gradle | 132 +- .../android/app/src/debug/AndroidManifest.xml | 2 +- .../android/app/src/main/AndroidManifest.xml | 2 +- .../main/java/com/gutenberg/MainActivity.java | 7 - .../java/com/gutenberg/MainApplication.java | 9 - .../MainApplicationReactNativeHost.java | 116 + .../components/MainComponentsRegistry.java | 36 + ...ApplicationTurboModuleManagerDelegate.java | 48 + .../android/app/src/main/jni/Android.mk | 48 + .../jni/MainApplicationModuleProvider.cpp | 24 + .../main/jni/MainApplicationModuleProvider.h | 16 + ...nApplicationTurboModuleManagerDelegate.cpp | 45 + ...ainApplicationTurboModuleManagerDelegate.h | 38 + .../src/main/jni/MainComponentsRegistry.cpp | 61 + .../app/src/main/jni/MainComponentsRegistry.h | 32 + .../android/app/src/main/jni/OnLoad.cpp | 11 + .../res/drawable/rn_edit_text_material.xml | 36 + .../app/src/main/res/values/styles.xml | 1 + .../react-native-editor/android/build.gradle | 24 +- .../android/gradle.properties | 13 +- .../android/settings.gradle | 7 + .../react-native-editor/ios/.ruby-version | 2 +- packages/react-native-editor/ios/Gemfile | 2 +- packages/react-native-editor/ios/Gemfile.lock | 81 +- .../GutenbergDemo.xcodeproj/project.pbxproj | 14 +- packages/react-native-editor/ios/Podfile | 7 +- packages/react-native-editor/ios/Podfile.lock | 483 +- .../ios/gutenbergTests/gutenbergTests.m | 16 +- packages/react-native-editor/package.json | 16 +- .../react-native.config.js | 15 + packages/react-native-editor/src/index.js | 5 - .../react-native-editor/src/initial-html.js | 4 +- packages/react-native-editor/src/setup.js | 3 +- .../test/__snapshots__/build.js.snap | 4 +- packages/reusable-blocks/CHANGELOG.md | 4 + packages/reusable-blocks/package.json | 4 +- .../reusable-blocks-manage-button.js | 10 +- .../reusable-blocks/src/store/test/actions.js | 4 +- packages/rich-text/CHANGELOG.md | 4 + packages/rich-text/package.json | 2 +- .../rich-text/src/component/index.native.js | 1 + packages/rich-text/src/create.js | 23 +- .../src/remove-unregistered-formatting.js | 0 packages/rich-text/src/store/selectors.js | 12 +- packages/rich-text/src/test/helpers/index.js | 4 + packages/rich-text/src/to-tree.js | 8 +- packages/scripts/CHANGELOG.md | 4 + packages/scripts/package.json | 4 +- packages/server-side-render/CHANGELOG.md | 4 + packages/server-side-render/package.json | 5 +- packages/server-side-render/src/index.js | 14 +- .../src/server-side-render.js | 4 +- packages/style-engine/CONTRIBUTING.md | 40 + packages/style-engine/README.md | 2 + .../class-wp-style-engine-css-rule.php | 6 +- .../class-wp-style-engine-processor.php | 2 +- packages/viewport/CHANGELOG.md | 4 + packages/viewport/package.json | 2 +- packages/warning/test/babel-plugin.js | 116 +- packages/widgets/CHANGELOG.md | 4 + packages/widgets/package.json | 4 +- phpunit/blocks/render-block-heading-test.php | 86 + ...rg-rest-block-patterns-controller-test.php | 123 +- ...rest-pattern-directory-controller-test.php | 52 +- phpunit/class-wp-theme-json-resolver-test.php | 130 + phpunit/class-wp-theme-json-test.php | 60 +- .../WP_HTML_Tag_Processor_Isolated_Test.php | 145 - phpunit/html/WP_HTML_Tag_Processor_Test.php | 1187 ---- .../wp-html-tag-processor-bookmark-test.php | 370 ++ ...est.php => wp-html-tag-processor-test.php} | 109 +- .../html/wp-html-tag-processor-wp-test.php | 91 - ...-wp-style-engine-css-declarations-test.php | 1 + .../class-wp-style-engine-css-rule-test.php | 28 + ...s-wp-style-engine-css-rules-store-test.php | 1 + .../class-wp-style-engine-processor-test.php | 1 + phpunit/style-engine/style-engine-test.php | 2 + schemas/json/block.json | 3 +- schemas/json/theme.json | 10 +- storybook/decorators/with-theme.js | 93 +- storybook/main.js | 4 +- storybook/preview.js | 8 +- storybook/stories/docs/inline-icon.js | 14 + storybook/stories/docs/introduction.story.mdx | 20 +- storybook/stories/playground/index.js | 3 + storybook/webpack.config.js | 9 + storybook/webpack/source-link-loader.js | 67 + test/e2e/specs/editor/blocks/buttons.spec.js | 130 + test/e2e/specs/editor/blocks/html.spec.js | 16 + test/e2e/specs/editor/blocks/list.spec.js | 8 +- .../e2e/specs/editor/blocks/paragraph.spec.js | 130 +- ...with-a-predefined-template-1-chromium.txt} | 0 .../specs/editor/plugins/format-api.spec.js | 22 + ...es.spec.js => post-type-templates.spec.js} | 2 +- ...ocks-with-content-only-lock-1-chromium.txt | 5 + ...normal-delete---not-forward-1-chromium.txt | 2 +- ...normal-delete---not-forward-2-chromium.txt | 4 +- ...s-block-selection-is-copied-1-chromium.txt | 2 +- ...s-block-selection-is-copied-2-chromium.txt | 2 +- ...-paste-preformatted-in-list-1-chromium.txt | 5 + ...one-block-on-forward-delete-2-chromium.txt | 9 + ...one-block-on-forward-delete-1-chromium.txt | 14 +- ...than-one-block-on-backspace-2-chromium.txt | 9 + ...than-one-block-on-backspace-1-chromium.txt | 10 +- .../various/autocomplete-and-mentions.spec.js | 28 + .../editor/various/content-only-lock.spec.js | 31 + .../editor/various/copy-cut-paste.spec.js | 15 + .../various/inner-blocks-templates.spec.js | 56 + .../editor/various/inserting-blocks.spec.js | 51 +- .../editor/various/splitting-merging.spec.js | 10 +- .../block-list-panel-preference.spec.js | 6 +- .../site-editor/site-editor-inserter.spec.js | 3 +- test/e2e/specs/site-editor/style-book.spec.js | 149 + .../site-editor/style-variations.spec.js | 10 + .../specs/site-editor/template-part.spec.js | 23 +- .../specs/site-editor/template-revert.spec.js | 6 +- test/e2e/specs/site-editor/title.spec.js | 9 +- .../specs/site-editor/writing-flow.spec.js | 6 +- .../specs/widgets/customizing-widgets.spec.js | 2 +- .../header.html | 0 .../category.html | 0 .../{block-templates => templates}/index.html | 0 .../singular.html | 0 test/emptytheme/theme.json | 1 + .../blocks-raw-handling.test.js.snap | 56 +- test/integration/blocks-raw-handling.test.js | 16 +- ...ore__heading__deprecated-1.serialized.html | 2 +- ...ore__heading__deprecated-2.serialized.html | 2 +- ...ore__heading__deprecated-3.serialized.html | 2 +- ...ore__heading__deprecated-4.serialized.html | 2 +- .../blocks/core__heading__deprecated-5.html | 3 + .../blocks/core__heading__deprecated-5.json | 13 + .../core__heading__deprecated-5.parsed.json | 14 + ...ore__heading__deprecated-5.serialized.html | 3 + .../blocks/core__heading__h2-color.html | 2 +- .../core__heading__h2-color.parsed.json | 4 +- .../core__heading__h2-color.serialized.html | 2 +- .../fixtures/blocks/core__heading__h2.html | 2 +- .../blocks/core__heading__h2.parsed.json | 4 +- .../blocks/core__heading__h2.serialized.html | 2 +- .../fixtures/blocks/core__heading__h4-em.html | 2 +- .../blocks/core__heading__h4-em.parsed.json | 6 +- .../core__heading__h4-em.serialized.html | 2 +- .../blocks/core__heading_align-textalign.html | 2 +- .../core__heading_align-textalign.parsed.json | 4 +- ...e__heading_align-textalign.serialized.html | 2 +- .../fixtures/blocks/core__page-list-item.html | 1 + .../fixtures/blocks/core__page-list-item.json | 11 + .../blocks/core__page-list-item.parsed.json | 14 + .../core__page-list-item.serialized.html | 1 + .../fixtures/blocks/core__page-list.json | 4 +- ...core__query__deprecated-2-with-colors.html | 6 + ...core__query__deprecated-2-with-colors.json | 72 + ...uery__deprecated-2-with-colors.parsed.json | 59 + ...__deprecated-2-with-colors.serialized.html | 7 + .../blocks/core__query__deprecated-3.html | 23 + .../blocks/core__query__deprecated-3.json | 133 + .../core__query__deprecated-3.parsed.json | 128 + .../core__query__deprecated-3.serialized.html | 25 + .../fixtures/documents/google-docs-out.html | 2 +- .../google-docs-with-comments-out.html | 2 +- .../fixtures/documents/markdown-out.html | 10 +- .../fixtures/documents/ms-word-out.html | 4 +- .../fixtures/documents/windows.html | 2 +- .../fixtures/documents/wordpress-out.html | 6 +- .../@wordpress/react-native-aztec/index.js | 5 +- .../integration-test-helpers/get-block.js | 5 +- .../get-inner-block.js | 4 +- .../initialize-editor.js | 3 +- test/native/setup.js | 3 + .../matchers/to-be-positioned-popover.js | 15 + test/unit/config/testing-library.js | 1 + 943 files changed, 28392 insertions(+), 18800 deletions(-) create mode 100644 .github/workflows/gradle-wrapper-validation.yml delete mode 100644 lib/compat/wordpress-6.0/block-editor-settings.php delete mode 100644 lib/compat/wordpress-6.0/block-gallery.php delete mode 100644 lib/compat/wordpress-6.0/block-patterns-update.php delete mode 100644 lib/compat/wordpress-6.0/block-patterns.php delete mode 100644 lib/compat/wordpress-6.0/block-template-utils.php delete mode 100644 lib/compat/wordpress-6.0/blocks.php delete mode 100644 lib/compat/wordpress-6.0/class-gutenberg-rest-edit-site-export-controller.php delete mode 100644 lib/compat/wordpress-6.0/class-gutenberg-rest-pattern-directory-controller-6-0.php delete mode 100644 lib/compat/wordpress-6.0/class-wp-rest-block-pattern-categories-controller.php delete mode 100644 lib/compat/wordpress-6.0/class-wp-theme-json-6-0.php delete mode 100644 lib/compat/wordpress-6.0/class-wp-theme-json-resolver-6-0.php delete mode 100644 lib/compat/wordpress-6.0/client-assets.php delete mode 100644 lib/compat/wordpress-6.0/edit-form-blocks.php delete mode 100644 lib/compat/wordpress-6.0/functions.php delete mode 100644 lib/compat/wordpress-6.0/get-global-styles-and-settings.php delete mode 100644 lib/compat/wordpress-6.0/post-lock.php delete mode 100644 lib/compat/wordpress-6.0/render-svg-filters.php delete mode 100644 lib/compat/wordpress-6.0/rest-api.php delete mode 100644 lib/compat/wordpress-6.0/site-editor.php delete mode 100644 lib/compat/wordpress-6.0/theme-i18n.json delete mode 100644 lib/compat/wordpress-6.0/theme.json delete mode 100644 lib/compat/wordpress-6.1/block-patterns.php rename lib/compat/wordpress-6.1/{class-gutenberg-rest-block-patterns-controller.php => class-gutenberg-rest-block-patterns-controller-6-1.php} (60%) create mode 100644 lib/compat/wordpress-6.2/block-editor-settings.php create mode 100644 lib/compat/wordpress-6.2/class-gutenberg-rest-block-patterns-controller-6-2.php rename lib/compat/{wordpress-6.0/class-gutenberg-rest-global-styles-controller.php => wordpress-6.2/class-gutenberg-rest-global-styles-controller-6-2.php} (53%) create mode 100644 lib/compat/wordpress-6.2/edit-form-blocks.php create mode 100644 lib/compat/wordpress-6.2/site-editor.php create mode 100644 lib/experimental/html/class-wp-html-span.php create mode 100644 lib/experimental/kses.php rename packages/block-editor/src/components/block-content-overlay/{style.scss => content.scss} (75%) create mode 100644 packages/block-editor/src/components/block-draggable/content.scss rename packages/block-editor/src/components/block-list-appender/{style.scss => content.scss} (100%) rename packages/block-editor/src/components/block-list/{style.scss => content.scss} (97%) create mode 100644 packages/block-editor/src/components/block-preview/content.scss delete mode 100644 packages/block-editor/src/components/block-preview/live.js rename packages/block-editor/src/components/default-block-appender/{style.scss => content.scss} (100%) create mode 100644 packages/block-editor/src/components/height-control/index.js create mode 100644 packages/block-editor/src/components/height-control/stories/index.js create mode 100644 packages/block-editor/src/components/height-control/style.scss rename packages/block-editor/src/components/inner-blocks/{style.scss => content.scss} (100%) create mode 100644 packages/block-editor/src/components/inserter/hooks/use-debounced-input.js create mode 100644 packages/block-editor/src/components/inserter/media-tab/hooks.js create mode 100644 packages/block-editor/src/components/inserter/media-tab/index.js create mode 100644 packages/block-editor/src/components/inserter/media-tab/media-list.js create mode 100644 packages/block-editor/src/components/inserter/media-tab/media-panel.js create mode 100644 packages/block-editor/src/components/inserter/media-tab/media-tab.js create mode 100644 packages/block-editor/src/components/inserter/media-tab/utils.js create mode 100644 packages/block-editor/src/components/inserter/mobile-tab-navigation.js rename packages/block-editor/src/components/inserter/stories/{ => utils}/fixtures.js (100%) create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/advanced-controls-panel.js create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/index.js create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/settings-tab.js create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/styles-tab.js create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/use-inspector-controls-tabs.js create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/use-is-list-view-tab-disabled.js create mode 100644 packages/block-editor/src/components/inspector-controls-tabs/utils.js rename packages/block-editor/src/components/media-placeholder/{style.scss => content.scss} (100%) create mode 100644 packages/block-editor/src/components/off-canvas-editor/appender.js create mode 100644 packages/block-editor/src/components/off-canvas-editor/block-edit-button.js create mode 100644 packages/block-editor/src/components/off-canvas-editor/link-ui.js create mode 100644 packages/block-editor/src/components/off-canvas-editor/update-attributes.js rename packages/block-editor/src/components/plain-text/{style.scss => content.scss} (100%) create mode 100644 packages/block-editor/src/components/rich-text/content.scss create mode 100644 packages/block-editor/src/components/rich-text/use-insert-replacement-text.js create mode 100644 packages/block-editor/src/components/use-setting/test/index.js create mode 100644 packages/block-editor/src/content.scss create mode 100644 packages/block-editor/src/hooks/child-layout.js create mode 100644 packages/block-editor/src/store/test/performance.js create mode 100644 packages/block-editor/src/utils/sorting.js create mode 100644 packages/block-editor/src/utils/test/sorting.js create mode 100644 packages/block-library/src/heading/index.php create mode 100644 packages/block-library/src/heading/test/__snapshots__/index.native.js.snap create mode 100644 packages/block-library/src/heading/test/index.native.js create mode 100644 packages/block-library/src/list-item/transforms.js create mode 100644 packages/block-library/src/navigation-link/link-ui.js create mode 100644 packages/block-library/src/navigation-link/update-attributes.js create mode 100644 packages/block-library/src/navigation/edit/are-blocks-dirty.js create mode 100644 packages/block-library/src/navigation/edit/menu-inspector-controls.js create mode 100644 packages/block-library/src/navigation/edit/test/are-blocks-dirty.js create mode 100644 packages/block-library/src/page-list-item/block.json create mode 100644 packages/block-library/src/page-list-item/edit.js create mode 100644 packages/block-library/src/page-list-item/index.js create mode 100644 packages/block-library/src/page-list-item/init.js create mode 100644 packages/block-library/src/page-list/constants.js create mode 100644 packages/block-library/src/page-list/convert-to-navigation-links.js create mode 100644 packages/blocks/src/api/raw-handling/test/paste-handler.js rename packages/components/src/alignment-matrix-control/{cell.js => cell.tsx} (74%) rename packages/components/src/alignment-matrix-control/{icon.js => icon.tsx} (77%) rename packages/components/src/alignment-matrix-control/{index.js => index.tsx} (70%) rename packages/components/src/alignment-matrix-control/stories/{index.js => index.tsx} (70%) rename packages/components/src/alignment-matrix-control/styles/{alignment-matrix-control-icon-styles.js => alignment-matrix-control-icon-styles.ts} (72%) rename packages/components/src/alignment-matrix-control/styles/{alignment-matrix-control-styles.js => alignment-matrix-control-styles.ts} (81%) rename packages/components/src/alignment-matrix-control/test/{index.js => index.tsx} (77%) create mode 100644 packages/components/src/alignment-matrix-control/types.ts rename packages/components/src/alignment-matrix-control/{utils.js => utils.tsx} (58%) create mode 100644 packages/components/src/base-control/hooks.ts create mode 100644 packages/components/src/base-control/test/index.tsx rename packages/components/src/base-field/{hook.js => hook.ts} (63%) rename packages/components/src/base-field/{index.js => index.ts} (100%) rename packages/components/src/base-field/{styles.js => styles.ts} (94%) rename packages/components/src/base-field/test/__snapshots__/{index.js.snap => index.tsx.snap} (96%) rename packages/components/src/base-field/test/{index.js => index.tsx} (83%) create mode 100644 packages/components/src/base-field/types.ts rename packages/components/src/dashicon/{index.js => index.tsx} (76%) rename packages/components/src/navigation/stories/{ => utils}/controlled-state.js (96%) rename packages/components/src/navigation/stories/{ => utils}/default.js (93%) rename packages/components/src/navigation/stories/{ => utils}/group.js (86%) rename packages/components/src/navigation/stories/{ => utils}/hide-if-empty.js (90%) rename packages/components/src/navigation/stories/{ => utils}/more-examples.js (96%) rename packages/components/src/navigation/stories/{ => utils}/search.js (89%) rename packages/components/src/number-control/stories/{index.js => index.tsx} (56%) rename packages/components/src/number-control/styles/{number-control-styles.js => number-control-styles.ts} (81%) delete mode 100644 packages/components/src/number-control/test/index.js create mode 100644 packages/components/src/number-control/test/index.tsx rename packages/components/src/snackbar/{index.js => index.tsx} (68%) rename packages/components/src/snackbar/{list.js => list.tsx} (65%) delete mode 100644 packages/components/src/snackbar/stories/index.js create mode 100644 packages/components/src/snackbar/stories/index.tsx create mode 100644 packages/components/src/snackbar/stories/list.tsx create mode 100644 packages/components/src/snackbar/types.ts create mode 100644 packages/components/src/theme/color-algorithms.ts create mode 100644 packages/components/src/theme/test/color-algorithms.ts rename packages/components/src/tools-panel/stories/{ => utils}/tools-panel-with-item-group-slot.js (92%) create mode 100644 packages/core-data/src/fetch/fetch-media.js create mode 100644 packages/e2e-test-utils-playwright/src/site-editor/index.ts create mode 100644 packages/e2e-test-utils-playwright/src/site-editor/toggle-canvas-mode.js delete mode 100644 packages/e2e-tests/specs/site-editor/multi-entity-editing.test.js create mode 100644 packages/edit-post/src/plugins/navigation-list-view-menu-item.js create mode 100644 packages/edit-site/src/components/block-editor/editor-canvas.js rename packages/edit-site/src/components/{editor/global-styles-renderer.js => global-styles-renderer/index.js} (84%) create mode 100644 packages/edit-site/src/components/global-styles/block-preview-panel.js create mode 100644 packages/edit-site/src/components/global-styles/custom-css.js create mode 100644 packages/edit-site/src/components/global-styles/screen-border.js create mode 100644 packages/edit-site/src/components/global-styles/screen-css.js create mode 100644 packages/edit-site/src/components/layout/index.js create mode 100644 packages/edit-site/src/components/layout/style.scss delete mode 100644 packages/edit-site/src/components/navigation-sidebar/index.js delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-panel/constants.js delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-panel/index.js delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-panel/style.scss delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-panel/template-hierarchy.js delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-toggle/index.js delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-toggle/style.scss delete mode 100644 packages/edit-site/src/components/navigation-sidebar/navigation-toggle/test/index.js create mode 100644 packages/edit-site/src/components/sidebar-navigation-item/index.js create mode 100644 packages/edit-site/src/components/sidebar-navigation-item/style.scss create mode 100644 packages/edit-site/src/components/sidebar-navigation-screen-main/index.js create mode 100644 packages/edit-site/src/components/sidebar-navigation-screen-templates/index.js create mode 100644 packages/edit-site/src/components/sidebar-navigation-screen-templates/style.scss create mode 100644 packages/edit-site/src/components/sidebar-navigation-screen/index.js create mode 100644 packages/edit-site/src/components/sidebar-navigation-screen/style.scss create mode 100644 packages/edit-site/src/components/sidebar/index.js create mode 100644 packages/edit-site/src/components/sidebar/style.scss create mode 100644 packages/edit-site/src/components/site-icon/index.js create mode 100644 packages/edit-site/src/components/site-icon/style.scss create mode 100644 packages/edit-site/src/components/site-title/index.js create mode 100644 packages/edit-site/src/components/style-book/index.js create mode 100644 packages/edit-site/src/components/style-book/style.scss rename packages/edit-site/src/components/{url-query-controller/index.js => sync-state-with-url/use-init-edited-entity-from-url.js} (77%) create mode 100644 packages/edit-site/src/components/sync-state-with-url/use-sync-sidebar-path-with-url.js create mode 100644 packages/format-library/src/unknown/index.js create mode 100644 packages/icons/src/library/border.js create mode 100644 packages/icons/src/library/drawer-left.js create mode 100644 packages/icons/src/library/drawer-right.js create mode 100644 packages/icons/src/library/seen.js create mode 100644 packages/icons/src/library/shadow.js create mode 100644 packages/react-native-editor/.bundle/config create mode 100644 packages/react-native-editor/android/app/src/main/java/com/gutenberg/newarchitecture/MainApplicationReactNativeHost.java create mode 100644 packages/react-native-editor/android/app/src/main/java/com/gutenberg/newarchitecture/components/MainComponentsRegistry.java create mode 100644 packages/react-native-editor/android/app/src/main/java/com/gutenberg/newarchitecture/modules/MainApplicationTurboModuleManagerDelegate.java create mode 100644 packages/react-native-editor/android/app/src/main/jni/Android.mk create mode 100644 packages/react-native-editor/android/app/src/main/jni/MainApplicationModuleProvider.cpp create mode 100644 packages/react-native-editor/android/app/src/main/jni/MainApplicationModuleProvider.h create mode 100644 packages/react-native-editor/android/app/src/main/jni/MainApplicationTurboModuleManagerDelegate.cpp create mode 100644 packages/react-native-editor/android/app/src/main/jni/MainApplicationTurboModuleManagerDelegate.h create mode 100644 packages/react-native-editor/android/app/src/main/jni/MainComponentsRegistry.cpp create mode 100644 packages/react-native-editor/android/app/src/main/jni/MainComponentsRegistry.h create mode 100644 packages/react-native-editor/android/app/src/main/jni/OnLoad.cpp create mode 100644 packages/react-native-editor/android/app/src/main/res/drawable/rn_edit_text_material.xml create mode 100644 packages/react-native-editor/react-native.config.js create mode 100644 packages/rich-text/src/remove-unregistered-formatting.js create mode 100644 packages/style-engine/CONTRIBUTING.md create mode 100644 phpunit/blocks/render-block-heading-test.php delete mode 100644 phpunit/html/WP_HTML_Tag_Processor_Isolated_Test.php delete mode 100644 phpunit/html/WP_HTML_Tag_Processor_Test.php create mode 100644 phpunit/html/wp-html-tag-processor-bookmark-test.php rename phpunit/html/{wp-html-tag-processor-standalone-test.php => wp-html-tag-processor-test.php} (94%) delete mode 100644 phpunit/html/wp-html-tag-processor-wp-test.php create mode 100644 storybook/stories/docs/inline-icon.js create mode 100644 storybook/webpack/source-link-loader.js rename test/e2e/specs/editor/plugins/__snapshots__/{templates-Using-a-CPT-with-a-predefined-templa-7538a--custom-post-types-with-a-predefined-template-1-chromium.txt => Post-type-templates-Using-a-CPT-with-a-predefi-fffe1--custom-post-types-with-a-predefined-template-1-chromium.txt} (100%) rename test/e2e/specs/editor/plugins/{templates.spec.js => post-type-templates.spec.js} (98%) create mode 100644 test/e2e/specs/editor/various/__snapshots__/Content-only-lock-should-be-able-to-edit-the-content-of-blocks-with-content-only-lock-1-chromium.txt create mode 100644 test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-preformatted-in-list-1-chromium.txt create mode 100644 test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-18edb-roduces-more-than-one-block-on-forward-delete-2-chromium.txt create mode 100644 test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-46dfa-rge-produces-more-than-one-block-on-backspace-2-chromium.txt create mode 100644 test/e2e/specs/editor/various/content-only-lock.spec.js create mode 100644 test/e2e/specs/editor/various/inner-blocks-templates.spec.js create mode 100644 test/e2e/specs/site-editor/style-book.spec.js rename test/emptytheme/{block-template-parts => parts}/header.html (100%) rename test/emptytheme/{block-templates => templates}/category.html (100%) rename test/emptytheme/{block-templates => templates}/index.html (100%) rename test/emptytheme/{block-templates => templates}/singular.html (100%) create mode 100644 test/integration/fixtures/blocks/core__heading__deprecated-5.html create mode 100644 test/integration/fixtures/blocks/core__heading__deprecated-5.json create mode 100644 test/integration/fixtures/blocks/core__heading__deprecated-5.parsed.json create mode 100644 test/integration/fixtures/blocks/core__heading__deprecated-5.serialized.html create mode 100644 test/integration/fixtures/blocks/core__page-list-item.html create mode 100644 test/integration/fixtures/blocks/core__page-list-item.json create mode 100644 test/integration/fixtures/blocks/core__page-list-item.parsed.json create mode 100644 test/integration/fixtures/blocks/core__page-list-item.serialized.html create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.html create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.json create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.parsed.json create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.serialized.html create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-3.html create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-3.json create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-3.parsed.json create mode 100644 test/integration/fixtures/blocks/core__query__deprecated-3.serialized.html create mode 100644 test/unit/config/matchers/to-be-positioned-popover.js diff --git a/.eslintrc.js b/.eslintrc.js index 22cb9209b2e9e..51bca0f3bc059 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -36,6 +36,160 @@ const typedFiles = glob( 'packages/*/package.json' ) .filter( ( fileName ) => require( join( __dirname, fileName ) ).types ) .map( ( fileName ) => fileName.replace( 'package.json', '**/*.js' ) ); +const restrictedImports = [ + { + name: 'framer-motion', + message: + 'Please use the Framer Motion API through `@wordpress/components` instead.', + }, + { + name: 'lodash', + importNames: [ + 'camelCase', + 'capitalize', + 'castArray', + 'chunk', + 'clamp', + 'cloneDeep', + 'compact', + 'concat', + 'countBy', + 'debounce', + 'deburr', + 'defaults', + 'defaultTo', + 'delay', + 'difference', + 'differenceWith', + 'dropRight', + 'each', + 'escape', + 'escapeRegExp', + 'every', + 'extend', + 'filter', + 'findIndex', + 'findKey', + 'findLast', + 'first', + 'flatMap', + 'flatten', + 'flattenDeep', + 'flow', + 'flowRight', + 'forEach', + 'fromPairs', + 'has', + 'identity', + 'includes', + 'invoke', + 'isArray', + 'isBoolean', + 'isEqual', + 'isFinite', + 'isFunction', + 'isMatch', + 'isNil', + 'isNumber', + 'isObject', + 'isObjectLike', + 'isPlainObject', + 'isString', + 'isUndefined', + 'keyBy', + 'keys', + 'last', + 'lowerCase', + 'mapKeys', + 'maxBy', + 'memoize', + 'negate', + 'noop', + 'nth', + 'omitBy', + 'once', + 'orderby', + 'overEvery', + 'partial', + 'partialRight', + 'pick', + 'random', + 'reduce', + 'reject', + 'repeat', + 'reverse', + 'size', + 'snakeCase', + 'some', + 'sortBy', + 'startCase', + 'startsWith', + 'stubFalse', + 'stubTrue', + 'sum', + 'sumBy', + 'take', + 'throttle', + 'times', + 'toString', + 'trim', + 'truncate', + 'unionBy', + 'uniq', + 'uniqBy', + 'uniqueId', + 'uniqWith', + 'upperFirst', + 'values', + 'without', + 'words', + 'xor', + 'zip', + ], + message: + 'This Lodash method is not recommended. Please use native functionality instead. If using `memoize`, please use `memize` instead.', + }, + { + name: 'reakit', + message: + 'Please use Reakit API through `@wordpress/components` instead.', + }, + { + name: 'redux', + importNames: [ 'combineReducers' ], + message: 'Please use `combineReducers` from `@wordpress/data` instead.', + }, + { + name: 'puppeteer-testing-library', + message: '`puppeteer-testing-library` is still experimental.', + }, + { + name: '@emotion/css', + message: + 'Please use `@emotion/react` and `@emotion/styled` in order to maintain iframe support. As a replacement for the `cx` function, please use the `useCx` hook defined in `@wordpress/components` instead.', + }, + { + name: '@wordpress/edit-post', + message: + "edit-post is a WordPress top level package that shouldn't be imported into other packages", + }, + { + name: '@wordpress/edit-site', + message: + "edit-site is a WordPress top level package that shouldn't be imported into other packages", + }, + { + name: '@wordpress/edit-widgets', + message: + "edit-widgets is a WordPress top level package that shouldn't be imported into other packages", + }, + { + name: '@wordpress/edit-navigation', + message: + "edit-navigation is a WordPress top level package that shouldn't be imported into other packages", + }, +]; + module.exports = { root: true, extends: [ @@ -70,137 +224,7 @@ module.exports = { 'no-restricted-imports': [ 'error', { - paths: [ - { - name: 'framer-motion', - message: - 'Please use the Framer Motion API through `@wordpress/components` instead.', - }, - { - name: 'lodash', - importNames: [ - 'camelCase', - 'capitalize', - 'castArray', - 'chunk', - 'clamp', - 'cloneDeep', - 'compact', - 'concat', - 'countBy', - 'debounce', - 'deburr', - 'defaults', - 'defaultTo', - 'delay', - 'difference', - 'differenceWith', - 'dropRight', - 'each', - 'escape', - 'escapeRegExp', - 'every', - 'extend', - 'findIndex', - 'findKey', - 'findLast', - 'first', - 'flatMap', - 'flatten', - 'flattenDeep', - 'flow', - 'flowRight', - 'forEach', - 'fromPairs', - 'has', - 'identity', - 'includes', - 'invoke', - 'isArray', - 'isBoolean', - 'isFinite', - 'isFunction', - 'isMatch', - 'isNil', - 'isNumber', - 'isObject', - 'isObjectLike', - 'isPlainObject', - 'isString', - 'isUndefined', - 'keyBy', - 'keys', - 'last', - 'lowerCase', - 'mapKeys', - 'maxBy', - 'memoize', - 'negate', - 'noop', - 'nth', - 'omitBy', - 'once', - 'overEvery', - 'partial', - 'partialRight', - 'random', - 'reduce', - 'reject', - 'repeat', - 'reverse', - 'size', - 'snakeCase', - 'some', - 'sortBy', - 'startCase', - 'startsWith', - 'stubFalse', - 'stubTrue', - 'sum', - 'sumBy', - 'take', - 'throttle', - 'times', - 'toString', - 'trim', - 'truncate', - 'unionBy', - 'uniq', - 'uniqBy', - 'uniqueId', - 'uniqWith', - 'upperFirst', - 'values', - 'without', - 'words', - 'xor', - 'zip', - ], - message: - 'This Lodash method is not recommended. Please use native functionality instead. If using `memoize`, please use `memize` instead.', - }, - { - name: 'reakit', - message: - 'Please use Reakit API through `@wordpress/components` instead.', - }, - { - name: 'redux', - importNames: [ 'combineReducers' ], - message: - 'Please use `combineReducers` from `@wordpress/data` instead.', - }, - { - name: 'puppeteer-testing-library', - message: - '`puppeteer-testing-library` is still experimental.', - }, - { - name: '@emotion/css', - message: - 'Please use `@emotion/react` and `@emotion/styled` in order to maintain iframe support. As a replacement for the `cx` function, please use the `useCx` hook defined in `@wordpress/components` instead.', - }, - ], + paths: restrictedImports, }, ], '@typescript-eslint/no-restricted-imports': [ @@ -347,7 +371,6 @@ module.exports = { 'plugin:testing-library/react', ], rules: { - 'testing-library/no-container': 'off', 'testing-library/no-node-access': 'off', }, }, @@ -418,5 +441,28 @@ module.exports = { plugins: [ 'ssr-friendly' ], extends: [ 'plugin:ssr-friendly/recommended' ], }, + { + files: [ 'packages/block-editor/**' ], + rules: { + 'no-restricted-imports': [ + 'error', + { + paths: [ + ...restrictedImports, + { + name: '@wordpress/api-fetch', + message: + "block-editor is a generic package that doesn't depend on a server or WordPress backend. To provide WordPress integration, consider passing settings to the BlockEditorProvider components.", + }, + { + name: '@wordpress/core-data', + message: + "block-editor is a generic package that doesn't depend on a server or WordPress backend. To provide WordPress integration, consider passing settings to the BlockEditorProvider components.", + }, + ], + }, + ], + }, + }, ], }; diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md index 2ac98e4c82aef..1798e22778e38 100644 --- a/.github/PULL_REQUEST_TEMPLATE.md +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -16,4 +16,7 @@ https://github.com/WordPress/gutenberg/blob/trunk/CONTRIBUTING.md --> +### Testing Instructions for Keyboard + + ## Screenshots or screencast diff --git a/.github/workflows/gradle-wrapper-validation.yml b/.github/workflows/gradle-wrapper-validation.yml new file mode 100644 index 0000000000000..390f121b570f4 --- /dev/null +++ b/.github/workflows/gradle-wrapper-validation.yml @@ -0,0 +1,10 @@ +name: 'Validate Gradle Wrapper' +on: [push, pull_request] + +jobs: + validation: + name: 'Validation' + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: gradle/wrapper-validation-action@v1 diff --git a/.github/workflows/performance.yml b/.github/workflows/performance.yml index e878724c1b5f1..675642df3c26c 100644 --- a/.github/workflows/performance.yml +++ b/.github/workflows/performance.yml @@ -61,7 +61,7 @@ jobs: WP_VERSION=$(awk -F ': ' '/^Tested up to/{print $2}' readme.txt) IFS=. read -ra WP_VERSION_ARRAY <<< "$WP_VERSION" WP_MAJOR="${WP_VERSION_ARRAY[0]}.${WP_VERSION_ARRAY[1]}" - ./bin/plugin/cli.js perf $GITHUB_SHA debd225d007f4e441ceec80fbd6fa96653f94737 --tests-branch $GITHUB_SHA --wp-version "$WP_MAJOR" + ./bin/plugin/cli.js perf $GITHUB_SHA debd225d007f4e441ceec80fbd6fa96653f94737 --tests-branch $GITHUB_SHA --wp-version "$WP_MAJOR" - uses: actions/github-script@d556feaca394842dc55e4734bf3bb9f685482fa0 # v6.3.3 if: github.event_name == 'push' diff --git a/.github/workflows/rnmobile-android-runner.yml b/.github/workflows/rnmobile-android-runner.yml index fec513d88e635..093c2a99ad4a0 100644 --- a/.github/workflows/rnmobile-android-runner.yml +++ b/.github/workflows/rnmobile-android-runner.yml @@ -14,11 +14,12 @@ concurrency: jobs: test: - runs-on: macos-latest + runs-on: macos-12 if: ${{ github.repository == 'WordPress/gutenberg' || github.event_name == 'pull_request' }} strategy: matrix: native-test-name: [gutenberg-editor-initial-html] + api-level: [29] steps: - name: checkout @@ -29,7 +30,6 @@ jobs: with: distribution: 'temurin' java-version: '11' - cache: 'gradle' - name: Use desired version of NodeJS uses: actions/setup-node@8c91899e586c5b171469028077307d293428b516 # v3.5.1 @@ -39,17 +39,39 @@ jobs: - run: npm ci - - name: Restore Gradle cache + - name: Gradle cache + uses: gradle/gradle-build-action@3fbe033aaae657f011f88f29be9e65ed26bd29ef # v2.3.3 + + - name: AVD cache uses: actions/cache@9b0c1fce7a93df8e3bb8926b0d6e9d89e92f20a7 # v3.0.11 + id: avd-cache + with: + path: | + ~/.android/avd/* + ~/.android/adb* + key: avd-${{ matrix.api-level }} + + - name: Create AVD and generate snapshot for caching + if: steps.avd-cache.outputs.cache-hit != 'true' + uses: reactivecircus/android-emulator-runner@50986b1464923454c95e261820bc626f38490ec0 # v2.27.0 with: - path: ~/.gradle/caches - key: ${{ runner.os }}-gradle-${{ hashFiles('**/*.gradle') }} + api-level: ${{ matrix.api-level }} + force-avd-creation: false + emulator-options: -no-window -gpu swiftshader_indirect -noaudio -no-boot-anim -camera-back none + disable-animations: false + arch: x86_64 + profile: Nexus 6 + script: echo "Generated AVD snapshot for caching." - - uses: reactivecircus/android-emulator-runner@50986b1464923454c95e261820bc626f38490ec0 # v2.27.0 + - name: Run tests + uses: reactivecircus/android-emulator-runner@50986b1464923454c95e261820bc626f38490ec0 # v2.27.0 with: - api-level: 28 - emulator-build: 7425822 # https://github.com/ReactiveCircus/android-emulator-runner/issues/160#issuecomment-868615730 - profile: pixel_xl + api-level: ${{ matrix.api-level }} + force-avd-creation: false + emulator-options: -no-snapshot-save -no-window -gpu swiftshader_indirect -noaudio -no-boot-anim -camera-back none + disable-animations: true + arch: x86_64 + profile: Nexus 6 script: npm run native test:e2e:android:local ${{ matrix.native-test-name }} - uses: actions/upload-artifact@83fd05a356d7e2593de66fc9913b3002723633cb # v3.1.1 diff --git a/bin/plugin/cli.js b/bin/plugin/cli.js index ed2fef4a5b05f..0101c9169171e 100755 --- a/bin/plugin/cli.js +++ b/bin/plugin/cli.js @@ -94,6 +94,10 @@ program .command( 'performance-tests [branches...]' ) .alias( 'perf' ) .option( ...ciOption ) + .option( + '--rounds ', + 'Run each test suite this many times for each branch; results are summarized, default = 1' + ) .option( '--tests-branch ', "Use this branch's performance test files" diff --git a/bin/plugin/commands/performance.js b/bin/plugin/commands/performance.js index 2aa8a2c9a7f5c..31a537f373731 100644 --- a/bin/plugin/commands/performance.js +++ b/bin/plugin/commands/performance.js @@ -22,6 +22,7 @@ const config = require( '../config' ); * @typedef WPPerformanceCommandOptions * * @property {boolean=} ci Run on CI. + * @property {number=} rounds Run each test suite this many times for each branch. * @property {string=} testsBranch The branch whose performance test files will be used for testing. * @property {string=} wpVersion The WordPress version to be used as the base install for testing. */ @@ -176,6 +177,7 @@ async function runTestSuite( testSuite, performanceTestDirectory ) { */ async function runPerformanceTests( branches, options ) { const runningInCI = !! process.env.CI || !! options.ci; + const TEST_ROUNDS = options.rounds || 1; // The default value doesn't work because commander provides an array. if ( branches.length === 0 ) { @@ -239,7 +241,7 @@ async function runPerformanceTests( branches, options ) { log( ' >> Installing dependencies and building packages' ); await runShellScript( - 'npm ci && npm run build:packages', + 'npm ci && node ./bin/packages/build.js', performanceTestDirectory ); log( ' >> Creating the environment folders' ); @@ -258,12 +260,26 @@ async function runPerformanceTests( branches, options ) { await runShellScript( 'mkdir ' + environmentDirectory ); await runShellScript( `cp -R ${ baseDirectory } ${ buildPath }` ); - log( ` >> Fetching the ${ formats.success( branch ) } branch` ); - // @ts-ignore - await SimpleGit( buildPath ).reset( 'hard' ).checkout( branch ); + const fancyBranch = formats.success( branch ); - log( ` >> Building the ${ formats.success( branch ) } branch` ); - await runShellScript( 'npm ci && npm run build', buildPath ); + if ( branch === options.testsBranch ) { + log( + ` >> Re-using the testing branch for ${ fancyBranch }` + ); + await runShellScript( + `cp -R ${ performanceTestDirectory } ${ buildPath }` + ); + } else { + log( ` >> Fetching the ${ fancyBranch } branch` ); + // @ts-ignore + await SimpleGit( buildPath ).reset( 'hard' ).checkout( branch ); + + log( ` >> Building the ${ fancyBranch } branch` ); + await runShellScript( + 'npm ci && npm run prebuild:packages && node ./bin/packages/build.js && npx wp-scripts build', + buildPath + ); + } await runShellScript( 'cp ' + @@ -330,7 +346,7 @@ async function runPerformanceTests( branches, options ) { /** @type {Array>} */ const rawResults = []; // Alternate three times between branches. - for ( let i = 0; i < 3; i++ ) { + for ( let i = 0; i < TEST_ROUNDS; i++ ) { rawResults[ i ] = {}; for ( const branch of branches ) { // @ts-ignore diff --git a/changelog.txt b/changelog.txt index 73cf3773d0a3d..2d52f25c1bf8a 100644 --- a/changelog.txt +++ b/changelog.txt @@ -1,5 +1,878 @@ == Changelog == += 14.7.3 = + +## Changelog + +### Bug fixes + +- Fix typing performance issue for container blocks. ([46527](https://github.com/WordPress/gutenberg/pull/46527)) + +## Contributors + +The following contributors merged PRs in this release: + +@youknowriad + + += 14.7.2 = + + + +## Changelog + +### Bug Fixes + +- Fix fatal error when using the plugin with PHP 8 and WordPress 6.0 by checking if block_type asset properties are set. ([46488](https://github.com/WordPress/gutenberg/pull/46488)) + + +## First time contributors + +The following PRs were merged by first time contributors: + + + +## Contributors + +The following contributors merged PRs in this release: + +@noahtallen + + += 14.7.1 = + + + +## Changelog + +### Bug Fixes + +#### Layout +- Only pass parentLayout to block edit if not empty. ([46390](https://github.com/WordPress/gutenberg/pull/46390)) + +## Contributors + +The following contributors merged PRs in this release: + +@tellthemachines + + += 14.7.0 = + + + +## Changelog + +### Enhancements + +#### Style Engine +- Style engine: Trim multiple selector strings. ([45873](https://github.com/WordPress/gutenberg/pull/45873)) + +#### Block Library +- Heading Block: Add a wp-block-heading CSS class. ([42122](https://github.com/WordPress/gutenberg/pull/42122)) +- Nav Block: Clarify explanation of how 'Convert to Links' works in Page List block. ([45394](https://github.com/WordPress/gutenberg/pull/45394)) +- Nav Block: Add label field to navigation link and navigation submenu. ([45964](https://github.com/WordPress/gutenberg/pull/45964)) +- Nav Block: Add link URL to the navigation submenu inspector controls. ([45816](https://github.com/WordPress/gutenberg/pull/45816)) +- Nav Block: Fix for navigation anchor links to close modal. ([45829](https://github.com/WordPress/gutenberg/pull/45829)) +- Template Part Block: Colorize template parts and Reusable blocks. ([45473](https://github.com/WordPress/gutenberg/pull/45473)) +- List: Allow pasting pre/code. ([45016](https://github.com/WordPress/gutenberg/pull/45016)) +- Page List: Enable page list to expand in list view. ([45776](https://github.com/WordPress/gutenberg/pull/45776)) +- Page List: Add a starting page for page list block's hierarchy. ([45861](https://github.com/WordPress/gutenberg/pull/45861)) +- Page List Item: Hide edit button. ([46163](https://github.com/WordPress/gutenberg/pull/46163)) +- Site Logo: Apply width to logo container in editor. ([45821](https://github.com/WordPress/gutenberg/pull/45821)) +- Table Block: Support colspan attribute in table HTML, including when pasting. ([45981](https://github.com/WordPress/gutenberg/pull/45981)) + +#### Components +- Add themeable background color. ([45466](https://github.com/WordPress/gutenberg/pull/45466)) +- Autocomplete: Only show UI on user input. ([45904](https://github.com/WordPress/gutenberg/pull/45904)) +- Bump `DateTimePicker` deprecated prop removal version. ([46006](https://github.com/WordPress/gutenberg/pull/46006)) +- ComboboxControl: Add new opt-in prop. ([45796](https://github.com/WordPress/gutenberg/pull/45796)) +- FocalPointPicker: Add new opt-in prop. ([45958](https://github.com/WordPress/gutenberg/pull/45958)) +- Global styles: Add onChange actions to color palette items. ([45681](https://github.com/WordPress/gutenberg/pull/45681)) +- InputControl: Add `help` prop. ([45931](https://github.com/WordPress/gutenberg/pull/45931)) +- RangeControl: Remove margin override and add new opt-in prop. ([45985](https://github.com/WordPress/gutenberg/pull/45985)) +- SearchControl: Remove margin overrides and add new opt-in prop. ([46081](https://github.com/WordPress/gutenberg/pull/46081)) +- Storybook: Opt in to story store v7. ([42486](https://github.com/WordPress/gutenberg/pull/42486)) +- ToggleControl text overflows when it has a long label. ([45962](https://github.com/WordPress/gutenberg/pull/45962)) +- useControlledValue: Let TypeScript infer the return type. ([46164](https://github.com/WordPress/gutenberg/pull/46164)) + +#### Inspector Controls +- Sidebar: Add list view tab for Navigation block et al. ([45483](https://github.com/WordPress/gutenberg/pull/45483)) +- Sidebar: Only render sidebar tabs possessing items to display. ([45991](https://github.com/WordPress/gutenberg/pull/45991)) +- Sidebar: Rename appearance tab to styles. ([46027](https://github.com/WordPress/gutenberg/pull/46027)) +- Sidebar: Split block tools into menu, settings, and appearance tabs. ([45005](https://github.com/WordPress/gutenberg/pull/45005)) + +#### Design Tools +- Min Height: Add height control component with slider. ([45875](https://github.com/WordPress/gutenberg/pull/45875)) +- Spacing: Make visualiser appear on focus. ([46096](https://github.com/WordPress/gutenberg/pull/46096)) + +#### Block Editor +- [Inserter]: Replace text in `Reusable` tab with an icon. ([45851](https://github.com/WordPress/gutenberg/pull/45851)) +- [Inserter]: Update pattern explorer button css. ([45735](https://github.com/WordPress/gutenberg/pull/45735)) +- [Inserter]: Add media tab. ([44918](https://github.com/WordPress/gutenberg/pull/44918)) + +#### Patterns +- [Pattern Directory]: Add categories endpoint. ([45749](https://github.com/WordPress/gutenberg/pull/45749)) +- [Patterns]: Update pattern category descriptions. ([46005](https://github.com/WordPress/gutenberg/pull/46005)) + +#### Nested / Inner Blocks +- Mark applying block templates not persistent. ([45843](https://github.com/WordPress/gutenberg/pull/45843)) + +#### Rich Text +- Create undo level before autocorrect. ([45670](https://github.com/WordPress/gutenberg/pull/45670)) + +#### Layout +- Add Layout controls to children of Flex layout blocks. ([45364](https://github.com/WordPress/gutenberg/pull/45364)) + + +### Bug Fixes + +#### Preferences +- Disable distraction free prefference effects on small viewports. ([45591](https://github.com/WordPress/gutenberg/pull/45591)) + +#### Block Library +- List Block: Fixed a bug that List block attributes were reset in 6.1.1. ([46000](https://github.com/WordPress/gutenberg/pull/46000)) +- Gallery: Use unbound query when fetching image details. ([46143](https://github.com/WordPress/gutenberg/pull/46143)) +- Heading: Add block classname deprecation. ([46138](https://github.com/WordPress/gutenberg/pull/46138)) +- Page List: If no parent page is set, still render all children. ([45967](https://github.com/WordPress/gutenberg/pull/45967)) +- Page List: Render the children correctly in the editor. ([46165](https://github.com/WordPress/gutenberg/pull/46165)) +- Post Author: Avoid errors when the user avatars are disabled. ([45989](https://github.com/WordPress/gutenberg/pull/45989)) +- Nav Block: Navigation menu doesn't appear when hamburger clicked on. ([45773](https://github.com/WordPress/gutenberg/pull/45773)) + +#### Block Editor +- Fix broken Link Control hook. ([46113](https://github.com/WordPress/gutenberg/pull/46113)) +- Fix inserter tab panel content buttons' position. ([45800](https://github.com/WordPress/gutenberg/pull/45800)) +- Block editor: rich text: Return early if __experimentalUndo is not defined. ([46152](https://github.com/WordPress/gutenberg/pull/46152)) + +#### Global Styles +- Global Style Context: Consider global user styles ready if a theme has none. ([46073](https://github.com/WordPress/gutenberg/pull/46073)) +- Merged data should consider origin to return early. ([45969](https://github.com/WordPress/gutenberg/pull/45969)) + +#### Components +- Remove CircleIndicatorWrapper `focus-visible` outline. ([45758](https://github.com/WordPress/gutenberg/pull/45758)) +- `ColorPalette`: Show `Clear` button even when `colors` array is empty. ([46001](https://github.com/WordPress/gutenberg/pull/46001)) + +#### Site Editor +- Fix template list width. ([45888](https://github.com/WordPress/gutenberg/pull/45888)) +- Prevent edit-post from being loaded in edit-site. ([45895](https://github.com/WordPress/gutenberg/pull/45895)) + +#### CSS & Styling +- Fix the editor area height. ([45799](https://github.com/WordPress/gutenberg/pull/45799)) + +#### Full Site Editing +- Ensure post-featured-image block is in_the_loop() for BC with core and plugins, and to fix lazy-loading. ([45534](https://github.com/WordPress/gutenberg/pull/45534)) + +### Accessibility + +- Add "Testing Instructions for Keyboard" to PR template to encourage accessibility testing. ([45957](https://github.com/WordPress/gutenberg/pull/45957)) +- BlockVariationPicker: Remove Unnecessary ARIA Role. ([45916](https://github.com/WordPress/gutenberg/pull/45916)) +- Sidebar Tabs: Set default tab to first available. ([45998](https://github.com/WordPress/gutenberg/pull/45998)) +- `TabPanel`: Support manual tab activation. ([46004](https://github.com/WordPress/gutenberg/pull/46004)) +- - Constrained tabbing: Fix unstable behavior in firefox. ([42653](https://github.com/WordPress/gutenberg/pull/42653)) + +### Performance + +- Work on refactor away from Lodash to reduce build size continued" ([see 13.7](https://make.wordpress.org/core/2022/07/20/whats-new-in-gutenberg-13-7-20-july/)) + +#### Global Styles +- Add `WP_Object_Cache` to the `gutenberg_get_global_settings` method. ([45372](https://github.com/WordPress/gutenberg/pull/45372)) +- Global styles WP_Query. ([46043](https://github.com/WordPress/gutenberg/pull/46043)) +- Ignore cached `wp_theme_has_theme_json` when `WP_DEBUG` is enabled. ([45882](https://github.com/WordPress/gutenberg/pull/45882)) +- Make `theme.json` object caches non persistent. ([46150](https://github.com/WordPress/gutenberg/pull/46150)) +- Remove `test_global_styles_user_cpt_change_invalidates_cached_stylesheet`. ([45993](https://github.com/WordPress/gutenberg/pull/45993)) +- Update `gutenberg_get_global_stylesheet` to use `WP_Object_Cache`. ([45679](https://github.com/WordPress/gutenberg/pull/45679)) +- Update which origins are queried for `gutenberg_get_global_settings`. ([45971](https://github.com/WordPress/gutenberg/pull/45971)) + +#### Post Editor +- useBlockEditorSettings: Return const empty array to avoid rerenders. ([46117](https://github.com/WordPress/gutenberg/pull/46117)) + +#### Block Editor +- Update the attributes reducer to use a map instead of a regular object. ([46146](https://github.com/WordPress/gutenberg/pull/46146)) + + +### Experiments + +#### Block Library +- Nav Block: Add basic edit button UI to Nav block offcanvas editor. ([45815](https://github.com/WordPress/gutenberg/pull/45815)) +- Nav Block: Add submenu menu item to list view. ([45794](https://github.com/WordPress/gutenberg/pull/45794)) +- Nav Block: Alternative: Add inserter to Nav block offcanvas experiment. ([45947](https://github.com/WordPress/gutenberg/pull/45947)) +- Nav Block: Display inserter popover in offcanvas UI. ([46013](https://github.com/WordPress/gutenberg/pull/46013)) +- Nav Block: List View - Stop child item selecting a parent which is already selected. ([45860](https://github.com/WordPress/gutenberg/pull/45860)) +- Nav Block: Add simple back button to inspector controls. ([45852](https://github.com/WordPress/gutenberg/pull/45852)) +- Nav Block: Move color controls to support panel. ([46049](https://github.com/WordPress/gutenberg/pull/46049)) +- Nav Block: Enable easier drag and drop for navigation building. ([45906](https://github.com/WordPress/gutenberg/pull/45906)) +- Nav Block: Hide the create new menu button if the experiment is enabled. ([46019](https://github.com/WordPress/gutenberg/pull/46019)) +- Navigation List view: Fix incorect class. ([46129](https://github.com/WordPress/gutenberg/pull/46129)) +- Navigation List view: Include offcanvas specific styles. ([45963](https://github.com/WordPress/gutenberg/pull/45963)) +- Navigation List view: Scroll horizontally when table overflows. ([45966](https://github.com/WordPress/gutenberg/pull/45966)) + + +### Documentation + +- (docs) Document the special case of shipping point releases when new release branch already exists. ([46083](https://github.com/WordPress/gutenberg/pull/46083)) +- Added InspectorControls import to example. ([45872](https://github.com/WordPress/gutenberg/pull/45872)) +- Fix NavigableRegion README. ([45879](https://github.com/WordPress/gutenberg/pull/45879)) +- Fix link & code markdown. ([45708](https://github.com/WordPress/gutenberg/pull/45708)) +- Navigation: Adds a warning about duplicate code for the future. ([45844](https://github.com/WordPress/gutenberg/pull/45844)) +- Storybook: Add link to component folder on GitHub, retire Storysource. ([45727](https://github.com/WordPress/gutenberg/pull/45727)) +- Style Engine: Add first draft of contributing doc. ([45930](https://github.com/WordPress/gutenberg/pull/45930)) +- Update applying-styles-with-stylesheets.md. ([45925](https://github.com/WordPress/gutenberg/pull/45925)) + +### Code Quality + +#### Components +- Cleanup the BlockPreview component. ([45936](https://github.com/WordPress/gutenberg/pull/45936)) +- Convert the `Snackbar` component to TypeScript. ([45472](https://github.com/WordPress/gutenberg/pull/45472)) +- Fix ESLint violations in `ContextSystemProvider` tests. ([46010](https://github.com/WordPress/gutenberg/pull/46010)) +- Fix ESLint violations in `NoticeList` tests. ([46011](https://github.com/WordPress/gutenberg/pull/46011)) +- Fix `no-node-access` in `Grid` tests. ([45900](https://github.com/WordPress/gutenberg/pull/45900)) +- Fix `no-node-access` in `Sandbox` tests. ([45908](https://github.com/WordPress/gutenberg/pull/45908)) +- Fix `no-node-access` in `Text` tests. ([45898](https://github.com/WordPress/gutenberg/pull/45898)) +- Fix `no-node-access` in `Theme` tests. ([45896](https://github.com/WordPress/gutenberg/pull/45896)) +- Fix `no-node-access` violation in `ControlLabel` tests. ([46007](https://github.com/WordPress/gutenberg/pull/46007)) +- Fix `no-node-access` violations in `Card` tests. ([46158](https://github.com/WordPress/gutenberg/pull/46158)) +- Fix `no-node-access` violations in `Disabled` tests. ([46156](https://github.com/WordPress/gutenberg/pull/46156)) +- Improve `BoxControl` tests. ([45968](https://github.com/WordPress/gutenberg/pull/45968)) +- Improve `Dropdown` tests. ([45911](https://github.com/WordPress/gutenberg/pull/45911)) +- LinkedButton: Remove unnecessary span tag. ([46063](https://github.com/WordPress/gutenberg/pull/46063)) +- TextControl: Restrict `type` prop in TypeScript. ([45433](https://github.com/WordPress/gutenberg/pull/45433)) +- Tooltip: Add readme and unit tests for `shortcut` prop. ([46092](https://github.com/WordPress/gutenberg/pull/46092)) +- `NumberControl`: Refactor styles/tests/stories to TypeScript, replace `fireEvent` with `user-event`. ([45990](https://github.com/WordPress/gutenberg/pull/45990)) +- useBaseField: Convert component to TypeScript. ([45712](https://github.com/WordPress/gutenberg/pull/45712)) +- Small refactoring to the NavigableRegion component. ([45849](https://github.com/WordPress/gutenberg/pull/45849)) + +#### Block Library +- ESLint: Fix minor ESLint warning in `LinkUI`. ([46161](https://github.com/WordPress/gutenberg/pull/46161)) +- Fix ESLint warnings in tests. ([46034](https://github.com/WordPress/gutenberg/pull/46034)) +- Fix invalid attribute markup in core/home-link block. ([46089](https://github.com/WordPress/gutenberg/pull/46089)) +- Link UI: Destructure the props earlier in the component. ([46209](https://github.com/WordPress/gutenberg/pull/46209)) +- Navigation Link UI: Try to align both files. ([46205](https://github.com/WordPress/gutenberg/pull/46205)) +- Navigation Menu Selector: Share the functions needed for the NavigationMenuSelector. ([46053](https://github.com/WordPress/gutenberg/pull/46053)) +- Navigation: Extract components. ([45850](https://github.com/WordPress/gutenberg/pull/45850)) +- Navigation: Reduce duplicate code. ([45779](https://github.com/WordPress/gutenberg/pull/45779)) +- Navigation: Remove unused clientId prop. ([46020](https://github.com/WordPress/gutenberg/pull/46020)) +- Post Featured Image: Only get the post title when rendering alt text. ([45835](https://github.com/WordPress/gutenberg/pull/45835)) +- Reduce prop drilling in Block Card component. ([46052](https://github.com/WordPress/gutenberg/pull/46052)) +- Refactor link creation UI to standalone component/file. ([46031](https://github.com/WordPress/gutenberg/pull/46031)) +- Remove Nav block specific classes from Nav offcanvas Link UI. ([46182](https://github.com/WordPress/gutenberg/pull/46182)) +- Remove WrappedNavigationMenuSelector. ([46056](https://github.com/WordPress/gutenberg/pull/46056)) +- Update offcanvas back button to select parent Nav block and limited to Nav block only. ([46037](https://github.com/WordPress/gutenberg/pull/46037)) +- Don't check if constants set by `wp_initial_constants()` are defined. ([45979](https://github.com/WordPress/gutenberg/pull/45979)) + +#### Block Editor +- LinkControl unit tests: Use user.type to type into search field. ([45802](https://github.com/WordPress/gutenberg/pull/45802)) +- Simplify api for link UI abstraction to use a single prop for the value. ([46189](https://github.com/WordPress/gutenberg/pull/46189)) +- URLInput: Keep the search results label in sync with the results list. ([45806](https://github.com/WordPress/gutenberg/pull/45806)) +- Use @wordpress/escape-html escapeHTML in Link UI in preference to Lodash method. ([46184](https://github.com/WordPress/gutenberg/pull/46184)) + +#### Global Styles +- Cleaner logic in wp_theme_has_theme_json. ([45950](https://github.com/WordPress/gutenberg/pull/45950)) +- Remove usage of wp_get_theme. ([45770](https://github.com/WordPress/gutenberg/pull/45770)) + +#### Post Editor +- Update BlockCard to pass className instead of isSynced prop. ([46021](https://github.com/WordPress/gutenberg/pull/46021)) + +#### Full Site Editing +- Block editor: Separate content styles for the iframe. ([44298](https://github.com/WordPress/gutenberg/pull/44298)) + +### Tools + +#### Build Tooling +- Bump caniuse-lite version. ([46093](https://github.com/WordPress/gutenberg/pull/46093)) + +#### Testing +- Fix Quote block's unwrap end-to-end test. ([46168](https://github.com/WordPress/gutenberg/pull/46168)) +- Remove 'response.deleted' check. ([45992](https://github.com/WordPress/gutenberg/pull/45992)) +- Warning: Fix ESLint warnings in tests. ([46033](https://github.com/WordPress/gutenberg/pull/46033)) +- ESLint: Enable `testing-library/no-container` rule. ([46160](https://github.com/WordPress/gutenberg/pull/46160)) +- Element: Fix `no-node-access` in `createInterpolateElement`. ([45894](https://github.com/WordPress/gutenberg/pull/45894)) +- Block Editor: Fix ESLint warnings in `MediaUpload` tests. ([46035](https://github.com/WordPress/gutenberg/pull/46035)) +- Block Editor: Fix `no-node-access` in `RecursionProvider` tests. ([45902](https://github.com/WordPress/gutenberg/pull/45902)) +- Block Editor: Fix block alignment tests for React 18. ([45937](https://github.com/WordPress/gutenberg/pull/45937)) +- Block Editor: Wait for popover positioning in `MediaReplaceFlow` tests. ([45863](https://github.com/WordPress/gutenberg/pull/45863)) +- Compose: Fix 'no-container' violations in 'useDisabled' tests. ([45797](https://github.com/WordPress/gutenberg/pull/45797)) +- Compose: Fix ESLint violations in `withGlobalEvents` tests. ([46012](https://github.com/WordPress/gutenberg/pull/46012)) + +## First time contributors + +The following PRs were merged by first time contributors: + +- @albarin: Remove 'response.deleted' check. ([45992](https://github.com/WordPress/gutenberg/pull/45992)) +- @artemiomorales: Clarify explanation of how 'Convert to Links' works in Page List block. ([45394](https://github.com/WordPress/gutenberg/pull/45394)) +- @coreyworrell: Fix for navigation anchor links to close modal. ([45829](https://github.com/WordPress/gutenberg/pull/45829)) +- @devanshijoshi9: Components: ToggleControl text overflows when it has a long label. ([45962](https://github.com/WordPress/gutenberg/pull/45962)) +- @flexseth: Added InspectorControls import to example. ([45872](https://github.com/WordPress/gutenberg/pull/45872)) +- @hiyascout: Update applying-styles-with-stylesheets.md. ([45925](https://github.com/WordPress/gutenberg/pull/45925)) +- @marissa-makes: BlockVariationPicker: Remove Unnecessary ARIA Role. ([45916](https://github.com/WordPress/gutenberg/pull/45916)) +- @mpkelly: Support `colspan` attribute in `table` HTML, including when pasting. ([45981](https://github.com/WordPress/gutenberg/pull/45981)) +- @TobiasBg: Fix invalid attribute markup in core/home-link block. ([46089](https://github.com/WordPress/gutenberg/pull/46089)) + + +## Contributors + +The following contributors merged PRs in this release: + +@aaronrobertshaw @adamziel @afercia @ajlende @albarin @alexstine @andrewserong @artemiomorales @brookewp @chad1008 @ciampo @coreyworrell @ddryo @devanshijoshi9 @draganescu @ellatrix @felixarntz @flexseth @fullofcaffeine @geriux @getdave @glendaviesnz @gvgvgvijayan @hiyascout @jsnajdr @kienstra @MaggieCabrera @Mamaduka @marissa-makes @mikachan @mirka @mmtr @mpkelly @ntsekouras @oandregal @ocean90 @oguzkocer @ramonjd @scruffian @SiobhyB @spacedmonkey @stokesman @t-hamano @tellthemachines @TobiasBg @tyxla @walbo @youknowriad + + += 14.6.1 = + +## Changelog + +### Fixes + +#### Global Styles +- Fix the `upgrader_process_complete` hook for `wp_theme_has_theme_json`. ([45881](https://github.com/WordPress/gutenberg/pull/45881)) +- Update `wp_theme_has_theme_json` to use `WP_Object_Cache`. ([45543](https://github.com/WordPress/gutenberg/pull/45543)) + +### Bug Fixes +- Tag Processor: Prevent bugs from pre-PHP8 strspn/strcspn behavior. ([45822](https://github.com/WordPress/gutenberg/pull/45822)) + +## Contributors + +The following contributors merged PRs in this release: + +@oandregal @dmsnell + + + += 14.6.0 = + +## Changelog + +### Enhancements + +#### Block Library + +- Latest posts: Add color support. ([41874](https://github.com/WordPress/gutenberg/pull/41874)) +- Latest posts and latest comments: Add spacing support. ([45110](https://github.com/WordPress/gutenberg/pull/45110)) +- Navigation: Adds a list view. ([45546](https://github.com/WordPress/gutenberg/pull/45546)) +- Navigation: Add a new ManageMenusButton component. ([45782](https://github.com/WordPress/gutenberg/pull/45782)) +- Navigation: Reposition the navigation selector. ([45555](https://github.com/WordPress/gutenberg/pull/45555)) +- Navigation Link: Add the URL field to the Navigation Link inspector controls. ([45751](https://github.com/WordPress/gutenberg/pull/45751)) +- Author: Make the Author selector display all users instead of just 10. ([45640](https://github.com/WordPress/gutenberg/pull/45640)) +- Columns: Add transform to unwrap the contents. ([45666](https://github.com/WordPress/gutenberg/pull/45666)) +- Read More: Add aria-label and screen reader text. ([45490](https://github.com/WordPress/gutenberg/pull/45490)) +- Group: Use a variation picker in the placeholder. ([43496](https://github.com/WordPress/gutenberg/pull/43496)) + +#### Components + +- Use new theming accent color in all components. ([45289](https://github.com/WordPress/gutenberg/pull/45289)) +- CheckboxControl: Replace margin overrides with new opt-in prop. ([45434](https://github.com/WordPress/gutenberg/pull/45434)) +- FocalPointPicker: Update the design of the focal point handle. ([45053](https://github.com/WordPress/gutenberg/pull/45053)) +- FontSizePicker: Update hint text to match the design. ([44966](https://github.com/WordPress/gutenberg/pull/44966)) +- CheckboxControl: Move icons out of labels. ([45535](https://github.com/WordPress/gutenberg/pull/45535)) + +#### Block Editor + +- Converts paragraphs to headings with keyboard shortcuts. ([44681](https://github.com/WordPress/gutenberg/pull/44681)) +- Restore the empty paragraph inserter. ([45542](https://github.com/WordPress/gutenberg/pull/45542)) +- Transform: Select all blocks if the result has more than one block. ([45015](https://github.com/WordPress/gutenberg/pull/45015)) +- Content-only locked patterns: Move "Modify" to the ellipsis menu. ([45391](https://github.com/WordPress/gutenberg/pull/45391)) +- Patterns: Adjust the space in the pattern explorer list. ([45730](https://github.com/WordPress/gutenberg/pull/45730)) +- Update: Lock icon to outline. ([45645](https://github.com/WordPress/gutenberg/pull/45645)) +- Don't use capital case for 'Distraction free' strings. ([45538](https://github.com/WordPress/gutenberg/pull/45538)) +- Replace Justification/Orientation controls with ToggleGroupControl. ([45637](https://github.com/WordPress/gutenberg/pull/45637)) + +#### Site Editor + +- Replace FSE with Site Editor. ([45699](https://github.com/WordPress/gutenberg/pull/45699)) + +#### Design Tools + +- Add a minHeight block support under dimensions. ([45300](https://github.com/WordPress/gutenberg/pull/45300)) +- Hide the block toolbar when the spacing visualizer is showing. ([45131](https://github.com/WordPress/gutenberg/pull/45131)) + +#### Global Styles + +- Elements: Add a text decoration control to link elements. ([45643](https://github.com/WordPress/gutenberg/pull/45643)) +- Global styles: Convert preset font size values to CSS vars. ([44967](https://github.com/WordPress/gutenberg/pull/44967)) +- Fluid typography: Adjust font size min and max rules. ([45536](https://github.com/WordPress/gutenberg/pull/45536)) +- Try generating random color palettes. ([40988](https://github.com/WordPress/gutenberg/pull/40988)) + +#### Plugin + +- Updates tested up to version to 6.1. ([45630](https://github.com/WordPress/gutenberg/pull/45630)) + +#### Patterns + +- Pattern Directory API: Add support for pagination parameters. ([45293](https://github.com/WordPress/gutenberg/pull/45293)) +- Update bundled patterns compat directory. ([45620](https://github.com/WordPress/gutenberg/pull/45620)) + +### Bug Fixes + +#### Block Library + +- Change the order of the pseudo-states in the pseudo-selectors array. ([45559](https://github.com/WordPress/gutenberg/pull/45559)) +- Cover: Avoid content loss when the templateLock value is all or contentOnly. ([45632](https://github.com/WordPress/gutenberg/pull/45632)) +- Fix alignment of create new post link. ([45638](https://github.com/WordPress/gutenberg/pull/45638)) +- Fix navigation appender position to prevent obstructing its items. ([43530](https://github.com/WordPress/gutenberg/pull/43530)) +- Fix: Button block text alignment. ([45663](https://github.com/WordPress/gutenberg/pull/45663)) +- Query Pagination: Fix positioning of the next link in editor when the parent is selected. ([45651](https://github.com/WordPress/gutenberg/pull/45651)) +- Site Logo: Use the correct home URL setting. ([45476](https://github.com/WordPress/gutenberg/pull/45476)) +- Switch background color to text color on the block separator. ([44943](https://github.com/WordPress/gutenberg/pull/44943)) +- Table Block: Apply borders and padding on both front end and editor. ([45069](https://github.com/WordPress/gutenberg/pull/45069)) +- Table block: Fix error in margin value. ([45674](https://github.com/WordPress/gutenberg/pull/45674)) +- Template Part Block: Update block isActive method. ([45672](https://github.com/WordPress/gutenberg/pull/45672)) +- Navigation: Fix overflowing menu name in the navigation selector dropdown. ([45647](https://github.com/WordPress/gutenberg/pull/45647)) + +#### Accessibility + +- Fix focus return when closing the Post publish panel. ([45623](https://github.com/WordPress/gutenberg/pull/45623)) +- Fix navigate regions backwards for macOS Firefox and Safari. ([45019](https://github.com/WordPress/gutenberg/pull/45019)) +- Fix the Save buttons labeling and tooltip. ([43952](https://github.com/WordPress/gutenberg/pull/43952)) +- Fix the navigate regions focus style. ([45369](https://github.com/WordPress/gutenberg/pull/45369)) +- Fix: Contrast checker appears unexpectedly on some blocks. ([45639](https://github.com/WordPress/gutenberg/pull/45639)) +- Fix: Contrast checker does not update properly. ([45686](https://github.com/WordPress/gutenberg/pull/45686)) + +#### Components + +- Autocomplete: Fix unexpected block insertion during IME composition. ([45510](https://github.com/WordPress/gutenberg/pull/45510)) +- Fix ESLint warning for Dashicon. ([45795](https://github.com/WordPress/gutenberg/pull/45795)) +- FormTokenField: Fix duplicate input in IME composition. ([45607](https://github.com/WordPress/gutenberg/pull/45607)) +- Making size prop work for icon components using dash icon strings. ([45593](https://github.com/WordPress/gutenberg/pull/45593)) +- ToolsPanel: Prevent calling deselect when panel remounts. ([45673](https://github.com/WordPress/gutenberg/pull/45673)) +- Color Picker: Prevent all number fields to become 0 when one of them is an empty string. ([45649](https://github.com/WordPress/gutenberg/pull/45649)) +- ToggleGroupControl: Only show the enclosing border when `isBlock`. ([45492](https://github.com/WordPress/gutenberg/pull/45492)) +- Autocomplete: Check key events more strictly in IME composition. ([45626](https://github.com/WordPress/gutenberg/pull/45626)) + +#### CSS & Styling + +- Inherit font from theme on overlay close button. ([45635](https://github.com/WordPress/gutenberg/pull/45635)) +- Navigation: Fix font inheritance when using text menu button. ([45514](https://github.com/WordPress/gutenberg/pull/45514)) +- Remove hover style to button on dark block tools UI. ([45653](https://github.com/WordPress/gutenberg/pull/45653)) +- Remove width from block mover button focus style. ([45665](https://github.com/WordPress/gutenberg/pull/45665)) +- Site editor hover/select: Fix double border. ([45589](https://github.com/WordPress/gutenberg/pull/45589)) +- Remove duplicate output of existing classnames in layout classnames. ([45499](https://github.com/WordPress/gutenberg/pull/45499)) + +#### Post Editor + +- BlockManagerCategory: Fix styles for indeterminate. ([45564](https://github.com/WordPress/gutenberg/pull/45564)) +- Fix: Updated names from List View to Document Overview. ([45524](https://github.com/WordPress/gutenberg/pull/45524)) +- Strip HTML from Post Title when pasting multiline title containing HTML. ([35825](https://github.com/WordPress/gutenberg/pull/35825)) + +#### Site Editor + +- Decode entities in template title and description. ([45716](https://github.com/WordPress/gutenberg/pull/45716)) +- Link to homeUrl from site editor view menu. ([45475](https://github.com/WordPress/gutenberg/pull/45475)) + +#### Block Editor + +- Fix Link UI popover positioning when inspector control input is focused. ([45661](https://github.com/WordPress/gutenberg/pull/45661)) +- Paste: Fix list only paste from Google documentation. ([45498](https://github.com/WordPress/gutenberg/pull/45498)) +- Make Manage Reusable blocks match similar links. ([45641](https://github.com/WordPress/gutenberg/pull/45641))([45689](https://github.com/WordPress/gutenberg/pull/45689)) +- List View: Disable branch expansion when block editing is locked. ([45541](https://github.com/WordPress/gutenberg/pull/45541)) +- Spacing visualizer: Fix display of unexpected visualizer for certain mouse actions. ([45739](https://github.com/WordPress/gutenberg/pull/45739)) + +### Experiments + +- A list view duplicate for use in navigation list view experiment. ([45544](https://github.com/WordPress/gutenberg/pull/45544)) +- Introduce experiment for inspector based navigation editing. ([45515](https://github.com/WordPress/gutenberg/pull/45515)) + +### Documentation + +- Add missing CHANGELOG entry. ([45691](https://github.com/WordPress/gutenberg/pull/45691)) +- Change Title: How to use JavaScript with Gutenberg. ([45323](https://github.com/WordPress/gutenberg/pull/45323)) +- Docs: Update the readme for the integration test fixtures. ([45581](https://github.com/WordPress/gutenberg/pull/45581)) +- Summarize "Available commands" section and refer them it to `scripts` documentation. ([45636](https://github.com/WordPress/gutenberg/pull/45636)) +- Update applying-styles-with-stylesheets.md. ([45604](https://github.com/WordPress/gutenberg/pull/45604)) +- [create-block] Reorganized sections to provide a better learning experience of this package. ([45676](https://github.com/WordPress/gutenberg/pull/45676)) +- Change "block style variations" references to "block style". ([45650](https://github.com/WordPress/gutenberg/pull/45650)) + +### Performance + +- Lodash: Refactor away from `_.reduce()`. ([45460](https://github.com/WordPress/gutenberg/pull/45460)) +- Lodash: Refactor block editor away from `_.reduce()`. ([45455](https://github.com/WordPress/gutenberg/pull/45455)) +- Lodash: Refactor blocks away from `_.reduce()`. ([45457](https://github.com/WordPress/gutenberg/pull/45457)) +- Lodash: Refactor site editor away from `_.reduce()`. ([45459](https://github.com/WordPress/gutenberg/pull/45459)) +- Lodash: Refactor post editor away from `_.reduce()`. ([45458](https://github.com/WordPress/gutenberg/pull/45458)) +- Do not look for block variants, if not supporting block-templates. ([45362](https://github.com/WordPress/gutenberg/pull/45362)) +- List: Disable nested list drop zone so dropping list items works. ([45321](https://github.com/WordPress/gutenberg/pull/45321)) +- Use low-level cache for get_user_data_from_wp_global_styles. ([45634](https://github.com/WordPress/gutenberg/pull/45634)) +- Update: Improve performance of block template object retrieval. ([45646](https://github.com/WordPress/gutenberg/pull/45646)) + +### Code Quality + +#### Block Editor + +- Block Editor: Improve `LinkControl` tests. ([45609](https://github.com/WordPress/gutenberg/pull/45609)) +- Block Editor: Improve `ResponsiveBlockControl` tests. ([45610](https://github.com/WordPress/gutenberg/pull/45610)) +- Block Editor: Improve `ReusableBlocksTab` tests. ([45652](https://github.com/WordPress/gutenberg/pull/45652)) +- LinkControl: Suppress errors on null values. ([45742](https://github.com/WordPress/gutenberg/pull/45742)) +- Simplify ResizableEditor component. ([45578](https://github.com/WordPress/gutenberg/pull/45578)) +- Remove duplicate colon. ([45763](https://github.com/WordPress/gutenberg/pull/45763)) +- Extract the manage menus button to a shared component to reduce duplicate code. ([45769](https://github.com/WordPress/gutenberg/pull/45769)) +- Backport pseudo selector comments from core. ([45619](https://github.com/WordPress/gutenberg/pull/45619)) +- unstableSubscribeStore: Support store descriptors. ([45481](https://github.com/WordPress/gutenberg/pull/45481)) + +#### Components + +- BaseField: Remove unnecessary `.firstChild` from tests. ([45687](https://github.com/WordPress/gutenberg/pull/45687)) +- DateTime: Remove unused types. ([45615](https://github.com/WordPress/gutenberg/pull/45615)) +- Draggable: Convert component to TypeScript. ([45471](https://github.com/WordPress/gutenberg/pull/45471)) +- Fix `no-container` violations in `FormGroup` tests. ([45662](https://github.com/WordPress/gutenberg/pull/45662)) +- Fix `testing-library/no-node-access` in `TreeGrid` tests. ([45554](https://github.com/WordPress/gutenberg/pull/45554)) +- FontSizePicker: Use components instead of helper functions. ([44891](https://github.com/WordPress/gutenberg/pull/44891)) +- Improve tests for `ToggleGroupControl`. ([45627](https://github.com/WordPress/gutenberg/pull/45627)) +- MenuGroup: Convert component to TypeScript. ([45617](https://github.com/WordPress/gutenberg/pull/45617)) +- Popover: Fix exhaustive-deps warning. ([45656](https://github.com/WordPress/gutenberg/pull/45656)) +- Refactor `ItemGroup` to pass `exhaustive-deps`. ([45531](https://github.com/WordPress/gutenberg/pull/45531)) +- Refactor `useFlex` to pass `exhaustive-deps`. ([45528](https://github.com/WordPress/gutenberg/pull/45528)) +- Refactor `withNotices` to pass `exhaustive-deps`. ([45530](https://github.com/WordPress/gutenberg/pull/45530)) +- Refactor`PaletteEditListView` to ignore `exhaustive-deps`. ([45467](https://github.com/WordPress/gutenberg/pull/45467)) +- TabPanel: Fix the `exhaustive-deps` warning. ([45660](https://github.com/WordPress/gutenberg/pull/45660)) +- ToolsPanel: Fix exhaustive-deps hook warning. ([45715](https://github.com/WordPress/gutenberg/pull/45715)) +- Truncate: Remove unnecessary `.firstChild` from tests. ([45694](https://github.com/WordPress/gutenberg/pull/45694)) +- View component: Rename index.js to index.ts. ([45667](https://github.com/WordPress/gutenberg/pull/45667)) +- `ColorPalette`, `BorderBox`, `BorderBoxControl`: Polish and DRY prop types, add default values. ([45463](https://github.com/WordPress/gutenberg/pull/45463)) +- `NavigatorScreen`: Satisfy `exhaustive-deps` eslint rule. ([45648](https://github.com/WordPress/gutenberg/pull/45648)) +- Fix `useCx` story to satisfy `exhaustive-deps` eslint rule. ([45614](https://github.com/WordPress/gutenberg/pull/45614)) +- URLPopover: Use new placement prop instead of legacy position prop. ([44391](https://github.com/WordPress/gutenberg/pull/44391)) +- Tidy and minor refactor of Link UI code. ([37833](https://github.com/WordPress/gutenberg/pull/37833)) + +#### Block Library + +- Avatar: Escape the 'get_author_posts_url()'. ([45427](https://github.com/WordPress/gutenberg/pull/45427)) +- Button: Remove unnecessary 'useCallback'. ([45584](https://github.com/WordPress/gutenberg/pull/45584)) +- Make unwrapping columns slighly more efficient. ([45684](https://github.com/WordPress/gutenberg/pull/45684)) +- Simplfy handling of save of Nav block uncontrolled inner blocks. ([45517](https://github.com/WordPress/gutenberg/pull/45517)) +- Lodash: Refactor block library away from `_.reduce()`. ([45456](https://github.com/WordPress/gutenberg/pull/45456)) + +### Tools + +#### Testing + +- Components: Add `exhaustive-deps` eslint rule. ([41166](https://github.com/WordPress/gutenberg/pull/41166)) +- Fix typos in Paragraph block end-to-end tests. ([45611](https://github.com/WordPress/gutenberg/pull/45611)) +- FontSizePicker: Fix a buggy unit test. ([45529](https://github.com/WordPress/gutenberg/pull/45529)) +- Ignore warnings for `window.wp` in Playwright. ([45598](https://github.com/WordPress/gutenberg/pull/45598)) +- Migrate mentions tests to playwright. ([43064](https://github.com/WordPress/gutenberg/pull/43064)) +- Navigation Toggle unit test: Unmount synchronously to cancel popover positioning. ([45726](https://github.com/WordPress/gutenberg/pull/45726)) +- React Native unit tests: Migrate getByA11yLabel usages. ([45454](https://github.com/WordPress/gutenberg/pull/45454)) +- Unit Tests: Rewrite ReactDOM.render usages to RTL. ([45453](https://github.com/WordPress/gutenberg/pull/45453)) +- E2E: Add site and widget editor supports for ensureSidebarOpened. ([45480](https://github.com/WordPress/gutenberg/pull/45480)) + +#### Build Tooling + +- Include TS and JSX files to testing-library lint. ([45533](https://github.com/WordPress/gutenberg/pull/45533)) +- Remove use of `set-output` in workflows. ([45357](https://github.com/WordPress/gutenberg/pull/45357)) + +#### Triage + +- Configure labels for GHA Dependabot PRs. ([45516](https://github.com/WordPress/gutenberg/pull/45516)) + +## First time contributors + +The following PRs were merged by first time contributors: + +- @edanzer: Template Part Block: Update block isActive method. ([45672](https://github.com/WordPress/gutenberg/pull/45672)) +- @TimBroddin: Fix alignment of create new post link. ([45638](https://github.com/WordPress/gutenberg/pull/45638)) +- @wojtekn: Make Author block selector to display all users instead of just 10. ([45640](https://github.com/WordPress/gutenberg/pull/45640)) + + +## Contributors + +The following contributors merged PRs in this release: + +@aaronrobertshaw @afercia @andrewserong @bph @brookewp @c4rl0sbr4v0 @carolinan @chad1008 @ciampo @Copons @DaisyOlsen @desrosj @dinhtungdu @draganescu @dsas @edanzer @ellatrix @enejb @flootr @getdave @glendaviesnz @hz-tyfoon @jasmussen @javierarce @jffng @jonathanbossenger @jorgefilipecosta @jsnajdr @juanmaguitar @juhi123 @kevin940726 @Mamaduka @matiasbenedetto @mikachan @mirka @mmtr @mtias @ndiego @nielslange @noisysocks @ntsekouras @peterwilsoncc @ramonjd @ryelle @scruffian @spacedmonkey @t-hamano @TimBroddin @tyxla @vcanales @walbo @wojtekn @youknowriad @yuliyan + + + + += 14.5.4 = + +## Changelog + +### Fixes + +#### Global Styles +- Fix the `upgrader_process_complete` hook for `wp_theme_has_theme_json`. ([45881](https://github.com/WordPress/gutenberg/pull/45881)) + +## Contributors + +The following contributors merged PRs in this release: + +@oandregal + + + += 14.5.3 = + + + +## Changelog + +### Performance + +#### Global Styles +- Update `wp_theme_has_theme_json` to use `WP_Object_Cache`. ([45543](https://github.com/WordPress/gutenberg/pull/45543)) + + +## First time contributors + +The following PRs were merged by first time contributors: + + + +## Contributors + +The following contributors merged PRs in this release: + +@oandregal + + += 14.5.2 = + + + +## Changelog + +### Bug Fixes + +- Tag Processor: Prevent bugs from pre-PHP8 strspn/strcspn behavior. ([45822](https://github.com/WordPress/gutenberg/pull/45822)) + +## Contributors + +The following contributors merged PRs in this release: + +@dmsnell + + += 14.6.0-rc.1 = + +## Changelog + +### Enhancements + +#### Block Library +- Add color block support to latest posts. ([41874](https://github.com/WordPress/gutenberg/pull/41874)) +- Adds a list view to the navigation block. ([45546](https://github.com/WordPress/gutenberg/pull/45546)) +- Latest posts and latest comments: Add spacing support. ([45110](https://github.com/WordPress/gutenberg/pull/45110)) +- Navigation Link: Add the URL field to the Navigation Link inspector controls. ([45751](https://github.com/WordPress/gutenberg/pull/45751)) +- Reposition navigation selector. ([45555](https://github.com/WordPress/gutenberg/pull/45555)) +- Make Author block selector to display all users instead of just 10. ([45640](https://github.com/WordPress/gutenberg/pull/45640)) +- Columns: Add transform to unwrap the contents. ([45666](https://github.com/WordPress/gutenberg/pull/45666)) +- Navigation: Add a new ManageMenusButton component. ([45782](https://github.com/WordPress/gutenberg/pull/45782)) +- Read More block: Add aria-label and screen reader text. ([45490](https://github.com/WordPress/gutenberg/pull/45490)) +- Group block: Use a variation picker in the placeholder. ([43496](https://github.com/WordPress/gutenberg/pull/43496)) + +#### Components +- CheckboxControl: Replace margin overrides with new opt-in prop. ([45434](https://github.com/WordPress/gutenberg/pull/45434)) +- FontSizePicker: Update hint text to match the design. ([44966](https://github.com/WordPress/gutenberg/pull/44966)) +- Update the design of the focal point handle. ([45053](https://github.com/WordPress/gutenberg/pull/45053)) +- Use new theming accent color in all components. ([45289](https://github.com/WordPress/gutenberg/pull/45289)) +- CheckboxControl: Move icons out of labels. ([45535](https://github.com/WordPress/gutenberg/pull/45535)) + +#### Block Editor +- Converts paragraphs to headings with keyboard shortcuts. ([44681](https://github.com/WordPress/gutenberg/pull/44681)) +- Restore the empty paragraph inserter. ([45542](https://github.com/WordPress/gutenberg/pull/45542)) +- Transform: Select all blocks if the result has more than one block. ([45015](https://github.com/WordPress/gutenberg/pull/45015)) +- Content-only locked patterns: Move "Modify" to elllipsis menu. ([45391](https://github.com/WordPress/gutenberg/pull/45391)) +- Patterns: Ajust the space in the pattern explorer list. ([45730](https://github.com/WordPress/gutenberg/pull/45730)) +- Update: Lock icon to outline. ([45645](https://github.com/WordPress/gutenberg/pull/45645)) +- Don't use capital case for 'Distraction free' strings. ([45538](https://github.com/WordPress/gutenberg/pull/45538)) +- Replace Justification/Orientation controls with ToggleGroupControl. ([45637](https://github.com/WordPress/gutenberg/pull/45637)) + +#### Site Editor +- Update: Improve performance of block template object retrieval. ([45646](https://github.com/WordPress/gutenberg/pull/45646)) +- Replace FSE with Site Editor. ([45699](https://github.com/WordPress/gutenberg/pull/45699)) + +#### Design Tools +- Add a minHeight block support under dimensions. ([45300](https://github.com/WordPress/gutenberg/pull/45300)) +- Hide block toolbar when spacing visualizer is showing. ([45131](https://github.com/WordPress/gutenberg/pull/45131)) + +#### Global Styles +- Elements: Add a text decoration control to link elements. ([45643](https://github.com/WordPress/gutenberg/pull/45643)) +- Global styles: Convert preset font size values to CSS vars. ([44967](https://github.com/WordPress/gutenberg/pull/44967)) +- Fluid typography: Adjust font size min and max rules. ([45536](https://github.com/WordPress/gutenberg/pull/45536)) +- Try generating random color palettes. ([40988](https://github.com/WordPress/gutenberg/pull/40988)) + + +#### Plugin +- Updates tested up to version to 6.1. ([45630](https://github.com/WordPress/gutenberg/pull/45630)) + + +### Bug Fixes + +#### Block Library +- Change the order of the pseudo-states in the pseudo selectors array. ([45559](https://github.com/WordPress/gutenberg/pull/45559)) +- Cover: Avoid content loss when the templateLock value is all or contentOnly. ([45632](https://github.com/WordPress/gutenberg/pull/45632)) +- Fix alignment of create new post link. ([45638](https://github.com/WordPress/gutenberg/pull/45638)) +- Fix navigation appender position to prevent obstructing its items. ([43530](https://github.com/WordPress/gutenberg/pull/43530)) +- Fix: Button block text alignment. ([45663](https://github.com/WordPress/gutenberg/pull/45663)) +- Query Pagination: Fix positioning of next link in editor when parent is selected. ([45651](https://github.com/WordPress/gutenberg/pull/45651)) +- Site Logo: Use the correct home URL setting. ([45476](https://github.com/WordPress/gutenberg/pull/45476)) +- Switch background color to text color on block separator. ([44943](https://github.com/WordPress/gutenberg/pull/44943)) +- Table Block: Apply borders and padding on both front end and editor. ([45069](https://github.com/WordPress/gutenberg/pull/45069)) +- Table block: Fix error in margin value. ([45674](https://github.com/WordPress/gutenberg/pull/45674)) +- Template Part Block: Update block isActive method. ([45672](https://github.com/WordPress/gutenberg/pull/45672)) +- Navigation: Fix overflowing menu name in the navigation selector dropdown. ([45647](https://github.com/WordPress/gutenberg/pull/45647)) + +#### Accessibility +- Fix focus return when closing the Post publish panel. ([45623](https://github.com/WordPress/gutenberg/pull/45623)) +- Fix navigate regions backwards for macOS Firefox and Safari. ([45019](https://github.com/WordPress/gutenberg/pull/45019)) +- Fix the Save buttons labelling and tooltip. ([43952](https://github.com/WordPress/gutenberg/pull/43952)) +- Fix the navigate regions focus style. ([45369](https://github.com/WordPress/gutenberg/pull/45369)) +- Fix: Contrast checker appears unexpectedly on some blocks. ([45639](https://github.com/WordPress/gutenberg/pull/45639)) +- Fix: Contrast checker does not update properly. ([45686](https://github.com/WordPress/gutenberg/pull/45686)) + +#### Components +- Autocomplete: Fix unexpected block insertion during IME composition. ([45510](https://github.com/WordPress/gutenberg/pull/45510)) +- Fix ESLint warning for Dashicon. ([45795](https://github.com/WordPress/gutenberg/pull/45795)) +- FormTokenField: Fix duplicate input in IME composition. ([45607](https://github.com/WordPress/gutenberg/pull/45607)) +- Making size prop work for icon components using dash icon strings. ([45593](https://github.com/WordPress/gutenberg/pull/45593)) +- ToolsPanel: Prevent calling deselect when panel remounts. ([45673](https://github.com/WordPress/gutenberg/pull/45673)) +- Color Picker: Prevent all number fields to become 0 when one of them is an empty string. ([45649](https://github.com/WordPress/gutenberg/pull/45649)) +- ToggleGroupControl: Only show enclosing border when `isBlock`. ([45492](https://github.com/WordPress/gutenberg/pull/45492)) +- Autocomplete: Check key events more strictly in IME composition. ([45626](https://github.com/WordPress/gutenberg/pull/45626)) + +#### CSS & Styling +- Inherit font from theme on overlay close button. ([45635](https://github.com/WordPress/gutenberg/pull/45635)) +- Navigation: Fix font inheritance when using text menu button. ([45514](https://github.com/WordPress/gutenberg/pull/45514)) +- Remove hover style to button on dark block tools UI. ([45653](https://github.com/WordPress/gutenberg/pull/45653)) +- Remove width from block mover button focus style. ([45665](https://github.com/WordPress/gutenberg/pull/45665)) +- Site editor hover/select: Fix double border. ([45589](https://github.com/WordPress/gutenberg/pull/45589)) +- Remove duplicate output of existing classnames in layout classnames. ([45499](https://github.com/WordPress/gutenberg/pull/45499)) + +#### Post Editor +- BlockManagerCategory: Fix styles for indeterminate. ([45564](https://github.com/WordPress/gutenberg/pull/45564)) +- Fix: Updated names from List View to Document Overview. ([45524](https://github.com/WordPress/gutenberg/pull/45524)) +- Strip HTML from Post Title when pasting multiline title containing HTML. ([35825](https://github.com/WordPress/gutenberg/pull/35825)) + +#### Site Editor +- Decode entities in template title and description. ([45716](https://github.com/WordPress/gutenberg/pull/45716)) +- Link to homeUrl from site editor view menu. ([45475](https://github.com/WordPress/gutenberg/pull/45475)) + +#### Block Editor +- Fix Link UI popover positioning when inspector control input is focused. ([45661](https://github.com/WordPress/gutenberg/pull/45661)) +- Paste: Fix list only paste from Google documentation. ([45498](https://github.com/WordPress/gutenberg/pull/45498)) +- Make Manage Reusable blocks match similar links. ([45641](https://github.com/WordPress/gutenberg/pull/45641))([45689](https://github.com/WordPress/gutenberg/pull/45689)) +- List View: Disable branch expansion when block editing is locked. ([45541](https://github.com/WordPress/gutenberg/pull/45541)) +- Spacing visualizer: Fix display of unexpected visualizer for certain mouse actions. ([45739](https://github.com/WordPress/gutenberg/pull/45739)) + + +### Performance + +- Lodash: Refactor away from `_.reduce()`. ([45460](https://github.com/WordPress/gutenberg/pull/45460)) +- Lodash: Refactor block editor away from `_.reduce()`. ([45455](https://github.com/WordPress/gutenberg/pull/45455)) +- Lodash: Refactor blocks away from `_.reduce()`. ([45457](https://github.com/WordPress/gutenberg/pull/45457)) +- Lodash: Refactor site editor away from `_.reduce()`. ([45459](https://github.com/WordPress/gutenberg/pull/45459)) +- Lodash: Refactor post editor away from `_.reduce()`. ([45458](https://github.com/WordPress/gutenberg/pull/45458)) + +#### Block Library +- Do not look for block variants, if not supporting block-templates. ([45362](https://github.com/WordPress/gutenberg/pull/45362)) +- List: Disable nested list drop zone so dropping list items works. ([45321](https://github.com/WordPress/gutenberg/pull/45321)) + +#### Global Styles +- Use low-level cache for get_user_data_from_wp_global_styles. ([45634](https://github.com/WordPress/gutenberg/pull/45634)) + + +### Experiments + +- Al list view duplicate for use in navigation list view experiment. ([45544](https://github.com/WordPress/gutenberg/pull/45544)) +- Introduce experiment for inspector based navigation editing. ([45515](https://github.com/WordPress/gutenberg/pull/45515)) + + +### Documentation + +- Add missing CHANGELOG entry. ([45691](https://github.com/WordPress/gutenberg/pull/45691)) +- Change Title: How to use JavaScript with Gutenberg. ([45323](https://github.com/WordPress/gutenberg/pull/45323)) +- Docs: Update the readme for the integration test fixtures. ([45581](https://github.com/WordPress/gutenberg/pull/45581)) +- Summarize "Available commands" section and refer them it to `scripts` documentation. ([45636](https://github.com/WordPress/gutenberg/pull/45636)) +- Update applying-styles-with-stylesheets.md. ([45604](https://github.com/WordPress/gutenberg/pull/45604)) +- [create-block] Reorganized sections to provide a better learning experience of this package. ([45676](https://github.com/WordPress/gutenberg/pull/45676)) +- Change "block style variations" references to "block style". ([45650](https://github.com/WordPress/gutenberg/pull/45650)) + + + +### Code Quality + +#### Block Editor +- Block Editor: Improve `LinkControl` tests. ([45609](https://github.com/WordPress/gutenberg/pull/45609)) +- Block Editor: Improve `ResponsiveBlockControl` tests. ([45610](https://github.com/WordPress/gutenberg/pull/45610)) +- Block Editor: Improve `ReusableBlocksTab` tests. ([45652](https://github.com/WordPress/gutenberg/pull/45652)) +- LinkControl: Suppress errors on null values. ([45742](https://github.com/WordPress/gutenberg/pull/45742)) +- Simplify ResizableEditor component. ([45578](https://github.com/WordPress/gutenberg/pull/45578)) +- Remove duplicate colon. ([45763](https://github.com/WordPress/gutenberg/pull/45763)) +- Extract the manage menus button to a shared component to reduce duplicate code. ([45769](https://github.com/WordPress/gutenberg/pull/45769)) +- Backport pseudo selector comments from core. ([45619](https://github.com/WordPress/gutenberg/pull/45619)) +- unstableSubscribeStore: Support store descriptors. ([45481](https://github.com/WordPress/gutenberg/pull/45481)) + + +#### Components +- BaseField: Remove unnecessary `.firstChild` from tests. ([45687](https://github.com/WordPress/gutenberg/pull/45687)) +- DateTime: Remove unused types. ([45615](https://github.com/WordPress/gutenberg/pull/45615)) +- Draggable: Convert component to TypeScript. ([45471](https://github.com/WordPress/gutenberg/pull/45471)) +- Fix `no-container` violations in `FormGroup` tests. ([45662](https://github.com/WordPress/gutenberg/pull/45662)) +- Fix `testing-library/no-node-access` in `TreeGrid` tests. ([45554](https://github.com/WordPress/gutenberg/pull/45554)) +- FontSizePicker: Use components instead of helper functions. ([44891](https://github.com/WordPress/gutenberg/pull/44891)) +- Improve tests for `ToggleGroupControl`. ([45627](https://github.com/WordPress/gutenberg/pull/45627)) +- MenuGroup: Convert component to TypeScript. ([45617](https://github.com/WordPress/gutenberg/pull/45617)) +- Popover: Fix exhaustive-deps warning. ([45656](https://github.com/WordPress/gutenberg/pull/45656)) +- Refactor `ItemGroup` to pass `exhaustive-deps`. ([45531](https://github.com/WordPress/gutenberg/pull/45531)) +- Refactor `useFlex` to pass `exhaustive-deps`. ([45528](https://github.com/WordPress/gutenberg/pull/45528)) +- Refactor `withNotices` to pass `exhaustive-deps`. ([45530](https://github.com/WordPress/gutenberg/pull/45530)) +- Refactor`PaletteEditListView` to ignore `exhaustive-deps`. ([45467](https://github.com/WordPress/gutenberg/pull/45467)) +- TabPanel: Fix the `exhaustive-deps` warning. ([45660](https://github.com/WordPress/gutenberg/pull/45660)) +- ToolsPanel: Fix exhaustive-deps hook warning. ([45715](https://github.com/WordPress/gutenberg/pull/45715)) +- Truncate: Remove unnecessary `.firstChild` from tests. ([45694](https://github.com/WordPress/gutenberg/pull/45694)) +- View component: Rename index.js to index.ts. ([45667](https://github.com/WordPress/gutenberg/pull/45667)) +- `ColorPalette`, `BorderBox`, `BorderBoxControl`: Polish and DRY prop types, add default values. ([45463](https://github.com/WordPress/gutenberg/pull/45463)) +- `NavigatorScreen`: Satisfy `exhaustive-deps` eslint rule. ([45648](https://github.com/WordPress/gutenberg/pull/45648)) +- Fix `useCx` story to satisfy `exhaustive-deps` eslint rule. ([45614](https://github.com/WordPress/gutenberg/pull/45614)) +- URLPopover: Use new placement prop instead of legacy position prop. ([44391](https://github.com/WordPress/gutenberg/pull/44391)) +- Tidy and minor refactor of Link UI code. ([37833](https://github.com/WordPress/gutenberg/pull/37833)) + +#### Block Library +- Avatar: Escape the 'get_author_posts_url()'. ([45427](https://github.com/WordPress/gutenberg/pull/45427)) +- Button: Remove unnecessary 'useCallback'. ([45584](https://github.com/WordPress/gutenberg/pull/45584)) +- Make unwrapping columns slighly more efficient. ([45684](https://github.com/WordPress/gutenberg/pull/45684)) +- Simplfy handling of save of Nav block uncontrolled inner blocks. ([45517](https://github.com/WordPress/gutenberg/pull/45517)) +- Lodash: Refactor block library away from `_.reduce()`. ([45456](https://github.com/WordPress/gutenberg/pull/45456)) + + +### Tools + +#### Testing +- Components: Add `exhaustive-deps` eslint rule. ([41166](https://github.com/WordPress/gutenberg/pull/41166)) +- Fix typos in Paragraph block end-to-end tests. ([45611](https://github.com/WordPress/gutenberg/pull/45611)) +- FontSizePicker: Fix a buggy unit test. ([45529](https://github.com/WordPress/gutenberg/pull/45529)) +- Ignore warnings for `window.wp` in Playwright. ([45598](https://github.com/WordPress/gutenberg/pull/45598)) +- Migrate mentions tests to playwright. ([43064](https://github.com/WordPress/gutenberg/pull/43064)) +- Navigation Toggle unit test: Unmount synchronously to cancel popover positioning. ([45726](https://github.com/WordPress/gutenberg/pull/45726)) +- React Native unit tests: Migrate getByA11yLabel usages. ([45454](https://github.com/WordPress/gutenberg/pull/45454)) +- Unit Tests: Rewrite ReactDOM.render usages to RTL. ([45453](https://github.com/WordPress/gutenberg/pull/45453)) +- E2E: Add site and widget editor supports for ensureSidebarOpened. ([45480](https://github.com/WordPress/gutenberg/pull/45480)) + +#### Build Tooling +- Include TS and JSX files to testing-library lint. ([45533](https://github.com/WordPress/gutenberg/pull/45533)) +- Remove use of `set-output` in workflows. ([45357](https://github.com/WordPress/gutenberg/pull/45357)) + +#### Triage +- Configure labels for GHA Dependabot PRs. ([45516](https://github.com/WordPress/gutenberg/pull/45516)) + +#### Patterns +- Pattern Directory API: Add support for pagination parameters. ([45293](https://github.com/WordPress/gutenberg/pull/45293)) +- Update bundled patterns compat directory. ([45620](https://github.com/WordPress/gutenberg/pull/45620)) + + + +## First time contributors + +The following PRs were merged by first time contributors: + +- @edanzer: Template Part Block: Update block isActive method. ([45672](https://github.com/WordPress/gutenberg/pull/45672)) +- @TimBroddin: Fix alignment of create new post link. ([45638](https://github.com/WordPress/gutenberg/pull/45638)) +- @wojtekn: Make Author block selector to display all users instead of just 10. ([45640](https://github.com/WordPress/gutenberg/pull/45640)) + + +## Contributors + +The following contributors merged PRs in this release: + +@aaronrobertshaw @afercia @andrewserong @bph @brookewp @c4rl0sbr4v0 @carolinan @chad1008 @ciampo @Copons @DaisyOlsen @desrosj @dinhtungdu @draganescu @dsas @edanzer @ellatrix @enejb @flootr @getdave @glendaviesnz @hz-tyfoon @jasmussen @javierarce @jffng @jonathanbossenger @jorgefilipecosta @jsnajdr @juanmaguitar @juhi123 @kevin940726 @Mamaduka @matiasbenedetto @mikachan @mirka @mmtr @mtias @ndiego @nielslange @noisysocks @ntsekouras @peterwilsoncc @ramonjd @ryelle @scruffian @spacedmonkey @t-hamano @TimBroddin @tyxla @vcanales @walbo @wojtekn @youknowriad @yuliyan + + = 14.5.1 = diff --git a/docs/contributors/code/release.md b/docs/contributors/code/release.md index 4ce35e189fd70..f463cb319f665 100644 --- a/docs/contributors/code/release.md +++ b/docs/contributors/code/release.md @@ -150,6 +150,8 @@ The method for point releases is nearly identical to the main Plugin release pro The point release should only contain the _specific commits_ required. To do this you should checkout the previous _minor_ stable (i.e. non-RC) release branch (e.g. `release/12.5`) locally and then cherry pick any commits that you require into that branch. +_IMPORTANT:_ If an RC already exists for a new version, you _need_ to cherry-pick the same commits in the respective release branch, as they will not be included automatically. E.g.: If you're about to release a new point-release for 12.5 and just cherry-picked into `release/12.5`, but 12.6.0-rc.1 is already out, then you need to cherry-pick the same commits into the `release/12.6` branch, or they won't be included in subsequent releases for 12.6! + The cherry-picking process can be automated with the [`npm run cherry-pick` script](/docs/contributors/code/auto-cherry-picking.md). You must also ensure that all PRs being included are assigned to the Github Milestone on which the point release is based. Bear in mind, that when PRs are _merged_ they are automatically assigned a milestone for the next _stable_ release. Therefore you will need to go back through each PR in Github and re-assign the Milestone. diff --git a/docs/explanations/faq.md b/docs/explanations/faq.md index c56dd2e0c3c6c..08fcc6cbdce52 100644 --- a/docs/explanations/faq.md +++ b/docs/explanations/faq.md @@ -408,9 +408,11 @@ _See:_ [Editor Styles](/docs/how-to-guides/themes/theme-support.md#editor-styles ## What browsers does Gutenberg support? -Gutenberg works in modern browsers, and Internet Explorer 11. +Gutenberg works in modern browsers. -Our [list of supported browsers can be found in the Make WordPress handbook](https://make.wordpress.org/core/handbook/best-practices/browser-support/). By “modern browsers” we generally mean the _current and past two versions_ of each major browser. +The [list of supported browsers can be found in the Make WordPress handbook](https://make.wordpress.org/core/handbook/best-practices/browser-support/). The term “modern browsers” generally refers to the _current and previous two versions_ of each major browser. + +Since WordPress 5.8, Gutenberg no longer supports any version of Internet Explorer. ## Should I be concerned that Gutenberg will make my plugin obsolete? diff --git a/docs/getting-started/create-block/README.md b/docs/getting-started/create-block/README.md index ae56e4b123f0e..05d17f708c920 100644 --- a/docs/getting-started/create-block/README.md +++ b/docs/getting-started/create-block/README.md @@ -18,6 +18,8 @@ From your plugins directory, to create your block run: npx @wordpress/create-block gutenpride --template @wordpress/create-block-tutorial-template ``` +> Remember that you should use Node.js v14. Other versions may result in an error in the terminal. See [Node Development Tools](https://developer.wordpress.org/block-editor/getting-started/devenv/#node-development-tools) for more info. + The [npx command](https://docs.npmjs.com/cli/v8/commands/npx) runs a command from a remote package, in this case our create-block package that will create a new directory called `gutenpride`, installs the necessary files, and builds the block plugin. If you want an interactive mode that prompts you for details, run the command without the `gutenpride` name. You now need to activate the plugin from inside wp-admin plugins page. diff --git a/docs/getting-started/create-block/attributes.md b/docs/getting-started/create-block/attributes.md index 5ec6f60c3d6a4..02a55f380dcee 100644 --- a/docs/getting-started/create-block/attributes.md +++ b/docs/getting-started/create-block/attributes.md @@ -23,7 +23,7 @@ Note: The text portion is equivalent to `innerText` attribute of a DOM element. ## Edit and Save -The **attributes** are passed to the `edit` and `save` functions, along with a **setAttributes** function to set the values. Additional parameters are also passed in to these functions, see [the edit/save documentation](/docs/reference-guides/block-api/block-edit-save.md) for more details. +The **attributes** are passed to both the `edit` and `save` functions. The **setAttributes** function is also passed, but only to the `edit` function. The **setAttributes** function is used to set the values. Additional parameters are also passed in to the `edit` and `save` functions, see [the edit/save documentation](/docs/reference-guides/block-api/block-edit-save.md) for more details. The `attributes` is a JavaScript object containing the values of each attribute, or default values if defined. The `setAttributes` is a function to update an attribute. diff --git a/docs/getting-started/create-block/block-anatomy.md b/docs/getting-started/create-block/block-anatomy.md index 3f43c018cea39..83dbcf1469632 100644 --- a/docs/getting-started/create-block/block-anatomy.md +++ b/docs/getting-started/create-block/block-anatomy.md @@ -29,7 +29,7 @@ registerBlockType( metadata.name, { } ); ``` -The first parameter in the **registerBlockType** function is the block name, this should match exactly to the name registered in the PHP file. +The first parameter in the **registerBlockType** function is the block name, this should match exactly to the `name` property in the `block.json` file. By importing the metadata from `block.json` and referencing the `name` property in the first parameter we ensure that they will match, and continue to match even if the name is subsequently changed in `block.json`. The second parameter to the function is the block object. See the [block registration documentation](/docs/reference-guides/block-api/block-registration.md) for full details. @@ -37,7 +37,7 @@ The last two block object properties are **edit** and **save**, these are the ke The results of the edit function is what the editor will render to the editor page when the block is inserted. -The results of the save function is what the editor will insert into the **post_content** field when the post is saved. The post_content field is the field in the WordPress database used to store the content of the post. +The results of the save function is what the editor will insert into the **post_content** field when the post is saved. The post_content field is the field in the **wp_posts** table in the WordPress database that is used to store the content of the post. Most of the properties are set in the `src/block.json` file. diff --git a/docs/how-to-guides/block-tutorial/README.md b/docs/how-to-guides/block-tutorial/README.md index b1ff5a19119c2..95aa4182430c0 100644 --- a/docs/how-to-guides/block-tutorial/README.md +++ b/docs/how-to-guides/block-tutorial/README.md @@ -2,7 +2,9 @@ The purpose of this tutorial is to step through the fundamentals of creating a new block type. Beginning with the simplest possible example, each new section will incrementally build upon the last to include more of the common functionality you could expect to need when implementing your own block types. -To follow along with this tutorial, you can [download the accompanying WordPress plugin](https://github.com/WordPress/gutenberg-examples) which includes all of the examples for you to try on your own site. At each step along the way, experiment by modifying the examples with your own ideas, and observe the effects they have on the block's behavior. +To follow along with this tutorial, you can download the [accompanying WordPress plugin](https://github.com/WordPress/gutenberg-examples) which includes all of the examples for you to try on your own site. At each step along the way, experiment by modifying the examples with your own ideas, and observe the effects they have on the block's behavior. + +> To find the latest version of the .zip file go to the repo's [releases page](https://github.com/WordPress/gutenberg-examples/releases) and look in the latest release under 'Assets'. Code snippets are provided in two formats "JSX" and "Plain". JSX refers to JavaScript code that uses JSX syntax which requires a build step. Plain refers to "classic" JavaScript that does not require building. You can change between them using tabs found above each code example. Using JSX, does require you to run [the JavaScript build step](/docs/how-to-guides/javascript/js-build-setup/) to compile your code to a browser compatible format. diff --git a/docs/how-to-guides/block-tutorial/applying-styles-with-stylesheets.md b/docs/how-to-guides/block-tutorial/applying-styles-with-stylesheets.md index 815adfdf80c26..e668a6ac76238 100644 --- a/docs/how-to-guides/block-tutorial/applying-styles-with-stylesheets.md +++ b/docs/how-to-guides/block-tutorial/applying-styles-with-stylesheets.md @@ -2,7 +2,7 @@ ## Overview -A block typically inserts markup (HTML) into post content that you want to style in someway. This guides walks through a few different ways you can use CSS with the block editor and how to work with styles and stylesheets. +A block typically inserts markup (HTML) into post content that you want to style in some way. This guide walks through a few different ways you can use CSS with the block editor and how to work with styles and stylesheets. ## Before you start diff --git a/docs/how-to-guides/block-tutorial/extending-the-query-loop-block.md b/docs/how-to-guides/block-tutorial/extending-the-query-loop-block.md index 3956971645f86..e43dcb9727088 100644 --- a/docs/how-to-guides/block-tutorial/extending-the-query-loop-block.md +++ b/docs/how-to-guides/block-tutorial/extending-the-query-loop-block.md @@ -192,6 +192,8 @@ Notice that we have also disabled the `postType` control. When the user selects Because our plugin uses custom attributes that we need to query, we want to add our own controls to allow the users to select those instead of the ones we have just disabled from the core inspector controls. We can do this via a [React HOC](https://reactjs.org/docs/higher-order-components.html) hooked into a [block filter](https://developer.wordpress.org/block-editor/reference-guides/filters/block-filters/), like so: ```jsx +import { InspectorControls } from '@wordpress/block-editor'; + export const withBookQueryControls = ( BlockEdit ) => ( props ) => { // We only want to add these controls if it is our variation, // so here we can implement a custom logic to check for that, similiar diff --git a/docs/reference-guides/block-api/block-metadata.md b/docs/reference-guides/block-api/block-metadata.md index 6185043353fc2..4acb0f914e037 100644 --- a/docs/reference-guides/block-api/block-metadata.md +++ b/docs/reference-guides/block-api/block-metadata.md @@ -252,7 +252,7 @@ The `ancestor` property makes a block available inside the specified block types An icon property should be specified to make it easier to identify a block. These can be any of WordPress' Dashicons (slug serving also as a fallback in non-js contexts). -**Note:** It's also possible to override this property on the client-side with the source of the SVG element. In addition, this property can be defined with JavaScript as an object containing background and foreground colors. This colors will appear with the icon when they are applicable e.g.: in the inserter. Custom SVG icons are automatically wrapped in the [wp.primitives.SVG](/packages/packages-primitives) component to add accessibility attributes (aria-hidden, role, and focusable). +**Note:** It's also possible to override this property on the client-side with the source of the SVG element. In addition, this property can be defined with JavaScript as an object containing background and foreground colors. This colors will appear with the icon when they are applicable e.g.: in the inserter. Custom SVG icons are automatically wrapped in the [wp.primitives.SVG](/packages/primitives/README.md) component to add accessibility attributes (aria-hidden, role, and focusable). ### Description diff --git a/docs/reference-guides/block-api/block-registration.md b/docs/reference-guides/block-api/block-registration.md index 5c38bf7f79c36..97077cb7efdfa 100644 --- a/docs/reference-guides/block-api/block-registration.md +++ b/docs/reference-guides/block-api/block-registration.md @@ -35,7 +35,7 @@ A block requires a few properties to be specified before it can be registered su - **Type:** `String` -This is the display title for your block, which can be translated with our translation functions. The title will display in the Inserter and in other areas of the editor. +This is the display title for your block, which can be translated with our translation functions. The title will display in the Inserter and in other areas of the editor. ```js // Our data object @@ -90,7 +90,7 @@ icon: 'book-alt', icon: , ``` -**Note:** Custom SVG icons are automatically wrapped in the [`wp.primitives.SVG` component](/packages/primitives/src/svg/) to add accessibility attributes (`aria-hidden`, `role`, and `focusable`). +**Note:** Custom SVG icons are automatically wrapped in the [`wp.primitives.SVG` component](/packages/primitives/README.md) to add accessibility attributes (`aria-hidden`, `role`, and `focusable`). An object can also be passed as icon, in this case, icon, as specified above, should be included in the src property. @@ -179,7 +179,7 @@ attributes: { - **Type:** `Object` -Example provides structured example data for the block. This data is used to construct a preview for the block to be shown in the Inspector Help Panel when the user mouses over the block. +Example provides structured example data for the block. This data is used to construct a preview for the block to be shown in the Inspector Help Panel when the user mouses over the block and in the Styles panel when the block is selected. The data provided in the example object should match the attributes defined. For example: diff --git a/docs/reference-guides/block-api/block-supports.md b/docs/reference-guides/block-api/block-supports.md index f3d15f118ae8f..68afd793231ef 100644 --- a/docs/reference-guides/block-api/block-supports.md +++ b/docs/reference-guides/block-api/block-supports.md @@ -351,36 +351,25 @@ supports: { Link color presets are sourced from the `editor-color-palette` [theme support](/docs/how-to-guides/themes/theme-support.md#block-color-palettes). -When the block declares support for `color.link`, the attributes definition is extended to include two new attributes: `linkColor` and `style`: - -- `linkColor`: attribute of `string` type with no default assigned. - - When a user chooses from the list of preset link colors, the preset slug is stored in the `linkColor` attribute. - - The block can apply a default preset text color by specifying its own attribute with a default e.g.: - - ```js - attributes: { - linkColor: { - type: 'string', - default: 'some-preset-link-color-slug', - } - } - ``` +When the block declares support for `color.link`, the attributes definition is extended to include the `style` attribute: - `style`: attribute of `object` type with no default assigned. - When a custom link color is selected (i.e. using the custom color picker), the custom color value is stored in the `style.color.link` attribute. + When a link color is selected, the color value is stored in the `style.elements.link.color.text` attribute. - The block can apply a default custom link color by specifying its own attribute with a default e.g.: + The block can apply a default link color by specifying its own attribute with a default e.g.: ```js attributes: { style: { type: 'object', default: { - color: { - link: '#ff0000', + elements: { + link: { + color: { + text: '#ff0000', + } + } } } } @@ -583,7 +572,7 @@ attributes: { } ``` -A spacing property may define an array of allowable sides – 'top', 'right', 'bottom', 'left' – that can be configured. When such arbitrary sides are defined, only UI controls for those sides are displayed. +A spacing property may define an array of allowable sides – 'top', 'right', 'bottom', 'left' – that can be configured. When such arbitrary sides are defined, only UI controls for those sides are displayed. Axial sides are defined with the `vertical` and `horizontal` terms, and display a single UI control for each axial pair (for example, `vertical` controls both the top and bottom sides). A spacing property may support arbitrary individual sides **or** axial sides, but not a mix of both. diff --git a/docs/reference-guides/core-blocks.md b/docs/reference-guides/core-blocks.md index 65e3cb1f65c8f..d2d66fc0d12de 100644 --- a/docs/reference-guides/core-blocks.md +++ b/docs/reference-guides/core-blocks.md @@ -284,7 +284,7 @@ Introduce new sections and organize content to help visitors (and search engines - **Name:** core/heading - **Category:** text -- **Supports:** __unstablePasteTextInline, align (full, wide), anchor, color (background, gradients, link, text), spacing (margin, padding), typography (fontSize, lineHeight), ~~className~~ +- **Supports:** __unstablePasteTextInline, align (full, wide), anchor, className, color (background, gradients, link, text), spacing (margin, padding), typography (fontSize, lineHeight) - **Attributes:** content, level, placeholder, textAlign ## Home Link @@ -429,7 +429,16 @@ Display a list of all pages. ([Source](https://github.com/WordPress/gutenberg/tr - **Name:** core/page-list - **Category:** widgets - **Supports:** ~~html~~, ~~reusable~~ -- **Attributes:** +- **Attributes:** parentPageID + +## Page List Item + +Displays a page inside a list of all pages. ([Source](https://github.com/WordPress/gutenberg/tree/trunk/packages/block-library/src/page-list-item)) + +- **Name:** core/page-list-item +- **Category:** widgets +- **Supports:** ~~html~~, ~~inserter~~, ~~lock~~, ~~reusable~~ +- **Attributes:** hasChildren, id, label, link, title ## Paragraph @@ -563,7 +572,7 @@ Contains the block elements used to render a post, like the title, date, feature - **Name:** core/post-template - **Category:** theme -- **Supports:** align, typography (fontSize, lineHeight), ~~html~~, ~~reusable~~ +- **Supports:** align, color (background, gradients, link, text), typography (fontSize, lineHeight), ~~html~~, ~~reusable~~ - **Attributes:** ## Post Terms @@ -608,7 +617,7 @@ An advanced block that allows displaying post types based on different query par - **Name:** core/query - **Category:** theme -- **Supports:** align (full, wide), color (background, gradients, link, text), ~~html~~ +- **Supports:** align (full, wide), ~~html~~ - **Attributes:** displayLayout, namespace, query, queryId, tagName ## No results diff --git a/docs/reference-guides/data/data-core-block-editor.md b/docs/reference-guides/data/data-core-block-editor.md index 8d7bceb5b4431..0580aa0141b2b 100644 --- a/docs/reference-guides/data/data-core-block-editor.md +++ b/docs/reference-guides/data/data-core-block-editor.md @@ -556,6 +556,18 @@ _Properties_ - _isDisabled_ `boolean`: Whether or not the user should be prevented from inserting this item. - _frecency_ `number`: Heuristic that combines frequency and recency. +### getLastInsertedBlockClientId + +Gets the client id of the last inserted block. + +_Parameters_ + +- _state_ `Object`: Global application state. + +_Returns_ + +- `string|undefined`: Client Id of the last inserted block. + ### getLastMultiSelectedBlockClientId Returns the client ID of the last block in the multi-selection set, or null @@ -801,7 +813,7 @@ _Parameters_ _Returns_ -- `?string`: Block Template Lock +- `string|false`: Block Template Lock ### hasBlockMovingClientId diff --git a/docs/reference-guides/data/data-core-edit-site.md b/docs/reference-guides/data/data-core-edit-site.md index 9ee5fea71801b..53fd4d138383a 100644 --- a/docs/reference-guides/data/data-core-edit-site.md +++ b/docs/reference-guides/data/data-core-edit-site.md @@ -20,8 +20,11 @@ _Returns_ ### getCurrentTemplateNavigationPanelSubMenu -Returns the current template or template part's corresponding -navigation panel's sub menu, to be used with `openNavigationPanelToMenu`. +> **Deprecated** + +### getCurrentTemplateTemplateParts + +Returns the template parts and their blocks for the current edited template. _Parameters_ @@ -29,11 +32,13 @@ _Parameters_ _Returns_ -- `string`: The current template or template part's sub menu. +- `Array`: Template parts and their blocks in an array. -### getCurrentTemplateTemplateParts +### getEditedPostContext -Returns the template parts and their blocks for the current edited template. +> **Deprecated** + +Returns the edited post's context object. _Parameters_ @@ -41,7 +46,7 @@ _Parameters_ _Returns_ -- `Array`: Template parts and their blocks in an array. +- `Object`: Page. ### getEditedPostId @@ -81,30 +86,16 @@ _Returns_ ### getHomeTemplateId -Returns the current home template ID. - -_Parameters_ - -- _state_ `Object`: Global application state. - -_Returns_ - -- `number?`: Home template ID. +> **Deprecated** ### getNavigationPanelActiveMenu -Returns the active menu in the navigation panel. - -_Parameters_ - -- _state_ `Object`: Global application state. - -_Returns_ - -- `string`: Active menu. +> **Deprecated** ### getPage +> **Deprecated** + Returns the current page object. _Parameters_ @@ -179,15 +170,7 @@ _Returns_ ### isNavigationOpened -Returns the current opened/closed state of the navigation panel. - -_Parameters_ - -- _state_ `Object`: Global application state. - -_Returns_ - -- `boolean`: True if the navigation panel should be open; false if closed. +> **Deprecated** ### isSaveViewOpened @@ -233,11 +216,9 @@ _Parameters_ ### openNavigationPanelToMenu -Opens the navigation panel and sets its active menu at the same time. +> **Deprecated** -_Parameters_ - -- _menu_ `string`: Identifies the menu to open. +Opens the navigation panel and sets its active menu at the same time. ### removeTemplate @@ -257,14 +238,21 @@ _Parameters_ - _options_ `[Object]`: - _options.allowUndo_ `[boolean]`: Whether to allow the user to undo reverting the template. Default true. -### setHomeTemplateId +### setEditedPostContext -Action that sets the home template ID to the template ID of the page resolved -from a given path. +Set's the current block editor context. _Parameters_ -- _homeTemplateId_ `number`: The template ID for the homepage. +- _context_ `Object`: The context object. + +_Returns_ + +- `number`: The resolved template ID for the page route. + +### setHomeTemplateId + +> **Deprecated** ### setIsInserterOpened @@ -290,11 +278,9 @@ _Parameters_ ### setIsNavigationPanelOpened -Sets whether the navigation panel should be open. +> **Deprecated** -_Parameters_ - -- _isOpen_ `boolean`: If true, opens the nav panel. If false, closes it. It does not toggle the state, but sets it directly. +Sets whether the navigation panel should be open. ### setIsSaveViewOpened @@ -306,11 +292,9 @@ _Parameters_ ### setNavigationPanelActiveMenu -Action that sets the active navigation panel menu. - -_Parameters_ +> **Deprecated** -- _menu_ `string`: Menu prop of active menu. +Action that sets the active navigation panel menu. _Returns_ diff --git a/docs/reference-guides/theme-json-reference/theme-json-living.md b/docs/reference-guides/theme-json-reference/theme-json-living.md index cf7624766c9ff..a694cf63e909f 100644 --- a/docs/reference-guides/theme-json-reference/theme-json-living.md +++ b/docs/reference-guides/theme-json-reference/theme-json-living.md @@ -225,6 +225,13 @@ Outline styles. | style | string | | | width | string | | +--- + +### css + +Sets custom CSS to apply styling not covered by other theme.json properties. + + --- diff --git a/gutenberg.php b/gutenberg.php index 9c93c6c823f62..4d4489ef4c061 100644 --- a/gutenberg.php +++ b/gutenberg.php @@ -3,9 +3,9 @@ * Plugin Name: Gutenberg * Plugin URI: https://github.com/WordPress/gutenberg * Description: Printing since 1440. This is the development plugin for the new block editor in core. - * Requires at least: 5.9 + * Requires at least: 6.0 * Requires PHP: 5.6 - * Version: 14.6.0-rc.1 + * Version: 14.8.0-rc.1 * Author: Gutenberg Team * Text Domain: gutenberg * diff --git a/lib/block-supports/duotone.php b/lib/block-supports/duotone.php index 4a9a23d7b5af3..b4a8397d72ece 100644 --- a/lib/block-supports/duotone.php +++ b/lib/block-supports/duotone.php @@ -387,7 +387,7 @@ function gutenberg_get_duotone_filter_svg( $preset ) { $svg = ob_get_clean(); - if ( ! defined( 'SCRIPT_DEBUG' ) || ! SCRIPT_DEBUG ) { + if ( ! SCRIPT_DEBUG ) { // Clean up the whitespace. $svg = preg_replace( "/[\r\n\t ]+/", ' ', $svg ); $svg = str_replace( '> <', '><', $svg ); @@ -464,7 +464,7 @@ function gutenberg_render_duotone_support( $block_content, $block ) { // !important is needed because these styles render before global styles, // and they should be overriding the duotone filters set by global styles. - $filter_style = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG + $filter_style = SCRIPT_DEBUG ? $selector . " {\n\tfilter: " . $filter_property . " !important;\n}\n" : $selector . '{filter:' . $filter_property . ' !important;}'; diff --git a/lib/block-supports/layout.php b/lib/block-supports/layout.php index a50e1fb883717..39869baa4e516 100644 --- a/lib/block-supports/layout.php +++ b/lib/block-supports/layout.php @@ -83,7 +83,7 @@ function gutenberg_get_layout_style( $selector, $layout, $has_block_gap_support $wide_max_width_value = $wide_size ? $wide_size : $content_size; // Make sure there is a single CSS rule, and all tags are stripped for security. - // TODO: Use `safecss_filter_attr` instead - once https://core.trac.wordpress.org/ticket/46197 is patched. + // TODO: Use `safecss_filter_attr` instead when the minimum required WP version is >= 6.1. $all_max_width_value = wp_strip_all_tags( explode( ';', $all_max_width_value )[0] ); $wide_max_width_value = wp_strip_all_tags( explode( ';', $wide_max_width_value )[0] ); @@ -316,23 +316,71 @@ function gutenberg_get_classnames_from_last_tag( $html ) { * @return string Filtered block content. */ function gutenberg_render_layout_support_flag( $block_content, $block ) { - $block_type = WP_Block_Type_Registry::get_instance()->get_registered( $block['blockName'] ); - $support_layout = block_has_support( $block_type, array( '__experimentalLayout' ), false ); + $block_type = WP_Block_Type_Registry::get_instance()->get_registered( $block['blockName'] ); + $support_layout = block_has_support( $block_type, array( '__experimentalLayout' ), false ); + $has_child_layout = isset( $block['attrs']['style']['layout']['selfStretch'] ); - if ( ! $support_layout ) { + if ( ! $support_layout + && ! $has_child_layout ) { return $block_content; } - $block_gap = gutenberg_get_global_settings( array( 'spacing', 'blockGap' ) ); - $global_layout_settings = gutenberg_get_global_settings( array( 'layout' ) ); - $has_block_gap_support = isset( $block_gap ) ? null !== $block_gap : false; - $default_block_layout = _wp_array_get( $block_type->supports, array( '__experimentalLayout', 'default' ), array() ); - $used_layout = isset( $block['attrs']['layout'] ) ? $block['attrs']['layout'] : $default_block_layout; + $outer_class_names = array(); + + if ( $has_child_layout && ( 'fixed' === $block['attrs']['style']['layout']['selfStretch'] || 'fill' === $block['attrs']['style']['layout']['selfStretch'] ) ) { + + $container_content_class = wp_unique_id( 'wp-container-content-' ); - if ( isset( $used_layout['inherit'] ) && $used_layout['inherit'] ) { - if ( ! $global_layout_settings ) { - return $block_content; + $child_layout_styles = array(); + + if ( 'fixed' === $block['attrs']['style']['layout']['selfStretch'] && isset( $block['attrs']['style']['layout']['flexSize'] ) ) { + $child_layout_styles[] = array( + 'selector' => ".$container_content_class", + 'declarations' => array( + 'flex-basis' => $block['attrs']['style']['layout']['flexSize'], + 'box-sizing' => 'border-box', + ), + ); + } elseif ( 'fill' === $block['attrs']['style']['layout']['selfStretch'] ) { + $child_layout_styles[] = array( + 'selector' => ".$container_content_class", + 'declarations' => array( + 'flex-grow' => '1', + ), + ); } + + gutenberg_style_engine_get_stylesheet_from_css_rules( + $child_layout_styles, + array( + 'context' => 'block-supports', + 'prettify' => false, + ) + ); + + $outer_class_names[] = $container_content_class; + + } + + // Return early if only child layout exists. + if ( ! $support_layout && ! empty( $outer_class_names ) ) { + $content = new WP_HTML_Tag_Processor( $block_content ); + $content->next_tag(); + $content->add_class( implode( ' ', $outer_class_names ) ); + return (string) $content; + } + + $global_settings = gutenberg_get_global_settings(); + $block_gap = _wp_array_get( $global_settings, array( 'spacing', 'blockGap' ), null ); + $has_block_gap_support = isset( $block_gap ); + $global_layout_settings = _wp_array_get( $global_settings, array( 'layout' ), null ); + $root_padding_aware_alignments = _wp_array_get( $global_settings, array( 'useRootPaddingAwareAlignments' ), false ); + + $default_block_layout = _wp_array_get( $block_type->supports, array( '__experimentalLayout', 'default' ), array() ); + $used_layout = isset( $block['attrs']['layout'] ) ? $block['attrs']['layout'] : $default_block_layout; + + if ( isset( $used_layout['inherit'] ) && $used_layout['inherit'] && ! $global_layout_settings ) { + return $block_content; } $class_names = array(); @@ -346,7 +394,7 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { } if ( - gutenberg_get_global_settings( array( 'useRootPaddingAwareAlignments' ) ) && + $root_padding_aware_alignments && isset( $used_layout['type'] ) && 'constrained' === $used_layout['type'] ) { @@ -428,13 +476,26 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { } } + $content_with_outer_classnames = ''; + + if ( ! empty( $outer_class_names ) ) { + $content_with_outer_classnames = new WP_HTML_Tag_Processor( $block_content ); + $content_with_outer_classnames->next_tag(); + foreach ( $outer_class_names as $outer_class_name ) { + $content_with_outer_classnames->add_class( $outer_class_name ); + } + + $content_with_outer_classnames = (string) $content_with_outer_classnames; + } + /** * The first chunk of innerContent contains the block markup up until the inner blocks start. * We want to target the opening tag of the inner blocks wrapper, which is the last tag in that chunk. */ $inner_content_classnames = isset( $block['innerContent'][0] ) && 'string' === gettype( $block['innerContent'][0] ) ? gutenberg_get_classnames_from_last_tag( $block['innerContent'][0] ) : ''; - $content = new WP_HTML_Tag_Processor( $block_content ); + $content = $content_with_outer_classnames ? new WP_HTML_Tag_Processor( $content_with_outer_classnames ) : new WP_HTML_Tag_Processor( $block_content ); + if ( $inner_content_classnames ) { $content->next_tag( array( 'class_name' => $inner_content_classnames ) ); foreach ( $class_names as $class_name ) { @@ -442,7 +503,9 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { } } else { $content->next_tag(); - $content->add_class( implode( ' ', $class_names ) ); + foreach ( $class_names as $class_name ) { + $content->add_class( $class_name ); + } } return (string) $content; diff --git a/lib/blocks.php b/lib/blocks.php index d991eac8ddbe4..add72e77062cb 100644 --- a/lib/blocks.php +++ b/lib/blocks.php @@ -23,7 +23,6 @@ function gutenberg_reregister_core_block_types() { 'columns', 'comments', 'group', - 'heading', 'html', 'list', 'list-item', @@ -69,6 +68,7 @@ function gutenberg_reregister_core_block_types() { 'home-link.php' => 'core/home-link', 'image.php' => 'core/image', 'gallery.php' => 'core/gallery', + 'heading.php' => 'core/heading', 'latest-comments.php' => 'core/latest-comments', 'latest-posts.php' => 'core/latest-posts', 'loginout.php' => 'core/loginout', @@ -190,7 +190,7 @@ function gutenberg_register_core_block_assets( $block_name ) { // When in production, use the plugin's version as the default asset version; // else (for development or test) default to use the current time. - $default_version = defined( 'GUTENBERG_VERSION' ) && ! ( defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ) ? GUTENBERG_VERSION : time(); + $default_version = defined( 'GUTENBERG_VERSION' ) && ! SCRIPT_DEBUG ? GUTENBERG_VERSION : time(); $style_path = "build/block-library/blocks/$block_name/"; $stylesheet_url = gutenberg_url( $style_path . 'style.css' ); diff --git a/lib/client-assets.php b/lib/client-assets.php index ea199750fa10b..b7a11724a5e09 100644 --- a/lib/client-assets.php +++ b/lib/client-assets.php @@ -193,7 +193,7 @@ function gutenberg_override_style( $styles, $handle, $src, $deps = array(), $ver function gutenberg_register_packages_scripts( $scripts ) { // When in production, use the plugin's version as the default asset version; // else (for development or test) default to use the current time. - $default_version = defined( 'GUTENBERG_VERSION' ) && ! ( defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ) ? GUTENBERG_VERSION : time(); + $default_version = defined( 'GUTENBERG_VERSION' ) && ! SCRIPT_DEBUG ? GUTENBERG_VERSION : time(); foreach ( glob( gutenberg_dir_path() . 'build/*/index.min.js' ) as $path ) { // Prefix `wp-` to package directory to get script handle. @@ -249,7 +249,16 @@ function gutenberg_register_packages_scripts( $scripts ) { function gutenberg_register_packages_styles( $styles ) { // When in production, use the plugin's version as the asset version; // else (for development or test) default to use the current time. - $version = defined( 'GUTENBERG_VERSION' ) && ! ( defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ) ? GUTENBERG_VERSION : time(); + $version = defined( 'GUTENBERG_VERSION' ) && ! SCRIPT_DEBUG ? GUTENBERG_VERSION : time(); + + gutenberg_override_style( + $styles, + 'wp-block-editor-content', + gutenberg_url( 'build/block-editor/content.css' ), + array(), + $version + ); + $styles->add_data( 'wp-block-editor-content', 'rtl', 'replace' ); // Editor Styles. gutenberg_override_style( @@ -315,6 +324,9 @@ function gutenberg_register_packages_styles( $styles ) { 'wp-reset-editor-styles', 'wp-block-library', 'wp-reusable-blocks', + // Until #37466, we can't specifically add them as editor styles yet, + // so we must hard-code it here as a dependency. + 'wp-block-editor-content', ); // Only load the default layout and margin styles for themes without theme.json file. @@ -486,7 +498,7 @@ function gutenberg_enqueue_stored_styles( $options = array() ) { $style_tag_id = 'core'; foreach ( $core_styles_keys as $style_key ) { // Adds comment if code is prettified to identify core styles sections in debugging. - $should_prettify = isset( $options['prettify'] ) ? true === $options['prettify'] : defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG; + $should_prettify = isset( $options['prettify'] ) ? true === $options['prettify'] : SCRIPT_DEBUG; if ( $should_prettify ) { $compiled_core_stylesheet .= "/**\n * Core styles: $style_key\n */\n"; } @@ -527,6 +539,37 @@ function gutenberg_enqueue_stored_styles( $options = array() ) { } } +/** + * Registers vendor JavaScript files to be used as dependencies of the editor + * and plugins. + * + * This function is called from a script during the plugin build process, so it + * should not call any WordPress PHP functions. + * + * @since 13.0 + * + * @param WP_Scripts $scripts WP_Scripts instance. + */ +function gutenberg_register_vendor_scripts( $scripts ) { + $extension = SCRIPT_DEBUG ? '.js' : '.min.js'; + + gutenberg_override_script( + $scripts, + 'react', + gutenberg_url( 'build/vendors/react' . $extension ), + // See https://github.com/pmmmwh/react-refresh-webpack-plugin/blob/main/docs/TROUBLESHOOTING.md#externalising-react. + SCRIPT_DEBUG ? array( 'wp-react-refresh-entry', 'wp-polyfill' ) : array( 'wp-polyfill' ) + ); + gutenberg_override_script( + $scripts, + 'react-dom', + gutenberg_url( 'build/vendors/react-dom' . $extension ), + array( 'react' ) + ); +} +add_action( 'wp_default_scripts', 'gutenberg_register_vendor_scripts' ); + + /* * Always remove the Core action hook while gutenberg_enqueue_stored_styles() exists to avoid styles being printed twice. * This is also because gutenberg_enqueue_stored_styles uses the Style Engine's `gutenberg_*` functions and `_Gutenberg` classes, diff --git a/lib/compat/wordpress-6.0/block-editor-settings.php b/lib/compat/wordpress-6.0/block-editor-settings.php deleted file mode 100644 index 8aae673bff334..0000000000000 --- a/lib/compat/wordpress-6.0/block-editor-settings.php +++ /dev/null @@ -1,35 +0,0 @@ - .alignleft { float: left; margin-right: 2em; }'; - $root_styles .= '.wp-site-blocks > .alignright { float: right; margin-left: 2em; }'; - $root_styles .= '.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }'; - - if ( - ( isset( $style['__unstableType'] ) && ( 'presets' === $style['__unstableType'] ) ) || - ( isset( $style['__unstableType'] ) && ( 'theme' === $style['__unstableType'] ) && str_contains( $style['css'], $root_styles ) ) - ) { - return true; - } - - return false; -} diff --git a/lib/compat/wordpress-6.0/block-gallery.php b/lib/compat/wordpress-6.0/block-gallery.php deleted file mode 100644 index 53c3a0c2247fd..0000000000000 --- a/lib/compat/wordpress-6.0/block-gallery.php +++ /dev/null @@ -1,57 +0,0 @@ -= 6.0. - * - * @return void. - */ -function gutenberg_check_gallery_block_v2_compatibility() { - $use_balance_tags = (int) get_option( 'use_balanceTags' ); - $v2_gallery_enabled = boolval( 1 !== $use_balance_tags || is_wp_version_compatible( '6.0' ) ) ? 'true' : 'false'; - - wp_add_inline_script( - 'wp-dom-ready', - 'wp.galleryBlockV2Enabled = ' . $v2_gallery_enabled . ';', - 'after' - ); -} -add_action( 'init', 'gutenberg_check_gallery_block_v2_compatibility' ); - -/** - * Prevent use_balanceTags being enabled on WordPress 5.8 or earlier as it breaks - * the layout of the new Gallery block. - * - * @since 12.1.0 - * @todo This should be removed when the minimum required WP version is >= 6.0. - * - * @param int $new_value The new value for use_balanceTags. - */ -function gutenberg_use_balancetags_check( $new_value ) { - global $wp_version; - - if ( 1 === (int) $new_value && version_compare( $wp_version, '6.0', '<' ) ) { - /* translators: %s: Minimum required version */ - $message = sprintf( __( 'Gutenberg requires WordPress %s or later in order to enable the “Correct invalidly nested XHTML automatically” option. Please upgrade WordPress before enabling.', 'gutenberg' ), '6.0' ); - add_settings_error( 'gutenberg_use_balancetags_check', 'gutenberg_use_balancetags_check', $message, 'error' ); - if ( class_exists( 'WP_CLI' ) ) { - WP_CLI::error( $message ); - } - return 0; - } - - return $new_value; -} -add_filter( 'pre_update_option_use_balanceTags', 'gutenberg_use_balancetags_check' ); diff --git a/lib/compat/wordpress-6.0/block-patterns-update.php b/lib/compat/wordpress-6.0/block-patterns-update.php deleted file mode 100644 index c17a65be2ae22..0000000000000 --- a/lib/compat/wordpress-6.0/block-patterns-update.php +++ /dev/null @@ -1,217 +0,0 @@ -is_registered( 'query' ) ) { - register_block_pattern_category( 'query', array( 'label' => __( 'Query', 'gutenberg' ) ) ); - } - - if ( ! $pattern_category_registry->is_registered( 'featured' ) ) { - register_block_pattern_category( 'featured', array( 'label' => __( 'Featured', 'gutenberg' ) ) ); - } - - $patterns = array( - 'query-standard-posts' => array( - 'title' => _x( 'Standard', 'Block pattern title', 'gutenberg' ), - 'blockTypes' => array( 'core/query' ), - 'categories' => array( 'query' ), - 'content' => ' -
- - - - - -
- - - -
- ', - ), - 'query-medium-posts' => array( - 'title' => _x( 'Image at left', 'Block pattern title', 'gutenberg' ), - 'blockTypes' => array( 'core/query' ), - 'categories' => array( 'query' ), - 'content' => ' -
- - -
-
- - -
-
-
- - -
- ', - ), - 'query-small-posts' => array( - 'title' => _x( 'Small image and title', 'Block pattern title', 'gutenberg' ), - 'blockTypes' => array( 'core/query' ), - 'categories' => array( 'query' ), - 'content' => ' -
- - -
-
- - -
-
- - -
- ', - ), - 'query-grid-posts' => array( - 'title' => _x( 'Grid', 'Block pattern title', 'gutenberg' ), - 'blockTypes' => array( 'core/query' ), - 'categories' => array( 'query' ), - 'content' => ' -
- - -
- -
- - -
- ', - ), - 'query-large-title-posts' => array( - 'title' => _x( 'Large title', 'Block pattern title', 'gutenberg' ), - 'blockTypes' => array( 'core/query' ), - 'categories' => array( 'query' ), - 'content' => ' -
-
- -
- - - -
-
- - - -
-
- -
-
- ', - ), - 'query-offset-posts' => array( - 'title' => _x( 'Offset', 'Block pattern title', 'gutenberg' ), - 'blockTypes' => array( 'core/query' ), - 'categories' => array( 'query' ), - 'content' => ' -
-
-
-
- - - - - - -
-
- - -
-
- - - - - - -
-
-
-
- ', - ), - // Initial block pattern to be used with block transformations with patterns. - 'social-links-shared-background-color' => array( - 'title' => _x( 'Social links with a shared background color', 'Block pattern title', 'gutenberg' ), - 'categories' => array( 'buttons' ), - 'blockTypes' => array( 'core/social-links' ), - 'viewportWidth' => 500, - 'content' => ' - - ', - ), - ); - - foreach ( $patterns as $name => $pattern ) { - $pattern_name = 'core/' . $name; - if ( ! WP_Block_Patterns_Registry::get_instance()->is_registered( $pattern_name ) ) { - register_block_pattern( $pattern_name, $pattern ); - } - } -} - -/** - * Deactivate the legacy patterns bundled with WordPress. - */ -function gutenberg_remove_core_patterns() { - $core_block_patterns = array( - 'text-two-columns', - 'two-buttons', - 'two-images', - 'text-two-columns-with-images', - 'text-three-columns-buttons', - 'large-header', - 'large-header-button', - 'three-buttons', - 'heading-paragraph', - 'quote', - 'query-standard-posts', - 'query-medium-posts', - 'query-small-posts', - 'query-grid-posts', - 'query-large-title-posts', - 'query-offset-posts', - 'social-links-shared-background-color', - ); - - foreach ( $core_block_patterns as $core_block_pattern ) { - $name = 'core/' . $core_block_pattern; - if ( WP_Block_Patterns_Registry::get_instance()->is_registered( $name ) ) { - unregister_block_pattern( $name ); - } - } -} - -add_action( - 'init', - function() { - if ( ! get_theme_support( 'core-block-patterns' ) || ! function_exists( 'unregister_block_pattern' ) ) { - return; - } - gutenberg_remove_core_patterns(); - gutenberg_register_gutenberg_patterns(); - } -); diff --git a/lib/compat/wordpress-6.0/block-patterns.php b/lib/compat/wordpress-6.0/block-patterns.php deleted file mode 100644 index c42ec73152219..0000000000000 --- a/lib/compat/wordpress-6.0/block-patterns.php +++ /dev/null @@ -1,44 +0,0 @@ -get_patterns(); - if ( empty( $pattern_settings ) ) { - return; - } - - $request = new WP_REST_Request( 'GET', '/wp/v2/pattern-directory/patterns' ); - $request['slug'] = $pattern_settings; - $response = rest_do_request( $request ); - if ( $response->is_error() ) { - return; - } - $patterns = $response->get_data(); - $patterns_registry = WP_Block_Patterns_Registry::get_instance(); - foreach ( $patterns as $pattern ) { - $pattern_name = sanitize_title( $pattern['title'] ); - // Some patterns might be already registered as core patterns with the `core` prefix. - $is_registered = $patterns_registry->is_registered( $pattern_name ) || $patterns_registry->is_registered( "core/$pattern_name" ); - if ( ! $is_registered ) { - register_block_pattern( $pattern_name, (array) $pattern ); - } - } - } -} diff --git a/lib/compat/wordpress-6.0/block-template-utils.php b/lib/compat/wordpress-6.0/block-template-utils.php deleted file mode 100644 index 1fcd3dae83e59..0000000000000 --- a/lib/compat/wordpress-6.0/block-template-utils.php +++ /dev/null @@ -1,132 +0,0 @@ -get( 'TextDomain' ); - $filename = get_temp_dir() . $theme_name . $obscura . '.zip'; - - $zip = new ZipArchive(); - if ( true !== $zip->open( $filename, ZipArchive::CREATE | ZipArchive::OVERWRITE ) ) { - return new WP_Error( 'unable_to_create_zip', __( 'Unable to open export file (archive) for writing.', 'gutenberg' ) ); - } - - $zip->addEmptyDir( 'templates' ); - $zip->addEmptyDir( 'parts' ); - - // Get path of the theme. - $theme_path = wp_normalize_path( get_stylesheet_directory() ); - - // Create recursive directory iterator. - $theme_files = new RecursiveIteratorIterator( - new RecursiveDirectoryIterator( $theme_path ), - RecursiveIteratorIterator::LEAVES_ONLY - ); - - // Make a copy of the current theme. - foreach ( $theme_files as $file ) { - // Skip directories as they are added automatically. - if ( ! $file->isDir() ) { - // Get real and relative path for current file. - $file_path = wp_normalize_path( $file ); - $relative_path = substr( $file_path, strlen( $theme_path ) + 1 ); - - if ( ! gutenberg_is_theme_directory_ignored( $relative_path ) ) { - $zip->addFile( $file_path, $relative_path ); - } - } - } - - // Load templates into the zip file. - $templates = gutenberg_get_block_templates(); - foreach ( $templates as $template ) { - $template->content = _remove_theme_attribute_in_block_template_content( $template->content ); - - $zip->addFromString( - 'templates/' . $template->slug . '.html', - $template->content - ); - } - - // Load template parts into the zip file. - $template_parts = gutenberg_get_block_templates( array(), 'wp_template_part' ); - foreach ( $template_parts as $template_part ) { - $zip->addFromString( - 'parts/' . $template_part->slug . '.html', - $template_part->content - ); - } - - // Load theme.json into the zip file. - $tree = WP_Theme_JSON_Resolver_Gutenberg::get_theme_data( array(), array( 'with_supports' => false ) ); - // Merge with user data. - $tree->merge( WP_Theme_JSON_Resolver_Gutenberg::get_user_data() ); - - $theme_json_raw = $tree->get_data(); - // If a version is defined, add a schema. - if ( $theme_json_raw['version'] ) { - global $wp_version; - $theme_json_version = 'wp/' . substr( $wp_version, 0, 3 ); - if ( defined( 'IS_GUTENBERG_PLUGIN' ) ) { - $theme_json_version = 'trunk'; - } - $schema = array( '$schema' => 'https://schemas.wp.org/' . $theme_json_version . '/theme.json' ); - $theme_json_raw = array_merge( $schema, $theme_json_raw ); - } - - // Convert to a string. - $theme_json_encoded = wp_json_encode( $theme_json_raw, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE ); - - // Replace 4 spaces with a tab. - $theme_json_tabbed = preg_replace( '~(?:^|\G)\h{4}~m', "\t", $theme_json_encoded ); - - // Add the theme.json file to the zip. - $zip->addFromString( - 'theme.json', - $theme_json_tabbed - ); - - // Save changes to the zip file. - $zip->close(); - - return $filename; -} diff --git a/lib/compat/wordpress-6.0/blocks.php b/lib/compat/wordpress-6.0/blocks.php deleted file mode 100644 index ed83f62f76d42..0000000000000 --- a/lib/compat/wordpress-6.0/blocks.php +++ /dev/null @@ -1,194 +0,0 @@ - 'comment_date_gmt', - 'order' => 'ASC', - 'status' => 'approve', - 'no_found_rows' => false, - ); - - if ( is_user_logged_in() ) { - $comment_args['include_unapproved'] = array( get_current_user_id() ); - } else { - $unapproved_email = wp_get_unapproved_comment_author_email(); - - if ( $unapproved_email ) { - $comment_args['include_unapproved'] = array( $unapproved_email ); - } - } - - if ( ! empty( $block->context['postId'] ) ) { - $comment_args['post_id'] = (int) $block->context['postId']; - } - - if ( get_option( 'thread_comments' ) ) { - $comment_args['hierarchical'] = 'threaded'; - } else { - $comment_args['hierarchical'] = false; - } - - if ( get_option( 'page_comments' ) === '1' || get_option( 'page_comments' ) === true ) { - $per_page = get_option( 'comments_per_page' ); - $default_page = get_option( 'default_comments_page' ); - if ( $per_page > 0 ) { - $comment_args['number'] = $per_page; - - $page = (int) get_query_var( 'cpage' ); - if ( $page ) { - $comment_args['paged'] = $page; - } elseif ( 'oldest' === $default_page ) { - $comment_args['paged'] = 1; - } elseif ( 'newest' === $default_page ) { - $max_num_pages = (int) ( new WP_Comment_Query( $comment_args ) )->max_num_pages; - if ( 0 !== $max_num_pages ) { - $comment_args['paged'] = $max_num_pages; - } - } - // Set the `cpage` query var to ensure the previous and next pagination links are correct - // when inheriting the Discussion Settings. - if ( 0 === $page && isset( $comment_args['paged'] ) && $comment_args['paged'] > 0 ) { - set_query_var( 'cpage', $comment_args['paged'] ); - } - } - } - - return $comment_args; - } -} - -if ( ! function_exists( 'get_comments_pagination_arrow' ) ) { - /** - * Helper function that returns the proper pagination arrow html for - * `CommentsPaginationNext` and `CommentsPaginationPrevious` blocks based - * on the provided `paginationArrow` from `CommentsPagination` context. - * - * It's used in CommentsPaginationNext and CommentsPaginationPrevious blocks. - * - * @since 6.0.0 - * - * @param WP_Block $block Block instance. - * @param string $pagination_type Type of the arrow we will be rendering. Default 'next'. Accepts 'next' or 'previous'. - * - * @return string|null Returns the constructed WP_Query arguments. - */ - function get_comments_pagination_arrow( $block, $pagination_type = 'next' ) { - $arrow_map = array( - 'none' => '', - 'arrow' => array( - 'next' => '→', - 'previous' => '←', - ), - 'chevron' => array( - 'next' => '»', - 'previous' => '«', - ), - ); - if ( ! empty( $block->context['comments/paginationArrow'] ) && ! empty( $arrow_map[ $block->context['comments/paginationArrow'] ][ $pagination_type ] ) ) { - $arrow_attribute = $block->context['comments/paginationArrow']; - $arrow = $arrow_map[ $block->context['comments/paginationArrow'] ][ $pagination_type ]; - $arrow_classes = "wp-block-comments-pagination-$pagination_type-arrow is-arrow-$arrow_attribute"; - return "$arrow"; - } - return null; - } -} - -/** - * Workaround for getting discussion settings as block editor settings - * so any user can access to them without needing to be an admin. - * - * @param array $settings Default editor settings. - * - * @return array Filtered editor settings. - */ -function gutenberg_extend_block_editor_settings_with_discussion_settings( $settings ) { - - $settings['__experimentalDiscussionSettings'] = array( - 'commentOrder' => get_option( 'comment_order' ), - 'commentsPerPage' => get_option( 'comments_per_page' ), - 'defaultCommentsPage' => get_option( 'default_comments_page' ), - 'pageComments' => get_option( 'page_comments' ), - 'threadComments' => get_option( 'thread_comments' ), - 'threadCommentsDepth' => get_option( 'thread_comments_depth' ), - 'defaultCommentStatus' => get_option( 'default_comment_status' ), - 'avatarURL' => get_avatar_url( - '', - array( - 'size' => 96, - 'force_default' => true, - 'default' => get_option( 'avatar_default' ), - ) - ), - ); - - return $settings; -} -add_filter( 'block_editor_settings_all', 'gutenberg_extend_block_editor_settings_with_discussion_settings' ); - -/** - * Mark the `children` attr of comments as embeddable so they can be included in - * REST API responses without additional requests. - * - * @return void - */ -function gutenberg_rest_comment_set_children_as_embeddable() { - add_filter( - 'rest_prepare_comment', - function ( $response ) { - $links = $response->get_links(); - if ( isset( $links['children'] ) ) { - $href = $links['children'][0]['href']; - $response->remove_link( 'children', $href ); - $response->add_link( 'children', $href, array( 'embeddable' => true ) ); - } - return $response; - } - ); -} -add_action( 'rest_api_init', 'gutenberg_rest_comment_set_children_as_embeddable' ); - -/** - * Registers the lock block attribute for block types. - * - * Once 6.0 is the minimum supported WordPress version for the Gutenberg - * plugin, this shim can be removed - * - * Doesn't need to be backported into Core. - * - * @param array $args Array of arguments for registering a block type. - * @return array $args - */ -function gutenberg_register_lock_attribute( $args ) { - // Setup attributes if needed. - if ( ! isset( $args['attributes'] ) || ! is_array( $args['attributes'] ) ) { - $args['attributes'] = array(); - } - - if ( ! array_key_exists( 'lock', $args['attributes'] ) ) { - $args['attributes']['lock'] = array( - 'type' => 'object', - ); - } - - return $args; -} -add_filter( 'register_block_type_args', 'gutenberg_register_lock_attribute' ); diff --git a/lib/compat/wordpress-6.0/class-gutenberg-rest-edit-site-export-controller.php b/lib/compat/wordpress-6.0/class-gutenberg-rest-edit-site-export-controller.php deleted file mode 100644 index e4cffb2bff268..0000000000000 --- a/lib/compat/wordpress-6.0/class-gutenberg-rest-edit-site-export-controller.php +++ /dev/null @@ -1,89 +0,0 @@ -namespace = 'wp-block-editor/v1'; - $this->rest_base = 'export'; - } - - /** - * Registers the necessary REST API routes. - */ - public function register_routes() { - register_rest_route( - $this->namespace, - '/' . $this->rest_base, - array( - array( - 'methods' => WP_REST_Server::READABLE, - 'callback' => array( $this, 'export' ), - 'permission_callback' => array( $this, 'permissions_check' ), - ), - ), - true // Override core route if already exists (WP 5.9). - ); - } - - /** - * Checks whether a given request has permission to export. - * - * @return WP_Error|bool True if the request has access, or WP_Error object. - */ - public function permissions_check() { - if ( current_user_can( 'edit_theme_options' ) ) { - return true; - } - - return new WP_Error( - 'rest_cannot_export_templates', - __( 'Sorry, you are not allowed to export templates and template parts.', 'gutenberg' ), - array( 'status' => rest_authorization_required_code() ) - ); - } - - /** - * Output a ZIP file with an export of the current templates - * template parts, theme.json and index.php from the site editor, - * and close the connection. - * - * @return WP_Error|void - */ - public function export() { - // Generate the export file. - $filename = gutenberg_generate_block_templates_export_file(); - - if ( is_wp_error( $filename ) ) { - $filename->add_data( array( 'status' => 500 ) ); - - return $filename; - } - - $theme_name = basename( get_stylesheet() ); - - header( 'Content-Type: application/zip' ); - header( 'Content-Disposition: attachment; filename=' . $theme_name . '.zip' ); - header( 'Content-Length: ' . filesize( $filename ) ); - flush(); - readfile( $filename ); - unlink( $filename ); - exit; - } -} diff --git a/lib/compat/wordpress-6.0/class-gutenberg-rest-pattern-directory-controller-6-0.php b/lib/compat/wordpress-6.0/class-gutenberg-rest-pattern-directory-controller-6-0.php deleted file mode 100644 index 331e6015edb0c..0000000000000 --- a/lib/compat/wordpress-6.0/class-gutenberg-rest-pattern-directory-controller-6-0.php +++ /dev/null @@ -1,48 +0,0 @@ -= 6.0. - * - * @param array $query_args Query arguments to generate a transient key from. - * @return string Transient key. - */ - protected function get_transient_key( $query_args ) { - if ( method_exists( get_parent_class( $this ), __FUNCTION__ ) ) { - return parent::get_transient_key( $query_args ); - } - - if ( isset( $query_args['slug'] ) ) { - // This is an additional precaution because the "sort" function expects an array. - $query_args['slug'] = wp_parse_list( $query_args['slug'] ); - - // Empty arrays should not affect the transient key. - if ( empty( $query_args['slug'] ) ) { - unset( $query_args['slug'] ); - } else { - // Sort the array so that the transient key doesn't depend on the order of slugs. - sort( $query_args['slug'] ); - } - } - - return 'wp_remote_block_patterns_' . md5( serialize( $query_args ) ); - } -} diff --git a/lib/compat/wordpress-6.0/class-wp-rest-block-pattern-categories-controller.php b/lib/compat/wordpress-6.0/class-wp-rest-block-pattern-categories-controller.php deleted file mode 100644 index 0f60781dae106..0000000000000 --- a/lib/compat/wordpress-6.0/class-wp-rest-block-pattern-categories-controller.php +++ /dev/null @@ -1,152 +0,0 @@ -namespace = 'wp/v2'; - $this->rest_base = 'block-patterns/categories'; - } - - /** - * Registers the routes for the objects of the controller. - * - * @see register_rest_route() - * - * @since 6.0.0 - */ - public function register_routes() { - register_rest_route( - $this->namespace, - '/' . $this->rest_base, - array( - array( - 'methods' => WP_REST_Server::READABLE, - 'callback' => array( $this, 'get_items' ), - 'permission_callback' => array( $this, 'get_items_permissions_check' ), - ), - 'schema' => array( $this, 'get_public_item_schema' ), - ) - ); - } - - /** - * Checks whether a given request has permission to read block patterns. - * - * @since 6.0.0 - * - * @param WP_REST_Request $request Full details about the request. - * - * @return WP_Error|bool True if the request has read access, WP_Error object otherwise. - */ - public function get_items_permissions_check( $request ) { // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable - if ( current_user_can( 'edit_posts' ) ) { - return true; - } - - foreach ( get_post_types( array( 'show_in_rest' => true ), 'objects' ) as $post_type ) { - if ( current_user_can( $post_type->cap->edit_posts ) ) { - return true; - } - } - - return new WP_Error( - 'rest_cannot_view', - __( 'Sorry, you are not allowed to view the registered block pattern categories.', 'gutenberg' ), - array( 'status' => rest_authorization_required_code() ) - ); - } - - /** - * Retrieves all block pattern categories. - * - * @since 6.0.0 - * - * @param WP_REST_Request $request Full details about the request. - * - * @return WP_Error|WP_REST_Response Response object on success, or WP_Error object on failure. - */ - public function get_items( $request ) { // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable - $response = array(); - $categories = WP_Block_Pattern_Categories_Registry::get_instance()->get_all_registered(); - foreach ( $categories as $category ) { - $prepared_category = $this->prepare_item_for_response( $category, $request ); - $response[] = $this->prepare_response_for_collection( $prepared_category ); - } - return rest_ensure_response( $response ); - } - - /** - * Prepare a raw block pattern category before it gets output in a REST API response. - * - * @since 6.0.0 - * - * @param object $item Raw category as registered, before any changes. - * @param WP_REST_Request $request Request object. - * @return WP_REST_Response - */ - public function prepare_item_for_response( $item, $request ) { - $fields = $this->get_fields_for_response( $request ); - $keys = array( 'name', 'label' ); - $data = array(); - foreach ( $keys as $key ) { - if ( rest_is_field_included( $key, $fields ) ) { - $data[ $key ] = $item[ $key ]; - } - } - - $context = ! empty( $request['context'] ) ? $request['context'] : 'view'; - $data = $this->add_additional_fields_to_object( $data, $request ); - $data = $this->filter_response_by_context( $data, $context ); - return rest_ensure_response( $data ); - } - - /** - * Retrieves the block pattern category schema, conforming to JSON Schema. - * - * @since 6.0.0 - * - * @return array Item schema data. - */ - public function get_item_schema() { - $schema = array( - '$schema' => 'http://json-schema.org/draft-04/schema#', - 'title' => 'block-pattern-category', - 'type' => 'object', - 'properties' => array( - 'name' => array( - 'description' => __( 'The category name.', 'gutenberg' ), - 'type' => 'string', - 'readonly' => true, - 'context' => array( 'view', 'edit', 'embed' ), - ), - 'label' => array( - 'description' => __( 'The category label, in human readable format.', 'gutenberg' ), - 'type' => 'string', - 'readonly' => true, - 'context' => array( 'view', 'edit', 'embed' ), - ), - ), - ); - - return $this->add_additional_fields_schema( $schema ); - } -} diff --git a/lib/compat/wordpress-6.0/class-wp-theme-json-6-0.php b/lib/compat/wordpress-6.0/class-wp-theme-json-6-0.php deleted file mode 100644 index 07820ccbd1dbe..0000000000000 --- a/lib/compat/wordpress-6.0/class-wp-theme-json-6-0.php +++ /dev/null @@ -1,680 +0,0 @@ - array( 'color', 'gradient' ), - 'background-color' => array( 'color', 'background' ), - 'border-radius' => array( 'border', 'radius' ), - 'border-top-left-radius' => array( 'border', 'radius', 'topLeft' ), - 'border-top-right-radius' => array( 'border', 'radius', 'topRight' ), - 'border-bottom-left-radius' => array( 'border', 'radius', 'bottomLeft' ), - 'border-bottom-right-radius' => array( 'border', 'radius', 'bottomRight' ), - 'border-color' => array( 'border', 'color' ), - 'border-width' => array( 'border', 'width' ), - 'border-style' => array( 'border', 'style' ), - 'border-top-color' => array( 'border', 'top', 'color' ), - 'border-top-width' => array( 'border', 'top', 'width' ), - 'border-top-style' => array( 'border', 'top', 'style' ), - 'border-right-color' => array( 'border', 'right', 'color' ), - 'border-right-width' => array( 'border', 'right', 'width' ), - 'border-right-style' => array( 'border', 'right', 'style' ), - 'border-bottom-color' => array( 'border', 'bottom', 'color' ), - 'border-bottom-width' => array( 'border', 'bottom', 'width' ), - 'border-bottom-style' => array( 'border', 'bottom', 'style' ), - 'border-left-color' => array( 'border', 'left', 'color' ), - 'border-left-width' => array( 'border', 'left', 'width' ), - 'border-left-style' => array( 'border', 'left', 'style' ), - 'color' => array( 'color', 'text' ), - 'font-family' => array( 'typography', 'fontFamily' ), - 'font-size' => array( 'typography', 'fontSize' ), - 'font-style' => array( 'typography', 'fontStyle' ), - 'font-weight' => array( 'typography', 'fontWeight' ), - 'letter-spacing' => array( 'typography', 'letterSpacing' ), - 'line-height' => array( 'typography', 'lineHeight' ), - 'margin' => array( 'spacing', 'margin' ), - 'margin-top' => array( 'spacing', 'margin', 'top' ), - 'margin-right' => array( 'spacing', 'margin', 'right' ), - 'margin-bottom' => array( 'spacing', 'margin', 'bottom' ), - 'margin-left' => array( 'spacing', 'margin', 'left' ), - 'padding' => array( 'spacing', 'padding' ), - 'padding-top' => array( 'spacing', 'padding', 'top' ), - 'padding-right' => array( 'spacing', 'padding', 'right' ), - 'padding-bottom' => array( 'spacing', 'padding', 'bottom' ), - 'padding-left' => array( 'spacing', 'padding', 'left' ), - '--wp--style--block-gap' => array( 'spacing', 'blockGap' ), - 'text-decoration' => array( 'typography', 'textDecoration' ), - 'text-transform' => array( 'typography', 'textTransform' ), - 'filter' => array( 'filter', 'duotone' ), - ); - - /** - * Presets are a set of values that serve - * to bootstrap some styles: colors, font sizes, etc. - * - * They are a unkeyed array of values such as: - * - * ```php - * array( - * array( - * 'slug' => 'unique-name-within-the-set', - * 'name' => 'Name for the UI', - * => 'value' - * ), - * ) - * ``` - * - * This contains the necessary metadata to process them: - * - * - path => Where to find the preset within the settings section. - * - prevent_override => Disables override of default presets by theme presets. - * The relationship between whether to override the defaults - * and whether the defaults are enabled is inverse: - * - If defaults are enabled => theme presets should not be overriden - * - If defaults are disabled => theme presets should be overriden - * For example, a theme sets defaultPalette to false, - * making the default palette hidden from the user. - * In that case, we want all the theme presets to be present, - * so they should override the defaults by setting this false. - * - use_default_names => whether to use the default names - * - value_key => the key that represents the value - * - value_func => optionally, instead of value_key, a function to generate - * the value that takes a preset as an argument - * (either value_key or value_func should be present) - * - css_vars => template string to use in generating the CSS Custom Property. - * Example output: "--wp--preset--duotone--blue: " will generate as many CSS Custom Properties as presets defined - * substituting the $slug for the slug's value for each preset value. - * - classes => array containing a structure with the classes to - * generate for the presets, where for each array item - * the key is the class name and the value the property name. - * The "$slug" substring will be replaced by the slug of each preset. - * For example: - * 'classes' => array( - * '.has-$slug-color' => 'color', - * '.has-$slug-background-color' => 'background-color', - * '.has-$slug-border-color' => 'border-color', - * ) - * - properties => array of CSS properties to be used by kses to - * validate the content of each preset - * by means of the remove_insecure_properties method. - */ - const PRESETS_METADATA = array( - array( - 'path' => array( 'color', 'palette' ), - 'prevent_override' => array( 'color', 'defaultPalette' ), - 'use_default_names' => false, - 'value_key' => 'color', - 'css_vars' => '--wp--preset--color--$slug', - 'classes' => array( - '.has-$slug-color' => 'color', - '.has-$slug-background-color' => 'background-color', - '.has-$slug-border-color' => 'border-color', - ), - 'properties' => array( 'color', 'background-color', 'border-color' ), - ), - array( - 'path' => array( 'color', 'gradients' ), - 'prevent_override' => array( 'color', 'defaultGradients' ), - 'use_default_names' => false, - 'value_key' => 'gradient', - 'css_vars' => '--wp--preset--gradient--$slug', - 'classes' => array( '.has-$slug-gradient-background' => 'background' ), - 'properties' => array( 'background' ), - ), - array( - 'path' => array( 'color', 'duotone' ), - 'prevent_override' => array( 'color', 'defaultDuotone' ), - 'use_default_names' => false, - 'value_func' => 'gutenberg_get_duotone_filter_property', - 'css_vars' => '--wp--preset--duotone--$slug', - 'classes' => array(), - 'properties' => array( 'filter' ), - ), - array( - 'path' => array( 'typography', 'fontSizes' ), - 'prevent_override' => false, - 'use_default_names' => true, - 'value_key' => 'size', - 'css_vars' => '--wp--preset--font-size--$slug', - 'classes' => array( '.has-$slug-font-size' => 'font-size' ), - 'properties' => array( 'font-size' ), - ), - array( - 'path' => array( 'typography', 'fontFamilies' ), - 'prevent_override' => false, - 'use_default_names' => false, - 'value_key' => 'fontFamily', - 'css_vars' => '--wp--preset--font-family--$slug', - 'classes' => array( '.has-$slug-font-family' => 'font-family' ), - 'properties' => array( 'font-family' ), - ), - ); - - /** - * The top-level keys a theme.json can have. - * - * @var string[] - */ - const VALID_TOP_LEVEL_KEYS = array( - 'customTemplates', - 'patterns', - 'settings', - 'styles', - 'templateParts', - 'version', - 'title', - ); - - const APPEARANCE_TOOLS_OPT_INS = array( - array( 'border', 'color' ), - array( 'border', 'radius' ), - array( 'border', 'style' ), - array( 'border', 'width' ), - array( 'color', 'link' ), - array( 'spacing', 'blockGap' ), - array( 'spacing', 'margin' ), - array( 'spacing', 'padding' ), - array( 'typography', 'lineHeight' ), - ); - - /** - * The valid properties under the settings key. - * - * @var array - */ - const VALID_SETTINGS = array( - 'appearanceTools' => null, - 'border' => array( - 'color' => null, - 'radius' => null, - 'style' => null, - 'width' => null, - ), - 'color' => array( - 'background' => null, - 'custom' => null, - 'customDuotone' => null, - 'customGradient' => null, - 'defaultDuotone' => null, - 'defaultGradients' => null, - 'defaultPalette' => null, - 'duotone' => null, - 'gradients' => null, - 'link' => null, - 'palette' => null, - 'text' => null, - ), - 'custom' => null, - 'layout' => array( - 'contentSize' => null, - 'wideSize' => null, - ), - 'spacing' => array( - 'blockGap' => null, - 'margin' => null, - 'padding' => null, - 'units' => null, - ), - 'typography' => array( - 'customFontSize' => null, - 'dropCap' => null, - 'fontFamilies' => null, - 'fontSizes' => null, - 'fontStyle' => null, - 'fontWeight' => null, - 'letterSpacing' => null, - 'lineHeight' => null, - 'textDecoration' => null, - 'textTransform' => null, - ), - ); - - /** - * The valid properties under the styles key. - * - * @var array - */ - const VALID_STYLES = array( - 'border' => array( - 'color' => null, - 'radius' => null, - 'style' => null, - 'width' => null, - 'top' => null, - 'right' => null, - 'bottom' => null, - 'left' => null, - ), - 'color' => array( - 'background' => null, - 'gradient' => null, - 'text' => null, - ), - 'filter' => array( - 'duotone' => null, - ), - 'spacing' => array( - 'margin' => null, - 'padding' => null, - 'blockGap' => 'top', - ), - 'typography' => array( - 'fontFamily' => null, - 'fontSize' => null, - 'fontStyle' => null, - 'fontWeight' => null, - 'letterSpacing' => null, - 'lineHeight' => null, - 'textDecoration' => null, - 'textTransform' => null, - ), - ); - - /** - * Returns the current theme's wanted patterns(slugs) to be - * registered from Pattern Directory. - * - * @return array - */ - public function get_patterns() { - if ( isset( $this->theme_json['patterns'] ) && is_array( $this->theme_json['patterns'] ) ) { - return $this->theme_json['patterns']; - } - return array(); - } - - /** - * Converts each style section into a list of rulesets - * containing the block styles to be appended to the stylesheet. - * - * See glossary at https://developer.mozilla.org/en-US/docs/Web/CSS/Syntax - * - * For each section this creates a new ruleset such as: - * - * block-selector { - * style-property-one: value; - * } - * - * @param array $style_nodes Nodes with styles. - * @return string The new stylesheet. - */ - protected function get_block_classes( $style_nodes ) { - $block_rules = ''; - - foreach ( $style_nodes as $metadata ) { - if ( null === $metadata['selector'] ) { - continue; - } - - $node = _wp_array_get( $this->theme_json, $metadata['path'], array() ); - $selector = $metadata['selector']; - $settings = _wp_array_get( $this->theme_json, array( 'settings' ) ); - $declarations = static::compute_style_properties( $node, $settings ); - - // 1. Separate the ones who use the general selector - // and the ones who use the duotone selector. - $declarations_duotone = array(); - foreach ( $declarations as $index => $declaration ) { - if ( 'filter' === $declaration['name'] ) { - unset( $declarations[ $index ] ); - $declarations_duotone[] = $declaration; - } - } - - /* - * Reset default browser margin on the root body element. - * This is set on the root selector **before** generating the ruleset - * from the `theme.json`. This is to ensure that if the `theme.json` declares - * `margin` in its `spacing` declaration for the `body` element then these - * user-generated values take precedence in the CSS cascade. - * @link https://github.com/WordPress/gutenberg/issues/36147. - */ - if ( static::ROOT_BLOCK_SELECTOR === $selector ) { - $block_rules .= 'body { margin: 0; }'; - } - - // 2. Generate the rules that use the general selector. - $block_rules .= static::to_ruleset( $selector, $declarations ); - - // 3. Generate the rules that use the duotone selector. - if ( isset( $metadata['duotone'] ) && ! empty( $declarations_duotone ) ) { - $selector_duotone = static::scope_selector( $metadata['selector'], $metadata['duotone'] ); - $block_rules .= static::to_ruleset( $selector_duotone, $declarations_duotone ); - } - - if ( static::ROOT_BLOCK_SELECTOR === $selector ) { - $block_rules .= '.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }'; - $block_rules .= '.wp-site-blocks > .alignright { float: right; margin-left: 2em; }'; - $block_rules .= '.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }'; - - $has_block_gap_support = _wp_array_get( $this->theme_json, array( 'settings', 'spacing', 'blockGap' ) ) !== null; - if ( $has_block_gap_support ) { - $block_rules .= '.wp-site-blocks > * { margin-block-start: 0; margin-block-end: 0; }'; - $block_rules .= '.wp-site-blocks > * + * { margin-block-start: var( --wp--style--block-gap ); }'; - } - } - } - - return $block_rules; - } - - /** - * Merge new incoming data. - * - * @param WP_Theme_JSON $incoming Data to merge. - */ - public function merge( $incoming ) { - $incoming_data = $incoming->get_raw_data(); - $this->theme_json = array_replace_recursive( $this->theme_json, $incoming_data ); - - /* - * The array_replace_recursive algorithm merges at the leaf level, - * but we don't want leaf arrays to be merged, so we overwrite it. - * - * For leaf values that are sequential arrays it will use the numeric indexes for replacement. - * We rather replace the existing with the incoming value, if it exists. - * This is the case of spacing.units. - * - * For leaf values that are associative arrays it will merge them as expected. - * This is also not the behavior we want for the current associative arrays (presets). - * We rather replace the existing with the incoming value, if it exists. - * This happens, for example, when we merge data from theme.json upon existing - * theme supports or when we merge anything coming from the same source twice. - * This is the case of color.palette, color.gradients, color.duotone, - * typography.fontSizes, or typography.fontFamilies. - * - * Additionally, for some preset types, we also want to make sure the - * values they introduce don't conflict with default values. We do so - * by checking the incoming slugs for theme presets and compare them - * with the equivalent default presets: if a slug is present as a default - * we remove it from the theme presets. - */ - $nodes = static::get_setting_nodes( $incoming_data ); - $slugs_global = static::get_default_slugs( $this->theme_json, array( 'settings' ) ); - foreach ( $nodes as $node ) { - $slugs_node = static::get_default_slugs( $this->theme_json, $node['path'] ); - $slugs = array_merge_recursive( $slugs_global, $slugs_node ); - - // Replace the spacing.units. - $path = array_merge( $node['path'], array( 'spacing', 'units' ) ); - $content = _wp_array_get( $incoming_data, $path, null ); - if ( isset( $content ) ) { - _wp_array_set( $this->theme_json, $path, $content ); - } - - // Replace the presets. - foreach ( static::PRESETS_METADATA as $preset ) { - $override_preset = ! static::get_metadata_boolean( $this->theme_json['settings'], $preset['prevent_override'], true ); - - foreach ( static::VALID_ORIGINS as $origin ) { - $base_path = array_merge( $node['path'], $preset['path'] ); - $path = array_merge( $base_path, array( $origin ) ); - $content = _wp_array_get( $incoming_data, $path, null ); - if ( ! isset( $content ) ) { - continue; - } - - if ( 'theme' === $origin && $preset['use_default_names'] ) { - foreach ( $content as &$item ) { - if ( ! array_key_exists( 'name', $item ) ) { - $name = static::get_name_from_defaults( $item['slug'], $base_path ); - if ( null !== $name ) { - $item['name'] = $name; - } - } - } - } - - if ( - ( 'theme' !== $origin ) || - ( 'theme' === $origin && $override_preset ) - ) { - _wp_array_set( $this->theme_json, $path, $content ); - } else { - $slugs_for_preset = _wp_array_get( $slugs, $preset['path'], array() ); - $content = static::filter_slugs( $content, $slugs_for_preset ); - _wp_array_set( $this->theme_json, $path, $content ); - } - } - } - } - } - - /** - * Converts all filter (duotone) presets into SVGs. - * - * @param array $origins List of origins to process. - * - * @return string SVG filters. - */ - public function get_svg_filters( $origins ) { - $blocks_metadata = static::get_blocks_metadata(); - $setting_nodes = static::get_setting_nodes( $this->theme_json, $blocks_metadata ); - - $filters = ''; - foreach ( $setting_nodes as $metadata ) { - $node = _wp_array_get( $this->theme_json, $metadata['path'], array() ); - if ( empty( $node['color']['duotone'] ) ) { - continue; - } - - $duotone_presets = $node['color']['duotone']; - - foreach ( $origins as $origin ) { - if ( ! isset( $duotone_presets[ $origin ] ) ) { - continue; - } - foreach ( $duotone_presets[ $origin ] as $duotone_preset ) { - $filters .= gutenberg_get_duotone_filter_svg( $duotone_preset ); - } - } - } - - return $filters; - } - - /** - * For metadata values that can either be booleans or paths to booleans, gets the value. - * - * ```php - * $data = array( - * 'color' => array( - * 'defaultPalette' => true - * ) - * ); - * - * static::get_metadata_boolean( $data, false ); - * // => false - * - * static::get_metadata_boolean( $data, array( 'color', 'defaultPalette' ) ); - * // => true - * ``` - * - * @param array $data The data to inspect. - * @param bool|array $path Boolean or path to a boolean. - * @param bool $default Default value if the referenced path is missing. - * @return boolean - */ - protected static function get_metadata_boolean( $data, $path, $default = false ) { - if ( is_bool( $path ) ) { - return $path; - } - - if ( is_array( $path ) ) { - $value = _wp_array_get( $data, $path ); - if ( null !== $value ) { - return $value; - } - } - - return $default; - } - - /** - * Returns a valid theme.json as provided by a theme. - * - * Unlike get_raw_data() this returns the presets flattened, as provided by a theme. - * This also uses appearanceTools instead of their opt-ins if all of them are true. - * - * @return string[] - */ - public function get_data() { - $output = $this->theme_json; - $nodes = static::get_setting_nodes( $output ); - - /** - * Flatten the theme & custom origins into a single one. - * - * For example, the following: - * - * { - * "settings": { - * "color": { - * "palette": { - * "theme": [ {} ], - * "custom": [ {} ] - * } - * } - * } - * } - * - * will be converted to: - * - * { - * "settings": { - * "color": { - * "palette": [ {} ] - * } - * } - * } - */ - foreach ( $nodes as $node ) { - foreach ( static::PRESETS_METADATA as $preset_metadata ) { - $path = array_merge( $node['path'], $preset_metadata['path'] ); - $preset = _wp_array_get( $output, $path, null ); - if ( null === $preset ) { - continue; - } - - $items = array(); - if ( isset( $preset['theme'] ) ) { - foreach ( $preset['theme'] as $item ) { - $slug = $item['slug']; - unset( $item['slug'] ); - $items[ $slug ] = $item; - } - } - if ( isset( $preset['custom'] ) ) { - foreach ( $preset['custom'] as $item ) { - $slug = $item['slug']; - unset( $item['slug'] ); - $items[ $slug ] = $item; - } - } - $flattened_preset = array(); - foreach ( $items as $slug => $value ) { - $flattened_preset[] = array_merge( array( 'slug' => (string) $slug ), $value ); - } - _wp_array_set( $output, $path, $flattened_preset ); - } - } - - // If all of the static::APPEARANCE_TOOLS_OPT_INS are true, - // this code unsets them and sets 'appearanceTools' instead. - foreach ( $nodes as $node ) { - $all_opt_ins_are_set = true; - foreach ( static::APPEARANCE_TOOLS_OPT_INS as $opt_in_path ) { - $full_path = array_merge( $node['path'], $opt_in_path ); - // Use "unset prop" as a marker instead of "null" because - // "null" can be a valid value for some props (e.g. blockGap). - $opt_in_value = _wp_array_get( $output, $full_path, 'unset prop' ); - if ( 'unset prop' === $opt_in_value ) { - $all_opt_ins_are_set = false; - break; - } - } - - if ( $all_opt_ins_are_set ) { - _wp_array_set( $output, array_merge( $node['path'], array( 'appearanceTools' ) ), true ); - foreach ( static::APPEARANCE_TOOLS_OPT_INS as $opt_in_path ) { - $full_path = array_merge( $node['path'], $opt_in_path ); - // Use "unset prop" as a marker instead of "null" because - // "null" can be a valid value for some props (e.g. blockGap). - $opt_in_value = _wp_array_get( $output, $full_path, 'unset prop' ); - if ( true !== $opt_in_value ) { - continue; - } - - // The following could be improved to be path independent. - // At the moment it relies on a couple of assumptions: - // - // - all opt-ins having a path of size 2. - // - there's two sources of settings: the top-level and the block-level. - if ( - ( 1 === count( $node['path'] ) ) && - ( 'settings' === $node['path'][0] ) - ) { - // Top-level settings. - unset( $output['settings'][ $opt_in_path[0] ][ $opt_in_path[1] ] ); - if ( empty( $output['settings'][ $opt_in_path[0] ] ) ) { - unset( $output['settings'][ $opt_in_path[0] ] ); - } - } elseif ( - ( 3 === count( $node['path'] ) ) && - ( 'settings' === $node['path'][0] ) && - ( 'blocks' === $node['path'][1] ) - ) { - // Block-level settings. - $block_name = $node['path'][2]; - unset( $output['settings']['blocks'][ $block_name ][ $opt_in_path[0] ][ $opt_in_path[1] ] ); - if ( empty( $output['settings']['blocks'][ $block_name ][ $opt_in_path[0] ] ) ) { - unset( $output['settings']['blocks'][ $block_name ][ $opt_in_path[0] ] ); - } - } - } - } - } - - wp_recursive_ksort( $output ); - - return $output; - } - - /** - * Enables some settings. - * - * @since 5.9.0 - * - * @param array $context The context to which the settings belong. - */ - protected static function do_opt_in_into_settings( &$context ) { - foreach ( static::APPEARANCE_TOOLS_OPT_INS as $path ) { - // Use "unset prop" as a marker instead of "null" because - // "null" can be a valid value for some props (e.g. blockGap). - if ( 'unset prop' === _wp_array_get( $context, $path, 'unset prop' ) ) { - _wp_array_set( $context, $path, true ); - } - } - - unset( $context['appearanceTools'] ); - } -} diff --git a/lib/compat/wordpress-6.0/class-wp-theme-json-resolver-6-0.php b/lib/compat/wordpress-6.0/class-wp-theme-json-resolver-6-0.php deleted file mode 100644 index 07b83049cfeaa..0000000000000 --- a/lib/compat/wordpress-6.0/class-wp-theme-json-resolver-6-0.php +++ /dev/null @@ -1,223 +0,0 @@ - true ) ); - - if ( null === static::$theme ) { - $theme_json_data = static::read_json_file( static::get_file_path_from_theme( 'theme.json' ) ); - $theme_json_data = static::translate( $theme_json_data, wp_get_theme()->get( 'TextDomain' ) ); - static::$theme = new WP_Theme_JSON_Gutenberg( $theme_json_data ); - - if ( wp_get_theme()->parent() ) { - // Get parent theme.json. - $parent_theme_json_data = static::read_json_file( static::get_file_path_from_theme( 'theme.json', true ) ); - $parent_theme_json_data = static::translate( $parent_theme_json_data, wp_get_theme()->parent()->get( 'TextDomain' ) ); - $parent_theme = new WP_Theme_JSON_Gutenberg( $parent_theme_json_data ); - - // Merge the child theme.json into the parent theme.json. - // The child theme takes precedence over the parent. - $parent_theme->merge( static::$theme ); - static::$theme = $parent_theme; - } - } - - if ( ! $options['with_supports'] ) { - return static::$theme; - } - - /* - * We want the presets and settings declared in theme.json - * to override the ones declared via theme supports. - * So we take theme supports, transform it to theme.json shape - * and merge the static::$theme upon that. - */ - $theme_support_data = WP_Theme_JSON_Gutenberg::get_from_editor_settings( get_default_block_editor_settings() ); - if ( ! wp_theme_has_theme_json() ) { - if ( ! isset( $theme_support_data['settings']['color'] ) ) { - $theme_support_data['settings']['color'] = array(); - } - - $default_palette = false; - if ( current_theme_supports( 'default-color-palette' ) ) { - $default_palette = true; - } - if ( ! isset( $theme_support_data['settings']['color']['palette'] ) ) { - // If the theme does not have any palette, we still want to show the core one. - $default_palette = true; - } - $theme_support_data['settings']['color']['defaultPalette'] = $default_palette; - - $default_gradients = false; - if ( current_theme_supports( 'default-gradient-presets' ) ) { - $default_gradients = true; - } - if ( ! isset( $theme_support_data['settings']['color']['gradients'] ) ) { - // If the theme does not have any gradients, we still want to show the core ones. - $default_gradients = true; - } - $theme_support_data['settings']['color']['defaultGradients'] = $default_gradients; - - // Classic themes without a theme.json don't support global duotone. - $theme_support_data['settings']['color']['defaultDuotone'] = false; - } - $with_theme_supports = new WP_Theme_JSON_Gutenberg( $theme_support_data ); - $with_theme_supports->merge( static::$theme ); - - return $with_theme_supports; - } - /** - * Returns the style variations defined by the theme. - * - * @return array - */ - public static function get_style_variations() { - $variations = array(); - $base_directory = get_stylesheet_directory() . '/styles'; - if ( is_dir( $base_directory ) ) { - $nested_files = new RecursiveIteratorIterator( new RecursiveDirectoryIterator( $base_directory ) ); - $nested_html_files = iterator_to_array( new RegexIterator( $nested_files, '/^.+\.json$/i', RecursiveRegexIterator::GET_MATCH ) ); - ksort( $nested_html_files ); - foreach ( $nested_html_files as $path => $file ) { - $decoded_file = wp_json_file_decode( $path, array( 'associative' => true ) ); - if ( is_array( $decoded_file ) ) { - $translated = static::translate( $decoded_file, wp_get_theme()->get( 'TextDomain' ) ); - $variation = ( new WP_Theme_JSON_Gutenberg( $translated ) )->get_raw_data(); - if ( empty( $variation['title'] ) ) { - $variation['title'] = basename( $path, '.json' ); - } - $variations[] = $variation; - } - } - } - return $variations; - } - - /** - * Returns the user's origin config. - * - * @return WP_Theme_JSON_Gutenberg Entity that holds styles for user data. - */ - public static function get_user_data() { - if ( null !== static::$user ) { - return static::$user; - } - - $config = array(); - $user_cpt = static::get_user_data_from_wp_global_styles( wp_get_theme() ); - - if ( array_key_exists( 'post_content', $user_cpt ) ) { - $decoded_data = json_decode( $user_cpt['post_content'], true ); - - $json_decoding_error = json_last_error(); - if ( JSON_ERROR_NONE !== $json_decoding_error ) { - trigger_error( 'Error when decoding a theme.json schema for user data. ' . json_last_error_msg() ); - return new WP_Theme_JSON_Gutenberg( $config, 'custom' ); - } - - // Very important to verify if the flag isGlobalStylesUserThemeJSON is true. - // If is not true the content was not escaped and is not safe. - if ( - is_array( $decoded_data ) && - isset( $decoded_data['isGlobalStylesUserThemeJSON'] ) && - $decoded_data['isGlobalStylesUserThemeJSON'] - ) { - unset( $decoded_data['isGlobalStylesUserThemeJSON'] ); - $config = $decoded_data; - } - } - static::$user = new WP_Theme_JSON_Gutenberg( $config, 'custom' ); - - return static::$user; - } - - /** - * There are three sources of data (origins) for a site: - * default, theme, and custom. The custom's has higher priority - * than the theme's, and the theme's higher than defaults's. - * - * Unlike the getters {@link get_core_data}, - * {@link get_theme_data}, and {@link get_user_data}, - * this method returns data after it has been merged - * with the previous origins. This means that if the same piece of data - * is declared in different origins (user, theme, and core), - * the last origin overrides the previous. - * - * For example, if the user has set a background color - * for the paragraph block, and the theme has done it as well, - * the user preference wins. - * - * @param string $origin Optional. To what level should we merge data. - * Valid values are 'theme' or 'custom'. - * Default is 'custom'. - * @return WP_Theme_JSON_Gutenberg - */ - public static function get_merged_data( $origin = 'custom' ) { - if ( is_array( $origin ) ) { - _deprecated_argument( __FUNCTION__, '5.9' ); - } - - $result = new WP_Theme_JSON_Gutenberg(); - $result->merge( static::get_core_data() ); - $result->merge( static::get_theme_data() ); - - if ( 'custom' === $origin ) { - $result->merge( static::get_user_data() ); - } - - return $result; - } -} diff --git a/lib/compat/wordpress-6.0/client-assets.php b/lib/compat/wordpress-6.0/client-assets.php deleted file mode 100644 index f9affb31af4b5..0000000000000 --- a/lib/compat/wordpress-6.0/client-assets.php +++ /dev/null @@ -1,117 +0,0 @@ -get_all_registered() as $block_type ) { - if ( ! empty( $block_type->style ) ) { - $style_handles[] = $block_type->style; - } - - if ( ! empty( $block_type->editor_style ) ) { - $style_handles[] = $block_type->editor_style; - } - - if ( ! empty( $block_type->script ) ) { - $script_handles[] = $block_type->script; - } - } - - $style_handles = array_unique( $style_handles ); - $done = wp_styles()->done; - - ob_start(); - - // We do not need reset styles for the iframed editor. - wp_styles()->done = array( 'wp-reset-editor-styles' ); - wp_styles()->do_items( $style_handles ); - wp_styles()->done = $done; - - $styles = ob_get_clean(); - - $script_handles = array_unique( $script_handles ); - $done = wp_scripts()->done; - - ob_start(); - - wp_scripts()->done = array(); - wp_scripts()->do_items( $script_handles ); - wp_scripts()->done = $done; - - $scripts = ob_get_clean(); - - return array( - 'styles' => $styles, - 'scripts' => $scripts, - ); -} - -add_filter( - 'block_editor_settings_all', - function( $settings ) { - // The `__unstableResolvedAssets` are generated by `_wp_get_iframed_editor_assets` for WP >= 6.0. - if ( function_exists( '_wp_get_iframed_editor_assets' ) ) { - return $settings; - } - - // In the future we can allow WP Dependency handles to be passed. - $settings['__unstableResolvedAssets'] = gutenberg_resolve_assets(); - return $settings; - } -); diff --git a/lib/compat/wordpress-6.0/edit-form-blocks.php b/lib/compat/wordpress-6.0/edit-form-blocks.php deleted file mode 100644 index b11851222b0ba..0000000000000 --- a/lib/compat/wordpress-6.0/edit-form-blocks.php +++ /dev/null @@ -1,66 +0,0 @@ - $user_path ) { - if ( is_string( $user_path ) && str_starts_with( $user_path, '/wp/v2/users/me' ) ) { - $preload_paths[ $user_index ] = '/wp/v2/users/me'; - break; - } - } - - return $preload_paths; -} -add_filter( 'block_editor_rest_api_preload_paths', 'gutenberg_optimize_preload_paths' ); - -/** - * Disables loading remote block patterns from REST while initializing the editor. - * Nowadays these loads are done in the `block-patterns/patterns` REST endpoint, and - * are undesired when initializing the block editor page, both in post and site editor. - * - * @param WP_Screen $current_screen WordPress current screen object. - */ -function gutenberg_disable_load_remote_patterns( $current_screen ) { - $is_site_editor = ( function_exists( 'gutenberg_is_edit_site_page' ) && gutenberg_is_edit_site_page( $current_screen->id ) ); - if ( $is_site_editor || $current_screen->is_block_editor() ) { - add_filter( 'should_load_remote_block_patterns', '__return_false' ); - } -} -add_action( 'current_screen', 'gutenberg_disable_load_remote_patterns' ); diff --git a/lib/compat/wordpress-6.0/functions.php b/lib/compat/wordpress-6.0/functions.php deleted file mode 100644 index 9368b211b0599..0000000000000 --- a/lib/compat/wordpress-6.0/functions.php +++ /dev/null @@ -1,27 +0,0 @@ -get_settings(); - return _wp_array_get( $settings, $path, $settings ); -} - -/** - * Function to get the styles resulting of merging core, theme, and user data. - * - * @param array $path Path to the specific style to retrieve. Optional. - * If empty, will return all styles. - * @param array $context { - * Metadata to know where to retrieve the $path from. Optional. - * - * @type string $block_name Which block to retrieve the styles from. - * If empty, it'll return the styles for the global context. - * @type string $origin Which origin to take data from. - * Valid values are 'all' (core, theme, and user) or 'base' (core and theme). - * If empty or unknown, 'all' is used. - * } - * - * @return array The styles to retrieve. - */ -function gutenberg_get_global_styles( $path = array(), $context = array() ) { - if ( ! empty( $context['block_name'] ) ) { - $path = array_merge( array( 'blocks', $context['block_name'] ), $path ); - } - $origin = 'custom'; - if ( isset( $context['origin'] ) && 'base' === $context['origin'] ) { - $origin = 'theme'; - } - $styles = WP_Theme_JSON_Resolver_Gutenberg::get_merged_data( $origin )->get_raw_data()['styles']; - return _wp_array_get( $styles, $path, $styles ); -} - -/** - * Returns a string containing the SVGs to be referenced as filters (duotone). - * - * @return string - */ -function gutenberg_get_global_styles_svg_filters() { - // Return cached value if it can be used and exists. - // It's cached by theme to make sure that theme switching clears the cache. - $transient_name = 'gutenberg_global_styles_svg_filters_' . get_stylesheet(); - $can_use_cached = ( - ( ! defined( 'WP_DEBUG' ) || ! WP_DEBUG ) && - ( ! defined( 'SCRIPT_DEBUG' ) || ! SCRIPT_DEBUG ) && - ( ! defined( 'REST_REQUEST' ) || ! REST_REQUEST ) && - ! is_admin() - ); - if ( $can_use_cached ) { - $cached = get_transient( $transient_name ); - if ( $cached ) { - return $cached; - } - } - - $supports_theme_json = wp_theme_has_theme_json(); - - $origins = array( 'default', 'theme', 'custom' ); - if ( ! $supports_theme_json ) { - $origins = array( 'default' ); - } - - $tree = WP_Theme_JSON_Resolver_Gutenberg::get_merged_data(); - $svgs = $tree->get_svg_filters( $origins ); - - if ( $can_use_cached ) { - // Cache for a minute, same as gutenberg_get_global_stylesheet. - set_transient( $transient_name, $svgs, MINUTE_IN_SECONDS ); - } - - return $svgs; -} diff --git a/lib/compat/wordpress-6.0/post-lock.php b/lib/compat/wordpress-6.0/post-lock.php deleted file mode 100644 index e36cdff3e7a22..0000000000000 --- a/lib/compat/wordpress-6.0/post-lock.php +++ /dev/null @@ -1,67 +0,0 @@ -display_name; - } - } - - return $response; -} -add_filter( 'heartbeat_received', 'gutenberg_refresh_post_lock', 20, 2 ); - -/** - * Updates post editor settings and adds avatar to the `postLock` user details. - * - * @param array $settings Default editor settings. - * @param WP_Block_Editor_Context $block_editor_context The current block editor context. - * - * @return array Filtered editor settings. - */ -function gutenberg_update_post_lock_details( $settings, $block_editor_context ) { - if ( empty( $block_editor_context->post ) ) { - return $settings; - } - - if ( ! isset( $settings['postLock']['user'] ) ) { - return $settings; - } - - $user_id = wp_check_post_lock( $block_editor_context->post->ID ); - if ( $user_id ) { - $settings['postLock']['user']['avatar'] = get_avatar_url( $user_id, array( 'size' => 128 ) ); - } - - return $settings; -} -add_filter( 'block_editor_settings_all', 'gutenberg_update_post_lock_details', 10, 2 ); diff --git a/lib/compat/wordpress-6.0/render-svg-filters.php b/lib/compat/wordpress-6.0/render-svg-filters.php deleted file mode 100644 index f13502b867fe9..0000000000000 --- a/lib/compat/wordpress-6.0/render-svg-filters.php +++ /dev/null @@ -1,38 +0,0 @@ -is_block_editor() - ) { - return; - } - - $filters = gutenberg_get_global_styles_svg_filters(); - if ( ! empty( $filters ) ) { - echo $filters; - } -} - -// Override actions introduced in 5.9.1 if they exist. -remove_action( 'wp_body_open', 'wp_global_styles_render_svg_filters' ); -remove_action( 'in_admin_header', 'wp_global_styles_render_svg_filters' ); -add_action( 'wp_body_open', 'gutenberg_global_styles_render_svg_filters' ); -add_action( 'in_admin_header', 'gutenberg_global_styles_render_svg_filters' ); diff --git a/lib/compat/wordpress-6.0/rest-api.php b/lib/compat/wordpress-6.0/rest-api.php deleted file mode 100644 index 0fa1852b18813..0000000000000 --- a/lib/compat/wordpress-6.0/rest-api.php +++ /dev/null @@ -1,65 +0,0 @@ -register_routes(); -} -add_action( 'rest_api_init', 'gutenberg_register_global_styles_endpoints' ); - -/** - * Registers the Edit Site's Export REST API routes. - * - * @return void - */ -function gutenberg_register_edit_site_export_endpoint() { - $editor_settings = new Gutenberg_REST_Edit_Site_Export_Controller(); - $editor_settings->register_routes(); -} -add_action( 'rest_api_init', 'gutenberg_register_edit_site_export_endpoint' ); - - -/** - * Register a core site settings. - * - * Note: Needs to be backported into the `register_initial_settings` function. - */ -function gutenberg_register_site_settings() { - register_setting( - 'reading', - 'show_on_front', - array( - 'show_in_rest' => true, - 'type' => 'string', - 'description' => __( 'What to show on the front page', 'gutenberg' ), - ) - ); - - register_setting( - 'reading', - 'page_on_front', - array( - 'show_in_rest' => true, - 'type' => 'number', - 'description' => __( 'The ID of the page that should be displayed on the front page', 'gutenberg' ), - ) - ); - - register_setting( - 'reading', - 'page_for_posts', - array( - 'show_in_rest' => true, - 'type' => 'number', - 'description' => __( 'The ID of the page that should display the latest posts', 'gutenberg' ), - ) - ); -} -add_action( 'rest_api_init', 'gutenberg_register_site_settings', 10 ); diff --git a/lib/compat/wordpress-6.0/site-editor.php b/lib/compat/wordpress-6.0/site-editor.php deleted file mode 100644 index 081657d0061fe..0000000000000 --- a/lib/compat/wordpress-6.0/site-editor.php +++ /dev/null @@ -1,95 +0,0 @@ - 'page', - 'postId' => $front_page_id, - ); - } - - $hierarchy = array( 'front-page', 'home', 'index' ); - $template = resolve_block_template( 'home', $hierarchy, '' ); - - if ( ! $template ) { - return null; - } - - return array( - 'postType' => 'wp_template', - 'postId' => $template->id, - ); -} - -/** - * Do a server-side redirection if missing `postType` and `postId` - * query args when visiting site editor. - * - * Note: This is a backward compatibility redirect for WP 5.9. - * - * @return void - */ -function gutenberg_site_editor_maybe_redirect() { - // Skip redirection for WP 6.0 and later. - if ( function_exists( '_resolve_home_block_template' ) ) { - return; - } - - // Check theme support. The action runs before checks in the Site Editor. - if ( ! wp_is_block_theme() ) { - return; - } - - if ( empty( $_GET['postType'] ) && empty( $_GET['postId'] ) ) { - $template = gutenberg_resolve_home_template(); - if ( ! empty( $_GET['styles'] ) ) { - $template['styles'] = sanitize_key( $_GET['styles'] ); - } - - $redirect_url = add_query_arg( - $template, - admin_url( 'site-editor.php' ) - ); - wp_safe_redirect( $redirect_url ); - exit; - } -} -add_action( 'load-site-editor.php', 'gutenberg_site_editor_maybe_redirect' ); - -/** - * Add home template settings for WP 5.9. - * - * @param array $settings Existing block editor settings. - * @param WP_Block_Editor_Context $context The current block editor context. - * @return array - */ -function gutenberg_site_editor_homepage_setting( $settings, $context ) { - if ( isset( $context->post ) ) { - return $settings; - } - - if ( ! isset( $settings['__unstableHomeTemplate'] ) ) { - $settings['__unstableHomeTemplate'] = gutenberg_resolve_home_template(); - } - - return $settings; -} -add_filter( 'block_editor_settings_all', 'gutenberg_site_editor_homepage_setting', 10, 2 ); diff --git a/lib/compat/wordpress-6.0/theme-i18n.json b/lib/compat/wordpress-6.0/theme-i18n.json deleted file mode 100644 index 282e520c338b4..0000000000000 --- a/lib/compat/wordpress-6.0/theme-i18n.json +++ /dev/null @@ -1,72 +0,0 @@ -{ - "title": "Style variation name", - "settings": { - "typography": { - "fontSizes": [ - { - "name": "Font size name" - } - ], - "fontFamilies": [ - { - "name": "Font family name" - } - ] - }, - "color": { - "palette": [ - { - "name": "Color name" - } - ], - "gradients": [ - { - "name": "Gradient name" - } - ], - "duotone": [ - { - "name": "Duotone name" - } - ] - }, - "blocks": { - "*": { - "typography": { - "fontSizes": [ - { - "name": "Font size name" - } - ], - "fontFamilies": [ - { - "name": "Font family name" - } - ] - }, - "color": { - "palette": [ - { - "name": "Color name" - } - ], - "gradients": [ - { - "name": "Gradient name" - } - ] - } - } - } - }, - "customTemplates": [ - { - "title": "Custom template name" - } - ], - "templateParts": [ - { - "title": "Template part name" - } - ] -} diff --git a/lib/compat/wordpress-6.0/theme.json b/lib/compat/wordpress-6.0/theme.json deleted file mode 100644 index 7691aa4a64e6a..0000000000000 --- a/lib/compat/wordpress-6.0/theme.json +++ /dev/null @@ -1,245 +0,0 @@ -{ - "version": 2, - "settings": { - "appearanceTools": false, - "border": { - "color": false, - "radius": false, - "style": false, - "width": false - }, - "color": { - "background": true, - "custom": true, - "customDuotone": true, - "customGradient": true, - "defaultDuotone": true, - "defaultGradients": true, - "defaultPalette": true, - "duotone": [ - { - "name": "Dark grayscale", - "colors": [ "#000000", "#7f7f7f" ], - "slug": "dark-grayscale" - }, - { - "name": "Grayscale", - "colors": [ "#000000", "#ffffff" ], - "slug": "grayscale" - }, - { - "name": "Purple and yellow", - "colors": [ "#8c00b7", "#fcff41" ], - "slug": "purple-yellow" - }, - { - "name": "Blue and red", - "colors": [ "#000097", "#ff4747" ], - "slug": "blue-red" - }, - { - "name": "Midnight", - "colors": [ "#000000", "#00a5ff" ], - "slug": "midnight" - }, - { - "name": "Magenta and yellow", - "colors": [ "#c7005a", "#fff278" ], - "slug": "magenta-yellow" - }, - { - "name": "Purple and green", - "colors": [ "#a60072", "#67ff66" ], - "slug": "purple-green" - }, - { - "name": "Blue and orange", - "colors": [ "#1900d8", "#ffa96b" ], - "slug": "blue-orange" - } - ], - "gradients": [ - { - "name": "Vivid cyan blue to vivid purple", - "gradient": "linear-gradient(135deg,rgba(6,147,227,1) 0%,rgb(155,81,224) 100%)", - "slug": "vivid-cyan-blue-to-vivid-purple" - }, - { - "name": "Light green cyan to vivid green cyan", - "gradient": "linear-gradient(135deg,rgb(122,220,180) 0%,rgb(0,208,130) 100%)", - "slug": "light-green-cyan-to-vivid-green-cyan" - }, - { - "name": "Luminous vivid amber to luminous vivid orange", - "gradient": "linear-gradient(135deg,rgba(252,185,0,1) 0%,rgba(255,105,0,1) 100%)", - "slug": "luminous-vivid-amber-to-luminous-vivid-orange" - }, - { - "name": "Luminous vivid orange to vivid red", - "gradient": "linear-gradient(135deg,rgba(255,105,0,1) 0%,rgb(207,46,46) 100%)", - "slug": "luminous-vivid-orange-to-vivid-red" - }, - { - "name": "Very light gray to cyan bluish gray", - "gradient": "linear-gradient(135deg,rgb(238,238,238) 0%,rgb(169,184,195) 100%)", - "slug": "very-light-gray-to-cyan-bluish-gray" - }, - { - "name": "Cool to warm spectrum", - "gradient": "linear-gradient(135deg,rgb(74,234,220) 0%,rgb(151,120,209) 20%,rgb(207,42,186) 40%,rgb(238,44,130) 60%,rgb(251,105,98) 80%,rgb(254,248,76) 100%)", - "slug": "cool-to-warm-spectrum" - }, - { - "name": "Blush light purple", - "gradient": "linear-gradient(135deg,rgb(255,206,236) 0%,rgb(152,150,240) 100%)", - "slug": "blush-light-purple" - }, - { - "name": "Blush bordeaux", - "gradient": "linear-gradient(135deg,rgb(254,205,165) 0%,rgb(254,45,45) 50%,rgb(107,0,62) 100%)", - "slug": "blush-bordeaux" - }, - { - "name": "Luminous dusk", - "gradient": "linear-gradient(135deg,rgb(255,203,112) 0%,rgb(199,81,192) 50%,rgb(65,88,208) 100%)", - "slug": "luminous-dusk" - }, - { - "name": "Pale ocean", - "gradient": "linear-gradient(135deg,rgb(255,245,203) 0%,rgb(182,227,212) 50%,rgb(51,167,181) 100%)", - "slug": "pale-ocean" - }, - { - "name": "Electric grass", - "gradient": "linear-gradient(135deg,rgb(202,248,128) 0%,rgb(113,206,126) 100%)", - "slug": "electric-grass" - }, - { - "name": "Midnight", - "gradient": "linear-gradient(135deg,rgb(2,3,129) 0%,rgb(40,116,252) 100%)", - "slug": "midnight" - } - ], - "link": false, - "palette": [ - { - "name": "Black", - "slug": "black", - "color": "#000000" - }, - { - "name": "Cyan bluish gray", - "slug": "cyan-bluish-gray", - "color": "#abb8c3" - }, - { - "name": "White", - "slug": "white", - "color": "#ffffff" - }, - { - "name": "Pale pink", - "slug": "pale-pink", - "color": "#f78da7" - }, - { - "name": "Vivid red", - "slug": "vivid-red", - "color": "#cf2e2e" - }, - { - "name": "Luminous vivid orange", - "slug": "luminous-vivid-orange", - "color": "#ff6900" - }, - { - "name": "Luminous vivid amber", - "slug": "luminous-vivid-amber", - "color": "#fcb900" - }, - { - "name": "Light green cyan", - "slug": "light-green-cyan", - "color": "#7bdcb5" - }, - { - "name": "Vivid green cyan", - "slug": "vivid-green-cyan", - "color": "#00d084" - }, - { - "name": "Pale cyan blue", - "slug": "pale-cyan-blue", - "color": "#8ed1fc" - }, - { - "name": "Vivid cyan blue", - "slug": "vivid-cyan-blue", - "color": "#0693e3" - }, - { - "name": "Vivid purple", - "slug": "vivid-purple", - "color": "#9b51e0" - } - ], - "text": true - }, - "spacing": { - "blockGap": null, - "margin": false, - "padding": false, - "units": [ "px", "em", "rem", "vh", "vw", "%" ] - }, - "typography": { - "customFontSize": true, - "dropCap": true, - "fontSizes": [ - { - "name": "Small", - "slug": "small", - "size": "13px" - }, - { - "name": "Medium", - "slug": "medium", - "size": "20px" - }, - { - "name": "Large", - "slug": "large", - "size": "36px" - }, - { - "name": "Extra Large", - "slug": "x-large", - "size": "42px" - } - ], - "fontStyle": true, - "fontWeight": true, - "letterSpacing": true, - "lineHeight": false, - "textDecoration": true, - "textTransform": true - }, - "blocks": { - "core/button": { - "border": { - "radius": true - } - }, - "core/pullquote": { - "border": { - "color": true, - "radius": true, - "style": true, - "width": true - } - } - } - }, - "styles": { - "spacing": { "blockGap": "24px" } - } -} diff --git a/lib/compat/wordpress-6.1/block-patterns.php b/lib/compat/wordpress-6.1/block-patterns.php deleted file mode 100644 index 7860aa6a1dd66..0000000000000 --- a/lib/compat/wordpress-6.1/block-patterns.php +++ /dev/null @@ -1,177 +0,0 @@ -

- * - * If applicable, this will collect from both parent and child theme. - * - * Other settable fields include: - * - * - Description - * - Viewport Width - * - Categories (comma-separated values) - * - Keywords (comma-separated values) - * - Block Types (comma-separated values) - * - Post Types (comma-separated values) - * - Inserter (yes/no) - * - * @since 6.0.0 - * @access private - */ -function gutenberg_register_theme_block_patterns() { - $default_headers = array( - 'title' => 'Title', - 'slug' => 'Slug', - 'description' => 'Description', - 'viewportWidth' => 'Viewport Width', - 'categories' => 'Categories', - 'keywords' => 'Keywords', - 'blockTypes' => 'Block Types', - 'postTypes' => 'Post Types', - 'inserter' => 'Inserter', - ); - - /* - * Register patterns for the active theme. If the theme is a child theme, - * let it override any patterns from the parent theme that shares the same slug. - */ - $themes = array(); - $stylesheet = get_stylesheet(); - $template = get_template(); - if ( $stylesheet !== $template ) { - $themes[] = wp_get_theme( $stylesheet ); - } - $themes[] = wp_get_theme( $template ); - - foreach ( $themes as $theme ) { - $dirpath = $theme->get_stylesheet_directory() . '/patterns/'; - if ( ! is_dir( $dirpath ) || ! is_readable( $dirpath ) ) { - continue; - } - if ( file_exists( $dirpath ) ) { - $files = glob( $dirpath . '*.php' ); - if ( $files ) { - foreach ( $files as $file ) { - $pattern_data = get_file_data( $file, $default_headers ); - - if ( empty( $pattern_data['slug'] ) ) { - _doing_it_wrong( - '_register_theme_block_patterns', - sprintf( - /* translators: %s: file name. */ - __( 'Could not register file "%s" as a block pattern ("Slug" field missing)', 'gutenberg' ), - $file - ), - '6.0.0' - ); - continue; - } - - if ( ! preg_match( '/^[A-z0-9\/_-]+$/', $pattern_data['slug'] ) ) { - _doing_it_wrong( - '_register_theme_block_patterns', - sprintf( - /* translators: %1s: file name; %2s: slug value found. */ - __( 'Could not register file "%1$s" as a block pattern (invalid slug "%2$s")', 'gutenberg' ), - $file, - $pattern_data['slug'] - ), - '6.0.0' - ); - } - - if ( WP_Block_Patterns_Registry::get_instance()->is_registered( $pattern_data['slug'] ) ) { - continue; - } - - // Title is a required property. - if ( ! $pattern_data['title'] ) { - _doing_it_wrong( - '_register_theme_block_patterns', - sprintf( - /* translators: %1s: file name; %2s: slug value found. */ - __( 'Could not register file "%s" as a block pattern ("Title" field missing)', 'gutenberg' ), - $file - ), - '6.0.0' - ); - continue; - } - - // For properties of type array, parse data as comma-separated. - foreach ( array( 'categories', 'keywords', 'blockTypes', 'postTypes' ) as $property ) { - if ( ! empty( $pattern_data[ $property ] ) ) { - $pattern_data[ $property ] = array_filter( - preg_split( - '/[\s,]+/', - (string) $pattern_data[ $property ] - ) - ); - } else { - unset( $pattern_data[ $property ] ); - } - } - - // Parse properties of type int. - foreach ( array( 'viewportWidth' ) as $property ) { - if ( ! empty( $pattern_data[ $property ] ) ) { - $pattern_data[ $property ] = (int) $pattern_data[ $property ]; - } else { - unset( $pattern_data[ $property ] ); - } - } - - // Parse properties of type bool. - foreach ( array( 'inserter' ) as $property ) { - if ( ! empty( $pattern_data[ $property ] ) ) { - $pattern_data[ $property ] = in_array( - strtolower( $pattern_data[ $property ] ), - array( 'yes', 'true' ), - true - ); - } else { - unset( $pattern_data[ $property ] ); - } - } - - // Translate the pattern metadata. - $text_domain = $theme->get( 'TextDomain' ); - //phpcs:ignore WordPress.WP.I18n.NonSingularStringLiteralText, WordPress.WP.I18n.NonSingularStringLiteralContext, WordPress.WP.I18n.NonSingularStringLiteralDomain, WordPress.WP.I18n.LowLevelTranslationFunction - $pattern_data['title'] = translate_with_gettext_context( $pattern_data['title'], 'Pattern title', $text_domain ); - if ( ! empty( $pattern_data['description'] ) ) { - //phpcs:ignore WordPress.WP.I18n.NonSingularStringLiteralText, WordPress.WP.I18n.NonSingularStringLiteralContext, WordPress.WP.I18n.NonSingularStringLiteralDomain, WordPress.WP.I18n.LowLevelTranslationFunction - $pattern_data['description'] = translate_with_gettext_context( $pattern_data['description'], 'Pattern description', $text_domain ); - } - - // The actual pattern content is the output of the file. - ob_start(); - include $file; - $pattern_data['content'] = ob_get_clean(); - if ( ! $pattern_data['content'] ) { - continue; - } - - register_block_pattern( $pattern_data['slug'], $pattern_data ); - } - } - } - } -} -remove_action( 'init', '_register_theme_block_patterns' ); -add_action( 'init', 'gutenberg_register_theme_block_patterns' ); diff --git a/lib/compat/wordpress-6.1/block-template-utils.php b/lib/compat/wordpress-6.1/block-template-utils.php index 088333990a0d5..928291b500084 100644 --- a/lib/compat/wordpress-6.1/block-template-utils.php +++ b/lib/compat/wordpress-6.1/block-template-utils.php @@ -84,7 +84,7 @@ function gutenberg_get_block_templates( $query = array(), $template_type = 'wp_t array( 'taxonomy' => 'wp_theme', 'field' => 'name', - 'terms' => wp_get_theme()->get_stylesheet(), + 'terms' => get_stylesheet(), ), ), ); @@ -429,7 +429,7 @@ function gutenberg_build_block_template_result_from_post( $post ) { $theme = $terms[0]->name; $template_file = _get_block_template_file( $post->post_type, $post->post_name ); - $has_theme_file = wp_get_theme()->get_stylesheet() === $theme && null !== $template_file; + $has_theme_file = get_stylesheet() === $theme && null !== $template_file; $template = new WP_Block_Template(); $template->wp_id = $post->ID; diff --git a/lib/compat/wordpress-6.1/blocks.php b/lib/compat/wordpress-6.1/blocks.php index cd61b70a07f09..3aa790dadcb1c 100644 --- a/lib/compat/wordpress-6.1/blocks.php +++ b/lib/compat/wordpress-6.1/blocks.php @@ -71,12 +71,12 @@ function gutenberg_block_type_metadata_view_script( $settings, $metadata ) { ! isset( $metadata['viewScript'] ) || ! empty( $settings['view_script'] ) || ! isset( $metadata['file'] ) || - ! str_starts_with( $metadata['file'], gutenberg_dir_path() ) + ! str_starts_with( $metadata['file'], wp_normalize_path( gutenberg_dir_path() ) ) ) { return $settings; } - $view_script_path = realpath( dirname( $metadata['file'] ) . '/' . remove_block_asset_path_prefix( $metadata['viewScript'] ) ); + $view_script_path = wp_normalize_path( realpath( dirname( $metadata['file'] ) . '/' . remove_block_asset_path_prefix( $metadata['viewScript'] ) ) ); if ( file_exists( $view_script_path ) ) { $view_script_id = str_replace( array( '.min.js', '.js' ), '', basename( remove_block_asset_path_prefix( $metadata['viewScript'] ) ) ); @@ -90,7 +90,7 @@ function gutenberg_block_type_metadata_view_script( $settings, $metadata ) { $view_script_version = isset( $view_asset['version'] ) ? $view_asset['version'] : false; $result = wp_register_script( $view_script_handle, - gutenberg_url( str_replace( gutenberg_dir_path(), '', $view_script_path ) ), + gutenberg_url( str_replace( wp_normalize_path( gutenberg_dir_path() ), '', $view_script_path ) ), $view_script_dependencies, $view_script_version ); diff --git a/lib/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller.php b/lib/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller-6-1.php similarity index 60% rename from lib/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller.php rename to lib/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller-6-1.php index c053a678083da..b40f3aa2497f1 100644 --- a/lib/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller.php +++ b/lib/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller-6-1.php @@ -1,6 +1,6 @@ namespace = 'wp/v2'; - $this->rest_base = 'block-patterns/patterns'; - } - - /** - * Registers the routes for the objects of the controller. - * - * @since 6.0.0 - */ - public function register_routes() { - register_rest_route( - $this->namespace, - '/' . $this->rest_base, - array( - array( - 'methods' => WP_REST_Server::READABLE, - 'callback' => array( $this, 'get_items' ), - 'permission_callback' => array( $this, 'get_items_permissions_check' ), - ), - 'schema' => array( $this, 'get_public_item_schema' ), - ), - true - ); - } - - /** - * Checks whether a given request has permission to read block patterns. - * - * @since 6.0.0 - * - * @param WP_REST_Request $request Full details about the request. - * @return true|WP_Error True if the request has read access, WP_Error object otherwise. - */ - public function get_items_permissions_check( $request ) { // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable - if ( current_user_can( 'edit_posts' ) ) { - return true; - } - - foreach ( get_post_types( array( 'show_in_rest' => true ), 'objects' ) as $post_type ) { - if ( current_user_can( $post_type->cap->edit_posts ) ) { - return true; - } - } - - return new WP_Error( - 'rest_cannot_view', - __( 'Sorry, you are not allowed to view the registered block patterns.', 'gutenberg' ), - array( 'status' => rest_authorization_required_code() ) - ); - } - - /** - * Retrieves all block patterns. - * - * @since 6.0.0 - * - * @param WP_REST_Request $request Full details about the request. - * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. - */ - public function get_items( $request ) { - if ( ! $this->remote_patterns_loaded ) { - // Load block patterns from w.org. - _load_remote_block_patterns(); // Patterns with the `core` keyword. - _load_remote_featured_patterns(); // Patterns in the `featured` category. - _register_remote_theme_patterns(); // Patterns requested by current theme. - - $this->remote_patterns_loaded = true; - } - - $response = array(); - $patterns = WP_Block_Patterns_Registry::get_instance()->get_all_registered(); - foreach ( $patterns as $pattern ) { - $prepared_pattern = $this->prepare_item_for_response( $pattern, $request ); - $response[] = $this->prepare_response_for_collection( $prepared_pattern ); - } - return rest_ensure_response( $response ); - } - +class Gutenberg_REST_Block_Patterns_Controller_6_1 extends WP_REST_Block_Patterns_Controller { /** * Prepare a raw block pattern before it gets output in a REST API response. * * @since 6.0.0 + * @since 6.1.0 Added `postTypes` property. * * @param array $item Raw pattern as registered, before any changes. * @param WP_REST_Request $request Request object. @@ -147,6 +55,7 @@ public function prepare_item_for_response( $item, $request ) { * Retrieves the block pattern schema, conforming to JSON Schema. * * @since 6.0.0 + * @since 6.1.0 Added `post_types` property. * * @return array Item schema data. */ diff --git a/lib/compat/wordpress-6.1/class-gutenberg-rest-templates-controller.php b/lib/compat/wordpress-6.1/class-gutenberg-rest-templates-controller.php index 879316c1c649e..e911b0f1e4d20 100644 --- a/lib/compat/wordpress-6.1/class-gutenberg-rest-templates-controller.php +++ b/lib/compat/wordpress-6.1/class-gutenberg-rest-templates-controller.php @@ -228,7 +228,7 @@ protected function prepare_item_for_database( $request ) { $changes->post_type = $this->post_type; $changes->post_status = 'publish'; $changes->tax_input = array( - 'wp_theme' => isset( $request['theme'] ) ? $request['theme'] : wp_get_theme()->get_stylesheet(), + 'wp_theme' => isset( $request['theme'] ) ? $request['theme'] : get_stylesheet(), ); } elseif ( 'custom' !== $template->source ) { $changes->post_name = $template->slug; diff --git a/lib/compat/wordpress-6.1/class-wp-theme-json-6-1.php b/lib/compat/wordpress-6.1/class-wp-theme-json-6-1.php index e965e0e824677..0c0ec0451aec4 100644 --- a/lib/compat/wordpress-6.1/class-wp-theme-json-6-1.php +++ b/lib/compat/wordpress-6.1/class-wp-theme-json-6-1.php @@ -14,7 +14,7 @@ * * @access private */ -class WP_Theme_JSON_6_1 extends WP_Theme_JSON_6_0 { +class WP_Theme_JSON_6_1 extends WP_Theme_JSON { /** * Defines which pseudo selectors are enabled for which elements. * diff --git a/lib/compat/wordpress-6.1/class-wp-theme-json-resolver-6-1.php b/lib/compat/wordpress-6.1/class-wp-theme-json-resolver-6-1.php index 73e012f33d1c7..1d4a83e3406fb 100644 --- a/lib/compat/wordpress-6.1/class-wp-theme-json-resolver-6-1.php +++ b/lib/compat/wordpress-6.1/class-wp-theme-json-resolver-6-1.php @@ -15,7 +15,7 @@ * * @access private */ -class WP_Theme_JSON_Resolver_6_1 extends WP_Theme_JSON_Resolver_6_0 { +class WP_Theme_JSON_Resolver_6_1 extends WP_Theme_JSON_Resolver { /** * Container for data coming from core. diff --git a/lib/compat/wordpress-6.1/get-global-styles-and-settings.php b/lib/compat/wordpress-6.1/get-global-styles-and-settings.php index 35c540ce1c57a..0829a09d084c7 100644 --- a/lib/compat/wordpress-6.1/get-global-styles-and-settings.php +++ b/lib/compat/wordpress-6.1/get-global-styles-and-settings.php @@ -56,86 +56,36 @@ function ( $item ) { } /** - * Returns the stylesheet resulting of merging core, theme, and user data. + * Repeated logic from `get_default_block_editor_settings` function. When implemented into core, + * remove logic from `get_default_block_editor_settings` and simple call this function instead. * - * @param array $types Types of styles to load. Optional. - * It accepts 'variables', 'styles', 'presets' as values. - * If empty, it'll load all for themes with theme.json support - * and only [ 'variables', 'presets' ] for themes without theme.json support. - * - * @return string Stylesheet. + * @return array */ -function gutenberg_get_global_stylesheet( $types = array() ) { - // Return cached value if it can be used and exists. - // It's cached by theme to make sure that theme switching clears the cache. - $can_use_cached = ( - ( empty( $types ) ) && - ( ! defined( 'WP_DEBUG' ) || ! WP_DEBUG ) && - ( ! defined( 'SCRIPT_DEBUG' ) || ! SCRIPT_DEBUG ) && - ( ! defined( 'REST_REQUEST' ) || ! REST_REQUEST ) && - ! is_admin() +function gutenberg_get_legacy_theme_supports_for_theme_json() { + $theme_settings = array( + 'disableCustomColors' => get_theme_support( 'disable-custom-colors' ), + 'disableCustomFontSizes' => get_theme_support( 'disable-custom-font-sizes' ), + 'disableCustomGradients' => get_theme_support( 'disable-custom-gradients' ), + 'enableCustomLineHeight' => get_theme_support( 'custom-line-height' ), + 'enableCustomSpacing' => get_theme_support( 'custom-spacing' ), + 'enableCustomUnits' => get_theme_support( 'custom-units' ), ); - $transient_name = 'gutenberg_global_styles_' . get_stylesheet(); - if ( $can_use_cached ) { - $cached = get_transient( $transient_name ); - if ( $cached ) { - return $cached; - } - } - $tree = WP_Theme_JSON_Resolver_Gutenberg::get_merged_data(); - $supports_theme_json = wp_theme_has_theme_json(); - if ( empty( $types ) && ! $supports_theme_json ) { - $types = array( 'variables', 'presets', 'base-layout-styles' ); - } elseif ( empty( $types ) ) { - $types = array( 'variables', 'styles', 'presets' ); - } - /* - * If variables are part of the stylesheet, - * we add them. - * - * This is so themes without a theme.json still work as before 5.9: - * they can override the default presets. - * See https://core.trac.wordpress.org/ticket/54782 - */ - $styles_variables = ''; - if ( in_array( 'variables', $types, true ) ) { - /* - * We only use the default, theme, and custom origins. - * This is because styles for blocks origin are added - * at a later phase (render cycle) so we only render the ones in use. - * @see wp_add_global_styles_for_blocks - */ - $origins = array( 'default', 'theme', 'custom' ); - $styles_variables = $tree->get_stylesheet( array( 'variables' ), $origins ); - $types = array_diff( $types, array( 'variables' ) ); + // Theme settings. + $color_palette = current( (array) get_theme_support( 'editor-color-palette' ) ); + if ( false !== $color_palette ) { + $theme_settings['colors'] = $color_palette; } - /* - * For the remaining types (presets, styles), we do consider origins: - * - * - themes without theme.json: only the classes for the presets defined by core - * - themes with theme.json: the presets and styles classes, both from core and the theme - */ - $styles_rest = ''; - if ( ! empty( $types ) ) { - /* - * We only use the default, theme, and custom origins. - * This is because styles for blocks origin are added - * at a later phase (render cycle) so we only render the ones in use. - * @see wp_add_global_styles_for_blocks - */ - $origins = array( 'default', 'theme', 'custom' ); - if ( ! $supports_theme_json ) { - $origins = array( 'default' ); - } - $styles_rest = $tree->get_stylesheet( $types, $origins ); + $font_sizes = current( (array) get_theme_support( 'editor-font-sizes' ) ); + if ( false !== $font_sizes ) { + $theme_settings['fontSizes'] = $font_sizes; } - $stylesheet = $styles_variables . $styles_rest; - if ( $can_use_cached ) { - // Cache for a minute. - // This cache doesn't need to be any longer, we only want to avoid spikes on high-traffic sites. - set_transient( $transient_name, $stylesheet, MINUTE_IN_SECONDS ); + + $gradient_presets = current( (array) get_theme_support( 'editor-gradient-presets' ) ); + if ( false !== $gradient_presets ) { + $theme_settings['gradients'] = $gradient_presets; } - return $stylesheet; + + return $theme_settings; } diff --git a/lib/compat/wordpress-6.1/rest-api.php b/lib/compat/wordpress-6.1/rest-api.php index f3391389d9e38..5dd06d3aad855 100644 --- a/lib/compat/wordpress-6.1/rest-api.php +++ b/lib/compat/wordpress-6.1/rest-api.php @@ -35,15 +35,6 @@ function gutenberg_update_post_types_rest_response( $response, $post_type ) { } add_filter( 'rest_prepare_post_type', 'gutenberg_update_post_types_rest_response', 10, 2 ); -/** - * Registers the block patterns REST API routes. - */ -function gutenberg_register_gutenberg_rest_block_patterns() { - $block_patterns = new Gutenberg_REST_Block_Patterns_Controller(); - $block_patterns->register_routes(); -} -add_action( 'rest_api_init', 'gutenberg_register_gutenberg_rest_block_patterns', 100 ); - /** * Exposes the site logo URL through the WordPress REST API. * diff --git a/lib/compat/wordpress-6.1/template-parts-screen.php b/lib/compat/wordpress-6.1/template-parts-screen.php index c8b5958bedcd4..1555b6ab1e250 100644 --- a/lib/compat/wordpress-6.1/template-parts-screen.php +++ b/lib/compat/wordpress-6.1/template-parts-screen.php @@ -118,17 +118,8 @@ static function( $classes ) { 'defaultTemplatePartAreas' => get_allowed_block_template_part_areas(), 'supportsLayout' => wp_theme_has_theme_json(), 'supportsTemplatePartsMode' => ! wp_is_block_theme() && current_theme_supports( 'block-template-parts' ), - '__unstableHomeTemplate' => gutenberg_resolve_home_template(), ); - /** - * We don't need home template resolution when block template parts are supported. - * Set the value to true to satisfy the editor initialization guard clause. - */ - if ( $custom_settings['supportsTemplatePartsMode'] ) { - $custom_settings['__unstableHomeTemplate'] = true; - } - // Add additional back-compat patterns registered by `current_screen` et al. $custom_settings['__experimentalAdditionalBlockPatterns'] = WP_Block_Patterns_Registry::get_instance()->get_all_registered( true ); $custom_settings['__experimentalAdditionalBlockPatternCategories'] = WP_Block_Pattern_Categories_Registry::get_instance()->get_all_registered( true ); @@ -143,7 +134,7 @@ static function( $classes ) { } $active_global_styles_id = WP_Theme_JSON_Resolver::get_user_global_styles_post_id(); - $active_theme = wp_get_theme()->get_stylesheet(); + $active_theme = get_stylesheet(); $preload_paths = array( array( '/wp/v2/media', 'OPTIONS' ), '/wp/v2/types?context=view', diff --git a/lib/compat/wordpress-6.2/block-editor-settings.php b/lib/compat/wordpress-6.2/block-editor-settings.php new file mode 100644 index 0000000000000..7323d34eb667c --- /dev/null +++ b/lib/compat/wordpress-6.2/block-editor-settings.php @@ -0,0 +1,28 @@ + gutenberg_get_global_stylesheet( array( 'custom-css' ) ), + '__unstableType' => 'user', + 'isGlobalStyles' => true, + ); + } + + return $settings; +} + +add_filter( 'block_editor_settings_all', 'gutenberg_get_block_editor_settings_6_2', PHP_INT_MAX ); diff --git a/lib/compat/wordpress-6.2/block-patterns.php b/lib/compat/wordpress-6.2/block-patterns.php index a28566d441e7a..1e529faf96321 100644 --- a/lib/compat/wordpress-6.2/block-patterns.php +++ b/lib/compat/wordpress-6.2/block-patterns.php @@ -6,9 +6,9 @@ */ /** - * Registers the block pattern categories REST API routes. + * Registers the block pattern categories. */ -function gutenberg_register_core_block_patterns_and_categories() { +function gutenberg_register_core_block_patterns_categories() { register_block_pattern_category( 'banner', array( @@ -18,54 +18,127 @@ function gutenberg_register_core_block_patterns_and_categories() { register_block_pattern_category( 'buttons', array( - 'label' => _x( 'Buttons', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Buttons', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Patterns that contain buttons and call to actions.', 'gutenberg' ), ) ); register_block_pattern_category( 'columns', array( - 'label' => _x( 'Columns', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Columns', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Multi-column patterns with more complex layouts.', 'gutenberg' ), ) ); register_block_pattern_category( - 'footer', + 'text', array( - 'label' => _x( 'Footers', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Text', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Patterns containing mostly text.', 'gutenberg' ), ) ); register_block_pattern_category( - 'gallery', + 'query', array( - 'label' => _x( 'Gallery', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Posts', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Display your latest posts in lists, grids or other layouts.', 'gutenberg' ), ) ); register_block_pattern_category( - 'header', + 'featured', array( - 'label' => _x( 'Headers', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Featured', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'A set of high quality curated patterns.', 'gutenberg' ), ) ); + + // Register new core block pattern categories. register_block_pattern_category( - 'text', + 'call-to-action', array( - 'label' => _x( 'Text', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Call to Action', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Sections whose purpose is to trigger a specific action.', 'gutenberg' ), ) ); register_block_pattern_category( - 'query', + 'team', + array( + 'label' => _x( 'Team', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'A variety of designs to display your team members.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'testimonials', + array( + 'label' => _x( 'Testimonials', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Share reviews and feedback about your brand/business.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'services', + array( + 'label' => _x( 'Services', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Briefly describe what your business does and how you can help.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'contact', + array( + 'label' => _x( 'Contact', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Display your contact information.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'about', + array( + 'label' => _x( 'About', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Introduce yourself.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'portfolio', + array( + 'label' => _x( 'Portfolio', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Showcase your latest work.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'gallery', + array( + 'label' => _x( 'Gallery', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Different layouts for displaying images.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'media', + array( + 'label' => _x( 'Media', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'Different layouts containing video or audio.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'posts', array( 'label' => _x( 'Posts', 'Block pattern category', 'gutenberg' ), - 'description' => __( 'Display post summaries in lists, grids, and other layouts.', 'gutenberg' ), + 'description' => __( 'Display your latest posts in lists, grids or other layouts.', 'gutenberg' ), ) ); + // Site building pattern categories. register_block_pattern_category( - 'featured', + 'footer', array( - 'label' => _x( 'Featured', 'Block pattern category', 'gutenberg' ), + 'label' => _x( 'Footers', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'A variety of footer designs displaying information and site navigation.', 'gutenberg' ), + ) + ); + register_block_pattern_category( + 'header', + array( + 'label' => _x( 'Headers', 'Block pattern category', 'gutenberg' ), + 'description' => __( 'A variety of header designs displaying your site title and navigation.', 'gutenberg' ), ) ); } -add_action( 'init', 'gutenberg_register_core_block_patterns_and_categories' ); +add_action( 'init', 'gutenberg_register_core_block_patterns_categories' ); /** * Registers Gutenberg-bundled patterns, with a focus on headers and footers @@ -111,3 +184,173 @@ function gutenberg_register_core_block_patterns() { } } add_action( 'init', 'gutenberg_register_core_block_patterns' ); + +/** + * Register any patterns that the active theme may provide under its + * `./patterns/` directory. Each pattern is defined as a PHP file and defines + * its metadata using plugin-style headers. The minimum required definition is: + * + * /** + * * Title: My Pattern + * * Slug: my-theme/my-pattern + * * + * + * The output of the PHP source corresponds to the content of the pattern, e.g.: + * + *

+ * + * If applicable, this will collect from both parent and child theme. + * + * Other settable fields include: + * + * - Description + * - Viewport Width + * - Categories (comma-separated values) + * - Keywords (comma-separated values) + * - Block Types (comma-separated values) + * - Post Types (comma-separated values) + * - Inserter (yes/no) + * + * @since 6.0.0 + * @access private + */ +function gutenberg_register_theme_block_patterns() { + $default_headers = array( + 'title' => 'Title', + 'slug' => 'Slug', + 'description' => 'Description', + 'viewportWidth' => 'Viewport Width', + 'categories' => 'Categories', + 'keywords' => 'Keywords', + 'blockTypes' => 'Block Types', + 'postTypes' => 'Post Types', + 'inserter' => 'Inserter', + ); + + /* + * Register patterns for the active theme. If the theme is a child theme, + * let it override any patterns from the parent theme that shares the same slug. + */ + $themes = array(); + $wp_theme = wp_get_theme(); + if ( $wp_theme->parent() ) { + $themes[] = $wp_theme->parent(); + } + $themes[] = $wp_theme; + + foreach ( $themes as $theme ) { + $dirpath = $theme->get_stylesheet_directory() . '/patterns/'; + if ( ! is_dir( $dirpath ) || ! is_readable( $dirpath ) ) { + continue; + } + if ( file_exists( $dirpath ) ) { + $files = glob( $dirpath . '*.php' ); + if ( $files ) { + foreach ( $files as $file ) { + $pattern_data = get_file_data( $file, $default_headers ); + + if ( empty( $pattern_data['slug'] ) ) { + _doing_it_wrong( + '_register_theme_block_patterns', + sprintf( + /* translators: %s: file name. */ + __( 'Could not register file "%s" as a block pattern ("Slug" field missing)', 'gutenberg' ), + $file + ), + '6.0.0' + ); + continue; + } + + if ( ! preg_match( '/^[A-z0-9\/_-]+$/', $pattern_data['slug'] ) ) { + _doing_it_wrong( + '_register_theme_block_patterns', + sprintf( + /* translators: %1s: file name; %2s: slug value found. */ + __( 'Could not register file "%1$s" as a block pattern (invalid slug "%2$s")', 'gutenberg' ), + $file, + $pattern_data['slug'] + ), + '6.0.0' + ); + } + + if ( WP_Block_Patterns_Registry::get_instance()->is_registered( $pattern_data['slug'] ) ) { + continue; + } + + // Title is a required property. + if ( ! $pattern_data['title'] ) { + _doing_it_wrong( + '_register_theme_block_patterns', + sprintf( + /* translators: %1s: file name; %2s: slug value found. */ + __( 'Could not register file "%s" as a block pattern ("Title" field missing)', 'gutenberg' ), + $file + ), + '6.0.0' + ); + continue; + } + + // For properties of type array, parse data as comma-separated. + foreach ( array( 'categories', 'keywords', 'blockTypes', 'postTypes' ) as $property ) { + if ( ! empty( $pattern_data[ $property ] ) ) { + $pattern_data[ $property ] = array_filter( + preg_split( + '/[\s,]+/', + (string) $pattern_data[ $property ] + ) + ); + } else { + unset( $pattern_data[ $property ] ); + } + } + + // Parse properties of type int. + foreach ( array( 'viewportWidth' ) as $property ) { + if ( ! empty( $pattern_data[ $property ] ) ) { + $pattern_data[ $property ] = (int) $pattern_data[ $property ]; + } else { + unset( $pattern_data[ $property ] ); + } + } + + // Parse properties of type bool. + foreach ( array( 'inserter' ) as $property ) { + if ( ! empty( $pattern_data[ $property ] ) ) { + $pattern_data[ $property ] = in_array( + strtolower( $pattern_data[ $property ] ), + array( 'yes', 'true' ), + true + ); + } else { + unset( $pattern_data[ $property ] ); + } + } + + // Translate the pattern metadata. + $text_domain = $theme->get( 'TextDomain' ); + //phpcs:ignore WordPress.WP.I18n.NonSingularStringLiteralText, WordPress.WP.I18n.NonSingularStringLiteralContext, WordPress.WP.I18n.NonSingularStringLiteralDomain, WordPress.WP.I18n.LowLevelTranslationFunction + $pattern_data['title'] = translate_with_gettext_context( $pattern_data['title'], 'Pattern title', $text_domain ); + if ( ! empty( $pattern_data['description'] ) ) { + //phpcs:ignore WordPress.WP.I18n.NonSingularStringLiteralText, WordPress.WP.I18n.NonSingularStringLiteralContext, WordPress.WP.I18n.NonSingularStringLiteralDomain, WordPress.WP.I18n.LowLevelTranslationFunction + $pattern_data['description'] = translate_with_gettext_context( $pattern_data['description'], 'Pattern description', $text_domain ); + } + + // The actual pattern content is the output of the file. + ob_start(); + include $file; + $pattern_data['content'] = ob_get_clean(); + if ( ! $pattern_data['content'] ) { + continue; + } + + register_block_pattern( $pattern_data['slug'], $pattern_data ); + } + } + } + } +} +remove_action( 'init', '_register_theme_block_patterns' ); +add_action( 'init', 'gutenberg_register_theme_block_patterns' ); diff --git a/lib/compat/wordpress-6.2/class-gutenberg-rest-block-patterns-controller-6-2.php b/lib/compat/wordpress-6.2/class-gutenberg-rest-block-patterns-controller-6-2.php new file mode 100644 index 0000000000000..d34160edb2af0 --- /dev/null +++ b/lib/compat/wordpress-6.2/class-gutenberg-rest-block-patterns-controller-6-2.php @@ -0,0 +1,107 @@ + 'call-to-action', + 'columns' => 'text', + 'query' => 'posts', + ); + + /** + * Registers the routes for the objects of the controller. + * + * @since 6.0.0 + */ + public function register_routes() { + register_rest_route( + $this->namespace, + '/' . $this->rest_base, + array( + array( + 'methods' => WP_REST_Server::READABLE, + 'callback' => array( $this, 'get_items' ), + 'permission_callback' => array( $this, 'get_items_permissions_check' ), + ), + 'schema' => array( $this, 'get_public_item_schema' ), + ), + true + ); + } + /** + * Retrieves all block patterns. + * + * @since 6.0.0 + * @since 6.2.0 Added migration for old core pattern categories to the new ones. + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. + */ + public function get_items( $request ) { + if ( ! $this->remote_patterns_loaded ) { + // Load block patterns from w.org. + _load_remote_block_patterns(); // Patterns with the `core` keyword. + _load_remote_featured_patterns(); // Patterns in the `featured` category. + _register_remote_theme_patterns(); // Patterns requested by current theme. + + $this->remote_patterns_loaded = true; + } + + $response = array(); + $patterns = WP_Block_Patterns_Registry::get_instance()->get_all_registered(); + foreach ( $patterns as $pattern ) { + $migrated_pattern = $this->migrate_pattern_categories( $pattern ); + $prepared_pattern = $this->prepare_item_for_response( $migrated_pattern, $request ); + $response[] = $this->prepare_response_for_collection( $prepared_pattern ); + } + return rest_ensure_response( $response ); + } + + /** + * Migrates old core pattern categories to new ones. + * + * Core pattern categories are being revamped and we need to handle the migration + * to the new ones and ensure backwards compatibility. + * + * @since 6.2.0 + * + * @param array $pattern Raw pattern as registered, before applying any changes. + * @return array Migrated pattern. + */ + protected function migrate_pattern_categories( $pattern ) { + if ( isset( $pattern['categories'] ) && is_array( $pattern['categories'] ) ) { + foreach ( $pattern['categories'] as $i => $category ) { + if ( array_key_exists( $category, static::$categories_migration ) ) { + $pattern['categories'][ $i ] = static::$categories_migration[ $category ]; + } + } + } + return $pattern; + } +} diff --git a/lib/compat/wordpress-6.0/class-gutenberg-rest-global-styles-controller.php b/lib/compat/wordpress-6.2/class-gutenberg-rest-global-styles-controller-6-2.php similarity index 53% rename from lib/compat/wordpress-6.0/class-gutenberg-rest-global-styles-controller.php rename to lib/compat/wordpress-6.2/class-gutenberg-rest-global-styles-controller-6-2.php index 2a50e9684c093..9f762dd961d05 100644 --- a/lib/compat/wordpress-6.0/class-gutenberg-rest-global-styles-controller.php +++ b/lib/compat/wordpress-6.2/class-gutenberg-rest-global-styles-controller-6-2.php @@ -9,111 +9,23 @@ /** * Base Global Styles REST API Controller. */ -class Gutenberg_REST_Global_Styles_Controller extends WP_REST_Global_Styles_Controller { +class Gutenberg_REST_Global_Styles_Controller_6_2 extends WP_REST_Global_Styles_Controller { /** * Registers the controllers routes. * * @return void */ public function register_routes() { - // List themes global styles. - register_rest_route( - $this->namespace, - '/' . $this->rest_base . '/themes/(?P[\/\s%\w\.\(\)\[\]\@_\-]+)/variations', - array( - array( - 'methods' => WP_REST_Server::READABLE, - 'callback' => array( $this, 'get_theme_items' ), - 'permission_callback' => array( $this, 'get_theme_items_permissions_check' ), - 'args' => array( - 'stylesheet' => array( - 'description' => __( 'The theme identifier', 'gutenberg' ), - 'type' => 'string', - ), - ), - ), - ) - ); - parent::register_routes(); } - /** - * Checks if a given request has access to read a single theme global styles config. - * - * @param WP_REST_Request $request Full details about the request. - * @return true|WP_Error True if the request has read access for the item, WP_Error object otherwise. - */ - public function get_theme_items_permissions_check( $request ) { // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable - // Verify if the current user has edit_theme_options capability. - // This capability is required to edit/view/delete templates. - if ( ! current_user_can( 'edit_theme_options' ) ) { - return new WP_Error( - 'rest_cannot_manage_global_styles', - __( 'Sorry, you are not allowed to access the global styles on this site.', 'gutenberg' ), - array( - 'status' => rest_authorization_required_code(), - ) - ); - } - - return true; - } - - /** - * Prepares a single global styles config for update. - * - * @since 5.9.0 - * - * @param WP_REST_Request $request Request object. - * @return stdClass Changes to pass to wp_update_post. - */ - protected function prepare_item_for_database( $request ) { - $changes = new stdClass(); - $changes->ID = $request['id']; - $post = get_post( $request['id'] ); - $existing_config = array(); - if ( $post ) { - $existing_config = json_decode( $post->post_content, true ); - $json_decoding_error = json_last_error(); - if ( JSON_ERROR_NONE !== $json_decoding_error || ! isset( $existing_config['isGlobalStylesUserThemeJSON'] ) || - ! $existing_config['isGlobalStylesUserThemeJSON'] ) { - $existing_config = array(); - } - } - if ( isset( $request['styles'] ) || isset( $request['settings'] ) ) { - $config = array(); - if ( isset( $request['styles'] ) ) { - $config['styles'] = $request['styles']; - } elseif ( isset( $existing_config['styles'] ) ) { - $config['styles'] = $existing_config['styles']; - } - if ( isset( $request['settings'] ) ) { - $config['settings'] = $request['settings']; - } elseif ( isset( $existing_config['settings'] ) ) { - $config['settings'] = $existing_config['settings']; - } - $config['isGlobalStylesUserThemeJSON'] = true; - $config['version'] = WP_Theme_JSON_Gutenberg::LATEST_SCHEMA; - $changes->post_content = wp_json_encode( $config ); - } - // Post title. - if ( isset( $request['title'] ) ) { - if ( is_string( $request['title'] ) ) { - $changes->post_title = $request['title']; - } elseif ( ! empty( $request['title']['raw'] ) ) { - $changes->post_title = $request['title']['raw']; - } - } - return $changes; - } - /** * Prepare a global styles config output for response. * * @since 5.9.0 + * @since 6.2 Handling of style.css was added to WP_Theme_JSON. * - * @param WP_Post $post Global Styles post object. + * @param WP_Post $post Global Styles post object. * @param WP_REST_Request $request Request object. * @return WP_REST_Response Response object. */ @@ -124,12 +36,15 @@ public function prepare_item_for_response( $post, $request ) { // phpcs:ignore V if ( $is_global_styles_user_theme_json ) { $config = ( new WP_Theme_JSON_Gutenberg( $raw_config, 'custom' ) )->get_raw_data(); } + // Base fields for every post. $data = array(); $fields = $this->get_fields_for_response( $request ); + if ( rest_is_field_included( 'id', $fields ) ) { $data['id'] = $post->ID; } + if ( rest_is_field_included( 'title', $fields ) ) { $data['title'] = array(); } @@ -138,94 +53,150 @@ public function prepare_item_for_response( $post, $request ) { // phpcs:ignore V } if ( rest_is_field_included( 'title.rendered', $fields ) ) { add_filter( 'protected_title_format', array( $this, 'protected_title_format' ) ); + $data['title']['rendered'] = get_the_title( $post->ID ); + remove_filter( 'protected_title_format', array( $this, 'protected_title_format' ) ); } + if ( rest_is_field_included( 'settings', $fields ) ) { $data['settings'] = ! empty( $config['settings'] ) && $is_global_styles_user_theme_json ? $config['settings'] : new stdClass(); } + if ( rest_is_field_included( 'styles', $fields ) ) { $data['styles'] = ! empty( $config['styles'] ) && $is_global_styles_user_theme_json ? $config['styles'] : new stdClass(); } + $context = ! empty( $request['context'] ) ? $request['context'] : 'view'; $data = $this->add_additional_fields_to_object( $data, $request ); $data = $this->filter_response_by_context( $data, $context ); + // Wrap the data in a response object. $response = rest_ensure_response( $data ); - $links = $this->prepare_links( $post->ID ); - $response->add_links( $links ); - if ( ! empty( $links['self']['href'] ) ) { - $actions = $this->get_available_actions(); - $self = $links['self']['href']; - foreach ( $actions as $rel ) { - $response->add_link( $rel, $self ); + + if ( rest_is_field_included( '_links', $fields ) || rest_is_field_included( '_embedded', $fields ) ) { + $links = $this->prepare_links( $post->ID ); + $response->add_links( $links ); + if ( ! empty( $links['self']['href'] ) ) { + $actions = $this->get_available_actions(); + $self = $links['self']['href']; + foreach ( $actions as $rel ) { + $response->add_link( $rel, $self ); + } } } + return $response; } /** - * Returns the given theme global styles config. + * Updates a single global style config. * * @since 5.9.0 + * @since 6.2.0 Added validation of styles.css property. * - * @param WP_REST_Request $request The request instance. - * @return WP_REST_Response|WP_Error + * @param WP_REST_Request $request Full details about the request. + * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. */ - public function get_theme_item( $request ) { - if ( wp_get_theme()->get_stylesheet() !== $request['stylesheet'] ) { - // This endpoint only supports the active theme for now. - return new WP_Error( - 'rest_theme_not_found', - __( 'Theme not found.', 'gutenberg' ), - array( 'status' => 404 ) - ); + public function update_item( $request ) { + $post_before = $this->get_post( $request['id'] ); + if ( is_wp_error( $post_before ) ) { + return $post_before; } - $theme = WP_Theme_JSON_Resolver_Gutenberg::get_merged_data( 'theme' ); - $data = array(); - $fields = $this->get_fields_for_response( $request ); - if ( rest_is_field_included( 'settings', $fields ) ) { - $data['settings'] = $theme->get_settings(); + + $changes = $this->prepare_item_for_database( $request ); + if ( is_wp_error( $changes ) ) { + return $changes; } - if ( rest_is_field_included( 'styles', $fields ) ) { - $raw_data = $theme->get_raw_data(); - $data['styles'] = isset( $raw_data['styles'] ) ? $raw_data['styles'] : array(); + + $result = wp_update_post( wp_slash( (array) $changes ), true, false ); + if ( is_wp_error( $result ) ) { + return $result; + } + + $post = get_post( $request['id'] ); + $fields_update = $this->update_additional_fields_for_object( $post, $request ); + if ( is_wp_error( $fields_update ) ) { + return $fields_update; } - $context = ! empty( $request['context'] ) ? $request['context'] : 'view'; - $data = $this->add_additional_fields_to_object( $data, $request ); - $data = $this->filter_response_by_context( $data, $context ); - $response = rest_ensure_response( $data ); - $links = array( - 'self' => array( - 'href' => rest_url( sprintf( '%s/%s/themes/%s', $this->namespace, $this->rest_base, $request['stylesheet'] ) ), - ), - ); - $response->add_links( $links ); - return $response; - } + wp_after_insert_post( $post, true, $post_before ); + $response = $this->prepare_item_for_response( $post, $request ); + + return rest_ensure_response( $response ); + } /** - * Returns the given theme global styles variations. + * Prepares a single global styles config for update. * - * @param WP_REST_Request $request The request instance. + * @since 5.9.0 + * @since 6.2.0 Added validation of styles.css property. * - * @return WP_REST_Response|WP_Error + * @param WP_REST_Request $request Request object. + * @return stdClass Changes to pass to wp_update_post. */ - public function get_theme_items( $request ) { - if ( wp_get_theme()->get_stylesheet() !== $request['stylesheet'] ) { - // This endpoint only supports the active theme for now. + protected function prepare_item_for_database( $request ) { + $changes = new stdClass(); + $changes->ID = $request['id']; + $post = get_post( $request['id'] ); + $existing_config = array(); + if ( $post ) { + $existing_config = json_decode( $post->post_content, true ); + $json_decoding_error = json_last_error(); + if ( JSON_ERROR_NONE !== $json_decoding_error || ! isset( $existing_config['isGlobalStylesUserThemeJSON'] ) || + ! $existing_config['isGlobalStylesUserThemeJSON'] ) { + $existing_config = array(); + } + } + if ( isset( $request['styles'] ) || isset( $request['settings'] ) ) { + $config = array(); + if ( isset( $request['styles'] ) ) { + $config['styles'] = $request['styles']; + $validate_custom_css = $this->validate_custom_css( $request['styles']['css'] ); + if ( is_wp_error( $validate_custom_css ) ) { + return $validate_custom_css; + } + } elseif ( isset( $existing_config['styles'] ) ) { + $config['styles'] = $existing_config['styles']; + } + if ( isset( $request['settings'] ) ) { + $config['settings'] = $request['settings']; + } elseif ( isset( $existing_config['settings'] ) ) { + $config['settings'] = $existing_config['settings']; + } + $config['isGlobalStylesUserThemeJSON'] = true; + $config['version'] = WP_Theme_JSON_Gutenberg::LATEST_SCHEMA; + $changes->post_content = wp_json_encode( $config ); + } + // Post title. + if ( isset( $request['title'] ) ) { + if ( is_string( $request['title'] ) ) { + $changes->post_title = $request['title']; + } elseif ( ! empty( $request['title']['raw'] ) ) { + $changes->post_title = $request['title']['raw']; + } + } + return $changes; + } + + /** + * Validate style.css as valid CSS. + * + * Currently just checks for invalid markup. + * + * @since 6.2.0 + * + * @param string $css CSS to validate. + * @return true|WP_Error True if the input was validated, otherwise WP_Error. + */ + private function validate_custom_css( $css ) { + if ( preg_match( '# 404 ) + 'rest_custom_css_illegal_markup', + __( 'Markup is not allowed in CSS.', 'gutenberg' ), + array( 'status' => 400 ) ); } - - $variations = WP_Theme_JSON_Resolver_Gutenberg::get_style_variations(); - $response = rest_ensure_response( $variations ); - - return $response; + return true; } - } diff --git a/lib/compat/wordpress-6.2/class-gutenberg-rest-pattern-directory-controller-6-2.php b/lib/compat/wordpress-6.2/class-gutenberg-rest-pattern-directory-controller-6-2.php index cc44c37b4d7c9..e8e456676e974 100644 --- a/lib/compat/wordpress-6.2/class-gutenberg-rest-pattern-directory-controller-6-2.php +++ b/lib/compat/wordpress-6.2/class-gutenberg-rest-pattern-directory-controller-6-2.php @@ -9,7 +9,128 @@ /** * Controller which provides REST endpoint for block patterns from wordpress.org/patterns. */ -class Gutenberg_REST_Pattern_Directory_Controller_6_2 extends Gutenberg_REST_Pattern_Directory_Controller_6_0 { +class Gutenberg_REST_Pattern_Directory_Controller_6_2 extends WP_REST_Pattern_Directory_Controller { + /** + * Registers the necessary REST API routes. + * + * @since 5.8.0 + * @since 6.2.0 Added pattern directory categories endpoint. + */ + public function register_routes() { + register_rest_route( + $this->namespace, + '/' . $this->rest_base . '/categories', + array( + array( + 'methods' => WP_REST_Server::READABLE, + 'callback' => array( $this, 'get_pattern_categories' ), + 'permission_callback' => array( $this, 'get_items_permissions_check' ), + ), + ) + ); + + parent::register_routes(); + } + + /** + * Retrieve block patterns categories. + * + * @since 6.2.0 + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. + */ + public function get_pattern_categories( $request ) { + $query_args = array( 'locale' => get_user_locale() ); + $transient_key = 'wp_remote_block_pattern_categories_' . md5( serialize( $query_args ) ); + + /** + * Use network-wide transient to improve performance. The locale is the only site + * configuration that affects the response, and it's included in the transient key. + */ + $raw_pattern_categories = get_site_transient( $transient_key ); + + if ( ! $raw_pattern_categories ) { + $api_url = 'http://api.wordpress.org/patterns/1.0/?categories&' . build_query( $query_args ); + if ( wp_http_supports( array( 'ssl' ) ) ) { + $api_url = set_url_scheme( $api_url, 'https' ); + } + + /** + * Default to a short TTL, to mitigate cache stampedes on high-traffic sites. + * This assumes that most errors will be short-lived, e.g., packet loss that causes the + * first request to fail, but a follow-up one will succeed. The value should be high + * enough to avoid stampedes, but low enough to not interfere with users manually + * re-trying a failed request. + */ + $cache_ttl = 5; + $wporg_response = wp_remote_get( $api_url ); + $raw_pattern_categories = json_decode( wp_remote_retrieve_body( $wporg_response ) ); + if ( is_wp_error( $wporg_response ) ) { + $raw_pattern_categories = $wporg_response; + + } elseif ( ! is_array( $raw_pattern_categories ) ) { + // HTTP request succeeded, but response data is invalid. + $raw_pattern_categories = new WP_Error( + 'pattern_directory_api_failed', + sprintf( + /* translators: %s: Support forums URL. */ + __( 'An unexpected error occurred. Something may be wrong with WordPress.org or this server’s configuration. If you continue to have problems, please try the support forums.', 'gutenberg' ), + __( 'https://wordpress.org/support/forums/', 'gutenberg' ) + ), + array( + 'response' => wp_remote_retrieve_body( $wporg_response ), + ) + ); + + } else { + // Response has valid data. + $cache_ttl = HOUR_IN_SECONDS; + } + + set_site_transient( $transient_key, $raw_pattern_categories, $cache_ttl ); + } + + if ( is_wp_error( $raw_pattern_categories ) ) { + $raw_pattern_categories->add_data( array( 'status' => 500 ) ); + + return $raw_pattern_categories; + } + + $response = array(); + + if ( $raw_pattern_categories ) { + foreach ( $raw_pattern_categories as $category ) { + $response[] = $this->prepare_response_for_collection( + $this->prepare_pattern_category_for_response( $category, $request ) + ); + } + } + + return new WP_REST_Response( $response ); + } + + /** + * Prepare a raw block pattern category before it gets output in a REST API response. + * + * @since 6.2.0 + * + * @param object $item Raw pattern category from api.wordpress.org, before any changes. + * @param WP_REST_Request $request Request object. + * @return WP_REST_Response + */ + public function prepare_pattern_category_for_response( $item, $request ) { + $raw_pattern_category = array( + 'id' => absint( $item->id ), + 'name' => sanitize_text_field( $item->name ), + 'slug' => sanitize_title_with_dashes( $item->slug ), + ); + + $prepared_pattern_category = $this->add_additional_fields_to_object( $raw_pattern_category, $request ); + + return new WP_REST_Response( $prepared_pattern_category ); + } + /** * Search and retrieve block patterns metadata * @@ -48,7 +169,7 @@ public function get_items( $request ) { $transient_key = $this->get_transient_key( $query_args ); - /* + /** * Use network-wide transient to improve performance. The locale is the only site * configuration that affects the response, and it's included in the transient key. */ @@ -60,7 +181,7 @@ public function get_items( $request ) { $api_url = set_url_scheme( $api_url, 'https' ); } - /* + /** * Default to a short TTL, to mitigate cache stampedes on high-traffic sites. * This assumes that most errors will be short-lived, e.g., packet loss that causes the * first request to fail, but a follow-up one will succeed. The value should be high diff --git a/lib/compat/wordpress-6.2/class-wp-theme-json-6-2.php b/lib/compat/wordpress-6.2/class-wp-theme-json-6-2.php index 4c6a1f50612a8..fe01eb273066e 100644 --- a/lib/compat/wordpress-6.2/class-wp-theme-json-6-2.php +++ b/lib/compat/wordpress-6.2/class-wp-theme-json-6-2.php @@ -97,6 +97,25 @@ class WP_Theme_JSON_6_2 extends WP_Theme_JSON_6_1 { 'box-shadow' => array( 'shadow' ), ); + /** + * Indirect metadata for style properties that are not directly output. + * + * Each element is a direct mapping from a CSS property name to the + * path to the value in theme.json & block attributes. + * + * Indirect properties are not output directly by `compute_style_properties`, + * but are used elsewhere in the processing of global styles. The indirect + * property is used to validate whether or not a style value is allowed. + * + * @since 6.2.0 + * @var array + */ + const INDIRECT_PROPERTIES_METADATA = array( + 'gap' => array( 'spacing', 'blockGap' ), + 'column-gap' => array( 'spacing', 'blockGap', 'left' ), + 'row-gap' => array( 'spacing', 'blockGap', 'top' ), + ); + /** * The valid properties under the settings key. * @@ -175,6 +194,7 @@ class WP_Theme_JSON_6_2 extends WP_Theme_JSON_6_1 { * @since 6.1.0 Added new side properties for `border`, * added new property `shadow`, * updated `blockGap` to be allowed at any level. + * @since 6.2.0 Added new property `css`. * @var array */ const VALID_STYLES = array( @@ -215,5 +235,157 @@ class WP_Theme_JSON_6_2 extends WP_Theme_JSON_6_1 { 'textDecoration' => null, 'textTransform' => null, ), + 'css' => null, ); + + /** + * Processes a style node and returns the same node + * without the insecure styles. + * + * @since 5.9.0 + * @since 6.2.0 Allow indirect properties used outside of `compute_style_properties`. + * + * @param array $input Node to process. + * @return array + */ + protected static function remove_insecure_styles( $input ) { + $output = array(); + $declarations = static::compute_style_properties( $input ); + + foreach ( $declarations as $declaration ) { + if ( static::is_safe_css_declaration( $declaration['name'], $declaration['value'] ) ) { + $path = static::PROPERTIES_METADATA[ $declaration['name'] ]; + + // Check the value isn't an array before adding so as to not + // double up shorthand and longhand styles. + $value = _wp_array_get( $input, $path, array() ); + if ( ! is_array( $value ) ) { + _wp_array_set( $output, $path, $value ); + } + } + } + + // Ensure indirect properties not handled by `compute_style_properties` are allowed. + foreach ( static::INDIRECT_PROPERTIES_METADATA as $property => $path ) { + $value = _wp_array_get( $input, $path, array() ); + if ( + isset( $value ) && + ! is_array( $value ) && + static::is_safe_css_declaration( $property, $value ) + ) { + _wp_array_set( $output, $path, $value ); + } + } + + return $output; + } + + /** + * Returns the stylesheet that results of processing + * the theme.json structure this object represents. + * + * @param array $types Types of styles to load. Will load all by default. It accepts: + * 'variables': only the CSS Custom Properties for presets & custom ones. + * 'styles': only the styles section in theme.json. + * 'presets': only the classes for the presets. + * 'custom-css': only the css from global styles.css. + * @param array $origins A list of origins to include. By default it includes VALID_ORIGINS. + * @param array $options An array of options for now used for internal purposes only (may change without notice). + * The options currently supported are 'scope' that makes sure all style are scoped to a given selector, + * and root_selector which overwrites and forces a given selector to be used on the root node. + * @return string Stylesheet. + */ + public function get_stylesheet( $types = array( 'variables', 'styles', 'presets' ), $origins = null, $options = array() ) { + if ( null === $origins ) { + $origins = static::VALID_ORIGINS; + } + + if ( is_string( $types ) ) { + // Dispatch error and map old arguments to new ones. + _deprecated_argument( __FUNCTION__, '5.9' ); + if ( 'block_styles' === $types ) { + $types = array( 'styles', 'presets' ); + } elseif ( 'css_variables' === $types ) { + $types = array( 'variables' ); + } else { + $types = array( 'variables', 'styles', 'presets' ); + } + } + + $blocks_metadata = static::get_blocks_metadata(); + $style_nodes = static::get_style_nodes( $this->theme_json, $blocks_metadata ); + $setting_nodes = static::get_setting_nodes( $this->theme_json, $blocks_metadata ); + + $root_style_key = array_search( static::ROOT_BLOCK_SELECTOR, array_column( $style_nodes, 'selector' ), true ); + $root_settings_key = array_search( static::ROOT_BLOCK_SELECTOR, array_column( $setting_nodes, 'selector' ), true ); + + if ( ! empty( $options['scope'] ) ) { + foreach ( $setting_nodes as &$node ) { + $node['selector'] = static::scope_selector( $options['scope'], $node['selector'] ); + } + foreach ( $style_nodes as &$node ) { + $node['selector'] = static::scope_selector( $options['scope'], $node['selector'] ); + } + } + + if ( ! empty( $options['root_selector'] ) ) { + if ( false !== $root_settings_key ) { + $setting_nodes[ $root_settings_key ]['selector'] = $options['root_selector']; + } + if ( false !== $root_style_key ) { + $setting_nodes[ $root_style_key ]['selector'] = $options['root_selector']; + } + } + + $stylesheet = ''; + + if ( in_array( 'variables', $types, true ) ) { + $stylesheet .= $this->get_css_variables( $setting_nodes, $origins ); + } + + if ( in_array( 'styles', $types, true ) ) { + if ( false !== $root_style_key ) { + $stylesheet .= $this->get_root_layout_rules( $style_nodes[ $root_style_key ]['selector'], $style_nodes[ $root_style_key ] ); + } + $stylesheet .= $this->get_block_classes( $style_nodes ); + } elseif ( in_array( 'base-layout-styles', $types, true ) ) { + $root_selector = static::ROOT_BLOCK_SELECTOR; + $columns_selector = '.wp-block-columns'; + if ( ! empty( $options['scope'] ) ) { + $root_selector = static::scope_selector( $options['scope'], $root_selector ); + $columns_selector = static::scope_selector( $options['scope'], $columns_selector ); + } + if ( ! empty( $options['root_selector'] ) ) { + $root_selector = $options['root_selector']; + } + // Base layout styles are provided as part of `styles`, so only output separately if explicitly requested. + // For backwards compatibility, the Columns block is explicitly included, to support a different default gap value. + $base_styles_nodes = array( + array( + 'path' => array( 'styles' ), + 'selector' => $root_selector, + ), + array( + 'path' => array( 'styles', 'blocks', 'core/columns' ), + 'selector' => $columns_selector, + 'name' => 'core/columns', + ), + ); + + foreach ( $base_styles_nodes as $base_style_node ) { + $stylesheet .= $this->get_layout_styles( $base_style_node ); + } + } + + if ( in_array( 'presets', $types, true ) ) { + $stylesheet .= $this->get_preset_classes( $setting_nodes, $origins ); + } + + // Load the custom CSS last so it has the highest specificity. + if ( in_array( 'custom-css', $types, true ) ) { + $stylesheet .= _wp_array_get( $this->theme_json, array( 'styles', 'css' ) ); + } + + return $stylesheet; + } } diff --git a/lib/compat/wordpress-6.2/class-wp-theme-json-resolver-6-2.php b/lib/compat/wordpress-6.2/class-wp-theme-json-resolver-6-2.php index e10710e0f4709..110c8bac7b147 100644 --- a/lib/compat/wordpress-6.2/class-wp-theme-json-resolver-6-2.php +++ b/lib/compat/wordpress-6.2/class-wp-theme-json-resolver-6-2.php @@ -17,6 +17,86 @@ */ class WP_Theme_JSON_Resolver_6_2 extends WP_Theme_JSON_Resolver_6_1 { + /** + * Returns the custom post type that contains the user's origin config + * for the active theme or a void array if none are found. + * + * This can also create and return a new draft custom post type. + * + * @param WP_Theme $theme The theme object. If empty, it + * defaults to the active theme. + * @param bool $create_post Optional. Whether a new custom post + * type should be created if none are + * found. Default false. + * @param array $post_status_filter Optional. Filter custom post type by + * post status. Default `array( 'publish' )`, + * so it only fetches published posts. + * @return array Custom Post Type for the user's origin config. + */ + public static function get_user_data_from_wp_global_styles( $theme, $create_post = false, $post_status_filter = array( 'publish' ) ) { + if ( ! $theme instanceof WP_Theme ) { + $theme = wp_get_theme(); + } + + /* + * Bail early if the theme does not support a theme.json. + * + * Since wp_theme_has_theme_json only supports the active + * theme, the extra condition for whether $theme is the active theme is + * present here. + */ + if ( $theme->get_stylesheet() === get_stylesheet() && ! wp_theme_has_theme_json() ) { + return array(); + } + + $user_cpt = array(); + $post_type_filter = 'wp_global_styles'; + $stylesheet = $theme->get_stylesheet(); + $args = array( + 'posts_per_page' => 1, + 'orderby' => 'date', + 'order' => 'desc', + 'post_type' => $post_type_filter, + 'post_status' => $post_status_filter, + 'ignore_sticky_posts' => true, + 'no_found_rows' => true, + 'update_post_meta_cache' => false, + 'update_post_term_cache' => false, + 'tax_query' => array( + array( + 'taxonomy' => 'wp_theme', + 'field' => 'name', + 'terms' => $stylesheet, + ), + ), + ); + + $global_style_query = new WP_Query(); + $recent_posts = $global_style_query->query( $args ); + if ( count( $recent_posts ) === 1 ) { + $user_cpt = get_object_vars( $recent_posts[0] ); + } elseif ( $create_post ) { + $cpt_post_id = wp_insert_post( + array( + 'post_content' => '{"version": ' . WP_Theme_JSON::LATEST_SCHEMA . ', "isGlobalStylesUserThemeJSON": true }', + 'post_status' => 'publish', + 'post_title' => 'Custom Styles', // Do not make string translatable, see https://core.trac.wordpress.org/ticket/54518. + 'post_type' => $post_type_filter, + 'post_name' => sprintf( 'wp-global-styles-%s', urlencode( $stylesheet ) ), + 'tax_input' => array( + 'wp_theme' => array( $stylesheet ), + ), + ), + true + ); + if ( ! is_wp_error( $cpt_post_id ) ) { + $user_cpt = get_object_vars( get_post( $cpt_post_id ) ); + } + } + + return $user_cpt; + } + /** * Determines whether the active theme has a theme.json file. * @@ -32,4 +112,119 @@ public static function theme_has_support() { return wp_theme_has_theme_json(); } + /** + * Returns the data merged from multiple origins. + * + * There are four sources of data (origins) for a site: + * + * - default => WordPress + * - blocks => each one of the blocks provides data for itself + * - theme => the active theme + * - custom => data provided by the user + * + * The custom's has higher priority than the theme's, the theme's higher than blocks', + * and block's higher than default's. + * + * Unlike the getters + * {@link https://developer.wordpress.org/reference/classes/wp_theme_json_resolver/get_core_data/ get_core_data}, + * {@link https://developer.wordpress.org/reference/classes/wp_theme_json_resolver/get_theme_data/ get_theme_data}, + * and {@link https://developer.wordpress.org/reference/classes/wp_theme_json_resolver/get_user_data/ get_user_data}, + * this method returns data after it has been merged with the previous origins. + * This means that if the same piece of data is declared in different origins + * (default, blocks, theme, custom), the last origin overrides the previous. + * + * For example, if the user has set a background color + * for the paragraph block, and the theme has done it as well, + * the user preference wins. + * + * @param string $origin Optional. To what level should we merge data:'default', 'blocks', 'theme' or 'custom'. + * 'custom' is used as default value as well as fallback value if the origin is unknown. + * + * @return WP_Theme_JSON + */ + public static function get_merged_data( $origin = 'custom' ) { + if ( is_array( $origin ) ) { + _deprecated_argument( __FUNCTION__, '5.9.0' ); + } + + $result = static::get_core_data(); + if ( 'default' === $origin ) { + $result->set_spacing_sizes(); + return $result; + } + + $result->merge( static::get_block_data() ); + if ( 'blocks' === $origin ) { + return $result; + } + + $result->merge( static::get_theme_data() ); + if ( 'theme' === $origin ) { + $result->set_spacing_sizes(); + return $result; + } + + $result->merge( static::get_user_data() ); + $result->set_spacing_sizes(); + return $result; + } + + /** + * Returns the user's origin config. + * + * @since 6.2 Added check for the WP_Theme_JSON_Gutenberg class to prevent $user + * values set in core fron overriding the new custom css values added to VALID_STYLES. + * This does not need to be backported to core as the new VALID_STYLES[css] value will + * be added to core with 6.2. + * + * @return WP_Theme_JSON_Gutenberg Entity that holds styles for user data. + */ + public static function get_user_data() { + if ( null !== static::$user && static::$user instanceof WP_Theme_JSON_Gutenberg ) { + return static::$user; + } + + $config = array(); + $user_cpt = static::get_user_data_from_wp_global_styles( wp_get_theme() ); + + if ( array_key_exists( 'post_content', $user_cpt ) ) { + $decoded_data = json_decode( $user_cpt['post_content'], true ); + + $json_decoding_error = json_last_error(); + if ( JSON_ERROR_NONE !== $json_decoding_error ) { + trigger_error( 'Error when decoding a theme.json schema for user data. ' . json_last_error_msg() ); + /** + * Filters the data provided by the user for global styles & settings. + * + * @param WP_Theme_JSON_Data_Gutenberg Class to access and update the underlying data. + */ + $theme_json = apply_filters( 'wp_theme_json_data_user', new WP_Theme_JSON_Data_Gutenberg( $config, 'custom' ) ); + $config = $theme_json->get_data(); + return new WP_Theme_JSON_Gutenberg( $config, 'custom' ); + } + + // Very important to verify if the flag isGlobalStylesUserThemeJSON is true. + // If is not true the content was not escaped and is not safe. + if ( + is_array( $decoded_data ) && + isset( $decoded_data['isGlobalStylesUserThemeJSON'] ) && + $decoded_data['isGlobalStylesUserThemeJSON'] + ) { + unset( $decoded_data['isGlobalStylesUserThemeJSON'] ); + $config = $decoded_data; + } + } + + /** + * Filters the data provided by the user for global styles & settings. + * + * @param WP_Theme_JSON_Data_Gutenberg Class to access and update the underlying data. + */ + $theme_json = apply_filters( 'wp_theme_json_data_user', new WP_Theme_JSON_Data_Gutenberg( $config, 'custom' ) ); + $config = $theme_json->get_data(); + + static::$user = new WP_Theme_JSON_Gutenberg( $config, 'custom' ); + + return static::$user; + } } diff --git a/lib/compat/wordpress-6.2/default-filters.php b/lib/compat/wordpress-6.2/default-filters.php index ac1ce19b42551..4698ead7f9bc3 100644 --- a/lib/compat/wordpress-6.2/default-filters.php +++ b/lib/compat/wordpress-6.2/default-filters.php @@ -18,10 +18,8 @@ */ /** - * Note for backport: we should also remove the existing filters: - * - * > add_action( 'switch_theme', array( 'WP_Theme_JSON_Resolver', 'clean_cached_data' ) ); - * > add_action( 'start_previewing_theme', array( 'WP_Theme_JSON_Resolver', 'clean_cached_data' ) ); + * When backporting to core, the existing filters hooked to WP_Theme_JSON_Resolver::clean_cached_data() + * need to be removed. */ -add_action( 'switch_theme', 'wp_theme_clean_theme_json_cached_data' ); -add_action( 'start_previewing_theme', 'wp_theme_clean_theme_json_cached_data' ); +add_action( 'start_previewing_theme', '_gutenberg_clean_theme_json_caches' ); +add_action( 'switch_theme', '_gutenberg_clean_theme_json_caches' ); diff --git a/lib/compat/wordpress-6.2/edit-form-blocks.php b/lib/compat/wordpress-6.2/edit-form-blocks.php new file mode 100644 index 0000000000000..5031550abc154 --- /dev/null +++ b/lib/compat/wordpress-6.2/edit-form-blocks.php @@ -0,0 +1,24 @@ +get_stylesheet( array( 'variables' ), $origins ); + $types = array_diff( $types, array( 'variables' ) ); + } + + /* + * For the remaining types (presets, styles), we do consider origins: + * + * - themes without theme.json: only the classes for the presets defined by core + * - themes with theme.json: the presets and styles classes, both from core and the theme + */ + $styles_rest = ''; + if ( ! empty( $types ) ) { + /* + * We only use the default, theme, and custom origins. + * This is because styles for blocks origin are added + * at a later phase (render cycle) so we only render the ones in use. + * @see wp_add_global_styles_for_blocks + */ + $origins = array( 'default', 'theme', 'custom' ); + if ( ! $supports_theme_json ) { + $origins = array( 'default' ); + } + $styles_rest = $tree->get_stylesheet( $types, $origins ); + } + $stylesheet = $styles_variables . $styles_rest; + if ( $can_use_cached ) { + wp_cache_set( $cache_key, $stylesheet, $cache_group ); + } + return $stylesheet; +} + +/** + * Function to get the settings resulting of merging core, theme, and user data. + * + * @param array $path Path to the specific setting to retrieve. Optional. + * If empty, will return all settings. + * @param array $context { + * Metadata to know where to retrieve the $path from. Optional. + * + * @type string $block_name Which block to retrieve the settings from. + * If empty, it'll return the settings for the global context. + * @type string $origin Which origin to take data from. + * Valid values are 'all' (core, theme, and user) or 'base' (core and theme). + * If empty or unknown, 'all' is used. + * } + * + * @return array The settings to retrieve. + */ +function gutenberg_get_global_settings( $path = array(), $context = array() ) { + if ( ! empty( $context['block_name'] ) ) { + $new_path = array( 'blocks', $context['block_name'] ); + foreach ( $path as $subpath ) { + $new_path[] = $subpath; + } + $path = $new_path; } + + // This is the default value when no origin is provided or when it is 'all'. + $origin = 'custom'; + if ( + ! wp_theme_has_theme_json() || + ( isset( $context['origin'] ) && 'base' === $context['origin'] ) + ) { + $origin = 'theme'; + } + + $cache_group = 'theme_json'; + $cache_key = 'gutenberg_get_global_settings_' . $origin; + $settings = wp_cache_get( $cache_key, $cache_group ); + + if ( false === $settings || WP_DEBUG ) { + $settings = WP_Theme_JSON_Resolver_Gutenberg::get_merged_data( $origin )->get_settings(); + wp_cache_set( $cache_key, $settings, $cache_group ); + } + + return _wp_array_get( $settings, $path, $settings ); +} + +/** + * Private function to clean the caches used by gutenberg_get_global_settings method. + * + * @access private + */ +function _gutenberg_clean_theme_json_caches() { + wp_cache_delete( 'wp_theme_has_theme_json', 'theme_json' ); + wp_cache_delete( 'gutenberg_get_global_stylesheet', 'theme_json' ); + wp_cache_delete( 'gutenberg_get_global_settings_custom', 'theme_json' ); + wp_cache_delete( 'gutenberg_get_global_settings_theme', 'theme_json' ); + WP_Theme_JSON_Resolver_Gutenberg::clean_cached_data(); +} + +/** + * Tell the cache mechanisms not to persist theme.json data across requests. + * The data stored under this cache group: + * + * - wp_theme_has_theme_json + * - gutenberg_get_global_settings + * - gutenberg_get_global_stylesheet + * + * There is some hooks consumers can use to modify parts + * of the theme.json logic. + * See https://make.wordpress.org/core/2022/10/10/filters-for-theme-json-data/ + * + * The rationale to make this cache group non persistent is to make sure derived data + * from theme.json is always fresh from the potential modifications done via hooks + * that can use dynamic data (modify the stylesheet depending on some option, + * or settings depending on user permissions, etc.). + * + * A different alternative considered was to invalidate the cache upon certain + * events such as options add/update/delete, user meta, etc. + * It was judged not enough, hence this approach. + * See https://github.com/WordPress/gutenberg/pull/45372 + */ +function _gutenberg_add_non_persistent_theme_json_cache_group() { + wp_cache_add_non_persistent_groups( 'theme_json' ); } +add_action( 'plugins_loaded', '_gutenberg_add_non_persistent_theme_json_cache_group' ); diff --git a/lib/compat/wordpress-6.2/rest-api.php b/lib/compat/wordpress-6.2/rest-api.php index 023a79e3db95f..4900a622b6c01 100644 --- a/lib/compat/wordpress-6.2/rest-api.php +++ b/lib/compat/wordpress-6.2/rest-api.php @@ -23,6 +23,15 @@ function gutenberg_register_rest_pattern_directory() { } add_action( 'rest_api_init', 'gutenberg_register_rest_pattern_directory' ); +/** + * Registers the block patterns REST API routes. + */ +function gutenberg_register_rest_block_patterns() { + $block_patterns = new Gutenberg_REST_Block_Patterns_Controller_6_2(); + $block_patterns->register_routes(); +} +add_action( 'rest_api_init', 'gutenberg_register_rest_block_patterns' ); + /** * Add extra collection params to pattern directory requests. * @@ -83,3 +92,12 @@ function gutenberg_pattern_directory_collection_params_6_2( $query_params ) { return $query_params; } add_filter( 'rest_pattern_directory_collection_params', 'gutenberg_pattern_directory_collection_params_6_2' ); + +/** + * Registers the Global Styles REST API routes. + */ +function gutenberg_register_global_styles_endpoints() { + $editor_settings = new Gutenberg_REST_Global_Styles_Controller_6_2(); + $editor_settings->register_routes(); +} +add_action( 'rest_api_init', 'gutenberg_register_global_styles_endpoints' ); diff --git a/lib/compat/wordpress-6.2/script-loader.php b/lib/compat/wordpress-6.2/script-loader.php index 9d67cbd3ef1ad..e08bfa6e46ca8 100644 --- a/lib/compat/wordpress-6.2/script-loader.php +++ b/lib/compat/wordpress-6.2/script-loader.php @@ -63,3 +63,82 @@ static function () use ( $style ) { $priority ); } + +/** + * Sets the content assets for the block editor. + * + * Note for core merge: see inline comment on what's been updated. + */ +function gutenberg_resolve_assets_override() { + global $pagenow; + + $script_handles = array(); + // Note for core merge: only 'wp-edit-blocks' should be in this array. + $style_handles = array( + 'wp-edit-blocks', + ); + + if ( current_theme_supports( 'wp-block-styles' ) ) { + $style_handles[] = 'wp-block-library-theme'; + } + + if ( 'widgets.php' === $pagenow || 'customize.php' === $pagenow ) { + $style_handles[] = 'wp-widgets'; + $style_handles[] = 'wp-edit-widgets'; + } + + $block_registry = WP_Block_Type_Registry::get_instance(); + + foreach ( $block_registry->get_all_registered() as $block_type ) { + // In older WordPress versions, like 6.0, these properties are not defined. + if ( isset( $block_type->style_handles ) && is_array( $block_type->style_handles ) ) { + $style_handles = array_merge( $style_handles, $block_type->style_handles ); + } + + if ( isset( $block_type->editor_style_handles ) && is_array( $block_type->editor_style_handles ) ) { + $style_handles = array_merge( $style_handles, $block_type->editor_style_handles ); + } + + if ( isset( $block_type->script_handles ) && is_array( $block_type->script_handles ) ) { + $script_handles = array_merge( $script_handles, $block_type->script_handles ); + } + } + + $style_handles = array_unique( $style_handles ); + $done = wp_styles()->done; + + ob_start(); + + // We do not need reset styles for the iframed editor. + wp_styles()->done = array( 'wp-reset-editor-styles' ); + wp_styles()->do_items( $style_handles ); + wp_styles()->done = $done; + + $styles = ob_get_clean(); + + $script_handles = array_unique( $script_handles ); + $done = wp_scripts()->done; + + ob_start(); + + wp_scripts()->done = array(); + wp_scripts()->do_items( $script_handles ); + wp_scripts()->done = $done; + + $scripts = ob_get_clean(); + + return array( + 'styles' => $styles, + 'scripts' => $scripts, + ); +} + +add_filter( + 'block_editor_settings_all', + function( $settings ) { + // We must override what core is passing now. + $settings['__unstableResolvedAssets'] = gutenberg_resolve_assets_override(); + return $settings; + }, + 100 +); diff --git a/lib/compat/wordpress-6.2/site-editor.php b/lib/compat/wordpress-6.2/site-editor.php new file mode 100644 index 0000000000000..b6246e49c6d11 --- /dev/null +++ b/lib/compat/wordpress-6.2/site-editor.php @@ -0,0 +1,24 @@ +post ) ) { + return $settings; + } + + unset( $settings['__unstableHomeTemplate'] ); + + return $settings; +} +add_filter( 'block_editor_settings_all', 'gutenberg_site_editor_unset_homepage_setting', 10, 2 ); diff --git a/lib/experimental/class-wp-rest-block-editor-settings-controller.php b/lib/experimental/class-wp-rest-block-editor-settings-controller.php index 4a258a70102bb..d95fb497d3a0a 100644 --- a/lib/experimental/class-wp-rest-block-editor-settings-controller.php +++ b/lib/experimental/class-wp-rest-block-editor-settings-controller.php @@ -168,6 +168,12 @@ public function get_item_schema() { 'context' => array( 'mobile' ), ), + '__experimentalBlockInspectorTabs' => array( + 'description' => __( 'Block inspector tab display overrides.', 'gutenberg' ), + 'type' => 'object', + 'context' => array( 'post-editor', 'site-editor', 'widgets-editor' ), + ), + 'alignWide' => array( 'description' => __( 'Enable/Disable Wide/Full Alignments.', 'gutenberg' ), 'type' => 'boolean', diff --git a/lib/experimental/class-wp-theme-json-resolver-gutenberg.php b/lib/experimental/class-wp-theme-json-resolver-gutenberg.php index 2f1ee93db30b5..566f6c4e4501e 100644 --- a/lib/experimental/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/experimental/class-wp-theme-json-resolver-gutenberg.php @@ -34,9 +34,15 @@ public static function get_theme_data( $deprecated = array(), $settings = array( } // When backporting to core, remove the instanceof Gutenberg class check, as it is only required for the Gutenberg plugin. - if ( null === static::$theme || ! static::$theme instanceof WP_Theme_JSON_Gutenberg ) { - $theme_json_data = static::read_json_file( static::get_file_path_from_theme( 'theme.json' ) ); - $theme_json_data = static::translate( $theme_json_data, wp_get_theme()->get( 'TextDomain' ) ); + if ( null === static::$theme || ! static::has_same_registered_blocks( 'theme' ) ) { + $theme_json_file = static::get_file_path_from_theme( 'theme.json' ); + $wp_theme = wp_get_theme(); + if ( '' !== $theme_json_file ) { + $theme_json_data = static::read_json_file( $theme_json_file ); + $theme_json_data = static::translate( $theme_json_data, $wp_theme->get( 'TextDomain' ) ); + } else { + $theme_json_data = array(); + } $theme_json_data = gutenberg_add_registered_webfonts_to_theme_json( $theme_json_data ); /** @@ -48,17 +54,21 @@ public static function get_theme_data( $deprecated = array(), $settings = array( $theme_json_data = $theme_json->get_data(); static::$theme = new WP_Theme_JSON_Gutenberg( $theme_json_data ); - if ( wp_get_theme()->parent() ) { + if ( $wp_theme->parent() ) { // Get parent theme.json. - $parent_theme_json_data = static::read_json_file( static::get_file_path_from_theme( 'theme.json', true ) ); - $parent_theme_json_data = static::translate( $parent_theme_json_data, wp_get_theme()->parent()->get( 'TextDomain' ) ); - $parent_theme_json_data = gutenberg_add_registered_webfonts_to_theme_json( $parent_theme_json_data ); - $parent_theme = new WP_Theme_JSON_Gutenberg( $parent_theme_json_data ); - - // Merge the child theme.json into the parent theme.json. - // The child theme takes precedence over the parent. - $parent_theme->merge( static::$theme ); - static::$theme = $parent_theme; + $parent_theme_json_file = static::get_file_path_from_theme( 'theme.json', true ); + if ( '' !== $parent_theme_json_file ) { + $parent_theme_json_data = static::read_json_file( $parent_theme_json_file ); + $parent_theme_json_data = gutenberg_add_registered_webfonts_to_theme_json( $parent_theme_json_data ); + $parent_theme = new WP_Theme_JSON_Gutenberg( $parent_theme_json_data ); + + /* + * Merge the child theme.json into the parent theme.json. + * The child theme takes precedence over the parent. + */ + $parent_theme->merge( static::$theme ); + static::$theme = $parent_theme; + } } } @@ -72,7 +82,7 @@ public static function get_theme_data( $deprecated = array(), $settings = array( * So we take theme supports, transform it to theme.json shape * and merge the static::$theme upon that. */ - $theme_support_data = WP_Theme_JSON_Gutenberg::get_from_editor_settings( get_default_block_editor_settings() ); + $theme_support_data = WP_Theme_JSON_Gutenberg::get_from_editor_settings( gutenberg_get_legacy_theme_supports_for_theme_json() ); if ( ! wp_theme_has_theme_json() ) { if ( ! isset( $theme_support_data['settings']['color'] ) ) { $theme_support_data['settings']['color'] = array(); @@ -164,47 +174,4 @@ private static function remove_JSON_comments( $array ) { return $array; } - /** - * Returns the data merged from multiple origins. - * - * There are three sources of data (origins) for a site: - * default, theme, and custom. The custom's has higher priority - * than the theme's, and the theme's higher than default's. - * - * Unlike the getters {@link get_core_data}, - * {@link get_theme_data}, and {@link get_user_data}, - * this method returns data after it has been merged - * with the previous origins. This means that if the same piece of data - * is declared in different origins (user, theme, and core), - * the last origin overrides the previous. - * - * For example, if the user has set a background color - * for the paragraph block, and the theme has done it as well, - * the user preference wins. - * - * @since 5.8.0 - * @since 5.9.0 Added user data, removed the `$settings` parameter, - * added the `$origin` parameter. - * - * @param string $origin Optional. To what level should we merge data. - * Valid values are 'theme' or 'custom'. Default 'custom'. - * @return WP_Theme_JSON - */ - public static function get_merged_data( $origin = 'custom' ) { - if ( is_array( $origin ) ) { - _deprecated_argument( __FUNCTION__, '5.9' ); - } - - $result = new WP_Theme_JSON_Gutenberg(); - $result->merge( static::get_core_data() ); - $result->merge( static::get_block_data() ); - $result->merge( static::get_theme_data() ); - if ( 'custom' === $origin ) { - $result->merge( static::get_user_data() ); - } - // Generate the default spacing sizes presets. - $result->set_spacing_sizes(); - - return $result; - } } diff --git a/lib/experimental/class-wp-webfonts.php b/lib/experimental/class-wp-webfonts.php index e40aca9e8671f..0c0cb06a27ffb 100644 --- a/lib/experimental/class-wp-webfonts.php +++ b/lib/experimental/class-wp-webfonts.php @@ -178,7 +178,9 @@ public function enqueue_webfont( $font_family_name ) { */ public static function get_font_slug( $to_convert ) { if ( is_array( $to_convert ) ) { - if ( isset( $to_convert['font-family'] ) ) { + if ( isset( $to_convert['slug'] ) ) { + return $to_convert['slug']; + } elseif ( isset( $to_convert['font-family'] ) ) { $to_convert = $to_convert['font-family']; } elseif ( isset( $to_convert['fontFamily'] ) ) { $to_convert = $to_convert['fontFamily']; @@ -188,6 +190,11 @@ public static function get_font_slug( $to_convert ) { } } + // If the font-family is a comma-separated list (example: "Inter, sans-serif" ), use just the first font. + if ( strpos( $to_convert, ',' ) !== false ) { + $to_convert = explode( ',', $to_convert )[0]; + } + return sanitize_title( $to_convert ); } diff --git a/lib/experimental/editor-settings.php b/lib/experimental/editor-settings.php index b0c05544d6b19..862a1bfe2b28c 100644 --- a/lib/experimental/editor-settings.php +++ b/lib/experimental/editor-settings.php @@ -86,6 +86,9 @@ function gutenberg_enable_experiments() { if ( $gutenberg_experiments && array_key_exists( 'gutenberg-off-canvas-navigation-editor', $gutenberg_experiments ) ) { wp_add_inline_script( 'wp-block-editor', 'window.__experimentalEnableOffCanvasNavigationEditor = true', 'before' ); } + if ( $gutenberg_experiments && array_key_exists( 'gutenberg-block-inspector-tabs', $gutenberg_experiments ) ) { + wp_add_inline_script( 'wp-block-editor', 'window.__experimentalEnableBlockInspectorTabs = true', 'before' ); + } } add_action( 'admin_init', 'gutenberg_enable_experiments' ); diff --git a/lib/experimental/html/class-wp-html-span.php b/lib/experimental/html/class-wp-html-span.php new file mode 100644 index 0000000000000..39e603662b17b --- /dev/null +++ b/lib/experimental/html/class-wp-html-span.php @@ -0,0 +1,52 @@ +start = $start; + $this->end = $end; + } +} diff --git a/lib/experimental/html/class-wp-html-tag-processor.php b/lib/experimental/html/class-wp-html-tag-processor.php index 99e347818ff4e..affbb6fb27b5c 100644 --- a/lib/experimental/html/class-wp-html-tag-processor.php +++ b/lib/experimental/html/class-wp-html-tag-processor.php @@ -180,6 +180,25 @@ * @since 6.2.0 */ class WP_HTML_Tag_Processor { + /** + * The maximum number of bookmarks allowed to exist at + * any given time. + * + * @see set_bookmark(); + * @since 6.2.0 + * @var int + */ + const MAX_BOOKMARKS = 10; + + /** + * Maximum number of times seek() can be called. + * Prevents accidental infinite loops. + * + * @see seek() + * @since 6.2.0 + * @var int + */ + const MAX_SEEK_OPS = 1000; /** * The HTML document to parse. @@ -221,6 +240,14 @@ class WP_HTML_Tag_Processor { */ private $sought_match_offset; + /** + * Whether to visit tag closers, e.g. , when walking an input document. + * + * @since 6.2.0 + * @var boolean + */ + private $stop_on_tag_closers; + /** * The updated HTML document. * @@ -276,6 +303,29 @@ class WP_HTML_Tag_Processor { */ private $tag_name_length; + /** + * Byte offset in input document where current tag token ends. + * + * Example: + * ``` + *
... + * 0 1 | + * 01234567890123456 + * --- tag name ends at 14 + * ``` + * + * @since 6.2.0 + * @var ?int + */ + private $tag_ends_at; + + /** + * Whether the current tag is an opening tag, e.g.
, or a closing tag, e.g.
. + * + * @var boolean + */ + private $is_closing_tag; + /** * Lazily-built index of attributes found within an HTML tag, keyed by the attribute name. * @@ -318,11 +368,11 @@ class WP_HTML_Tag_Processor { * * Example: * - * // Add the `WP-block-group` class, remove the `WP-group` class. - * $class_changes = [ + * // Add the `wp-block-group` class, remove the `wp-group` class. + * $classname_updates = [ * // Indexed by a comparable class name - * 'wp-block-group' => new WP_Class_Name_Operation( 'WP-block-group', WP_Class_Name_Operation::ADD ), - * 'wp-group' => new WP_Class_Name_Operation( 'WP-group', WP_Class_Name_Operation::REMOVE ) + * 'wp-block-group' => WP_HTML_Tag_Processor::ADD_CLASS, + * 'wp-group' => WP_HTML_Tag_Processor::REMOVE_CLASS * ]; * * @@ -331,6 +381,15 @@ class WP_HTML_Tag_Processor { */ private $classname_updates = array(); + /** + * Tracks a semantic location in the original HTML which + * shifts with updates as they are applied to the document. + * + * @since 6.2.0 + * @var WP_HTML_Span[] + */ + private $bookmarks = array(); + const ADD_CLASS = true; const REMOVE_CLASS = false; const SKIP_CLASS = null; @@ -365,6 +424,16 @@ class WP_HTML_Tag_Processor { */ private $attribute_updates = array(); + /** + * Tracks how many times we've performed a `seek()` + * so that we can prevent accidental infinite loops. + * + * @see seek + * @since 6.2.0 + * @var int + */ + private $seek_count = 0; + /** * Constructor. * @@ -397,6 +466,10 @@ public function next_tag( $query = null ) { $already_found = 0; do { + if ( $this->parsed_bytes >= strlen( $this->html ) ) { + return false; + } + /* * Unfortunately we can't try to search for only the tag name we want because that might * lead us to skip over other tags and lose track of our place. So we need to search for @@ -408,7 +481,16 @@ public function next_tag( $query = null ) { return false; } - $this->parse_tag_opener_attributes(); + while ( $this->parse_next_attribute() ) { + continue; + } + + $tag_ends_at = strpos( $this->html, '>', $this->parsed_bytes ); + if ( false === $tag_ends_at ) { + return false; + } + $this->tag_ends_at = $tag_ends_at; + $this->parsed_bytes = $tag_ends_at; if ( $this->matches() ) { ++$already_found; @@ -419,10 +501,15 @@ public function next_tag( $query = null ) { if ( 's' === $t || 'S' === $t || 't' === $t || 'T' === $t ) { $tag_name = $this->get_tag(); - if ( 'SCRIPT' === $tag_name ) { - $this->skip_script_data(); - } elseif ( 'TEXTAREA' === $tag_name || 'TITLE' === $tag_name ) { - $this->skip_rcdata( $tag_name ); + if ( 'SCRIPT' === $tag_name && ! $this->skip_script_data() ) { + $this->parsed_bytes = strlen( $this->html ); + return false; + } elseif ( + ( 'TEXTAREA' === $tag_name || 'TITLE' === $tag_name ) && + ! $this->skip_rcdata( $tag_name ) + ) { + $this->parsed_bytes = strlen( $this->html ); + return false; } } } while ( $already_found < $this->sought_match_offset ); @@ -430,6 +517,123 @@ public function next_tag( $query = null ) { return true; } + + /** + * Sets a bookmark in the HTML document. + * + * Bookmarks represent specific places or tokens in the HTML + * document, such as a tag opener or closer. When applying + * edits to a document, such as setting an attribute, the + * text offsets of that token may shift; the bookmark is + * kept updated with those shifts and remains stable unless + * the entire span of text in which the token sits is removed. + * + * Release bookmarks when they are no longer needed. + * + * Example: + * ``` + *

Surprising fact you may not know!

+ * ^ ^ + * \-|-- this `H2` opener bookmark tracks the token + * + *

Surprising fact you may no… + * ^ ^ + * \-|-- it shifts with edits + * ``` + * + * Bookmarks provide the ability to seek to a previously-scanned + * place in the HTML document. This avoids the need to re-scan + * the entire thing. + * + * Example: + * ``` + *
  • One
  • Two
  • Three
+ * ^^^^ + * want to note this last item + * + * $p = new WP_HTML_Tag_Processor( $html ); + * $in_list = false; + * while ( $p->next_tag( [ 'tag_closers' => $in_list ? 'visit' : 'skip' ] ) ) { + * if ( 'UL' === $p->get_tag() ) { + * if ( $p->is_tag_closer() ) { + * $in_list = false; + * $p->set_bookmark( 'resume' ); + * if ( $p->seek( 'last-li' ) ) { + * $p->add_class( 'last-li' ); + * } + * $p->seek( 'resume' ); + * $p->release_bookmark( 'last-li' ); + * $p->release_bookmark( 'resume' ); + * } else { + * $in_list = true; + * } + * } + * + * if ( 'LI' === $p->get_tag() ) { + * $p->set_bookmark( 'last-li' ); + * } + * } + * ``` + * + * Because bookmarks maintain their position they don't + * expose any internal offsets for the HTML document + * and can't be used with normal string functions. + * + * Because bookmarks allocate memory and require processing + * for every applied update they are limited and require + * a name. They should not be created inside a loop. + * + * Bookmarks are a powerful tool to enable complicated behavior; + * consider double-checking that you need this tool if you are + * reaching for it, as inappropriate use could lead to broken + * HTML structure or unwanted processing overhead. + * + * @param string $name Identifies this particular bookmark. + * @return false|void + * @throws Exception Throws on invalid bookmark name if WP_DEBUG set. + */ + public function set_bookmark( $name ) { + if ( null === $this->tag_name_starts_at ) { + return false; + } + + if ( ! array_key_exists( $name, $this->bookmarks ) && count( $this->bookmarks ) >= self::MAX_BOOKMARKS ) { + if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) { + throw new Exception( "Tried to jump to a non-existent HTML bookmark {$name}." ); + } + return false; + } + + $this->bookmarks[ $name ] = new WP_HTML_Span( + $this->tag_name_starts_at - 1, + $this->tag_ends_at + ); + + return true; + } + + + /** + * Removes a bookmark if you no longer need to use it. + * + * Releasing a bookmark frees up the small performance + * overhead they require, mainly in the form of compute + * costs when modifying the document. + * + * @param string $name Name of the bookmark to remove. + * @return bool + */ + public function release_bookmark( $name ) { + if ( ! array_key_exists( $name, $this->bookmarks ) ) { + return false; + } + + unset( $this->bookmarks[ $name ] ); + + return true; + } + + /** * Skips the contents of the title and textarea tags until an appropriate * tag closer is found. @@ -449,9 +653,9 @@ private function skip_rcdata( $tag_name ) { $at = strpos( $this->html, ' $doc_length ) { + if ( false === $at || ( $at + $tag_length ) >= $doc_length ) { $this->parsed_bytes = $doc_length; - return; + return false; } $at += 2; @@ -486,12 +690,17 @@ private function skip_rcdata( $tag_name ) { continue; } - $this->skip_tag_closer_attributes(); + while ( $this->parse_next_attribute() ) { + continue; + } $at = $this->parsed_bytes; + if ( $at >= strlen( $this->html ) ) { + return false; + } if ( '>' === $html[ $at ] || '/' === $html[ $at ] ) { ++$this->parsed_bytes; - return; + return true; } } @@ -586,6 +795,9 @@ private function skip_script_data() { * We also have to make sure we terminate the script tag opener/closer * to avoid making partial matches on strings like `= $doc_length ) { + continue; + } $at += 6; $c = $html[ $at ]; if ( ' ' !== $c && "\t" !== $c && "\r" !== $c && "\n" !== $c && '/' !== $c && '>' !== $c ) { @@ -605,11 +817,17 @@ private function skip_script_data() { if ( $is_closing ) { $this->parsed_bytes = $at; - $this->skip_tag_closer_attributes(); + if ( $this->parsed_bytes >= $doc_length ) { + return false; + } + + while ( $this->parse_next_attribute() ) { + continue; + } if ( '>' === $html[ $this->parsed_bytes ] ) { ++$this->parsed_bytes; - return; + return true; } } @@ -637,6 +855,13 @@ private function parse_next_tag() { return false; } + if ( '/' === $this->html[ $at + 1 ] ) { + $this->is_closing_tag = true; + $at++; + } else { + $this->is_closing_tag = false; + } + /* * HTML tag names must start with [a-zA-Z] otherwise they are not tags. * For example, "<3" is rendered as text, not a tag opener. This means @@ -658,6 +883,13 @@ private function parse_next_tag() { return true; } + // If we didn't find a tag opener, and we can't be + // transitioning into different markup states, then + // we can abort because there aren't any more tags. + if ( $at + 1 >= strlen( $html ) ) { + return false; + } + // parse_next_attribute() ) { - continue; - } - } - - /** - * Skips all attributes of the current tag. - * - * @since 6.2.0 - */ - private function skip_tag_closer_attributes() { - while ( $this->parse_next_attribute( 'tag-closer' ) ) { - continue; - } - } - /** * Parses the next attribute. * - * @param string $context tag-opener or tag-closer. * @since 6.2.0 */ - private function parse_next_attribute( $context = 'tag-opener' ) { + private function parse_next_attribute() { // Skip whitespace and slashes. $this->parsed_bytes += strspn( $this->html, " \t\f\r\n/", $this->parsed_bytes ); + if ( $this->parsed_bytes >= strlen( $this->html ) ) { + return false; + } /* * Treat the equal sign ("=") as a part of the attribute name if it is the @@ -793,20 +1005,29 @@ private function parse_next_attribute( $context = 'tag-opener' ) { : strcspn( $this->html, "=/> \t\f\r\n", $this->parsed_bytes ); // No attribute, just tag closer. - if ( 0 === $name_length ) { + if ( 0 === $name_length || $this->parsed_bytes + $name_length >= strlen( $this->html ) ) { return false; } $attribute_start = $this->parsed_bytes; $attribute_name = substr( $this->html, $attribute_start, $name_length ); $this->parsed_bytes += $name_length; + if ( $this->parsed_bytes >= strlen( $this->html ) ) { + return false; + } $this->skip_whitespace(); + if ( $this->parsed_bytes >= strlen( $this->html ) ) { + return false; + } $has_value = '=' === $this->html[ $this->parsed_bytes ]; if ( $has_value ) { ++$this->parsed_bytes; $this->skip_whitespace(); + if ( $this->parsed_bytes >= strlen( $this->html ) ) { + return false; + } switch ( $this->html[ $this->parsed_bytes ] ) { case "'": @@ -830,7 +1051,11 @@ private function parse_next_attribute( $context = 'tag-opener' ) { $attribute_end = $attribute_start + $name_length; } - if ( 'tag-opener' !== $context ) { + if ( $attribute_end >= strlen( $this->html ) ) { + return false; + } + + if ( $this->is_closing_tag ) { return true; } @@ -872,6 +1097,8 @@ private function after_tag() { $this->apply_attributes_updates(); $this->tag_name_starts_at = null; $this->tag_name_length = null; + $this->tag_ends_at = null; + $this->is_closing_tag = null; $this->attributes = array(); } @@ -1032,9 +1259,77 @@ private function apply_attributes_updates() { $this->updated_bytes = $diff->end; } + foreach ( $this->bookmarks as $bookmark ) { + /** + * As we loop through $this->attribute_updates, we keep comparing + * $bookmark->start and $bookmark->end to $diff->start. We can't + * change it and still expect the correct result, so let's accumulate + * the deltas separately and apply them all at once after the loop. + */ + $head_delta = 0; + $tail_delta = 0; + + foreach ( $this->attribute_updates as $diff ) { + $update_head = $bookmark->start >= $diff->start; + $update_tail = $bookmark->end >= $diff->start; + + if ( ! $update_head && ! $update_tail ) { + break; + } + + $delta = strlen( $diff->text ) - ( $diff->end - $diff->start ); + + if ( $update_head ) { + $head_delta += $delta; + } + + if ( $update_tail ) { + $tail_delta += $delta; + } + } + + $bookmark->start += $head_delta; + $bookmark->end += $tail_delta; + } + $this->attribute_updates = array(); } + /** + * Move the current pointer in the Tag Processor to a given bookmark's location. + * + * In order to prevent accidental infinite loops, there's a + * maximum limit on the number of times seek() can be called. + * + * @param string $bookmark_name Jump to the place in the document identified by this bookmark name. + * @return bool + * @throws Exception Throws on invalid bookmark name if WP_DEBUG set. + */ + public function seek( $bookmark_name ) { + if ( ! array_key_exists( $bookmark_name, $this->bookmarks ) ) { + if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) { + throw new Exception( 'Invalid bookmark name' ); + } + return false; + } + + if ( ++$this->seek_count > self::MAX_SEEK_OPS ) { + if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) { + throw new Exception( 'Too many calls to seek() - this can lead to performance issues.' ); + } + return false; + } + + // Flush out any pending updates to the document. + $this->get_updated_html(); + + // Point this tag processor before the sought tag opener and consume it. + $this->parsed_bytes = $this->bookmarks[ $bookmark_name ]->start; + $this->updated_bytes = $this->parsed_bytes; + $this->updated_html = substr( $this->html, 0, $this->updated_bytes ); + return $this->next_tag(); + } + /** * Sort function to arrange objects with a start property in ascending order. * @@ -1117,6 +1412,25 @@ public function get_tag() { return strtoupper( $tag_name ); } + /** + * Indicates if the current tag token is a tag closer. + * + * Example: + * + * $p = new WP_HTML_Tag_Processor( '
' ); + * $p->next_tag( [ 'tag_name' => 'div', 'tag_closers' => 'visit' ] ); + * $p->is_tag_closer() === false; + * + * $p->next_tag( [ 'tag_name' => 'div', 'tag_closers' => 'visit' ] ); + * $p->is_tag_closer() === true; + *
+ * + * @return bool + */ + public function is_tag_closer() { + return $this->is_closing_tag; + } + /** * Updates or creates a new attribute on the currently matched tag with the value passed. * @@ -1133,8 +1447,8 @@ public function get_tag() { * @throws Exception When WP_DEBUG is true and the attribute name is invalid. */ public function set_attribute( $name, $value ) { - if ( null === $this->tag_name_starts_at ) { - return; + if ( $this->is_closing_tag || null === $this->tag_name_starts_at ) { + return false; } /* @@ -1174,7 +1488,7 @@ public function set_attribute( $name, $value ) { ']~Ssu', $name ) ) { - if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) { + if ( WP_DEBUG ) { throw new Exception( 'Invalid attribute name' ); } @@ -1244,8 +1558,8 @@ public function set_attribute( $name, $value ) { * @param string $name The attribute name to remove. */ public function remove_attribute( $name ) { - if ( ! isset( $this->attributes[ $name ] ) ) { - return; + if ( $this->is_closing_tag || ! isset( $this->attributes[ $name ] ) ) { + return false; } /* @@ -1274,6 +1588,10 @@ public function remove_attribute( $name ) { * @param string $class_name The class name to add. */ public function add_class( $class_name ) { + if ( $this->is_closing_tag ) { + return false; + } + if ( null !== $this->tag_name_starts_at ) { $this->classname_updates[ $class_name ] = self::ADD_CLASS; } @@ -1287,6 +1605,10 @@ public function add_class( $class_name ) { * @param string $class_name The class name to remove. */ public function remove_class( $class_name ) { + if ( $this->is_closing_tag ) { + return false; + } + if ( null !== $this->tag_name_starts_at ) { $this->classname_updates[ $class_name ] = self::REMOVE_CLASS; } @@ -1312,45 +1634,31 @@ public function __toString() { * @return string The processed HTML. */ public function get_updated_html() { - // Short-circuit if there are no updates to apply. + // Short-circuit if there are no new updates to apply. if ( ! count( $this->classname_updates ) && ! count( $this->attribute_updates ) ) { return $this->updated_html . substr( $this->html, $this->updated_bytes ); } - /* - * Parsing is in progress – let's apply the attribute updates without moving on to the next tag. - * - * In practice: - * 1. Apply the attributes updates to the original HTML - * 2. Replace the original HTML with the updated HTML - * 3. Point this tag processor to the current tag name's end in that updated HTML - */ - - // Find tag name's end in the updated markup. - $markup_updated_up_to_a_tag_name_end = $this->updated_html . substr( $this->html, $this->updated_bytes, $this->tag_name_starts_at + $this->tag_name_length - $this->updated_bytes ); - $updated_tag_name_ends_at = strlen( $markup_updated_up_to_a_tag_name_end ); - $updated_tag_name_starts_at = $updated_tag_name_ends_at - $this->tag_name_length; + // Otherwise: apply the updates, rewind before the current tag, and parse it again. + $delta_between_updated_html_end_and_current_tag_end = substr( + $this->html, + $this->updated_bytes, + $this->tag_name_starts_at + $this->tag_name_length - $this->updated_bytes + ); + $updated_html_up_to_current_tag_name_end = $this->updated_html . $delta_between_updated_html_end_and_current_tag_end; - // Apply attributes updates. - $this->updated_html = $markup_updated_up_to_a_tag_name_end; - $this->updated_bytes = $this->tag_name_starts_at + $this->tag_name_length; + // 1. Apply the attributes updates to the original HTML $this->class_name_updates_to_attributes_updates(); $this->apply_attributes_updates(); - // Replace $this->html with the updated markup. - $this->html = $this->updated_html . substr( $this->html, $this->updated_bytes ); - - // Rewind this processor to the tag name's end. - $this->tag_name_starts_at = $updated_tag_name_starts_at; - $this->parsed_bytes = $updated_tag_name_ends_at; + // 2. Replace the original HTML with the updated HTML + $this->html = $this->updated_html . substr( $this->html, $this->updated_bytes ); + $this->updated_html = $updated_html_up_to_current_tag_name_end; + $this->updated_bytes = strlen( $this->updated_html ); - // Restore the previous version of the updated_html as we are not finished with the current_tag yet. - $this->updated_html = $markup_updated_up_to_a_tag_name_end; - $this->updated_bytes = $updated_tag_name_ends_at; - - // Parse the attributes in the updated markup. - $this->attributes = array(); - $this->parse_tag_opener_attributes(); + // 3. Point this tag processor at the original tag opener and consume it + $this->parsed_bytes = strlen( $updated_html_up_to_current_tag_name_end ) - $this->tag_name_length - 2; + $this->next_tag(); return $this->html; } @@ -1365,6 +1673,7 @@ public function get_updated_html() { * * @type string|null $tag_name Which tag to find, or `null` for "any tag." * @type string|null $class_name Tag must contain this class name to match. + * @type string $tag_closers "visit" or "skip": whether to stop on tag closers, e.g.

. * } */ private function parse_query( $query ) { @@ -1376,6 +1685,7 @@ private function parse_query( $query ) { $this->sought_tag_name = null; $this->sought_class_name = null; $this->sought_match_offset = 1; + $this->stop_on_tag_closers = false; // A single string value means "find the tag of this name". if ( is_string( $query ) ) { @@ -1399,6 +1709,10 @@ private function parse_query( $query ) { if ( isset( $query['match_offset'] ) && is_int( $query['match_offset'] ) && 0 < $query['match_offset'] ) { $this->sought_match_offset = $query['match_offset']; } + + if ( isset( $query['tag_closers'] ) ) { + $this->stop_on_tag_closers = 'visit' === $query['tag_closers']; + } } @@ -1410,6 +1724,10 @@ private function parse_query( $query ) { * @return boolean */ private function matches() { + if ( $this->is_closing_tag && ! $this->stop_on_tag_closers ) { + return false; + } + // Do we match a case-insensitive HTML tag name? if ( null !== $this->sought_tag_name ) { /* diff --git a/lib/experimental/html/index.php b/lib/experimental/html/index.php index e7d41f8cdf486..a31dbaf48c6b2 100644 --- a/lib/experimental/html/index.php +++ b/lib/experimental/html/index.php @@ -7,5 +7,6 @@ // All class files necessary for the HTML Tag Processor. require_once __DIR__ . '/class-wp-html-attribute-token.php'; +require_once __DIR__ . '/class-wp-html-span.php'; require_once __DIR__ . '/class-wp-html-text-replacement.php'; require_once __DIR__ . '/class-wp-html-tag-processor.php'; diff --git a/lib/experimental/kses.php b/lib/experimental/kses.php new file mode 100644 index 0000000000000..fd7531617939f --- /dev/null +++ b/lib/experimental/kses.php @@ -0,0 +1,66 @@ + 'gutenberg-zoomed-out-view', ) ); + add_settings_field( 'gutenberg-off-canvas-navigation-editor', __( 'Off canvas navigation editor ', 'gutenberg' ), @@ -63,6 +64,7 @@ function gutenberg_initialize_experiments_settings() { 'id' => 'gutenberg-off-canvas-navigation-editor', ) ); + add_settings_field( 'gutenberg-color-randomizer', __( 'Color randomizer ', 'gutenberg' ), @@ -75,6 +77,18 @@ function gutenberg_initialize_experiments_settings() { ) ); + add_settings_field( + 'gutenberg-block-inspector-tabs', + __( 'Block inspector tabs ', 'gutenberg' ), + 'gutenberg_display_experiment_field', + 'gutenberg-experiments', + 'gutenberg_experiments_section', + array( + 'label' => __( 'Test a new block inspector view splitting settings and appearance controls into tabs', 'gutenberg' ), + 'id' => 'gutenberg-block-inspector-tabs', + ) + ); + register_setting( 'gutenberg-experiments', 'gutenberg-experiments' @@ -112,22 +126,3 @@ function gutenberg_display_experiment_section() { (int) get_option( 'use_balanceTags' ) !== 1 || is_wp_version_compatible( '5.9' ), - ); - return array_merge( $settings, $experiments_settings ); -} - -add_filter( 'block_editor_settings_all', 'gutenberg_experiments_editor_settings' ); diff --git a/lib/load.php b/lib/load.php index 70dbf94453525..b634b85b977e3 100644 --- a/lib/load.php +++ b/lib/load.php @@ -35,25 +35,18 @@ function gutenberg_is_experiment_enabled( $name ) { require_once __DIR__ . '/experimental/class-wp-rest-block-editor-settings-controller.php'; } - // WordPress 6.0 compat. - require_once __DIR__ . '/compat/wordpress-6.0/class-gutenberg-rest-global-styles-controller.php'; - require_once __DIR__ . '/compat/wordpress-6.0/class-gutenberg-rest-pattern-directory-controller-6-0.php'; - require_once __DIR__ . '/compat/wordpress-6.0/class-gutenberg-rest-edit-site-export-controller.php'; - if ( ! class_exists( 'WP_REST_Block_Pattern_Categories_Controller' ) ) { - require_once __DIR__ . '/compat/wordpress-6.0/class-wp-rest-block-pattern-categories-controller.php'; - } - require_once __DIR__ . '/compat/wordpress-6.0/rest-api.php'; - // WordPress 6.1 compat. - require_once __DIR__ . '/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller.php'; + require_once __DIR__ . '/compat/wordpress-6.1/class-gutenberg-rest-block-patterns-controller-6-1.php'; require_once __DIR__ . '/compat/wordpress-6.1/class-gutenberg-rest-templates-controller.php'; require_once __DIR__ . '/compat/wordpress-6.1/rest-api.php'; // WordPress 6.2 compat. + require_once __DIR__ . '/compat/wordpress-6.2/class-gutenberg-rest-block-patterns-controller-6-2.php'; require_once __DIR__ . '/compat/wordpress-6.2/class-gutenberg-rest-block-pattern-categories-controller.php'; require_once __DIR__ . '/compat/wordpress-6.2/class-gutenberg-rest-pattern-directory-controller-6-2.php'; require_once __DIR__ . '/compat/wordpress-6.2/rest-api.php'; require_once __DIR__ . '/compat/wordpress-6.2/block-patterns.php'; + require_once __DIR__ . '/compat/wordpress-6.2/class-gutenberg-rest-global-styles-controller-6-2.php'; // Experimental. if ( ! class_exists( 'WP_Rest_Customizer_Nonces' ) ) { @@ -67,26 +60,9 @@ function gutenberg_is_experiment_enabled( $name ) { // Gutenberg plugin compat. require __DIR__ . '/compat/plugin/edit-site-routes-backwards-compat.php'; -// WordPress 6.0 compat. -require __DIR__ . '/compat/wordpress-6.0/block-gallery.php'; -require __DIR__ . '/compat/wordpress-6.0/block-editor-settings.php'; -require __DIR__ . '/compat/wordpress-6.0/get-global-styles-and-settings.php'; -require __DIR__ . '/compat/wordpress-6.0/render-svg-filters.php'; -require __DIR__ . '/compat/wordpress-6.0/post-lock.php'; -require __DIR__ . '/compat/wordpress-6.0/blocks.php'; -require __DIR__ . '/compat/wordpress-6.0/block-template-utils.php'; -require __DIR__ . '/compat/wordpress-6.0/functions.php'; -require __DIR__ . '/compat/wordpress-6.0/class-wp-theme-json-6-0.php'; -require __DIR__ . '/compat/wordpress-6.0/class-wp-theme-json-resolver-6-0.php'; -require __DIR__ . '/compat/wordpress-6.0/block-patterns.php'; -require __DIR__ . '/compat/wordpress-6.0/site-editor.php'; -require __DIR__ . '/compat/wordpress-6.0/edit-form-blocks.php'; -require __DIR__ . '/compat/wordpress-6.0/block-patterns-update.php'; -require __DIR__ . '/compat/wordpress-6.0/client-assets.php'; - // WordPress 6.1 compat. -require __DIR__ . '/compat/wordpress-6.1/blocks.php'; require __DIR__ . '/compat/wordpress-6.1/block-editor-settings.php'; +require __DIR__ . '/compat/wordpress-6.1/blocks.php'; require __DIR__ . '/compat/wordpress-6.1/persisted-preferences.php'; require __DIR__ . '/compat/wordpress-6.1/get-global-styles-and-settings.php'; require __DIR__ . '/compat/wordpress-6.1/class-wp-theme-json-data-gutenberg.php'; @@ -96,7 +72,6 @@ function gutenberg_is_experiment_enabled( $name ) { require __DIR__ . '/compat/wordpress-6.1/wp-theme-get-post-templates.php'; require __DIR__ . '/compat/wordpress-6.1/script-loader.php'; require __DIR__ . '/compat/wordpress-6.1/date-settings.php'; -require __DIR__ . '/compat/wordpress-6.1/block-patterns.php'; require __DIR__ . '/compat/wordpress-6.1/edit-form-blocks.php'; require __DIR__ . '/compat/wordpress-6.1/template-parts-screen.php'; require __DIR__ . '/compat/wordpress-6.1/theme.php'; @@ -107,6 +82,9 @@ function gutenberg_is_experiment_enabled( $name ) { require __DIR__ . '/compat/wordpress-6.2/default-filters.php'; require __DIR__ . '/compat/wordpress-6.2/class-wp-theme-json-resolver-6-2.php'; require __DIR__ . '/compat/wordpress-6.2/class-wp-theme-json-6-2.php'; +require __DIR__ . '/compat/wordpress-6.2/edit-form-blocks.php'; +require __DIR__ . '/compat/wordpress-6.2/site-editor.php'; +require __DIR__ . '/compat/wordpress-6.2/block-editor-settings.php'; // Experimental features. remove_action( 'plugins_loaded', '_wp_theme_json_webfonts_handler' ); // Turns off WP 6.0's stopgap handler for Webfonts API. @@ -123,6 +101,7 @@ function gutenberg_is_experiment_enabled( $name ) { require __DIR__ . '/experimental/webfonts.php'; require __DIR__ . '/experimental/navigation-theme-opt-in.php'; require __DIR__ . '/experimental/navigation-page.php'; +require __DIR__ . '/experimental/kses.php'; // Plugin specific code. require __DIR__ . '/blocks.php'; diff --git a/package-lock.json b/package-lock.json index 9f0f34dd3c5bf..66631f6f47804 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "gutenberg", - "version": "14.6.0-rc.1", + "version": "14.8.0-rc.1", "lockfileVersion": 1, "requires": true, "dependencies": { @@ -94,6 +94,12 @@ "tunnel": "0.0.6" } }, + "@adobe/css-tools": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.0.1.tgz", + "integrity": "sha512-+u76oB43nOHrF4DDWRLWDCtci7f3QJoEBigemIdIeTi1ODqjx6Tad9NCVnPRwewWlKkVab5PlK8DCtPTyX7S8g==", + "dev": true + }, "@axe-core/puppeteer": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/@axe-core/puppeteer/-/puppeteer-4.0.0.tgz", @@ -443,6 +449,11 @@ "@babel/types": "^7.16.0" } }, + "@babel/helper-string-parser": { + "version": "7.19.4", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.19.4.tgz", + "integrity": "sha512-nHtDoQcuqFmwYNYPz3Rah5ph2p8PFeFCsZk9A/48dPc/rGocJ5J3hAAZ7pb76VWX3fZKu+uEr/FhH5jLx7umrw==" + }, "@babel/helper-validator-identifier": { "version": "7.15.7", "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.15.7.tgz", @@ -525,7 +536,6 @@ "version": "7.16.0", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.16.0.tgz", "integrity": "sha512-nyYmIo7ZqKsY6P4lnVmBlxp9B3a96CscbLotlsNuktMHahkDwoPYEjXrZHU0Tj844Z9f1IthVxQln57mhkcExw==", - "dev": true, "requires": { "@babel/helper-plugin-utils": "^7.14.5", "@babel/helper-remap-async-to-generator": "^7.16.0", @@ -906,7 +916,6 @@ "version": "7.8.4", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", - "dev": true, "requires": { "@babel/helper-plugin-utils": "^7.8.0" } @@ -974,7 +983,6 @@ "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz", "integrity": "sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==", - "dev": true, "requires": { "@babel/helper-plugin-utils": "^7.8.3" } @@ -1279,7 +1287,6 @@ "version": "7.16.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.16.0.tgz", "integrity": "sha512-LogN88uO+7EhxWc8WZuQ8vxdSyVGxhkh8WTC3tzlT8LccMuQdA81e9SGV6zY7kY2LjDhhDOFdQVxdGwPyBCnvg==", - "dev": true, "requires": { "@babel/helper-create-regexp-features-plugin": "^7.16.0" } @@ -1423,6 +1430,7 @@ "version": "7.16.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.16.0.tgz", "integrity": "sha512-JAvGxgKuwS2PihiSFaDrp94XOzzTUeDeOQlcKzVAyaPap7BnZXK/lvMDiubkPTdotPKOIZq9xWXWnggUMYiExg==", + "dev": true, "requires": { "regenerator-transform": "^0.14.2" } @@ -1785,6 +1793,7 @@ "version": "1.0.3", "resolved": "https://registry.npmjs.org/@cnakazawa/watch/-/watch-1.0.3.tgz", "integrity": "sha512-r5160ogAvGyHsal38Kux7YYtodEKOj89RGb28ht1jh3SJb08VwRwAKKJL0bGb04Zd/3r9FL3BFIc3bBidYffCA==", + "dev": true, "requires": { "exec-sh": "^0.3.2", "minimist": "^1.2.0" @@ -2892,17 +2901,17 @@ } }, "@jest/create-cache-key-function": { - "version": "27.3.1", - "resolved": "https://registry.npmjs.org/@jest/create-cache-key-function/-/create-cache-key-function-27.3.1.tgz", - "integrity": "sha512-21lx0HRgkznc5Tc2WGiXVYQQ6Vdfohs6CkLV2FLogLRb52f6v9SiSIjTNflu23lzEmY4EalLgQLxCfhgvREV6w==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/create-cache-key-function/-/create-cache-key-function-27.5.1.tgz", + "integrity": "sha512-dmH1yW+makpTSURTy8VzdUwFnfQh1G8R+DxO2Ho2FFmBbKFEVm+3jWdvFhE2VqB/LATCTokkP0dotjyQyw5/AQ==", "requires": { - "@jest/types": "^27.2.5" + "@jest/types": "^27.5.1" }, "dependencies": { "@jest/types": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.4.2.tgz", - "integrity": "sha512-j35yw0PMTPpZsUoOBiuHzr1zTYoad1cVIE0ajEjcrJONxxrko/IRGKkXx3os0Nsi4Hu3+5VmDbVfq5WhG/pWAg==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.5.1.tgz", + "integrity": "sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw==", "requires": { "@types/istanbul-lib-coverage": "^2.0.0", "@types/istanbul-reports": "^3.0.0", @@ -2957,6 +2966,23 @@ } } }, + "@jest/expect-utils": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.2.1.tgz", + "integrity": "sha512-yr4aHNg5Z1CjKby5ozm7sKjgBlCOorlAoFcvrOQ/4rbZRfgZQdnmh7cth192PYIgiPZo2bBXvqdOApnAMWFJZg==", + "dev": true, + "requires": { + "jest-get-type": "^29.2.0" + }, + "dependencies": { + "jest-get-type": { + "version": "29.2.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.2.0.tgz", + "integrity": "sha512-uXNJlg8hKFEnDgFsrCjznB+sTxdkuqiCL6zMgA75qEbAJjJYTs9XPrvDctrEig2GDow22T/LvHgO57iJhXB/UA==", + "dev": true + } + } + }, "@jest/fake-timers": { "version": "27.4.2", "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-27.4.2.tgz", @@ -3750,14 +3776,12 @@ "@jridgewell/resolve-uri": { "version": "3.0.7", "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.0.7.tgz", - "integrity": "sha512-8cXDaBBHOr2pQ7j77Y6Vp5VDT2sIqWyWQ56TjEq4ih/a4iST3dItRe8Q9fp0rrIl9DoKhWQtUQz/YpOxLkXbNA==", - "dev": true + "integrity": "sha512-8cXDaBBHOr2pQ7j77Y6Vp5VDT2sIqWyWQ56TjEq4ih/a4iST3dItRe8Q9fp0rrIl9DoKhWQtUQz/YpOxLkXbNA==" }, "@jridgewell/set-array": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.1.tgz", - "integrity": "sha512-Ct5MqZkLGEXTVmQYbGtx9SVqD2fqwvdubdps5D3djjAkgkKwT918VNOz65pEHFaYTeWcukmJmH5SwsA9Tn2ObQ==", - "dev": true + "integrity": "sha512-Ct5MqZkLGEXTVmQYbGtx9SVqD2fqwvdubdps5D3djjAkgkKwT918VNOz65pEHFaYTeWcukmJmH5SwsA9Tn2ObQ==" }, "@jridgewell/source-map": { "version": "0.3.2", @@ -3772,14 +3796,12 @@ "@jridgewell/sourcemap-codec": { "version": "1.4.13", "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.13.tgz", - "integrity": "sha512-GryiOJmNcWbovBxTfZSF71V/mXbgcV3MewDe3kIMCLyIh5e7SKAeUZs+rMnJ8jkMolZ/4/VsdBmMrw3l+VdZ3w==", - "dev": true + "integrity": "sha512-GryiOJmNcWbovBxTfZSF71V/mXbgcV3MewDe3kIMCLyIh5e7SKAeUZs+rMnJ8jkMolZ/4/VsdBmMrw3l+VdZ3w==" }, "@jridgewell/trace-mapping": { "version": "0.3.13", "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.13.tgz", "integrity": "sha512-o1xbKhp9qnIAoHJSWd6KlCZfqslL4valSF81H8ImioOAxluWYWOpWkpyktY2vnt4tbrX9XYaxovq6cgowaJp2w==", - "dev": true, "requires": { "@jridgewell/resolve-uri": "^3.0.3", "@jridgewell/sourcemap-codec": "^1.4.10" @@ -6936,6 +6958,99 @@ "integrity": "sha512-H1rQc1ZOHANWBvPcW+JpGwr+juXSxM8Q8YCkm3GhZd8REu1fHR3z99CErO1p9pkcfcxZnMdIZdIsXkOHY0NilA==", "dev": true }, + "@motionone/animation": { + "version": "10.14.0", + "resolved": "https://registry.npmjs.org/@motionone/animation/-/animation-10.14.0.tgz", + "integrity": "sha512-h+1sdyBP8vbxEBW5gPFDnj+m2DCqdlAuf2g6Iafb1lcMnqjsRXWlPw1AXgvUMXmreyhqmPbJqoNfIKdytampRQ==", + "requires": { + "@motionone/easing": "^10.14.0", + "@motionone/types": "^10.14.0", + "@motionone/utils": "^10.14.0", + "tslib": "^2.3.1" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } + } + }, + "@motionone/dom": { + "version": "10.13.1", + "resolved": "https://registry.npmjs.org/@motionone/dom/-/dom-10.13.1.tgz", + "integrity": "sha512-zjfX+AGMIt/fIqd/SL1Lj93S6AiJsEA3oc5M9VkUr+Gz+juRmYN1vfvZd6MvEkSqEjwPQgcjN7rGZHrDB9APfQ==", + "requires": { + "@motionone/animation": "^10.13.1", + "@motionone/generators": "^10.13.1", + "@motionone/types": "^10.13.0", + "@motionone/utils": "^10.13.1", + "hey-listen": "^1.0.8", + "tslib": "^2.3.1" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } + } + }, + "@motionone/easing": { + "version": "10.14.0", + "resolved": "https://registry.npmjs.org/@motionone/easing/-/easing-10.14.0.tgz", + "integrity": "sha512-2vUBdH9uWTlRbuErhcsMmt1jvMTTqvGmn9fHq8FleFDXBlHFs5jZzHJT9iw+4kR1h6a4SZQuCf72b9ji92qNYA==", + "requires": { + "@motionone/utils": "^10.14.0", + "tslib": "^2.3.1" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } + } + }, + "@motionone/generators": { + "version": "10.14.0", + "resolved": "https://registry.npmjs.org/@motionone/generators/-/generators-10.14.0.tgz", + "integrity": "sha512-6kRHezoFfIjFN7pPpaxmkdZXD36tQNcyJe3nwVqwJ+ZfC0e3rFmszR8kp9DEVFs9QL/akWjuGPSLBI1tvz+Vjg==", + "requires": { + "@motionone/types": "^10.14.0", + "@motionone/utils": "^10.14.0", + "tslib": "^2.3.1" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } + } + }, + "@motionone/types": { + "version": "10.14.0", + "resolved": "https://registry.npmjs.org/@motionone/types/-/types-10.14.0.tgz", + "integrity": "sha512-3bNWyYBHtVd27KncnJLhksMFQ5o2MSdk1cA/IZqsHtA9DnRM1SYgN01CTcJ8Iw8pCXF5Ocp34tyAjY7WRpOJJQ==" + }, + "@motionone/utils": { + "version": "10.14.0", + "resolved": "https://registry.npmjs.org/@motionone/utils/-/utils-10.14.0.tgz", + "integrity": "sha512-sLWBLPzRqkxmOTRzSaD3LFQXCPHvDzyHJ1a3VP9PRzBxyVd2pv51/gMOsdAcxQ9n+MIeGJnxzXBYplUHKj4jkw==", + "requires": { + "@motionone/types": "^10.14.0", + "hey-listen": "^1.0.8", + "tslib": "^2.3.1" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } + } + }, "@mrmlnc/readdir-enhanced": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", @@ -8132,56 +8247,21 @@ "integrity": "sha512-O/ohFq1CAQLfoNc376Z3W6gvVcCJlje5AVk0JhsI8Q40hn+NXAWCnOM1bEePfC0uDMtp0/RCK6FotUvkQ6c4Zw==" }, "@react-native-community/blur": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/@react-native-community/blur/-/blur-3.6.0.tgz", - "integrity": "sha512-GtDBhpX2pQcjl4VopOC8FktrVufrEfYRwVeMQ2WWckqKIv2BdwvlvWvj88L1WmEdBr9UNcm3rtgz+d+YXkmirA==", - "requires": { - "prop-types": "^15.5.10" - } + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@react-native-community/blur/-/blur-4.2.0.tgz", + "integrity": "sha512-StgP5zQJOCHqDRjmcKnzVkJ920S6DYBKRJfigSUnlkNQp+HzZtVtyKq0j5a7x84NtHcV7j8Uy5mz1Lx9ZKRKfA==" }, - "@react-native-community/cli": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/@react-native-community/cli/-/cli-6.3.1.tgz", - "integrity": "sha512-UQ77AkGvPzdwJt6qhYXUyDMP1v2rdCcIlrhU48FOcAhGX+N/LCL9Cp/Ic6CkiiSHJdktbgiEEJ2srprXH8nzVg==", + "@react-native-community/cli-clean": { + "version": "8.0.4", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-clean/-/cli-clean-8.0.4.tgz", + "integrity": "sha512-IwS1M1NHg6+qL8PThZYMSIMYbZ6Zbx+lIck9PLBskbosFo24M3lCOflOl++Bggjakp6mR+sRXxLMexid/GeOsQ==", "requires": { - "@react-native-community/cli-debugger-ui": "^6.0.0-rc.0", - "@react-native-community/cli-hermes": "^6.3.0", - "@react-native-community/cli-plugin-metro": "^6.2.0", - "@react-native-community/cli-server-api": "^6.2.0", - "@react-native-community/cli-tools": "^6.2.0", - "@react-native-community/cli-types": "^6.0.0", - "appdirsjs": "^1.2.4", + "@react-native-community/cli-tools": "^8.0.4", "chalk": "^4.1.2", - "command-exists": "^1.2.8", - "commander": "^2.19.0", - "cosmiconfig": "^5.1.0", - "deepmerge": "^3.2.0", - "envinfo": "^7.7.2", "execa": "^1.0.0", - "find-up": "^4.1.0", - "fs-extra": "^8.1.0", - "glob": "^7.1.3", - "graceful-fs": "^4.1.3", - "joi": "^17.2.1", - "leven": "^3.1.0", - "lodash": "^4.17.15", - "minimist": "^1.2.0", - "node-stream-zip": "^1.9.1", - "ora": "^3.4.0", - "pretty-format": "^26.6.2", - "prompts": "^2.4.0", - "semver": "^6.3.0", - "serve-static": "^1.13.1", - "strip-ansi": "^5.2.0", - "sudo-prompt": "^9.0.0", - "wcwidth": "^1.0.1" + "prompts": "^2.4.0" }, "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==" - }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", @@ -8212,22 +8292,6 @@ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" }, - "commander": { - "version": "2.20.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", - "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" - }, - "cosmiconfig": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", - "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", - "requires": { - "import-fresh": "^2.0.0", - "is-directory": "^0.3.1", - "js-yaml": "^3.13.1", - "parse-json": "^4.0.0" - } - }, "cross-spawn": { "version": "6.0.5", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", @@ -8238,25 +8302,8 @@ "semver": "^5.5.0", "shebang-command": "^1.2.0", "which": "^1.2.9" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" - } } }, - "deepmerge": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-3.3.0.tgz", - "integrity": "sha512-GRQOafGHwMHpjPx9iCvTgpu9NojZ49q794EEL94JVEw6VaeA8XTUyBKvAkOOjBX9oJNiV6G3P+T+tihFjo2TqA==" - }, - "envinfo": { - "version": "7.8.1", - "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.8.1.tgz", - "integrity": "sha512-/o+BXHmB7ocbHEAs6F2EnG0ogybVVUdkRunTT2glZU9XAaGmhqskrvKwqXuDfNjEO0LZKWdejEEpnq8aM0tOaw==" - }, "execa": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", @@ -8271,15 +8318,6 @@ "strip-eof": "^1.0.0" } }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, "get-stream": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", @@ -8288,217 +8326,139 @@ "pump": "^3.0.0" } }, - "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" - } - }, "has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, - "import-fresh": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", - "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", "requires": { - "caller-path": "^2.0.0", - "resolve-from": "^3.0.0" + "end-of-stream": "^1.1.0", + "once": "^1.3.1" } }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", "requires": { - "p-locate": "^4.1.0" + "has-flag": "^4.0.0" } - }, - "ora": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/ora/-/ora-3.4.0.tgz", - "integrity": "sha512-eNwHudNbO1folBP3JsZ19v9azXWtQZjICdr3Q0TDPIaeBQ3mXLrh54wM+er0+hSp+dWKf+Z8KM58CYzEyIYxYg==", + } + } + }, + "@react-native-community/cli-config": { + "version": "8.0.6", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-config/-/cli-config-8.0.6.tgz", + "integrity": "sha512-mjVpVvdh8AviiO8xtqeX+BkjqE//NMDnISwsLWSJUfNCwTAPmdR8PGbhgP5O4hWHyJ3WkepTopl0ya7Tfi3ifw==", + "requires": { + "@react-native-community/cli-tools": "^8.0.4", + "cosmiconfig": "^5.1.0", + "deepmerge": "^3.2.0", + "glob": "^7.1.3", + "joi": "^17.2.1" + }, + "dependencies": { + "cosmiconfig": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", + "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", "requires": { - "chalk": "^2.4.2", - "cli-cursor": "^2.1.0", - "cli-spinners": "^2.0.0", - "log-symbols": "^2.2.0", - "strip-ansi": "^5.2.0", - "wcwidth": "^1.0.1" - }, - "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "requires": { - "color-convert": "^1.9.0" - } - }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - } - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" - }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" - }, - "supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "requires": { - "has-flag": "^3.0.0" - } - } + "import-fresh": "^2.0.0", + "is-directory": "^0.3.1", + "js-yaml": "^3.13.1", + "parse-json": "^4.0.0" } }, - "p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "deepmerge": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-3.3.0.tgz", + "integrity": "sha512-GRQOafGHwMHpjPx9iCvTgpu9NojZ49q794EEL94JVEw6VaeA8XTUyBKvAkOOjBX9oJNiV6G3P+T+tihFjo2TqA==" + }, + "glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", "requires": { - "p-try": "^2.0.0" + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" } }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "import-fresh": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", + "integrity": "sha512-eZ5H8rcgYazHbKC3PG4ClHNykCSxtAhxSSEM+2mb+7evD2CKF5V7c0dNum7AdpDh0ZdICwZY9sRSn8f+KH96sg==", "requires": { - "p-limit": "^2.2.0" + "caller-path": "^2.0.0", + "resolve-from": "^3.0.0" } }, - "p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==" + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "requires": { + "brace-expansion": "^1.1.7" + } }, "parse-json": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", - "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "integrity": "sha512-aOIos8bujGN93/8Ox/jPLh7RwVnPEysynVFE+fQZyg6jKELEHwzgKdLRFHUgXJL6kylijVSBC4BvN9OmsB48Rw==", "requires": { "error-ex": "^1.3.1", "json-parse-better-errors": "^1.0.1" } - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==" - }, - "prompts": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", - "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", - "requires": { - "kleur": "^3.0.3", - "sisteransi": "^1.0.5" - } - }, - "pump": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", - "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", - "requires": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" - } - }, - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==" - }, - "sisteransi": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", - "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==" - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "requires": { - "ansi-regex": "^4.1.0" - } - }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "requires": { - "has-flag": "^4.0.0" - } } } }, "@react-native-community/cli-debugger-ui": { - "version": "6.0.0-rc.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-debugger-ui/-/cli-debugger-ui-6.0.0-rc.0.tgz", - "integrity": "sha512-achYcPPoWa9D02C5tn6TBzjeY443wQTyx37urptc75JpZ7gR5YHsDyIEEWa3DDYp1va9zx/iGg+uZ/hWw07GAw==", + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-debugger-ui/-/cli-debugger-ui-8.0.0.tgz", + "integrity": "sha512-u2jq06GZwZ9sRERzd9FIgpW6yv4YOW4zz7Ym/B8eSzviLmy3yI/8mxJtvlGW+J8lBsfMcQoqJpqI6Rl1nZy9yQ==", "requires": { "serve-static": "^1.13.1" } }, - "@react-native-community/cli-hermes": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-hermes/-/cli-hermes-6.3.0.tgz", - "integrity": "sha512-Uhbm9bubyZLZ12vFCIfWbE/Qi3SBTbYIN/TC08EudTLhv/KbPomCQnmFsnJ7AXQFuOZJs73mBxoEAYSbRbwyVA==", + "@react-native-community/cli-doctor": { + "version": "8.0.6", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-doctor/-/cli-doctor-8.0.6.tgz", + "integrity": "sha512-ZQqyT9mJMVeFEVIwj8rbDYGCA2xXjJfsQjWk2iTRZ1CFHfhPSUuUiG8r6mJmTinAP9t+wYcbbIYzNgdSUKnDMw==", "requires": { - "@react-native-community/cli-platform-android": "^6.3.0", - "@react-native-community/cli-tools": "^6.2.0", + "@react-native-community/cli-config": "^8.0.6", + "@react-native-community/cli-platform-ios": "^8.0.6", + "@react-native-community/cli-tools": "^8.0.4", "chalk": "^4.1.2", + "command-exists": "^1.2.8", + "envinfo": "^7.7.2", + "execa": "^1.0.0", "hermes-profile-transformer": "^0.0.6", - "ip": "^1.1.5" + "ip": "^1.1.5", + "node-stream-zip": "^1.9.1", + "ora": "^5.4.1", + "prompts": "^2.4.0", + "semver": "^6.3.0", + "strip-ansi": "^5.2.0", + "sudo-prompt": "^9.0.0", + "wcwidth": "^1.0.1" }, "dependencies": { - "@react-native-community/cli-platform-android": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-platform-android/-/cli-platform-android-6.3.0.tgz", - "integrity": "sha512-d5ufyYcvrZoHznYm5bjBXaiHIJv552t5gYtQpnUsxBhHSQ8QlaNmlLUyeSPRDfOw4ND9b0tPHqs4ufwx6vp/fQ==", - "requires": { - "@react-native-community/cli-tools": "^6.2.0", - "chalk": "^4.1.2", - "execa": "^1.0.0", - "fs-extra": "^8.1.0", - "glob": "^7.1.3", - "jetifier": "^1.6.2", - "lodash": "^4.17.15", - "logkitty": "^0.7.1", - "slash": "^3.0.0", - "xmldoc": "^1.1.2" - } + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" }, "ansi-styles": { "version": "4.3.0", @@ -8517,6 +8477,19 @@ "supports-color": "^7.1.0" } }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-spinners": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", + "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==" + }, "color-convert": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", @@ -8540,6 +8513,13 @@ "semver": "^5.5.0", "shebang-command": "^1.2.0", "which": "^1.2.9" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + } } }, "execa": { @@ -8564,24 +8544,50 @@ "pump": "^3.0.0" } }, - "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" - } - }, "has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==" + }, + "onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "requires": { + "mimic-fn": "^2.1.0" + } + }, + "ora": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz", + "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==", + "requires": { + "bl": "^4.1.0", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.5.0", + "is-interactive": "^1.0.0", + "is-unicode-supported": "^0.1.0", + "log-symbols": "^4.1.0", + "strip-ansi": "^6.0.0", + "wcwidth": "^1.0.1" + }, + "dependencies": { + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "requires": { + "ansi-regex": "^5.0.1" + } + } + } + }, "pump": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", @@ -8591,10 +8597,91 @@ "once": "^1.3.1" } }, + "restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "requires": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + } + }, "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==" + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "requires": { + "ansi-regex": "^4.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==" + } + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "@react-native-community/cli-hermes": { + "version": "8.0.5", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-hermes/-/cli-hermes-8.0.5.tgz", + "integrity": "sha512-Zm0wM6SfgYAEX1kfJ1QBvTayabvh79GzmjHyuSnEROVNPbl4PeCG4WFbwy489tGwOP9Qx9fMT5tRIFCD8bp6/g==", + "requires": { + "@react-native-community/cli-platform-android": "^8.0.5", + "@react-native-community/cli-tools": "^8.0.4", + "chalk": "^4.1.2", + "hermes-profile-transformer": "^0.0.6", + "ip": "^1.1.5" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, "supports-color": { "version": "7.2.0", @@ -8607,11 +8694,11 @@ } }, "@react-native-community/cli-platform-android": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-platform-android/-/cli-platform-android-6.2.0.tgz", - "integrity": "sha512-QLxwClcbxVhuIGsQiIpqRnoJzRdpN2B+y/Yt2OGgDHXGbuOXulgt4D+8AhvZXrB4jyAcEUlFg/048v3RGQQudw==", + "version": "8.0.5", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-platform-android/-/cli-platform-android-8.0.5.tgz", + "integrity": "sha512-z1YNE4T1lG5o9acoQR1GBvf7mq6Tzayqo/za5sHVSOJAC9SZOuVN/gg/nkBa9a8n5U7qOMFXfwhTMNqA474gXA==", "requires": { - "@react-native-community/cli-tools": "^6.2.0", + "@react-native-community/cli-tools": "^8.0.4", "chalk": "^4.1.2", "execa": "^1.0.0", "fs-extra": "^8.1.0", @@ -8619,8 +8706,7 @@ "jetifier": "^1.6.2", "lodash": "^4.17.15", "logkitty": "^0.7.1", - "slash": "^3.0.0", - "xmldoc": "^1.1.2" + "slash": "^3.0.0" }, "dependencies": { "ansi-styles": { @@ -8688,14 +8774,14 @@ } }, "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", "requires": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", - "minimatch": "^3.0.4", + "minimatch": "^3.1.1", "once": "^1.3.0", "path-is-absolute": "^1.0.0" } @@ -8705,6 +8791,14 @@ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "requires": { + "brace-expansion": "^1.1.7" + } + }, "pump": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", @@ -8730,24 +8824,24 @@ } }, "@react-native-community/cli-platform-ios": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-platform-ios/-/cli-platform-ios-6.2.0.tgz", - "integrity": "sha512-k15MhExxLiLDDZOeuPgvTxbp0CsoLQQpk2Du0HjZDePqqWcKJylQqMZru1o8HuQHPcEr+b71HIs5V+lKyFYpfg==", + "version": "8.0.6", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-platform-ios/-/cli-platform-ios-8.0.6.tgz", + "integrity": "sha512-CMR6mu/LVx6JVfQRDL9uULsMirJT633bODn+IrYmrwSz250pnhON16We8eLPzxOZHyDjm7JPuSgHG3a/BPiRuQ==", "requires": { - "@react-native-community/cli-tools": "^6.2.0", + "@react-native-community/cli-tools": "^8.0.4", "chalk": "^4.1.2", + "execa": "^1.0.0", "glob": "^7.1.3", "js-yaml": "^3.13.1", "lodash": "^4.17.15", - "ora": "^3.4.0", - "plist": "^3.0.2", - "xcode": "^2.0.0" + "ora": "^5.4.1", + "plist": "^3.0.2" }, "dependencies": { "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==" + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" }, "ansi-styles": { "version": "4.3.0", @@ -8766,6 +8860,19 @@ "supports-color": "^7.1.0" } }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-spinners": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", + "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==" + }, "color-convert": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", @@ -8779,15 +8886,49 @@ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "execa": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", + "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "requires": { + "cross-spawn": "^6.0.0", + "get-stream": "^4.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + } + }, + "get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "requires": { + "pump": "^3.0.0" + } + }, "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", "requires": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", - "minimatch": "^3.0.4", + "minimatch": "^3.1.1", "once": "^1.3.0", "path-is-absolute": "^1.0.0" } @@ -8797,71 +8938,72 @@ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==" + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "requires": { + "mimic-fn": "^2.1.0" + } + }, "ora": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/ora/-/ora-3.4.0.tgz", - "integrity": "sha512-eNwHudNbO1folBP3JsZ19v9azXWtQZjICdr3Q0TDPIaeBQ3mXLrh54wM+er0+hSp+dWKf+Z8KM58CYzEyIYxYg==", + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz", + "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==", "requires": { - "chalk": "^2.4.2", - "cli-cursor": "^2.1.0", - "cli-spinners": "^2.0.0", - "log-symbols": "^2.2.0", - "strip-ansi": "^5.2.0", + "bl": "^4.1.0", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.5.0", + "is-interactive": "^1.0.0", + "is-unicode-supported": "^0.1.0", + "log-symbols": "^4.1.0", + "strip-ansi": "^6.0.0", "wcwidth": "^1.0.1" - }, - "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "requires": { - "color-convert": "^1.9.0" - } - }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - } - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" - }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" - }, - "supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "requires": { - "has-flag": "^3.0.0" - } - } } }, + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "requires": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + } + }, + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + }, "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", "requires": { - "ansi-regex": "^4.1.0" + "ansi-regex": "^5.0.1" } }, "supports-color": { @@ -8875,19 +9017,19 @@ } }, "@react-native-community/cli-plugin-metro": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-plugin-metro/-/cli-plugin-metro-6.2.0.tgz", - "integrity": "sha512-JfmzuFNzOr+dFTUQJo1rV0t87XAqgHRTMYXNleQVt8otOVCk1FSCgKlgqMdvQc/FCx2ZjoMWEEV/g0LrPI8Etw==", + "version": "8.0.4", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-plugin-metro/-/cli-plugin-metro-8.0.4.tgz", + "integrity": "sha512-UWzY1eMcEr/6262R2+d0Is5M3L/7Y/xXSDIFMoc5Rv5Wucl3hJM/TxHXmByvHpuJf6fJAfqOskyt4bZCvbI+wQ==", "requires": { - "@react-native-community/cli-server-api": "^6.2.0", - "@react-native-community/cli-tools": "^6.2.0", + "@react-native-community/cli-server-api": "^8.0.4", + "@react-native-community/cli-tools": "^8.0.4", "chalk": "^4.1.2", - "metro": "^0.66.1", - "metro-config": "^0.66.1", - "metro-core": "^0.66.1", - "metro-react-native-babel-transformer": "^0.66.1", - "metro-resolver": "^0.66.1", - "metro-runtime": "^0.66.1", + "metro": "^0.70.1", + "metro-config": "^0.70.1", + "metro-core": "^0.70.1", + "metro-react-native-babel-transformer": "^0.70.1", + "metro-resolver": "^0.70.1", + "metro-runtime": "^0.70.1", "readline": "^1.3.0" }, "dependencies": { @@ -8926,6 +9068,112 @@ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, + "hermes-parser": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.6.0.tgz", + "integrity": "sha512-Vf58jBZca2+QBLR9h7B7mdg8oFz2g5ILz1iVouZ5DOrOrAfBmPfJjdjDT8jrO0f+iJ4/hSRrQHqHIjSnTaLUDQ==", + "requires": { + "hermes-estree": "0.6.0" + } + }, + "metro-react-native-babel-preset": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.70.3.tgz", + "integrity": "sha512-4Nxc1zEiHEu+GTdEMEsHnRgfaBkg8f/Td3+FcQ8NTSvs+xL3LBrQy6N07idWSQZHIdGFf+tTHvRfSIWLD8u8Tg==", + "requires": { + "@babel/core": "^7.14.0", + "@babel/plugin-proposal-async-generator-functions": "^7.0.0", + "@babel/plugin-proposal-class-properties": "^7.0.0", + "@babel/plugin-proposal-export-default-from": "^7.0.0", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.0.0", + "@babel/plugin-proposal-object-rest-spread": "^7.0.0", + "@babel/plugin-proposal-optional-catch-binding": "^7.0.0", + "@babel/plugin-proposal-optional-chaining": "^7.0.0", + "@babel/plugin-syntax-dynamic-import": "^7.0.0", + "@babel/plugin-syntax-export-default-from": "^7.0.0", + "@babel/plugin-syntax-flow": "^7.2.0", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.0.0", + "@babel/plugin-syntax-optional-chaining": "^7.0.0", + "@babel/plugin-transform-arrow-functions": "^7.0.0", + "@babel/plugin-transform-async-to-generator": "^7.0.0", + "@babel/plugin-transform-block-scoping": "^7.0.0", + "@babel/plugin-transform-classes": "^7.0.0", + "@babel/plugin-transform-computed-properties": "^7.0.0", + "@babel/plugin-transform-destructuring": "^7.0.0", + "@babel/plugin-transform-exponentiation-operator": "^7.0.0", + "@babel/plugin-transform-flow-strip-types": "^7.0.0", + "@babel/plugin-transform-function-name": "^7.0.0", + "@babel/plugin-transform-literals": "^7.0.0", + "@babel/plugin-transform-modules-commonjs": "^7.0.0", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.0.0", + "@babel/plugin-transform-parameters": "^7.0.0", + "@babel/plugin-transform-react-display-name": "^7.0.0", + "@babel/plugin-transform-react-jsx": "^7.0.0", + "@babel/plugin-transform-react-jsx-self": "^7.0.0", + "@babel/plugin-transform-react-jsx-source": "^7.0.0", + "@babel/plugin-transform-runtime": "^7.0.0", + "@babel/plugin-transform-shorthand-properties": "^7.0.0", + "@babel/plugin-transform-spread": "^7.0.0", + "@babel/plugin-transform-sticky-regex": "^7.0.0", + "@babel/plugin-transform-template-literals": "^7.0.0", + "@babel/plugin-transform-typescript": "^7.5.0", + "@babel/plugin-transform-unicode-regex": "^7.0.0", + "@babel/template": "^7.0.0", + "react-refresh": "^0.4.0" + } + }, + "metro-react-native-babel-transformer": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-transformer/-/metro-react-native-babel-transformer-0.70.3.tgz", + "integrity": "sha512-WKBU6S/G50j9cfmFM4k4oRYprd8u3qjleD4so1E2zbTNILg+gYla7ZFGCAvi2G0ZcqS2XuGCR375c2hF6VVvwg==", + "requires": { + "@babel/core": "^7.14.0", + "babel-preset-fbjs": "^3.4.0", + "hermes-parser": "0.6.0", + "metro-babel-transformer": "0.70.3", + "metro-react-native-babel-preset": "0.70.3", + "metro-source-map": "0.70.3", + "nullthrows": "^1.1.1" + } + }, + "metro-source-map": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.70.3.tgz", + "integrity": "sha512-zsYtZGrwRbbGEFHtmMqqeCH9K9aTGNVPsurMOWCUeQA3VGyVGXPGtLMC+CdAM9jLpUyg6jw2xh0esxi+tYH7Uw==", + "requires": { + "@babel/traverse": "^7.14.0", + "@babel/types": "^7.0.0", + "invariant": "^2.2.4", + "metro-symbolicate": "0.70.3", + "nullthrows": "^1.1.1", + "ob1": "0.70.3", + "source-map": "^0.5.6", + "vlq": "^1.0.0" + } + }, + "metro-symbolicate": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.70.3.tgz", + "integrity": "sha512-JTYkF1dpeDUssQ84juE1ycnhHki2ylJBBdJE1JHtfu5oC+z1ElDbBdPHq90Uvt8HbRov/ZAnxvv7Zy6asS+WCA==", + "requires": { + "invariant": "^2.2.4", + "metro-source-map": "0.70.3", + "nullthrows": "^1.1.1", + "source-map": "^0.5.6", + "through2": "^2.0.1", + "vlq": "^1.0.0" + } + }, + "ob1": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.70.3.tgz", + "integrity": "sha512-Vy9GGhuXgDRY01QA6kdhToPd8AkLdLpX9GjH5kpqluVqTu70mgOm7tpGoJDZGaNbr9nJlJgnipqHJQRPORixIQ==" + }, + "react-refresh": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.4.3.tgz", + "integrity": "sha512-Hwln1VNuGl/6bVwnd0Xdn1e84gT/8T9aYNL+HAKDArLCS7LWjwr7StE30IEYbIkx0Vi3vs+coQxe+SQDbGbbpA==" + }, "supports-color": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", @@ -8937,47 +9185,50 @@ } }, "@react-native-community/cli-server-api": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-server-api/-/cli-server-api-6.2.0.tgz", - "integrity": "sha512-OnbnYclhoDpjge33QO5Slhfn0DsmLzzAgyrSCnb24HhSqwq7ObjMHaLpoEhpajzLG71wq5oKh0APEQjiL4Mknw==", + "version": "8.0.4", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-server-api/-/cli-server-api-8.0.4.tgz", + "integrity": "sha512-Orr14njx1E70CVrUA8bFdl+mrnbuXUjf1Rhhm0RxUadFpvkHuOi5dh8Bryj2MKtf8eZrpEwZ7tuQPhJEULW16A==", "requires": { - "@react-native-community/cli-debugger-ui": "^6.0.0-rc.0", - "@react-native-community/cli-tools": "^6.2.0", + "@react-native-community/cli-debugger-ui": "^8.0.0", + "@react-native-community/cli-tools": "^8.0.4", "compression": "^1.7.1", "connect": "^3.6.5", "errorhandler": "^1.5.0", - "nocache": "^2.1.0", + "nocache": "^3.0.1", "pretty-format": "^26.6.2", "serve-static": "^1.13.1", - "ws": "^1.1.0" + "ws": "^7.5.1" }, "dependencies": { "ws": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/ws/-/ws-1.1.5.tgz", - "integrity": "sha512-o3KqipXNUdS7wpQzBHSe180lBGO60SoK0yVo3CYJgb2MkobuWuBX6dhkYP5ORCLd55y+SaflMOV5fqAB53ux4w==", - "requires": { - "options": ">=0.0.5", - "ultron": "1.0.x" - } + "version": "7.5.9", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.9.tgz", + "integrity": "sha512-F+P9Jil7UiSKSkppIiD94dN07AwvFixvLIj1Og1Rl9GGMuNipJnV9JzjD6XuqmAeiswGvUmNLjr5cFuXwNS77Q==" } } }, "@react-native-community/cli-tools": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-tools/-/cli-tools-6.2.1.tgz", - "integrity": "sha512-7RbOkZLT/3YG8CAYYM70ajRKIOgVxK/b4t9KNsPq+2uen99MGezfeglC8s1cs3vBNVVxCo0a2JbXg18bUd8eqA==", + "version": "8.0.4", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-tools/-/cli-tools-8.0.4.tgz", + "integrity": "sha512-ePN9lGxh6LRFiotyddEkSmuqpQhnq2iw9oiXYr4EFWpIEy0yCigTuSTiDF68+c8M9B+7bTwkRpz/rMPC4ViO5Q==", "requires": { "appdirsjs": "^1.2.4", "chalk": "^4.1.2", + "find-up": "^5.0.0", "lodash": "^4.17.15", "mime": "^2.4.1", "node-fetch": "^2.6.0", "open": "^6.2.0", + "ora": "^5.4.1", "semver": "^6.3.0", "shell-quote": "^1.7.3" }, "dependencies": { + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" + }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", @@ -8995,6 +9246,19 @@ "supports-color": "^7.1.0" } }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-spinners": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", + "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==" + }, "color-convert": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", @@ -9008,21 +9272,105 @@ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" }, + "find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "requires": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + } + }, "has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" }, + "locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "requires": { + "p-locate": "^5.0.0" + } + }, "mime": { "version": "2.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-2.6.0.tgz", "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==" }, + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==" + }, + "onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "requires": { + "mimic-fn": "^2.1.0" + } + }, + "ora": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz", + "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==", + "requires": { + "bl": "^4.1.0", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.5.0", + "is-interactive": "^1.0.0", + "is-unicode-supported": "^0.1.0", + "log-symbols": "^4.1.0", + "strip-ansi": "^6.0.0", + "wcwidth": "^1.0.1" + } + }, + "p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "requires": { + "yocto-queue": "^0.1.0" + } + }, + "p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "requires": { + "p-limit": "^3.0.2" + } + }, + "path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==" + }, + "restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "requires": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + } + }, "semver": { "version": "6.3.0", "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==" }, + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "requires": { + "ansi-regex": "^5.0.1" + } + }, "supports-color": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", @@ -9034,49 +9382,11 @@ } }, "@react-native-community/cli-types": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/@react-native-community/cli-types/-/cli-types-6.0.0.tgz", - "integrity": "sha512-K493Fk2DMJC0ZM8s8gnfseKxGasIhuDaCUDeLZcoCSFlrjKEuEs1BKKEJiev0CARhKEXKOyyp/uqYM9nWhisNw==", + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/@react-native-community/cli-types/-/cli-types-8.0.0.tgz", + "integrity": "sha512-1lZS1PEvMlFaN3Se1ksyoFWzMjk+YfKi490GgsqKJln9gvFm8tqVPdnXttI5Uf2DQf3BMse8Bk8dNH4oV6Ewow==", "requires": { - "ora": "^3.4.0" - }, - "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==" - }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - } - }, - "ora": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/ora/-/ora-3.4.0.tgz", - "integrity": "sha512-eNwHudNbO1folBP3JsZ19v9azXWtQZjICdr3Q0TDPIaeBQ3mXLrh54wM+er0+hSp+dWKf+Z8KM58CYzEyIYxYg==", - "requires": { - "chalk": "^2.4.2", - "cli-cursor": "^2.1.0", - "cli-spinners": "^2.0.0", - "log-symbols": "^2.2.0", - "strip-ansi": "^5.2.0", - "wcwidth": "^1.0.1" - } - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "requires": { - "ansi-regex": "^4.1.0" - } - } + "joi": "^17.2.1" } }, "@react-native-community/slider": { @@ -9094,9 +9404,9 @@ "integrity": "sha512-KrwSpS1tKI70wuKl68DwJZYEvXktDHdZMG0k2AXD/rJVSlB23/X2CB2cutVR0HwNMJIal9HOUOBB2rVfa6UGtQ==" }, "@react-native/normalize-color": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@react-native/normalize-color/-/normalize-color-1.0.0.tgz", - "integrity": "sha512-xUNRvNmCl3UGCPbbHvfyFMnpvLPoOjDCcp5bT9m2k+TF/ZBklEQwhPZlkrxRx2NhgFh1X3a5uL7mJ7ZR+8G7Qg==" + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@react-native/normalize-color/-/normalize-color-2.0.0.tgz", + "integrity": "sha512-Wip/xsc5lw8vsBlmY2MO/gFLp3MvuZ2baBZjDeTjjndMgM0h5sxz7AZR62RDPGgstp8Np7JzjvVqVT7tpFZqsw==" }, "@react-native/polyfills": { "version": "2.0.0", @@ -9831,87 +10141,6 @@ } } }, - "@storybook/addon-storysource": { - "version": "6.5.7", - "resolved": "https://registry.npmjs.org/@storybook/addon-storysource/-/addon-storysource-6.5.7.tgz", - "integrity": "sha512-4cquq0Oqf9skPdIYkVtO5lAPS7+83TyY4vT52HQ5LSpSekZELQkRxN793jNMQdu2IGQ/l9WqvyzFGaN62bRg7Q==", - "dev": true, - "requires": { - "@storybook/addons": "6.5.7", - "@storybook/api": "6.5.7", - "@storybook/client-logger": "6.5.7", - "@storybook/components": "6.5.7", - "@storybook/router": "6.5.7", - "@storybook/source-loader": "6.5.7", - "@storybook/theming": "6.5.7", - "core-js": "^3.8.2", - "estraverse": "^5.2.0", - "loader-utils": "^2.0.0", - "prop-types": "^15.7.2", - "react-syntax-highlighter": "^15.4.5", - "regenerator-runtime": "^0.13.7" - }, - "dependencies": { - "estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", - "dev": true - }, - "parse-entities": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/parse-entities/-/parse-entities-2.0.0.tgz", - "integrity": "sha512-kkywGpCcRYhqQIchaWqZ875wzpS/bMKhz5HnN3p7wveJTkTtyAB/AlnS0f8DFSqYW1T82t6yEAkEcB+A1I3MbQ==", - "dev": true, - "requires": { - "character-entities": "^1.0.0", - "character-entities-legacy": "^1.0.0", - "character-reference-invalid": "^1.0.0", - "is-alphanumerical": "^1.0.0", - "is-decimal": "^1.0.0", - "is-hexadecimal": "^1.0.0" - } - }, - "prismjs": { - "version": "1.28.0", - "resolved": "https://registry.npmjs.org/prismjs/-/prismjs-1.28.0.tgz", - "integrity": "sha512-8aaXdYvl1F7iC7Xm1spqSaY/OJBpYW3v+KJ+F17iYxvdc8sfjW194COK5wVhMZX45tGteiBQgdvD/nhxcRwylw==", - "dev": true - }, - "react-syntax-highlighter": { - "version": "15.5.0", - "resolved": "https://registry.npmjs.org/react-syntax-highlighter/-/react-syntax-highlighter-15.5.0.tgz", - "integrity": "sha512-+zq2myprEnQmH5yw6Gqc8lD55QHnpKaU8TOcFeC/Lg/MQSs8UknEA0JC4nTZGFAXC2J2Hyj/ijJ7NlabyPi2gg==", - "dev": true, - "requires": { - "@babel/runtime": "^7.3.1", - "highlight.js": "^10.4.1", - "lowlight": "^1.17.0", - "prismjs": "^1.27.0", - "refractor": "^3.6.0" - } - }, - "refractor": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/refractor/-/refractor-3.6.0.tgz", - "integrity": "sha512-MY9W41IOWxxk31o+YvFCNyNzdkc9M20NoZK5vq6jkv4I/uh2zkWcfudj0Q1fovjUQJrNewS9NMzeTtqPf+n5EA==", - "dev": true, - "requires": { - "hastscript": "^6.0.0", - "parse-entities": "^2.0.0", - "prismjs": "~1.27.0" - }, - "dependencies": { - "prismjs": { - "version": "1.27.0", - "resolved": "https://registry.npmjs.org/prismjs/-/prismjs-1.27.0.tgz", - "integrity": "sha512-t13BGPUlFDR7wRB5kQDG4jjl7XeuH6jbJGt11JHPL96qwsEHNX2+68tFXqc1/k+/jALsbSWJKUOT/hcYAZ5LkA==", - "dev": true - } - } - } - } - }, "@storybook/addon-toolbars": { "version": "6.5.7", "resolved": "https://registry.npmjs.org/@storybook/addon-toolbars/-/addon-toolbars-6.5.7.tgz", @@ -15891,16 +16120,16 @@ } }, "@testing-library/jest-dom": { - "version": "5.16.4", - "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-5.16.4.tgz", - "integrity": "sha512-Gy+IoFutbMQcky0k+bqqumXZ1cTGswLsFqmNLzNdSKkU9KGV2u9oXhukCbbJ9/LRPKiqwxEE8VpV/+YZlfkPUA==", + "version": "5.16.5", + "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-5.16.5.tgz", + "integrity": "sha512-N5ixQ2qKpi5OLYfwQmUb/5mSV9LneAcaUfp32pn4yCnpb8r/Yz0pXFPck21dIicKmi+ta5WRAknkZCfA8refMA==", "dev": true, "requires": { + "@adobe/css-tools": "^4.0.1", "@babel/runtime": "^7.9.2", "@types/testing-library__jest-dom": "^5.9.1", "aria-query": "^5.0.0", "chalk": "^3.0.0", - "css": "^3.0.0", "css.escape": "^1.5.1", "dom-accessibility-api": "^0.5.6", "lodash": "^4.17.15", @@ -15917,16 +16146,13 @@ } }, "aria-query": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.0.0.tgz", - "integrity": "sha512-V+SM7AbUwJ+EBnB8+DXs0hPZHO0W6pqBcc0dW90OwtVG02PswOu/teuARoLQjdDOH+t9pJgGnW5/Qmouf3gPJg==", - "dev": true - }, - "atob": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", - "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", - "dev": true + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.1.1.tgz", + "integrity": "sha512-4cPQjOYM2mqq7mZG8CSxkUvL2Yv/x29VhGq5LKehTsxRnoVQps1YGt9NyjcNQsznEsD4rr8a6zGxqeNTqJWjpA==", + "dev": true, + "requires": { + "deep-equal": "^2.0.5" + } }, "chalk": { "version": "3.0.0", @@ -15953,15 +16179,37 @@ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "css": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/css/-/css-3.0.0.tgz", - "integrity": "sha512-DG9pFfwOrzc+hawpmqX/dHYHJG+Bsdb0klhyi1sDneOgGOXy9wQIC8hzyVp1e4NRYDBdxcylvywPkkXCHAzTyQ==", + "deep-equal": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-2.0.5.tgz", + "integrity": "sha512-nPiRgmbAtm1a3JsnLCf6/SLfXcjyN5v8L1TXzdCmHrXJ4hx+gW/w1YCcn7z8gJtSiDArZCgYtbao3QqLm/N1Sw==", "dev": true, "requires": { - "inherits": "^2.0.4", - "source-map": "^0.6.1", - "source-map-resolve": "^0.6.0" + "call-bind": "^1.0.0", + "es-get-iterator": "^1.1.1", + "get-intrinsic": "^1.0.1", + "is-arguments": "^1.0.4", + "is-date-object": "^1.0.2", + "is-regex": "^1.1.1", + "isarray": "^2.0.5", + "object-is": "^1.1.4", + "object-keys": "^1.1.1", + "object.assign": "^4.1.2", + "regexp.prototype.flags": "^1.3.0", + "side-channel": "^1.0.3", + "which-boxed-primitive": "^1.0.1", + "which-collection": "^1.0.1", + "which-typed-array": "^1.1.2" + } + }, + "define-properties": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.4.tgz", + "integrity": "sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA==", + "dev": true, + "requires": { + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" } }, "has-flag": { @@ -15970,18 +16218,79 @@ "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, "indent-string": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", "dev": true }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "is-date-object": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", + "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + } + } + }, + "isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", "dev": true }, + "object.assign": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", + "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + } + } + }, "redent": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/redent/-/redent-3.0.0.tgz", @@ -15992,22 +16301,6 @@ "strip-indent": "^3.0.0" } }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "source-map-resolve": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.6.0.tgz", - "integrity": "sha512-KXBr9d/fO/bWo97NXsPIAW1bFSBOuCnjbNTBMO7N59hsv5i9yzRDfcYwwt0l04+VqnKC+EwzvJZIP/qkuMgR/w==", - "dev": true, - "requires": { - "atob": "^2.1.2", - "decode-uri-component": "^0.2.0" - } - }, "strip-indent": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-3.0.0.tgz", @@ -16029,29 +16322,49 @@ } }, "@testing-library/react": { - "version": "12.1.5", - "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-12.1.5.tgz", - "integrity": "sha512-OfTXCJUFgjd/digLUuPxa0+/3ZxsQmE7ub9kcbW/wi96Bh3o/p5vrETcBGfP17NWPGqeYYl5LTRpwyGoMC4ysg==", + "version": "13.4.0", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-13.4.0.tgz", + "integrity": "sha512-sXOGON+WNTh3MLE9rve97ftaZukN3oNf2KjDy7YTx6hcTO2uuLHuCGynMDhFwGw/jYf4OJ2Qk0i4i79qMNNkyw==", "dev": true, "requires": { "@babel/runtime": "^7.12.5", - "@testing-library/dom": "^8.0.0", - "@types/react-dom": "<18.0.0" + "@testing-library/dom": "^8.5.0", + "@types/react-dom": "^18.0.0" + }, + "dependencies": { + "@types/react-dom": { + "version": "18.0.6", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.0.6.tgz", + "integrity": "sha512-/5OFZgfIPSwy+YuIBP/FgJnQnsxhZhjjrnxudMddeblOouIodEQ75X14Rr4wGSG/bknL+Omy9iWlLo1u/9GzAA==", + "dev": true, + "requires": { + "@types/react": "*" + } + } } }, "@testing-library/react-native": { - "version": "9.1.0", - "resolved": "https://registry.npmjs.org/@testing-library/react-native/-/react-native-9.1.0.tgz", - "integrity": "sha512-YBCSOIMYlh8gI0VG7ExRe80hNpfhC+i7j0cvpwiopUYtbpft8bMJXO35A4zEk7BkiWXEq6bYZ7VDJR3muSLhyQ==", + "version": "11.3.0", + "resolved": "https://registry.npmjs.org/@testing-library/react-native/-/react-native-11.3.0.tgz", + "integrity": "sha512-wCbH7TJ2uVwtkQPRQTZbM3Y/mzwK5zxwkOPKfR2UdVdz/RtCC2UVyDnJMaBNwG/cNle5tc92sQDfp3Gn2oyftg==", "dev": true, "requires": { - "pretty-format": "^27.0.0" + "pretty-format": "^29.0.3" }, "dependencies": { - "ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "@jest/schemas": { + "version": "29.0.0", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.0.0.tgz", + "integrity": "sha512-3Ab5HgYIIAnS0HjqJHQYZS+zXc4tUmTmBH3z83ajI6afXp8X3ZtdLX+nXx+I7LNkJD7uN9LAVhgnjDgZa2z0kA==", + "dev": true, + "requires": { + "@sinclair/typebox": "^0.24.1" + } + }, + "@sinclair/typebox": { + "version": "0.24.50", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.24.50.tgz", + "integrity": "sha512-k8ETQOOQDg5FtK7y9KJWpsGLik+QlPmIi8zzl/dGUgshV2QitprkFlCR/AemjWOTyKn9UwSSGRTzLVotvgCjYQ==", "dev": true }, "ansi-styles": { @@ -16061,28 +16374,28 @@ "dev": true }, "pretty-format": { - "version": "27.5.1", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", - "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.2.1.tgz", + "integrity": "sha512-Y41Sa4aLCtKAXvwuIpTvcFBkyeYp2gdFWzXGA+ZNES3VwURIB165XO/z7CjETwzCCS53MjW/rLMyyqEnTtaOfA==", "dev": true, "requires": { - "ansi-regex": "^5.0.1", + "@jest/schemas": "^29.0.0", "ansi-styles": "^5.0.0", - "react-is": "^17.0.1" + "react-is": "^18.0.0" } }, "react-is": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", - "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.2.0.tgz", + "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==", "dev": true } } }, "@testing-library/user-event": { - "version": "14.2.0", - "resolved": "https://registry.npmjs.org/@testing-library/user-event/-/user-event-14.2.0.tgz", - "integrity": "sha512-+hIlG4nJS6ivZrKnOP7OGsDu9Fxmryj9vCl8x0ZINtTJcCHs2zLsYif5GzuRiBF2ck5GZG2aQr7Msg+EHlnYVQ==", + "version": "14.4.3", + "resolved": "https://registry.npmjs.org/@testing-library/user-event/-/user-event-14.4.3.tgz", + "integrity": "sha512-kCUc5MEwaEMakkO5x7aoD+DLi02ehmEM2QCGWvNqAS1dV/fAvORWEjnjsEIvml59M7Y5kCkWN6fCCyPOe8OL6Q==", "dev": true }, "@tootallnate/once": { @@ -16311,43 +16624,216 @@ } }, "@types/jest": { - "version": "28.1.1", - "resolved": "https://registry.npmjs.org/@types/jest/-/jest-28.1.1.tgz", - "integrity": "sha512-C2p7yqleUKtCkVjlOur9BWVA4HgUQmEj/HWCt5WzZ5mLXrWnyIfl0wGuArc+kBXsy0ZZfLp+7dywB4HtSVYGVA==", + "version": "29.2.0", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.2.0.tgz", + "integrity": "sha512-KO7bPV21d65PKwv3LLsD8Jn3E05pjNjRZvkm+YTacWhVmykAb07wW6IkZUmQAltwQafNcDUEUrMO2h3jeBSisg==", "dev": true, "requires": { - "jest-matcher-utils": "^27.0.0", - "pretty-format": "^27.0.0" + "expect": "^29.0.0", + "pretty-format": "^29.0.0" }, "dependencies": { - "ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "@jest/schemas": { + "version": "29.0.0", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.0.0.tgz", + "integrity": "sha512-3Ab5HgYIIAnS0HjqJHQYZS+zXc4tUmTmBH3z83ajI6afXp8X3ZtdLX+nXx+I7LNkJD7uN9LAVhgnjDgZa2z0kA==", + "dev": true, + "requires": { + "@sinclair/typebox": "^0.24.1" + } + }, + "@jest/types": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.2.1.tgz", + "integrity": "sha512-O/QNDQODLnINEPAI0cl9U6zUIDXEWXt6IC1o2N2QENuos7hlGUIthlKyV4p6ki3TvXFX071blj8HUhgLGquPjw==", + "dev": true, + "requires": { + "@jest/schemas": "^29.0.0", + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^17.0.8", + "chalk": "^4.0.0" + } + }, + "@sinclair/typebox": { + "version": "0.24.50", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.24.50.tgz", + "integrity": "sha512-k8ETQOOQDg5FtK7y9KJWpsGLik+QlPmIi8zzl/dGUgshV2QitprkFlCR/AemjWOTyKn9UwSSGRTzLVotvgCjYQ==", "dev": true }, + "@types/yargs": { + "version": "17.0.13", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.13.tgz", + "integrity": "sha512-9sWaruZk2JGxIQU+IhI1fhPYRcQ0UuTNuKuCW9bR5fp7qi2Llf7WDzNa17Cy7TKnh3cdxDOiyTu6gaLS0eDatg==", + "dev": true, + "requires": { + "@types/yargs-parser": "*" + } + }, "ansi-styles": { "version": "5.2.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", "dev": true }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "ci-info": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.5.0.tgz", + "integrity": "sha512-yH4RezKOGlOhxkmhbeNuC4eYZKAUsEaGtBuBzDDP1eFUKiccDWzBABxBfOx31IDwDIXMTxWuwAxUGModvkbuVw==", + "dev": true + }, + "diff-sequences": { + "version": "29.2.0", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.2.0.tgz", + "integrity": "sha512-413SY5JpYeSBZxmenGEmCVQ8mCgtFJF0w9PROdaS6z987XC2Pd2GOKqOITLtMftmyFZqgtCOb/QA7/Z3ZXfzIw==", + "dev": true + }, + "expect": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.2.1.tgz", + "integrity": "sha512-BJtA754Fba0YWRWHgjKUMTA3ltWarKgITXHQnbZ2mTxTXC4yMQlR0FI7HkB3fJYkhWBf4qjNiqvg3LDtXCcVRQ==", + "dev": true, + "requires": { + "@jest/expect-utils": "^29.2.1", + "jest-get-type": "^29.2.0", + "jest-matcher-utils": "^29.2.1", + "jest-message-util": "^29.2.1", + "jest-util": "^29.2.1" + } + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "graceful-fs": { + "version": "4.2.10", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", + "dev": true + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "jest-diff": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.2.1.tgz", + "integrity": "sha512-gfh/SMNlQmP3MOUgdzxPOd4XETDJifADpT937fN1iUGz+9DgOu2eUPHH25JDkLVcLwwqxv3GzVyK4VBUr9fjfA==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "diff-sequences": "^29.2.0", + "jest-get-type": "^29.2.0", + "pretty-format": "^29.2.1" + } + }, + "jest-get-type": { + "version": "29.2.0", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.2.0.tgz", + "integrity": "sha512-uXNJlg8hKFEnDgFsrCjznB+sTxdkuqiCL6zMgA75qEbAJjJYTs9XPrvDctrEig2GDow22T/LvHgO57iJhXB/UA==", + "dev": true + }, + "jest-matcher-utils": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.2.1.tgz", + "integrity": "sha512-hUTBh7H/Mnb6GTpihbLh8uF5rjAMdekfW/oZNXUMAXi7bbmym2HiRpzgqf/zzkjgejMrVAkPdVSQj+32enlUww==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "jest-diff": "^29.2.1", + "jest-get-type": "^29.2.0", + "pretty-format": "^29.2.1" + } + }, + "jest-message-util": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.2.1.tgz", + "integrity": "sha512-Dx5nEjw9V8C1/Yj10S/8ivA8F439VS8vTq1L7hEgwHFn9ovSKNpYW/kwNh7UglaEgXO42XxzKJB+2x0nSglFVw==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.2.1", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.2.1", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + } + }, + "jest-util": { + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.2.1.tgz", + "integrity": "sha512-P5VWDj25r7kj7kl4pN2rG/RN2c1TLfYYYZYULnS/35nFDjBai+hBeo3MDrYZS7p6IoY3YHZnt2vq4L6mKnLk0g==", + "dev": true, + "requires": { + "@jest/types": "^29.2.1", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + } + }, + "micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dev": true, + "requires": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + } + }, + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true + }, "pretty-format": { - "version": "27.5.1", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", - "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "version": "29.2.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.2.1.tgz", + "integrity": "sha512-Y41Sa4aLCtKAXvwuIpTvcFBkyeYp2gdFWzXGA+ZNES3VwURIB165XO/z7CjETwzCCS53MjW/rLMyyqEnTtaOfA==", "dev": true, "requires": { - "ansi-regex": "^5.0.1", + "@jest/schemas": "^29.0.0", "ansi-styles": "^5.0.0", - "react-is": "^17.0.1" + "react-is": "^18.0.0" } }, "react-is": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", - "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.2.0.tgz", + "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==", "dev": true + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "requires": { + "is-number": "^7.0.0" + } } } }, @@ -16520,20 +17006,13 @@ } }, "@types/react": { - "version": "17.0.37", - "resolved": "https://registry.npmjs.org/@types/react/-/react-17.0.37.tgz", - "integrity": "sha512-2FS1oTqBGcH/s0E+CjrCCR9+JMpsu9b69RTFO+40ua43ZqP5MmQ4iUde/dMjWR909KxZwmOQIFq6AV6NjEG5xg==", + "version": "18.0.21", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.0.21.tgz", + "integrity": "sha512-7QUCOxvFgnD5Jk8ZKlUAhVcRj7GuJRjnjjiY/IUBWKgOlnvDvTMLD4RTF7NPyVmbRhNrbomZiOepg7M/2Kj1mA==", "requires": { "@types/prop-types": "*", "@types/scheduler": "*", "csstype": "^3.0.2" - }, - "dependencies": { - "csstype": { - "version": "3.0.10", - "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.0.10.tgz", - "integrity": "sha512-2u44ZG2OcNUO9HDp/Jl8C07x6pU/eTR3ncV91SiK3dhG9TWvRVsCoJw14Ckx5DgWkzGA3waZWO3d7pgqpUI/XA==" - } } }, "@types/react-dates": { @@ -16556,9 +17035,9 @@ } }, "@types/react-dom": { - "version": "17.0.11", - "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-17.0.11.tgz", - "integrity": "sha512-f96K3k+24RaLGVu/Y2Ng3e1EbZ8/cVJvypZWd7cy0ofCBaf2lcM46xNhycMZ2xGwbBjRql7hOlZ+e2WlJ5MH3Q==", + "version": "18.0.6", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.0.6.tgz", + "integrity": "sha512-/5OFZgfIPSwy+YuIBP/FgJnQnsxhZhjjrnxudMddeblOouIodEQ75X14Rr4wGSG/bknL+Omy9iWlLo1u/9GzAA==", "requires": { "@types/react": "*" } @@ -16638,9 +17117,9 @@ "dev": true }, "@types/testing-library__jest-dom": { - "version": "5.14.3", - "resolved": "https://registry.npmjs.org/@types/testing-library__jest-dom/-/testing-library__jest-dom-5.14.3.tgz", - "integrity": "sha512-oKZe+Mf4ioWlMuzVBaXQ9WDnEm1+umLx0InILg+yvZVBBDmzV5KfZyLrCvadtWcx8+916jLmHafcmqqffl+iIw==", + "version": "5.14.5", + "resolved": "https://registry.npmjs.org/@types/testing-library__jest-dom/-/testing-library__jest-dom-5.14.5.tgz", + "integrity": "sha512-SBwbxYoyPIvxHbeHxTZX2Pe/74F/tX2/D3mMvzabdeJ25bBojfW0TyB8BHrbq/9zaaKICJZjLP+8r6AeZMFCuQ==", "dev": true, "requires": { "@types/jest": "*" @@ -17579,6 +18058,7 @@ "@wordpress/deprecated": "file:packages/deprecated", "@wordpress/dom": "file:packages/dom", "@wordpress/element": "file:packages/element", + "@wordpress/escape-html": "file:packages/escape-html", "@wordpress/hooks": "file:packages/hooks", "@wordpress/html-entities": "file:packages/html-entities", "@wordpress/i18n": "file:packages/i18n", @@ -17599,6 +18079,7 @@ "colord": "^2.7.0", "diff": "^4.0.2", "dom-scroll-into-view": "^1.2.1", + "fast-deep-equal": "^3.1.3", "inherits": "^2.0.3", "lodash": "^4.17.21", "react-autosize-textarea": "^7.1.0", @@ -17626,6 +18107,7 @@ "@wordpress/deprecated": "file:packages/deprecated", "@wordpress/dom": "file:packages/dom", "@wordpress/element": "file:packages/element", + "@wordpress/escape-html": "file:packages/escape-html", "@wordpress/hooks": "file:packages/hooks", "@wordpress/html-entities": "file:packages/html-entities", "@wordpress/i18n": "file:packages/i18n", @@ -17643,6 +18125,7 @@ "colord": "^2.7.0", "escape-html": "^1.0.3", "fast-average-color": "^9.1.1", + "fast-deep-equal": "^3.1.3", "lodash": "^4.17.21", "memize": "^1.1.0", "micromodal": "^0.4.10", @@ -17681,6 +18164,7 @@ "@wordpress/shortcode": "file:packages/shortcode", "change-case": "^4.1.2", "colord": "^2.7.0", + "fast-deep-equal": "^3.1.3", "hpq": "^1.3.0", "is-plain-object": "^5.0.0", "lodash": "^4.17.21", @@ -17729,7 +18213,8 @@ "date-fns": "^2.28.0", "dom-scroll-into-view": "^1.2.1", "downshift": "^6.0.15", - "framer-motion": "^6.2.8", + "fast-deep-equal": "^3.1.3", + "framer-motion": "^7.6.1", "gradient-parser": "^0.1.5", "highlight-words-core": "^1.2.2", "lodash": "^4.17.21", @@ -17783,6 +18268,7 @@ "@wordpress/url": "file:packages/url", "change-case": "^4.1.2", "equivalent-key-map": "^0.2.2", + "fast-deep-equal": "^3.1.3", "lodash": "^4.17.21", "memize": "^1.1.0", "rememo": "^4.0.0", @@ -17836,6 +18322,7 @@ "@wordpress/preferences": "file:packages/preferences", "@wordpress/widgets": "file:packages/widgets", "classnames": "^2.3.1", + "fast-deep-equal": "^3.1.3", "lodash": "^4.17.21" } }, @@ -18059,6 +18546,7 @@ "classnames": "^2.3.1", "colord": "^2.9.2", "downloadjs": "^1.4.7", + "fast-deep-equal": "^3.1.3", "history": "^5.1.0", "lodash": "^4.17.21", "react-autosize-textarea": "^7.1.0", @@ -18155,13 +18643,25 @@ "version": "file:packages/element", "requires": { "@babel/runtime": "^7.16.0", - "@types/react": "^17.0.37", - "@types/react-dom": "^17.0.11", + "@types/react": "^18.0.21", + "@types/react-dom": "^18.0.6", "@wordpress/escape-html": "file:packages/escape-html", "change-case": "^4.1.2", "is-plain-object": "^5.0.0", - "react": "^17.0.2", - "react-dom": "^17.0.2" + "react": "^18.2.0", + "react-dom": "^18.2.0" + }, + "dependencies": { + "@types/react": { + "version": "18.0.21", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.0.21.tgz", + "integrity": "sha512-7QUCOxvFgnD5Jk8ZKlUAhVcRj7GuJRjnjjiY/IUBWKgOlnvDvTMLD4RTF7NPyVmbRhNrbomZiOepg7M/2Kj1mA==", + "requires": { + "@types/prop-types": "*", + "@types/scheduler": "*", + "csstype": "^3.0.2" + } + } } }, "@wordpress/env": { @@ -18600,7 +19100,7 @@ "requires": { "@babel/runtime": "^7.16.0", "@react-native-clipboard/clipboard": "1.9.0", - "@react-native-community/blur": "3.6.0", + "@react-native-community/blur": "4.2.0", "@react-native-community/slider": "https://raw.githubusercontent.com/wordpress-mobile/react-native-slider/v3.0.2-wp-3/react-native-community-slider-3.0.2-wp-3.tgz", "@react-native-masked-view/masked-view": "0.2.6", "@react-navigation/core": "5.12.0", @@ -18623,26 +19123,26 @@ "fast-average-color": "^9.1.1", "gettext-parser": "^1.3.1", "jed": "^1.1.1", - "jsdom-jscore-rn": "git+https://github.com/iamcco/jsdom-jscore-rn.git#a562f3d57c27c13e5bfc8cf82d496e69a3ba2800", + "jsdom-jscore-rn": "0.1.8", "node-fetch": "^2.6.0", - "react-native": "0.66.2", + "react-native": "0.69.4", "react-native-fast-image": "8.5.11", "react-native-gesture-handler": "https://raw.githubusercontent.com/wordpress-mobile/react-native-gesture-handler/2.3.2-wp-2/react-native-gesture-handler-2.3.2-wp-2.tgz", "react-native-get-random-values": "1.4.0", - "react-native-hr": "git+https://github.com/Riglerr/react-native-hr.git#2d01a5cf77212d100e8b99e0310cce5234f977b3", + "react-native-hr": "https://raw.githubusercontent.com/wordpress-mobile/react-native-hr/1.1.3-wp-1/react-native-hr-1.1.3.tgz", "react-native-hsv-color-picker": "https://raw.githubusercontent.com/wordpress-mobile/react-native-hsv-color-picker/v1.0.1-wp-3/react-native-hsv-color-picker-1.0.1-wp-3.tgz", "react-native-keyboard-aware-scroll-view": "https://raw.githubusercontent.com/wordpress-mobile/react-native-keyboard-aware-scroll-view/v0.8.8-wp-1/react-native-keyboard-aware-scroll-view-0.8.8-wp-1.tgz", "react-native-linear-gradient": "https://raw.githubusercontent.com/wordpress-mobile/react-native-linear-gradient/v2.5.6-wp-3/react-native-linear-gradient-2.5.6-wp-3.tgz", "react-native-modal": "^11.10.0", "react-native-prompt-android": "https://raw.githubusercontent.com/wordpress-mobile/react-native-prompt-android/v1.0.0-wp-3/react-native-prompt-android-1.0.0-wp-3.tgz", - "react-native-reanimated": "https://raw.githubusercontent.com/wordpress-mobile/react-native-reanimated/2.4.1-wp-4/react-native-reanimated-2.4.1-wp-4.tgz", + "react-native-reanimated": "https://raw.githubusercontent.com/wordpress-mobile/react-native-reanimated/2.9.1-wp-2/react-native-reanimated-2.9.1-wp-2.tgz", "react-native-safe-area": "^0.5.0", "react-native-safe-area-context": "3.2.0", "react-native-sass-transformer": "^1.1.1", "react-native-screens": "2.9.0", "react-native-svg": "9.13.6", "react-native-url-polyfill": "^1.1.2", - "react-native-video": "https://raw.githubusercontent.com/wordpress-mobile/react-native-video/5.2.0-wp-4/react-native-video-5.2.0-wp-4.tgz", + "react-native-video": "https://raw.githubusercontent.com/wordpress-mobile/react-native-video/5.2.0-wp-5/react-native-video-5.2.0-wp-5.tgz", "react-native-webview": "11.6.2" } }, @@ -19082,6 +19582,7 @@ "@wordpress/element": "file:packages/element", "@wordpress/i18n": "file:packages/i18n", "@wordpress/url": "file:packages/url", + "fast-deep-equal": "^3.1.3", "lodash": "^4.17.21" } }, @@ -19529,9 +20030,9 @@ }, "dependencies": { "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==" + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==" }, "slice-ansi": { "version": "2.1.0", @@ -19600,6 +20101,7 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, "requires": { "micromatch": "^3.1.4", "normalize-path": "^2.1.1" @@ -19608,7 +20110,7 @@ "app-root-dir": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/app-root-dir/-/app-root-dir-1.0.2.tgz", - "integrity": "sha1-OBh+wt6nV3//Az/8sSFyaS/24Rg=", + "integrity": "sha512-jlpIfsOoNoafl92Sz//64uQHGSyMrD2vYG5d8o2a4qGvyNCvXur7bzIsWtAC/6flI2RYAp3kv8rsfBtaLm7w0g==", "dev": true }, "app-root-path": { @@ -19617,9 +20119,9 @@ "integrity": "sha512-91IFKeKk7FjfmezPKkwtaRvSpnUc4gDwPAjA1YZ9Gn0q0PPeW+vbeUsZuyDwjI7+QTHhcLen2v25fi/AmhvbJA==" }, "appdirsjs": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/appdirsjs/-/appdirsjs-1.2.6.tgz", - "integrity": "sha512-D8wJNkqMCeQs3kLasatELsddox/Xqkhp+J07iXGyL54fVN7oc+nmNfYzGuCs1IEP6uBw+TfpuO3JKwc+lECy4w==" + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/appdirsjs/-/appdirsjs-1.2.7.tgz", + "integrity": "sha512-Quji6+8kLBC3NnBeo14nPDq0+2jUs5s3/xEye+udFHumHhRk4M7aAMXp/PBJqkKYGuuyR9M/6Dq7d2AViiGmhw==" }, "appium": { "version": "1.22.3", @@ -27361,6 +27863,11 @@ "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==" }, + "async": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.4.tgz", + "integrity": "sha512-iAB+JbDEGXhyIUavoDl9WP/Jj106Kz9DEn1DPgYw5ruDn0e3Wgi3sKFm55sASdGBNOQB8F59d9qQ7deqrHA8wQ==" + }, "async-each": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", @@ -27454,6 +27961,12 @@ "resolved": "https://registry.npmjs.org/autosize/-/autosize-4.0.2.tgz", "integrity": "sha512-jnSyH2d+qdfPGpWlcuhGiHmqBJ6g3X+8T+iRwFrHPLVcdoGJE/x6Qicm6aDHfTsbgZKxyV8UU/YB2p4cjKDRRA==" }, + "available-typed-arrays": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz", + "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==", + "dev": true + }, "aws-sign2": { "version": "0.7.0", "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", @@ -27890,7 +28403,7 @@ "babel-plugin-add-react-displayname": { "version": "0.0.5", "resolved": "https://registry.npmjs.org/babel-plugin-add-react-displayname/-/babel-plugin-add-react-displayname-0.0.5.tgz", - "integrity": "sha1-M51M3be2X9YtHfnbn+BN4TQSK9U=", + "integrity": "sha512-LY3+Y0XVDYcShHHorshrDbt4KFWL4bSeniCtl4SYZbask+Syngk1uMPCeN9+nSiZo6zX5s0RTq/J9Pnaaf/KHw==", "dev": true }, "babel-plugin-apply-mdx-type-prop": { @@ -28313,7 +28826,7 @@ "batch-processor": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/batch-processor/-/batch-processor-1.0.0.tgz", - "integrity": "sha1-dclcMrdI4IUNEMKxaPa9vpiRrOg=", + "integrity": "sha512-xoLQD8gmmR32MeuBHgH0Tzd5PuSZx71ZsbhVxOCRbgktZEPe4SQy7s9Z50uPp0F/f7iw2XmkHN2xkgbMfckMDA==", "dev": true }, "bcrypt-pbkdf": { @@ -28375,7 +28888,8 @@ "big-integer": { "version": "1.6.48", "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.48.tgz", - "integrity": "sha512-j51egjPa7/i+RdiRuJbPdJ2FIUYYPhvYLjzoYbcMMm62ooO6F94fETG4MTs46zPAF9Brs04OajboA/qTGuz78w==" + "integrity": "sha512-j51egjPa7/i+RdiRuJbPdJ2FIUYYPhvYLjzoYbcMMm62ooO6F94fETG4MTs46zPAF9Brs04OajboA/qTGuz78w==", + "dev": true }, "big.js": { "version": "5.2.2", @@ -28451,7 +28965,6 @@ "version": "4.1.0", "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", - "dev": true, "requires": { "buffer": "^5.5.0", "inherits": "^2.0.4", @@ -28461,14 +28974,12 @@ "base64-js": { "version": "1.5.1", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", - "dev": true + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" }, "buffer": { "version": "5.7.1", "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", - "dev": true, "requires": { "base64-js": "^1.3.1", "ieee754": "^1.1.13" @@ -28477,20 +28988,17 @@ "ieee754": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", - "dev": true + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" }, "inherits": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" }, "readable-stream": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "dev": true, "requires": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", @@ -28709,14 +29217,6 @@ } } }, - "bplist-creator": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/bplist-creator/-/bplist-creator-0.1.0.tgz", - "integrity": "sha512-sXaHZicyEEmY86WyueLTQesbeoH/mquvarJaQNbjuOQO+7gbFcDEWqKmcWA4cOTLzFlfgvkiVxolk1k5bBIpmg==", - "requires": { - "stream-buffers": "2.2.x" - } - }, "brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -29264,8 +29764,7 @@ "camelcase": { "version": "5.3.1", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" }, "camelcase-css": { "version": "2.0.1", @@ -29311,9 +29810,9 @@ } }, "caniuse-lite": { - "version": "1.0.30001352", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001352.tgz", - "integrity": "sha512-GUgH8w6YergqPQDGWhJGt8GDRnY0L/iJVQcU3eJ46GYf52R8tk0Wxp0PymuFVZboJYXGiCqwozAYZNRjVj6IcA==" + "version": "1.0.30001434", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001434.tgz", + "integrity": "sha512-aOBHrLmTQw//WFa2rcF1If9fa3ypkC1wzqqiKHgfdrXTWcU8C4gKVZT77eQAPWN1APys3+uQ0Df07rKauXGEYA==" }, "capital-case": { "version": "1.0.4", @@ -29329,6 +29828,7 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/capture-exit/-/capture-exit-2.0.0.tgz", "integrity": "sha512-PiT/hQmTonHhl/HFGN+Lx3JJUznrVYJ3+AQsnthneZbvW7x+f08Tk7yLJTLEOUvBTbduLeeBkxEaYXUOUrRq6g==", + "dev": true, "requires": { "rsvp": "^4.8.4" } @@ -29807,6 +30307,7 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, "requires": { "restore-cursor": "^2.0.0" } @@ -29814,7 +30315,8 @@ "cli-spinners": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.2.0.tgz", - "integrity": "sha512-tgU3fKwzYjiLEQgPMD9Jt+JjHVL9kW93FiIMX/l7rivvOD4/LL0Mf7gda3+4U2KJBloybwgj5KEoQgGRioMiKQ==" + "integrity": "sha512-tgU3fKwzYjiLEQgPMD9Jt+JjHVL9kW93FiIMX/l7rivvOD4/LL0Mf7gda3+4U2KJBloybwgj5KEoQgGRioMiKQ==", + "dev": true }, "cli-table3": { "version": "0.6.2", @@ -31772,9 +32274,9 @@ "dev": true }, "dayjs": { - "version": "1.10.7", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.10.7.tgz", - "integrity": "sha512-P6twpd70BcPK34K26uJ1KT3wlhpuOAPoMwJzpsIWUxHZ7wpmbdZL/hQqBDfz7hGurYSa5PhzdhDHtt319hL3ig==" + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.6.tgz", + "integrity": "sha512-zZbY5giJAinCG+7AGaw0wIhNZ6J8AhWuSXKvuc1KAyMiRsvGQWqh4L+MomvhdAYjN+lqvVCMq1I41e3YHvXkyQ==" }, "debug": { "version": "3.1.0", @@ -32216,7 +32718,18 @@ "depd": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", - "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", + "dev": true + }, + "deprecated-react-native-prop-types": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/deprecated-react-native-prop-types/-/deprecated-react-native-prop-types-2.3.0.tgz", + "integrity": "sha512-pWD0voFtNYxrVqvBMYf5gq3NA2GCpfodS1yNynTPc93AYA/KEMGeWDqqeUB6R2Z9ZofVhks2aeJXiuQqKNpesA==", + "requires": { + "@react-native/normalize-color": "*", + "invariant": "*", + "prop-types": "*" + } }, "deprecation": { "version": "2.0.0", @@ -32237,7 +32750,8 @@ "destroy": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", - "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=", + "dev": true }, "detab": { "version": "2.0.4", @@ -32604,9 +33118,9 @@ } }, "domutils": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.5.1.tgz", - "integrity": "sha1-3NhIiib1Y9YQeeSMn3t+Mjc2gs8=", + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", + "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", "requires": { "dom-serializer": "0", "domelementtype": "1" @@ -32865,8 +33379,7 @@ "envinfo": { "version": "7.8.1", "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.8.1.tgz", - "integrity": "sha512-/o+BXHmB7ocbHEAs6F2EnG0ogybVVUdkRunTT2glZU9XAaGmhqskrvKwqXuDfNjEO0LZKWdejEEpnq8aM0tOaw==", - "dev": true + "integrity": "sha512-/o+BXHmB7ocbHEAs6F2EnG0ogybVVUdkRunTT2glZU9XAaGmhqskrvKwqXuDfNjEO0LZKWdejEEpnq8aM0tOaw==" }, "equivalent-key-map": { "version": "0.2.2", @@ -34506,7 +35019,7 @@ "eventemitter2": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-1.0.5.tgz", - "integrity": "sha1-+YNhBRexc3wLncZDvsqTiTwE3xg=" + "integrity": "sha512-EUFhWUYzqqBZlzBMI+dPU8rnKXfQZEUnitnccQuEIAnvWFHCpt3+4fts2+4dpxLtlsiseVXCMFg37KjYChSxpg==" }, "eventemitter3": { "version": "4.0.7", @@ -34533,7 +35046,8 @@ "exec-sh": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/exec-sh/-/exec-sh-0.3.2.tgz", - "integrity": "sha512-9sLAvzhI5nc8TpuQUh4ahMdCrWT00wPWz7j47/emR5+2qEfoZP5zzUXvx+vdx+H6ohhnsYC31iX04QLYJK8zTg==" + "integrity": "sha512-9sLAvzhI5nc8TpuQUh4ahMdCrWT00wPWz7j47/emR5+2qEfoZP5zzUXvx+vdx+H6ohhnsYC31iX04QLYJK8zTg==", + "dev": true }, "execa": { "version": "4.0.2", @@ -35254,8 +35768,7 @@ "fast-deep-equal": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", - "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", - "dev": true + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" }, "fast-diff": { "version": "1.2.0", @@ -35852,6 +36365,15 @@ "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==", "dev": true }, + "for-each": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", + "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", + "dev": true, + "requires": { + "is-callable": "^1.1.3" + } + }, "for-in": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", @@ -36109,24 +36631,39 @@ } }, "framer-motion": { - "version": "6.2.8", - "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-6.2.8.tgz", - "integrity": "sha512-4PtBWFJ6NqR350zYVt9AsFDtISTqsdqna79FvSYPfYDXuuqFmiKtZdkTnYPslnsOMedTW0pEvaQ7eqjD+sA+HA==", + "version": "7.6.15", + "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-7.6.15.tgz", + "integrity": "sha512-z3LPTWWq3zlmtxtZyCHWyucNRHg8iXXeLo9cFQwCIgOT8mp3gWFzU+KrGQ9h3QJFx79G9MVqAcuIN6dVhhuFQQ==", "requires": { "@emotion/is-prop-valid": "^0.8.2", - "framesync": "6.0.1", + "@motionone/dom": "10.13.1", + "framesync": "6.1.2", "hey-listen": "^1.0.8", - "popmotion": "11.0.3", - "style-value-types": "5.0.0", - "tslib": "^2.1.0" + "popmotion": "11.0.5", + "style-value-types": "5.1.2", + "tslib": "2.4.0" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } } }, "framesync": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/framesync/-/framesync-6.0.1.tgz", - "integrity": "sha512-fUY88kXvGiIItgNC7wcTOl0SNRCVXMKSWW2Yzfmn7EKNc+MpCzcz9DhdHcdjbrtN3c6R4H5dTY2jiCpPdysEjA==", + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/framesync/-/framesync-6.1.2.tgz", + "integrity": "sha512-jBTqhX6KaQVDyus8muwZbBeGGP0XgujBRbQ7gM7BRdS3CadCZIHiawyzYLnafYcvZIh5j8WE7cxZKFn7dXhu9g==", "requires": { - "tslib": "^2.1.0" + "tslib": "2.4.0" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } } }, "fresh": { @@ -36167,9 +36704,9 @@ }, "dependencies": { "graceful-fs": { - "version": "4.2.4", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", - "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==" + "version": "4.2.10", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==" } } }, @@ -36446,6 +36983,12 @@ "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", "dev": true }, + "functions-have-names": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", + "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", + "dev": true + }, "gauge": { "version": "4.0.4", "resolved": "https://registry.npmjs.org/gauge/-/gauge-4.0.4.tgz", @@ -37125,7 +37668,7 @@ "has-glob": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-glob/-/has-glob-1.0.0.tgz", - "integrity": "sha1-mqqe7b/7G6OZCnsAEPtnjuAIEgc=", + "integrity": "sha512-D+8A457fBShSEI3tFCj65PAbT++5sKiFtdCdOam0gnfBgw9D277OERk+HM9qYJXmdVLZ/znez10SqHN0BBQ50g==", "dev": true, "requires": { "is-glob": "^3.0.0" @@ -37134,7 +37677,7 @@ "is-glob": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", - "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "integrity": "sha512-UFpDDrPgM6qpnFNI+rh/p3bUaq9hKLZN8bMUWzxmcnZVS3omf4IPK+BrewlnWjO1WmUsMYuSjKh4UJuV4+Lqmw==", "dev": true, "requires": { "is-extglob": "^2.1.0" @@ -37466,14 +38009,23 @@ } }, "hermes-engine": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/hermes-engine/-/hermes-engine-0.9.0.tgz", - "integrity": "sha512-r7U+Y4P2Qg/igFVZN+DpT7JFfXUn1MM4dFne8aW+cCrF6RRymof+VqrUHs1kl07j8h8V2CNesU19RKgWbr3qPw==" + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/hermes-engine/-/hermes-engine-0.11.0.tgz", + "integrity": "sha512-7aMUlZja2IyLYAcZ69NBnwJAR5ZOYlSllj0oMpx08a8HzxHOys0eKCzfphrf6D0vX1JGO1QQvVsQKe6TkYherw==" + }, + "hermes-estree": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.6.0.tgz", + "integrity": "sha512-2YTGzJCkhdmT6VuNprWjXnvTvw/3iPNw804oc7yknvQpNKo+vJGZmtvLLCghOZf0OwzKaNAzeIMp71zQbNl09w==" }, "hermes-parser": { - "version": "0.4.7", - "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.4.7.tgz", - "integrity": "sha512-jc+zCtXbtwTiXoMAoXOHepxAaGVFIp89wwE9qcdwnMd/uGVEtPoY8FaFSsx0ThPvyKirdR2EsIIDVrpbSXz1Ag==" + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.6.0.tgz", + "integrity": "sha512-Vf58jBZca2+QBLR9h7B7mdg8oFz2g5ILz1iVouZ5DOrOrAfBmPfJjdjDT8jrO0f+iJ4/hSRrQHqHIjSnTaLUDQ==", + "dev": true, + "requires": { + "hermes-estree": "0.6.0" + } }, "hermes-profile-transformer": { "version": "0.0.6", @@ -37484,9 +38036,9 @@ }, "dependencies": { "source-map": { - "version": "0.7.3", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", - "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==" + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", + "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==" } } }, @@ -37705,7 +38257,7 @@ "htmlparser2-without-node-native": { "version": "3.9.2", "resolved": "https://registry.npmjs.org/htmlparser2-without-node-native/-/htmlparser2-without-node-native-3.9.2.tgz", - "integrity": "sha1-s+0FDYd9D/NGWWnjOYd7f59mMfY=", + "integrity": "sha512-+FplQXqmY5fRx6vCIp2P5urWaoBCpTNJMXnKP/6mNCcyb+AZWWJzA8D03peXfozlxDL+vpgLK5dJblqEgu8j6A==", "requires": { "domelementtype": "^1.3.0", "domhandler": "^2.3.0", @@ -37729,22 +38281,32 @@ "dev": true }, "http-errors": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.1.tgz", - "integrity": "sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", + "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", "requires": { - "depd": "~1.1.2", + "depd": "2.0.0", "inherits": "2.0.4", "setprototypeof": "1.2.0", - "statuses": ">= 1.5.0 < 2", + "statuses": "2.0.1", "toidentifier": "1.0.1" }, "dependencies": { + "depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" + }, "inherits": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" }, + "statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==" + }, "toidentifier": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", @@ -38630,6 +39192,7 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", + "dev": true, "requires": { "ci-info": "^2.0.0" } @@ -38785,8 +39348,7 @@ "is-interactive": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", - "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", - "dev": true + "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==" }, "is-ip": { "version": "3.1.0", @@ -38984,6 +39546,231 @@ "text-extensions": "^1.0.0" } }, + "is-typed-array": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.9.tgz", + "integrity": "sha512-kfrlnTTn8pZkfpJMUgYD7YZ3qzeJgWUn8XfVYBARc4wnmNOmLbmuuaAs3q5fvB0UJOn6yHAKaGTPM7d6ezoD/A==", + "dev": true, + "requires": { + "available-typed-arrays": "^1.0.5", + "call-bind": "^1.0.2", + "es-abstract": "^1.20.0", + "for-each": "^0.3.3", + "has-tostringtag": "^1.0.0" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "define-properties": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.4.tgz", + "integrity": "sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA==", + "dev": true, + "requires": { + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + } + }, + "es-abstract": { + "version": "1.20.4", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.20.4.tgz", + "integrity": "sha512-0UtvRN79eMe2L+UNEF1BwRe364sj/DXhQ/k5FmivgoSdpM90b8Jc0mDzKMGo7QS0BVbOP/bTwBKNnDc9rNzaPA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "function.prototype.name": "^1.1.5", + "get-intrinsic": "^1.1.3", + "get-symbol-description": "^1.0.0", + "has": "^1.0.3", + "has-property-descriptors": "^1.0.0", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.3", + "is-callable": "^1.2.7", + "is-negative-zero": "^2.0.2", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "is-string": "^1.0.7", + "is-weakref": "^1.0.2", + "object-inspect": "^1.12.2", + "object-keys": "^1.1.1", + "object.assign": "^4.1.4", + "regexp.prototype.flags": "^1.4.3", + "safe-regex-test": "^1.0.0", + "string.prototype.trimend": "^1.0.5", + "string.prototype.trimstart": "^1.0.5", + "unbox-primitive": "^1.0.2" + }, + "dependencies": { + "get-intrinsic": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", + "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.3" + } + } + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "function.prototype.name": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", + "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.0", + "functions-have-names": "^1.2.2" + } + }, + "has-bigints": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", + "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==", + "dev": true + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, + "is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "dev": true + }, + "is-negative-zero": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", + "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "dev": true + }, + "is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "is-shared-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", + "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "object-inspect": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.2.tgz", + "integrity": "sha512-z+cPxW0QGUp0mcqcsgQyLVRDoXFQbXOwBaqyF7VIgI4TWNQsDHrBpUQslRmIfAoYWdYzs6UlKJtB2XJpTaNSpQ==", + "dev": true + }, + "object.assign": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", + "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + } + }, + "regexp.prototype.flags": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz", + "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "functions-have-names": "^1.2.2" + } + }, + "string.prototype.trimstart": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.5.tgz", + "integrity": "sha512-THx16TJCGlsN0o6dl2o6ncWUsdgnLRSA23rRE5pyGBw/mLr3Ej/R2LaqCtgP8VNMGZsvMWnf9ooZPyY2bHvUFg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.19.5" + } + }, + "unbox-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", + "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + } + } + } + }, "is-typedarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", @@ -38993,8 +39780,7 @@ "is-unicode-supported": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", - "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", - "dev": true + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==" }, "is-utf8": { "version": "0.2.1", @@ -39003,6 +39789,12 @@ "dev": true, "optional": true }, + "is-weakmap": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.1.tgz", + "integrity": "sha512-NSBR4kH5oVj1Uwvv970ruUkCV7O1mzgVFO4/rev2cLRda9Tm9HrL70ZPut4rOHgY0FNrUu9BCbXA2sdQ+x0chA==", + "dev": true + }, "is-weakref": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.1.tgz", @@ -39012,6 +39804,45 @@ "call-bind": "^1.0.0" } }, + "is-weakset": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.2.tgz", + "integrity": "sha512-t2yVvttHkQktwnNNmBQ98AhENLdPUTDTE21uPqAQ0ARwQfGeQKRVS0NNurH7bTf7RrvcVn1OOge45CnBeHCSmg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.1" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "get-intrinsic": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", + "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.3" + } + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + } + } + }, "is-whitespace-character": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/is-whitespace-character/-/is-whitespace-character-1.0.4.tgz", @@ -39021,7 +39852,7 @@ "is-window": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-window/-/is-window-1.0.2.tgz", - "integrity": "sha1-LIlspT25feRdPDMTOmXYyfVjSA0=", + "integrity": "sha512-uj00kdXyZb9t9RcAUAwMZAnkBUwdYGhYlt7djMXhfyhUCzwNba50tIiBKR7q0l7tdoBtFVw/3JmLY6fI3rmZmg==", "dev": true }, "is-windows": { @@ -40159,13 +40990,13 @@ "jest-get-type": { "version": "26.3.0", "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", - "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", - "dev": true + "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==" }, "jest-haste-map": { "version": "26.6.2", "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-26.6.2.tgz", "integrity": "sha512-easWIJXIw71B2RdR8kgqpjQrbMRWQBgiBwXYEhtGUTaX+doCjBheluShdDMeR8IMfJiTqH4+zfhtg29apJf/8w==", + "dev": true, "requires": { "@jest/types": "^26.6.2", "@types/graceful-fs": "^4.1.2", @@ -40187,6 +41018,7 @@ "version": "3.1.2", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", + "dev": true, "requires": { "normalize-path": "^3.0.0", "picomatch": "^2.0.4" @@ -40196,6 +41028,7 @@ "version": "3.0.2", "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, "requires": { "fill-range": "^7.0.1" } @@ -40204,6 +41037,7 @@ "version": "7.0.1", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, "requires": { "to-regex-range": "^5.0.1" } @@ -40211,17 +41045,20 @@ "graceful-fs": { "version": "4.2.8", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", - "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==" + "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", + "dev": true }, "is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true }, "micromatch": { "version": "4.0.4", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz", "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==", + "dev": true, "requires": { "braces": "^3.0.1", "picomatch": "^2.2.3" @@ -40230,19 +41067,22 @@ "picomatch": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", - "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==" + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "dev": true } } }, "normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true }, "to-regex-range": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, "requires": { "is-number": "^7.0.0" } @@ -40763,7 +41603,8 @@ "jest-regex-util": { "version": "26.0.0", "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-26.0.0.tgz", - "integrity": "sha512-Gv3ZIs/nA48/Zvjrl34bf+oD76JHiGDUxNOVgUjh3j890sblXryjY4rss71fPtD/njchl6PSE2hIhvyWa1eT0A==" + "integrity": "sha512-Gv3ZIs/nA48/Zvjrl34bf+oD76JHiGDUxNOVgUjh3j890sblXryjY4rss71fPtD/njchl6PSE2hIhvyWa1eT0A==", + "dev": true }, "jest-resolve": { "version": "27.4.5", @@ -41698,6 +42539,7 @@ "version": "26.6.2", "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-26.6.2.tgz", "integrity": "sha512-S5wqyz0DXnNJPd/xfIzZ5Xnp1HrJWBczg8mMfMpN78OJ5eDxXyf+Ygld9wX1DnUWbIbhM1YDY95NjR4CBXkb2g==", + "dev": true, "requires": { "@types/node": "*", "graceful-fs": "^4.2.4" @@ -41706,7 +42548,8 @@ "graceful-fs": { "version": "4.2.8", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", - "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==" + "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", + "dev": true } } }, @@ -42043,6 +42886,7 @@ "version": "26.6.2", "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-26.6.2.tgz", "integrity": "sha512-MDW0fKfsn0OI7MS7Euz6h8HNDXVQ0gaM9uW6RjfDmd1DAFcaxX9OqIakHIqhbnmF08Cf2DLDG+ulq8YQQ0Lp0Q==", + "dev": true, "requires": { "@jest/types": "^26.6.2", "@types/node": "*", @@ -42056,6 +42900,7 @@ "version": "3.0.2", "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, "requires": { "fill-range": "^7.0.1" } @@ -42064,6 +42909,7 @@ "version": "7.0.1", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, "requires": { "to-regex-range": "^5.0.1" } @@ -42071,17 +42917,20 @@ "graceful-fs": { "version": "4.2.8", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", - "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==" + "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", + "dev": true }, "is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true }, "micromatch": { "version": "4.0.4", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz", "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==", + "dev": true, "requires": { "braces": "^3.0.1", "picomatch": "^2.2.3" @@ -42090,12 +42939,14 @@ "picomatch": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", - "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==" + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "dev": true }, "to-regex-range": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, "requires": { "is-number": "^7.0.0" } @@ -42347,6 +43198,7 @@ "version": "26.6.2", "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.6.2.tgz", "integrity": "sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==", + "dev": true, "requires": { "@types/node": "*", "merge-stream": "^2.0.0", @@ -42356,12 +43208,14 @@ "has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true }, "supports-color": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, "requires": { "has-flag": "^4.0.0" } @@ -42398,7 +43252,7 @@ "js-string-escape": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/js-string-escape/-/js-string-escape-1.0.1.tgz", - "integrity": "sha1-4mJbrbwNZ8dTPp7cEGjFh65BN+8=", + "integrity": "sha512-Smw4xcfIQ5LVjAOuJCvN/zIodzA/BBSsluuoSykP+lUvScIi4U6RJLfwHet5cxFnCswUjISV8oAXaqaJDY3chg==", "dev": true }, "js-tokens": { @@ -42428,40 +43282,115 @@ "integrity": "sha512-KmxeBlRjwoqCnBBKGsihFtvsBHyUFlBxJPK4FzeYcIuBfdjv6jFys44JITAgSTbQD+vIdwMEfyZklsuQX0yI1Q==" }, "jscodeshift": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/jscodeshift/-/jscodeshift-0.11.0.tgz", - "integrity": "sha512-SdRK2C7jjs4k/kT2mwtO07KJN9RnjxtKn03d9JVj6c3j9WwaLcFYsICYDnLAzY0hp+wG2nxl+Cm2jWLiNVYb8g==", - "requires": { - "@babel/core": "^7.1.6", - "@babel/parser": "^7.1.6", - "@babel/plugin-proposal-class-properties": "^7.1.0", - "@babel/plugin-proposal-nullish-coalescing-operator": "^7.1.0", - "@babel/plugin-proposal-optional-chaining": "^7.1.0", - "@babel/plugin-transform-modules-commonjs": "^7.1.0", - "@babel/preset-flow": "^7.0.0", - "@babel/preset-typescript": "^7.1.0", - "@babel/register": "^7.0.0", + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/jscodeshift/-/jscodeshift-0.13.1.tgz", + "integrity": "sha512-lGyiEbGOvmMRKgWk4vf+lUrCWO/8YR8sUR3FKF1Cq5fovjZDlIcw3Hu5ppLHAnEXshVffvaM0eyuY/AbOeYpnQ==", + "requires": { + "@babel/core": "^7.13.16", + "@babel/parser": "^7.13.16", + "@babel/plugin-proposal-class-properties": "^7.13.0", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.13.8", + "@babel/plugin-proposal-optional-chaining": "^7.13.12", + "@babel/plugin-transform-modules-commonjs": "^7.13.8", + "@babel/preset-flow": "^7.13.13", + "@babel/preset-typescript": "^7.13.0", + "@babel/register": "^7.13.16", "babel-core": "^7.0.0-bridge.0", - "colors": "^1.1.2", + "chalk": "^4.1.2", "flow-parser": "0.*", "graceful-fs": "^4.2.4", "micromatch": "^3.1.10", "neo-async": "^2.5.0", "node-dir": "^0.1.17", - "recast": "^0.20.3", - "temp": "^0.8.1", + "recast": "^0.20.4", + "temp": "^0.8.4", "write-file-atomic": "^2.3.0" }, "dependencies": { - "colors": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", - "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==" + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } }, "graceful-fs": { - "version": "4.2.8", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", - "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==" + "version": "4.2.10", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==" + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "requires": { + "glob": "^7.1.3" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "requires": { + "has-flag": "^4.0.0" + } + }, + "temp": { + "version": "0.8.4", + "resolved": "https://registry.npmjs.org/temp/-/temp-0.8.4.tgz", + "integrity": "sha512-s0ZZzd0BzYv5tLSptZooSjK8oj6C+c19p7Vqta9+6NPOf7r+fxq0cJe6/oN4LTC79sy5NY8ucOJNgwsKCSbfqg==", + "requires": { + "rimraf": "~2.6.2" + } } } }, @@ -42635,8 +43564,9 @@ } }, "jsdom-jscore-rn": { - "version": "git+https://github.com/iamcco/jsdom-jscore-rn.git#a562f3d57c27c13e5bfc8cf82d496e69a3ba2800", - "from": "git+https://github.com/iamcco/jsdom-jscore-rn.git#a562f3d57c27c13e5bfc8cf82d496e69a3ba2800", + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/jsdom-jscore-rn/-/jsdom-jscore-rn-0.1.8.tgz", + "integrity": "sha512-Sm0BCCQL3RRmJwztZa9B4QIOxgFOiu7vpsJjLsjG55HlRzOgUdmTOHrhV4SZkjXef5sBF+E8qKnrhsBUO7Gxjg==", "requires": { "htmlparser2-without-node-native": "^3.9.2", "querystring": "^0.2.0" @@ -43862,23 +44792,12 @@ "dev": true }, "log-symbols": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", - "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", "requires": { - "chalk": "^2.0.1" - }, - "dependencies": { - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - } - } + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" } }, "log-update": { @@ -43919,11 +44838,6 @@ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" - }, "emoji-regex": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", @@ -44637,8 +45551,7 @@ "memoize-one": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/memoize-one/-/memoize-one-5.2.1.tgz", - "integrity": "sha512-zYiwtZUcYyXKo/np96AGZAckk+FWWsUdJ3cHGGmld7+AhvcWmQyGCYUh1hc4Q/pkOhb65dQR/pqCyK0cOaHz4Q==", - "dev": true + "integrity": "sha512-zYiwtZUcYyXKo/np96AGZAckk+FWWsUdJ3cHGGmld7+AhvcWmQyGCYUh1hc4Q/pkOhb65dQR/pqCyK0cOaHz4Q==" }, "memoizerific": { "version": "1.11.3", @@ -44884,9 +45797,9 @@ "dev": true }, "metro": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro/-/metro-0.66.2.tgz", - "integrity": "sha512-uNsISfcQ3iKKSHoN5Q+LAh0l3jeeg7ZcNZ/4BAHGsk02erA0OP+l2m+b5qYVoPptHz9Oc3KyG5oGJoTu41pWjg==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro/-/metro-0.70.3.tgz", + "integrity": "sha512-uEWS7xg8oTetQDABYNtsyeUjdLhH3KAvLFpaFFoJqUpOk2A3iygszdqmjobFl6W4zrvKDJS+XxdMR1roYvUhTw==", "requires": { "@babel/code-frame": "^7.0.0", "@babel/core": "^7.14.0", @@ -44897,7 +45810,7 @@ "@babel/types": "^7.0.0", "absolute-path": "^0.0.0", "accepts": "^1.3.7", - "async": "^2.4.0", + "async": "^3.2.2", "chalk": "^4.0.0", "ci-info": "^2.0.0", "connect": "^3.6.5", @@ -44905,31 +45818,29 @@ "denodeify": "^1.2.1", "error-stack-parser": "^2.0.6", "fs-extra": "^1.0.0", - "graceful-fs": "^4.1.3", - "hermes-parser": "0.4.7", + "graceful-fs": "^4.2.4", + "hermes-parser": "0.6.0", "image-size": "^0.6.0", "invariant": "^2.2.4", - "jest-haste-map": "^26.5.2", - "jest-worker": "^26.0.0", + "jest-haste-map": "^27.3.1", + "jest-worker": "^27.2.0", "lodash.throttle": "^4.1.1", - "metro-babel-register": "0.66.2", - "metro-babel-transformer": "0.66.2", - "metro-cache": "0.66.2", - "metro-cache-key": "0.66.2", - "metro-config": "0.66.2", - "metro-core": "0.66.2", - "metro-hermes-compiler": "0.66.2", - "metro-inspector-proxy": "0.66.2", - "metro-minify-uglify": "0.66.2", - "metro-react-native-babel-preset": "0.66.2", - "metro-resolver": "0.66.2", - "metro-runtime": "0.66.2", - "metro-source-map": "0.66.2", - "metro-symbolicate": "0.66.2", - "metro-transform-plugins": "0.66.2", - "metro-transform-worker": "0.66.2", + "metro-babel-transformer": "0.70.3", + "metro-cache": "0.70.3", + "metro-cache-key": "0.70.3", + "metro-config": "0.70.3", + "metro-core": "0.70.3", + "metro-hermes-compiler": "0.70.3", + "metro-inspector-proxy": "0.70.3", + "metro-minify-uglify": "0.70.3", + "metro-react-native-babel-preset": "0.70.3", + "metro-resolver": "0.70.3", + "metro-runtime": "0.70.3", + "metro-source-map": "0.70.3", + "metro-symbolicate": "0.70.3", + "metro-transform-plugins": "0.70.3", + "metro-transform-worker": "0.70.3", "mime-types": "^2.1.27", - "mkdirp": "^0.5.1", "node-fetch": "^2.2.0", "nullthrows": "^1.1.1", "rimraf": "^2.5.4", @@ -44938,27 +45849,51 @@ "strip-ansi": "^6.0.0", "temp": "0.8.3", "throat": "^5.0.0", - "ws": "^1.1.5", + "ws": "^7.5.1", "yargs": "^15.3.1" }, "dependencies": { + "@jest/types": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.5.1.tgz", + "integrity": "sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw==", + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^16.0.0", + "chalk": "^4.0.0" + } + }, + "@types/yargs": { + "version": "16.0.4", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-16.0.4.tgz", + "integrity": "sha512-T8Yc9wt/5LbJyCaLiHPReJa0kApcIgJ7Bn735GjItUfh08Z1pJvu8QZqb9s+mMvKV6WUQRV7K2R46YbjMXTTJw==", + "requires": { + "@types/yargs-parser": "*" + } + }, "ansi-regex": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" }, - "async": { - "version": "2.6.4", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", - "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==", + "anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", "requires": { - "lodash": "^4.17.14" + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" } }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "requires": { + "fill-range": "^7.0.1" + } }, "debug": { "version": "2.6.9", @@ -44973,6 +45908,14 @@ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "requires": { + "to-regex-range": "^5.0.1" + } + }, "find-up": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", @@ -44985,7 +45928,7 @@ "fs-extra": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-1.0.0.tgz", - "integrity": "sha1-zTzl9+fLYUWIP8rjGR6Yd/hYeVA=", + "integrity": "sha512-VerQV6vEKuhDWD2HGOybV6v5I73syoc/cXAbKlgTC7M/oFVEtklWlp9QH2Ijw3IaWDOQcMkldSPa7zXy79Z/UQ==", "requires": { "graceful-fs": "^4.1.2", "jsonfile": "^2.1.0", @@ -44993,27 +45936,119 @@ } }, "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", "requires": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", - "minimatch": "^3.0.4", + "minimatch": "^3.1.1", "once": "^1.3.0", "path-is-absolute": "^1.0.0" } }, + "graceful-fs": { + "version": "4.2.10", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==" + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" + }, + "hermes-parser": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.6.0.tgz", + "integrity": "sha512-Vf58jBZca2+QBLR9h7B7mdg8oFz2g5ILz1iVouZ5DOrOrAfBmPfJjdjDT8jrO0f+iJ4/hSRrQHqHIjSnTaLUDQ==", + "requires": { + "hermes-estree": "0.6.0" + } + }, "is-fullwidth-code-point": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" + }, + "jest-haste-map": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-27.5.1.tgz", + "integrity": "sha512-7GgkZ4Fw4NFbMSDSpZwXeBiIbx+t/46nJ2QitkOjvwPYyZmqttu2TDSimMHP1EkPOi4xUZAN1doE5Vd25H4Jng==", + "requires": { + "@jest/types": "^27.5.1", + "@types/graceful-fs": "^4.1.2", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "fsevents": "^2.3.2", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^27.5.1", + "jest-serializer": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "micromatch": "^4.0.4", + "walker": "^1.0.7" + } + }, + "jest-regex-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-27.5.1.tgz", + "integrity": "sha512-4bfKq2zie+x16okqDXjXn9ql2B0dScQu+vcwe4TvFVhkVyuWLqpZrZtXxLLWoXYgn0E87I6r6GRYHF7wFZBUvg==" + }, + "jest-serializer": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-27.5.1.tgz", + "integrity": "sha512-jZCyo6iIxO1aqUxpuBlwTDMkzOAJS4a3eYz3YzgxxVQFwLeSA7Jfq5cbqCY+JLvTDrWirgusI/0KwxKMgrdf7w==", + "requires": { + "@types/node": "*", + "graceful-fs": "^4.2.9" + } + }, + "jest-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-27.5.1.tgz", + "integrity": "sha512-Kv2o/8jNvX1MQ0KGtw480E/w4fBCDOnH6+6DmeKi6LZUIlKA5kwY0YNdlzaWTiVgxqAqik11QyxDOKk543aKXw==", + "requires": { + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "dependencies": { + "ci-info": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.6.2.tgz", + "integrity": "sha512-lVZdhvbEudris15CLytp2u6Y0p5EKfztae9Fqa189MfNmln9F33XuH69v5fvNfiRN5/0eAUz2yJL3mo+nhaRKg==" + }, + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" + } + } + }, + "jest-worker": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", + "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "requires": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + } + }, "jsonfile": { "version": "2.4.0", "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-2.4.0.tgz", - "integrity": "sha1-NzaitCi4e72gzIO1P6PWM6NcKug=", + "integrity": "sha512-PKllAqbgLgxHaj8TElYymKCAgrASebJrWpTnEkOaTowt23VKXXN0sUeriJ+eh7y6ufb/CC5ap11pz71/cM0hUw==", "requires": { "graceful-fs": "^4.1.6" } @@ -45026,23 +46061,13 @@ "p-locate": "^4.1.0" } }, - "metro-babel-transformer": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-babel-transformer/-/metro-babel-transformer-0.66.2.tgz", - "integrity": "sha512-aJ/7fc/Xkofw8Fqa51OTDhBzBz26mmpIWrXAZcPdQ8MSTt883EWncxeCEjasc79NJ89BRi7sOkkaWZo2sXlKvw==", - "requires": { - "@babel/core": "^7.14.0", - "hermes-parser": "0.4.7", - "metro-source-map": "0.66.2", - "nullthrows": "^1.1.1" - } - }, "metro-react-native-babel-preset": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.66.2.tgz", - "integrity": "sha512-H/nLBAz0MgfDloSe1FjyH4EnbokHFdncyERvLPXDACY3ROVRCeUyFNo70ywRGXW2NMbrV4H7KUyU4zkfWhC2HQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.70.3.tgz", + "integrity": "sha512-4Nxc1zEiHEu+GTdEMEsHnRgfaBkg8f/Td3+FcQ8NTSvs+xL3LBrQy6N07idWSQZHIdGFf+tTHvRfSIWLD8u8Tg==", "requires": { "@babel/core": "^7.14.0", + "@babel/plugin-proposal-async-generator-functions": "^7.0.0", "@babel/plugin-proposal-class-properties": "^7.0.0", "@babel/plugin-proposal-export-default-from": "^7.0.0", "@babel/plugin-proposal-nullish-coalescing-operator": "^7.0.0", @@ -45062,17 +46087,15 @@ "@babel/plugin-transform-destructuring": "^7.0.0", "@babel/plugin-transform-exponentiation-operator": "^7.0.0", "@babel/plugin-transform-flow-strip-types": "^7.0.0", - "@babel/plugin-transform-for-of": "^7.0.0", "@babel/plugin-transform-function-name": "^7.0.0", "@babel/plugin-transform-literals": "^7.0.0", "@babel/plugin-transform-modules-commonjs": "^7.0.0", - "@babel/plugin-transform-object-assign": "^7.0.0", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.0.0", "@babel/plugin-transform-parameters": "^7.0.0", "@babel/plugin-transform-react-display-name": "^7.0.0", "@babel/plugin-transform-react-jsx": "^7.0.0", "@babel/plugin-transform-react-jsx-self": "^7.0.0", "@babel/plugin-transform-react-jsx-source": "^7.0.0", - "@babel/plugin-transform-regenerator": "^7.0.0", "@babel/plugin-transform-runtime": "^7.0.0", "@babel/plugin-transform-shorthand-properties": "^7.0.0", "@babel/plugin-transform-spread": "^7.0.0", @@ -45085,46 +46108,80 @@ } }, "metro-source-map": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.66.2.tgz", - "integrity": "sha512-038tFmB7vSh73VQcDWIbr5O1m+WXWyYafDaOy+1A/2K308YP0oj33gbEgDnZsLZDwcJ+xt1x6KUEBIzlX4YGeQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.70.3.tgz", + "integrity": "sha512-zsYtZGrwRbbGEFHtmMqqeCH9K9aTGNVPsurMOWCUeQA3VGyVGXPGtLMC+CdAM9jLpUyg6jw2xh0esxi+tYH7Uw==", "requires": { "@babel/traverse": "^7.14.0", "@babel/types": "^7.0.0", "invariant": "^2.2.4", - "metro-symbolicate": "0.66.2", + "metro-symbolicate": "0.70.3", "nullthrows": "^1.1.1", - "ob1": "0.66.2", + "ob1": "0.70.3", "source-map": "^0.5.6", "vlq": "^1.0.0" } }, "metro-symbolicate": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.66.2.tgz", - "integrity": "sha512-u+DeQHyAFXVD7mVP+GST/894WHJ3i/U8oEJFnT7U3P52ZuLgX8n4tMNxhqZU12RcLR6etF8143aP0Ktx1gFLEQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.70.3.tgz", + "integrity": "sha512-JTYkF1dpeDUssQ84juE1ycnhHki2ylJBBdJE1JHtfu5oC+z1ElDbBdPHq90Uvt8HbRov/ZAnxvv7Zy6asS+WCA==", "requires": { "invariant": "^2.2.4", - "metro-source-map": "0.66.2", + "metro-source-map": "0.70.3", "nullthrows": "^1.1.1", "source-map": "^0.5.6", "through2": "^2.0.1", "vlq": "^1.0.0" } }, + "micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "requires": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "dependencies": { + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" + } + } + }, "mime-db": { - "version": "1.51.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.51.0.tgz", - "integrity": "sha512-5y8A56jg7XVQx2mbv1lu49NR4dokRnhZYTtL+KGfaa27uq4pSTXkwQkFJl4pkRMyNFz/EtYDSkiiEHx3F7UN6g==" + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" }, "mime-types": { - "version": "2.1.34", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.34.tgz", - "integrity": "sha512-6cP692WwGIs9XXdOO4++N+7qjqv0rqxxVvJ3VHPh/Sc9mVZcQP+ZGhkKiTvWMQRr2tbHkJP/Yn7Y0npb3ZBs4A==", + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", "requires": { - "mime-db": "1.51.0" + "mime-db": "1.52.0" + } + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "requires": { + "brace-expansion": "^1.1.7" } }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" + }, + "ob1": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.70.3.tgz", + "integrity": "sha512-Vy9GGhuXgDRY01QA6kdhToPd8AkLdLpX9GjH5kpqluVqTu70mgOm7tpGoJDZGaNbr9nJlJgnipqHJQRPORixIQ==" + }, "p-limit": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", @@ -45182,20 +46239,27 @@ "ansi-regex": "^5.0.1" } }, - "strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=" + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "requires": { + "has-flag": "^4.0.0" + } }, - "ws": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/ws/-/ws-1.1.5.tgz", - "integrity": "sha512-o3KqipXNUdS7wpQzBHSe180lBGO60SoK0yVo3CYJgb2MkobuWuBX6dhkYP5ORCLd55y+SaflMOV5fqAB53ux4w==", + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", "requires": { - "options": ">=0.0.5", - "ultron": "1.0.x" + "is-number": "^7.0.0" } }, + "ws": { + "version": "7.5.9", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.9.tgz", + "integrity": "sha512-F+P9Jil7UiSKSkppIiD94dN07AwvFixvLIj1Og1Rl9GGMuNipJnV9JzjD6XuqmAeiswGvUmNLjr5cFuXwNS77Q==" + }, "yargs": { "version": "15.4.1", "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", @@ -45225,55 +46289,90 @@ } } }, - "metro-babel-register": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-babel-register/-/metro-babel-register-0.66.2.tgz", - "integrity": "sha512-3F+vsVubUPJYKfVMeol8/7pd8CC287Rw92QYzJD8LEmI980xcgwMUEVBZ0UIAUwlLgiJG/f4Mwhuji2EeBXrPg==", - "requires": { - "@babel/core": "^7.14.0", - "@babel/plugin-proposal-nullish-coalescing-operator": "^7.0.0", - "@babel/plugin-proposal-optional-chaining": "^7.0.0", - "@babel/plugin-syntax-class-properties": "^7.0.0", - "@babel/plugin-transform-flow-strip-types": "^7.0.0", - "@babel/plugin-transform-modules-commonjs": "^7.0.0", - "@babel/register": "^7.0.0", - "escape-string-regexp": "^1.0.5" - } - }, "metro-babel-transformer": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-babel-transformer/-/metro-babel-transformer-0.66.2.tgz", - "integrity": "sha512-aJ/7fc/Xkofw8Fqa51OTDhBzBz26mmpIWrXAZcPdQ8MSTt883EWncxeCEjasc79NJ89BRi7sOkkaWZo2sXlKvw==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-babel-transformer/-/metro-babel-transformer-0.70.3.tgz", + "integrity": "sha512-bWhZRMn+mIOR/s3BDpFevWScz9sV8FGktVfMlF1eJBLoX24itHDbXvTktKBYi38PWIKcHedh6THSFpJogfuwNA==", "requires": { "@babel/core": "^7.14.0", - "hermes-parser": "0.4.7", - "metro-source-map": "0.66.2", + "hermes-parser": "0.6.0", + "metro-source-map": "0.70.3", "nullthrows": "^1.1.1" + }, + "dependencies": { + "hermes-parser": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.6.0.tgz", + "integrity": "sha512-Vf58jBZca2+QBLR9h7B7mdg8oFz2g5ILz1iVouZ5DOrOrAfBmPfJjdjDT8jrO0f+iJ4/hSRrQHqHIjSnTaLUDQ==", + "requires": { + "hermes-estree": "0.6.0" + } + }, + "metro-source-map": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.70.3.tgz", + "integrity": "sha512-zsYtZGrwRbbGEFHtmMqqeCH9K9aTGNVPsurMOWCUeQA3VGyVGXPGtLMC+CdAM9jLpUyg6jw2xh0esxi+tYH7Uw==", + "requires": { + "@babel/traverse": "^7.14.0", + "@babel/types": "^7.0.0", + "invariant": "^2.2.4", + "metro-symbolicate": "0.70.3", + "nullthrows": "^1.1.1", + "ob1": "0.70.3", + "source-map": "^0.5.6", + "vlq": "^1.0.0" + } + }, + "metro-symbolicate": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.70.3.tgz", + "integrity": "sha512-JTYkF1dpeDUssQ84juE1ycnhHki2ylJBBdJE1JHtfu5oC+z1ElDbBdPHq90Uvt8HbRov/ZAnxvv7Zy6asS+WCA==", + "requires": { + "invariant": "^2.2.4", + "metro-source-map": "0.70.3", + "nullthrows": "^1.1.1", + "source-map": "^0.5.6", + "through2": "^2.0.1", + "vlq": "^1.0.0" + } + }, + "ob1": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.70.3.tgz", + "integrity": "sha512-Vy9GGhuXgDRY01QA6kdhToPd8AkLdLpX9GjH5kpqluVqTu70mgOm7tpGoJDZGaNbr9nJlJgnipqHJQRPORixIQ==" + } } }, "metro-cache": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-cache/-/metro-cache-0.66.2.tgz", - "integrity": "sha512-5QCYJtJOHoBSbL3H4/Fpl36oA697C3oYHqsce+Hk/dh2qtODUGpS3gOBhvP1B8iB+H8jJMyR75lZq129LJEsIQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-cache/-/metro-cache-0.70.3.tgz", + "integrity": "sha512-iCix/+z812fUqa6KlOxaTkY6LQQDoXIe/VljXkGIvpygSCmYyhjQpfQVZEVVPezFmUBYXNdabdQ6cYx6JX3yMg==", "requires": { - "metro-core": "0.66.2", - "mkdirp": "^0.5.1", + "metro-core": "0.70.3", "rimraf": "^2.5.4" }, "dependencies": { "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", "requires": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", - "minimatch": "^3.0.4", + "minimatch": "^3.1.1", "once": "^1.3.0", "path-is-absolute": "^1.0.0" } }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "requires": { + "brace-expansion": "^1.1.7" + } + }, "rimraf": { "version": "2.7.1", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", @@ -45285,47 +46384,27 @@ } }, "metro-cache-key": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-cache-key/-/metro-cache-key-0.66.2.tgz", - "integrity": "sha512-WtkNmRt41qOpHh1MkNA4nLiQ/m7iGL90ysSKD+fcLqlUnOBKJptPQm0ZUv8Kfqk18ddWX2KmsSbq+Sf3I6XohQ==" + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-cache-key/-/metro-cache-key-0.70.3.tgz", + "integrity": "sha512-0zpw+IcpM3hmGd5sKMdxNv3sbOIUYnMUvx1/yaM6vNRReSPmOLX0bP8fYf3CGgk8NEreZ1OHbVsuw7bdKt40Mw==" }, "metro-config": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-config/-/metro-config-0.66.2.tgz", - "integrity": "sha512-0C+PrKKIBNNzLZUKN/8ZDJS2U5FLMOTXDWbvBHIdqb6YXz8WplXR2+xlSlaSCCi5b+GR7cWFWUNeKA4GQS1/AQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-config/-/metro-config-0.70.3.tgz", + "integrity": "sha512-SSCDjSTygoCgzoj61DdrBeJzZDRwQxUEfcgc6t6coxWSExXNR4mOngz0q4SAam49Bmjq9J2Jft6qUKnUTPrRgA==", "requires": { "cosmiconfig": "^5.0.5", "jest-validate": "^26.5.2", - "metro": "0.66.2", - "metro-cache": "0.66.2", - "metro-core": "0.66.2", - "metro-runtime": "0.66.2" + "metro": "0.70.3", + "metro-cache": "0.70.3", + "metro-core": "0.70.3", + "metro-runtime": "0.70.3" }, "dependencies": { - "@jest/types": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", - "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", - "requires": { - "@types/istanbul-lib-coverage": "^2.0.0", - "@types/istanbul-reports": "^3.0.0", - "@types/node": "*", - "@types/yargs": "^15.0.0", - "chalk": "^4.0.0" - } - }, - "@types/istanbul-reports": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.1.tgz", - "integrity": "sha512-c3mAZEuK0lvBp8tmuL74XRKn1+y2dcwOUpH7x4WrF6gk1GIgiluDRgMYQtw2OFcBvAJWlt6ASU3tSqxp0Uu0Aw==", - "requires": { - "@types/istanbul-lib-report": "*" - } - }, "camelcase": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.1.tgz", - "integrity": "sha512-tVI4q5jjFV5CavAU8DXfza/TJcZutVKo/5Foskmsqcm0MsL91moHvwiGNnqaa2o6PF/7yT5ikDRcVcl8Rj6LCA==" + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==" }, "cosmiconfig": { "version": "5.2.1", @@ -45341,17 +46420,12 @@ "import-fresh": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", - "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", + "integrity": "sha512-eZ5H8rcgYazHbKC3PG4ClHNykCSxtAhxSSEM+2mb+7evD2CKF5V7c0dNum7AdpDh0ZdICwZY9sRSn8f+KH96sg==", "requires": { "caller-path": "^2.0.0", "resolve-from": "^3.0.0" } }, - "jest-get-type": { - "version": "26.3.0", - "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", - "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==" - }, "jest-validate": { "version": "26.6.2", "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-26.6.2.tgz", @@ -45368,7 +46442,7 @@ "parse-json": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", - "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "integrity": "sha512-aOIos8bujGN93/8Ox/jPLh7RwVnPEysynVFE+fQZyg6jKELEHwzgKdLRFHUgXJL6kylijVSBC4BvN9OmsB48Rw==", "requires": { "error-ex": "^1.3.1", "json-parse-better-errors": "^1.0.1" @@ -45377,28 +46451,196 @@ } }, "metro-core": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-core/-/metro-core-0.66.2.tgz", - "integrity": "sha512-JieLZkef/516yxXYvQxWnf3OWw5rcgWRy76K8JV/wr/i8LGVGulPAXlIi445/QZzXVydzRVASKAEVqyxM5F4mA==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-core/-/metro-core-0.70.3.tgz", + "integrity": "sha512-NzfHB/w5R7yLaOeU1tzPTbBzCRsYSvpKJkLMP0yudszKZzIAZqNdjoEJ9GZ688Wi0ynZxcU0BxukXh4my80ZBw==", "requires": { - "jest-haste-map": "^26.5.2", + "jest-haste-map": "^27.3.1", "lodash.throttle": "^4.1.1", - "metro-resolver": "0.66.2" + "metro-resolver": "0.70.3" + }, + "dependencies": { + "@jest/types": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.5.1.tgz", + "integrity": "sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw==", + "requires": { + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^16.0.0", + "chalk": "^4.0.0" + } + }, + "@types/yargs": { + "version": "16.0.4", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-16.0.4.tgz", + "integrity": "sha512-T8Yc9wt/5LbJyCaLiHPReJa0kApcIgJ7Bn735GjItUfh08Z1pJvu8QZqb9s+mMvKV6WUQRV7K2R46YbjMXTTJw==", + "requires": { + "@types/yargs-parser": "*" + } + }, + "anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "requires": { + "fill-range": "^7.0.1" + } + }, + "ci-info": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.6.2.tgz", + "integrity": "sha512-lVZdhvbEudris15CLytp2u6Y0p5EKfztae9Fqa189MfNmln9F33XuH69v5fvNfiRN5/0eAUz2yJL3mo+nhaRKg==" + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "graceful-fs": { + "version": "4.2.10", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==" + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" + }, + "jest-haste-map": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-27.5.1.tgz", + "integrity": "sha512-7GgkZ4Fw4NFbMSDSpZwXeBiIbx+t/46nJ2QitkOjvwPYyZmqttu2TDSimMHP1EkPOi4xUZAN1doE5Vd25H4Jng==", + "requires": { + "@jest/types": "^27.5.1", + "@types/graceful-fs": "^4.1.2", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "fsevents": "^2.3.2", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^27.5.1", + "jest-serializer": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "micromatch": "^4.0.4", + "walker": "^1.0.7" + } + }, + "jest-regex-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-27.5.1.tgz", + "integrity": "sha512-4bfKq2zie+x16okqDXjXn9ql2B0dScQu+vcwe4TvFVhkVyuWLqpZrZtXxLLWoXYgn0E87I6r6GRYHF7wFZBUvg==" + }, + "jest-serializer": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-27.5.1.tgz", + "integrity": "sha512-jZCyo6iIxO1aqUxpuBlwTDMkzOAJS4a3eYz3YzgxxVQFwLeSA7Jfq5cbqCY+JLvTDrWirgusI/0KwxKMgrdf7w==", + "requires": { + "@types/node": "*", + "graceful-fs": "^4.2.9" + } + }, + "jest-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-27.5.1.tgz", + "integrity": "sha512-Kv2o/8jNvX1MQ0KGtw480E/w4fBCDOnH6+6DmeKi6LZUIlKA5kwY0YNdlzaWTiVgxqAqik11QyxDOKk543aKXw==", + "requires": { + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "dependencies": { + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" + } + } + }, + "jest-worker": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", + "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "requires": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + } + }, + "micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "requires": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "dependencies": { + "picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" + } + } + }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "requires": { + "has-flag": "^4.0.0" + } + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "requires": { + "is-number": "^7.0.0" + } + } } }, "metro-hermes-compiler": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-hermes-compiler/-/metro-hermes-compiler-0.66.2.tgz", - "integrity": "sha512-nCVL1g9uR6vrw5+X1wjwZruRyMkndnzGRMqjqoljf+nGEqBTD607CR7elXw4fMWn/EM+1y0Vdq5altUu9LdgCA==" + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-hermes-compiler/-/metro-hermes-compiler-0.70.3.tgz", + "integrity": "sha512-W6WttLi4E72JL/NyteQ84uxYOFMibe0PUr9aBKuJxxfCq6QRnJKOVcNY0NLW0He2tneXGk+8ZsNz8c0flEvYqg==" }, "metro-inspector-proxy": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-inspector-proxy/-/metro-inspector-proxy-0.66.2.tgz", - "integrity": "sha512-gnLc9121eznwP0iiA9tCBW8qZjwIsCgwHWMF1g1Qaki9le9tzeJv3dK4/lFNGxyfSaLO7vahQEhsEYsiRnTROg==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-inspector-proxy/-/metro-inspector-proxy-0.70.3.tgz", + "integrity": "sha512-qQoNdPGrmyoJSWYkxSDpTaAI8xyqVdNDVVj9KRm1PG8niSuYmrCCFGLLFsMvkVYwsCWUGHoGBx0UoAzVp14ejw==", "requires": { "connect": "^3.6.5", "debug": "^2.2.0", - "ws": "^1.1.5", + "ws": "^7.5.1", "yargs": "^15.3.1" }, "dependencies": { @@ -45407,11 +46649,6 @@ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" - }, "debug": { "version": "2.6.9", "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", @@ -45492,13 +46729,9 @@ } }, "ws": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/ws/-/ws-1.1.5.tgz", - "integrity": "sha512-o3KqipXNUdS7wpQzBHSe180lBGO60SoK0yVo3CYJgb2MkobuWuBX6dhkYP5ORCLd55y+SaflMOV5fqAB53ux4w==", - "requires": { - "options": ">=0.0.5", - "ultron": "1.0.x" - } + "version": "7.5.9", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.9.tgz", + "integrity": "sha512-F+P9Jil7UiSKSkppIiD94dN07AwvFixvLIj1Og1Rl9GGMuNipJnV9JzjD6XuqmAeiswGvUmNLjr5cFuXwNS77Q==" }, "yargs": { "version": "15.4.1", @@ -45530,41 +46763,21 @@ } }, "metro-minify-uglify": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-minify-uglify/-/metro-minify-uglify-0.66.2.tgz", - "integrity": "sha512-7TUK+L5CmB5x1PVnFbgmjzHW4CUadq9H5jgp0HfFoWT1skXAyEsx0DHkKDXwnot0khnNhBOEfl62ctQOnE110Q==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-minify-uglify/-/metro-minify-uglify-0.70.3.tgz", + "integrity": "sha512-oHyjV9WDqOlDE1FPtvs6tIjjeY/oP1PNUPYL1wqyYtqvjN+zzAOrcbsAAL1sv+WARaeiMsWkF2bwtNo+Hghoog==", "requires": { "uglify-es": "^3.1.9" - }, - "dependencies": { - "commander": { - "version": "2.13.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.13.0.tgz", - "integrity": "sha512-MVuS359B+YzaWqjCL/c+22gfryv+mCBPHAv3zyVI2GN8EY6IRP8VwtasXn8jyyhvvq84R4ImN1OKRtcbIasjYA==" - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" - }, - "uglify-es": { - "version": "3.3.9", - "resolved": "https://registry.npmjs.org/uglify-es/-/uglify-es-3.3.9.tgz", - "integrity": "sha512-r+MU0rfv4L/0eeW3xZrd16t4NZfK8Ld4SWVglYBb7ez5uXFWHuVRs6xCTrf1yirs9a4j4Y27nn7SRfO6v67XsQ==", - "requires": { - "commander": "~2.13.0", - "source-map": "~0.6.1" - } - } } }, "metro-react-native-babel-preset": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.66.2.tgz", - "integrity": "sha512-H/nLBAz0MgfDloSe1FjyH4EnbokHFdncyERvLPXDACY3ROVRCeUyFNo70ywRGXW2NMbrV4H7KUyU4zkfWhC2HQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.70.3.tgz", + "integrity": "sha512-4Nxc1zEiHEu+GTdEMEsHnRgfaBkg8f/Td3+FcQ8NTSvs+xL3LBrQy6N07idWSQZHIdGFf+tTHvRfSIWLD8u8Tg==", "dev": true, "requires": { "@babel/core": "^7.14.0", + "@babel/plugin-proposal-async-generator-functions": "^7.0.0", "@babel/plugin-proposal-class-properties": "^7.0.0", "@babel/plugin-proposal-export-default-from": "^7.0.0", "@babel/plugin-proposal-nullish-coalescing-operator": "^7.0.0", @@ -45584,17 +46797,15 @@ "@babel/plugin-transform-destructuring": "^7.0.0", "@babel/plugin-transform-exponentiation-operator": "^7.0.0", "@babel/plugin-transform-flow-strip-types": "^7.0.0", - "@babel/plugin-transform-for-of": "^7.0.0", "@babel/plugin-transform-function-name": "^7.0.0", "@babel/plugin-transform-literals": "^7.0.0", "@babel/plugin-transform-modules-commonjs": "^7.0.0", - "@babel/plugin-transform-object-assign": "^7.0.0", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.0.0", "@babel/plugin-transform-parameters": "^7.0.0", "@babel/plugin-transform-react-display-name": "^7.0.0", "@babel/plugin-transform-react-jsx": "^7.0.0", "@babel/plugin-transform-react-jsx-self": "^7.0.0", "@babel/plugin-transform-react-jsx-source": "^7.0.0", - "@babel/plugin-transform-regenerator": "^7.0.0", "@babel/plugin-transform-runtime": "^7.0.0", "@babel/plugin-transform-shorthand-properties": "^7.0.0", "@babel/plugin-transform-spread": "^7.0.0", @@ -45615,134 +46826,60 @@ } }, "metro-react-native-babel-transformer": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-react-native-babel-transformer/-/metro-react-native-babel-transformer-0.66.2.tgz", - "integrity": "sha512-z1ab7ihIT0pJrwgi9q2IH+LcW/xUWMQ0hH+Mrk7wbKQB0RnJdXFoxphrfoVHBHMUu+TBPetUcEkKawkK1e7Cng==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-transformer/-/metro-react-native-babel-transformer-0.70.3.tgz", + "integrity": "sha512-WKBU6S/G50j9cfmFM4k4oRYprd8u3qjleD4so1E2zbTNILg+gYla7ZFGCAvi2G0ZcqS2XuGCR375c2hF6VVvwg==", + "dev": true, "requires": { "@babel/core": "^7.14.0", "babel-preset-fbjs": "^3.4.0", - "hermes-parser": "0.4.7", - "metro-babel-transformer": "0.66.2", - "metro-react-native-babel-preset": "0.66.2", - "metro-source-map": "0.66.2", + "hermes-parser": "0.6.0", + "metro-babel-transformer": "0.70.3", + "metro-react-native-babel-preset": "0.70.3", + "metro-source-map": "0.70.3", "nullthrows": "^1.1.1" - }, - "dependencies": { - "metro-babel-transformer": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-babel-transformer/-/metro-babel-transformer-0.66.2.tgz", - "integrity": "sha512-aJ/7fc/Xkofw8Fqa51OTDhBzBz26mmpIWrXAZcPdQ8MSTt883EWncxeCEjasc79NJ89BRi7sOkkaWZo2sXlKvw==", - "requires": { - "@babel/core": "^7.14.0", - "hermes-parser": "0.4.7", - "metro-source-map": "0.66.2", - "nullthrows": "^1.1.1" - } - }, - "metro-react-native-babel-preset": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.66.2.tgz", - "integrity": "sha512-H/nLBAz0MgfDloSe1FjyH4EnbokHFdncyERvLPXDACY3ROVRCeUyFNo70ywRGXW2NMbrV4H7KUyU4zkfWhC2HQ==", - "requires": { - "@babel/core": "^7.14.0", - "@babel/plugin-proposal-class-properties": "^7.0.0", - "@babel/plugin-proposal-export-default-from": "^7.0.0", - "@babel/plugin-proposal-nullish-coalescing-operator": "^7.0.0", - "@babel/plugin-proposal-object-rest-spread": "^7.0.0", - "@babel/plugin-proposal-optional-catch-binding": "^7.0.0", - "@babel/plugin-proposal-optional-chaining": "^7.0.0", - "@babel/plugin-syntax-dynamic-import": "^7.0.0", - "@babel/plugin-syntax-export-default-from": "^7.0.0", - "@babel/plugin-syntax-flow": "^7.2.0", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.0.0", - "@babel/plugin-syntax-optional-chaining": "^7.0.0", - "@babel/plugin-transform-arrow-functions": "^7.0.0", - "@babel/plugin-transform-async-to-generator": "^7.0.0", - "@babel/plugin-transform-block-scoping": "^7.0.0", - "@babel/plugin-transform-classes": "^7.0.0", - "@babel/plugin-transform-computed-properties": "^7.0.0", - "@babel/plugin-transform-destructuring": "^7.0.0", - "@babel/plugin-transform-exponentiation-operator": "^7.0.0", - "@babel/plugin-transform-flow-strip-types": "^7.0.0", - "@babel/plugin-transform-for-of": "^7.0.0", - "@babel/plugin-transform-function-name": "^7.0.0", - "@babel/plugin-transform-literals": "^7.0.0", - "@babel/plugin-transform-modules-commonjs": "^7.0.0", - "@babel/plugin-transform-object-assign": "^7.0.0", - "@babel/plugin-transform-parameters": "^7.0.0", - "@babel/plugin-transform-react-display-name": "^7.0.0", - "@babel/plugin-transform-react-jsx": "^7.0.0", - "@babel/plugin-transform-react-jsx-self": "^7.0.0", - "@babel/plugin-transform-react-jsx-source": "^7.0.0", - "@babel/plugin-transform-regenerator": "^7.0.0", - "@babel/plugin-transform-runtime": "^7.0.0", - "@babel/plugin-transform-shorthand-properties": "^7.0.0", - "@babel/plugin-transform-spread": "^7.0.0", - "@babel/plugin-transform-sticky-regex": "^7.0.0", - "@babel/plugin-transform-template-literals": "^7.0.0", - "@babel/plugin-transform-typescript": "^7.5.0", - "@babel/plugin-transform-unicode-regex": "^7.0.0", - "@babel/template": "^7.0.0", - "react-refresh": "^0.4.0" - } - }, - "metro-source-map": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.66.2.tgz", - "integrity": "sha512-038tFmB7vSh73VQcDWIbr5O1m+WXWyYafDaOy+1A/2K308YP0oj33gbEgDnZsLZDwcJ+xt1x6KUEBIzlX4YGeQ==", - "requires": { - "@babel/traverse": "^7.14.0", - "@babel/types": "^7.0.0", - "invariant": "^2.2.4", - "metro-symbolicate": "0.66.2", - "nullthrows": "^1.1.1", - "ob1": "0.66.2", - "source-map": "^0.5.6", - "vlq": "^1.0.0" - } - }, - "react-refresh": { - "version": "0.4.3", - "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.4.3.tgz", - "integrity": "sha512-Hwln1VNuGl/6bVwnd0Xdn1e84gT/8T9aYNL+HAKDArLCS7LWjwr7StE30IEYbIkx0Vi3vs+coQxe+SQDbGbbpA==" - } } }, "metro-resolver": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-resolver/-/metro-resolver-0.66.2.tgz", - "integrity": "sha512-pXQAJR/xauRf4kWFj2/hN5a77B4jLl0Fom5I3PHp6Arw/KxSBp0cnguXpGLwNQ6zQC0nxKCoYGL9gQpzMnN7Hw==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-resolver/-/metro-resolver-0.70.3.tgz", + "integrity": "sha512-5Pc5S/Gs4RlLbziuIWtvtFd9GRoILlaRC8RZDVq5JZWcWHywKy/PjNmOBNhpyvtRlzpJfy/ssIfLhu8zINt1Mw==", "requires": { "absolute-path": "^0.0.0" } }, "metro-runtime": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-runtime/-/metro-runtime-0.66.2.tgz", - "integrity": "sha512-vFhKBk2ot9FS4b+2v0OTa/guCF/QDAOJubY0CNg7PzCS5+w4y3IvZIcPX4SSS1t8pYEZBLvtdtTDarlDl81xmg==" + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-runtime/-/metro-runtime-0.70.3.tgz", + "integrity": "sha512-22xU7UdXZacniTIDZgN2EYtmfau2pPyh97Dcs+cWrLcJYgfMKjWBtesnDcUAQy3PHekDYvBdJZkoQUeskYTM+w==", + "requires": { + "@babel/runtime": "^7.0.0" + } }, "metro-source-map": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.66.2.tgz", - "integrity": "sha512-038tFmB7vSh73VQcDWIbr5O1m+WXWyYafDaOy+1A/2K308YP0oj33gbEgDnZsLZDwcJ+xt1x6KUEBIzlX4YGeQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.70.3.tgz", + "integrity": "sha512-zsYtZGrwRbbGEFHtmMqqeCH9K9aTGNVPsurMOWCUeQA3VGyVGXPGtLMC+CdAM9jLpUyg6jw2xh0esxi+tYH7Uw==", + "dev": true, "requires": { "@babel/traverse": "^7.14.0", "@babel/types": "^7.0.0", "invariant": "^2.2.4", - "metro-symbolicate": "0.66.2", + "metro-symbolicate": "0.70.3", "nullthrows": "^1.1.1", - "ob1": "0.66.2", + "ob1": "0.70.3", "source-map": "^0.5.6", "vlq": "^1.0.0" } }, "metro-symbolicate": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.66.2.tgz", - "integrity": "sha512-u+DeQHyAFXVD7mVP+GST/894WHJ3i/U8oEJFnT7U3P52ZuLgX8n4tMNxhqZU12RcLR6etF8143aP0Ktx1gFLEQ==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.70.3.tgz", + "integrity": "sha512-JTYkF1dpeDUssQ84juE1ycnhHki2ylJBBdJE1JHtfu5oC+z1ElDbBdPHq90Uvt8HbRov/ZAnxvv7Zy6asS+WCA==", + "dev": true, "requires": { "invariant": "^2.2.4", - "metro-source-map": "0.66.2", + "metro-source-map": "0.70.3", "nullthrows": "^1.1.1", "source-map": "^0.5.6", "through2": "^2.0.1", @@ -45750,9 +46887,9 @@ } }, "metro-transform-plugins": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-transform-plugins/-/metro-transform-plugins-0.66.2.tgz", - "integrity": "sha512-KTvqplh0ut7oDKovvDG6yzXM02R6X+9b2oVG+qYq8Zd3aCGTi51ASx4ThCNkAHyEvCuJdYg9fxXTL+j+wvhB5w==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-transform-plugins/-/metro-transform-plugins-0.70.3.tgz", + "integrity": "sha512-dQRIJoTkWZN2IVS2KzgS1hs7ZdHDX3fS3esfifPkqFAEwHiLctCf0EsPgIknp0AjMLvmGWfSLJigdRB/dc0ASw==", "requires": { "@babel/core": "^7.14.0", "@babel/generator": "^7.14.0", @@ -45762,23 +46899,58 @@ } }, "metro-transform-worker": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/metro-transform-worker/-/metro-transform-worker-0.66.2.tgz", - "integrity": "sha512-dO4PtYOMGB7Vzte8aIzX39xytODhmbJrBYPu+zYzlDjyefJZT7BkZ0LkPIThtyJi96xWcGqi9JBSo0CeRupAHw==", + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-transform-worker/-/metro-transform-worker-0.70.3.tgz", + "integrity": "sha512-MtVVsnHhhBOp9GRLCdAb2mD1dTCsIzT4+m34KMRdBDCEbDIb90YafT5prpU8qbj5uKd0o2FOQdrJ5iy5zQilHw==", "requires": { "@babel/core": "^7.14.0", "@babel/generator": "^7.14.0", "@babel/parser": "^7.14.0", "@babel/types": "^7.0.0", "babel-preset-fbjs": "^3.4.0", - "metro": "0.66.2", - "metro-babel-transformer": "0.66.2", - "metro-cache": "0.66.2", - "metro-cache-key": "0.66.2", - "metro-hermes-compiler": "0.66.2", - "metro-source-map": "0.66.2", - "metro-transform-plugins": "0.66.2", + "metro": "0.70.3", + "metro-babel-transformer": "0.70.3", + "metro-cache": "0.70.3", + "metro-cache-key": "0.70.3", + "metro-hermes-compiler": "0.70.3", + "metro-source-map": "0.70.3", + "metro-transform-plugins": "0.70.3", "nullthrows": "^1.1.1" + }, + "dependencies": { + "metro-source-map": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.70.3.tgz", + "integrity": "sha512-zsYtZGrwRbbGEFHtmMqqeCH9K9aTGNVPsurMOWCUeQA3VGyVGXPGtLMC+CdAM9jLpUyg6jw2xh0esxi+tYH7Uw==", + "requires": { + "@babel/traverse": "^7.14.0", + "@babel/types": "^7.0.0", + "invariant": "^2.2.4", + "metro-symbolicate": "0.70.3", + "nullthrows": "^1.1.1", + "ob1": "0.70.3", + "source-map": "^0.5.6", + "vlq": "^1.0.0" + } + }, + "metro-symbolicate": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.70.3.tgz", + "integrity": "sha512-JTYkF1dpeDUssQ84juE1ycnhHki2ylJBBdJE1JHtfu5oC+z1ElDbBdPHq90Uvt8HbRov/ZAnxvv7Zy6asS+WCA==", + "requires": { + "invariant": "^2.2.4", + "metro-source-map": "0.70.3", + "nullthrows": "^1.1.1", + "source-map": "^0.5.6", + "through2": "^2.0.1", + "vlq": "^1.0.0" + } + }, + "ob1": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.70.3.tgz", + "integrity": "sha512-Vy9GGhuXgDRY01QA6kdhToPd8AkLdLpX9GjH5kpqluVqTu70mgOm7tpGoJDZGaNbr9nJlJgnipqHJQRPORixIQ==" + } } }, "microevent.ts": { @@ -45854,7 +47026,8 @@ "mimic-fn": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", - "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==" + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true }, "mimic-response": { "version": "1.0.1", @@ -46275,11 +47448,6 @@ } } }, - "mockdate": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/mockdate/-/mockdate-3.0.5.tgz", - "integrity": "sha512-iniQP4rj1FhBdBYS/+eQv7j1tadJ9lJtdzgOpvsOHng/GbcDh2Fhdeq+ZRldrPYdXvCyfFUmFeEwEGXZB5I/AQ==" - }, "modify-values": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/modify-values/-/modify-values-1.0.1.tgz", @@ -46470,9 +47638,9 @@ } }, "nocache": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/nocache/-/nocache-2.1.0.tgz", - "integrity": "sha512-0L9FvHG3nfnnmaEQPjT9xhfN4ISk0A8/2j4M37Np4mcDesJjHgEUfgPhdCyZuFI954tjokaIj/A3NdpFNdEh4Q==" + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/nocache/-/nocache-3.0.4.tgz", + "integrity": "sha512-WDD0bdg9mbq6F4mRxEYcPWwfA1vxd0mrvKOyxI7Xj/atfRHVeutzuWByG//jfm4uPzp0y4Kj051EORCBSQMycw==" }, "nock": { "version": "12.0.3", @@ -46685,6 +47853,7 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, "requires": { "remove-trailing-separator": "^1.0.1" } @@ -47331,7 +48500,7 @@ "num2fraction": { "version": "1.2.2", "resolved": "https://registry.npmjs.org/num2fraction/-/num2fraction-1.2.2.tgz", - "integrity": "sha1-b2gragJ6Tp3fpFZM0lidHU5mnt4=", + "integrity": "sha512-Y1wZESM7VUThYY+4W+X4ySH2maqcA+p7UR+w8VWNWVAd6lwuXXWz/w/Cz43J/dI2I+PS6wD5N+bJUF+gjWvIqg==", "dev": true }, "number-is-nan": { @@ -47748,9 +48917,10 @@ "dev": true }, "ob1": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.66.2.tgz", - "integrity": "sha512-RFewnL/RjE0qQBOuM+2bbY96zmJPIge/aDtsiDbLSb+MOiK8CReAhBHDgL+zrA3F1hQk00lMWpUwYcep750plA==" + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.70.3.tgz", + "integrity": "sha512-Vy9GGhuXgDRY01QA6kdhToPd8AkLdLpX9GjH5kpqluVqTu70mgOm7tpGoJDZGaNbr9nJlJgnipqHJQRPORixIQ==", + "dev": true }, "object-assign": { "version": "4.1.1", @@ -48627,6 +49797,7 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, "requires": { "mimic-fn": "^1.0.0" } @@ -48659,11 +49830,6 @@ "wordwrap": "~1.0.0" } }, - "options": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/options/-/options-0.0.6.tgz", - "integrity": "sha1-7CLTEoBrtT5zF3Pnza788cZDEo8=" - }, "ora": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/ora/-/ora-4.0.2.tgz", @@ -48809,7 +49975,7 @@ "p-defer": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", - "integrity": "sha1-n26xgvbJqozXQwBKfU+WsZaw+ww=", + "integrity": "sha512-wB3wfAxZpk2AzOfUMJNL+d36xothRSyj8EXOa4f6GMqYDN9BJaaSISbsk+wS9abmnebVw95C2Kb5t85UmpCxuw==", "dev": true }, "p-event": { @@ -49585,11 +50751,6 @@ "version": "1.5.1", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" - }, - "xmlbuilder": { - "version": "15.1.1", - "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-15.1.1.tgz", - "integrity": "sha512-yMqGBqtXyeN1e3TGYvgNgDVZ3j84W4cwkOXQswghol6APgZWaff9lnbvN7MHYJOiXsvGPXtjTYJEiC9J2wv9Eg==" } } }, @@ -49632,14 +50793,21 @@ } }, "popmotion": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/popmotion/-/popmotion-11.0.3.tgz", - "integrity": "sha512-Y55FLdj3UxkR7Vl3s7Qr4e9m0onSnP8W7d/xQLsoJM40vs6UKHFdygs6SWryasTZYqugMjm3BepCF4CWXDiHgA==", + "version": "11.0.5", + "resolved": "https://registry.npmjs.org/popmotion/-/popmotion-11.0.5.tgz", + "integrity": "sha512-la8gPM1WYeFznb/JqF4GiTkRRPZsfaj2+kCxqQgr2MJylMmIKUwBfWW8Wa5fml/8gmtlD5yI01MP1QCZPWmppA==", "requires": { - "framesync": "6.0.1", + "framesync": "6.1.2", "hey-listen": "^1.0.8", - "style-value-types": "5.0.0", - "tslib": "^2.1.0" + "style-value-types": "5.1.2", + "tslib": "2.4.0" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } } }, "portfinder": { @@ -50383,7 +51551,7 @@ "pretty-hrtime": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/pretty-hrtime/-/pretty-hrtime-1.0.3.tgz", - "integrity": "sha1-t+PqQkNaTJsnWdmeDyAesZWALuE=", + "integrity": "sha512-66hKPCr+72mlfiSjlEB1+45IjXSqvVAIy6mocupoww4tBFE9R9IhwwUGoI4G++Tc9Aq+2rxOt0RFU6gPcrte0A==", "dev": true }, "prismjs": { @@ -50395,7 +51563,8 @@ "private": { "version": "0.1.8", "resolved": "https://registry.npmjs.org/private/-/private-0.1.8.tgz", - "integrity": "sha512-VvivMrbvd2nKkiG38qjULzlc+4Vx4wm/whI9pQD35YrARNnhxeiRktSOhSukRLFNlzg6Br/cJPet5J/u19r/mg==" + "integrity": "sha512-VvivMrbvd2nKkiG38qjULzlc+4Vx4wm/whI9pQD35YrARNnhxeiRktSOhSukRLFNlzg6Br/cJPet5J/u19r/mg==", + "dev": true }, "proc-log": { "version": "2.0.1", @@ -50420,6 +51589,14 @@ "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", "dev": true }, + "promise": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/promise/-/promise-8.3.0.tgz", + "integrity": "sha512-rZPNPKTOYVNEEKFaq1HqTgOwZD+4/YHS5ukLzQCypkj+OkYx7iv0mA91lJlpPPZ8vMau3IIGj5Qlwrx+8iiSmg==", + "requires": { + "asap": "~2.0.6" + } + }, "promise-all-reject-late": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/promise-all-reject-late/-/promise-all-reject-late-1.0.1.tgz", @@ -50931,7 +52108,6 @@ "version": "2.4.2", "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", - "dev": true, "requires": { "kleur": "^3.0.3", "sisteransi": "^1.0.5" @@ -51436,12 +52612,11 @@ } }, "react": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react/-/react-17.0.2.tgz", - "integrity": "sha512-gnhPt75i/dq/z3/6q/0asP78D0u592D5L1pd7M8P+dck6Fu/jJeL6iVVK23fptSUZj8Vjf++7wXA8UNclGQcbA==", + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react/-/react-18.2.0.tgz", + "integrity": "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==", "requires": { - "loose-envify": "^1.1.0", - "object-assign": "^4.1.1" + "loose-envify": "^1.1.0" } }, "react-autosize-textarea": { @@ -51460,9 +52635,9 @@ "integrity": "sha512-nkP1w1LGe8PzhtOQO10xSDTsMWAYVj/Wm5c7ORk7CBngjCE7hHsknFRtosT5qMYkwWs8wSiU0sBwZ8dyzRbNEQ==" }, "react-devtools-core": { - "version": "4.21.0", - "resolved": "https://registry.npmjs.org/react-devtools-core/-/react-devtools-core-4.21.0.tgz", - "integrity": "sha512-clGWwJHV5MHwTwYyKc+7FZHwzdbzrD2/AoZSkicUcr6YLc3Za9a9FaLhccWDHfjQ+ron9yzNhDT6Tv+FiPkD3g==", + "version": "4.24.0", + "resolved": "https://registry.npmjs.org/react-devtools-core/-/react-devtools-core-4.24.0.tgz", + "integrity": "sha512-Rw7FzYOOzcfyUPaAm9P3g0tFdGqGq2LLiAI+wjYcp6CsF3DeeMrRS3HZAho4s273C29G/DJhx0e8BpRE/QZNGg==", "requires": { "shell-quote": "^1.6.1", "ws": "^7" @@ -51519,13 +52694,22 @@ "dev": true }, "react-dom": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-17.0.2.tgz", - "integrity": "sha512-s4h96KtLDUQlsENhMn1ar8t2bEa+q/YAtj8pPPdIjPDGBDIVNsrD9aXNWqspUe6AzKCIG0C1HZZLqLV7qpOBGA==", + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", + "integrity": "sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==", "requires": { "loose-envify": "^1.1.0", - "object-assign": "^4.1.1", - "scheduler": "^0.20.2" + "scheduler": "^0.23.0" + }, + "dependencies": { + "scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "requires": { + "loose-envify": "^1.1.0" + } + } } }, "react-easy-crop": { @@ -51575,11 +52759,6 @@ "integrity": "sha512-rtGImPZ0YyLrscKI9xTpV8psd6I8VAtjKCzQDlzyDvqJA8XOW78TXYQwNRNd8g8JZnDu8q9Fu/1v4HPAVwVdHA==", "dev": true }, - "react-freeze": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/react-freeze/-/react-freeze-1.0.3.tgz", - "integrity": "sha512-ZnXwLQnGzrDpHBHiC56TXFXvmolPeMjTn1UOm610M4EXGzbEDR7oOIyS2ZiItgbs6eZc4oU/a0hpk8PrcKvv5g==" - }, "react-input-autosize": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/react-input-autosize/-/react-input-autosize-3.0.0.tgz", @@ -51612,49 +52791,304 @@ "dev": true }, "react-native": { - "version": "0.66.2", - "resolved": "https://registry.npmjs.org/react-native/-/react-native-0.66.2.tgz", - "integrity": "sha512-d5Kp23kqAH0EmcyxyfjAr4rhVVzyK/J0cZ/JuopX16CBD4Nkad65KcJi1pyOpbhpP9L1aUUs+mRyK5kg0uLqJQ==", + "version": "0.69.4", + "resolved": "https://registry.npmjs.org/react-native/-/react-native-0.69.4.tgz", + "integrity": "sha512-rqNMialM/T4pHRKWqTIpOxA65B/9kUjtnepxwJqvsdCeMP9Q2YdSx4VASFR9RoEFYcPRU41yGf6EKrChNfns3g==", "requires": { "@jest/create-cache-key-function": "^27.0.1", - "@react-native-community/cli": "^6.0.0", - "@react-native-community/cli-platform-android": "^6.0.0", - "@react-native-community/cli-platform-ios": "^6.0.0", + "@react-native-community/cli": "^8.0.4", + "@react-native-community/cli-platform-android": "^8.0.4", + "@react-native-community/cli-platform-ios": "^8.0.4", "@react-native/assets": "1.0.0", - "@react-native/normalize-color": "1.0.0", + "@react-native/normalize-color": "2.0.0", "@react-native/polyfills": "2.0.0", "abort-controller": "^3.0.0", "anser": "^1.4.9", "base64-js": "^1.1.2", "event-target-shim": "^5.0.1", - "hermes-engine": "~0.9.0", + "hermes-engine": "~0.11.0", "invariant": "^2.2.4", "jsc-android": "^250230.2.1", - "metro-babel-register": "0.66.2", - "metro-react-native-babel-transformer": "0.66.2", - "metro-runtime": "0.66.2", - "metro-source-map": "0.66.2", + "memoize-one": "^5.0.0", + "metro-react-native-babel-transformer": "0.70.3", + "metro-runtime": "0.70.3", + "metro-source-map": "0.70.3", + "mkdirp": "^0.5.1", "nullthrows": "^1.1.1", "pretty-format": "^26.5.2", "promise": "^8.0.3", - "prop-types": "^15.7.2", - "react-devtools-core": "^4.13.0", - "react-native-codegen": "^0.0.7", + "react-devtools-core": "4.24.0", + "react-native-codegen": "^0.69.1", + "react-native-gradle-plugin": "^0.0.7", "react-refresh": "^0.4.0", + "react-shallow-renderer": "16.15.0", "regenerator-runtime": "^0.13.2", - "scheduler": "^0.20.2", + "scheduler": "^0.21.0", "stacktrace-parser": "^0.1.3", - "use-subscription": "^1.0.0", + "use-sync-external-store": "^1.0.0", "whatwg-fetch": "^3.0.0", "ws": "^6.1.4" }, "dependencies": { - "promise": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/promise/-/promise-8.1.0.tgz", - "integrity": "sha512-W04AqnILOL/sPRXziNicCjSNRruLAuIHEOVBazepu0545DDNGYHz7ar9ZgZ1fMU8/MA4mVxp5rkBWRi6OXIy3Q==", + "@react-native-community/cli": { + "version": "8.0.6", + "resolved": "https://registry.npmjs.org/@react-native-community/cli/-/cli-8.0.6.tgz", + "integrity": "sha512-E36hU/if3quQCfJHGWVkpsCnwtByRCwORuAX0r6yr1ebKktpKeEO49zY9PAu/Z1gfyxCtgluXY0HfRxjKRFXTg==", + "requires": { + "@react-native-community/cli-clean": "^8.0.4", + "@react-native-community/cli-config": "^8.0.6", + "@react-native-community/cli-debugger-ui": "^8.0.0", + "@react-native-community/cli-doctor": "^8.0.6", + "@react-native-community/cli-hermes": "^8.0.5", + "@react-native-community/cli-plugin-metro": "^8.0.4", + "@react-native-community/cli-server-api": "^8.0.4", + "@react-native-community/cli-tools": "^8.0.4", + "@react-native-community/cli-types": "^8.0.0", + "chalk": "^4.1.2", + "commander": "^2.19.0", + "execa": "^1.0.0", + "find-up": "^4.1.0", + "fs-extra": "^8.1.0", + "graceful-fs": "^4.1.3", + "leven": "^3.1.0", + "lodash": "^4.17.15", + "minimist": "^1.2.0", + "prompts": "^2.4.0", + "semver": "^6.3.0" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" + }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", "requires": { - "asap": "~2.0.6" + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + } + } + }, + "execa": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", + "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "requires": { + "cross-spawn": "^6.0.0", + "get-stream": "^4.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + } + }, + "find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "requires": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + } + }, + "get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "requires": { + "pump": "^3.0.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" + }, + "hermes-parser": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.6.0.tgz", + "integrity": "sha512-Vf58jBZca2+QBLR9h7B7mdg8oFz2g5ILz1iVouZ5DOrOrAfBmPfJjdjDT8jrO0f+iJ4/hSRrQHqHIjSnTaLUDQ==", + "requires": { + "hermes-estree": "0.6.0" + } + }, + "locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "requires": { + "p-locate": "^4.1.0" + } + }, + "metro-react-native-babel-preset": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-preset/-/metro-react-native-babel-preset-0.70.3.tgz", + "integrity": "sha512-4Nxc1zEiHEu+GTdEMEsHnRgfaBkg8f/Td3+FcQ8NTSvs+xL3LBrQy6N07idWSQZHIdGFf+tTHvRfSIWLD8u8Tg==", + "requires": { + "@babel/core": "^7.14.0", + "@babel/plugin-proposal-async-generator-functions": "^7.0.0", + "@babel/plugin-proposal-class-properties": "^7.0.0", + "@babel/plugin-proposal-export-default-from": "^7.0.0", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.0.0", + "@babel/plugin-proposal-object-rest-spread": "^7.0.0", + "@babel/plugin-proposal-optional-catch-binding": "^7.0.0", + "@babel/plugin-proposal-optional-chaining": "^7.0.0", + "@babel/plugin-syntax-dynamic-import": "^7.0.0", + "@babel/plugin-syntax-export-default-from": "^7.0.0", + "@babel/plugin-syntax-flow": "^7.2.0", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.0.0", + "@babel/plugin-syntax-optional-chaining": "^7.0.0", + "@babel/plugin-transform-arrow-functions": "^7.0.0", + "@babel/plugin-transform-async-to-generator": "^7.0.0", + "@babel/plugin-transform-block-scoping": "^7.0.0", + "@babel/plugin-transform-classes": "^7.0.0", + "@babel/plugin-transform-computed-properties": "^7.0.0", + "@babel/plugin-transform-destructuring": "^7.0.0", + "@babel/plugin-transform-exponentiation-operator": "^7.0.0", + "@babel/plugin-transform-flow-strip-types": "^7.0.0", + "@babel/plugin-transform-function-name": "^7.0.0", + "@babel/plugin-transform-literals": "^7.0.0", + "@babel/plugin-transform-modules-commonjs": "^7.0.0", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.0.0", + "@babel/plugin-transform-parameters": "^7.0.0", + "@babel/plugin-transform-react-display-name": "^7.0.0", + "@babel/plugin-transform-react-jsx": "^7.0.0", + "@babel/plugin-transform-react-jsx-self": "^7.0.0", + "@babel/plugin-transform-react-jsx-source": "^7.0.0", + "@babel/plugin-transform-runtime": "^7.0.0", + "@babel/plugin-transform-shorthand-properties": "^7.0.0", + "@babel/plugin-transform-spread": "^7.0.0", + "@babel/plugin-transform-sticky-regex": "^7.0.0", + "@babel/plugin-transform-template-literals": "^7.0.0", + "@babel/plugin-transform-typescript": "^7.5.0", + "@babel/plugin-transform-unicode-regex": "^7.0.0", + "@babel/template": "^7.0.0", + "react-refresh": "^0.4.0" + } + }, + "metro-react-native-babel-transformer": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-react-native-babel-transformer/-/metro-react-native-babel-transformer-0.70.3.tgz", + "integrity": "sha512-WKBU6S/G50j9cfmFM4k4oRYprd8u3qjleD4so1E2zbTNILg+gYla7ZFGCAvi2G0ZcqS2XuGCR375c2hF6VVvwg==", + "requires": { + "@babel/core": "^7.14.0", + "babel-preset-fbjs": "^3.4.0", + "hermes-parser": "0.6.0", + "metro-babel-transformer": "0.70.3", + "metro-react-native-babel-preset": "0.70.3", + "metro-source-map": "0.70.3", + "nullthrows": "^1.1.1" + } + }, + "metro-source-map": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-source-map/-/metro-source-map-0.70.3.tgz", + "integrity": "sha512-zsYtZGrwRbbGEFHtmMqqeCH9K9aTGNVPsurMOWCUeQA3VGyVGXPGtLMC+CdAM9jLpUyg6jw2xh0esxi+tYH7Uw==", + "requires": { + "@babel/traverse": "^7.14.0", + "@babel/types": "^7.0.0", + "invariant": "^2.2.4", + "metro-symbolicate": "0.70.3", + "nullthrows": "^1.1.1", + "ob1": "0.70.3", + "source-map": "^0.5.6", + "vlq": "^1.0.0" + } + }, + "metro-symbolicate": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/metro-symbolicate/-/metro-symbolicate-0.70.3.tgz", + "integrity": "sha512-JTYkF1dpeDUssQ84juE1ycnhHki2ylJBBdJE1JHtfu5oC+z1ElDbBdPHq90Uvt8HbRov/ZAnxvv7Zy6asS+WCA==", + "requires": { + "invariant": "^2.2.4", + "metro-source-map": "0.70.3", + "nullthrows": "^1.1.1", + "source-map": "^0.5.6", + "through2": "^2.0.1", + "vlq": "^1.0.0" + } + }, + "ob1": { + "version": "0.70.3", + "resolved": "https://registry.npmjs.org/ob1/-/ob1-0.70.3.tgz", + "integrity": "sha512-Vy9GGhuXgDRY01QA6kdhToPd8AkLdLpX9GjH5kpqluVqTu70mgOm7tpGoJDZGaNbr9nJlJgnipqHJQRPORixIQ==" + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "requires": { + "p-limit": "^2.2.0" + } + }, + "p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==" + }, + "path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==" + }, + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" } }, "react-refresh": { @@ -51662,6 +53096,19 @@ "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.4.3.tgz", "integrity": "sha512-Hwln1VNuGl/6bVwnd0Xdn1e84gT/8T9aYNL+HAKDArLCS7LWjwr7StE30IEYbIkx0Vi3vs+coQxe+SQDbGbbpA==" }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==" + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "requires": { + "has-flag": "^4.0.0" + } + }, "ws": { "version": "6.2.2", "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.2.tgz", @@ -51681,12 +53128,13 @@ } }, "react-native-codegen": { - "version": "0.0.7", - "resolved": "https://registry.npmjs.org/react-native-codegen/-/react-native-codegen-0.0.7.tgz", - "integrity": "sha512-dwNgR8zJ3ALr480QnAmpTiqvFo+rDtq6V5oCggKhYFlRjzOmVSFn3YD41u8ltvKS5G2nQ8gCs2vReFFnRGLYng==", + "version": "0.69.2", + "resolved": "https://registry.npmjs.org/react-native-codegen/-/react-native-codegen-0.69.2.tgz", + "integrity": "sha512-yPcgMHD4mqLbckqnWjFBaxomDnBREfRjDi2G/WxNyPBQLD+PXUEmZTkDx6QoOXN+Bl2SkpnNOSsLE2+/RUHoPw==", "requires": { + "@babel/parser": "^7.14.0", "flow-parser": "^0.121.0", - "jscodeshift": "^0.11.0", + "jscodeshift": "^0.13.1", "nullthrows": "^1.1.1" } }, @@ -51714,9 +53162,14 @@ "fast-base64-decode": "^1.0.0" } }, + "react-native-gradle-plugin": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/react-native-gradle-plugin/-/react-native-gradle-plugin-0.0.7.tgz", + "integrity": "sha512-+4JpbIx42zGTONhBTIXSyfyHICHC29VTvhkkoUOJAh/XHPEixpuBduYgf6Y4y9wsN1ARlQhBBoptTvXvAFQf5g==" + }, "react-native-hr": { - "version": "git+https://github.com/Riglerr/react-native-hr.git#2d01a5cf77212d100e8b99e0310cce5234f977b3", - "from": "git+https://github.com/Riglerr/react-native-hr.git#2d01a5cf77212d100e8b99e0310cce5234f977b3" + "version": "https://raw.githubusercontent.com/wordpress-mobile/react-native-hr/1.1.3-wp-1/react-native-hr-1.1.3.tgz", + "integrity": "sha512-iwKdUpLc7lGN0+yiVfKxvvouMEXJeR4sRmfXFKXNnWOaBRdcf412zPa5vK4yEGPh5s38GDR29jYMM/RRw5U4nw==" }, "react-native-hsv-color-picker": { "version": "https://raw.githubusercontent.com/wordpress-mobile/react-native-hsv-color-picker/v1.0.1-wp-3/react-native-hsv-color-picker-1.0.1-wp-3.tgz", @@ -51757,26 +53210,258 @@ "integrity": "sha512-JyNcFTqyQh1YVOTMamH29/wzspXH1+AlDbxiuW07gkhdw0BspCdq2P0Vf6JzPdiDxhv/v1+ix4t6RjAKGWulGQ==" }, "react-native-reanimated": { - "version": "https://raw.githubusercontent.com/wordpress-mobile/react-native-reanimated/2.4.1-wp-4/react-native-reanimated-2.4.1-wp-4.tgz", - "integrity": "sha512-kPA7eM0Rn96ZzKId4QPaBnJ8UwC7NwT/CrruszF7fY+Xb1DGAY3vP7ZJu1ZsNbsxdK2ERNWZ9bnocXop4QV7cg==", + "version": "https://raw.githubusercontent.com/wordpress-mobile/react-native-reanimated/2.9.1-wp-2/react-native-reanimated-2.9.1-wp-2.tgz", + "integrity": "sha512-Lb3OyTv8mnC8Flzlz29CWQCLds7at6wnTy1R46lNT0Rv6CLArer9KWOicYO0iKBHBifKHNXQFz4WRZY40sFFGQ==", "requires": { - "@babel/plugin-transform-object-assign": "^7.10.4", + "@babel/plugin-proposal-export-namespace-from": "^7.17.12", + "@babel/plugin-transform-object-assign": "^7.16.7", + "@babel/preset-typescript": "^7.16.7", "@types/invariant": "^2.2.35", "invariant": "^2.2.4", "lodash.isequal": "^4.5.0", - "mockdate": "^3.0.2", - "react-native-screens": "^3.4.0", + "setimmediate": "^1.0.5", "string-hash-64": "^1.0.3" }, "dependencies": { - "react-native-screens": { - "version": "3.18.1", - "resolved": "https://registry.npmjs.org/react-native-screens/-/react-native-screens-3.18.1.tgz", - "integrity": "sha512-GtEC1AbvpvtKDJldavuONF/hXW1aEZO7qz8SioyHrV9L9/nDiy+iTviMFmeEoix9KwmEHHxoYxc0xiDrFxKHyA==", + "@babel/code-frame": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.18.6.tgz", + "integrity": "sha512-TDCmlK5eOvH+eH7cdAFlNXeVJqWIQ7gW9tY1GJIpUtFb6CmjVyq2VM3u71bOyR8CRihcCgMUYoDNyLXao3+70Q==", + "requires": { + "@babel/highlight": "^7.18.6" + } + }, + "@babel/generator": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.20.5.tgz", + "integrity": "sha512-jl7JY2Ykn9S0yj4DQP82sYvPU+T3g0HFcWTqDLqiuA9tGRNIj9VfbtXGAYTTkyNEnQk1jkMGOdYka8aG/lulCA==", + "requires": { + "@babel/types": "^7.20.5", + "@jridgewell/gen-mapping": "^0.3.2", + "jsesc": "^2.5.1" + } + }, + "@babel/helper-annotate-as-pure": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.18.6.tgz", + "integrity": "sha512-duORpUiYrEpzKIop6iNbjnwKLAKnJ47csTyRACyEmWj0QdUrm5aqNJGHSSEQSUAvNW0ojX0dOmK9dZduvkfeXA==", + "requires": { + "@babel/types": "^7.18.6" + } + }, + "@babel/helper-create-class-features-plugin": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.20.5.tgz", + "integrity": "sha512-3RCdA/EmEaikrhayahwToF0fpweU/8o2p8vhc1c/1kftHOdTKuC65kik/TLc+qfbS8JKw4qqJbne4ovICDhmww==", + "requires": { + "@babel/helper-annotate-as-pure": "^7.18.6", + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-function-name": "^7.19.0", + "@babel/helper-member-expression-to-functions": "^7.18.9", + "@babel/helper-optimise-call-expression": "^7.18.6", + "@babel/helper-replace-supers": "^7.19.1", + "@babel/helper-split-export-declaration": "^7.18.6" + } + }, + "@babel/helper-environment-visitor": { + "version": "7.18.9", + "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.18.9.tgz", + "integrity": "sha512-3r/aACDJ3fhQ/EVgFy0hpj8oHyHpQc+LPtJoY9SzTThAsStm4Ptegq92vqKoE3vD706ZVFWITnMnxucw+S9Ipg==" + }, + "@babel/helper-function-name": { + "version": "7.19.0", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.19.0.tgz", + "integrity": "sha512-WAwHBINyrpqywkUH0nTnNgI5ina5TFn85HKS0pbPDfxFfhyR/aNQEn4hGi1P1JyT//I0t4OgXUlofzWILRvS5w==", + "requires": { + "@babel/template": "^7.18.10", + "@babel/types": "^7.19.0" + } + }, + "@babel/helper-hoist-variables": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.18.6.tgz", + "integrity": "sha512-UlJQPkFqFULIcyW5sbzgbkxn2FKRgwWiRexcuaR8RNJRy8+LLveqPjwZV/bwrLZCN0eUHD/x8D0heK1ozuoo6Q==", + "requires": { + "@babel/types": "^7.18.6" + } + }, + "@babel/helper-member-expression-to-functions": { + "version": "7.18.9", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.18.9.tgz", + "integrity": "sha512-RxifAh2ZoVU67PyKIO4AMi1wTenGfMR/O/ae0CCRqwgBAt5v7xjdtRw7UoSbsreKrQn5t7r89eruK/9JjYHuDg==", + "requires": { + "@babel/types": "^7.18.9" + } + }, + "@babel/helper-optimise-call-expression": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.18.6.tgz", + "integrity": "sha512-HP59oD9/fEHQkdcbgFCnbmgH5vIQTJbxh2yf+CdM89/glUNnuzr87Q8GIjGEnOktTROemO0Pe0iPAYbqZuOUiA==", + "requires": { + "@babel/types": "^7.18.6" + } + }, + "@babel/helper-plugin-utils": { + "version": "7.20.2", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.20.2.tgz", + "integrity": "sha512-8RvlJG2mj4huQ4pZ+rU9lqKi9ZKiRmuvGuM2HlWmkmgOhbs6zEAw6IEiJ5cQqGbDzGZOhwuOQNtZMi/ENLjZoQ==" + }, + "@babel/helper-replace-supers": { + "version": "7.19.1", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.19.1.tgz", + "integrity": "sha512-T7ahH7wV0Hfs46SFh5Jz3s0B6+o8g3c+7TMxu7xKfmHikg7EAZ3I2Qk9LFhjxXq8sL7UkP5JflezNwoZa8WvWw==", + "requires": { + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-member-expression-to-functions": "^7.18.9", + "@babel/helper-optimise-call-expression": "^7.18.6", + "@babel/traverse": "^7.19.1", + "@babel/types": "^7.19.0" + } + }, + "@babel/helper-split-export-declaration": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.18.6.tgz", + "integrity": "sha512-bde1etTx6ZyTmobl9LLMMQsaizFVZrquTEHOqKeQESMKo4PlObf+8+JA25ZsIpZhT/WEd39+vOdLXAFG/nELpA==", + "requires": { + "@babel/types": "^7.18.6" + } + }, + "@babel/helper-validator-identifier": { + "version": "7.19.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz", + "integrity": "sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==" + }, + "@babel/helper-validator-option": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.18.6.tgz", + "integrity": "sha512-XO7gESt5ouv/LRJdrVjkShckw6STTaB7l9BrpBaAHDeF5YZT+01PCwmR0SJHnkW6i8OwW/EVWRShfi4j2x+KQw==" + }, + "@babel/highlight": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.18.6.tgz", + "integrity": "sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g==", + "requires": { + "@babel/helper-validator-identifier": "^7.18.6", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + } + }, + "@babel/parser": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.20.5.tgz", + "integrity": "sha512-r27t/cy/m9uKLXQNWWebeCUHgnAZq0CpG1OwKRxzJMP1vpSU4bSIK2hq+/cp0bQxetkXx38n09rNu8jVkcK/zA==" + }, + "@babel/plugin-proposal-export-namespace-from": { + "version": "7.18.9", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.18.9.tgz", + "integrity": "sha512-k1NtHyOMvlDDFeb9G5PhUXuGj8m/wiwojgQVEhJ/fsVsMCpLyOP4h0uGEjYJKrRI+EVPlb5Jk+Gt9P97lOGwtA==", + "requires": { + "@babel/helper-plugin-utils": "^7.18.9", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3" + } + }, + "@babel/plugin-syntax-typescript": { + "version": "7.20.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.20.0.tgz", + "integrity": "sha512-rd9TkG+u1CExzS4SM1BlMEhMXwFLKVjOAFFCDx9PbX5ycJWDoWMcwdJH9RhkPu1dOgn5TrxLot/Gx6lWFuAUNQ==", + "requires": { + "@babel/helper-plugin-utils": "^7.19.0" + } + }, + "@babel/plugin-transform-typescript": { + "version": "7.20.2", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.20.2.tgz", + "integrity": "sha512-jvS+ngBfrnTUBfOQq8NfGnSbF9BrqlR6hjJ2yVxMkmO5nL/cdifNbI30EfjRlN4g5wYWNnMPyj5Sa6R1pbLeag==", + "requires": { + "@babel/helper-create-class-features-plugin": "^7.20.2", + "@babel/helper-plugin-utils": "^7.20.2", + "@babel/plugin-syntax-typescript": "^7.20.0" + } + }, + "@babel/preset-typescript": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/preset-typescript/-/preset-typescript-7.18.6.tgz", + "integrity": "sha512-s9ik86kXBAnD760aybBucdpnLsAt0jK1xqJn2juOn9lkOvSHV60os5hxoVJsPzMQxvnUJFAlkont2DvvaYEBtQ==", + "requires": { + "@babel/helper-plugin-utils": "^7.18.6", + "@babel/helper-validator-option": "^7.18.6", + "@babel/plugin-transform-typescript": "^7.18.6" + } + }, + "@babel/template": { + "version": "7.18.10", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.18.10.tgz", + "integrity": "sha512-TI+rCtooWHr3QJ27kJxfjutghu44DLnasDMwpDqCXVTal9RLp3RSYNh4NdBrRP2cQAoG9A8juOQl6P6oZG4JxA==", + "requires": { + "@babel/code-frame": "^7.18.6", + "@babel/parser": "^7.18.10", + "@babel/types": "^7.18.10" + } + }, + "@babel/traverse": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.20.5.tgz", + "integrity": "sha512-WM5ZNN3JITQIq9tFZaw1ojLU3WgWdtkxnhM1AegMS+PvHjkM5IXjmYEGY7yukz5XS4sJyEf2VzWjI8uAavhxBQ==", + "requires": { + "@babel/code-frame": "^7.18.6", + "@babel/generator": "^7.20.5", + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-function-name": "^7.19.0", + "@babel/helper-hoist-variables": "^7.18.6", + "@babel/helper-split-export-declaration": "^7.18.6", + "@babel/parser": "^7.20.5", + "@babel/types": "^7.20.5", + "debug": "^4.1.0", + "globals": "^11.1.0" + } + }, + "@babel/types": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.20.5.tgz", + "integrity": "sha512-c9fst/h2/dcF7H+MJKZ2T0KjEQ8hY/BNnDk/H3XY8C4Aw/eWQXWn/lWntHF9ooUBnGmEvbfGrTgLWc+um0YDUg==", + "requires": { + "@babel/helper-string-parser": "^7.19.4", + "@babel/helper-validator-identifier": "^7.19.1", + "to-fast-properties": "^2.0.0" + } + }, + "@jridgewell/gen-mapping": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.2.tgz", + "integrity": "sha512-mh65xKQAzI6iBcFzwv28KVWSmCkdRBWoOh+bYQGW3+6OZvbbN3TqMGo5hqYxQniRcH9F2VZIoJCm4pa3BPDK/A==", + "requires": { + "@jridgewell/set-array": "^1.0.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.9" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", "requires": { - "react-freeze": "^1.0.0", - "warn-once": "^0.1.0" + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "requires": { + "ms": "2.1.2" } + }, + "globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==" + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" } } }, @@ -51892,9 +53577,10 @@ } }, "react-native-video": { - "version": "https://raw.githubusercontent.com/wordpress-mobile/react-native-video/5.2.0-wp-4/react-native-video-5.2.0-wp-4.tgz", - "integrity": "sha512-Y/kCq6hGDOMkT4VLuGtSgJaCF5QWUIipYvqTpgLaPXF33tT9evqTtKXt6OjSFyuaoxXv9rrxaNoN70Fai2A91Q==", + "version": "https://raw.githubusercontent.com/wordpress-mobile/react-native-video/5.2.0-wp-5/react-native-video-5.2.0-wp-5.tgz", + "integrity": "sha512-kUU86AmVQ73tQsTyi93Uqa4ti23bHcBIKzgU2wY6lyPTH2hqceUFOdRolRh7AnYhcBjJRGfNK8R8TfUBecsdKA==", "requires": { + "deprecated-react-native-prop-types": "^2.2.0", "prop-types": "^15.7.2" } }, @@ -52010,20 +53696,18 @@ } }, "react-shallow-renderer": { - "version": "16.14.1", - "resolved": "https://registry.npmjs.org/react-shallow-renderer/-/react-shallow-renderer-16.14.1.tgz", - "integrity": "sha512-rkIMcQi01/+kxiTE9D3fdS959U1g7gs+/rborw++42m1O9FAQiNI/UNRZExVUoAOprn4umcXf+pFRou8i4zuBg==", - "dev": true, + "version": "16.15.0", + "resolved": "https://registry.npmjs.org/react-shallow-renderer/-/react-shallow-renderer-16.15.0.tgz", + "integrity": "sha512-oScf2FqQ9LFVQgA73vr86xl2NaOIX73rh+YFqcOp68CWj56tSfgtGKrEbyhCj0rSijyG9M1CYprTh39fBi5hzA==", "requires": { "object-assign": "^4.1.1", - "react-is": "^16.12.0 || ^17.0.0" + "react-is": "^16.12.0 || ^17.0.0 || ^18.0.0" }, "dependencies": { "react-is": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", - "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", - "dev": true + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.2.0.tgz", + "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==" } } }, @@ -52053,22 +53737,30 @@ } }, "react-test-renderer": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-test-renderer/-/react-test-renderer-17.0.2.tgz", - "integrity": "sha512-yaQ9cB89c17PUb0x6UfWRs7kQCorVdHlutU1boVPEsB8IDZH6n9tHxMacc3y0JoXOJUsZb/t/Mb8FUWMKaM7iQ==", + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-test-renderer/-/react-test-renderer-18.2.0.tgz", + "integrity": "sha512-JWD+aQ0lh2gvh4NM3bBM42Kx+XybOxCpgYK7F8ugAlpaTSnWsX+39Z4XkOykGZAHrjwwTZT3x3KxswVWxHPUqA==", "dev": true, "requires": { - "object-assign": "^4.1.1", - "react-is": "^17.0.2", - "react-shallow-renderer": "^16.13.1", - "scheduler": "^0.20.2" + "react-is": "^18.2.0", + "react-shallow-renderer": "^16.15.0", + "scheduler": "^0.23.0" }, "dependencies": { "react-is": { - "version": "17.0.2", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", - "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.2.0.tgz", + "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==", "dev": true + }, + "scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "dev": true, + "requires": { + "loose-envify": "^1.1.0" + } } } }, @@ -52512,6 +54204,7 @@ "version": "0.14.4", "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.4.tgz", "integrity": "sha512-EaJaKPBI9GvKpvUz2mz4fhx7WPgvwRLY9v3hlNHWmAuJHI13T4nwKnNvm5RWJzEdnI5g5UwtOww+S8IdoUC2bw==", + "dev": true, "requires": { "@babel/runtime": "^7.8.4", "private": "^0.1.8" @@ -52689,7 +54382,7 @@ "relateurl": { "version": "0.2.7", "resolved": "https://registry.npmjs.org/relateurl/-/relateurl-0.2.7.tgz", - "integrity": "sha1-VNvzd+UUQKypCkzSdGANP/LYiKk=", + "integrity": "sha512-G08Dxvm4iDN3MLM0EsP62EDV9IuhXPR6blNz6Utcp7zyV3tr4HVNINt6MpaRWbxoOHT3Q7YN2P+jaHX8vUbgog==", "dev": true }, "remark": { @@ -53099,7 +54792,8 @@ "remove-trailing-separator": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", - "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=" + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", + "dev": true }, "renderkid": { "version": "3.0.0", @@ -53464,6 +55158,7 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, "requires": { "onetime": "^2.0.0", "signal-exit": "^3.0.2" @@ -53536,7 +55231,8 @@ "rsvp": { "version": "4.8.4", "resolved": "https://registry.npmjs.org/rsvp/-/rsvp-4.8.4.tgz", - "integrity": "sha512-6FomvYPfs+Jy9TfXmBpBuMWNH94SgCsZmJKcanySzgNNP6LjWxBvyLTa9KaMfDDM5oxRfrKDB0r/qeRsLwnBfA==" + "integrity": "sha512-6FomvYPfs+Jy9TfXmBpBuMWNH94SgCsZmJKcanySzgNNP6LjWxBvyLTa9KaMfDDM5oxRfrKDB0r/qeRsLwnBfA==", + "dev": true }, "rtlcss": { "version": "4.0.0", @@ -53638,6 +55334,56 @@ "ret": "~0.1.10" } }, + "safe-regex-test": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz", + "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.3", + "is-regex": "^1.1.4" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "get-intrinsic": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", + "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.3" + } + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, + "is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + } + } + }, "safer-buffer": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", @@ -53647,6 +55393,7 @@ "version": "4.1.0", "resolved": "https://registry.npmjs.org/sane/-/sane-4.1.0.tgz", "integrity": "sha512-hhbzAgTIX8O7SHfp2c8/kREfEn4qO/9q8C9beyY6+tvZ87EpoZ3i1RIEvp27YBswnNbY9mWd6paKVmKbAgLfZA==", + "dev": true, "requires": { "@cnakazawa/watch": "^1.0.3", "anymatch": "^2.0.0", @@ -53663,6 +55410,7 @@ "version": "6.0.5", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, "requires": { "nice-try": "^1.0.4", "path-key": "^2.0.1", @@ -53674,7 +55422,8 @@ "semver": { "version": "5.7.0", "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.0.tgz", - "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==" + "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==", + "dev": true } } }, @@ -53682,6 +55431,7 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "dev": true, "requires": { "cross-spawn": "^6.0.0", "get-stream": "^4.0.0", @@ -53696,6 +55446,7 @@ "version": "4.1.0", "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "dev": true, "requires": { "pump": "^3.0.0" } @@ -53704,6 +55455,7 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "dev": true, "requires": { "end-of-stream": "^1.1.0", "once": "^1.3.1" @@ -53847,11 +55599,6 @@ "neo-async": "^2.6.2" } }, - "sax": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", - "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" - }, "saxes": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", @@ -53862,12 +55609,11 @@ } }, "scheduler": { - "version": "0.20.2", - "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.20.2.tgz", - "integrity": "sha512-2eWfGgAqqWFGqtdMmcL5zCMK1U8KlXv8SQFGglL3CEtd0aDVDWgeF/YoCmvln55m5zSk3J/20hTaSBeSObsQDQ==", + "version": "0.21.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.21.0.tgz", + "integrity": "sha512-1r87x5fz9MXqswA2ERLo0EbOAU74DpIUO090gIasYTqlVoJeMcl+Z1Rg7WHz+qtPujhS/hGIt9kxZOYBV3faRQ==", "requires": { - "loose-envify": "^1.1.0", - "object-assign": "^4.1.1" + "loose-envify": "^1.1.0" } }, "schema-utils": { @@ -53948,23 +55694,23 @@ "dev": true }, "send": { - "version": "0.17.2", - "resolved": "https://registry.npmjs.org/send/-/send-0.17.2.tgz", - "integrity": "sha512-UJYB6wFSJE3G00nEivR5rgWp8c2xXvJ3OPWPhmuteU0IKj8nKbG3DrjiOmLwpnHGYWAVwA69zmTm++YG0Hmwww==", + "version": "0.18.0", + "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", + "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", "requires": { "debug": "2.6.9", - "depd": "~1.1.2", - "destroy": "~1.0.4", + "depd": "2.0.0", + "destroy": "1.2.0", "encodeurl": "~1.0.2", "escape-html": "~1.0.3", "etag": "~1.8.1", "fresh": "0.5.2", - "http-errors": "1.8.1", + "http-errors": "2.0.0", "mime": "1.6.0", "ms": "2.1.3", - "on-finished": "~2.3.0", + "on-finished": "2.4.1", "range-parser": "~1.2.1", - "statuses": "~1.5.0" + "statuses": "2.0.1" }, "dependencies": { "debug": { @@ -53978,10 +55724,20 @@ "ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" } } }, + "depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" + }, + "destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==" + }, "mime": { "version": "1.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", @@ -53991,6 +55747,19 @@ "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" + }, + "on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "requires": { + "ee-first": "1.1.1" + } + }, + "statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==" } } }, @@ -54090,14 +55859,14 @@ } }, "serve-static": { - "version": "1.14.2", - "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.2.tgz", - "integrity": "sha512-+TMNA9AFxUEGuC0z2mevogSnn9MXKb4fa7ngeRMJaaGv8vTwnIEkKi+QGvPt33HSnf8pRS+WGM0EbMtCJLKMBQ==", + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", + "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", "requires": { "encodeurl": "~1.0.2", "escape-html": "~1.0.3", "parseurl": "~1.3.3", - "send": "0.17.2" + "send": "0.18.0" } }, "set-blocking": { @@ -54137,8 +55906,7 @@ "setimmediate": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", - "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", - "dev": true + "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=" }, "setprototypeof": { "version": "1.2.0", @@ -54369,26 +56137,6 @@ "resolved": "https://registry.npmjs.org/simple-html-tokenizer/-/simple-html-tokenizer-0.5.7.tgz", "integrity": "sha512-APW9iYbkJ5cijjX4Ljhf3VG8SwYPUJT5gZrwci/wieMabQxWFiV5VwsrP5c6GMRvXKEQMGkAB1d9dvW66dTqpg==" }, - "simple-plist": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/simple-plist/-/simple-plist-1.3.1.tgz", - "integrity": "sha512-iMSw5i0XseMnrhtIzRb7XpQEXepa9xhWxGUojHBL43SIpQuDQkh3Wpy67ZbDzZVr6EKxvwVChnVpdl8hEVLDiw==", - "requires": { - "bplist-creator": "0.1.0", - "bplist-parser": "0.3.1", - "plist": "^3.0.5" - }, - "dependencies": { - "bplist-parser": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/bplist-parser/-/bplist-parser-0.3.1.tgz", - "integrity": "sha512-PyJxiNtA5T2PlLIeBot4lbp7rj4OadzjnMZD/G5zuBNt8ei/yCU7+wW0h2bag9vr8c+/WuRWmSxbqAl9hL1rBA==", - "requires": { - "big-integer": "1.6.x" - } - } - } - }, "simple-swizzle": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", @@ -54426,8 +56174,7 @@ "sisteransi": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", - "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", - "dev": true + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==" }, "slash": { "version": "3.0.0", @@ -55354,7 +57101,8 @@ "statuses": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", - "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true }, "store2": { "version": "2.12.0", @@ -55362,6 +57110,12 @@ "integrity": "sha512-7t+/wpKLanLzSnQPX8WAcuLCCeuSHoWdQuh9SB3xD0kNOM38DNf+0Oa+wmvxmYueRzkmh6IcdKFtvTa+ecgPDw==", "dev": true }, + "storybook-source-link": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/storybook-source-link/-/storybook-source-link-2.0.3.tgz", + "integrity": "sha512-Vw/ECmTObbcGbS6mX2bolQUF6c/Z4iBtcJBQh6T/3uFDz8pmzGo840hfSJDKXwwatHWDZ1V70jDLCDb4fbXQYg==", + "dev": true + }, "stream-browserify": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/stream-browserify/-/stream-browserify-2.0.2.tgz", @@ -55372,11 +57126,6 @@ "readable-stream": "^2.0.2" } }, - "stream-buffers": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/stream-buffers/-/stream-buffers-2.2.0.tgz", - "integrity": "sha1-kdX1Ew0c75bc+n9yaUUYh0HQnuQ=" - }, "stream-each": { "version": "1.2.2", "resolved": "https://registry.npmjs.org/stream-each/-/stream-each-1.2.2.tgz", @@ -56052,14 +57801,248 @@ "dev": true }, "object.assign": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", - "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3", + "has-symbols": "^1.0.1", + "object-keys": "^1.1.1" + } + }, + "regexp.prototype.flags": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz", + "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "functions-have-names": "^1.2.2" + } + }, + "string.prototype.trimend": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.5.tgz", + "integrity": "sha512-I7RGvmjV4pJ7O3kdf+LXFpVfdNOxtCW/2C8f6jNiW4+PQchwxkCDzlk1/7p+Wl4bqFIZeF47qAHXLuHHWKAxog==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.19.5" + } + }, + "string.prototype.trimstart": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.5.tgz", + "integrity": "sha512-THx16TJCGlsN0o6dl2o6ncWUsdgnLRSA23rRE5pyGBw/mLr3Ej/R2LaqCtgP8VNMGZsvMWnf9ooZPyY2bHvUFg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.19.5" + } + }, + "unbox-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", + "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + } + } + } + }, + "string.prototype.trimend": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.5.tgz", + "integrity": "sha512-I7RGvmjV4pJ7O3kdf+LXFpVfdNOxtCW/2C8f6jNiW4+PQchwxkCDzlk1/7p+Wl4bqFIZeF47qAHXLuHHWKAxog==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.19.5" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "define-properties": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.4.tgz", + "integrity": "sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA==", + "dev": true, + "requires": { + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + } + }, + "es-abstract": { + "version": "1.20.4", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.20.4.tgz", + "integrity": "sha512-0UtvRN79eMe2L+UNEF1BwRe364sj/DXhQ/k5FmivgoSdpM90b8Jc0mDzKMGo7QS0BVbOP/bTwBKNnDc9rNzaPA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "function.prototype.name": "^1.1.5", + "get-intrinsic": "^1.1.3", + "get-symbol-description": "^1.0.0", + "has": "^1.0.3", + "has-property-descriptors": "^1.0.0", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.3", + "is-callable": "^1.2.7", + "is-negative-zero": "^2.0.2", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "is-string": "^1.0.7", + "is-weakref": "^1.0.2", + "object-inspect": "^1.12.2", + "object-keys": "^1.1.1", + "object.assign": "^4.1.4", + "regexp.prototype.flags": "^1.4.3", + "safe-regex-test": "^1.0.0", + "string.prototype.trimend": "^1.0.5", + "string.prototype.trimstart": "^1.0.5", + "unbox-primitive": "^1.0.2" + }, + "dependencies": { + "get-intrinsic": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", + "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.3" + } + } + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "function.prototype.name": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", + "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.0", + "functions-have-names": "^1.2.2" + } + }, + "has-bigints": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", + "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==", + "dev": true + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, + "is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "dev": true + }, + "is-negative-zero": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", + "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "dev": true + }, + "is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "is-shared-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", + "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "object-inspect": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.2.tgz", + "integrity": "sha512-z+cPxW0QGUp0mcqcsgQyLVRDoXFQbXOwBaqyF7VIgI4TWNQsDHrBpUQslRmIfAoYWdYzs6UlKJtB2XJpTaNSpQ==", + "dev": true + }, + "object.assign": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", + "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", "dev": true, "requires": { - "call-bind": "^1.0.0", - "define-properties": "^1.1.3", - "has-symbols": "^1.0.1", + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "has-symbols": "^1.0.3", "object-keys": "^1.1.1" } }, @@ -56074,17 +58057,6 @@ "functions-have-names": "^1.2.2" } }, - "string.prototype.trimend": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.5.tgz", - "integrity": "sha512-I7RGvmjV4pJ7O3kdf+LXFpVfdNOxtCW/2C8f6jNiW4+PQchwxkCDzlk1/7p+Wl4bqFIZeF47qAHXLuHHWKAxog==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.19.5" - } - }, "string.prototype.trimstart": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.5.tgz", @@ -56250,12 +58222,19 @@ } }, "style-value-types": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/style-value-types/-/style-value-types-5.0.0.tgz", - "integrity": "sha512-08yq36Ikn4kx4YU6RD7jWEv27v4V+PUsOGa4n/as8Et3CuODMJQ00ENeAVXAeydX4Z2j1XHZF1K2sX4mGl18fA==", + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/style-value-types/-/style-value-types-5.1.2.tgz", + "integrity": "sha512-Vs9fNreYF9j6W2VvuDTP7kepALi7sk0xtk2Tu8Yxi9UoajJdEVpNpCov0HsLTqXvNGKX+Uv09pkozVITi1jf3Q==", "requires": { "hey-listen": "^1.0.8", - "tslib": "^2.1.0" + "tslib": "2.4.0" + }, + "dependencies": { + "tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + } } }, "stylehacks": { @@ -58144,17 +60123,33 @@ "integrity": "sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA==", "dev": true }, + "uglify-es": { + "version": "3.3.9", + "resolved": "https://registry.npmjs.org/uglify-es/-/uglify-es-3.3.9.tgz", + "integrity": "sha512-r+MU0rfv4L/0eeW3xZrd16t4NZfK8Ld4SWVglYBb7ez5uXFWHuVRs6xCTrf1yirs9a4j4Y27nn7SRfO6v67XsQ==", + "requires": { + "commander": "~2.13.0", + "source-map": "~0.6.1" + }, + "dependencies": { + "commander": { + "version": "2.13.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.13.0.tgz", + "integrity": "sha512-MVuS359B+YzaWqjCL/c+22gfryv+mCBPHAv3zyVI2GN8EY6IRP8VwtasXn8jyyhvvq84R4ImN1OKRtcbIasjYA==" + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" + } + } + }, "uglify-js": { "version": "3.13.7", "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.13.7.tgz", "integrity": "sha512-1Psi2MmnZJbnEsgJJIlfnd7tFlJfitusmR7zDI8lXlFI0ACD4/Rm/xdrU8bh6zF0i74aiVoBtkRiFulkrmh3AA==", "dev": true }, - "ultron": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.0.2.tgz", - "integrity": "sha1-rOEWq1V80Zc4ak6I9GhTeMiy5Po=" - }, "unbox-primitive": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.1.tgz", @@ -58970,11 +60965,6 @@ "makeerror": "1.0.x" } }, - "warn-once": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/warn-once/-/warn-once-0.1.1.tgz", - "integrity": "sha512-VkQZJbO8zVImzYFteBXvBOZEl1qL175WH8VmZcxF2fZAoudNhNDvHi+doCaAEdU2l2vtcIwa2zn0QK5+I1HQ3Q==" - }, "warning": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/warning/-/warning-4.0.3.tgz", @@ -60295,11 +62285,249 @@ } } }, + "which-collection": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.1.tgz", + "integrity": "sha512-W8xeTUwaln8i3K/cY1nGXzdnVZlidBcagyNFtBdD5kxnb4TvGKR7FfSIS3mYpwWS1QUCutfKz8IY8RjftB0+1A==", + "dev": true, + "requires": { + "is-map": "^2.0.1", + "is-set": "^2.0.1", + "is-weakmap": "^2.0.1", + "is-weakset": "^2.0.1" + } + }, "which-module": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=" }, + "which-typed-array": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.8.tgz", + "integrity": "sha512-Jn4e5PItbcAHyLoRDwvPj1ypu27DJbtdYXUa5zsinrUx77Uvfb0cXwwnGMTn7cjUfhhqgVQnVJCwF+7cgU7tpw==", + "dev": true, + "requires": { + "available-typed-arrays": "^1.0.5", + "call-bind": "^1.0.2", + "es-abstract": "^1.20.0", + "for-each": "^0.3.3", + "has-tostringtag": "^1.0.0", + "is-typed-array": "^1.1.9" + }, + "dependencies": { + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "define-properties": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.4.tgz", + "integrity": "sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA==", + "dev": true, + "requires": { + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + } + }, + "es-abstract": { + "version": "1.20.4", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.20.4.tgz", + "integrity": "sha512-0UtvRN79eMe2L+UNEF1BwRe364sj/DXhQ/k5FmivgoSdpM90b8Jc0mDzKMGo7QS0BVbOP/bTwBKNnDc9rNzaPA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "function.prototype.name": "^1.1.5", + "get-intrinsic": "^1.1.3", + "get-symbol-description": "^1.0.0", + "has": "^1.0.3", + "has-property-descriptors": "^1.0.0", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.3", + "is-callable": "^1.2.7", + "is-negative-zero": "^2.0.2", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "is-string": "^1.0.7", + "is-weakref": "^1.0.2", + "object-inspect": "^1.12.2", + "object-keys": "^1.1.1", + "object.assign": "^4.1.4", + "regexp.prototype.flags": "^1.4.3", + "safe-regex-test": "^1.0.0", + "string.prototype.trimend": "^1.0.5", + "string.prototype.trimstart": "^1.0.5", + "unbox-primitive": "^1.0.2" + }, + "dependencies": { + "get-intrinsic": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.3.tgz", + "integrity": "sha512-QJVz1Tj7MS099PevUG5jvnt9tSkXN8K14dxQlikJuPt4uD9hHAHjLyLBiLR5zELelBdD9QNRAXZzsJx0WaDL9A==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.3" + } + } + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "function.prototype.name": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", + "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.0", + "functions-have-names": "^1.2.2" + } + }, + "has-bigints": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", + "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==", + "dev": true + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, + "is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "dev": true + }, + "is-negative-zero": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", + "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "dev": true + }, + "is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "is-shared-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", + "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "object-inspect": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.2.tgz", + "integrity": "sha512-z+cPxW0QGUp0mcqcsgQyLVRDoXFQbXOwBaqyF7VIgI4TWNQsDHrBpUQslRmIfAoYWdYzs6UlKJtB2XJpTaNSpQ==", + "dev": true + }, + "object.assign": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", + "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + } + }, + "regexp.prototype.flags": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz", + "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "functions-have-names": "^1.2.2" + } + }, + "string.prototype.trimstart": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.5.tgz", + "integrity": "sha512-THx16TJCGlsN0o6dl2o6ncWUsdgnLRSA23rRE5pyGBw/mLr3Ej/R2LaqCtgP8VNMGZsvMWnf9ooZPyY2bHvUFg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.4", + "es-abstract": "^1.19.5" + } + }, + "unbox-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", + "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + } + } + } + }, "wicg-inert": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/wicg-inert/-/wicg-inert-3.1.2.tgz", @@ -60627,22 +62855,6 @@ "integrity": "sha1-R0tQhlrzpJqcRlfwWs0UVFj3fYI=", "dev": true }, - "xcode": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/xcode/-/xcode-2.1.0.tgz", - "integrity": "sha512-uCrmPITrqTEzhn0TtT57fJaNaw8YJs1aCzs+P/QqxsDbvPZSv7XMPPwXrKvHtD6pLjBM/NaVwraWJm8q83Y4iQ==", - "requires": { - "simple-plist": "^1.0.0", - "uuid": "^3.3.2" - }, - "dependencies": { - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==" - } - } - }, "xml": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/xml/-/xml-1.0.1.tgz", @@ -60655,20 +62867,17 @@ "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", "dev": true }, + "xmlbuilder": { + "version": "15.1.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-15.1.1.tgz", + "integrity": "sha512-yMqGBqtXyeN1e3TGYvgNgDVZ3j84W4cwkOXQswghol6APgZWaff9lnbvN7MHYJOiXsvGPXtjTYJEiC9J2wv9Eg==" + }, "xmlchars": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", "dev": true }, - "xmldoc": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/xmldoc/-/xmldoc-1.1.2.tgz", - "integrity": "sha512-ruPC/fyPNck2BD1dpz0AZZyrEwMOrWTO5lDdIXS91rs3wtm4j+T8Rp2o+zoOYkkAxJTZRPOSnOGei1egoRmKMQ==", - "requires": { - "sax": "^1.2.1" - } - }, "xtend": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", @@ -60818,8 +63027,7 @@ "yocto-queue": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", - "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", - "dev": true + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==" }, "zip-stream": { "version": "2.1.3", diff --git a/package.json b/package.json index 98f8d07607f67..b1ebbf430f35f 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "gutenberg", - "version": "14.6.0-rc.1", + "version": "14.8.0-rc.1", "private": true, "description": "A new WordPress editor experience.", "author": "The WordPress Contributors", @@ -107,16 +107,15 @@ "@storybook/addon-controls": "6.5.7", "@storybook/addon-docs": "6.5.7", "@storybook/addon-knobs": "6.2.9", - "@storybook/addon-storysource": "6.5.7", "@storybook/addon-toolbars": "6.5.7", "@storybook/addon-viewport": "6.5.7", "@storybook/builder-webpack5": "6.5.7", "@storybook/manager-webpack5": "6.5.7", "@storybook/react": "6.5.7", - "@testing-library/jest-dom": "5.16.4", - "@testing-library/react": "12.1.5", - "@testing-library/react-native": "9.1.0", - "@testing-library/user-event": "14.2.0", + "@testing-library/jest-dom": "5.16.5", + "@testing-library/react": "13.4.0", + "@testing-library/react-native": "11.3.0", + "@testing-library/user-event": "14.4.3", "@types/classnames": "2.3.1", "@types/eslint": "7.28.0", "@types/estree": "0.0.50", @@ -127,7 +126,9 @@ "@types/npm-package-arg": "6.1.1", "@types/prettier": "2.4.4", "@types/qs": "6.9.7", + "@types/react": "18.0.21", "@types/react-dates": "21.8.3", + "@types/react-dom": "18.0.6", "@types/requestidlecallback": "0.3.4", "@types/semver": "7.3.8", "@types/sprintf-js": "1.1.2", @@ -205,8 +206,8 @@ "lint-staged": "10.0.1", "lodash": "4.17.21", "make-dir": "3.0.0", - "metro-react-native-babel-preset": "0.66.2", - "metro-react-native-babel-transformer": "0.66.2", + "metro-react-native-babel-preset": "0.70.3", + "metro-react-native-babel-transformer": "0.70.3", "mkdirp": "0.5.1", "nock": "12.0.3", "node-fetch": "2.6.1", @@ -217,12 +218,12 @@ "postcss-loader": "6.2.1", "prettier": "npm:wp-prettier@2.6.2", "progress": "2.0.3", - "react": "17.0.2", - "react-dom": "17.0.2", - "react-native": "0.66.2", + "react": "18.2.0", + "react-dom": "18.2.0", + "react-native": "0.69.4", "react-native-url-polyfill": "1.1.2", "react-refresh": "0.10.0", - "react-test-renderer": "17.0.2", + "react-test-renderer": "18.2.0", "redux": "4.1.2", "resize-observer-polyfill": "1.5.1", "rimraf": "3.0.2", @@ -234,6 +235,7 @@ "snapshot-diff": "0.8.1", "source-map-loader": "3.0.0", "sprintf-js": "1.1.1", + "storybook-source-link": "2.0.3", "style-loader": "3.2.1", "terser-webpack-plugin": "5.1.4", "typescript": "4.4.2", diff --git a/packages/annotations/package.json b/packages/annotations/package.json index 5d7021f010411..8a60fcc93422c 100644 --- a/packages/annotations/package.json +++ b/packages/annotations/package.json @@ -33,9 +33,6 @@ "rememo": "^4.0.0", "uuid": "^8.3.0" }, - "peerDependencies": { - "react": "^17.0.0" - }, "publishConfig": { "access": "public" } diff --git a/packages/babel-plugin-makepot/index.js b/packages/babel-plugin-makepot/index.js index 3dc1b3ed26651..6bd5891a60438 100644 --- a/packages/babel-plugin-makepot/index.js +++ b/packages/babel-plugin-makepot/index.js @@ -33,7 +33,7 @@ */ const { po } = require( 'gettext-parser' ); -const { pick, isEqual, merge, isEmpty } = require( 'lodash' ); +const { merge, isEmpty } = require( 'lodash' ); const { relative, sep } = require( 'path' ); const { writeFileSync } = require( 'fs' ); @@ -182,10 +182,7 @@ function isValidTranslationKey( key ) { * @return {boolean} Whether valid translation keys match. */ function isSameTranslation( a, b ) { - return isEqual( - pick( a, VALID_TRANSLATION_KEYS ), - pick( b, VALID_TRANSLATION_KEYS ) - ); + return VALID_TRANSLATION_KEYS.every( ( key ) => a[ key ] === b[ key ] ); } /** diff --git a/packages/base-styles/CHANGELOG.md b/packages/base-styles/CHANGELOG.md index 85fc4cb684b63..a532f63be7006 100644 --- a/packages/base-styles/CHANGELOG.md +++ b/packages/base-styles/CHANGELOG.md @@ -2,6 +2,10 @@ ## Unreleased +### Enhancements + +- Lighten the border color in the `input-style__neutral` mixin ([#46252](https://github.com/WordPress/gutenberg/pull/46252)). + ## 4.13.0 (2022-11-16) ## 4.12.0 (2022-11-02) diff --git a/packages/base-styles/_colors.scss b/packages/base-styles/_colors.scss index 03f6bfbdd566d..2ce58b64e43b8 100644 --- a/packages/base-styles/_colors.scss +++ b/packages/base-styles/_colors.scss @@ -1,3 +1,5 @@ +@import "./functions"; + /** * Colors */ diff --git a/packages/base-styles/_default-custom-properties.scss b/packages/base-styles/_default-custom-properties.scss index 9f2979ea039bb..52dfeb3899d77 100644 --- a/packages/base-styles/_default-custom-properties.scss +++ b/packages/base-styles/_default-custom-properties.scss @@ -4,4 +4,6 @@ // It also provides default CSS variables for npm package consumers. :root { @include admin-scheme(#007cba); + --wp-block-synced-color: #7a00df; + --wp-block-synced-color--rgb: #{hex-to-rgb(#7a00df)}; } diff --git a/packages/base-styles/_mixins.scss b/packages/base-styles/_mixins.scss index 50af0938bccdb..9eee5820eba81 100644 --- a/packages/base-styles/_mixins.scss +++ b/packages/base-styles/_mixins.scss @@ -131,7 +131,7 @@ box-shadow: 0 0 0 transparent; transition: box-shadow 0.1s linear; border-radius: $radius-block-ui; - border: $border-width solid $gray-700; + border: $border-width solid $gray-600; @include reduce-motion("transition"); } @@ -550,3 +550,36 @@ } /* stylelint-enable function-comma-space-after */ } + +@mixin custom-scrollbars-on-hover() { + visibility: hidden; + + $handle-color: #757575; + $track-color: #1e1e1e; + + // WebKit + &::-webkit-scrollbar { + width: 12px; + height: 12px; + } + &::-webkit-scrollbar-track { + background-color: $track-color; + } + &::-webkit-scrollbar-thumb { + background-color: $handle-color; + border-radius: 8px; + border: 3px solid transparent; + background-clip: padding-box; + } + + // Firefox 109+ and Chrome 111+ + scrollbar-color: $handle-color $track-color; // Syntax, "dark", "light", or "#handle-color #track-color" + scrollbar-width: thin; + scrollbar-gutter: stable; + + &:hover, + &:focus, + & > * { + visibility: visible; + } +} diff --git a/packages/base-styles/_variables.scss b/packages/base-styles/_variables.scss index cc922ba5a035a..c4b0edd50e61b 100644 --- a/packages/base-styles/_variables.scss +++ b/packages/base-styles/_variables.scss @@ -57,15 +57,15 @@ $admin-sidebar-width-big: 190px; $admin-sidebar-width-collapsed: 36px; $modal-min-width: 360px; $spinner-size: 16px; +$canvas-padding: $grid-unit-30; /** * Shadows. */ -$shadow-popover: 0 2px 6px rgba($black, 0.05); -$shadow-modal: 0 10px 10px rgba($black, 0.25); - +$shadow-popover: 0 0.7px 1px rgba($black, 0.1), 0 1.2px 1.7px -0.2px rgba($black, 0.1), 0 2.3px 3.3px -0.5px rgba($black, 0.1); +$shadow-modal: 0 0.7px 1px rgba($black, 0.15), 0 2.7px 3.8px -0.2px rgba($black, 0.15), 0 5.5px 7.8px -0.3px rgba($black, 0.15), 0.1px 11.5px 16.4px -0.5px rgba($black, 0.15); /** * Editor widths. diff --git a/packages/base-styles/_z-index.scss b/packages/base-styles/_z-index.scss index 14305a9361197..ec3be48f1b975 100644 --- a/packages/base-styles/_z-index.scss +++ b/packages/base-styles/_z-index.scss @@ -7,15 +7,13 @@ $z-layers: ( ".block-editor-block-switcher__arrow": 1, ".block-editor-block-list__block {core/image aligned wide or fullwide}": 20, ".block-library-classic__toolbar": 31, // When scrolled to top this toolbar needs to sit over block-editor-block-toolbar - ".block-editor-block-list__layout .reusable-block-indicator": 1, ".block-editor-block-list__block-selection-button": 22, ".components-form-toggle__input": 1, ".edit-post-text-editor__toolbar": 1, ".edit-site-code-editor__toolbar": 1, - ".edit-post-sidebar__panel-tab.is-active": 1, // These next three share a stacking context - ".block-library-template-part__selection-search": 1, // higher sticky element + ".block-library-template-part__selection-search": 2, // higher sticky element // These next two share a stacking context ".interface-complementary-area .components-panel" : 0, // lower scrolling content @@ -109,14 +107,6 @@ $z-layers: ( // Show interface skeleton footer above interface skeleton drawer ".interface-interface-skeleton__footer": 90, - // Show the navigation toggle above the skeleton header - ".edit-site-navigation-toggle": 31, - // Show the navigation link above the skeleton header - ".edit-site-navigation-link": 31, - - // Show the FSE template previews above the editor and any open block toolbars - ".edit-site-navigation-panel__preview": 32, - // Above the block list and the header. ".block-editor-block-popover": 31, diff --git a/packages/block-directory/CHANGELOG.md b/packages/block-directory/CHANGELOG.md index d65f7d56149a6..f7f7566370060 100644 --- a/packages/block-directory/CHANGELOG.md +++ b/packages/block-directory/CHANGELOG.md @@ -2,6 +2,10 @@ ## Unreleased +### Breaking Changes + +- Updated dependencies to require React 18 ([45235](https://github.com/WordPress/gutenberg/pull/45235)) + ## 3.20.0 (2022-11-16) ## 3.19.0 (2022-11-02) diff --git a/packages/block-directory/package.json b/packages/block-directory/package.json index 1083e6281f7ab..0055a4846d7b6 100644 --- a/packages/block-directory/package.json +++ b/packages/block-directory/package.json @@ -47,8 +47,8 @@ "change-case": "^4.1.2" }, "peerDependencies": { - "react": "^17.0.0", - "react-dom": "^17.0.0" + "react": "^18.0.0", + "react-dom": "^18.0.0" }, "publishConfig": { "access": "public" diff --git a/packages/block-directory/src/plugins/installed-blocks-pre-publish-panel/index.js b/packages/block-directory/src/plugins/installed-blocks-pre-publish-panel/index.js index a51b68fcaffef..f0d46b7d1e1de 100644 --- a/packages/block-directory/src/plugins/installed-blocks-pre-publish-panel/index.js +++ b/packages/block-directory/src/plugins/installed-blocks-pre-publish-panel/index.js @@ -2,7 +2,6 @@ * WordPress dependencies */ import { _n, sprintf } from '@wordpress/i18n'; -import { PluginPrePublishPanel } from '@wordpress/edit-post'; import { useSelect } from '@wordpress/data'; import { blockDefault } from '@wordpress/icons'; @@ -12,6 +11,10 @@ import { blockDefault } from '@wordpress/icons'; import CompactList from '../../components/compact-list'; import { store as blockDirectoryStore } from '../../store'; +// We shouldn't import the edit-post package directly +// because it would include the wp-edit-post in all pages loading the block-directory script. +const { PluginPrePublishPanel } = window?.wp?.editPost ?? {}; + export default function InstalledBlocksPrePublishPanel() { const newBlockTypes = useSelect( ( select ) => select( blockDirectoryStore ).getNewBlockTypes(), diff --git a/packages/block-editor/CHANGELOG.md b/packages/block-editor/CHANGELOG.md index 4d37bf5737599..6abc73d7073b6 100644 --- a/packages/block-editor/CHANGELOG.md +++ b/packages/block-editor/CHANGELOG.md @@ -2,6 +2,14 @@ ## Unreleased +### Breaking Changes + +- Updated dependencies to require React 18 ([45235](https://github.com/WordPress/gutenberg/pull/45235)) + +### Enhancements + +- `URLInput`: the `renderSuggestions` callback prop now receives `currentInputValue` as a new parameter ([45806](https://github.com/WordPress/gutenberg/pull/45806)). + ## 10.5.0 (2022-11-16) ### Enhancement diff --git a/packages/block-editor/package.json b/packages/block-editor/package.json index e90f7aaf4619f..dce5352d5812a 100644 --- a/packages/block-editor/package.json +++ b/packages/block-editor/package.json @@ -44,6 +44,7 @@ "@wordpress/deprecated": "file:../deprecated", "@wordpress/dom": "file:../dom", "@wordpress/element": "file:../element", + "@wordpress/escape-html": "file:../escape-html", "@wordpress/hooks": "file:../hooks", "@wordpress/html-entities": "file:../html-entities", "@wordpress/i18n": "file:../i18n", @@ -64,6 +65,7 @@ "colord": "^2.7.0", "diff": "^4.0.2", "dom-scroll-into-view": "^1.2.1", + "fast-deep-equal": "^3.1.3", "inherits": "^2.0.3", "lodash": "^4.17.21", "react-autosize-textarea": "^7.1.0", @@ -73,8 +75,8 @@ "traverse": "^0.6.6" }, "peerDependencies": { - "react": "^17.0.0", - "react-dom": "^17.0.0" + "react": "^18.0.0", + "react-dom": "^18.0.0" }, "publishConfig": { "access": "public" diff --git a/packages/block-editor/src/autocompleters/block.js b/packages/block-editor/src/autocompleters/block.js index e5bcd2e78dbbf..9c1747fe867b6 100644 --- a/packages/block-editor/src/autocompleters/block.js +++ b/packages/block-editor/src/autocompleters/block.js @@ -1,8 +1,3 @@ -/** - * External dependencies - */ -import { orderBy } from 'lodash'; - /** * WordPress dependencies */ @@ -20,6 +15,7 @@ import { searchBlockItems } from '../components/inserter/search-items'; import useBlockTypesState from '../components/inserter/hooks/use-block-types-state'; import BlockIcon from '../components/block-icon'; import { store as blockEditorStore } from '../store'; +import { orderBy } from '../utils/sorting'; const noop = () => {}; const SHOWN_BLOCK_TYPES = 9; @@ -68,7 +64,7 @@ function createBlockCompleter() { collections, filterValue ) - : orderBy( items, [ 'frecency' ], [ 'desc' ] ); + : orderBy( items, 'frecency', 'desc' ); return initialFilteredItems .filter( ( item ) => item.name !== selectedBlockName ) diff --git a/packages/block-editor/src/autocompleters/link.js b/packages/block-editor/src/autocompleters/link.js index d439c5e6587aa..ce9af28f19d00 100644 --- a/packages/block-editor/src/autocompleters/link.js +++ b/packages/block-editor/src/autocompleters/link.js @@ -1,6 +1,8 @@ /** * WordPress dependencies */ +// Disable Reason: Needs to be refactored. +// eslint-disable-next-line no-restricted-imports import apiFetch from '@wordpress/api-fetch'; import { addQueryArgs } from '@wordpress/url'; import { Icon, page, post } from '@wordpress/icons'; diff --git a/packages/block-editor/src/components/alignment-control/test/index.js b/packages/block-editor/src/components/alignment-control/test/index.js index 14f0d9a04e033..c792b3e9b7f82 100644 --- a/packages/block-editor/src/components/alignment-control/test/index.js +++ b/packages/block-editor/src/components/alignment-control/test/index.js @@ -52,7 +52,7 @@ describe( 'AlignmentUI', () => { advanceTimers: jest.advanceTimersByTime, } ); - render( + const { unmount } = render( { name: /^Align text \w+$/, } ) ).toHaveLength( 3 ); + + // Cancel running effects, like delayed dropdown menu popover positioning. + unmount(); } ); test( 'should call on change with undefined when a control is already active', async () => { diff --git a/packages/block-editor/src/components/block-alignment-control/test/index.js b/packages/block-editor/src/components/block-alignment-control/test/index.js index fd9f0f782e25d..abf2a0a7c448a 100644 --- a/packages/block-editor/src/components/block-alignment-control/test/index.js +++ b/packages/block-editor/src/components/block-alignment-control/test/index.js @@ -47,7 +47,7 @@ describe( 'BlockAlignmentUI', () => { advanceTimers: jest.advanceTimersByTime, } ); - render( + const { unmount } = render( { name: /^Align \w+$/, } ) ).toHaveLength( 3 ); + + // Cancel running effects, like delayed dropdown menu popover positioning. + unmount(); } ); test( 'should call onChange with undefined, when the control is already active', async () => { diff --git a/packages/block-editor/src/components/block-card/index.js b/packages/block-editor/src/components/block-card/index.js index 21832629b9557..dab832861ae66 100644 --- a/packages/block-editor/src/components/block-card/index.js +++ b/packages/block-editor/src/components/block-card/index.js @@ -1,14 +1,24 @@ +/** + * External dependencies + */ +import classnames from 'classnames'; + /** * WordPress dependencies */ import deprecated from '@wordpress/deprecated'; +import { Button } from '@wordpress/components'; +import { chevronLeft, chevronRight } from '@wordpress/icons'; +import { __, isRTL } from '@wordpress/i18n'; +import { useSelect, useDispatch } from '@wordpress/data'; /** * Internal dependencies */ import BlockIcon from '../block-icon'; +import { store as blockEditorStore } from '../../store'; -function BlockCard( { title, icon, description, blockType } ) { +function BlockCard( { title, icon, description, blockType, className } ) { if ( blockType ) { deprecated( '`blockType` property in `BlockCard component`', { since: '5.7', @@ -16,8 +26,42 @@ function BlockCard( { title, icon, description, blockType } ) { } ); ( { title, icon, description } = blockType ); } + + const isOffCanvasNavigationEditorEnabled = + window?.__experimentalEnableOffCanvasNavigationEditor === true; + + const { parentNavBlockClientId } = useSelect( ( select ) => { + const { getSelectedBlockClientId, getBlockParentsByBlockName } = + select( blockEditorStore ); + + const _selectedBlockClientId = getSelectedBlockClientId(); + + return { + parentNavBlockClientId: getBlockParentsByBlockName( + _selectedBlockClientId, + 'core/navigation', + true + )[ 0 ], + }; + }, [] ); + + const { selectBlock } = useDispatch( blockEditorStore ); + return ( -
+
+ { isOffCanvasNavigationEditorEnabled && parentNavBlockClientId && ( + ); - const button = screen.getByRole( 'button' ); - fireEvent.mouseDown( button.parentElement ); + + fireEvent.mouseDown( screen.getByTestId( 'selection-clearer' ) ); expect( mockClearSelectedBlock ).toBeCalled(); } ); @@ -64,12 +64,12 @@ describe( 'BlockSelectionClearer component', () => { } ) ); render( - + ); - const button = screen.getByRole( 'button' ); - fireEvent.mouseDown( button.parentElement ); + + fireEvent.mouseDown( screen.getByTestId( 'selection-clearer' ) ); expect( mockClearSelectedBlock ).toBeCalled(); } ); @@ -82,12 +82,12 @@ describe( 'BlockSelectionClearer component', () => { } ) ); render( - + ); - const button = screen.getByRole( 'button' ); - fireEvent.mouseDown( button.parentElement ); + + fireEvent.mouseDown( screen.getByTestId( 'selection-clearer' ) ); expect( mockClearSelectedBlock ).not.toBeCalled(); } ); @@ -106,12 +106,12 @@ describe( 'BlockSelectionClearer component', () => { } ) ); render( - + ); - const button = screen.getByRole( 'button' ); - fireEvent.mouseDown( button.parentElement ); + + fireEvent.mouseDown( screen.getByTestId( 'selection-clearer' ) ); expect( mockClearSelectedBlock ).not.toBeCalled(); } ); diff --git a/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js b/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js index 87fc6bc5e0585..9ebc961440f7a 100644 --- a/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js +++ b/packages/block-editor/src/components/block-settings-menu/block-settings-dropdown.js @@ -189,6 +189,11 @@ export function BlockSettingsDropdown( { }, } ); + // This can occur when the selected block (the parent) + // displays child blocks within a List View. + const parentBlockIsSelected = + selectedBlockClientIds?.includes( firstParentClientId ); + return ( - { !! firstParentClientId && ( - - } - onClick={ () => - selectBlock( firstParentClientId ) - } - > - { sprintf( - /* translators: %s: Name of the block's parent. */ - __( 'Select parent block (%s)' ), - parentBlockType.title - ) } - - ) } + { ! parentBlockIsSelected && + !! firstParentClientId && ( + + } + onClick={ () => + selectBlock( + firstParentClientId + ) + } + > + { sprintf( + /* translators: %s: Name of the block's parent. */ + __( + 'Select parent block (%s)' + ), + parentBlockType.title + ) } + + ) } { count === 1 && ( jest.fn() ); jest.mock( '../../block-title/use-block-display-title', () => @@ -210,6 +211,7 @@ describe( 'BlockSwitcherDropdownMenu', () => { } ), '[ArrowDown]' ); + await act( () => Promise.resolve() ); expect( screen.getByRole( 'button', { @@ -252,6 +254,7 @@ describe( 'BlockSwitcherDropdownMenu', () => { expanded: false, } ) ); + await act( () => Promise.resolve() ); expect( screen.getByRole( 'button', { @@ -282,6 +285,7 @@ describe( 'BlockSwitcherDropdownMenu', () => { expanded: false, } ) ); + await act( () => Promise.resolve() ); expect( within( diff --git a/packages/block-editor/src/components/block-toolbar/index.js b/packages/block-editor/src/components/block-toolbar/index.js index b2160b3db4b6b..6727851e1647b 100644 --- a/packages/block-editor/src/components/block-toolbar/index.js +++ b/packages/block-editor/src/components/block-toolbar/index.js @@ -9,7 +9,12 @@ import classnames from 'classnames'; import { useSelect, useDispatch } from '@wordpress/data'; import { useRef } from '@wordpress/element'; import { useViewportMatch } from '@wordpress/compose'; -import { getBlockType, hasBlockSupport } from '@wordpress/blocks'; +import { + getBlockType, + hasBlockSupport, + isReusableBlock, + isTemplatePart, +} from '@wordpress/blocks'; import { ToolbarGroup } from '@wordpress/components'; /** @@ -109,11 +114,13 @@ const BlockToolbar = ( { hideDragHandle } ) => { const shouldShowVisualToolbar = isValid && isVisual; const isMultiToolbar = blockClientIds.length > 1; + const isSynced = + isReusableBlock( blockType ) || isTemplatePart( blockType ); - const classes = classnames( - 'block-editor-block-toolbar', - shouldShowMovers && 'is-showing-movers' - ); + const classes = classnames( 'block-editor-block-toolbar', { + 'is-showing-movers': shouldShowMovers, + 'is-synced': isSynced, + } ); return (
diff --git a/packages/block-editor/src/components/block-toolbar/style.scss b/packages/block-editor/src/components/block-toolbar/style.scss index 00d8d0da47f34..41faa1a46eaed 100644 --- a/packages/block-editor/src/components/block-toolbar/style.scss +++ b/packages/block-editor/src/components/block-toolbar/style.scss @@ -39,6 +39,16 @@ border-right: $border-width solid $gray-300; } + &.is-synced .block-editor-block-switcher .components-button .block-editor-block-icon { + color: var(--wp-block-synced-color); + } + + &.is-synced .components-toolbar-button.block-editor-block-switcher__no-switcher-icon { + &:disabled .block-editor-block-icon.has-colors { + color: var(--wp-block-synced-color); + } + } + > :last-child, > :last-child .components-toolbar-group, > :last-child .components-toolbar { diff --git a/packages/block-editor/src/components/block-tools/insertion-point.js b/packages/block-editor/src/components/block-tools/insertion-point.js index b7cd30462e359..2008cbc6b03ee 100644 --- a/packages/block-editor/src/components/block-tools/insertion-point.js +++ b/packages/block-editor/src/components/block-tools/insertion-point.js @@ -79,7 +79,6 @@ function InbetweenInsertionPointPopover( { isInserterShown: insertionPoint?.__unstableWithInserter, }; }, [] ); - const isVertical = orientation === 'vertical'; const disableMotion = useReducedMotion(); @@ -105,65 +104,22 @@ function InbetweenInsertionPointPopover( { } } - // Define animation variants for the line element. - const horizontalLine = { - start: { - width: 0, - top: '50%', - bottom: '50%', - x: 0, - }, - rest: { - width: 4, - top: 0, - bottom: 0, - x: -2, - }, - hover: { - width: 4, - top: 0, - bottom: 0, - x: -2, - }, - }; - const verticalLine = { - start: { - height: 0, - left: '50%', - right: '50%', - y: 0, - }, - rest: { - height: 4, - left: 0, - right: 0, - y: -2, - }, - hover: { - height: 4, - left: 0, - right: 0, - y: -2, - }, - }; const lineVariants = { // Initial position starts from the center and invisible. start: { - ...( ! isVertical ? horizontalLine.start : verticalLine.start ), opacity: 0, + scale: 0, }, // The line expands to fill the container. If the inserter is visible it // is delayed so it appears orchestrated. rest: { - ...( ! isVertical ? horizontalLine.rest : verticalLine.rest ), opacity: 1, - borderRadius: '2px', + scale: 1, transition: { delay: isInserterShown ? 0.5 : 0, type: 'tween' }, }, hover: { - ...( ! isVertical ? horizontalLine.hover : verticalLine.hover ), opacity: 1, - borderRadius: '2px', + scale: 1, transition: { delay: 0.5, type: 'tween' }, }, }; diff --git a/packages/block-editor/src/components/block-tools/style.scss b/packages/block-editor/src/components/block-tools/style.scss index e301859c16549..4bb2e07fcabb0 100644 --- a/packages/block-editor/src/components/block-tools/style.scss +++ b/packages/block-editor/src/components/block-tools/style.scss @@ -13,17 +13,22 @@ .block-editor-block-list__insertion-point-indicator { position: absolute; background: var(--wp-admin-theme-color); + border-radius: 2px; + transform-origin: center; + opacity: 0; + will-change: transform, opacity; .block-editor-block-list__insertion-point.is-vertical > & { - top: 50%; - height: $border-width; + top: calc(50% - 2px); + height: 4px; + width: 100%; } .block-editor-block-list__insertion-point.is-horizontal > & { top: 0; - right: 0; - left: 50%; - width: $border-width; + bottom: 0; + left: calc(50% - 2px); + width: 4px; } } @@ -32,6 +37,8 @@ // Don't show on mobile. display: none; position: absolute; + will-change: transform; + @include break-mobile() { display: flex; } diff --git a/packages/block-editor/src/components/block-variation-picker/index.js b/packages/block-editor/src/components/block-variation-picker/index.js index 24bdfa272c34b..27e214ec39bb0 100644 --- a/packages/block-editor/src/components/block-variation-picker/index.js +++ b/packages/block-editor/src/components/block-variation-picker/index.js @@ -49,10 +49,7 @@ function BlockVariationPicker( { className="block-editor-block-variation-picker__variation" label={ variation.description || variation.title } /> - + { variation.title } diff --git a/packages/block-editor/src/components/block-vertical-alignment-control/test/index.js b/packages/block-editor/src/components/block-vertical-alignment-control/test/index.js index 2e54ee9026854..f2645357a197c 100644 --- a/packages/block-editor/src/components/block-vertical-alignment-control/test/index.js +++ b/packages/block-editor/src/components/block-vertical-alignment-control/test/index.js @@ -45,7 +45,7 @@ describe( 'BlockVerticalAlignmentUI', () => { advanceTimers: jest.advanceTimersByTime, } ); - render( + const { unmount } = render( { name: /^Align \w+$/, } ) ).toHaveLength( 3 ); + + // Cancel running effects, like delayed dropdown menu popover positioning. + unmount(); } ); it( 'should call onChange with undefined, when the control is already active', async () => { diff --git a/packages/block-editor/src/components/default-block-appender/style.scss b/packages/block-editor/src/components/default-block-appender/content.scss similarity index 100% rename from packages/block-editor/src/components/default-block-appender/style.scss rename to packages/block-editor/src/components/default-block-appender/content.scss diff --git a/packages/block-editor/src/components/height-control/index.js b/packages/block-editor/src/components/height-control/index.js new file mode 100644 index 0000000000000..cc1eea0b4bad8 --- /dev/null +++ b/packages/block-editor/src/components/height-control/index.js @@ -0,0 +1,123 @@ +/** + * WordPress dependencies + */ +import { useMemo } from '@wordpress/element'; +import { + BaseControl, + RangeControl, + Flex, + FlexItem, + __experimentalSpacer as Spacer, + __experimentalUseCustomUnits as useCustomUnits, + __experimentalUnitControl as UnitControl, + __experimentalParseQuantityAndUnitFromRawValue as parseQuantityAndUnitFromRawValue, +} from '@wordpress/components'; +import { __ } from '@wordpress/i18n'; + +/** + * Internal dependencies + */ +import useSetting from '../use-setting'; + +const RANGE_CONTROL_CUSTOM_SETTINGS = { + px: { max: 1000, step: 1 }, + '%': { max: 100, step: 1 }, + vw: { max: 100, step: 1 }, + vh: { max: 100, step: 1 }, + em: { max: 50, step: 0.1 }, + rem: { max: 50, step: 0.1 }, +}; + +export default function HeightControl( { + onChange, + label = __( 'Height' ), + value, +} ) { + const customRangeValue = parseFloat( value ); + + const units = useCustomUnits( { + availableUnits: useSetting( 'spacing.units' ) || [ + '%', + 'px', + 'em', + 'rem', + 'vh', + 'vw', + ], + } ); + + const selectedUnit = + useMemo( + () => parseQuantityAndUnitFromRawValue( value ), + [ value ] + )[ 1 ] || + units[ 0 ]?.value || + 'px'; + + const handleSliderChange = ( next ) => { + onChange( [ next, selectedUnit ].join( '' ) ); + }; + + const handleUnitChange = ( newUnit ) => { + // Attempt to smooth over differences between currentUnit and newUnit. + // This should slightly improve the experience of switching between unit types. + const [ currentValue, currentUnit ] = + parseQuantityAndUnitFromRawValue( value ); + + if ( [ 'em', 'rem' ].includes( newUnit ) && currentUnit === 'px' ) { + // Convert pixel value to an approximate of the new unit, assuming a root size of 16px. + onChange( ( currentValue / 16 ).toFixed( 2 ) + newUnit ); + } else if ( + [ 'em', 'rem' ].includes( currentUnit ) && + newUnit === 'px' + ) { + // Convert to pixel value assuming a root size of 16px. + onChange( Math.round( currentValue * 16 ) + newUnit ); + } else if ( + [ 'vh', 'vw', '%' ].includes( newUnit ) && + currentValue > 100 + ) { + // When converting to `vh`, `vw`, or `%` units, cap the new value at 100. + onChange( 100 + newUnit ); + } + }; + + return ( +
+ + { label } + + + + + + + + + + + +
+ ); +} diff --git a/packages/block-editor/src/components/height-control/stories/index.js b/packages/block-editor/src/components/height-control/stories/index.js new file mode 100644 index 0000000000000..f4b586a96b0e3 --- /dev/null +++ b/packages/block-editor/src/components/height-control/stories/index.js @@ -0,0 +1,21 @@ +/** + * WordPress dependencies + */ +import { useState } from '@wordpress/element'; + +/** + * Internal dependencies + */ +import HeightControl from '../'; + +export default { + component: HeightControl, + title: 'BlockEditor/HeightControl', +}; + +const Template = ( props ) => { + const [ value, setValue ] = useState(); + return ; +}; + +export const Default = Template.bind( {} ); diff --git a/packages/block-editor/src/components/height-control/style.scss b/packages/block-editor/src/components/height-control/style.scss new file mode 100644 index 0000000000000..add0866835f76 --- /dev/null +++ b/packages/block-editor/src/components/height-control/style.scss @@ -0,0 +1,5 @@ +.block-editor-height-control { + border: 0; + margin: 0; + padding: 0; +} diff --git a/packages/block-editor/src/components/iframe/index.js b/packages/block-editor/src/components/iframe/index.js index 25e36037ebac5..b7413b54a2e44 100644 --- a/packages/block-editor/src/components/iframe/index.js +++ b/packages/block-editor/src/components/iframe/index.js @@ -188,7 +188,17 @@ async function loadScript( head, { id, src } ) { } function Iframe( - { contentRef, children, head, tabIndex = 0, assets, isZoomedOut, ...props }, + { + contentRef, + children, + head, + tabIndex = 0, + assets, + scale = 1, + frameSize = 0, + readonly, + ...props + }, ref ) { const [ , forceRender ] = useReducer( () => ( {} ) ); @@ -322,7 +332,7 @@ function Iframe( { head } , element ) } + + + ); + } +); + addFilter( 'blocks.registerBlockType', 'core/layout/addAttribute', @@ -426,6 +481,11 @@ addFilter( 'core/editor/layout/with-layout-styles', withLayoutStyles ); +addFilter( + 'editor.BlockListBlock', + 'core/editor/layout/with-child-layout-styles', + withChildLayoutStyles +); addFilter( 'editor.BlockEdit', 'core/editor/layout/with-inspector-controls', diff --git a/packages/block-editor/src/hooks/min-height.js b/packages/block-editor/src/hooks/min-height.js index 3167edba8a829..e123f0cee98b2 100644 --- a/packages/block-editor/src/hooks/min-height.js +++ b/packages/block-editor/src/hooks/min-height.js @@ -2,16 +2,13 @@ * WordPress dependencies */ import { getBlockSupport } from '@wordpress/blocks'; -import { - __experimentalUseCustomUnits as useCustomUnits, - __experimentalUnitControl as UnitControl, -} from '@wordpress/components'; import { __ } from '@wordpress/i18n'; /** * Internal dependencies */ import useSetting from '../components/use-setting'; +import HeightControl from '../components/height-control'; import { DIMENSIONS_SUPPORT_KEY } from './dimensions'; import { cleanEmptyObject } from './utils'; @@ -81,17 +78,6 @@ export function MinHeightEdit( props ) { setAttributes, } = props; - const units = useCustomUnits( { - availableUnits: useSetting( 'dimensions.units' ) || [ - '%', - 'px', - 'em', - 'rem', - 'vh', - 'vw', - ], - } ); - if ( useIsMinHeightDisabled( props ) ) { return null; } @@ -109,13 +95,10 @@ export function MinHeightEdit( props ) { }; return ( - ); } diff --git a/packages/block-editor/src/store/reducer.js b/packages/block-editor/src/store/reducer.js index 924b563f5710e..9c99126f54ba1 100644 --- a/packages/block-editor/src/store/reducer.js +++ b/packages/block-editor/src/store/reducer.js @@ -1,7 +1,8 @@ /** * External dependencies */ -import { omit, mapValues, isEqual, isEmpty } from 'lodash'; +import fastDeepEqual from 'fast-deep-equal/es6'; +import { omit, isEmpty } from 'lodash'; /** * WordPress dependencies @@ -28,16 +29,18 @@ const identity = ( x ) => x; * @return {Object} Block order map object. */ function mapBlockOrder( blocks, rootClientId = '' ) { - const result = { [ rootClientId ]: [] }; - + const result = new Map(); + const current = []; + result.set( rootClientId, current ); blocks.forEach( ( block ) => { const { clientId, innerBlocks } = block; - - result[ rootClientId ].push( clientId ); - - Object.assign( result, mapBlockOrder( innerBlocks, clientId ) ); + current.push( clientId ); + mapBlockOrder( innerBlocks, clientId ).forEach( + ( order, subClientId ) => { + result.set( subClientId, order ); + } + ); } ); - return result; } @@ -51,15 +54,18 @@ function mapBlockOrder( blocks, rootClientId = '' ) { * @return {Object} Block order map object. */ function mapBlockParents( blocks, rootClientId = '' ) { - return blocks.reduce( - ( result, block ) => - Object.assign( - result, - { [ block.clientId ]: rootClientId }, - mapBlockParents( block.innerBlocks, block.clientId ) - ), - {} - ); + const result = []; + const stack = [ [ rootClientId, blocks ] ]; + while ( stack.length ) { + const [ parent, currentBlocks ] = stack.shift(); + currentBlocks.forEach( ( { innerBlocks, ...block } ) => { + result.push( [ block.clientId, parent ] ); + if ( innerBlocks?.length ) { + stack.push( [ block.clientId, innerBlocks ] ); + } + } ); + } + return result; } /** @@ -70,16 +76,28 @@ function mapBlockParents( blocks, rootClientId = '' ) { * @param {Array} blocks Blocks to flatten. * @param {Function} transform Transforming function to be applied to each block. * - * @return {Object} Flattened object. + * @return {Array} Flattened object. */ function flattenBlocks( blocks, transform = identity ) { - const result = {}; + const result = []; const stack = [ ...blocks ]; while ( stack.length ) { const { innerBlocks, ...block } = stack.shift(); stack.push( ...innerBlocks ); - result[ block.clientId ] = transform( block ); + result.push( [ block.clientId, transform( block ) ] ); + } + + return result; +} + +function getFlattenedClientIds( blocks ) { + const result = {}; + const stack = [ ...blocks ]; + while ( stack.length ) { + const { innerBlocks, ...block } = stack.shift(); + stack.push( ...innerBlocks ); + result[ block.clientId ] = true; } return result; @@ -92,7 +110,7 @@ function flattenBlocks( blocks, transform = identity ) { * * @param {Array} blocks Blocks to flatten. * - * @return {Object} Flattened block attributes object. + * @return {Array} Flattened block attributes object. */ function getFlattenedBlocksWithoutAttributes( blocks ) { return flattenBlocks( blocks, ( block ) => omit( block, 'attributes' ) ); @@ -105,29 +123,12 @@ function getFlattenedBlocksWithoutAttributes( blocks ) { * * @param {Array} blocks Blocks to flatten. * - * @return {Object} Flattened block attributes object. + * @return {Array} Flattened block attributes object. */ function getFlattenedBlockAttributes( blocks ) { return flattenBlocks( blocks, ( block ) => block.attributes ); } -/** - * Returns an object against which it is safe to perform mutating operations, - * given the original object and its current working copy. - * - * @param {Object} original Original object. - * @param {Object} working Working object. - * - * @return {Object} Mutation-safe object. - */ -function getMutateSafeObject( original, working ) { - if ( original === working ) { - return { ...original }; - } - - return working; -} - /** * Returns true if the two object arguments have the same keys, or false * otherwise. @@ -138,7 +139,7 @@ function getMutateSafeObject( original, working ) { * @return {boolean} Whether the two objects have the same keys. */ export function hasSameKeys( a, b ) { - return isEqual( Object.keys( a ), Object.keys( b ) ); + return fastDeepEqual( Object.keys( a ), Object.keys( b ) ); } /** @@ -156,13 +157,13 @@ export function isUpdatingSameBlockAttribute( action, lastAction ) { action.type === 'UPDATE_BLOCK_ATTRIBUTES' && lastAction !== undefined && lastAction.type === 'UPDATE_BLOCK_ATTRIBUTES' && - isEqual( action.clientIds, lastAction.clientIds ) && + fastDeepEqual( action.clientIds, lastAction.clientIds ) && hasSameKeys( action.attributes, lastAction.attributes ) ); } -function buildBlockTree( state, blocks ) { - const result = {}; +function updateBlockTreeForBlocks( state, blocks ) { + const treeToUpdate = state.tree; const stack = [ ...blocks ]; const flattenedBlocks = [ ...blocks ]; while ( stack.length ) { @@ -172,33 +173,34 @@ function buildBlockTree( state, blocks ) { } // Create objects before mutating them, that way it's always defined. for ( const block of flattenedBlocks ) { - result[ block.clientId ] = {}; + treeToUpdate.set( block.clientId, {} ); } for ( const block of flattenedBlocks ) { - result[ block.clientId ] = Object.assign( result[ block.clientId ], { - ...state.byClientId[ block.clientId ], - attributes: state.attributes[ block.clientId ], - innerBlocks: block.innerBlocks.map( - ( subBlock ) => result[ subBlock.clientId ] - ), - } ); + treeToUpdate.set( + block.clientId, + Object.assign( treeToUpdate.get( block.clientId ), { + ...state.byClientId.get( block.clientId ), + attributes: state.attributes.get( block.clientId ), + innerBlocks: block.innerBlocks.map( ( subBlock ) => + treeToUpdate.get( subBlock.clientId ) + ), + } ) + ); } - - return result; } function updateParentInnerBlocksInTree( state, - tree, updatedClientIds, updateChildrenOfUpdatedClientIds = false ) { + const treeToUpdate = state.tree; const uncontrolledParents = new Set( [] ); const controlledParents = new Set(); for ( const clientId of updatedClientIds ) { let current = updateChildrenOfUpdatedClientIds ? clientId - : state.parents[ clientId ]; + : state.parents.get( clientId ); do { if ( state.controlledInnerBlocks[ current ] ) { // Should stop on controlled blocks. @@ -208,7 +210,7 @@ function updateParentInnerBlocksInTree( } else { // Else continue traversing up through parents. uncontrolledParents.add( current ); - current = state.parents[ current ]; + current = state.parents.get( current ); } } while ( current !== undefined ); } @@ -216,27 +218,23 @@ function updateParentInnerBlocksInTree( // To make sure the order of assignments doesn't matter, // we first create empty objects and mutates the inner blocks later. for ( const clientId of uncontrolledParents ) { - tree[ clientId ] = { - ...tree[ clientId ], - }; + treeToUpdate.set( clientId, { ...treeToUpdate.get( clientId ) } ); } for ( const clientId of uncontrolledParents ) { - tree[ clientId ].innerBlocks = ( state.order[ clientId ] || [] ).map( - ( subClientId ) => tree[ subClientId ] - ); + treeToUpdate.get( clientId ).innerBlocks = ( + state.order.get( clientId ) || [] + ).map( ( subClientId ) => treeToUpdate.get( subClientId ) ); } // Controlled parent blocks, need a dedicated key for their inner blocks // to be used when doing getBlocks( controlledBlockClientId ). for ( const clientId of controlledParents ) { - tree[ 'controlled||' + clientId ] = { - innerBlocks: ( state.order[ clientId ] || [] ).map( - ( subClientId ) => tree[ subClientId ] + treeToUpdate.set( 'controlled||' + clientId, { + innerBlocks: ( state.order.get( clientId ) || [] ).map( + ( subClientId ) => treeToUpdate.get( subClientId ) ), - }; + } ); } - - return tree; } /** @@ -257,82 +255,70 @@ const withBlockTree = return state; } - newState.tree = state.tree ? state.tree : {}; + newState.tree = state.tree ? state.tree : new Map(); switch ( action.type ) { case 'RECEIVE_BLOCKS': case 'INSERT_BLOCKS': { - const subTree = buildBlockTree( newState, action.blocks ); - newState.tree = updateParentInnerBlocksInTree( + newState.tree = new Map( newState.tree ); + updateBlockTreeForBlocks( newState, action.blocks ); + updateParentInnerBlocksInTree( newState, - { - ...newState.tree, - ...subTree, - }, action.rootClientId ? [ action.rootClientId ] : [ '' ], true ); break; } case 'UPDATE_BLOCK': - newState.tree = updateParentInnerBlocksInTree( + newState.tree = new Map( newState.tree ); + newState.tree.set( action.clientId, { + ...newState.tree.get( action.clientId ), + ...newState.byClientId.get( action.clientId ), + attributes: newState.attributes.get( action.clientId ), + } ); + updateParentInnerBlocksInTree( newState, - { - ...newState.tree, - [ action.clientId ]: { - ...newState.tree[ action.clientId ], - ...newState.byClientId[ action.clientId ], - attributes: newState.attributes[ action.clientId ], - }, - }, [ action.clientId ], false ); break; case 'UPDATE_BLOCK_ATTRIBUTES': { - const newSubTree = action.clientIds.reduce( - ( result, clientId ) => { - result[ clientId ] = { - ...newState.tree[ clientId ], - attributes: newState.attributes[ clientId ], - }; - return result; - }, - {} - ); - newState.tree = updateParentInnerBlocksInTree( + newState.tree = new Map( newState.tree ); + action.clientIds.forEach( ( clientId ) => { + newState.tree.set( clientId, { + ...newState.tree.get( clientId ), + attributes: newState.attributes.get( clientId ), + } ); + } ); + updateParentInnerBlocksInTree( newState, - { - ...newState.tree, - ...newSubTree, - }, action.clientIds, false ); break; } case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': { - const subTree = buildBlockTree( newState, action.blocks ); - newState.tree = updateParentInnerBlocksInTree( - newState, - { - ...omit( - newState.tree, - action.replacedClientIds.concat( - // Controlled inner blocks are only removed - // if the block doesn't move to another position - // otherwise their content will be lost. - action.replacedClientIds - .filter( - ( clientId ) => ! subTree[ clientId ] - ) - .map( - ( clientId ) => - 'controlled||' + clientId - ) + const inserterClientIds = getFlattenedClientIds( + action.blocks + ); + newState.tree = new Map( newState.tree ); + action.replacedClientIds + .concat( + // Controlled inner blocks are only removed + // if the block doesn't move to another position + // otherwise their content will be lost. + action.replacedClientIds + .filter( + ( clientId ) => ! inserterClientIds[ clientId ] ) - ), - ...subTree, - }, + .map( ( clientId ) => 'controlled||' + clientId ) + ) + .forEach( ( key ) => { + newState.tree.delete( key ); + } ); + + updateBlockTreeForBlocks( newState, action.blocks ); + updateParentInnerBlocksInTree( + newState, action.blocks.map( ( b ) => b.clientId ), false ); @@ -341,18 +327,19 @@ const withBlockTree = const parentsOfRemovedBlocks = []; for ( const clientId of action.clientIds ) { if ( - state.parents[ clientId ] !== undefined && - ( state.parents[ clientId ] === '' || - newState.byClientId[ state.parents[ clientId ] ] ) + state.parents.get( clientId ) !== undefined && + ( state.parents.get( clientId ) === '' || + newState.byClientId.get( + state.parents.get( clientId ) + ) ) ) { parentsOfRemovedBlocks.push( - state.parents[ clientId ] + state.parents.get( clientId ) ); } } - newState.tree = updateParentInnerBlocksInTree( + updateParentInnerBlocksInTree( newState, - newState.tree, parentsOfRemovedBlocks, true ); @@ -362,25 +349,29 @@ const withBlockTree = const parentsOfRemovedBlocks = []; for ( const clientId of action.clientIds ) { if ( - state.parents[ clientId ] !== undefined && - ( state.parents[ clientId ] === '' || - newState.byClientId[ state.parents[ clientId ] ] ) + state.parents.get( clientId ) !== undefined && + ( state.parents.get( clientId ) === '' || + newState.byClientId.get( + state.parents.get( clientId ) + ) ) ) { parentsOfRemovedBlocks.push( - state.parents[ clientId ] + state.parents.get( clientId ) ); } } - newState.tree = updateParentInnerBlocksInTree( - newState, - omit( - newState.tree, - action.removedClientIds.concat( - action.removedClientIds.map( - ( clientId ) => 'controlled||' + clientId - ) + newState.tree = new Map( newState.tree ); + action.removedClientIds + .concat( + action.removedClientIds.map( + ( clientId ) => 'controlled||' + clientId ) - ), + ) + .forEach( ( key ) => { + newState.tree.delete( key ); + } ); + updateParentInnerBlocksInTree( + newState, parentsOfRemovedBlocks, true ); @@ -395,9 +386,9 @@ const withBlockTree = if ( action.toRootClientId ) { updatedBlockUids.push( action.toRootClientId ); } - newState.tree = updateParentInnerBlocksInTree( + newState.tree = new Map( newState.tree ); + updateParentInnerBlocksInTree( newState, - newState.tree, updatedBlockUids, true ); @@ -408,39 +399,35 @@ const withBlockTree = const updatedBlockUids = [ action.rootClientId ? action.rootClientId : '', ]; - newState.tree = updateParentInnerBlocksInTree( + newState.tree = new Map( newState.tree ); + updateParentInnerBlocksInTree( newState, - newState.tree, updatedBlockUids, true ); break; } case 'SAVE_REUSABLE_BLOCK_SUCCESS': { - const updatedBlockUids = Object.entries( newState.attributes ) - .filter( ( [ clientId, attributes ] ) => { - return ( - newState.byClientId[ clientId ].name === - 'core/block' && - attributes.ref === action.updatedId - ); - } ) - .map( ( [ clientId ] ) => clientId ); - - newState.tree = updateParentInnerBlocksInTree( + const updatedBlockUids = []; + newState.attributes.forEach( ( attributes, clientId ) => { + if ( + newState.byClientId.get( clientId ).name === + 'core/block' && + attributes.ref === action.updatedId + ) { + updatedBlockUids.push( clientId ); + } + } ); + newState.tree = new Map( newState.tree ); + updatedBlockUids.forEach( ( clientId ) => { + newState.tree.set( clientId, { + ...newState.byClientId.get( clientId ), + attributes: newState.attributes.get( clientId ), + innerBlocks: newState.tree.get( clientId ).innerBlocks, + } ); + } ); + updateParentInnerBlocksInTree( newState, - { - ...newState.tree, - ...updatedBlockUids.reduce( ( result, clientId ) => { - result[ clientId ] = { - ...newState.byClientId[ clientId ], - attributes: newState.attributes[ clientId ], - innerBlocks: - newState.tree[ clientId ].innerBlocks, - }; - return result; - }, {} ), - }, updatedBlockUids, false ); @@ -549,7 +536,7 @@ const withInnerBlocksRemoveCascade = ( reducer ) => ( state, action ) => { let result = clientIds; for ( let i = 0; i < result.length; i++ ) { if ( - ! state.order[ result[ i ] ] || + ! state.order.get( result[ i ] ) || ( action.keepControlledInnerBlocks && action.keepControlledInnerBlocks[ result[ i ] ] ) ) { @@ -560,7 +547,7 @@ const withInnerBlocksRemoveCascade = ( reducer ) => ( state, action ) => { result = [ ...result ]; } - result.push( ...state.order[ result[ i ] ] ); + result.push( ...state.order.get( result[ i ] ) ); } return result; }; @@ -601,23 +588,22 @@ const withBlockReset = ( reducer ) => ( state, action ) => { if ( action.type === 'RESET_BLOCKS' ) { const newState = { ...state, - byClientId: getFlattenedBlocksWithoutAttributes( action.blocks ), - attributes: getFlattenedBlockAttributes( action.blocks ), + byClientId: new Map( + getFlattenedBlocksWithoutAttributes( action.blocks ) + ), + attributes: new Map( getFlattenedBlockAttributes( action.blocks ) ), order: mapBlockOrder( action.blocks ), - parents: mapBlockParents( action.blocks ), + parents: new Map( mapBlockParents( action.blocks ) ), controlledInnerBlocks: {}, }; - const subTree = buildBlockTree( newState, action.blocks ); - newState.tree = { - ...subTree, - // Root. - '': { - innerBlocks: action.blocks.map( - ( subBlock ) => subTree[ subBlock.clientId ] - ), - }, - }; + newState.tree = new Map( state?.tree ); + updateBlockTreeForBlocks( newState, action.blocks ); + newState.tree.set( '', { + innerBlocks: action.blocks.map( ( subBlock ) => + newState.tree.get( subBlock.clientId ) + ), + } ); return newState; } @@ -663,11 +649,11 @@ const withReplaceInnerBlocks = ( reducer ) => ( state, action ) => { // marked block in the block state so that they can be reattached to the // marked block when we re-insert everything a few lines below. let stateAfterBlocksRemoval = state; - if ( state.order[ action.rootClientId ] ) { + if ( state.order.get( action.rootClientId ) ) { stateAfterBlocksRemoval = reducer( stateAfterBlocksRemoval, { type: 'REMOVE_BLOCKS', keepControlledInnerBlocks: nestedControllers, - clientIds: state.order[ action.rootClientId ], + clientIds: state.order.get( action.rootClientId ), } ); } let stateAfterInsert = stateAfterBlocksRemoval; @@ -681,25 +667,20 @@ const withReplaceInnerBlocks = ( reducer ) => ( state, action ) => { // We need to re-attach the controlled inner blocks to the blocks tree and // preserve their block order. Otherwise, an inner block controller's blocks // will be deleted entirely from its entity. - stateAfterInsert.order = { - ...stateAfterInsert.order, - ...Object.keys( nestedControllers ).reduce( ( result, key ) => { - if ( state.order[ key ] ) { - result[ key ] = state.order[ key ]; - } - return result; - }, {} ), - }; - stateAfterInsert.tree = { - ...stateAfterInsert.tree, - ...Object.keys( nestedControllers ).reduce( ( result, _key ) => { - const key = `controlled||${ _key }`; - if ( state.tree[ key ] ) { - result[ key ] = state.tree[ key ]; - } - return result; - }, {} ), - }; + const stateAfterInsertOrder = new Map( stateAfterInsert.order ); + Object.keys( nestedControllers ).forEach( ( key ) => { + if ( state.order.get( key ) ) { + stateAfterInsertOrder.set( key, state.order.get( key ) ); + } + } ); + stateAfterInsert.order = stateAfterInsertOrder; + stateAfterInsert.tree = new Map( stateAfterInsert.tree ); + Object.keys( nestedControllers ).forEach( ( _key ) => { + const key = `controlled||${ _key }`; + if ( state.tree.has( key ) ) { + stateAfterInsert.tree.set( key, state.tree.get( key ) ); + } + } ); } return stateAfterInsert; }; @@ -724,21 +705,16 @@ const withSaveReusableBlock = ( reducer ) => ( state, action ) => { } state = { ...state }; - - state.attributes = mapValues( - state.attributes, - ( attributes, clientId ) => { - const { name } = state.byClientId[ clientId ]; - if ( name === 'core/block' && attributes.ref === id ) { - return { - ...attributes, - ref: updatedId, - }; - } - - return attributes; + state.attributes = new Map( state.attributes ); + state.attributes.forEach( ( attributes, clientId ) => { + const { name } = state.byClientId.get( clientId ); + if ( name === 'core/block' && attributes.ref === id ) { + state.attributes.set( clientId, { + ...attributes, + ref: updatedId, + } ); } - ); + } ); } return reducer( state, action ); @@ -784,18 +760,24 @@ export const blocks = pipe( withIgnoredBlockChange, withResetControlledBlocks )( { - byClientId( state = {}, action ) { + // The state is using a Map instead of a plain object for performance reasons. + // You can run the "./test/performance.js" unit test to check the impact + // code changes can have on this reducer. + byClientId( state = new Map(), action ) { switch ( action.type ) { case 'RECEIVE_BLOCKS': - case 'INSERT_BLOCKS': - return { - ...state, - ...getFlattenedBlocksWithoutAttributes( action.blocks ), - }; - - case 'UPDATE_BLOCK': + case 'INSERT_BLOCKS': { + const newState = new Map( state ); + getFlattenedBlocksWithoutAttributes( action.blocks ).forEach( + ( [ key, value ] ) => { + newState.set( key, value ); + } + ); + return newState; + } + case 'UPDATE_BLOCK': { // Ignore updates if block isn't known. - if ( ! state[ action.clientId ] ) { + if ( ! state.has( action.clientId ) ) { return state; } @@ -805,142 +787,184 @@ export const blocks = pipe( return state; } - return { - ...state, - [ action.clientId ]: { - ...state[ action.clientId ], - ...changes, - }, - }; + const newState = new Map( state ); + newState.set( action.clientId, { + ...state.get( action.clientId ), + ...changes, + } ); + return newState; + } - case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': + case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': { if ( ! action.blocks ) { return state; } - return { - ...omit( state, action.replacedClientIds ), - ...getFlattenedBlocksWithoutAttributes( action.blocks ), - }; + const newState = new Map( state ); + action.replacedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); - case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': - return omit( state, action.removedClientIds ); + getFlattenedBlocksWithoutAttributes( action.blocks ).forEach( + ( [ key, value ] ) => { + newState.set( key, value ); + } + ); + return newState; + } + + case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': { + const newState = new Map( state ); + action.removedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + return newState; + } } return state; }, - attributes( state = {}, action ) { + // The state is using a Map instead of a plain object for performance reasons. + // You can run the "./test/performance.js" unit test to check the impact + // code changes can have on this reducer. + attributes( state = new Map(), action ) { switch ( action.type ) { case 'RECEIVE_BLOCKS': - case 'INSERT_BLOCKS': - return { - ...state, - ...getFlattenedBlockAttributes( action.blocks ), - }; + case 'INSERT_BLOCKS': { + const newState = new Map( state ); + getFlattenedBlockAttributes( action.blocks ).forEach( + ( [ key, value ] ) => { + newState.set( key, value ); + } + ); + return newState; + } - case 'UPDATE_BLOCK': + case 'UPDATE_BLOCK': { // Ignore updates if block isn't known or there are no attribute changes. if ( - ! state[ action.clientId ] || + ! state.get( action.clientId ) || ! action.updates.attributes ) { return state; } - return { - ...state, - [ action.clientId ]: { - ...state[ action.clientId ], - ...action.updates.attributes, - }, - }; + const newState = new Map( state ); + newState.set( action.clientId, { + ...state.get( action.clientId ), + ...action.updates.attributes, + } ); + return newState; + } case 'UPDATE_BLOCK_ATTRIBUTES': { // Avoid a state change if none of the block IDs are known. - if ( action.clientIds.every( ( id ) => ! state[ id ] ) ) { + if ( action.clientIds.every( ( id ) => ! state.get( id ) ) ) { return state; } - const next = action.clientIds.reduce( - ( accumulator, id ) => ( { - ...accumulator, - [ id ]: Object.entries( - action.uniqueByBlock - ? action.attributes[ id ] - : action.attributes ?? {} - ).reduce( ( result, [ key, value ] ) => { - // Consider as updates only changed values. - if ( value !== result[ key ] ) { - result = getMutateSafeObject( - state[ id ], - result - ); - result[ key ] = value; - } - - return result; - }, state[ id ] ), - } ), - {} - ); - - if ( - action.clientIds.every( - ( id ) => next[ id ] === state[ id ] - ) - ) { - return state; + let hasChange = false; + const newState = new Map( state ); + for ( const clientId of action.clientIds ) { + const updatedAttributeEntries = Object.entries( + action.uniqueByBlock + ? action.attributes[ clientId ] + : action.attributes ?? {} + ); + if ( updatedAttributeEntries.length === 0 ) { + continue; + } + let hasUpdatedAttributes = false; + const existingAttributes = state.get( clientId ); + const newAttributes = {}; + updatedAttributeEntries.forEach( ( [ key, value ] ) => { + if ( existingAttributes[ key ] !== value ) { + hasUpdatedAttributes = true; + newAttributes[ key ] = value; + } + } ); + hasChange = hasChange || hasUpdatedAttributes; + if ( hasUpdatedAttributes ) { + newState.set( clientId, { + ...existingAttributes, + ...newAttributes, + } ); + } } - return { ...state, ...next }; + return hasChange ? newState : state; } - case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': + case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': { if ( ! action.blocks ) { return state; } - return { - ...omit( state, action.replacedClientIds ), - ...getFlattenedBlockAttributes( action.blocks ), - }; + const newState = new Map( state ); + action.replacedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + getFlattenedBlockAttributes( action.blocks ).forEach( + ( [ key, value ] ) => { + newState.set( key, value ); + } + ); + return newState; + } - case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': - return omit( state, action.removedClientIds ); + case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': { + const newState = new Map( state ); + action.removedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + return newState; + } } return state; }, - order( state = {}, action ) { + // The state is using a Map instead of a plain object for performance reasons. + // You can run the "./test/performance.js" unit test to check the impact + // code changes can have on this reducer. + order( state = new Map(), action ) { switch ( action.type ) { case 'RECEIVE_BLOCKS': { const blockOrder = mapBlockOrder( action.blocks ); - return { - ...state, - ...omit( blockOrder, '' ), - '': ( state?.[ '' ] || [] ).concat( blockOrder[ '' ] ), - }; + const newState = new Map( state ); + blockOrder.forEach( ( order, clientId ) => { + if ( clientId !== '' ) { + newState.set( clientId, order ); + } + } ); + newState.set( + '', + ( state.get( '' ) ?? [] ).concat( blockOrder[ '' ] ) + ); + return newState; } case 'INSERT_BLOCKS': { const { rootClientId = '' } = action; - const subState = state[ rootClientId ] || []; + const subState = state.get( rootClientId ) || []; const mappedBlocks = mapBlockOrder( action.blocks, rootClientId ); const { index = subState.length } = action; - - return { - ...state, - ...mappedBlocks, - [ rootClientId ]: insertAt( + const newState = new Map( state ); + mappedBlocks.forEach( ( order, clientId ) => { + newState.set( clientId, order ); + } ); + newState.set( + rootClientId, + insertAt( subState, - mappedBlocks[ rootClientId ], + mappedBlocks.get( rootClientId ), index - ), - }; + ) + ); + return newState; } case 'MOVE_BLOCKS_TO_POSITION': { @@ -949,65 +973,68 @@ export const blocks = pipe( toRootClientId = '', clientIds, } = action; - const { index = state[ toRootClientId ].length } = action; + const { index = state.get( toRootClientId ).length } = action; // Moving inside the same parent block. if ( fromRootClientId === toRootClientId ) { - const subState = state[ toRootClientId ]; + const subState = state.get( toRootClientId ); const fromIndex = subState.indexOf( clientIds[ 0 ] ); - return { - ...state, - [ toRootClientId ]: moveTo( - state[ toRootClientId ], + const newState = new Map( state ); + newState.set( + toRootClientId, + moveTo( + state.get( toRootClientId ), fromIndex, index, clientIds.length - ), - }; + ) + ); + return newState; } // Moving from a parent block to another. - return { - ...state, - [ fromRootClientId ]: - state[ fromRootClientId ]?.filter( - ( id ) => ! clientIds.includes( id ) - ) ?? [], - [ toRootClientId ]: insertAt( - state[ toRootClientId ], - clientIds, - index - ), - }; + const newState = new Map( state ); + newState.set( + fromRootClientId, + state + .get( fromRootClientId ) + ?.filter( ( id ) => ! clientIds.includes( id ) ) ?? [] + ); + newState.set( + toRootClientId, + insertAt( state.get( toRootClientId ), clientIds, index ) + ); + return newState; } case 'MOVE_BLOCKS_UP': { const { clientIds, rootClientId = '' } = action; const firstClientId = clientIds[ 0 ]; - const subState = state[ rootClientId ]; + const subState = state.get( rootClientId ); if ( ! subState.length || firstClientId === subState[ 0 ] ) { return state; } const firstIndex = subState.indexOf( firstClientId ); - - return { - ...state, - [ rootClientId ]: moveTo( + const newState = new Map( state ); + newState.set( + rootClientId, + moveTo( subState, firstIndex, firstIndex - 1, clientIds.length - ), - }; + ) + ); + return newState; } case 'MOVE_BLOCKS_DOWN': { const { clientIds, rootClientId = '' } = action; const firstClientId = clientIds[ 0 ]; const lastClientId = clientIds[ clientIds.length - 1 ]; - const subState = state[ rootClientId ]; + const subState = state.get( rootClientId ); if ( ! subState.length || @@ -1017,16 +1044,17 @@ export const blocks = pipe( } const firstIndex = subState.indexOf( firstClientId ); - - return { - ...state, - [ rootClientId ]: moveTo( + const newState = new Map( state ); + newState.set( + rootClientId, + moveTo( subState, firstIndex, firstIndex + 1, clientIds.length - ), - }; + ) + ); + return newState; } case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': { @@ -1036,55 +1064,52 @@ export const blocks = pipe( } const mappedBlocks = mapBlockOrder( action.blocks ); + const newState = new Map( state ); + action.replacedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + mappedBlocks.forEach( ( order, clientId ) => { + if ( clientId !== '' ) { + newState.set( clientId, order ); + } + } ); + newState.forEach( ( order, clientId ) => { + const newSubOrder = Object.values( order ).reduce( + ( result, subClientId ) => { + if ( subClientId === clientIds[ 0 ] ) { + return [ ...result, ...mappedBlocks.get( '' ) ]; + } - return pipe( [ - ( nextState ) => - omit( nextState, action.replacedClientIds ), - ( nextState ) => ( { - ...nextState, - ...omit( mappedBlocks, '' ), - } ), - ( nextState ) => - mapValues( nextState, ( subState ) => - Object.values( subState ).reduce( - ( result, clientId ) => { - if ( clientId === clientIds[ 0 ] ) { - return [ - ...result, - ...mappedBlocks[ '' ], - ]; - } - - if ( - clientIds.indexOf( clientId ) === -1 - ) { - result.push( clientId ); - } - - return result; - }, - [] - ) - ), - ] )( state ); + if ( clientIds.indexOf( subClientId ) === -1 ) { + result.push( subClientId ); + } + + return result; + }, + [] + ); + newState.set( clientId, newSubOrder ); + } ); + return newState; } - case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': - return pipe( [ - // Remove inner block ordering for removed blocks. - ( nextState ) => omit( nextState, action.removedClientIds ), - - // Remove deleted blocks from other blocks' orderings. - ( nextState ) => - mapValues( - nextState, - ( subState ) => - subState?.filter( - ( id ) => - ! action.removedClientIds.includes( id ) - ) ?? [] - ), - ] )( state ); + case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': { + const newState = new Map( state ); + // Remove inner block ordering for removed blocks. + action.removedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + newState.forEach( ( order, clientId ) => { + const newSubOrder = + order?.filter( + ( id ) => ! action.removedClientIds.includes( id ) + ) ?? []; + if ( newSubOrder.length !== order.length ) { + newState.set( clientId, newSubOrder ); + } + } ); + return newState; + } } return state; @@ -1092,44 +1117,55 @@ export const blocks = pipe( // While technically redundant data as the inverse of `order`, it serves as // an optimization for the selectors which derive the ancestry of a block. - parents( state = {}, action ) { + parents( state = new Map(), action ) { switch ( action.type ) { - case 'RECEIVE_BLOCKS': - return { - ...state, - ...mapBlockParents( action.blocks ), - }; - - case 'INSERT_BLOCKS': - return { - ...state, - ...mapBlockParents( - action.blocks, - action.rootClientId || '' - ), - }; - + case 'RECEIVE_BLOCKS': { + const newState = new Map( state ); + mapBlockParents( action.blocks ).forEach( + ( [ key, value ] ) => { + newState.set( key, value ); + } + ); + return newState; + } + case 'INSERT_BLOCKS': { + const newState = new Map( state ); + mapBlockParents( + action.blocks, + action.rootClientId || '' + ).forEach( ( [ key, value ] ) => { + newState.set( key, value ); + } ); + return newState; + } case 'MOVE_BLOCKS_TO_POSITION': { - return { - ...state, - ...action.clientIds.reduce( ( accumulator, id ) => { - accumulator[ id ] = action.toRootClientId || ''; - return accumulator; - }, {} ), - }; + const newState = new Map( state ); + action.clientIds.forEach( ( id ) => { + newState.set( id, action.toRootClientId || '' ); + } ); + return newState; } - case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': - return { - ...omit( state, action.replacedClientIds ), - ...mapBlockParents( - action.blocks, - state[ action.clientIds[ 0 ] ] - ), - }; - - case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': - return omit( state, action.removedClientIds ); + case 'REPLACE_BLOCKS_AUGMENTED_WITH_CHILDREN': { + const newState = new Map( state ); + action.replacedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + mapBlockParents( + action.blocks, + state.get( action.clientIds[ 0 ] ) + ).forEach( ( [ key, value ] ) => { + newState.set( key, value ); + } ); + return newState; + } + case 'REMOVE_BLOCKS_AUGMENTED_WITH_CHILDREN': { + const newState = new Map( state ); + action.removedClientIds.forEach( ( clientId ) => { + newState.delete( clientId ); + } ); + return newState; + } } return state; @@ -1492,7 +1528,7 @@ export function insertionPoint( state = null, action ) { }; // Bail out updates if the states are the same. - return isEqual( state, nextState ) ? state : nextState; + return fastDeepEqual( state, nextState ) ? state : nextState; } case 'HIDE_INSERTION_POINT': @@ -1617,7 +1653,7 @@ export const blockListSettings = ( state = {}, action ) => { return state; } - if ( isEqual( state[ clientId ], action.settings ) ) { + if ( fastDeepEqual( state[ clientId ], action.settings ) ) { return state; } diff --git a/packages/block-editor/src/store/selectors.js b/packages/block-editor/src/store/selectors.js index 0345b29ff9332..3c048a7d58a29 100644 --- a/packages/block-editor/src/store/selectors.js +++ b/packages/block-editor/src/store/selectors.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { map, find, filter, orderBy } from 'lodash'; +import { map, find } from 'lodash'; import createSelector from 'rememo'; /** @@ -27,6 +27,7 @@ import deprecated from '@wordpress/deprecated'; * Internal dependencies */ import { mapRichTextSettings } from './utils'; +import { orderBy } from '../utils/sorting'; /** * A block selection object. @@ -65,12 +66,12 @@ const EMPTY_ARRAY = []; * @return {string} Block name. */ export function getBlockName( state, clientId ) { - const block = state.blocks.byClientId[ clientId ]; + const block = state.blocks.byClientId.get( clientId ); const socialLinkName = 'core/social-link'; if ( Platform.OS !== 'web' && block?.name === socialLinkName ) { - const attributes = state.blocks.attributes[ clientId ]; - const { service } = attributes; + const attributes = state.blocks.attributes.get( clientId ); + const { service } = attributes ?? {}; return service ? `${ socialLinkName }-${ service }` : socialLinkName; } @@ -86,7 +87,7 @@ export function getBlockName( state, clientId ) { * @return {boolean} Is Valid. */ export function isBlockValid( state, clientId ) { - const block = state.blocks.byClientId[ clientId ]; + const block = state.blocks.byClientId.get( clientId ); return !! block && block.isValid; } @@ -100,12 +101,12 @@ export function isBlockValid( state, clientId ) { * @return {Object?} Block attributes. */ export function getBlockAttributes( state, clientId ) { - const block = state.blocks.byClientId[ clientId ]; + const block = state.blocks.byClientId.get( clientId ); if ( ! block ) { return null; } - return state.blocks.attributes[ clientId ]; + return state.blocks.attributes.get( clientId ); } /** @@ -130,29 +131,27 @@ export function getBlockAttributes( state, clientId ) { * @return {Object} Parsed block object. */ export function getBlock( state, clientId ) { - const block = state.blocks.byClientId[ clientId ]; - if ( ! block ) { + if ( ! state.blocks.byClientId.has( clientId ) ) { return null; } - return state.blocks.tree[ clientId ]; + return state.blocks.tree.get( clientId ); } export const __unstableGetBlockWithoutInnerBlocks = createSelector( ( state, clientId ) => { - const block = state.blocks.byClientId[ clientId ]; - if ( ! block ) { + if ( ! state.blocks.byClientId.has( clientId ) ) { return null; } return { - ...block, + ...state.blocks.byClientId.get( clientId ), attributes: getBlockAttributes( state, clientId ), }; }, ( state, clientId ) => [ - state.blocks.byClientId[ clientId ], - state.blocks.attributes[ clientId ], + state.blocks.byClientId.get( clientId ), + state.blocks.attributes.get( clientId ), ] ); @@ -171,7 +170,7 @@ export function getBlocks( state, rootClientId ) { ! rootClientId || ! areInnerBlocksControlled( state, rootClientId ) ? rootClientId || '' : 'controlled||' + rootClientId; - return state.blocks.tree[ treeKey ]?.innerBlocks || EMPTY_ARRAY; + return state.blocks.tree.get( treeKey )?.innerBlocks || EMPTY_ARRAY; } /** @@ -274,7 +273,7 @@ export const getGlobalBlockCount = createSelector( return clientIds.length; } return clientIds.reduce( ( accumulator, clientId ) => { - const block = state.blocks.byClientId[ clientId ]; + const block = state.blocks.byClientId.get( clientId ); return block.name === blockName ? accumulator + 1 : accumulator; }, 0 ); }, @@ -296,7 +295,7 @@ export const __experimentalGetGlobalBlocksByName = createSelector( } const clientIds = getClientIdsWithDescendants( state ); const foundBlocks = clientIds.filter( ( clientId ) => { - const block = state.blocks.byClientId[ clientId ]; + const block = state.blocks.byClientId.get( clientId ); return block.name === blockName; } ); return foundBlocks.length > 0 ? foundBlocks : EMPTY_ARRAY; @@ -322,7 +321,7 @@ export const getBlocksByClientId = createSelector( ( state, clientIds ) => map( Array.isArray( clientIds ) ? clientIds : [ clientIds ], - ( clientId ) => state.blocks.tree[ clientId ] + ( clientId ) => state.blocks.tree.get( clientId ) ) ); @@ -463,8 +462,8 @@ export function getSelectedBlock( state ) { * @return {?string} Root client ID, if exists */ export function getBlockRootClientId( state, clientId ) { - return state.blocks.parents[ clientId ] !== undefined - ? state.blocks.parents[ clientId ] + return state.blocks.parents.has( clientId ) + ? state.blocks.parents.get( clientId ) : null; } @@ -481,8 +480,8 @@ export const getBlockParents = createSelector( ( state, clientId, ascending = false ) => { const parents = []; let current = clientId; - while ( !! state.blocks.parents[ current ] ) { - current = state.blocks.parents[ current ]; + while ( !! state.blocks.parents.get( current ) ) { + current = state.blocks.parents.get( current ); parents.push( current ); } @@ -509,24 +508,20 @@ export const getBlockParentsByBlockName = createSelector( ( state, clientId, blockName, ascending = false ) => { const parents = getBlockParents( state, clientId, ascending ); return map( - filter( - map( parents, ( id ) => ( { - id, - name: getBlockName( state, id ), - } ) ), - ( { name } ) => { - if ( Array.isArray( blockName ) ) { - return blockName.includes( name ); - } - return name === blockName; + map( parents, ( id ) => ( { + id, + name: getBlockName( state, id ), + } ) ).filter( ( { name } ) => { + if ( Array.isArray( blockName ) ) { + return blockName.includes( name ); } - ), + return name === blockName; + } ), ( { id } ) => id ); }, ( state ) => [ state.blocks.parents ] ); - /** * Given a block client ID, returns the root of the hierarchy from which the block is nested, return the block itself for root level blocks. * @@ -540,7 +535,7 @@ export function getBlockHierarchyRootClientId( state, clientId ) { let parent; do { parent = current; - current = state.blocks.parents[ current ]; + current = state.blocks.parents.get( current ); } while ( current ); return parent; } @@ -617,7 +612,7 @@ export function getAdjacentBlockClientId( state, startClientId, modifier = 1 ) { } const { order } = state.blocks; - const orderSet = order[ rootClientId ]; + const orderSet = order.get( rootClientId ); const index = orderSet.indexOf( startClientId ); const nextIndex = index + 1 * modifier; @@ -1132,7 +1127,7 @@ export const __unstableGetSelectedBlocksWithPartialSelection = ( state ) => { * @return {Array} Ordered client IDs of editor blocks. */ export function getBlockOrder( state, rootClientId ) { - return state.blocks.order[ rootClientId || '' ] || EMPTY_ARRAY; + return state.blocks.order.get( rootClientId || '' ) || EMPTY_ARRAY; } /** @@ -1431,19 +1426,14 @@ export function getTemplate( state ) { * @param {Object} state Editor state. * @param {?string} rootClientId Optional block root client ID. * - * @return {?string} Block Template Lock + * @return {string|false} Block Template Lock */ export function getTemplateLock( state, rootClientId ) { if ( ! rootClientId ) { - return state.settings.templateLock; - } - - const blockListSettings = getBlockListSettings( state, rootClientId ); - if ( ! blockListSettings ) { - return undefined; + return state.settings.templateLock ?? false; } - return blockListSettings.templateLock; + return getBlockListSettings( state, rootClientId )?.templateLock ?? false; } const checkAllowList = ( list, item, defaultResult = null ) => { @@ -1598,7 +1588,7 @@ export const canInsertBlockType = createSelector( canInsertBlockTypeUnmemoized, ( state, blockName, rootClientId ) => [ state.blockListSettings[ rootClientId ], - state.blocks.byClientId[ rootClientId ], + state.blocks.byClientId.get( rootClientId ), state.settings.allowedBlockTypes, state.settings.templateLock, ] @@ -2111,7 +2101,7 @@ export const getBlockTransformItems = createSelector( 'desc' ); }, - ( state, rootClientId ) => [ + ( state, blocks, rootClientId ) => [ state.blockListSettings[ rootClientId ], state.blocks.byClientId, state.preferences.insertUsage, @@ -2167,7 +2157,7 @@ export const __experimentalGetAllowedBlocks = createSelector( return; } - return filter( getBlockTypes(), ( blockType ) => + return getBlockTypes().filter( ( blockType ) => canIncludeBlockTypeInInserter( state, blockType, rootClientId ) ); }, @@ -2214,7 +2204,7 @@ export const __experimentalGetDirectInsertBlock = createSelector( }, ( state, rootClientId ) => [ state.blockListSettings[ rootClientId ], - state.blocks.tree[ rootClientId ], + state.blocks.tree.get( rootClientId ), ] ); @@ -2292,8 +2282,7 @@ const getAllAllowedPatterns = createSelector( export const __experimentalGetAllowedPatterns = createSelector( ( state, rootClientId = null ) => { const availableParsedPatterns = getAllAllowedPatterns( state ); - const patternsAllowed = filter( - availableParsedPatterns, + const patternsAllowed = availableParsedPatterns.filter( ( { blocks } ) => blocks.every( ( { name } ) => canInsertBlockType( state, name, rootClientId ) @@ -2307,7 +2296,7 @@ export const __experimentalGetAllowedPatterns = createSelector( state.settings.allowedBlockTypes, state.settings.templateLock, state.blockListSettings[ rootClientId ], - state.blocks.byClientId[ rootClientId ], + state.blocks.byClientId.get( rootClientId ), ] ); @@ -2340,7 +2329,7 @@ export const __experimentalGetPatternsByBlockTypes = createSelector( ) ); }, - ( state, rootClientId ) => [ + ( state, blockNames, rootClientId ) => [ ...__experimentalGetAllowedPatterns.getDependants( state, rootClientId @@ -2401,7 +2390,7 @@ export const __experimentalGetPatternTransformItems = createSelector( rootClientId ); }, - ( state, rootClientId ) => [ + ( state, blocks, rootClientId ) => [ ...__experimentalGetPatternsByBlockTypes.getDependants( state, rootClientId @@ -2664,6 +2653,16 @@ export function wasBlockJustInserted( state, clientId, source ) { ); } +/** + * Gets the client id of the last inserted block. + * + * @param {Object} state Global application state. + * @return {string|undefined} Client Id of the last inserted block. + */ +export function getLastInsertedBlockClientId( state ) { + return state?.lastBlockInserted?.clientId; +} + /** * Tells if the block is visible on the canvas or not. * @@ -2701,8 +2700,8 @@ export const __unstableGetContentLockingParent = createSelector( ( state, clientId ) => { let current = clientId; let result; - while ( !! state.blocks.parents[ current ] ) { - current = state.blocks.parents[ current ]; + while ( state.blocks.parents.has( current ) ) { + current = state.blocks.parents.get( current ); if ( getTemplateLock( state, current ) === 'contentOnly' ) { result = current; } diff --git a/packages/block-editor/src/store/test/performance.js b/packages/block-editor/src/store/test/performance.js new file mode 100644 index 0000000000000..2da3c1a41db2a --- /dev/null +++ b/packages/block-editor/src/store/test/performance.js @@ -0,0 +1,71 @@ +/** + * Internal dependencies + */ +import reducer from '../reducer'; + +describe( 'performance', () => { + const state = reducer( undefined, { type: '@@init' } ); + const blocks = []; + for ( let i = 0; i < 100000; i++ ) { + blocks.push( { + clientId: `block-${ i }`, + attributes: { content: `paragraph ${ i }` }, + innerBlocks: [], + } ); + } + + let preparedState; + + it( 'should reset blocks', () => { + preparedState = reducer( state, { + type: 'RESET_BLOCKS', + blocks, + } ); + expect( preparedState ).toBeDefined(); + } ); + + it( 'should update blocks', () => { + const updatedState = reducer( preparedState, { + type: 'UPDATE_BLOCK_ATTRIBUTES', + clientIds: [ 'block-10' ], + attributes: { + content: 'updated paragraph 10', + }, + } ); + + expect( updatedState ).toBeDefined(); + } ); + + it( 'should replace blocks (Enter in paragraphs)', () => { + const updatedState = reducer( preparedState, { + type: 'REPLACE_BLOCKS', + clientIds: [ 'block-10' ], + blocks: [ + { + clientId: `block-10`, + attributes: { content: `paragraph 10` }, + innerBlocks: [], + }, + { + clientId: `block-10-2`, + attributes: { content: '' }, + innerBlocks: [], + }, + ], + indexToSelect: 10, + initialPosition: 0, + } ); + + expect( updatedState ).toBeDefined(); + } ); + + it( 'should move blocks', () => { + const updatedState = reducer( preparedState, { + type: 'MOVE_BLOCKS_DOWN', + clientIds: [ 'block-10' ], + rootClientId: '', + } ); + + expect( updatedState ).toBeDefined(); + } ); +} ); diff --git a/packages/block-editor/src/store/test/reducer.js b/packages/block-editor/src/store/test/reducer.js index 9f2e623c70111..61a51a5f28cd0 100644 --- a/packages/block-editor/src/store/test/reducer.js +++ b/packages/block-editor/src/store/test/reducer.js @@ -210,38 +210,48 @@ describe( 'state', () => { } ); it( 'can replace a child block', () => { const existingState = deepFreeze( { - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - 'chicken-child': { - clientId: 'chicken-child', - name: 'core/test-child-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - 'chicken-child': { - attr: true, - }, - }, - order: { - '': [ 'chicken' ], - chicken: [ 'chicken-child' ], - 'chicken-child': [], - }, - parents: { - chicken: '', - 'chicken-child': 'chicken', - }, - tree: { - '': {}, - chicken: {}, - 'chicken-child': {}, - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + 'chicken-child': { + clientId: 'chicken-child', + name: 'core/test-child-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + 'chicken-child': { + attr: true, + }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ 'chicken-child' ], + 'chicken-child': [], + } ) + ), + parents: new Map( + Object.entries( { + chicken: '', + 'chicken-child': 'chicken', + } ) + ), + tree: new Map( + Object.entries( { + '': {}, + chicken: {}, + 'chicken-child': {}, + } ) + ), controlledInnerBlocks: {}, } ); @@ -264,66 +274,84 @@ describe( 'state', () => { expect( restState ).toEqual( { isPersistentChange: true, isIgnoredChange: false, - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - [ newChildBlockId ]: { - clientId: newChildBlockId, - name: 'core/test-child-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - [ newChildBlockId ]: { - attr: false, - attr2: 'perfect', - }, - }, - order: { - '': [ 'chicken' ], - chicken: [ newChildBlockId ], - [ newChildBlockId ]: [], - }, - parents: { - [ newChildBlockId ]: 'chicken', - chicken: '', - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + [ newChildBlockId ]: { + clientId: newChildBlockId, + name: 'core/test-child-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + [ newChildBlockId ]: { + attr: false, + attr2: 'perfect', + }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ newChildBlockId ], + [ newChildBlockId ]: [], + } ) + ), + parents: new Map( + Object.entries( { + [ newChildBlockId ]: 'chicken', + chicken: '', + } ) + ), controlledInnerBlocks: {}, } ); - expect( state.tree.chicken ).not.toBe( - existingState.tree.chicken + expect( state.tree.get( 'chicken' ) ).not.toBe( + existingState.tree.get( 'chicken' ) ); } ); it( 'can insert a child block', () => { const existingState = deepFreeze( { - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - }, - order: { - '': [ 'chicken' ], - chicken: [], - }, - parents: { - chicken: '', - }, - tree: { - '': { - innerBlocks: [], - }, - chicken: {}, - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [], + } ) + ), + parents: new Map( + Object.entries( { + chicken: '', + } ) + ), + tree: new Map( + Object.entries( { + '': { + innerBlocks: [], + }, + chicken: {}, + } ) + ), controlledInnerBlocks: {}, } ); @@ -346,46 +374,54 @@ describe( 'state', () => { expect( restState ).toEqual( { isPersistentChange: true, isIgnoredChange: false, - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - [ newChildBlockId ]: { - clientId: newChildBlockId, - name: 'core/test-child-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - [ newChildBlockId ]: { - attr: false, - attr2: 'perfect', - }, - }, - order: { - '': [ 'chicken' ], - chicken: [ newChildBlockId ], - [ newChildBlockId ]: [], - }, - parents: { - [ newChildBlockId ]: 'chicken', - chicken: '', - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + [ newChildBlockId ]: { + clientId: newChildBlockId, + name: 'core/test-child-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + [ newChildBlockId ]: { + attr: false, + attr2: 'perfect', + }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ newChildBlockId ], + [ newChildBlockId ]: [], + } ) + ), + parents: new Map( + Object.entries( { + [ newChildBlockId ]: 'chicken', + chicken: '', + } ) + ), controlledInnerBlocks: {}, } ); - expect( state.tree.chicken ).not.toBe( - existingState.tree.chicken + expect( state.tree.get( 'chicken' ) ).not.toBe( + existingState.tree.get( 'chicken' ) ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.chicken + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'chicken' ) ); - expect( state.tree.chicken.innerBlocks[ 0 ] ).toBe( - state.tree[ newChildBlockId ] + expect( state.tree.get( 'chicken' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( newChildBlockId ) ); - expect( state.tree[ newChildBlockId ] ).toEqual( { + expect( state.tree.get( newChildBlockId ) ).toEqual( { clientId: newChildBlockId, innerBlocks: [], isValid: true, @@ -399,44 +435,52 @@ describe( 'state', () => { it( 'can replace multiple child blocks', () => { const existingState = deepFreeze( { - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - 'chicken-child': { - clientId: 'chicken-child', - name: 'core/test-child-block', - isValid: true, - }, - 'chicken-child-2': { - clientId: 'chicken-child', - name: 'core/test-child-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - 'chicken-child': { - attr: true, - }, - 'chicken-child-2': { - attr2: 'ok', - }, - }, - order: { - '': [ 'chicken' ], - chicken: [ 'chicken-child', 'chicken-child-2' ], - 'chicken-child': [], - 'chicken-child-2': [], - }, - parents: { - chicken: '', - 'chicken-child': 'chicken', - 'chicken-child-2': 'chicken', - }, - tree: {}, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + 'chicken-child': { + clientId: 'chicken-child', + name: 'core/test-child-block', + isValid: true, + }, + 'chicken-child-2': { + clientId: 'chicken-child', + name: 'core/test-child-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + 'chicken-child': { + attr: true, + }, + 'chicken-child-2': { + attr2: 'ok', + }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ 'chicken-child', 'chicken-child-2' ], + 'chicken-child': [], + 'chicken-child-2': [], + } ) + ), + parents: new Map( + Object.entries( { + chicken: '', + 'chicken-child': 'chicken', + 'chicken-child-2': 'chicken', + } ) + ), + tree: new Map(), controlledInnerBlocks: {}, } ); @@ -470,75 +514,83 @@ describe( 'state', () => { expect( restState ).toEqual( { isPersistentChange: true, isIgnoredChange: false, - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - [ newChildBlockId1 ]: { - clientId: newChildBlockId1, - name: 'core/test-child-block', - isValid: true, - }, - [ newChildBlockId2 ]: { - clientId: newChildBlockId2, - name: 'core/test-child-block', - isValid: true, - }, - [ newChildBlockId3 ]: { - clientId: newChildBlockId3, - name: 'core/test-child-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - [ newChildBlockId1 ]: { - attr: false, - attr2: 'perfect', - }, - [ newChildBlockId2 ]: { - attr: true, - attr2: 'not-perfect', - }, - [ newChildBlockId3 ]: { - attr2: 'hello', - }, - }, - order: { - '': [ 'chicken' ], - chicken: [ - newChildBlockId1, - newChildBlockId2, - newChildBlockId3, - ], - [ newChildBlockId1 ]: [], - [ newChildBlockId2 ]: [], - [ newChildBlockId3 ]: [], - }, - parents: { - chicken: '', - [ newChildBlockId1 ]: 'chicken', - [ newChildBlockId2 ]: 'chicken', - [ newChildBlockId3 ]: 'chicken', - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + [ newChildBlockId1 ]: { + clientId: newChildBlockId1, + name: 'core/test-child-block', + isValid: true, + }, + [ newChildBlockId2 ]: { + clientId: newChildBlockId2, + name: 'core/test-child-block', + isValid: true, + }, + [ newChildBlockId3 ]: { + clientId: newChildBlockId3, + name: 'core/test-child-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + [ newChildBlockId1 ]: { + attr: false, + attr2: 'perfect', + }, + [ newChildBlockId2 ]: { + attr: true, + attr2: 'not-perfect', + }, + [ newChildBlockId3 ]: { + attr2: 'hello', + }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ + newChildBlockId1, + newChildBlockId2, + newChildBlockId3, + ], + [ newChildBlockId1 ]: [], + [ newChildBlockId2 ]: [], + [ newChildBlockId3 ]: [], + } ) + ), + parents: new Map( + Object.entries( { + chicken: '', + [ newChildBlockId1 ]: 'chicken', + [ newChildBlockId2 ]: 'chicken', + [ newChildBlockId3 ]: 'chicken', + } ) + ), controlledInnerBlocks: {}, } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.chicken + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'chicken' ) ); - expect( state.tree.chicken.innerBlocks[ 0 ] ).toBe( - state.tree[ newChildBlockId1 ] + expect( state.tree.get( 'chicken' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( newChildBlockId1 ) ); - expect( state.tree.chicken.innerBlocks[ 1 ] ).toBe( - state.tree[ newChildBlockId2 ] + expect( state.tree.get( 'chicken' ).innerBlocks[ 1 ] ).toBe( + state.tree.get( newChildBlockId2 ) ); - expect( state.tree.chicken.innerBlocks[ 2 ] ).toBe( - state.tree[ newChildBlockId3 ] + expect( state.tree.get( 'chicken' ).innerBlocks[ 2 ] ).toBe( + state.tree.get( newChildBlockId3 ) ); - expect( state.tree[ newChildBlockId1 ] ).toEqual( { + expect( state.tree.get( newChildBlockId1 ) ).toEqual( { innerBlocks: [], clientId: newChildBlockId1, name: 'core/test-child-block', @@ -552,42 +604,52 @@ describe( 'state', () => { it( 'can replace a child block that has other children', () => { const existingState = deepFreeze( { - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - 'chicken-child': { - clientId: 'chicken-child', - name: 'core/test-child-block', - isValid: true, - }, - 'chicken-grand-child': { - clientId: 'chicken-child', - name: 'core/test-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - 'chicken-child': {}, - 'chicken-grand-child': {}, - }, - order: { - '': [ 'chicken' ], - chicken: [ 'chicken-child' ], - 'chicken-child': [ 'chicken-grand-child' ], - 'chicken-grand-child': [], - }, - parents: { - chicken: '', - 'chicken-child': 'chicken', - 'chicken-grand-child': 'chicken-child', - }, - tree: { - chicken: {}, - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + 'chicken-child': { + clientId: 'chicken-child', + name: 'core/test-child-block', + isValid: true, + }, + 'chicken-grand-child': { + clientId: 'chicken-child', + name: 'core/test-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + 'chicken-child': {}, + 'chicken-grand-child': {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ 'chicken-child' ], + 'chicken-child': [ 'chicken-grand-child' ], + 'chicken-grand-child': [], + } ) + ), + parents: new Map( + Object.entries( { + chicken: '', + 'chicken-child': 'chicken', + 'chicken-grand-child': 'chicken-child', + } ) + ), + tree: new Map( + Object.entries( { + chicken: {}, + } ) + ), controlledInnerBlocks: {}, } ); @@ -607,37 +669,45 @@ describe( 'state', () => { expect( restState ).toEqual( { isPersistentChange: true, isIgnoredChange: false, - byClientId: { - chicken: { - clientId: 'chicken', - name: 'core/test-parent-block', - isValid: true, - }, - [ newChildBlockId ]: { - clientId: newChildBlockId, - name: 'core/test-block', - isValid: true, - }, - }, - attributes: { - chicken: {}, - [ newChildBlockId ]: {}, - }, - order: { - '': [ 'chicken' ], - chicken: [ newChildBlockId ], - [ newChildBlockId ]: [], - }, - parents: { - chicken: '', - [ newChildBlockId ]: 'chicken', - }, + byClientId: new Map( + Object.entries( { + chicken: { + clientId: 'chicken', + name: 'core/test-parent-block', + isValid: true, + }, + [ newChildBlockId ]: { + clientId: newChildBlockId, + name: 'core/test-block', + isValid: true, + }, + } ) + ), + attributes: new Map( + Object.entries( { + chicken: {}, + [ newChildBlockId ]: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'chicken' ], + chicken: [ newChildBlockId ], + [ newChildBlockId ]: [], + } ) + ), + parents: new Map( + Object.entries( { + chicken: '', + [ newChildBlockId ]: 'chicken', + } ) + ), controlledInnerBlocks: {}, } ); // The block object of the parent should be updated. - expect( state.tree.chicken ).not.toBe( - existingState.tree.chicken + expect( state.tree.get( 'chicken' ) ).not.toBe( + existingState.tree.get( 'chicken' ) ); } ); } ); @@ -646,13 +716,13 @@ describe( 'state', () => { const state = blocks( undefined, {} ); expect( state ).toEqual( { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), isPersistentChange: true, isIgnoredChange: false, - tree: {}, + tree: new Map(), controlledInnerBlocks: {}, } ); } ); @@ -664,20 +734,20 @@ describe( 'state', () => { blocks: [ { clientId: 'bananas', innerBlocks: [] } ], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 1 ); - expect( Object.values( state.byClientId )[ 0 ].clientId ).toBe( + expect( state.byClientId.size ).toBe( 1 ); + expect( state.byClientId.get( 'bananas' ).clientId ).toBe( 'bananas' ); - expect( state.order ).toEqual( { + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ 'bananas' ], bananas: [], } ); - expect( state.tree.bananas ).toEqual( { + expect( state.tree.get( 'bananas' ) ).toEqual( { clientId: 'bananas', innerBlocks: [], } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.bananas + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'bananas' ) ); } ); } ); @@ -696,8 +766,8 @@ describe( 'state', () => { ], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 2 ); - expect( state.order ).toEqual( { + expect( state.byClientId.size ).toBe( 2 ); + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ 'bananas' ], apples: [], bananas: [ 'apples' ], @@ -727,21 +797,21 @@ describe( 'state', () => { ], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 2 ); - expect( Object.values( state.byClientId )[ 1 ].clientId ).toBe( - 'ribs' - ); - expect( state.order ).toEqual( { + expect( state.byClientId.size ).toBe( 2 ); + expect( state.byClientId.get( 'ribs' ).clientId ).toBe( 'ribs' ); + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ 'chicken', 'ribs' ], chicken: [], ribs: [], } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.chicken + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'chicken' ) ); - expect( state.tree[ '' ].innerBlocks[ 1 ] ).toBe( state.tree.ribs ); - expect( state.tree.chicken ).toEqual( { + expect( state.tree.get( '' ).innerBlocks[ 1 ] ).toBe( + state.tree.get( 'ribs' ) + ); + expect( state.tree.get( 'chicken' ) ).toEqual( { clientId: 'chicken', name: 'core/test-block', attributes: {}, @@ -773,24 +843,22 @@ describe( 'state', () => { ], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 1 ); - expect( Object.values( state.byClientId )[ 0 ].name ).toBe( + expect( state.byClientId.size ).toBe( 1 ); + expect( state.byClientId.get( 'wings' ).name ).toBe( 'core/freeform' ); - expect( Object.values( state.byClientId )[ 0 ].clientId ).toBe( - 'wings' - ); - expect( state.order ).toEqual( { + expect( state.byClientId.get( 'wings' ).clientId ).toBe( 'wings' ); + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ 'wings' ], wings: [], } ); - expect( state.parents ).toEqual( { + expect( Object.fromEntries( state.parents ) ).toEqual( { wings: '', } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.wings + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'wings' ) ); - expect( state.tree.wings ).toEqual( { + expect( state.tree.get( 'wings' ) ).toEqual( { clientId: 'wings', name: 'core/freeform', innerBlocks: [], @@ -815,8 +883,8 @@ describe( 'state', () => { blocks: [], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 0 ); - expect( state.tree[ '' ].innerBlocks ).toHaveLength( 0 ); + expect( state.byClientId.size ).toBe( 0 ); + expect( state.tree.get( '' ).innerBlocks ).toHaveLength( 0 ); } ); it( 'should replace the block and remove references to its inner blocks', () => { @@ -850,18 +918,18 @@ describe( 'state', () => { ], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 1 ); - expect( state.order ).toEqual( { + expect( state.byClientId.size ).toBe( 1 ); + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ 'wings' ], wings: [], } ); - expect( state.parents ).toEqual( { + expect( Object.fromEntries( state.parents ) ).toEqual( { wings: '', } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.wings + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'wings' ) ); - expect( state.tree.wings ).toEqual( { + expect( state.tree.get( 'wings' ) ).toEqual( { clientId: 'wings', name: 'core/freeform', innerBlocks: [], @@ -885,21 +953,21 @@ describe( 'state', () => { blocks: [ replacementBlock ], } ); - expect( state.order ).toEqual( { + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ wrapperBlock.clientId ], [ wrapperBlock.clientId ]: [ replacementBlock.clientId ], [ replacementBlock.clientId ]: [], } ); - expect( state.parents ).toEqual( { + expect( Object.fromEntries( state.parents ) ).toEqual( { [ wrapperBlock.clientId ]: '', [ replacementBlock.clientId ]: wrapperBlock.clientId, } ); - expect( state.tree[ wrapperBlock.clientId ].innerBlocks[ 0 ] ).toBe( - state.tree[ replacementBlock.clientId ] - ); - expect( state.tree[ replacementBlock.clientId ] ).toEqual( { + expect( + state.tree.get( wrapperBlock.clientId ).innerBlocks[ 0 ] + ).toBe( state.tree.get( replacementBlock.clientId ) ); + expect( state.tree.get( replacementBlock.clientId ) ).toEqual( { clientId: replacementBlock.clientId, name: 'core/test-block', innerBlocks: [], @@ -932,22 +1000,22 @@ describe( 'state', () => { ], } ); - expect( Object.keys( replacedState.byClientId ) ).toHaveLength( 1 ); - expect( Object.values( originalState.byClientId )[ 0 ].name ).toBe( + expect( replacedState.byClientId.size ).toBe( 1 ); + expect( originalState.byClientId.get( 'chicken' ).name ).toBe( 'core/test-block' ); - expect( Object.values( replacedState.byClientId )[ 0 ].name ).toBe( + expect( replacedState.byClientId.get( 'chicken' ).name ).toBe( 'core/freeform' ); - expect( - Object.values( replacedState.byClientId )[ 0 ].clientId - ).toBe( 'chicken' ); - expect( replacedState.order ).toEqual( { + expect( replacedState.byClientId.get( 'chicken' ).clientId ).toBe( + 'chicken' + ); + expect( Object.fromEntries( replacedState.order ) ).toEqual( { '': [ 'chicken' ], chicken: [], } ); - expect( originalState.tree.chicken ).not.toBe( - replacedState.tree.chicken + expect( originalState.tree.get( 'chicken' ) ).not.toBe( + replacedState.tree.get( 'chicken' ) ); const nestedBlock = { @@ -977,16 +1045,16 @@ describe( 'state', () => { blocks: [ replacementNestedBlock ], } ); - expect( replacedNestedState.order ).toEqual( { + expect( Object.fromEntries( replacedNestedState.order ) ).toEqual( { '': [ wrapperBlock.clientId ], [ wrapperBlock.clientId ]: [ replacementNestedBlock.clientId ], [ replacementNestedBlock.clientId ]: [], } ); - expect( originalNestedState.byClientId.chicken.name ).toBe( + expect( originalNestedState.byClientId.get( 'chicken' ).name ).toBe( 'core/test-block' ); - expect( replacedNestedState.byClientId.chicken.name ).toBe( + expect( replacedNestedState.byClientId.get( 'chicken' ).name ).toBe( 'core/freeform' ); } ); @@ -1013,19 +1081,19 @@ describe( 'state', () => { }, } ); - expect( state.byClientId.chicken ).toEqual( { + expect( state.byClientId.get( 'chicken' ) ).toEqual( { clientId: 'chicken', name: 'core/test-block', isValid: true, } ); - expect( state.attributes.chicken ).toEqual( { + expect( state.attributes.get( 'chicken' ) ).toEqual( { content: 'ribs', } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.chicken + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'chicken' ) ); - expect( state.tree.chicken ).toEqual( { + expect( state.tree.get( 'chicken' ) ).toEqual( { clientId: 'chicken', name: 'core/test-block', innerBlocks: [], @@ -1058,20 +1126,20 @@ describe( 'state', () => { updatedId: 3, } ); - expect( state.byClientId.chicken ).toEqual( { + expect( state.byClientId.get( 'chicken' ) ).toEqual( { clientId: 'chicken', name: 'core/block', isValid: false, } ); - expect( state.attributes.chicken ).toEqual( { + expect( state.attributes.get( 'chicken' ) ).toEqual( { ref: 3, } ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( - state.tree.chicken + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'chicken' ) ); - expect( state.tree.chicken ).toEqual( { + expect( state.tree.get( 'chicken' ) ).toEqual( { clientId: 'chicken', name: 'core/block', isValid: false, @@ -1105,12 +1173,16 @@ describe( 'state', () => { clientIds: [ 'ribs' ], } ); - expect( state.order[ '' ] ).toEqual( [ 'ribs', 'chicken' ] ); - expect( state.tree[ '' ].innerBlocks[ 0 ] ).toBe( state.tree.ribs ); - expect( state.tree[ '' ].innerBlocks[ 1 ] ).toBe( - state.tree.chicken + expect( state.order.get( '' ) ).toEqual( [ 'ribs', 'chicken' ] ); + expect( state.tree.get( '' ).innerBlocks[ 0 ] ).toBe( + state.tree.get( 'ribs' ) + ); + expect( state.tree.get( '' ).innerBlocks[ 1 ] ).toBe( + state.tree.get( 'chicken' ) + ); + expect( state.tree.get( 'chicken' ) ).toBe( + original.tree.get( 'chicken' ) ); - expect( state.tree.chicken ).toBe( original.tree.chicken ); } ); it( 'should move the nested block up', () => { @@ -1130,7 +1202,7 @@ describe( 'state', () => { rootClientId: wrapperBlock.clientId, } ); - expect( state.order ).toEqual( { + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ wrapperBlock.clientId ], [ wrapperBlock.clientId ]: [ movedBlock.clientId, @@ -1140,14 +1212,14 @@ describe( 'state', () => { [ siblingBlock.clientId ]: [], } ); - expect( state.tree[ wrapperBlock.clientId ].innerBlocks[ 0 ] ).toBe( - state.tree[ movedBlock.clientId ] - ); - expect( state.tree[ wrapperBlock.clientId ].innerBlocks[ 1 ] ).toBe( - state.tree[ siblingBlock.clientId ] - ); - expect( state.tree[ movedBlock.clientId ] ).toBe( - original.tree[ movedBlock.clientId ] + expect( + state.tree.get( wrapperBlock.clientId ).innerBlocks[ 0 ] + ).toBe( state.tree.get( movedBlock.clientId ) ); + expect( + state.tree.get( wrapperBlock.clientId ).innerBlocks[ 1 ] + ).toBe( state.tree.get( siblingBlock.clientId ) ); + expect( state.tree.get( movedBlock.clientId ) ).toBe( + original.tree.get( movedBlock.clientId ) ); } ); @@ -1180,7 +1252,7 @@ describe( 'state', () => { clientIds: [ 'ribs', 'veggies' ], } ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.order.get( '' ) ).toEqual( [ 'ribs', 'veggies', 'chicken', @@ -1206,7 +1278,7 @@ describe( 'state', () => { rootClientId: wrapperBlock.clientId, } ); - expect( state.order ).toEqual( { + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ wrapperBlock.clientId ], [ wrapperBlock.clientId ]: [ movedBlockA.clientId, @@ -1268,7 +1340,7 @@ describe( 'state', () => { clientIds: [ 'chicken' ], } ); - expect( state.order[ '' ] ).toEqual( [ 'ribs', 'chicken' ] ); + expect( state.order.get( '' ) ).toEqual( [ 'ribs', 'chicken' ] ); } ); it( 'should move the nested block down', () => { @@ -1288,7 +1360,7 @@ describe( 'state', () => { rootClientId: wrapperBlock.clientId, } ); - expect( state.order ).toEqual( { + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ wrapperBlock.clientId ], [ wrapperBlock.clientId ]: [ siblingBlock.clientId, @@ -1328,7 +1400,7 @@ describe( 'state', () => { clientIds: [ 'chicken', 'ribs' ], } ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.order.get( '' ) ).toEqual( [ 'veggies', 'chicken', 'ribs', @@ -1354,7 +1426,7 @@ describe( 'state', () => { rootClientId: wrapperBlock.clientId, } ); - expect( state.order ).toEqual( { + expect( Object.fromEntries( state.order ) ).toEqual( { '': [ wrapperBlock.clientId ], [ wrapperBlock.clientId ]: [ siblingBlock.clientId, @@ -1416,21 +1488,23 @@ describe( 'state', () => { clientIds: [ 'chicken' ], } ); - expect( state.order[ '' ] ).toEqual( [ 'ribs' ] ); - expect( state.order ).not.toHaveProperty( 'chicken' ); - expect( state.parents ).toEqual( { + expect( state.order.get( '' ) ).toEqual( [ 'ribs' ] ); + expect( Object.fromEntries( state.order ) ).not.toHaveProperty( + 'chicken' + ); + expect( Object.fromEntries( state.parents ) ).toEqual( { ribs: '', } ); - expect( state.byClientId ).toEqual( { + expect( Object.fromEntries( state.byClientId ) ).toEqual( { ribs: { clientId: 'ribs', name: 'core/test-block', }, } ); - expect( state.attributes ).toEqual( { + expect( Object.fromEntries( state.attributes ) ).toEqual( { ribs: {}, } ); - expect( state.tree[ '' ].innerBlocks ).toHaveLength( 1 ); + expect( state.tree.get( '' ).innerBlocks ).toHaveLength( 1 ); } ); it( 'should remove multiple blocks', () => { @@ -1462,19 +1536,23 @@ describe( 'state', () => { clientIds: [ 'chicken', 'veggies' ], } ); - expect( state.order[ '' ] ).toEqual( [ 'ribs' ] ); - expect( state.order ).not.toHaveProperty( 'chicken' ); - expect( state.order ).not.toHaveProperty( 'veggies' ); - expect( state.parents ).toEqual( { + expect( state.order.get( '' ) ).toEqual( [ 'ribs' ] ); + expect( Object.fromEntries( state.order ) ).not.toHaveProperty( + 'chicken' + ); + expect( Object.fromEntries( state.order ) ).not.toHaveProperty( + 'veggies' + ); + expect( Object.fromEntries( state.parents ) ).toEqual( { ribs: '', } ); - expect( state.byClientId ).toEqual( { + expect( Object.fromEntries( state.byClientId ) ).toEqual( { ribs: { clientId: 'ribs', name: 'core/test-block', }, } ); - expect( state.attributes ).toEqual( { + expect( Object.fromEntries( state.attributes ) ).toEqual( { ribs: {}, } ); } ); @@ -1496,11 +1574,11 @@ describe( 'state', () => { clientIds: [ block.clientId ], } ); - expect( state.byClientId ).toEqual( {} ); - expect( state.order ).toEqual( { + expect( state.byClientId ).toEqual( new Map() ); + expect( Object.fromEntries( state.order ) ).toEqual( { '': [], } ); - expect( state.parents ).toEqual( {} ); + expect( Object.fromEntries( state.parents ) ).toEqual( {} ); } ); it( 'should insert at the specified index', () => { @@ -1534,8 +1612,8 @@ describe( 'state', () => { ], } ); - expect( Object.keys( state.byClientId ) ).toHaveLength( 3 ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.byClientId.size ).toBe( 3 ); + expect( state.order.get( '' ) ).toEqual( [ 'kumquat', 'persimmon', 'loquat', @@ -1572,7 +1650,7 @@ describe( 'state', () => { index: 0, } ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.order.get( '' ) ).toEqual( [ 'ribs', 'chicken', 'veggies', @@ -1609,7 +1687,7 @@ describe( 'state', () => { index: 2, } ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.order.get( '' ) ).toEqual( [ 'chicken', 'veggies', 'ribs', @@ -1646,7 +1724,7 @@ describe( 'state', () => { index: 1, } ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.order.get( '' ) ).toEqual( [ 'chicken', 'ribs', 'veggies', @@ -1683,7 +1761,7 @@ describe( 'state', () => { index: 0, } ); - expect( state.order[ '' ] ).toEqual( [ + expect( state.order.get( '' ) ).toEqual( [ 'ribs', 'veggies', 'chicken', @@ -1722,8 +1800,11 @@ describe( 'state', () => { index: 0, } ); - expect( state.order[ '' ] ).toEqual( [ 'chicken' ] ); - expect( state.order.chicken ).toEqual( [ 'ribs', 'veggies' ] ); + expect( state.order.get( '' ) ).toEqual( [ 'chicken' ] ); + expect( state.order.get( 'chicken' ) ).toEqual( [ + 'ribs', + 'veggies', + ] ); } ); describe( 'blocks', () => { @@ -1795,7 +1876,42 @@ describe( 'state', () => { }, } ); - expect( state.attributes.kumquat.updated ).toBe( true ); + expect( state.attributes.get( 'kumquat' ).updated ).toBe( + true + ); + } ); + + it( 'should not updated equal attributes', () => { + const original = deepFreeze( + blocks( undefined, { + type: 'RESET_BLOCKS', + blocks: [ + { + clientId: 'kumquat', + attributes: {}, + innerBlocks: [], + }, + ], + } ) + ); + const state = blocks( original, { + type: 'UPDATE_BLOCK_ATTRIBUTES', + clientIds: [ 'kumquat' ], + attributes: { + updated: true, + }, + } ); + const updatedState = blocks( state, { + type: 'UPDATE_BLOCK_ATTRIBUTES', + clientIds: [ 'kumquat' ], + attributes: { + updated: true, + }, + } ); + + expect( state.attributes.get( 'kumquat' ) ).toBe( + updatedState.attributes.get( 'kumquat' ) + ); } ); it( 'should return with attribute block updates when attributes are unique by block', () => { @@ -1820,7 +1936,9 @@ describe( 'state', () => { uniqueByBlock: true, } ); - expect( state.attributes.kumquat.updated ).toBe( true ); + expect( state.attributes.get( 'kumquat' ).updated ).toBe( + true + ); } ); it( 'should accumulate attribute block updates', () => { @@ -1846,7 +1964,7 @@ describe( 'state', () => { }, } ); - expect( state.attributes.kumquat ).toEqual( { + expect( state.attributes.get( 'kumquat' ) ).toEqual( { updated: true, moreUpdated: true, } ); @@ -1914,7 +2032,7 @@ describe( 'state', () => { clientIds: [ 'kumquat' ], } ); - expect( state.attributes.kumquat ).toEqual( {} ); + expect( state.attributes.get( 'kumquat' ) ).toEqual( {} ); } ); } ); @@ -2111,105 +2229,115 @@ describe( 'state', () => { expect( state.controlledInnerBlocks.chicken ).toBe( true ); // The previous content of the block should be removed expect( state.byClientId.child ).toBeUndefined(); - expect( state.tree.child ).toBeUndefined(); - expect( state.tree.chicken.innerBlocks ).toEqual( [] ); + expect( state.tree.get( 'child' ) ).toBeUndefined(); + expect( state.tree.get( 'chicken' ).innerBlocks ).toEqual( + [] + ); } ); it( 'should preserve the controlled blocks in state and re-attach them in other pieces of state(order, tree, etc..), when we replace inner blocks', () => { const initialState = { - byClientId: { - 'group-id': { - clientId: 'group-id', - name: 'core/group', - isValid: true, - }, - 'reusable-id': { - clientId: 'reusable-id', - name: 'core/block', - isValid: true, - }, - 'paragraph-id': { - clientId: 'paragraph-id', - name: 'core/paragraph', - isValid: true, - }, - }, - order: { - '': [ 'group-id' ], - 'group-id': [ 'reusable-id' ], - 'reusable-id': [ 'paragraph-id' ], - 'paragraph-id': [], - }, + byClientId: new Map( + Object.entries( { + 'group-id': { + clientId: 'group-id', + name: 'core/group', + isValid: true, + }, + 'reusable-id': { + clientId: 'reusable-id', + name: 'core/block', + isValid: true, + }, + 'paragraph-id': { + clientId: 'paragraph-id', + name: 'core/paragraph', + isValid: true, + }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'group-id' ], + 'group-id': [ 'reusable-id' ], + 'reusable-id': [ 'paragraph-id' ], + 'paragraph-id': [], + } ) + ), controlledInnerBlocks: { 'reusable-id': true, }, - parents: { - 'group-id': '', - 'reusable-id': 'group-id', - 'paragraph-id': 'reusable-id', - }, - tree: { - 'group-id': { - clientId: 'group-id', - name: 'core/group', - isValid: true, - innerBlocks: [ - { - clientId: 'reusable-id', - name: 'core/block', - isValid: true, - attributes: { - ref: 687, + parents: new Map( + Object.entries( { + 'group-id': '', + 'reusable-id': 'group-id', + 'paragraph-id': 'reusable-id', + } ) + ), + tree: new Map( + Object.entries( { + 'group-id': { + clientId: 'group-id', + name: 'core/group', + isValid: true, + innerBlocks: [ + { + clientId: 'reusable-id', + name: 'core/block', + isValid: true, + attributes: { + ref: 687, + }, + innerBlocks: [], }, - innerBlocks: [], + ], + }, + 'reusable-id': { + clientId: 'reusable-id', + name: 'core/block', + isValid: true, + attributes: { + ref: 687, }, - ], - }, - 'reusable-id': { - clientId: 'reusable-id', - name: 'core/block', - isValid: true, - attributes: { - ref: 687, + innerBlocks: [], }, - innerBlocks: [], - }, - '': { - innerBlocks: [ - { - clientId: 'group-id', - name: 'core/group', - isValid: true, - innerBlocks: [ - { - clientId: 'reusable-id', - name: 'core/block', - isValid: true, - attributes: { - ref: 687, + '': { + innerBlocks: [ + { + clientId: 'group-id', + name: 'core/group', + isValid: true, + innerBlocks: [ + { + clientId: 'reusable-id', + name: 'core/block', + isValid: true, + attributes: { + ref: 687, + }, + innerBlocks: [], }, - innerBlocks: [], - }, - ], - }, - ], - }, - 'paragraph-id': { - clientId: 'paragraph-id', - name: 'core/paragraph', - isValid: true, - innerBlocks: [], - }, - 'controlled||reusable-id': { - innerBlocks: [ - { - clientId: 'paragraph-id', - name: 'core/paragraph', - isValid: true, - innerBlocks: [], - }, - ], - }, - }, + ], + }, + ], + }, + 'paragraph-id': { + clientId: 'paragraph-id', + name: 'core/paragraph', + isValid: true, + innerBlocks: [], + }, + 'controlled||reusable-id': { + innerBlocks: [ + { + clientId: 'paragraph-id', + name: 'core/paragraph', + isValid: true, + innerBlocks: [], + }, + ], + }, + } ) + ), }; // We will dispatch an action that replaces the inner // blocks with the same inner blocks, which contain @@ -2231,11 +2359,15 @@ describe( 'state', () => { updateSelection: false, }; const state = blocks( initialState, action ); - expect( state.order ).toEqual( - expect.objectContaining( initialState.order ) + expect( Object.fromEntries( state.order ) ).toEqual( + expect.objectContaining( + Object.fromEntries( initialState.order ) + ) ); - expect( state.tree ).toEqual( - expect.objectContaining( initialState.tree ) + expect( Object.fromEntries( state.tree ) ).toEqual( + expect.objectContaining( + Object.fromEntries( initialState.tree ) + ) ); } ); } ); diff --git a/packages/block-editor/src/store/test/selectors.js b/packages/block-editor/src/store/test/selectors.js index 95038fe2f58b8..ee4e9ee4c167a 100644 --- a/packages/block-editor/src/store/test/selectors.js +++ b/packages/block-editor/src/store/test/selectors.js @@ -73,6 +73,7 @@ const { __experimentalGetPatternTransformItems, wasBlockJustInserted, __experimentalGetGlobalBlocksByName, + getLastInsertedBlockClientId, } = selectors; describe( 'selectors', () => { @@ -202,10 +203,10 @@ describe( 'selectors', () => { it( 'returns null if no block by clientId', () => { const state = { blocks: { - byClientId: {}, + byClientId: new Map(), attributes: {}, - order: {}, - parents: {}, + order: new Map(), + parents: new Map(), }, }; @@ -220,22 +221,31 @@ describe( 'selectors', () => { it( 'returns block name', () => { const state = { blocks: { - byClientId: { - 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': { - clientId: 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1', - name: 'core/paragraph', - }, - }, - attributes: { - 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': {}, - }, - order: { - '': [ 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1' ], - 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': [], - }, - parents: { - 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': '', - }, + byClientId: new Map( + Object.entries( { + 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': { + clientId: + 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1', + name: 'core/paragraph', + }, + } ) + ), + attributes: new Map( + Object.entries( { + 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1' ], + 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': [], + } ) + ), + parents: new Map( + Object.entries( { + 'afd1cb17-2c08-4e7a-91be-007ba7ddc3a1': '', + } ) + ), }, }; @@ -252,54 +262,68 @@ describe( 'selectors', () => { it( 'should return the block', () => { const state = { blocks: { - byClientId: { - 123: { clientId: 123, name: 'core/paragraph' }, - }, - attributes: { - 123: {}, - }, - order: { - '': [ 123 ], - 123: [], - }, - parents: { - 123: '', - }, - tree: { - 123: { - clientId: 123, - name: 'core/paragraph', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + 123: { clientId: '123', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 123: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '123' ], + 123: [], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + } ) + ), + tree: new Map( + Object.entries( { + 123: { + clientId: '123', + name: 'core/paragraph', + attributes: {}, + innerBlocks: [], + }, + } ) + ), controlledInnerBlocks: {}, }, }; - expect( getBlock( state, 123 ) ).toBe( state.blocks.tree[ 123 ] ); + expect( getBlock( state, '123' ) ).toBe( + state.blocks.tree.get( '123' ) + ); } ); it( 'should return null if the block is not present in state', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, - tree: { - 123: { - clientId: 123, - name: 'core/paragraph', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), + tree: new Map( + Object.entries( { + 123: { + clientId: '123', + name: 'core/paragraph', + attributes: {}, + innerBlocks: [], + }, + } ) + ), controlledInnerBlocks: {}, }, }; - expect( getBlock( state, 123 ) ).toBe( null ); + expect( getBlock( state, '123' ) ).toBe( null ); } ); } ); @@ -307,47 +331,57 @@ describe( 'selectors', () => { it( 'should return the ordered blocks', () => { const state = { blocks: { - byClientId: { - 23: { clientId: 23, name: 'core/heading' }, - 123: { clientId: 123, name: 'core/paragraph' }, - }, - attributes: { - 23: {}, - 123: {}, - }, - order: { - '': [ 123, 23 ], - }, - parents: { - 123: '', - 23: '', - }, - tree: { - '': { - innerBlocks: [ - { - clientId: 123, - name: 'core/paragraph', - attributes: {}, - innerBlocks: [], - }, - { - clientId: 23, - name: 'core/heading', - attributes: {}, - innerBlocks: [], - }, - ], - }, - 123: {}, - 23: {}, - }, + byClientId: new Map( + Object.entries( { + 23: { clientId: '23', name: 'core/heading' }, + 123: { clientId: '123', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 23: {}, + 123: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 23: '', + } ) + ), + tree: new Map( + Object.entries( { + '': { + innerBlocks: [ + { + clientId: '123', + name: 'core/paragraph', + attributes: {}, + innerBlocks: [], + }, + { + clientId: '23', + name: 'core/heading', + attributes: {}, + innerBlocks: [], + }, + ], + }, + 123: {}, + 23: {}, + } ) + ), controlledInnerBlocks: {}, }, }; expect( getBlocks( state ) ).toBe( - state.blocks.tree[ '' ].innerBlocks + state.blocks.tree.get( '' ).innerBlocks ); } ); } ); @@ -356,30 +390,34 @@ describe( 'selectors', () => { it( 'should return parent blocks', () => { const state = { blocks: { - parents: { - 'client-id-01': '', - 'client-id-02': 'client-id-01', - 'client-id-03': 'client-id-02', - 'client-id-04': 'client-id-03', - }, - byClientId: { - 'client-id-01': { - clientId: 'client-id-01', - name: 'core/columns', - }, - 'client-id-02': { - clientId: 'client-id-02', - name: 'core/navigation', - }, - 'client-id-03': { - clientId: 'client-id-03', - name: 'core/navigation-link', - }, - 'client-id-04': { - clientId: 'client-id-04', - name: 'core/paragraph', - }, - }, + parents: new Map( + Object.entries( { + 'client-id-01': '', + 'client-id-02': 'client-id-01', + 'client-id-03': 'client-id-02', + 'client-id-04': 'client-id-03', + } ) + ), + byClientId: new Map( + Object.entries( { + 'client-id-01': { + clientId: 'client-id-01', + name: 'core/columns', + }, + 'client-id-02': { + clientId: 'client-id-02', + name: 'core/navigation', + }, + 'client-id-03': { + clientId: 'client-id-03', + name: 'core/navigation-link', + }, + 'client-id-04': { + clientId: 'client-id-04', + name: 'core/paragraph', + }, + } ) + ), cache: { 'client-id-01': {}, 'client-id-02': {}, @@ -403,35 +441,39 @@ describe( 'selectors', () => { describe( 'getBlockParentsByBlockName', () => { const state = { blocks: { - parents: { - 'client-id-01': '', - 'client-id-02': 'client-id-01', - 'client-id-03': 'client-id-02', - 'client-id-04': 'client-id-03', - 'client-id-05': 'client-id-04', - }, - byClientId: { - 'client-id-01': { - clientId: 'client-id-01', - name: 'core/navigation', - }, - 'client-id-02': { - clientId: 'client-id-02', - name: 'core/columns', - }, - 'client-id-03': { - clientId: 'client-id-03', - name: 'core/navigation', - }, - 'client-id-04': { - clientId: 'client-id-04', - name: 'core/navigation-link', - }, - 'client-id-05': { - clientId: 'client-id-05', - name: 'core/navigation-link', - }, - }, + parents: new Map( + Object.entries( { + 'client-id-01': '', + 'client-id-02': 'client-id-01', + 'client-id-03': 'client-id-02', + 'client-id-04': 'client-id-03', + 'client-id-05': 'client-id-04', + } ) + ), + byClientId: new Map( + Object.entries( { + 'client-id-01': { + clientId: 'client-id-01', + name: 'core/navigation', + }, + 'client-id-02': { + clientId: 'client-id-02', + name: 'core/columns', + }, + 'client-id-03': { + clientId: 'client-id-03', + name: 'core/navigation', + }, + 'client-id-04': { + clientId: 'client-id-04', + name: 'core/navigation-link', + }, + 'client-id-05': { + clientId: 'client-id-05', + name: 'core/navigation-link', + }, + } ) + ), cache: { 'client-id-01': {}, 'client-id-02': {}, @@ -487,89 +529,124 @@ describe( 'selectors', () => { it( 'should return the ids of any descendants in sequential order, given an array of clientIds', () => { const state = { blocks: { - byClientId: { - 'uuid-2': { clientId: 'uuid-2', name: 'core/image' }, - 'uuid-4': { - clientId: 'uuid-4', - name: 'core/paragraph', - }, - 'uuid-6': { - clientId: 'uuid-6', - name: 'core/paragraph', - }, - 'uuid-8': { clientId: 'uuid-8', name: 'core/block' }, - 'uuid-10': { - clientId: 'uuid-10', - name: 'core/columns', - }, - 'uuid-12': { clientId: 'uuid-12', name: 'core/column' }, - 'uuid-14': { clientId: 'uuid-14', name: 'core/column' }, - 'uuid-16': { clientId: 'uuid-16', name: 'core/quote' }, - 'uuid-18': { clientId: 'uuid-18', name: 'core/block' }, - 'uuid-20': { - clientId: 'uuid-20', - name: 'core/gallery', - }, - 'uuid-22': { clientId: 'uuid-22', name: 'core/block' }, - 'uuid-24': { - clientId: 'uuid-24', - name: 'core/columns', - }, - 'uuid-26': { clientId: 'uuid-26', name: 'core/column' }, - 'uuid-28': { clientId: 'uuid-28', name: 'core/column' }, - 'uuid-30': { - clientId: 'uuid-30', - name: 'core/paragraph', - }, - }, - attributes: { - 'uuid-2': {}, - 'uuid-4': {}, - 'uuid-6': {}, - 'uuid-8': {}, - 'uuid-10': {}, - 'uuid-12': {}, - 'uuid-14': {}, - 'uuid-16': {}, - 'uuid-18': {}, - 'uuid-20': {}, - 'uuid-22': {}, - 'uuid-24': {}, - 'uuid-26': {}, - 'uuid-28': {}, - 'uuid-30': {}, - }, - order: { - '': [ 'uuid-6', 'uuid-8', 'uuid-10', 'uuid-22' ], - 'uuid-2': [], - 'uuid-4': [], - 'uuid-6': [], - 'uuid-8': [], - 'uuid-10': [ 'uuid-12', 'uuid-14' ], - 'uuid-12': [ 'uuid-16' ], - 'uuid-14': [ 'uuid-18' ], - 'uuid-16': [], - 'uuid-18': [ 'uuid-24' ], - 'uuid-20': [], - 'uuid-22': [], - 'uuid-24': [ 'uuid-26', 'uuid-28' ], - 'uuid-26': [], - 'uuid-28': [ 'uuid-30' ], - }, - parents: { - 'uuid-6': '', - 'uuid-8': '', - 'uuid-10': '', - 'uuid-22': '', - 'uuid-12': 'uuid-10', - 'uuid-14': 'uuid-10', - 'uuid-16': 'uuid-12', - 'uuid-18': 'uuid-14', - 'uuid-24': 'uuid-18', - 'uuid-26': 'uuid-24', - 'uuid-28': 'uuid-24', - 'uuid-30': 'uuid-28', - }, + byClientId: new Map( + Object.entries( { + 'uuid-2': { + clientId: 'uuid-2', + name: 'core/image', + }, + 'uuid-4': { + clientId: 'uuid-4', + name: 'core/paragraph', + }, + 'uuid-6': { + clientId: 'uuid-6', + name: 'core/paragraph', + }, + 'uuid-8': { + clientId: 'uuid-8', + name: 'core/block', + }, + 'uuid-10': { + clientId: 'uuid-10', + name: 'core/columns', + }, + 'uuid-12': { + clientId: 'uuid-12', + name: 'core/column', + }, + 'uuid-14': { + clientId: 'uuid-14', + name: 'core/column', + }, + 'uuid-16': { + clientId: 'uuid-16', + name: 'core/quote', + }, + 'uuid-18': { + clientId: 'uuid-18', + name: 'core/block', + }, + 'uuid-20': { + clientId: 'uuid-20', + name: 'core/gallery', + }, + 'uuid-22': { + clientId: 'uuid-22', + name: 'core/block', + }, + 'uuid-24': { + clientId: 'uuid-24', + name: 'core/columns', + }, + 'uuid-26': { + clientId: 'uuid-26', + name: 'core/column', + }, + 'uuid-28': { + clientId: 'uuid-28', + name: 'core/column', + }, + 'uuid-30': { + clientId: 'uuid-30', + name: 'core/paragraph', + }, + } ) + ), + attributes: new Map( + Object.entries( { + 'uuid-2': {}, + 'uuid-4': {}, + 'uuid-6': {}, + 'uuid-8': {}, + 'uuid-10': {}, + 'uuid-12': {}, + 'uuid-14': {}, + 'uuid-16': {}, + 'uuid-18': {}, + 'uuid-20': {}, + 'uuid-22': {}, + 'uuid-24': {}, + 'uuid-26': {}, + 'uuid-28': {}, + 'uuid-30': {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'uuid-6', 'uuid-8', 'uuid-10', 'uuid-22' ], + 'uuid-2': [], + 'uuid-4': [], + 'uuid-6': [], + 'uuid-8': [], + 'uuid-10': [ 'uuid-12', 'uuid-14' ], + 'uuid-12': [ 'uuid-16' ], + 'uuid-14': [ 'uuid-18' ], + 'uuid-16': [], + 'uuid-18': [ 'uuid-24' ], + 'uuid-20': [], + 'uuid-22': [], + 'uuid-24': [ 'uuid-26', 'uuid-28' ], + 'uuid-26': [], + 'uuid-28': [ 'uuid-30' ], + } ) + ), + parents: new Map( + Object.entries( { + 'uuid-6': '', + 'uuid-8': '', + 'uuid-10': '', + 'uuid-22': '', + 'uuid-12': 'uuid-10', + 'uuid-14': 'uuid-10', + 'uuid-16': 'uuid-12', + 'uuid-18': 'uuid-14', + 'uuid-24': 'uuid-18', + 'uuid-26': 'uuid-24', + 'uuid-28': 'uuid-24', + 'uuid-30': 'uuid-28', + } ) + ), controlledInnerBlocks: {}, }, }; @@ -592,89 +669,124 @@ describe( 'selectors', () => { it( 'should return the ids for top-level blocks and their descendants of any depth (for nested blocks) in sequential order.', () => { const state = { blocks: { - byClientId: { - 'uuid-2': { clientId: 'uuid-2', name: 'core/image' }, - 'uuid-4': { - clientId: 'uuid-4', - name: 'core/paragraph', - }, - 'uuid-6': { - clientId: 'uuid-6', - name: 'core/paragraph', - }, - 'uuid-8': { clientId: 'uuid-8', name: 'core/block' }, - 'uuid-10': { - clientId: 'uuid-10', - name: 'core/columns', - }, - 'uuid-12': { clientId: 'uuid-12', name: 'core/column' }, - 'uuid-14': { clientId: 'uuid-14', name: 'core/column' }, - 'uuid-16': { clientId: 'uuid-16', name: 'core/quote' }, - 'uuid-18': { clientId: 'uuid-18', name: 'core/block' }, - 'uuid-20': { - clientId: 'uuid-20', - name: 'core/gallery', - }, - 'uuid-22': { clientId: 'uuid-22', name: 'core/block' }, - 'uuid-24': { - clientId: 'uuid-24', - name: 'core/columns', - }, - 'uuid-26': { clientId: 'uuid-26', name: 'core/column' }, - 'uuid-28': { clientId: 'uuid-28', name: 'core/column' }, - 'uuid-30': { - clientId: 'uuid-30', - name: 'core/paragraph', - }, - }, - attributes: { - 'uuid-2': {}, - 'uuid-4': {}, - 'uuid-6': {}, - 'uuid-8': {}, - 'uuid-10': {}, - 'uuid-12': {}, - 'uuid-14': {}, - 'uuid-16': {}, - 'uuid-18': {}, - 'uuid-20': {}, - 'uuid-22': {}, - 'uuid-24': {}, - 'uuid-26': {}, - 'uuid-28': {}, - 'uuid-30': {}, - }, - order: { - '': [ 'uuid-6', 'uuid-8', 'uuid-10', 'uuid-22' ], - 'uuid-2': [], - 'uuid-4': [], - 'uuid-6': [], - 'uuid-8': [], - 'uuid-10': [ 'uuid-12', 'uuid-14' ], - 'uuid-12': [ 'uuid-16' ], - 'uuid-14': [ 'uuid-18' ], - 'uuid-16': [], - 'uuid-18': [ 'uuid-24' ], - 'uuid-20': [], - 'uuid-22': [], - 'uuid-24': [ 'uuid-26', 'uuid-28' ], - 'uuid-26': [], - 'uuid-28': [ 'uuid-30' ], - }, - parents: { - 'uuid-6': '', - 'uuid-8': '', - 'uuid-10': '', - 'uuid-22': '', - 'uuid-12': 'uuid-10', - 'uuid-14': 'uuid-10', - 'uuid-16': 'uuid-12', - 'uuid-18': 'uuid-14', - 'uuid-24': 'uuid-18', - 'uuid-26': 'uuid-24', - 'uuid-28': 'uuid-24', - 'uuid-30': 'uuid-28', - }, + byClientId: new Map( + Object.entries( { + 'uuid-2': { + clientId: 'uuid-2', + name: 'core/image', + }, + 'uuid-4': { + clientId: 'uuid-4', + name: 'core/paragraph', + }, + 'uuid-6': { + clientId: 'uuid-6', + name: 'core/paragraph', + }, + 'uuid-8': { + clientId: 'uuid-8', + name: 'core/block', + }, + 'uuid-10': { + clientId: 'uuid-10', + name: 'core/columns', + }, + 'uuid-12': { + clientId: 'uuid-12', + name: 'core/column', + }, + 'uuid-14': { + clientId: 'uuid-14', + name: 'core/column', + }, + 'uuid-16': { + clientId: 'uuid-16', + name: 'core/quote', + }, + 'uuid-18': { + clientId: 'uuid-18', + name: 'core/block', + }, + 'uuid-20': { + clientId: 'uuid-20', + name: 'core/gallery', + }, + 'uuid-22': { + clientId: 'uuid-22', + name: 'core/block', + }, + 'uuid-24': { + clientId: 'uuid-24', + name: 'core/columns', + }, + 'uuid-26': { + clientId: 'uuid-26', + name: 'core/column', + }, + 'uuid-28': { + clientId: 'uuid-28', + name: 'core/column', + }, + 'uuid-30': { + clientId: 'uuid-30', + name: 'core/paragraph', + }, + } ) + ), + attributes: new Map( + Object.entries( { + 'uuid-2': {}, + 'uuid-4': {}, + 'uuid-6': {}, + 'uuid-8': {}, + 'uuid-10': {}, + 'uuid-12': {}, + 'uuid-14': {}, + 'uuid-16': {}, + 'uuid-18': {}, + 'uuid-20': {}, + 'uuid-22': {}, + 'uuid-24': {}, + 'uuid-26': {}, + 'uuid-28': {}, + 'uuid-30': {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'uuid-6', 'uuid-8', 'uuid-10', 'uuid-22' ], + 'uuid-2': [], + 'uuid-4': [], + 'uuid-6': [], + 'uuid-8': [], + 'uuid-10': [ 'uuid-12', 'uuid-14' ], + 'uuid-12': [ 'uuid-16' ], + 'uuid-14': [ 'uuid-18' ], + 'uuid-16': [], + 'uuid-18': [ 'uuid-24' ], + 'uuid-20': [], + 'uuid-22': [], + 'uuid-24': [ 'uuid-26', 'uuid-28' ], + 'uuid-26': [], + 'uuid-28': [ 'uuid-30' ], + } ) + ), + parents: new Map( + Object.entries( { + 'uuid-6': '', + 'uuid-8': '', + 'uuid-10': '', + 'uuid-22': '', + 'uuid-12': 'uuid-10', + 'uuid-14': 'uuid-10', + 'uuid-16': 'uuid-12', + 'uuid-18': 'uuid-14', + 'uuid-24': 'uuid-18', + 'uuid-26': 'uuid-24', + 'uuid-28': 'uuid-24', + 'uuid-30': 'uuid-28', + } ) + ), }, }; expect( getClientIdsWithDescendants( state ) ).toEqual( [ @@ -698,17 +810,23 @@ describe( 'selectors', () => { it( 'should return the number of top-level blocks in the post', () => { const state = { blocks: { - byClientId: { - 23: { clientId: 23, name: 'core/heading' }, - 123: { clientId: 123, name: 'core/paragraph' }, - }, - attributes: { - 23: {}, - 123: {}, - }, - order: { - '': [ 123, 23 ], - }, + byClientId: new Map( + Object.entries( { + 23: { clientId: '23', name: 'core/heading' }, + 123: { clientId: '123', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 23: {}, + 123: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), }, }; @@ -718,25 +836,33 @@ describe( 'selectors', () => { it( 'should return the number of blocks in a nested context', () => { const state = { blocks: { - byClientId: { - 123: { clientId: 123, name: 'core/columns' }, - 456: { clientId: 456, name: 'core/paragraph' }, - 789: { clientId: 789, name: 'core/paragraph' }, - }, - attributes: { - 123: {}, - 456: {}, - 789: {}, - }, - order: { - '': [ 123 ], - 123: [ 456, 789 ], - }, - parents: { - 123: '', - 456: 123, - 789: 123, - }, + byClientId: new Map( + Object.entries( { + 123: { clientId: '123', name: 'core/columns' }, + 456: { clientId: '456', name: 'core/paragraph' }, + 789: { clientId: '789', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 123: {}, + 456: {}, + 789: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '123' ], + 123: [ '456', '789' ], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 456: '123', + 789: '123', + } ) + ), }, }; @@ -790,23 +916,31 @@ describe( 'selectors', () => { describe( 'getGlobalBlockCount', () => { const state = { blocks: { - byClientId: { - 123: { clientId: 123, name: 'core/heading' }, - 456: { clientId: 456, name: 'core/paragraph' }, - 789: { clientId: 789, name: 'core/paragraph' }, - }, - attributes: { - 123: {}, - 456: {}, - 789: {}, - }, - order: { - '': [ 123, 456 ], - }, - parents: { - 123: '', - 456: '', - }, + byClientId: new Map( + Object.entries( { + 123: { clientId: '123', name: 'core/heading' }, + 456: { clientId: '456', name: 'core/paragraph' }, + 789: { clientId: '789', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 123: {}, + 456: {}, + 789: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '123', '456' ], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 456: '', + } ) + ), }, }; @@ -821,10 +955,10 @@ describe( 'selectors', () => { it( 'should return 0 if no blocks exist', () => { const emptyState = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), }, }; expect( getGlobalBlockCount( emptyState ) ).toBe( 0 ); @@ -837,46 +971,54 @@ describe( 'selectors', () => { describe( '__experimentalGetGlobalBlocksByName', () => { const state = { blocks: { - byClientId: { - 123: { clientId: 123, name: 'core/heading' }, - 456: { clientId: 456, name: 'core/paragraph' }, - 789: { clientId: 789, name: 'core/paragraph' }, - 1011: { clientId: 1011, name: 'core/group' }, - 1213: { clientId: 1213, name: 'core/paragraph' }, - 1415: { clientId: 1213, name: 'core/paragraph' }, - }, - attributes: { - 123: {}, - 456: {}, - 789: {}, - 1011: {}, - 1213: {}, - 1415: {}, - }, - order: { - '': [ 123, 456, 1011 ], - 1011: [ 1415, 1213 ], - }, - parents: { - 123: '', - 456: '', - 1011: '', - 1213: 1011, - 1415: 1011, - }, + byClientId: new Map( + Object.entries( { + 123: { clientId: '123', name: 'core/heading' }, + 456: { clientId: '456', name: 'core/paragraph' }, + 789: { clientId: '789', name: 'core/paragraph' }, + 1011: { clientId: '1011', name: 'core/group' }, + 1213: { clientId: '1213', name: 'core/paragraph' }, + 1415: { clientId: '1213', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 123: {}, + 456: {}, + 789: {}, + 1011: {}, + 1213: {}, + 1415: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '123', '456', '1011' ], + 1011: [ '1415', '1213' ], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 456: '', + 1011: '', + 1213: '1011', + 1415: '1011', + } ) + ), }, }; it( 'should return the clientIds of blocks of a given type', () => { expect( __experimentalGetGlobalBlocksByName( state, 'core/heading' ) - ).toStrictEqual( [ 123 ] ); + ).toStrictEqual( [ '123' ] ); } ); it( 'should return the clientIds of blocks of a given type even if blocks are nested', () => { expect( __experimentalGetGlobalBlocksByName( state, 'core/paragraph' ) - ).toStrictEqual( [ 456, 1415, 1213 ] ); + ).toStrictEqual( [ '456', '1415', '1213' ] ); } ); it( 'Should return empty array if no blocks match. The empty array should be the same reference', () => { @@ -911,8 +1053,8 @@ describe( 'selectors', () => { it( 'should return null if there is multi selection', () => { const state = { selection: { - selectionStart: { clientId: 23 }, - selectionEnd: { clientId: 123 }, + selectionStart: { clientId: '23' }, + selectionEnd: { clientId: '123' }, }, }; @@ -922,19 +1064,21 @@ describe( 'selectors', () => { it( 'should return the selected block ClientId', () => { const state = { blocks: { - byClientId: { - 23: { - name: 'fake block', - }, - }, + byClientId: new Map( + Object.entries( { + 23: { + name: 'fake block', + }, + } ) + ), }, selection: { - selectionStart: { clientId: 23 }, - selectionEnd: { clientId: 23 }, + selectionStart: { clientId: '23' }, + selectionEnd: { clientId: '23' }, }, }; - expect( getSelectedBlockClientId( state ) ).toEqual( 23 ); + expect( getSelectedBlockClientId( state ) ).toEqual( '23' ); } ); } ); @@ -942,31 +1086,41 @@ describe( 'selectors', () => { it( 'should return null if no block is selected', () => { const state = { blocks: { - byClientId: { - 23: { clientId: 23, name: 'core/heading' }, - 123: { clientId: 123, name: 'core/paragraph' }, - }, - attributes: { - 23: {}, - 123: {}, - }, - order: { - '': [ 23, 123 ], - 23: [], - 123: [], - }, - parents: { - 23: '', - 123: '', - }, - tree: { - 23: { - clientId: 23, - name: 'core/heading', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + 23: { clientId: '23', name: 'core/heading' }, + 123: { clientId: '123', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 23: {}, + 123: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '23', '123' ], + 23: [], + 123: [], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), + tree: new Map( + Object.entries( { + 23: { + clientId: '23', + name: 'core/heading', + attributes: {}, + innerBlocks: [], + }, + } ) + ), }, selection: { selectionStart: {}, @@ -980,35 +1134,45 @@ describe( 'selectors', () => { it( 'should return null if there is multi selection', () => { const state = { blocks: { - byClientId: { - 23: { clientId: 23, name: 'core/heading' }, - 123: { clientId: 123, name: 'core/paragraph' }, - }, - attributes: { - 23: {}, - 123: {}, - }, - order: { - '': [ 23, 123 ], - 23: [], - 123: [], - }, - parents: { - 123: '', - 23: '', - }, - tree: { - 23: { - clientId: 23, - name: 'core/heading', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + 23: { clientId: '23', name: 'core/heading' }, + 123: { clientId: '123', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 23: {}, + 123: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '23', '123' ], + 23: [], + 123: [], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 23: '', + } ) + ), + tree: new Map( + Object.entries( { + 23: { + clientId: '23', + name: 'core/heading', + attributes: {}, + innerBlocks: [], + }, + } ) + ), }, selection: { - selectionStart: { clientId: 23 }, - selectionEnd: { clientId: 123 }, + selectionStart: { clientId: '23' }, + selectionEnd: { clientId: '123' }, }, }; @@ -1018,41 +1182,51 @@ describe( 'selectors', () => { it( 'should return the selected block', () => { const state = { blocks: { - byClientId: { - 23: { clientId: 23, name: 'core/heading' }, - 123: { clientId: 123, name: 'core/paragraph' }, - }, - attributes: { - 23: {}, - 123: {}, - }, - order: { - '': [ 23, 123 ], - 23: [], - 123: [], - }, - parents: { - 123: '', - 23: '', - }, - tree: { - 23: { - clientId: 23, - name: 'core/heading', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + 23: { clientId: '23', name: 'core/heading' }, + 123: { clientId: '123', name: 'core/paragraph' }, + } ) + ), + attributes: new Map( + Object.entries( { + 23: {}, + 123: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ '23', '123' ], + 23: [], + 123: [], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 23: '', + } ) + ), + tree: new Map( + Object.entries( { + 23: { + clientId: '23', + name: 'core/heading', + attributes: {}, + innerBlocks: [], + }, + } ) + ), controlledInnerBlocks: {}, }, selection: { - selectionStart: { clientId: 23 }, - selectionEnd: { clientId: 23 }, + selectionStart: { clientId: '23' }, + selectionEnd: { clientId: '23' }, }, }; expect( getSelectedBlock( state ) ).toEqual( - getBlock( state, 23 ) + getBlock( state, '23' ) ); } ); } ); @@ -1061,8 +1235,8 @@ describe( 'selectors', () => { it( 'should return null if the block does not exist', () => { const state = { blocks: { - order: {}, - parents: {}, + order: new Map(), + parents: new Map(), }, }; @@ -1072,20 +1246,24 @@ describe( 'selectors', () => { it( 'should return root ClientId relative the block ClientId', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456, 56 ], - }, - parents: { - 123: '', - 23: '', - 456: 123, - 56: 123, - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456', '56' ], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 23: '', + 456: '123', + 56: '123', + } ) + ), }, }; - expect( getBlockRootClientId( state, 56 ) ).toBe( 123 ); + expect( getBlockRootClientId( state, '56' ) ).toBe( '123' ); } ); } ); @@ -1093,8 +1271,8 @@ describe( 'selectors', () => { it( 'should return the given block if the block has no parents', () => { const state = { blocks: { - order: {}, - parents: {}, + order: new Map(), + parents: new Map(), }, }; @@ -1104,16 +1282,20 @@ describe( 'selectors', () => { it( 'should return root ClientId relative the block ClientId', () => { const state = { blocks: { - order: { - '': [ 'a', 'b' ], - a: [ 'c', 'd' ], - }, - parents: { - a: '', - b: '', - c: 'a', - d: 'a', - }, + order: new Map( + Object.entries( { + '': [ 'a', 'b' ], + a: [ 'c', 'd' ], + } ) + ), + parents: new Map( + Object.entries( { + a: '', + b: '', + c: 'a', + d: 'a', + } ) + ), }, }; @@ -1123,18 +1305,22 @@ describe( 'selectors', () => { it( 'should return the top level root ClientId relative the block ClientId', () => { const state = { blocks: { - order: { - '': [ 'a', 'b' ], - a: [ 'c', 'd' ], - d: [ 'e' ], - }, - parents: { - a: '', - b: '', - c: 'a', - d: 'a', - e: 'd', - }, + order: new Map( + Object.entries( { + '': [ 'a', 'b' ], + a: [ 'c', 'd' ], + d: [ 'e' ], + } ) + ), + parents: new Map( + Object.entries( { + a: '', + b: '', + c: 'a', + d: 'a', + e: 'd', + } ) + ), }, }; @@ -1146,13 +1332,17 @@ describe( 'selectors', () => { it( 'should return empty if there is no selection', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 123: '', - 23: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 123: '', + 23: '', + } ) + ), }, selection: { selectionStart: {}, @@ -1166,75 +1356,95 @@ describe( 'selectors', () => { it( 'should return the selected block clientId if there is a selection', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 2 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '2' }, }, }; - expect( getSelectedBlockClientIds( state ) ).toEqual( [ 2 ] ); + expect( getSelectedBlockClientIds( state ) ).toEqual( [ '2' ] ); } ); it( 'should return selected block clientIds if there is multi selection', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; - expect( getSelectedBlockClientIds( state ) ).toEqual( [ 4, 3, 2 ] ); + expect( getSelectedBlockClientIds( state ) ).toEqual( [ + '4', + '3', + '2', + ] ); } ); it( 'should return selected block clientIds if there is multi selection (nested context)', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - 4: [ 9, 8, 7, 6 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - 6: 4, - 7: 4, - 8: 4, - 9: 4, - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + 4: [ '9', '8', '7', '6' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + 6: '4', + 7: '4', + 8: '4', + 9: '4', + } ) + ), }, selection: { - selectionStart: { clientId: 7 }, - selectionEnd: { clientId: 9 }, + selectionStart: { clientId: '7' }, + selectionEnd: { clientId: '9' }, }, }; - expect( getSelectedBlockClientIds( state ) ).toEqual( [ 9, 8, 7 ] ); + expect( getSelectedBlockClientIds( state ) ).toEqual( [ + '9', + '8', + '7', + ] ); } ); } ); @@ -1242,13 +1452,17 @@ describe( 'selectors', () => { it( 'should return empty if there is no multi selection', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, selection: { selectionStart: {}, @@ -1262,55 +1476,67 @@ describe( 'selectors', () => { it( 'should return selected block clientIds if there is multi selection', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; expect( getMultiSelectedBlockClientIds( state ) ).toEqual( [ - 4, 3, 2, + '4', + '3', + '2', ] ); } ); it( 'should return selected block clientIds if there is multi selection (nested context)', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - 4: [ 9, 8, 7, 6 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - 6: 4, - 7: 4, - 8: 4, - 9: 4, - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + 4: [ '9', '8', '7', '6' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + 6: '4', + 7: '4', + 8: '4', + 9: '4', + } ) + ), }, selection: { - selectionStart: { clientId: 7 }, - selectionEnd: { clientId: 9 }, + selectionStart: { clientId: '7' }, + selectionEnd: { clientId: '9' }, }, }; expect( getMultiSelectedBlockClientIds( state ) ).toEqual( [ - 9, 8, 7, + '9', + '8', + '7', ] ); } ); } ); @@ -1319,10 +1545,10 @@ describe( 'selectors', () => { it( 'should return the same reference on subsequent invocations of empty selection', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), }, selection: { selectionStart: {}, @@ -1351,12 +1577,12 @@ describe( 'selectors', () => { it( 'returns multi selection start', () => { const state = { selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; - expect( getMultiSelectedBlocksStartClientId( state ) ).toBe( 2 ); + expect( getMultiSelectedBlocksStartClientId( state ) ).toBe( '2' ); } ); } ); @@ -1375,12 +1601,12 @@ describe( 'selectors', () => { it( 'returns multi selection end', () => { const state = { selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; - expect( getMultiSelectedBlocksEndClientId( state ) ).toBe( 4 ); + expect( getMultiSelectedBlocksEndClientId( state ) ).toBe( '4' ); } ); } ); @@ -1388,35 +1614,43 @@ describe( 'selectors', () => { it( 'should return the ordered block ClientIds of top-level blocks by default', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, }; - expect( getBlockOrder( state ) ).toEqual( [ 123, 23 ] ); + expect( getBlockOrder( state ) ).toEqual( [ '123', '23' ] ); } ); it( 'should return the ordered block ClientIds at a specified rootClientId', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456 ], - }, - parents: { - 23: '', - 123: '', - 456: 123, - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + 456: '123', + } ) + ), }, }; - expect( getBlockOrder( state, '123' ) ).toEqual( [ 456 ] ); + expect( getBlockOrder( state, '123' ) ).toEqual( [ '456' ] ); } ); } ); @@ -1424,36 +1658,44 @@ describe( 'selectors', () => { it( 'should return the block order', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, }; - expect( getBlockIndex( state, 23 ) ).toBe( 1 ); + expect( getBlockIndex( state, '23' ) ).toBe( 1 ); } ); it( 'should return the block order (nested context)', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456, 56 ], - }, - parents: { - 23: '', - 123: '', - 56: 123, - 456: 123, - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456', '56' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + 56: '123', + 456: '123', + } ) + ), }, }; - expect( getBlockIndex( state, 56 ) ).toBe( 1 ); + expect( getBlockIndex( state, '56' ) ).toBe( 1 ); } ); } ); @@ -1461,73 +1703,91 @@ describe( 'selectors', () => { it( 'should return the previous block', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, }; - expect( getPreviousBlockClientId( state, 23 ) ).toEqual( 123 ); + expect( getPreviousBlockClientId( state, '23' ) ).toEqual( '123' ); } ); it( 'should return the previous block (nested context)', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456, 56 ], - }, - parents: { - 23: '', - 123: '', - 456: 123, - 56: 123, - }, - }, - }; - - expect( getPreviousBlockClientId( state, 56, '123' ) ).toEqual( - 456 + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456', '56' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + 456: '123', + 56: '123', + } ) + ), + }, + }; + + expect( getPreviousBlockClientId( state, '56', '123' ) ).toEqual( + '456' ); } ); it( 'should return null for the first block', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, }; - expect( getPreviousBlockClientId( state, 123 ) ).toBeNull(); + expect( getPreviousBlockClientId( state, '123' ) ).toBeNull(); } ); it( 'should return null for the first block (nested context)', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456, 56 ], - }, - parents: { - 23: '', - 123: '', - 456: 123, - 56: 123, - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456', '56' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + 456: '123', + 56: '123', + } ) + ), }, }; - expect( getPreviousBlockClientId( state, 456, '123' ) ).toBeNull(); + expect( + getPreviousBlockClientId( state, '456', '123' ) + ).toBeNull(); } ); } ); @@ -1535,71 +1795,89 @@ describe( 'selectors', () => { it( 'should return the following block', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, }; - expect( getNextBlockClientId( state, 123 ) ).toEqual( 23 ); + expect( getNextBlockClientId( state, '123' ) ).toEqual( '23' ); } ); it( 'should return the following block (nested context)', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456, 56 ], - }, - parents: { - 23: '', - 123: '', - 456: 123, - 56: 123, - }, - }, - }; - - expect( getNextBlockClientId( state, 456, '123' ) ).toEqual( 56 ); + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456', '56' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + 456: '123', + 56: '123', + } ) + ), + }, + }; + + expect( getNextBlockClientId( state, '456', '123' ) ).toEqual( + '56' + ); } ); it( 'should return null for the last block', () => { const state = { - blocks: { - order: { - '': [ 123, 23 ], - }, - parents: { - 23: '', - 123: '', - }, + blocks: { + order: new Map( + Object.entries( { + '': [ '123', '23' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + } ) + ), }, }; - expect( getNextBlockClientId( state, 23 ) ).toBeNull(); + expect( getNextBlockClientId( state, '23' ) ).toBeNull(); } ); it( 'should return null for the last block (nested context)', () => { const state = { blocks: { - order: { - '': [ 123, 23 ], - 123: [ 456, 56 ], - }, - parents: { - 23: '', - 123: '', - 456: 123, - 56: 123, - }, + order: new Map( + Object.entries( { + '': [ '123', '23' ], + 123: [ '456', '56' ], + } ) + ), + parents: new Map( + Object.entries( { + 23: '', + 123: '', + 456: '123', + 56: '123', + } ) + ), }, }; - expect( getNextBlockClientId( state, 56, '123' ) ).toBeNull(); + expect( getNextBlockClientId( state, '56', '123' ) ).toBeNull(); } ); } ); @@ -1607,23 +1885,23 @@ describe( 'selectors', () => { it( 'should return true if the block is selected', () => { const state = { selection: { - selectionStart: { clientId: 123 }, - selectionEnd: { clientId: 123 }, + selectionStart: { clientId: '123' }, + selectionEnd: { clientId: '123' }, }, }; - expect( isBlockSelected( state, 123 ) ).toBe( true ); + expect( isBlockSelected( state, '123' ) ).toBe( true ); } ); it( 'should return false if a multi-selection range exists', () => { const state = { selection: { - selectionStart: { clientId: 123 }, - selectionEnd: { clientId: 124 }, + selectionStart: { clientId: '123' }, + selectionEnd: { clientId: '124' }, }, }; - expect( isBlockSelected( state, 123 ) ).toBe( false ); + expect( isBlockSelected( state, '123' ) ).toBe( false ); } ); it( 'should return false if the block is not selected', () => { @@ -1634,7 +1912,7 @@ describe( 'selectors', () => { }, }; - expect( isBlockSelected( state, 23 ) ).toBe( false ); + expect( isBlockSelected( state, '23' ) ).toBe( false ); } ); } ); @@ -1642,87 +1920,103 @@ describe( 'selectors', () => { it( 'should return false if the selected block is a child of the given ClientId', () => { const state = { selection: { - selectionStart: { clientId: 5 }, - selectionEnd: { clientId: 5 }, + selectionStart: { clientId: '5' }, + selectionEnd: { clientId: '5' }, }, blocks: { - order: { - 4: [ 3, 2, 1 ], - }, - parents: { - 1: 4, - 2: 4, - 3: 4, - }, + order: new Map( + Object.entries( { + 4: [ '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '4', + 2: '4', + 3: '4', + } ) + ), }, }; - expect( hasSelectedInnerBlock( state, 4 ) ).toBe( false ); + expect( hasSelectedInnerBlock( state, '4' ) ).toBe( false ); } ); it( 'should return true if the selected block is a child of the given ClientId', () => { const state = { selection: { - selectionStart: { clientId: 3 }, - selectionEnd: { clientId: 3 }, + selectionStart: { clientId: '3' }, + selectionEnd: { clientId: '3' }, }, blocks: { - order: { - 4: [ 3, 2, 1 ], - }, - parents: { - 1: 4, - 2: 4, - 3: 4, - }, + order: new Map( + Object.entries( { + 4: [ '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '4', + 2: '4', + 3: '4', + } ) + ), }, }; - expect( hasSelectedInnerBlock( state, 4 ) ).toBe( true ); + expect( hasSelectedInnerBlock( state, '4' ) ).toBe( true ); } ); it( 'should return true if a multi selection exists that contains children of the block with the given ClientId', () => { const state = { blocks: { - order: { - 6: [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: 6, - 2: 6, - 3: 6, - 4: 6, - 5: 6, - }, + order: new Map( + Object.entries( { + 6: [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '6', + 2: '6', + 3: '6', + 4: '6', + 5: '6', + } ) + ), }, selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; - expect( hasSelectedInnerBlock( state, 6 ) ).toBe( true ); + expect( hasSelectedInnerBlock( state, '6' ) ).toBe( true ); } ); it( 'should return false if a multi selection exists bot does not contains children of the block with the given ClientId', () => { const state = { blocks: { - order: { - 3: [ 2, 1 ], - 6: [ 5, 4 ], - }, - parents: { - 1: 3, - 2: 3, - 4: 6, - 5: 6, - }, + order: new Map( + Object.entries( { + 3: [ '2', '1' ], + 6: [ '5', '4' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '3', + 2: '3', + 4: '6', + 5: '6', + } ) + ), }, selection: { - selectionStart: { clientId: 5 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '5' }, + selectionEnd: { clientId: '4' }, }, }; - expect( hasSelectedInnerBlock( state, 3 ) ).toBe( false ); + expect( hasSelectedInnerBlock( state, '3' ) ).toBe( false ); } ); } ); @@ -1730,70 +2024,82 @@ describe( 'selectors', () => { it( 'should return true if the block is selected but not the last', () => { const state = { selection: { - selectionStart: { clientId: 5 }, - selectionEnd: { clientId: 3 }, + selectionStart: { clientId: '5' }, + selectionEnd: { clientId: '3' }, }, blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, }; - expect( isBlockWithinSelection( state, 4 ) ).toBe( true ); + expect( isBlockWithinSelection( state, '4' ) ).toBe( true ); } ); it( 'should return false if the block is the last selected', () => { const state = { selection: { - selectionStart: { clientId: 5 }, - selectionEnd: { clientId: 3 }, + selectionStart: { clientId: '5' }, + selectionEnd: { clientId: '3' }, }, blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, }; - expect( isBlockWithinSelection( state, 3 ) ).toBe( false ); + expect( isBlockWithinSelection( state, '3' ) ).toBe( false ); } ); it( 'should return false if the block is not selected', () => { const state = { selection: { - selectionStart: { clientId: 5 }, - selectionEnd: { clientId: 3 }, + selectionStart: { clientId: '5' }, + selectionEnd: { clientId: '3' }, }, blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, }; - expect( isBlockWithinSelection( state, 2 ) ).toBe( false ); + expect( isBlockWithinSelection( state, '2' ) ).toBe( false ); } ); it( 'should return false if there is no selection', () => { @@ -1803,20 +2109,24 @@ describe( 'selectors', () => { selectionEnd: {}, }, blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, }; - expect( isBlockWithinSelection( state, 4 ) ).toBe( false ); + expect( isBlockWithinSelection( state, '4' ) ).toBe( false ); } ); } ); @@ -1866,58 +2176,66 @@ describe( 'selectors', () => { describe( 'isBlockMultiSelected', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; it( 'should return true if the block is multi selected', () => { - expect( isBlockMultiSelected( state, 3 ) ).toBe( true ); + expect( isBlockMultiSelected( state, '3' ) ).toBe( true ); } ); it( 'should return false if the block is not multi selected', () => { - expect( isBlockMultiSelected( state, 5 ) ).toBe( false ); + expect( isBlockMultiSelected( state, '5' ) ).toBe( false ); } ); } ); describe( 'isFirstMultiSelectedBlock', () => { const state = { blocks: { - order: { - '': [ 5, 4, 3, 2, 1 ], - }, - parents: { - 1: '', - 2: '', - 3: '', - 4: '', - 5: '', - }, + order: new Map( + Object.entries( { + '': [ '5', '4', '3', '2', '1' ], + } ) + ), + parents: new Map( + Object.entries( { + 1: '', + 2: '', + 3: '', + 4: '', + 5: '', + } ) + ), }, selection: { - selectionStart: { clientId: 2 }, - selectionEnd: { clientId: 4 }, + selectionStart: { clientId: '2' }, + selectionEnd: { clientId: '4' }, }, }; it( 'should return true if the block is first in multi selection', () => { - expect( isFirstMultiSelectedBlock( state, 4 ) ).toBe( true ); + expect( isFirstMultiSelectedBlock( state, '4' ) ).toBe( true ); } ); it( 'should return false if the block is not first in multi selection', () => { - expect( isFirstMultiSelectedBlock( state, 3 ) ).toBe( false ); + expect( isFirstMultiSelectedBlock( state, '3' ) ).toBe( false ); } ); } ); @@ -1927,7 +2245,7 @@ describe( 'selectors', () => { blocksMode: {}, }; - expect( getBlockMode( state, 123 ) ).toEqual( 'visual' ); + expect( getBlockMode( state, '123' ) ).toEqual( 'visual' ); } ); it( 'should return the block mode', () => { @@ -1937,7 +2255,7 @@ describe( 'selectors', () => { }, }; - expect( getBlockMode( state, 123 ) ).toEqual( 'html' ); + expect( getBlockMode( state, '123' ) ).toEqual( 'html' ); } ); } ); @@ -2031,10 +2349,12 @@ describe( 'selectors', () => { const state = { draggedBlocks: [ 'block-1_grandparent' ], blocks: { - parents: { - 'block-1': 'block-1_parent', - 'block-1_parent': 'block-1_grandparent', - }, + parents: new Map( + Object.entries( { + 'block-1': 'block-1_parent', + 'block-1_parent': 'block-1_grandparent', + } ) + ), }, }; expect( isAncestorBeingDragged( state, 'block-1' ) ).toBe( true ); @@ -2044,10 +2364,12 @@ describe( 'selectors', () => { const state = { draggedBlocks: [ 'block-2_grandparent' ], blocks: { - parents: { - 'block-1': 'block-1_parent', - 'block-1_parent': 'block-1_grandparent', - }, + parents: new Map( + Object.entries( { + 'block-1': 'block-1_parent', + 'block-1_parent': 'block-1_grandparent', + } ) + ), }, }; expect( isAncestorBeingDragged( state, 'block-1' ) ).toBe( false ); @@ -2057,10 +2379,12 @@ describe( 'selectors', () => { const state = { draggedBlocks: [], blocks: { - parents: { - 'block-1': 'block-1_parent', - 'block-1_parent': 'block-1_grandparent', - }, + parents: new Map( + Object.entries( { + 'block-1': 'block-1_parent', + 'block-1_parent': 'block-1_grandparent', + } ) + ), }, }; expect( isAncestorBeingDragged( state, 'block-1' ) ).toBe( false ); @@ -2093,23 +2417,31 @@ describe( 'selectors', () => { selectionEnd: { clientId: 'clientId2' }, }, blocks: { - byClientId: { - clientId1: { clientId: 'clientId1' }, - clientId2: { clientId: 'clientId2' }, - }, - attributes: { - clientId1: {}, - clientId2: {}, - }, - order: { - '': [ 'clientId1' ], - clientId1: [ 'clientId2' ], - clientId2: [], - }, - parents: { - clientId1: '', - clientId2: 'clientId1', - }, + byClientId: new Map( + Object.entries( { + clientId1: { clientId: 'clientId1' }, + clientId2: { clientId: 'clientId2' }, + } ) + ), + attributes: new Map( + Object.entries( { + clientId1: {}, + clientId2: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'clientId1' ], + clientId1: [ 'clientId2' ], + clientId2: [], + } ) + ), + parents: new Map( + Object.entries( { + clientId1: '', + clientId2: 'clientId1', + } ) + ), }, insertionPoint: { rootClientId: undefined, @@ -2130,19 +2462,27 @@ describe( 'selectors', () => { selectionEnd: { clientId: 'clientId1' }, }, blocks: { - byClientId: { - clientId1: { clientId: 'clientId1' }, - }, - attributes: { - clientId1: {}, - }, - order: { - '': [ 'clientId1' ], - clientId1: [], - }, - parents: { - clientId1: '', - }, + byClientId: new Map( + Object.entries( { + clientId1: { clientId: 'clientId1' }, + } ) + ), + attributes: new Map( + Object.entries( { + clientId1: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'clientId1' ], + clientId1: [], + } ) + ), + parents: new Map( + Object.entries( { + clientId1: '', + } ) + ), }, insertionPoint: null, }; @@ -2160,23 +2500,31 @@ describe( 'selectors', () => { selectionEnd: { clientId: 'clientId2' }, }, blocks: { - byClientId: { - clientId1: { clientId: 'clientId1' }, - clientId2: { clientId: 'clientId2' }, - }, - attributes: { - clientId1: {}, - clientId2: {}, - }, - order: { - '': [ 'clientId1' ], - clientId1: [ 'clientId2' ], - clientId2: [], - }, - parents: { - clientId1: '', - clientId2: 'clientId1', - }, + byClientId: new Map( + Object.entries( { + clientId1: { clientId: 'clientId1' }, + clientId2: { clientId: 'clientId2' }, + } ) + ), + attributes: new Map( + Object.entries( { + clientId1: {}, + clientId2: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'clientId1' ], + clientId1: [ 'clientId2' ], + clientId2: [], + } ) + ), + parents: new Map( + Object.entries( { + clientId1: '', + clientId2: 'clientId1', + } ) + ), }, insertionPoint: null, }; @@ -2194,23 +2542,31 @@ describe( 'selectors', () => { selectionEnd: { clientId: 'clientId2' }, }, blocks: { - byClientId: { - clientId1: { clientId: 'clientId1' }, - clientId2: { clientId: 'clientId2' }, - }, - attributes: { - clientId1: {}, - clientId2: {}, - }, - order: { - '': [ 'clientId1', 'clientId2' ], - clientId1: [], - clientId2: [], - }, - parents: { - clientId1: '', - clientId2: '', - }, + byClientId: new Map( + Object.entries( { + clientId1: { clientId: 'clientId1' }, + clientId2: { clientId: 'clientId2' }, + } ) + ), + attributes: new Map( + Object.entries( { + clientId1: {}, + clientId2: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'clientId1', 'clientId2' ], + clientId1: [], + clientId2: [], + } ) + ), + parents: new Map( + Object.entries( { + clientId1: '', + clientId2: '', + } ) + ), }, insertionPoint: null, }; @@ -2228,23 +2584,31 @@ describe( 'selectors', () => { selectionEnd: {}, }, blocks: { - byClientId: { - clientId1: { clientId: 'clientId1' }, - clientId2: { clientId: 'clientId2' }, - }, - attributes: { - clientId1: {}, - clientId2: {}, - }, - order: { - '': [ 'clientId1', 'clientId2' ], - clientId1: [], - clientId2: [], - }, - parents: { - clientId1: '', - clientId2: '', - }, + byClientId: new Map( + Object.entries( { + clientId1: { clientId: 'clientId1' }, + clientId2: { clientId: 'clientId2' }, + } ) + ), + attributes: new Map( + Object.entries( { + clientId1: {}, + clientId2: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'clientId1', 'clientId2' ], + clientId1: [], + clientId2: [], + } ) + ), + parents: new Map( + Object.entries( { + clientId1: '', + clientId2: '', + } ) + ), }, insertionPoint: null, }; @@ -2281,8 +2645,8 @@ describe( 'selectors', () => { it( 'should deny blocks that are not registered', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, + byClientId: new Map(), + attributes: new Map(), }, blockListSettings: {}, settings: {}, @@ -2293,8 +2657,8 @@ describe( 'selectors', () => { it( 'should deny blocks that are not allowed by the editor', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, + byClientId: new Map(), + attributes: new Map(), }, blockListSettings: {}, settings: { @@ -2309,8 +2673,8 @@ describe( 'selectors', () => { it( 'should allow blocks that are allowed by the editor', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, + byClientId: new Map(), + attributes: new Map(), }, blockListSettings: {}, settings: { @@ -2325,8 +2689,8 @@ describe( 'selectors', () => { it( 'should deny blocks when the editor has a template lock', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, + byClientId: new Map(), + attributes: new Map(), }, blockListSettings: {}, settings: { @@ -2341,8 +2705,8 @@ describe( 'selectors', () => { it( 'should deny blocks that restrict parent from being inserted into the root', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, + byClientId: new Map(), + attributes: new Map(), }, blockListSettings: {}, settings: {}, @@ -2355,12 +2719,16 @@ describe( 'selectors', () => { it( 'should deny blocks that restrict parent from being inserted into a restricted parent', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-a' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-a' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: {}, settings: {}, @@ -2373,12 +2741,16 @@ describe( 'selectors', () => { it( 'should allow blocks to be inserted into an allowed parent', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-b' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-b' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: { block1: {}, @@ -2393,12 +2765,16 @@ describe( 'selectors', () => { it( 'should deny blocks from being inserted into a block that does not allow inner blocks', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-b' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-b' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: { block1: {}, @@ -2413,12 +2789,16 @@ describe( 'selectors', () => { it( 'should deny restricted blocks from being inserted into a block that restricts allowedBlocks', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-a' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-a' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: { block1: { @@ -2435,12 +2815,16 @@ describe( 'selectors', () => { it( 'should allow allowed blocks to be inserted into a block that restricts allowedBlocks', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-a' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-a' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: { block1: { @@ -2457,12 +2841,16 @@ describe( 'selectors', () => { it( 'should prioritise parent over allowedBlocks', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-b' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-b' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: { block1: { @@ -2479,12 +2867,16 @@ describe( 'selectors', () => { it( 'should deny blocks that restrict parent to core/post-content when not in editor root', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-c' }, - }, - attributes: { - block1: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-c' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + } ) + ), }, blockListSettings: {}, settings: {}, @@ -2497,8 +2889,8 @@ describe( 'selectors', () => { it( 'should allow blocks that restrict parent to core/post-content when in editor root', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, + byClientId: new Map(), + attributes: new Map(), }, blockListSettings: {}, settings: {}, @@ -2511,17 +2903,23 @@ describe( 'selectors', () => { it( 'should allow blocks to be inserted in a descendant of a required ancestor', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-ancestor' }, - block2: { name: 'core/block' }, - }, - attributes: { - block1: {}, - block2: {}, - }, - parents: { - block2: 'block1', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-ancestor' }, + block2: { name: 'core/block' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + } ) + ), + parents: new Map( + Object.entries( { + block2: 'block1', + } ) + ), }, blockListSettings: { block1: {}, @@ -2541,20 +2939,26 @@ describe( 'selectors', () => { it( 'should allow blocks to be inserted if both parent and ancestor restrictions are met', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-ancestor' }, - block2: { name: 'core/block' }, - block3: { name: 'core/test-block-parent' }, - }, - attributes: { - block1: {}, - block2: {}, - block3: {}, - }, - parents: { - block2: 'block1', - block3: 'block2', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-ancestor' }, + block2: { name: 'core/block' }, + block3: { name: 'core/test-block-parent' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + block3: {}, + } ) + ), + parents: new Map( + Object.entries( { + block2: 'block1', + block3: 'block2', + } ) + ), }, blockListSettings: { block1: {}, @@ -2575,19 +2979,25 @@ describe( 'selectors', () => { it( 'should deny blocks from being inserted outside a required ancestor', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-ancestor' }, - block2: { name: 'core/block' }, - block3: { name: 'core/block' }, - }, - attributes: { - block1: {}, - block2: {}, - block3: {}, - }, - parents: { - block3: 'block2', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-ancestor' }, + block2: { name: 'core/block' }, + block3: { name: 'core/block' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + block3: {}, + } ) + ), + parents: new Map( + Object.entries( { + block3: 'block2', + } ) + ), }, blockListSettings: { block1: {}, @@ -2608,19 +3018,25 @@ describe( 'selectors', () => { it( 'should deny blocks from being inserted outside of a required ancestor, even if parent matches', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-ancestor' }, - block2: { name: 'core/block' }, - block3: { name: 'core/test-block-parent' }, - }, - attributes: { - block1: {}, - block2: {}, - block3: {}, - }, - parents: { - block3: 'block2', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-ancestor' }, + block2: { name: 'core/block' }, + block3: { name: 'core/test-block-parent' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + block3: {}, + } ) + ), + parents: new Map( + Object.entries( { + block3: 'block2', + } ) + ), }, blockListSettings: { block1: {}, @@ -2641,17 +3057,23 @@ describe( 'selectors', () => { it( 'should deny blocks from being inserted inside ancestor if parent restricts allowedBlocks', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-ancestor' }, - block2: { name: 'core/block' }, - }, - attributes: { - block1: {}, - block2: {}, - }, - parents: { - block2: 'block1', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-ancestor' }, + block2: { name: 'core/block' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + } ) + ), + parents: new Map( + Object.entries( { + block2: 'block1', + } ) + ), }, blockListSettings: { block1: {}, @@ -2673,17 +3095,23 @@ describe( 'selectors', () => { it( 'should deny blocks from being inserted inside ancestor if parent restriction is not met', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-ancestor' }, - block2: { name: 'core/block' }, - }, - attributes: { - block1: {}, - block2: {}, - }, - parents: { - block2: 'block1', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-ancestor' }, + block2: { name: 'core/block' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + } ) + ), + parents: new Map( + Object.entries( { + block2: 'block1', + } ) + ), }, blockListSettings: { block1: {}, @@ -2705,16 +3133,20 @@ describe( 'selectors', () => { it( 'should allow blocks', () => { const state = { blocks: { - byClientId: { - 1: { name: 'core/test-block-a' }, - 2: { name: 'core/test-block-b' }, - 3: { name: 'core/test-block-c' }, - }, - attributes: { - 1: {}, - 2: {}, - 3: {}, - }, + byClientId: new Map( + Object.entries( { + 1: { name: 'core/test-block-a' }, + 2: { name: 'core/test-block-b' }, + 3: { name: 'core/test-block-c' }, + } ) + ), + attributes: new Map( + Object.entries( { + 1: {}, + 2: {}, + 3: {}, + } ) + ), }, blockListSettings: { 1: { @@ -2732,16 +3164,20 @@ describe( 'selectors', () => { it( 'should deny blocks', () => { const state = { blocks: { - byClientId: { - 1: { name: 'core/test-block-a' }, - 2: { name: 'core/test-block-b' }, - 3: { name: 'core/test-block-c' }, - }, - attributes: { - 1: {}, - 2: {}, - 3: {}, - }, + byClientId: new Map( + Object.entries( { + 1: { name: 'core/test-block-a' }, + 2: { name: 'core/test-block-b' }, + 3: { name: 'core/test-block-c' }, + } ) + ), + attributes: new Map( + Object.entries( { + 1: {}, + 2: {}, + 3: {}, + } ) + ), }, blockListSettings: { 1: { @@ -2758,15 +3194,17 @@ describe( 'selectors', () => { it( 'should properly list block type and reusable block items', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, - tree: { - '': { - innerBlocks: [], - }, - }, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), + tree: new Map( + Object.entries( { + '': { + innerBlocks: [], + }, + } ) + ), }, settings: { __experimentalReusableBlocks: [ @@ -2827,35 +3265,45 @@ describe( 'selectors', () => { it( 'should correctly cache the return values', () => { const state = { blocks: { - byClientId: { - block3: { name: 'core/test-block-a' }, - block4: { name: 'core/test-block-a' }, - }, - attributes: { - block3: {}, - block4: {}, - }, - order: { - '': [ 'block3', 'block4' ], - }, - parents: { - block3: '', - block4: '', - }, - tree: { - block3: { - clientId: 'block3', - name: 'core/test-block-a', - attributes: {}, - innerBlocks: [], - }, - block4: { - clientId: 'block4', - name: 'core/test-block-a', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + block3: { name: 'core/test-block-a' }, + block4: { name: 'core/test-block-a' }, + } ) + ), + attributes: new Map( + Object.entries( { + block3: {}, + block4: {}, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'block3', 'block4' ], + } ) + ), + parents: new Map( + Object.entries( { + block3: '', + block4: '', + } ) + ), + tree: new Map( + Object.entries( { + block3: { + clientId: 'block3', + name: 'core/test-block-a', + attributes: {}, + innerBlocks: [], + }, + block4: { + clientId: 'block4', + name: 'core/test-block-a', + attributes: {}, + innerBlocks: [], + }, + } ) + ), controlledInnerBlocks: {}, }, settings: { @@ -2934,28 +3382,36 @@ describe( 'selectors', () => { it( 'should set isDisabled when a block with `multiple: false` has been used', () => { const state = { blocks: { - byClientId: { - block1: { - clientId: 'block1', - name: 'core/test-block-b', - }, - }, - attributes: { - block1: { attribute: {} }, - }, - order: { - '': [ 'block1' ], - }, - tree: { - block1: { - clientId: 'block1', - name: 'core/test-block-b', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + block1: { + clientId: 'block1', + name: 'core/test-block-b', + }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: { attribute: {} }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'block1' ], + } ) + ), + tree: new Map( + Object.entries( { + block1: { + clientId: 'block1', + name: 'core/test-block-b', + attributes: {}, + innerBlocks: [], + }, + } ) + ), controlledInnerBlocks: {}, - parents: {}, + parents: new Map(), }, preferences: { insertUsage: {}, @@ -2973,10 +3429,10 @@ describe( 'selectors', () => { it( 'should set a frecency', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), cache: {}, }, preferences: { @@ -3068,10 +3524,10 @@ describe( 'selectors', () => { it( 'should properly return block type items', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), cache: {}, }, settings: {}, @@ -3108,10 +3564,10 @@ describe( 'selectors', () => { it( 'should support single block object', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), cache: {}, }, settings: {}, @@ -3125,19 +3581,25 @@ describe( 'selectors', () => { it( 'should return only eligible blocks for transformation - `allowedBlocks`', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/with-tranforms-b' }, - block2: { name: 'core/with-tranforms-a' }, - }, - attributes: { - block1: {}, - block2: {}, - }, - order: {}, - parents: { - block1: '', - block2: 'block1', - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/with-tranforms-b' }, + block2: { name: 'core/with-tranforms-a' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + } ) + ), + order: new Map(), + parents: new Map( + Object.entries( { + block1: '', + block2: 'block1', + } ) + ), cache: {}, controlledInnerBlocks: {}, }, @@ -3160,28 +3622,36 @@ describe( 'selectors', () => { it( 'should take into account the usage of blocks settings `multiple` - if multiple blocks of the same type are allowed', () => { const state = { blocks: { - byClientId: { - block1: { - clientId: 'block1', - name: 'core/with-tranforms-c', - }, - }, - attributes: { - block1: { attribute: {} }, - }, - order: { - '': [ 'block1' ], - }, - tree: { - block1: { - clientId: 'block1', - name: 'core/with-tranforms-c', - attributes: {}, - innerBlocks: [], - }, - }, + byClientId: new Map( + Object.entries( { + block1: { + clientId: 'block1', + name: 'core/with-tranforms-c', + }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: { attribute: {} }, + } ) + ), + order: new Map( + Object.entries( { + '': [ 'block1' ], + } ) + ), + tree: new Map( + Object.entries( { + block1: { + clientId: 'block1', + name: 'core/with-tranforms-c', + attributes: {}, + innerBlocks: [], + }, + } ) + ), controlledInnerBlocks: {}, - parents: {}, + parents: new Map(), }, preferences: { insertUsage: {}, @@ -3208,10 +3678,10 @@ describe( 'selectors', () => { it( 'should set frecency', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), cache: {}, }, preferences: { @@ -3264,7 +3734,7 @@ describe( 'selectors', () => { } ); describe( 'getTemplateLock', () => { - it( 'should return the general template lock if no clientId was set', () => { + it( 'should return the general template lock if no clientId was specified', () => { const state = { settings: { templateLock: 'all' }, }; @@ -3272,7 +3742,7 @@ describe( 'selectors', () => { expect( getTemplateLock( state ) ).toBe( 'all' ); } ); - it( 'should return undefined if the specified clientId was not found', () => { + it( 'should return false if the specified clientId was not found', () => { const state = { settings: { templateLock: 'all' }, blockListSettings: { @@ -3282,10 +3752,10 @@ describe( 'selectors', () => { }, }; - expect( getTemplateLock( state, 'ribs' ) ).toBe( undefined ); + expect( getTemplateLock( state, 'ribs' ) ).toBe( false ); } ); - it( 'should return undefined if template lock was not set on the specified block', () => { + it( 'should return false if template lock was not set on the specified block', () => { const state = { settings: { templateLock: 'all' }, blockListSettings: { @@ -3295,7 +3765,7 @@ describe( 'selectors', () => { }, }; - expect( getTemplateLock( state, 'ribs' ) ).toBe( undefined ); + expect( getTemplateLock( state, 'chicken' ) ).toBe( false ); } ); it( 'should return the template lock for the specified clientId', () => { @@ -3429,20 +3899,24 @@ describe( 'selectors', () => { describe( 'getLowestCommonAncestorWithSelectedBlock', () => { const blocks = { - order: { - '': [ 'a', 'b' ], - a: [ 'c', 'd' ], - d: [ 'e' ], - b: [ 'f' ], - }, - parents: { - a: '', - b: '', - c: 'a', - d: 'a', - e: 'd', - f: 'b', - }, + order: new Map( + Object.entries( { + '': [ 'a', 'b' ], + a: [ 'c', 'd' ], + d: [ 'e' ], + b: [ 'f' ], + } ) + ), + parents: new Map( + Object.entries( { + a: '', + b: '', + c: 'a', + d: 'a', + e: 'd', + f: 'b', + } ) + ), }; it( 'should not be defined if there is no block selected', () => { @@ -3541,35 +4015,39 @@ describe( 'selectors', () => { }, }, blocks: { - parents: { - 'client-id-01': '', - 'client-id-02': 'client-id-01', - 'client-id-03': 'client-id-02', - 'client-id-04': 'client-id-03', - 'client-id-05': 'client-id-03', - }, - byClientId: { - 'client-id-01': { - clientId: 'client-id-01', - name: 'core/columns', - }, - 'client-id-02': { - clientId: 'client-id-02', - name: 'core/navigation', - }, - 'client-id-03': { - clientId: 'client-id-03', - name: 'core/navigation-link', - }, - 'client-id-04': { - clientId: 'client-id-04', - name: 'core/navigation-link', - }, - 'client-id-05': { - clientId: 'client-id-05', - name: 'core/navigation-link', - }, - }, + parents: new Map( + Object.entries( { + 'client-id-01': '', + 'client-id-02': 'client-id-01', + 'client-id-03': 'client-id-02', + 'client-id-04': 'client-id-03', + 'client-id-05': 'client-id-03', + } ) + ), + byClientId: new Map( + Object.entries( { + 'client-id-01': { + clientId: 'client-id-01', + name: 'core/columns', + }, + 'client-id-02': { + clientId: 'client-id-02', + name: 'core/navigation', + }, + 'client-id-03': { + clientId: 'client-id-03', + name: 'core/navigation-link', + }, + 'client-id-04': { + clientId: 'client-id-04', + name: 'core/navigation-link', + }, + 'client-id-05': { + clientId: 'client-id-05', + name: 'core/navigation-link', + }, + } ) + ), cache: { 'client-id-01': {}, 'client-id-02': {}, @@ -3577,9 +4055,11 @@ describe( 'selectors', () => { 'client-id-04': {}, 'client-id-05': {}, }, - order: { - 'client-id-03': [ 'client-id-04', 'client-id-05' ], - }, + order: new Map( + Object.entries( { + 'client-id-03': [ 'client-id-04', 'client-id-05' ], + } ) + ), controlledInnerBlocks: {}, }, }; @@ -3611,14 +4091,18 @@ describe( 'selectors', () => { describe( '__experimentalGetAllowedPatterns', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-a' }, - block2: { name: 'core/test-block-b' }, - }, - attributes: { - block1: {}, - block2: {}, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-a' }, + block2: { name: 'core/test-block-b' }, + } ) + ), + attributes: new Map( + Object.entries( { + block1: {}, + block2: {}, + } ) + ), }, blockListSettings: { block1: { @@ -3670,7 +4154,7 @@ describe( 'selectors', () => { it( 'should return empty array if only patterns hidden from UI exist', () => { expect( __experimentalGetAllowedPatterns( { - blocks: { byClientId: {} }, + blocks: { byClientId: new Map() }, blockListSettings: {}, settings: { __experimentalBlockPatterns: [ @@ -3742,9 +4226,11 @@ describe( 'selectors', () => { describe( '__experimentalGetPatternsByBlockTypes', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-a' }, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-a' }, + } ) + ), }, blockListSettings: { block1: { @@ -3816,10 +4302,12 @@ describe( 'selectors', () => { describe( '__experimentalGetPatternTransformItems', () => { const state = { blocks: { - byClientId: { - block1: { name: 'core/test-block-a' }, - block2: { name: 'core/test-block-b' }, - }, + byClientId: new Map( + Object.entries( { + block1: { name: 'core/test-block-a' }, + block2: { name: 'core/test-block-b' }, + } ) + ), controlledInnerBlocks: { 'block2-clientId': true }, }, blockListSettings: { @@ -4073,10 +4561,10 @@ describe( 'getInserterItems with core blocks prioritization', () => { it( 'should prioritize core blocks by sorting them at the top of the returned list', () => { const state = { blocks: { - byClientId: {}, - attributes: {}, - order: {}, - parents: {}, + byClientId: new Map(), + attributes: new Map(), + order: new Map(), + parents: new Map(), cache: {}, }, settings: {}, @@ -4104,11 +4592,13 @@ describe( '__unstableGetClientIdWithClientIdsTree', () => { it( "should return a stripped down block object containing only its client ID and its inner blocks' client IDs", () => { const state = { blocks: { - order: { - '': [ 'foo' ], - foo: [ 'bar', 'baz' ], - bar: [ 'qux' ], - }, + order: new Map( + Object.entries( { + '': [ 'foo' ], + foo: [ 'bar', 'baz' ], + bar: [ 'qux' ], + } ) + ), }, }; @@ -4130,11 +4620,13 @@ describe( '__unstableGetClientIdsTree', () => { it( "should return the full content tree starting from the given root, consisting of stripped down block object containing only its client ID and its inner blocks' client IDs", () => { const state = { blocks: { - order: { - '': [ 'foo' ], - foo: [ 'bar', 'baz' ], - bar: [ 'qux' ], - }, + order: new Map( + Object.entries( { + '': [ 'foo' ], + foo: [ 'bar', 'baz' ], + bar: [ 'qux' ], + } ) + ), }, }; @@ -4150,11 +4642,13 @@ describe( '__unstableGetClientIdsTree', () => { it( "should return the full content tree starting from the root, consisting of stripped down block object containing only its client ID and its inner blocks' client IDs", () => { const state = { blocks: { - order: { - '': [ 'foo' ], - foo: [ 'bar', 'baz' ], - bar: [ 'qux' ], - }, + order: new Map( + Object.entries( { + '': [ 'foo' ], + foo: [ 'bar', 'baz' ], + bar: [ 'qux' ], + } ) + ), }, }; @@ -4172,3 +4666,23 @@ describe( '__unstableGetClientIdsTree', () => { ] ); } ); } ); + +describe( 'getLastInsertedBlockClientId', () => { + it( 'should return undefined if no blocks have been inserted', () => { + const state = { + lastBlockInserted: {}, + }; + + expect( getLastInsertedBlockClientId( state ) ).toEqual( undefined ); + } ); + + it( 'should return clientId if blocks have been inserted', () => { + const state = { + lastBlockInserted: { + clientId: '123456', + }, + }; + + expect( getLastInsertedBlockClientId( state ) ).toEqual( '123456' ); + } ); +} ); diff --git a/packages/block-editor/src/style.scss b/packages/block-editor/src/style.scss index 853a030f04594..76da3c16325db 100644 --- a/packages/block-editor/src/style.scss +++ b/packages/block-editor/src/style.scss @@ -2,14 +2,11 @@ @import "./components/block-alignment-control/style.scss"; @import "./components/block-icon/style.scss"; @import "./components/block-inspector/style.scss"; -@import "./components/block-list/style.scss"; @import "./components/block-tools/style.scss"; -@import "./components/block-list-appender/style.scss"; @import "./components/block-lock/style.scss"; @import "./components/block-breadcrumb/style.scss"; @import "./components/block-card/style.scss"; @import "./components/block-compare/style.scss"; -@import "./components/block-content-overlay/style.scss"; @import "./components/block-draggable/style.scss"; @import "./components/block-mover/style.scss"; @import "./components/block-navigation/style.scss"; @@ -28,21 +25,18 @@ @import "./components/button-block-appender/style.scss"; @import "./components/colors-gradients/style.scss"; @import "./components/contrast-checker/style.scss"; -@import "./components/default-block-appender/style.scss"; @import "./components/date-format-picker/style.scss"; @import "./components/duotone-control/style.scss"; @import "./components/font-appearance-control/style.scss"; +@import "./components/height-control/style.scss"; @import "./components/image-size-control/style.scss"; -@import "./components/inner-blocks/style.scss"; @import "./components/inserter-list-item/style.scss"; @import "./components/inspector-popover-header/style.scss"; @import "./components/justify-content-control/style.scss"; @import "./components/link-control/style.scss"; @import "./components/list-view/style.scss"; @import "./components/media-replace-flow/style.scss"; -@import "./components/media-placeholder/style.scss"; @import "./components/multi-selection-inspector/style.scss"; -@import "./components/plain-text/style.scss"; @import "./components/responsive-block-control/style.scss"; @import "./components/rich-text/style.scss"; @import "./components/skip-to-selected-block/style.scss"; @@ -65,4 +59,7 @@ @import "./components/preview-options/style.scss"; @import "./components/spacing-sizes-control/style.scss"; +// Experiments. +@import "./components/off-canvas-editor/style.scss"; + @include wordpress-admin-schemes(); diff --git a/packages/block-editor/src/utils/sorting.js b/packages/block-editor/src/utils/sorting.js new file mode 100644 index 0000000000000..84cf65a08e827 --- /dev/null +++ b/packages/block-editor/src/utils/sorting.js @@ -0,0 +1,54 @@ +/** + * Recursive stable sorting comparator function. + * + * @param {string|Function} field Field to sort by. + * @param {Array} items Items to sort. + * @param {string} order Order, 'asc' or 'desc'. + * @return {Function} Comparison function to be used in a `.sort()`. + */ +const comparator = ( field, items, order ) => { + return ( a, b ) => { + let cmpA, cmpB; + + if ( typeof field === 'function' ) { + cmpA = field( a ); + cmpB = field( b ); + } else { + cmpA = a[ field ]; + cmpB = b[ field ]; + } + + if ( cmpA > cmpB ) { + return order === 'asc' ? 1 : -1; + } else if ( cmpB > cmpA ) { + return order === 'asc' ? -1 : 1; + } + + const orderA = items.findIndex( ( item ) => item === a ); + const orderB = items.findIndex( ( item ) => item === b ); + + // Stable sort: maintaining original array order + if ( orderA > orderB ) { + return 1; + } else if ( orderB > orderA ) { + return -1; + } + + return 0; + }; +}; + +/** + * Order items by a certain key. + * Supports decorator functions that allow complex picking of a comparison field. + * Sorts in ascending order by default, but supports descending as well. + * Stable sort - maintains original order of equal items. + * + * @param {Array} items Items to order. + * @param {string|Function} field Field to order by. + * @param {string} order Sorting order, `asc` or `desc`. + * @return {Array} Sorted items. + */ +export function orderBy( items, field, order = 'asc' ) { + return items.concat().sort( comparator( field, items, order ) ); +} diff --git a/packages/block-editor/src/utils/test/sorting.js b/packages/block-editor/src/utils/test/sorting.js new file mode 100644 index 0000000000000..f1038cda5809c --- /dev/null +++ b/packages/block-editor/src/utils/test/sorting.js @@ -0,0 +1,49 @@ +/** + * Internal dependencies + */ +import { orderBy } from '../sorting'; + +describe( 'orderBy', () => { + it( 'should not mutate original input', () => { + const input = []; + expect( orderBy( input, 'x' ) ).not.toBe( input ); + } ); + + it( 'should sort items by a field when it is specified as a string', () => { + const input = [ { x: 2 }, { x: 1 }, { x: 3 } ]; + const expected = [ { x: 1 }, { x: 2 }, { x: 3 } ]; + expect( orderBy( input, 'x' ) ).toEqual( expected ); + } ); + + it( 'should support functions for picking the field', () => { + const input = [ { x: 2 }, { x: 1 }, { x: 3 } ]; + const expected = [ { x: 1 }, { x: 2 }, { x: 3 } ]; + expect( orderBy( input, ( item ) => item.x ) ).toEqual( expected ); + } ); + + it( 'should support sorting in a descending order', () => { + const input = [ { x: 2 }, { x: 1 }, { x: 3 } ]; + const expected = [ { x: 3 }, { x: 2 }, { x: 1 } ]; + expect( orderBy( input, 'x', 'desc' ) ).toEqual( expected ); + } ); + + it( 'should maintain original order of equal items', () => { + const a = { x: 1, a: 1 }; + const b = { x: 1, b: 2 }; + const c = { x: 0 }; + const d = { x: 1, b: 4 }; + const input = [ a, b, c, d ]; + const expected = [ c, a, b, d ]; + expect( orderBy( input, 'x' ) ).toEqual( expected ); + } ); + + it( 'should maintain original order of equal items in descencing order', () => { + const a = { x: 1, a: 1 }; + const b = { x: 1, b: 2 }; + const c = { x: 0 }; + const d = { x: 1, b: 4 }; + const input = [ a, b, c, d ]; + const expected = [ a, b, d, c ]; + expect( orderBy( input, 'x', 'desc' ) ).toEqual( expected ); + } ); +} ); diff --git a/packages/block-library/CHANGELOG.md b/packages/block-library/CHANGELOG.md index 1dce136b8f867..c11fbb8ab8d9b 100644 --- a/packages/block-library/CHANGELOG.md +++ b/packages/block-library/CHANGELOG.md @@ -2,6 +2,10 @@ ## Unreleased +### Breaking Changes + +- Updated dependencies to require React 18 ([45235](https://github.com/WordPress/gutenberg/pull/45235)) + ## 7.19.0 (2022-11-16) ## 7.18.0 (2022-11-02) diff --git a/packages/block-library/package.json b/packages/block-library/package.json index d5671a8d30064..7510c3b22a7c2 100644 --- a/packages/block-library/package.json +++ b/packages/block-library/package.json @@ -45,6 +45,7 @@ "@wordpress/deprecated": "file:../deprecated", "@wordpress/dom": "file:../dom", "@wordpress/element": "file:../element", + "@wordpress/escape-html": "file:../escape-html", "@wordpress/hooks": "file:../hooks", "@wordpress/html-entities": "file:../html-entities", "@wordpress/i18n": "file:../i18n", @@ -62,14 +63,15 @@ "colord": "^2.7.0", "escape-html": "^1.0.3", "fast-average-color": "^9.1.1", + "fast-deep-equal": "^3.1.3", "lodash": "^4.17.21", "memize": "^1.1.0", "micromodal": "^0.4.10", "remove-accents": "^0.4.2" }, "peerDependencies": { - "react": "^17.0.0", - "react-dom": "^17.0.0" + "react": "^18.0.0", + "react-dom": "^18.0.0" }, "publishConfig": { "access": "public" diff --git a/packages/block-library/src/audio/edit.native.js b/packages/block-library/src/audio/edit.native.js index 30ee44ff3f948..fbc39ba190d3e 100644 --- a/packages/block-library/src/audio/edit.native.js +++ b/packages/block-library/src/audio/edit.native.js @@ -2,7 +2,6 @@ * External dependencies */ import { TouchableWithoutFeedback } from 'react-native'; -import { isEmpty } from 'lodash'; /** * WordPress dependencies @@ -228,7 +227,7 @@ function AudioEdit( { - isEmpty( caption ) + ! caption ? /* translators: accessibility text. Empty Audio caption. */ __( 'Audio caption. Empty' ) : sprintf( diff --git a/packages/block-library/src/audio/index.js b/packages/block-library/src/audio/index.js index 0d3d05936ef90..aa1bba1c341d2 100644 --- a/packages/block-library/src/audio/index.js +++ b/packages/block-library/src/audio/index.js @@ -23,6 +23,7 @@ export const settings = { attributes: { src: 'https://upload.wikimedia.org/wikipedia/commons/d/dd/Armstrong_Small_Step.ogg', }, + viewportWidth: 350, }, transforms, deprecated, diff --git a/packages/block-library/src/avatar/edit.js b/packages/block-library/src/avatar/edit.js index 7df772677ebb7..2741b7adcd3da 100644 --- a/packages/block-library/src/avatar/edit.js +++ b/packages/block-library/src/avatar/edit.js @@ -35,6 +35,7 @@ const AvatarInspectorControls = ( { setAttributes( { diff --git a/packages/block-library/src/avatar/user-control.js b/packages/block-library/src/avatar/user-control.js index 49370b39716ef..598c05011eaed 100644 --- a/packages/block-library/src/avatar/user-control.js +++ b/packages/block-library/src/avatar/user-control.js @@ -33,6 +33,7 @@ function UserControl( { value, onChange } ) { return ( select( blockEditorStore ).canRemoveBlock( clientId ), + const { canRemove, innerBlockCount } = useSelect( + ( select ) => { + const { canRemoveBlock, getBlockCount } = + select( blockEditorStore ); + return { + canRemove: canRemoveBlock( clientId ), + innerBlockCount: getBlockCount( clientId ), + }; + }, [ clientId ] ); @@ -109,7 +116,11 @@ export default function ReusableBlockEdit( { attributes: { ref }, clientId } ) { convertBlockToStatic( clientId ) } - label={ __( 'Convert to regular blocks' ) } + label={ + innerBlockCount > 1 + ? __( 'Convert to regular blocks' ) + : __( 'Convert to regular block' ) + } icon={ ungroup } showTooltip /> diff --git a/packages/block-library/src/block/edit.native.js b/packages/block-library/src/block/edit.native.js index ce2a3999e2b55..96472470efebd 100644 --- a/packages/block-library/src/block/edit.native.js +++ b/packages/block-library/src/block/edit.native.js @@ -77,7 +77,7 @@ export default function ReusableBlockEdit( { styles.spinnerDark ); - const { hasResolved, isEditing, isMissing } = useSelect( + const { hasResolved, isEditing, isMissing, innerBlockCount } = useSelect( ( select ) => { const persistedBlock = select( coreStore ).getEntityRecord( 'postType', @@ -88,6 +88,9 @@ export default function ReusableBlockEdit( { 'getEntityRecord', [ 'postType', 'wp_block', ref ] ); + + const { getBlockCount } = select( blockEditorStore ); + return { hasResolved: hasResolvedBlock, isEditing: @@ -95,6 +98,7 @@ export default function ReusableBlockEdit( { reusableBlocksStore ).__experimentalIsEditingReusableBlock( clientId ), isMissing: hasResolvedBlock && ! persistedBlock, + innerBlockCount: getBlockCount( clientId ), }; }, [ ref, clientId ] @@ -122,13 +126,13 @@ export default function ReusableBlockEdit( { } const onConvertToRegularBlocks = useCallback( () => { - createSuccessNotice( - sprintf( - /* translators: %s: name of the reusable block */ - __( '%s converted to regular blocks' ), - title - ) - ); + const successNotice = + innerBlockCount > 1 + ? /* translators: %s: name of the reusable block */ + __( '%s converted to regular blocks' ) + : /* translators: %s: name of the reusable block */ + __( '%s converted to regular block' ); + createSuccessNotice( sprintf( successNotice, title ) ); clearSelectedBlock(); // Convert action is executed at the end of the current JavaScript execution block @@ -162,12 +166,20 @@ export default function ReusableBlockEdit( { { infoTitle } - { __( - 'Alternatively, you can detach and edit these blocks separately by tapping “Convert to regular blocks”.' - ) } + { innerBlockCount > 1 + ? __( + 'Alternatively, you can detach and edit these blocks separately by tapping “Convert to regular blocks”.' + ) + : __( + 'Alternatively, you can detach and edit this block separately by tapping “Convert to regular block”.' + ) } 1 + ? __( 'Convert to regular blocks' ) + : __( 'Convert to regular block' ) + } separatorType="topFullWidth" onPress={ onConvertToRegularBlocks } labelStyle={ actionButtonStyle } diff --git a/packages/block-library/src/block/editor.scss b/packages/block-library/src/block/editor.scss index b3eb1a67e9945..c49675392a525 100644 --- a/packages/block-library/src/block/editor.scss +++ b/packages/block-library/src/block/editor.scss @@ -14,3 +14,20 @@ display: none; } } + +.edit-post-visual-editor .block-editor-block-list__block:not(.remove-outline).is-reusable { + &.is-highlighted, + &.is-selected { + box-shadow: inset 0 0 0 var(--wp-admin-border-width-focus) var(--wp-block-synced-color); + } + + &.block-editor-block-list__block:not([contenteditable]):focus { + &::after { + box-shadow: 0 0 0 var(--wp-admin-border-width-focus) var(--wp-block-synced-color); + // Show a light color for dark themes. + .is-dark-theme & { + box-shadow: 0 0 0 var(--wp-admin-border-width-focus) $dark-theme-focus; + } + } + } +} diff --git a/packages/block-library/src/block/index.php b/packages/block-library/src/block/index.php index 2cfd672aa3ada..d51b35d68b23d 100644 --- a/packages/block-library/src/block/index.php +++ b/packages/block-library/src/block/index.php @@ -27,8 +27,7 @@ function render_block_core_block( $attributes ) { if ( isset( $seen_refs[ $attributes['ref'] ] ) ) { // WP_DEBUG_DISPLAY must only be honored when WP_DEBUG. This precedent // is set in `wp_debug_mode()`. - $is_debug = defined( 'WP_DEBUG' ) && WP_DEBUG && - defined( 'WP_DEBUG_DISPLAY' ) && WP_DEBUG_DISPLAY; + $is_debug = WP_DEBUG && WP_DEBUG_DISPLAY; return $is_debug ? // translators: Visible only in the front end, this warning takes the place of a faulty block. diff --git a/packages/block-library/src/block/test/edit.native.js b/packages/block-library/src/block/test/edit.native.js index a683426a751e2..4652f8ba20f38 100644 --- a/packages/block-library/src/block/test/edit.native.js +++ b/packages/block-library/src/block/test/edit.native.js @@ -5,7 +5,6 @@ import { getEditorHtml, initializeEditor, fireEvent, - waitFor, within, } from 'test/helpers'; @@ -24,7 +23,7 @@ const getMockedReusableBlock = ( id ) => ( { content: { raw: ` -

First Reusable block

+

First Reusable block

@@ -78,17 +77,16 @@ describe( 'Reusable block', () => { return Promise.resolve( response ); } ); - const { getByLabelText, getByTestId, getByText } = - await initializeEditor( { - initialHtml: '', - capabilities: { reusableBlock: true }, - } ); + const screen = await initializeEditor( { + initialHtml: '', + capabilities: { reusableBlock: true }, + } ); // Open the inserter menu. - fireEvent.press( await waitFor( () => getByLabelText( 'Add block' ) ) ); + fireEvent.press( await screen.findByLabelText( 'Add block' ) ); // Navigate to reusable tab. - const reusableSegment = await waitFor( () => getByText( 'Reusable' ) ); + const reusableSegment = await screen.findByText( 'Reusable' ); // onLayout event is required by Segment component. fireEvent( reusableSegment, 'layout', { nativeEvent: { @@ -99,7 +97,9 @@ describe( 'Reusable block', () => { } ); fireEvent.press( reusableSegment ); - const reusableBlockList = getByTestId( 'InserterUI-ReusableBlocks' ); + const reusableBlockList = screen.getByTestId( + 'InserterUI-ReusableBlocks' + ); // onScroll event used to force the FlatList to render all items. fireEvent.scroll( reusableBlockList, { nativeEvent: { @@ -110,13 +110,11 @@ describe( 'Reusable block', () => { } ); // Insert a reusable block. - fireEvent.press( - await waitFor( () => getByText( `Reusable block - 1` ) ) - ); + fireEvent.press( await screen.findByText( `Reusable block - 1` ) ); // Get the reusable block. - const reusableBlock = await waitFor( () => - getByLabelText( /Reusable block Block\. Row 1/ ) + const [ reusableBlock ] = await screen.findAllByLabelText( + /Reusable block Block\. Row 1/ ); expect( reusableBlock ).toBeDefined(); @@ -128,18 +126,16 @@ describe( 'Reusable block', () => { const id = 3; const initialHtml = ``; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); - const reusableBlock = await waitFor( () => - getByLabelText( /Reusable block Block\. Row 1/ ) + const [ reusableBlock ] = await screen.findAllByLabelText( + /Reusable block Block\. Row 1/ ); - const blockDeleted = await waitFor( () => - within( reusableBlock ).getByText( - 'Block has been deleted or is unavailable.' - ) + const blockDeleted = within( reusableBlock ).getByText( + 'Block has been deleted or is unavailable.' ); expect( reusableBlock ).toBeDefined(); @@ -163,17 +159,17 @@ describe( 'Reusable block', () => { return Promise.resolve( response ); } ); - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); - const reusableBlock = await waitFor( () => - getByLabelText( /Reusable block Block\. Row 1/ ) + const [ reusableBlock ] = await screen.findByLabelText( + /Reusable block Block\. Row 1/ ); - const innerBlockListWrapper = await waitFor( () => - within( reusableBlock ).getByTestId( 'block-list-wrapper' ) - ); + const innerBlockListWrapper = await within( + reusableBlock + ).findByTestId( 'block-list-wrapper' ); // onLayout event has to be explicitly dispatched in BlockList component, // otherwise the inner blocks are not rendered. @@ -185,10 +181,10 @@ describe( 'Reusable block', () => { }, } ); - const headingInnerBlock = await waitFor( () => - within( reusableBlock ).getByLabelText( - 'Heading Block. Row 1. Level 2. First Reusable block' - ) + const [ headingInnerBlock ] = await within( + reusableBlock + ).findAllByLabelText( + 'Heading Block. Row 1. Level 2. First Reusable block' ); expect( reusableBlock ).toBeDefined(); diff --git a/packages/block-library/src/button/edit.native.js b/packages/block-library/src/button/edit.native.js index 39082488f1f65..214be33c7df2e 100644 --- a/packages/block-library/src/button/edit.native.js +++ b/packages/block-library/src/button/edit.native.js @@ -31,6 +31,7 @@ import { useMobileGlobalStylesColors, } from '@wordpress/components'; import { link } from '@wordpress/icons'; +// eslint-disable-next-line no-restricted-imports import { store as editPostStore } from '@wordpress/edit-post'; /** diff --git a/packages/block-library/src/buttons/test/edit.native.js b/packages/block-library/src/buttons/test/edit.native.js index a6a25ee0590a9..9d23413fd79be 100644 --- a/packages/block-library/src/buttons/test/edit.native.js +++ b/packages/block-library/src/buttons/test/edit.native.js @@ -3,7 +3,6 @@ */ import { fireEvent, - waitFor, getEditorHtml, within, getBlock, @@ -50,20 +49,20 @@ describe( 'Buttons block', () => {
`; - const { getByLabelText } = await initializeEditor( { + const editor = await initializeEditor( { initialHtml, } ); - const buttonsBlock = await waitFor( () => - getByLabelText( /Buttons Block\. Row 1/ ) + const [ buttonsBlock ] = await editor.findAllByLabelText( + /Buttons Block\. Row 1/ ); fireEvent.press( buttonsBlock ); // onLayout event has to be explicitly dispatched in BlockList component, // otherwise the inner blocks are not rendered. - const innerBlockListWrapper = await waitFor( () => - within( buttonsBlock ).getByTestId( 'block-list-wrapper' ) - ); + const innerBlockListWrapper = await within( + buttonsBlock + ).findByTestId( 'block-list-wrapper' ); fireEvent( innerBlockListWrapper, 'layout', { nativeEvent: { layout: { @@ -72,22 +71,22 @@ describe( 'Buttons block', () => { }, } ); - const buttonInnerBlock = await waitFor( () => - within( buttonsBlock ).getByLabelText( /Button Block\. Row 1/ ) - ); + const [ buttonInnerBlock ] = await within( + buttonsBlock + ).findAllByLabelText( /Button Block\. Row 1/ ); fireEvent.press( buttonInnerBlock ); - const settingsButton = await waitFor( () => - getByLabelText( 'Open Settings' ) + const settingsButton = await editor.findByLabelText( + 'Open Settings' ); fireEvent.press( settingsButton ); - const radiusStepper = await waitFor( () => - getByLabelText( /Border Radius/ ) + const radiusStepper = await editor.findByLabelText( + /Border Radius/ ); - const incrementButton = await waitFor( () => - within( radiusStepper ).getByTestId( 'Increment' ) + const incrementButton = await within( radiusStepper ).findByTestId( + 'Increment' ); fireEvent( incrementButton, 'onPressIn' ); @@ -98,15 +97,14 @@ describe( 'Buttons block', () => { const screen = await initializeEditor( { initialHtml: BUTTONS_HTML, } ); - const { getByLabelText } = screen; // Get block const buttonsBlock = await getBlock( screen, 'Buttons' ); // Trigger inner blocks layout - const innerBlockListWrapper = await waitFor( () => - within( buttonsBlock ).getByTestId( 'block-list-wrapper' ) - ); + const innerBlockListWrapper = await within( + buttonsBlock + ).findByTestId( 'block-list-wrapper' ); fireEvent( innerBlockListWrapper, 'layout', { nativeEvent: { layout: { @@ -125,14 +123,14 @@ describe( 'Buttons block', () => { fireEvent.press( appenderButton ); // Check for new button - const secondButtonBlock = await waitFor( () => - within( buttonsBlock ).getByLabelText( /Button Block\. Row 2/ ) - ); + const [ secondButtonBlock ] = await within( + buttonsBlock + ).findAllByLabelText( /Button Block\. Row 2/ ); expect( secondButtonBlock ).toBeVisible(); // Add a Paragraph block using the empty placeholder at the bottom - const paragraphPlaceholder = await waitFor( () => - getByLabelText( 'Add paragraph block' ) + const paragraphPlaceholder = await screen.findByLabelText( + 'Add paragraph block' ); fireEvent.press( paragraphPlaceholder ); @@ -148,21 +146,15 @@ describe( 'Buttons block', () => { const screen = await initializeEditor( { initialHtml: BUTTONS_HTML, } ); - const { - getByLabelText, - getByTestId, - queryAllByLabelText, - getByText, - } = screen; // Get block const buttonsBlock = await getBlock( screen, 'Buttons' ); fireEvent.press( buttonsBlock ); // Trigger inner blocks layout - const innerBlockListWrapper = await waitFor( () => - within( buttonsBlock ).getByTestId( 'block-list-wrapper' ) - ); + const innerBlockListWrapper = await within( + buttonsBlock + ).findByTestId( 'block-list-wrapper' ); fireEvent( innerBlockListWrapper, 'layout', { nativeEvent: { layout: { @@ -176,9 +168,9 @@ describe( 'Buttons block', () => { fireEvent.press( buttonBlock ); // Open the block inserter - fireEvent.press( getByLabelText( 'Add block' ) ); + fireEvent.press( screen.getByLabelText( 'Add block' ) ); - const blockList = getByTestId( 'InserterUI-Blocks' ); + const blockList = screen.getByTestId( 'InserterUI-Blocks' ); // onScroll event used to force the FlatList to render all items fireEvent.scroll( blockList, { nativeEvent: { @@ -190,11 +182,11 @@ describe( 'Buttons block', () => { // Check the Add block here placeholder is not visible const addBlockHerePlaceholders = - queryAllByLabelText( 'ADD BLOCK HERE' ); + screen.queryAllByLabelText( 'ADD BLOCK HERE' ); expect( addBlockHerePlaceholders.length ).toBe( 0 ); // Add a new Button block - fireEvent.press( await waitFor( () => getByText( 'Button' ) ) ); + fireEvent.press( await screen.findByText( 'Button' ) ); // Get new button const secondButtonBlock = await getBlock( screen, 'Button', { @@ -214,15 +206,14 @@ describe( 'Buttons block', () => { const screen = await initializeEditor( { initialHtml: BUTTONS_HTML, } ); - const { getByLabelText } = screen; // Get block const buttonsBlock = await getBlock( screen, 'Buttons' ); // Trigger inner blocks layout - const innerBlockListWrapper = await waitFor( () => - within( buttonsBlock ).getByTestId( 'block-list-wrapper' ) - ); + const innerBlockListWrapper = await within( + buttonsBlock + ).findByTestId( 'block-list-wrapper' ); fireEvent( innerBlockListWrapper, 'layout', { nativeEvent: { layout: { @@ -236,13 +227,13 @@ describe( 'Buttons block', () => { fireEvent.press( buttonBlock ); // Open block actions menu - const blockActionsButton = getByLabelText( + const blockActionsButton = screen.getByLabelText( /Open Block Actions Menu/ ); fireEvent.press( blockActionsButton ); // Delete block - const deleteButton = getByLabelText( /Remove block/ ); + const deleteButton = screen.getByLabelText( /Remove block/ ); fireEvent.press( deleteButton ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -260,22 +251,20 @@ describe( 'Buttons block', () => { const initialHtml = `
`; - const { getByLabelText, getByText } = await initializeEditor( { - initialHtml, - } ); + const screen = await initializeEditor( { initialHtml } ); - const block = await waitFor( () => - getByLabelText( /Buttons Block\. Row 1/ ) + const [ block ] = await screen.findAllByLabelText( + /Buttons Block\. Row 1/ ); fireEvent.press( block ); fireEvent.press( - getByLabelText( 'Change items justification' ) + screen.getByLabelText( 'Change items justification' ) ); // Select alignment option. fireEvent.press( - await waitFor( () => getByText( justificationOption ) ) + await screen.findByText( justificationOption ) ); expect( getEditorHtml() ).toMatchSnapshot(); diff --git a/packages/block-library/src/code/transforms.js b/packages/block-library/src/code/transforms.js index b19d584fb7b05..fcfb41fb0262c 100644 --- a/packages/block-library/src/code/transforms.js +++ b/packages/block-library/src/code/transforms.js @@ -43,7 +43,9 @@ const transforms = { type: 'block', blocks: [ 'core/paragraph' ], transform: ( { content } ) => { - return createBlock( 'core/paragraph', { content } ); + return createBlock( 'core/paragraph', { + content: content.replace( /\n/g, '
' ), + } ); }, }, ], diff --git a/packages/block-library/src/columns/edit.js b/packages/block-library/src/columns/edit.js index 8879f0e52f2e7..b978c411107d5 100644 --- a/packages/block-library/src/columns/edit.js +++ b/packages/block-library/src/columns/edit.js @@ -95,6 +95,7 @@ function ColumnsEditContainer( { updateColumns( count, value ) } diff --git a/packages/block-library/src/columns/edit.native.js b/packages/block-library/src/columns/edit.native.js index 96855e3da17e4..6331097f96b99 100644 --- a/packages/block-library/src/columns/edit.native.js +++ b/packages/block-library/src/columns/edit.native.js @@ -2,7 +2,7 @@ * External dependencies */ import { View, Dimensions } from 'react-native'; -import { map } from 'lodash'; + /** * WordPress dependencies */ @@ -332,11 +332,6 @@ const ColumnsEditContainerWrapper = withDispatch( width: value, } ); }, - updateBlockSettings( settings ) { - const { clientId } = ownProps; - const { updateBlockListSettings } = dispatch( blockEditorStore ); - updateBlockListSettings( clientId, settings ); - }, /** * Updates the column columnCount, including necessary revisions to child Column * blocks to grant required or redistribute available space. @@ -450,7 +445,7 @@ const ColumnsEdit = ( props ) => { const { columnCount, isDefaultColumns, - innerWidths = [], + innerBlocks, hasParents, parentBlockAlignment, editorSidebarOpened, @@ -463,24 +458,20 @@ const ColumnsEdit = ( props ) => { getBlockAttributes, } = select( blockEditorStore ); const { isEditorSidebarOpened } = select( 'core/edit-post' ); - const innerBlocks = getBlocks( clientId ); - const isContentEmpty = map( - innerBlocks, - ( innerBlock ) => innerBlock.innerBlocks.length + const innerBlocksList = getBlocks( clientId ); + + const isContentEmpty = innerBlocksList.every( + ( innerBlock ) => innerBlock.innerBlocks.length === 0 ); - const innerColumnsWidths = innerBlocks.map( ( inn ) => ( { - clientId: inn.clientId, - attributes: { width: inn.attributes.width }, - } ) ); const parents = getBlockParents( clientId, true ); return { columnCount: getBlockCount( clientId ), - isDefaultColumns: ! isContentEmpty.filter( Boolean ).length, - innerWidths: innerColumnsWidths, - hasParents: !! parents.length, + isDefaultColumns: isContentEmpty, + innerBlocks: innerBlocksList, + hasParents: parents.length > 0, parentBlockAlignment: getBlockAttributes( parents[ 0 ] )?.align, editorSidebarOpened: isSelected && isEditorSidebarOpened(), }; @@ -488,13 +479,14 @@ const ColumnsEdit = ( props ) => { [ clientId, isSelected ] ); - const memoizedInnerWidths = useMemo( () => { - return innerWidths; - }, [ - // The JSON.stringify is used because innerWidth is always a new reference. - // The innerBlocks is a new reference after each attribute change of any nested block. - JSON.stringify( innerWidths ), - ] ); + const innerWidths = useMemo( + () => + innerBlocks.map( ( inn ) => ( { + clientId: inn.clientId, + attributes: { width: inn.attributes.width }, + } ) ), + [ innerBlocks ] + ); const [ isVisible, setIsVisible ] = useState( false ); @@ -514,7 +506,7 @@ const ColumnsEdit = ( props ) => { setAttributes( { diff --git a/packages/block-library/src/cover/edit.native.js b/packages/block-library/src/cover/edit.native.js index 900298354ded2..67bf968c3a8f4 100644 --- a/packages/block-library/src/cover/edit.native.js +++ b/packages/block-library/src/cover/edit.native.js @@ -58,6 +58,7 @@ import { } from '@wordpress/element'; import { cover as icon, replace, image, warning } from '@wordpress/icons'; import { getProtocol } from '@wordpress/url'; +// eslint-disable-next-line no-restricted-imports import { store as editPostStore } from '@wordpress/edit-post'; /** @@ -87,6 +88,36 @@ const INNER_BLOCKS_TEMPLATE = [ ], ]; +function useIsScreenReaderEnabled() { + const [ isScreenReaderEnabled, setIsScreenReaderEnabled ] = + useState( false ); + + useEffect( () => { + let mounted = true; + + const changeListener = AccessibilityInfo.addEventListener( + 'screenReaderChanged', + ( enabled ) => setIsScreenReaderEnabled( enabled ) + ); + + AccessibilityInfo.isScreenReaderEnabled().then( + ( screenReaderEnabled ) => { + if ( mounted && screenReaderEnabled ) { + setIsScreenReaderEnabled( screenReaderEnabled ); + } + } + ); + + return () => { + mounted = false; + + changeListener.remove(); + }; + }, [] ); + + return isScreenReaderEnabled; +} + const Cover = ( { attributes, getStylesFromColorScheme, @@ -117,29 +148,11 @@ const Cover = ( { overlayColor, isDark, } = attributes; - const [ isScreenReaderEnabled, setIsScreenReaderEnabled ] = - useState( false ); + const isScreenReaderEnabled = useIsScreenReaderEnabled(); useEffect( () => { - let isCurrent = true; - // Sync with local media store. mediaUploadSync(); - const a11yInfoChangeSubscription = AccessibilityInfo.addEventListener( - 'screenReaderChanged', - setIsScreenReaderEnabled - ); - - AccessibilityInfo.isScreenReaderEnabled().then( () => { - if ( isCurrent ) { - setIsScreenReaderEnabled(); - } - } ); - - return () => { - isCurrent = false; - a11yInfoChangeSubscription.remove(); - }; }, [] ); const convertedMinHeight = useConvertUnitToMobile( diff --git a/packages/block-library/src/cover/edit/inspector-controls.js b/packages/block-library/src/cover/edit/inspector-controls.js index cfcf012badd62..d3f65672e0932 100644 --- a/packages/block-library/src/cover/edit/inspector-controls.js +++ b/packages/block-library/src/cover/edit/inspector-controls.js @@ -162,6 +162,7 @@ export default function CoverInspectorControls( { ) } { showFocalPointPicker && ( diff --git a/packages/block-library/src/cover/test/edit.native.js b/packages/block-library/src/cover/test/edit.native.js index 10dbff032160a..6a5bef376c18f 100644 --- a/packages/block-library/src/cover/test/edit.native.js +++ b/packages/block-library/src/cover/test/edit.native.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { AccessibilityInfo, Image } from 'react-native'; +import { Image } from 'react-native'; import { getEditorHtml, initializeEditor, @@ -90,10 +90,6 @@ beforeAll( () => { const getSizeSpy = jest.spyOn( Image, 'getSize' ); getSizeSpy.mockImplementation( ( _url, callback ) => callback( 300, 200 ) ); - AccessibilityInfo.isScreenReaderEnabled.mockResolvedValue( - Promise.resolve( true ) - ); - // Register required blocks. paragraph.init(); cover.init(); @@ -103,7 +99,6 @@ beforeAll( () => { afterAll( () => { // Restore mocks. Image.getSize.mockRestore(); - AccessibilityInfo.isScreenReaderEnabled.mockReset(); // Clean up registered blocks. unregisterBlockType( paragraph.name ); @@ -134,32 +129,32 @@ describe( 'when no media is attached', () => { describe( 'when an image is attached', () => { it( 'edits the image', async () => { - const { getByLabelText, getByText } = render( + const screen = render( ); - fireEvent.press( getByLabelText( 'Edit image' ) ); - const editButton = await waitFor( () => getByText( 'Edit' ) ); + fireEvent.press( screen.getByLabelText( 'Edit image' ) ); + const editButton = await screen.findByText( 'Edit' ); fireEvent.press( editButton ); expect( requestMediaEditor ).toHaveBeenCalled(); } ); it( 'replaces the image', async () => { - const { getByLabelText, getByText } = render( + const screen = render( ); - fireEvent.press( getByLabelText( 'Edit image' ) ); - const replaceButton = await waitFor( () => getByText( 'Replace' ) ); + fireEvent.press( screen.getByLabelText( 'Edit image' ) ); + const replaceButton = await screen.findByText( 'Replace' ); fireEvent.press( replaceButton ); - const mediaLibraryButton = await waitFor( () => - getByText( 'WordPress Media Library' ) + const mediaLibraryButton = await screen.findByText( + 'WordPress Media Library' ); fireEvent.press( mediaLibraryButton ); @@ -167,17 +162,17 @@ describe( 'when an image is attached', () => { } ); it( 'clears the image within image edit button', async () => { - const { getByLabelText, getAllByText } = render( + const screen = render( ); - fireEvent.press( getByLabelText( 'Edit image' ) ); - const clearMediaButton = await waitFor( () => - getAllByText( 'Clear Media' ) + fireEvent.press( screen.getByLabelText( 'Edit image' ) ); + const [ clearMediaButton ] = await screen.findAllByText( + 'Clear Media' ); - fireEvent.press( clearMediaButton[ 0 ] ); + fireEvent.press( clearMediaButton ); expect( setAttributes ).toHaveBeenCalledWith( expect.objectContaining( { @@ -209,22 +204,22 @@ describe( 'when an image is attached', () => { } ); it( 'edits the focal point with a slider', async () => { - const { getByText, getByLabelText, getByTestId } = render( + const screen = render( ); - const editFocalPointButton = await waitFor( () => - getByText( 'Edit focal point' ) + const editFocalPointButton = await screen.findByText( + 'Edit focal point' ); fireEvent.press( editFocalPointButton ); fireEvent( - getByTestId( 'Slider Y-Axis Position' ), + screen.getByTestId( 'Slider Y-Axis Position' ), 'valueChange', '52' ); - fireEvent.press( getByLabelText( 'Apply' ) ); + fireEvent.press( screen.getByLabelText( 'Apply' ) ); expect( setAttributes ).toHaveBeenCalledWith( expect.objectContaining( { @@ -234,21 +229,24 @@ describe( 'when an image is attached', () => { } ); it( 'edits the focal point with a text input', async () => { - const { getByText, getByLabelText } = render( + const screen = render( ); - const editFocalPointButton = await waitFor( () => - getByText( 'Edit focal point' ) + const editFocalPointButton = await screen.findByText( + 'Edit focal point' ); fireEvent.press( editFocalPointButton ); fireEvent.press( - getByText( ( attributes.focalPoint.x * 100 ).toString() ) + screen.getByText( ( attributes.focalPoint.x * 100 ).toString() ) ); - fireEvent.changeText( getByLabelText( 'X-Axis Position' ), '99' ); - fireEvent.press( getByLabelText( 'Apply' ) ); + fireEvent.changeText( + screen.getByLabelText( 'X-Axis Position' ), + '99' + ); + fireEvent.press( screen.getByLabelText( 'Apply' ) ); expect( setAttributes ).toHaveBeenCalledWith( expect.objectContaining( { @@ -282,15 +280,13 @@ describe( 'when an image is attached', () => { } ); it( 'clears the media within cell button', async () => { - const { getByText } = render( + const screen = render( ); - const clearMediaButton = await waitFor( () => - getByText( 'Clear Media' ) - ); + const clearMediaButton = await screen.findByText( 'Clear Media' ); fireEvent.press( clearMediaButton ); expect( setAttributes ).toHaveBeenCalledWith( @@ -334,97 +330,79 @@ describe( 'when an image is attached', () => { describe( 'color settings', () => { it( 'sets a color for the overlay background when the placeholder is visible', async () => { - const { getByTestId, getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml: COVER_BLOCK_PLACEHOLDER_HTML, } ); - const block = await waitFor( () => - getByLabelText( 'Cover block. Empty' ) - ); + const block = await screen.findByLabelText( 'Cover block. Empty' ); expect( block ).toBeDefined(); // Select a color from the placeholder palette. - const colorPalette = await waitFor( () => - getByTestId( 'color-palette' ) - ); + const colorPalette = await screen.findByTestId( 'color-palette' ); const colorButton = within( colorPalette ).getByTestId( COLOR_PINK ); expect( colorButton ).toBeDefined(); fireEvent.press( colorButton ); // Wait for the block to be created. - const coverBlockWithOverlay = await waitFor( () => - getByLabelText( /Cover Block\. Row 1/ ) + const [ coverBlockWithOverlay ] = await screen.findAllByLabelText( + /Cover Block\. Row 1/ ); fireEvent.press( coverBlockWithOverlay ); // Open Block Settings. - const settingsButton = await waitFor( () => - getByLabelText( 'Open Settings' ) - ); + const settingsButton = await screen.findByLabelText( 'Open Settings' ); fireEvent.press( settingsButton ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = screen.getByTestId( 'block-settings-modal' ); await waitFor( () => blockSettingsModal.props.isVisible ); // Open the overlay color settings. - const colorOverlay = await waitFor( () => - getByLabelText( 'Color. Empty' ) - ); + const colorOverlay = await screen.findByLabelText( 'Color. Empty' ); expect( colorOverlay ).toBeDefined(); fireEvent.press( colorOverlay ); // Find the selected color. - const colorPaletteButton = await waitFor( () => - getByTestId( COLOR_PINK ) - ); + const colorPaletteButton = await screen.findByTestId( COLOR_PINK ); expect( colorPaletteButton ).toBeDefined(); // Select another color. - const newColorButton = await waitFor( () => getByTestId( COLOR_RED ) ); + const newColorButton = await screen.findByTestId( COLOR_RED ); fireEvent.press( newColorButton ); expect( getEditorHtml() ).toMatchSnapshot(); } ); it( 'sets a gradient overlay background when a solid background was already selected', async () => { - const { getByTestId, getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml: COVER_BLOCK_SOLID_COLOR_HTML, } ); // Wait for the block to be created. - const coverBlock = await waitFor( () => - getByLabelText( /Cover Block\. Row 1/ ) + const [ coverBlock ] = await screen.findAllByLabelText( + /Cover Block\. Row 1/ ); - expect( coverBlock ).toBeDefined(); fireEvent.press( coverBlock ); // Open Block Settings. - const settingsButton = await waitFor( () => - getByLabelText( 'Open Settings' ) - ); + const settingsButton = await screen.findByLabelText( 'Open Settings' ); fireEvent.press( settingsButton ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = screen.getByTestId( 'block-settings-modal' ); await waitFor( () => blockSettingsModal.props.isVisible ); // Open the overlay color settings. - const colorOverlay = await waitFor( () => - getByLabelText( 'Color. Empty' ) - ); - expect( colorOverlay ).toBeDefined(); + const colorOverlay = await screen.findByLabelText( 'Color. Empty' ); fireEvent.press( colorOverlay ); // Find the selected color. - const colorButton = await waitFor( () => getByTestId( COLOR_GRAY ) ); + const colorButton = await screen.findByTestId( COLOR_GRAY ); expect( colorButton ).toBeDefined(); // Open the gradients. - const gradientsButton = await waitFor( () => - getByLabelText( 'Gradient' ) - ); + const gradientsButton = await screen.findByLabelText( 'Gradient' ); expect( gradientsButton ).toBeDefined(); fireEvent( gradientsButton, 'layout', { @@ -433,10 +411,7 @@ describe( 'color settings', () => { fireEvent.press( gradientsButton ); // Find the gradient color. - const newGradientButton = await waitFor( () => - getByTestId( GRADIENT_GREEN ) - ); - expect( newGradientButton ).toBeDefined(); + const newGradientButton = await screen.findByTestId( GRADIENT_GREEN ); fireEvent.press( newGradientButton ); // Dismiss the Block Settings modal. @@ -446,62 +421,48 @@ describe( 'color settings', () => { } ); it( 'toggles between solid colors and gradients', async () => { - const { getByTestId, getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml: COVER_BLOCK_PLACEHOLDER_HTML, } ); - const block = await waitFor( () => - getByLabelText( 'Cover block. Empty' ) - ); + const block = await screen.findByLabelText( 'Cover block. Empty' ); expect( block ).toBeDefined(); // Select a color from the placeholder palette. - const colorPalette = await waitFor( () => - getByTestId( 'color-palette' ) - ); + const colorPalette = await screen.findByTestId( 'color-palette' ); const colorButton = within( colorPalette ).getByTestId( COLOR_PINK ); expect( colorButton ).toBeDefined(); fireEvent.press( colorButton ); // Wait for the block to be created. - const coverBlockWithOverlay = await waitFor( () => - getByLabelText( /Cover Block\. Row 1/ ) + const [ coverBlockWithOverlay ] = await screen.findAllByLabelText( + /Cover Block\. Row 1/ ); fireEvent.press( coverBlockWithOverlay ); // Open Block Settings. - const settingsButton = await waitFor( () => - getByLabelText( 'Open Settings' ) - ); + const settingsButton = await screen.findByLabelText( 'Open Settings' ); fireEvent.press( settingsButton ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = screen.getByTestId( 'block-settings-modal' ); await waitFor( () => blockSettingsModal.props.isVisible ); // Open the overlay color settings. - const colorOverlay = await waitFor( () => - getByLabelText( 'Color. Empty' ) - ); - expect( colorOverlay ).toBeDefined(); + const colorOverlay = await screen.findByLabelText( 'Color. Empty' ); fireEvent.press( colorOverlay ); // Find the selected color. - const colorPaletteButton = await waitFor( () => - getByTestId( COLOR_PINK ) - ); + const colorPaletteButton = await screen.findByTestId( COLOR_PINK ); expect( colorPaletteButton ).toBeDefined(); // Select another color. - const newColorButton = await waitFor( () => getByTestId( COLOR_RED ) ); + const newColorButton = await screen.findByTestId( COLOR_RED ); fireEvent.press( newColorButton ); // Open the gradients. - const gradientsButton = await waitFor( () => - getByLabelText( 'Gradient' ) - ); - expect( gradientsButton ).toBeDefined(); + const gradientsButton = await screen.findByLabelText( 'Gradient' ); fireEvent( gradientsButton, 'layout', { nativeEvent: { layout: { width: 80, height: 26 } }, @@ -509,20 +470,14 @@ describe( 'color settings', () => { fireEvent.press( gradientsButton ); // Find the gradient color. - const newGradientButton = await waitFor( () => - getByTestId( GRADIENT_GREEN ) - ); - expect( newGradientButton ).toBeDefined(); + const newGradientButton = await screen.findByTestId( GRADIENT_GREEN ); fireEvent.press( newGradientButton ); // Go back to the settings list. - fireEvent.press( await waitFor( () => getByLabelText( 'Go back' ) ) ); + fireEvent.press( await screen.findByLabelText( 'Go back' ) ); // Find the color setting. - const colorSetting = await waitFor( () => - getByLabelText( 'Color. Empty' ) - ); - expect( colorSetting ).toBeDefined(); + const colorSetting = await screen.findByLabelText( 'Color. Empty' ); fireEvent.press( colorSetting ); // Dismiss the Block Settings modal. @@ -532,42 +487,34 @@ describe( 'color settings', () => { } ); it( 'clears the selected overlay color and mantains the inner blocks', async () => { - const { getByTestId, getByLabelText, getByText } = - await initializeEditor( { - initialHtml: COVER_BLOCK_SOLID_COLOR_HTML, - } ); + const screen = await initializeEditor( { + initialHtml: COVER_BLOCK_SOLID_COLOR_HTML, + } ); // Wait for the block to be created. - const coverBlock = await waitFor( () => - getByLabelText( /Cover Block\. Row 1/ ) + const [ coverBlock ] = await screen.findAllByLabelText( + /Cover Block\. Row 1/ ); - expect( coverBlock ).toBeDefined(); fireEvent.press( coverBlock ); // Open Block Settings. - const settingsButton = await waitFor( () => - getByLabelText( 'Open Settings' ) - ); + const settingsButton = await screen.findByLabelText( 'Open Settings' ); fireEvent.press( settingsButton ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = screen.getByTestId( 'block-settings-modal' ); await waitFor( () => blockSettingsModal.props.isVisible ); // Open the overlay color settings. - const colorOverlay = await waitFor( () => - getByLabelText( 'Color. Empty' ) - ); - expect( colorOverlay ).toBeDefined(); + const colorOverlay = await screen.findByLabelText( 'Color. Empty' ); fireEvent.press( colorOverlay ); // Find the selected color. - const colorButton = await waitFor( () => getByTestId( COLOR_GRAY ) ); + const colorButton = await screen.findByTestId( COLOR_GRAY ); expect( colorButton ).toBeDefined(); // Reset the selected color. - const resetButton = await waitFor( () => getByText( 'Reset' ) ); - expect( resetButton ).toBeDefined(); + const resetButton = await screen.findByText( 'Reset' ); fireEvent.press( resetButton ); expect( getEditorHtml() ).toMatchSnapshot(); diff --git a/packages/block-library/src/embed/embed-controls.native.js b/packages/block-library/src/embed/embed-controls.native.js index b1249308ebdf5..524f2da619bcb 100644 --- a/packages/block-library/src/embed/embed-controls.native.js +++ b/packages/block-library/src/embed/embed-controls.native.js @@ -9,6 +9,7 @@ import { __ } from '@wordpress/i18n'; import { PanelBody, ToggleControl } from '@wordpress/components'; import { InspectorControls } from '@wordpress/block-editor'; import { useDispatch } from '@wordpress/data'; +// eslint-disable-next-line no-restricted-imports import { store as editPostStore } from '@wordpress/edit-post'; function getResponsiveHelp( checked ) { diff --git a/packages/block-library/src/embed/embed-preview.native.js b/packages/block-library/src/embed/embed-preview.native.js index bb8df663baf28..87abc38348d56 100644 --- a/packages/block-library/src/embed/embed-preview.native.js +++ b/packages/block-library/src/embed/embed-preview.native.js @@ -2,7 +2,6 @@ * External dependencies */ import { TouchableWithoutFeedback } from 'react-native'; -import { isEmpty } from 'lodash'; import classnames from 'classnames/dedupe'; /** @@ -52,7 +51,7 @@ const EmbedPreview = ( { styles[ `embed-preview__sandbox--align-${ align }` ]; function accessibilityLabelCreator( caption ) { - return isEmpty( caption ) + return ! caption ? /* translators: accessibility text. Empty Embed caption. */ __( 'Embed caption. Empty' ) : sprintf( diff --git a/packages/block-library/src/embed/test/index.native.js b/packages/block-library/src/embed/test/index.native.js index 60bc6604b8cff..d4b7fa99ded83 100644 --- a/packages/block-library/src/embed/test/index.native.js +++ b/packages/block-library/src/embed/test/index.native.js @@ -29,6 +29,7 @@ import { requestPreview } from '@wordpress/react-native-bridge'; */ import * as paragraph from '../../paragraph'; import * as embed from '..'; +import { WebView } from 'react-native-webview'; // Override modal mock to prevent unmounting it when is not visible. // This is required to be able to trigger onClose and onDismiss events when @@ -121,55 +122,54 @@ const MOST_USED_PROVIDERS = embed.settings.variations.filter( ( { name } ) => // Return specified mocked responses for the oembed endpoint. const mockEmbedResponses = ( mockedResponses ) => { - fetchRequest.mockImplementation( ( { path } ) => { - if ( path.startsWith( '/wp/v2/themes' ) ) { - return Promise.resolve( [ - { theme_supports: { 'responsive-embeds': true } }, - ] ); - } - - if ( path.startsWith( '/wp/v2/block-patterns/categories' ) ) { - return Promise.resolve( [] ); - } - + fetchRequest.mockImplementation( async ( req ) => { const matchedEmbedResponse = mockedResponses.find( ( mockedResponse ) => - path === + req.path === `/oembed/1.0/proxy?url=${ encodeURIComponent( mockedResponse.url ) }` ); - return Promise.resolve( matchedEmbedResponse || {} ); + + return matchedEmbedResponse || mockOtherResponses( req ); } ); }; +async function mockOtherResponses( { path } ) { + if ( path.startsWith( '/wp/v2/themes' ) ) { + return [ { theme_supports: { 'responsive-embeds': true } } ]; + } + + if ( path.startsWith( '/wp/v2/block-patterns/patterns' ) ) { + return []; + } + + if ( path.startsWith( '/wp/v2/block-patterns/categories' ) ) { + return []; + } + + return {}; +} + const insertEmbedBlock = async ( blockTitle = 'Embed' ) => { - const editor = await initializeEditor( { - initialHtml: '', - } ); - const { getByLabelText, getByText } = editor; + const editor = await initializeEditor( { initialHtml: '' } ); // Open inserter menu. - fireEvent.press( await waitFor( () => getByLabelText( 'Add block' ) ) ); + fireEvent.press( await editor.findByLabelText( 'Add block' ) ); // Insert embed block. - fireEvent.press( await waitFor( () => getByText( blockTitle ) ) ); + fireEvent.press( await editor.findByText( blockTitle ) ); // Return the embed block. - const block = await waitFor( () => - getByLabelText( /Embed Block\. Row 1/ ) - ); + const [ block ] = await editor.findAllByLabelText( /Embed Block\. Row 1/ ); return { ...editor, block }; }; const initializeWithEmbedBlock = async ( initialHtml, selectBlock = true ) => { const editor = await initializeEditor( { initialHtml } ); - const { getByLabelText } = editor; - const block = await waitFor( () => - getByLabelText( /Embed Block\. Row 1/ ) - ); + const [ block ] = await editor.findAllByLabelText( /Embed Block\. Row 1/ ); if ( selectBlock ) { // Select block. @@ -232,10 +232,12 @@ describe( 'Embed block', () => { describe( 'set URL upon block insertion', () => { it( 'sets empty URL when dismissing edit URL modal', async () => { - const { getByTestId } = await insertEmbedBlock(); + const editor = await insertEmbedBlock(); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Dismiss the edit URL modal. @@ -248,15 +250,16 @@ describe( 'Embed block', () => { it( 'sets a valid URL when dismissing edit URL modal', async () => { const expectedURL = 'https://twitter.com/notnownikki'; - const { getByPlaceholderText, getByTestId } = - await insertEmbedBlock(); + const editor = await insertEmbedBlock(); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Set an URL. - const linkTextInput = getByPlaceholderText( 'Add link' ); + const linkTextInput = editor.getByPlaceholderText( 'Add link' ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, expectedURL ); @@ -264,8 +267,13 @@ describe( 'Embed block', () => { fireEvent( embedEditURLModal, 'backdropPress' ); fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); - const blockSettingsModal = await waitFor( () => - getByTestId( 'block-settings-modal' ) + // Wait until the WebView with the rich preview appears + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + // Wait until responsiveness settings appear, driven by `theme_supports.responsive-embeds` + await editor.findByText( 'Media settings' ); + + const blockSettingsModal = await editor.findByTestId( + 'block-settings-modal' ); // Get Twitter link field. const twitterLinkField = within( @@ -282,23 +290,26 @@ describe( 'Embed block', () => { // Mock clipboard. Clipboard.getString.mockResolvedValue( clipboardURL ); - const { getByTestId, getByText } = await insertEmbedBlock(); + const editor = await insertEmbedBlock(); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Get embed link with auto-pasted URL. - const autopastedLinkField = await waitFor( () => - getByText( clipboardURL ) - ); + const autopastedLinkField = await editor.findByText( clipboardURL ); // Dismiss the edit URL modal. fireEvent( embedEditURLModal, 'backdropPress' ); fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); - const blockSettingsModal = await waitFor( () => - getByTestId( 'block-settings-modal' ) + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + await editor.findByText( 'Media settings' ); + + const blockSettingsModal = await editor.findByTestId( + 'block-settings-modal' ); // Get Twitter link field. const twitterLinkField = within( @@ -315,15 +326,15 @@ describe( 'Embed block', () => { describe( 'set URL when empty block', () => { it( 'sets empty URL when dismissing edit URL modal', async () => { - const { getByTestId, getByText } = await initializeWithEmbedBlock( - EMPTY_EMBED_HTML - ); + const editor = await initializeWithEmbedBlock( EMPTY_EMBED_HTML ); // Edit URL. - fireEvent.press( await waitFor( () => getByText( 'ADD LINK' ) ) ); + fireEvent.press( await editor.findByText( 'ADD LINK' ) ); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Dismiss the edit URL modal. @@ -336,18 +347,19 @@ describe( 'Embed block', () => { it( 'sets a valid URL when dismissing edit URL modal', async () => { const expectedURL = 'https://twitter.com/notnownikki'; - const { getByPlaceholderText, getByTestId, getByText } = - await initializeWithEmbedBlock( EMPTY_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( EMPTY_EMBED_HTML ); // Edit URL. - fireEvent.press( getByText( 'ADD LINK' ) ); + fireEvent.press( editor.getByText( 'ADD LINK' ) ); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Set an URL. - const linkTextInput = getByPlaceholderText( 'Add link' ); + const linkTextInput = editor.getByPlaceholderText( 'Add link' ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, expectedURL ); @@ -355,8 +367,11 @@ describe( 'Embed block', () => { fireEvent( embedEditURLModal, 'backdropPress' ); fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); - const blockSettingsModal = await waitFor( () => - getByTestId( 'block-settings-modal' ) + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + await editor.findByText( 'Media settings' ); + + const blockSettingsModal = await editor.findByTestId( + 'block-settings-modal' ); // Get Twitter link field. const twitterLinkField = within( @@ -373,26 +388,29 @@ describe( 'Embed block', () => { // Mock clipboard. Clipboard.getString.mockResolvedValue( clipboardURL ); - const { getByTestId, getByText } = await initializeWithEmbedBlock( - EMPTY_EMBED_HTML - ); + const editor = await initializeWithEmbedBlock( EMPTY_EMBED_HTML ); // Edit URL. - fireEvent.press( getByText( 'ADD LINK' ) ); + fireEvent.press( editor.getByText( 'ADD LINK' ) ); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Get embed link. - const embedLink = await waitFor( () => getByText( clipboardURL ) ); + const embedLink = await editor.findByText( clipboardURL ); // Dismiss the edit URL modal. fireEvent( embedEditURLModal, 'backdropPress' ); fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); - const blockSettingsModal = await waitFor( () => - getByTestId( 'block-settings-modal' ) + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + await editor.findByText( 'Media settings' ); + + const blockSettingsModal = await editor.findByTestId( + 'block-settings-modal' ); // Get Twitter link field. const twitterLinkField = within( @@ -409,16 +427,17 @@ describe( 'Embed block', () => { describe( 'edit URL', () => { it( 'keeps the previous URL if no URL is set', async () => { - const { getByLabelText, getByTestId } = - await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( + RICH_TEXT_EMBED_HTML + ); // Open Block Settings. - fireEvent.press( - await waitFor( () => getByLabelText( 'Open Settings' ) ) - ); + fireEvent.press( await editor.findByLabelText( 'Open Settings' ) ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = editor.getByTestId( + 'block-settings-modal' + ); await waitFor( () => blockSettingsModal.props.isVisible ); // Dismiss the Block Settings modal. @@ -432,16 +451,17 @@ describe( 'Embed block', () => { const initialURL = 'https://twitter.com/notnownikki'; const expectedURL = 'https://www.youtube.com/watch?v=lXMskKTw3Bc'; - const { getByLabelText, getByDisplayValue, getByTestId } = - await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( + RICH_TEXT_EMBED_HTML + ); // Open Block Settings. - fireEvent.press( - await waitFor( () => getByLabelText( 'Open Settings' ) ) - ); + fireEvent.press( await editor.findByLabelText( 'Open Settings' ) ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = editor.getByTestId( + 'block-settings-modal' + ); await waitFor( () => blockSettingsModal.props.isVisible ); // Start editing link. @@ -452,7 +472,7 @@ describe( 'Embed block', () => { ); // Replace URL. - const linkTextInput = getByDisplayValue( initialURL ); + const linkTextInput = editor.getByDisplayValue( initialURL ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, expectedURL ); @@ -460,12 +480,13 @@ describe( 'Embed block', () => { fireEvent( blockSettingsModal, 'backdropPress' ); fireEvent( blockSettingsModal, MODAL_DISMISS_EVENT ); + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + await editor.findByText( 'Media settings' ); + // Get YouTube link field. - const youtubeLinkField = await waitFor( () => - within( blockSettingsModal ).getByLabelText( - `YouTube link, ${ expectedURL }` - ) - ); + const youtubeLinkField = await within( + blockSettingsModal + ).findByLabelText( `YouTube link, ${ expectedURL }` ); expect( youtubeLinkField ).toBeDefined(); expect( getEditorHtml() ).toMatchSnapshot(); @@ -475,20 +496,17 @@ describe( 'Embed block', () => { const previousURL = 'https://twitter.com/notnownikki'; const invalidURL = 'http://'; - const { - getByLabelText, - getByDisplayValue, - getByTestId, - getByText, - } = await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( + RICH_TEXT_EMBED_HTML + ); // Open Block Settings. - fireEvent.press( - await waitFor( () => getByLabelText( 'Open Settings' ) ) - ); + fireEvent.press( await editor.findByLabelText( 'Open Settings' ) ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = editor.getByTestId( + 'block-settings-modal' + ); await waitFor( () => blockSettingsModal.props.isVisible ); // Start editing link. @@ -499,7 +517,7 @@ describe( 'Embed block', () => { ); // Replace URL. - const linkTextInput = getByDisplayValue( previousURL ); + const linkTextInput = editor.getByDisplayValue( previousURL ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, invalidURL ); @@ -507,8 +525,8 @@ describe( 'Embed block', () => { fireEvent( blockSettingsModal, 'backdropPress' ); fireEvent( blockSettingsModal, MODAL_DISMISS_EVENT ); - const errorNotice = await waitFor( () => - getByText( 'Invalid URL. Please enter a valid URL.' ) + const errorNotice = await editor.findByText( + 'Invalid URL. Please enter a valid URL.' ); expect( errorNotice ).toBeDefined(); @@ -518,20 +536,17 @@ describe( 'Embed block', () => { it( 'sets empty state when setting an empty URL', async () => { const previousURL = 'https://twitter.com/notnownikki'; - const { - getByLabelText, - getByDisplayValue, - getByTestId, - getByPlaceholderText, - } = await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( + RICH_TEXT_EMBED_HTML + ); // Open Block Settings. - fireEvent.press( - await waitFor( () => getByLabelText( 'Open Settings' ) ) - ); + fireEvent.press( await editor.findByLabelText( 'Open Settings' ) ); // Get Block Settings modal. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = editor.getByTestId( + 'block-settings-modal' + ); // Start editing link. fireEvent.press( @@ -541,7 +556,7 @@ describe( 'Embed block', () => { ); // Replace URL with empty value. - const linkTextInput = getByDisplayValue( previousURL ); + const linkTextInput = editor.getByDisplayValue( previousURL ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, '' ); @@ -550,8 +565,8 @@ describe( 'Embed block', () => { fireEvent( blockSettingsModal, MODAL_DISMISS_EVENT ); // Get empty embed link. - const emptyLinkTextInput = await waitFor( () => - getByPlaceholderText( 'Add link' ) + const emptyLinkTextInput = await editor.findByPlaceholderText( + 'Add link' ); expect( emptyLinkTextInput ).toBeDefined(); @@ -560,10 +575,12 @@ describe( 'Embed block', () => { // This test case covers the bug fixed in PR #35460. it( 'edits URL after dismissing two times the edit URL bottom sheet with empty value', async () => { - const { block, getByTestId, getByText } = await insertEmbedBlock(); + const editor = await insertEmbedBlock(); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Dismiss the edit URL modal. @@ -571,10 +588,10 @@ describe( 'Embed block', () => { fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); // Select block. - fireEvent.press( block ); + fireEvent.press( editor.block ); // Edit URL. - fireEvent.press( getByText( 'ADD LINK' ) ); + fireEvent.press( editor.getByText( 'ADD LINK' ) ); // Wait for edit URL modal to be visible. await waitFor( () => embedEditURLModal.props.isVisible ); @@ -584,7 +601,7 @@ describe( 'Embed block', () => { fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); // Edit URL. - fireEvent.press( getByText( 'ADD LINK' ) ); + fireEvent.press( editor.getByText( 'ADD LINK' ) ); // Wait for edit URL modal to be visible. const isVisibleThirdTime = await waitFor( @@ -599,19 +616,16 @@ describe( 'Embed block', () => { const badURL = 'https://youtu.be/BAD_URL'; const expectedURL = 'https://twitter.com/notnownikki'; - const { - getByLabelText, - getByDisplayValue, - getByPlaceholderText, - getByTestId, - } = await insertEmbedBlock(); + const editor = await insertEmbedBlock(); // Wait for edit URL modal to be visible. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); await waitFor( () => embedEditURLModal.props.isVisible ); // Set an bad URL. - let linkTextInput = getByPlaceholderText( 'Add link' ); + let linkTextInput = editor.getByPlaceholderText( 'Add link' ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, badURL ); @@ -620,12 +634,12 @@ describe( 'Embed block', () => { fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); // Open Block Settings. - fireEvent.press( - await waitFor( () => getByLabelText( 'Open Settings' ) ) - ); + fireEvent.press( await editor.findByLabelText( 'Open Settings' ) ); // Wait for Block Settings to be visible. - const blockSettingsModal = getByTestId( 'block-settings-modal' ); + const blockSettingsModal = editor.getByTestId( + 'block-settings-modal' + ); await waitFor( () => blockSettingsModal.props.isVisible ); // Start editing link. @@ -636,7 +650,7 @@ describe( 'Embed block', () => { ); // Replace URL. - linkTextInput = getByDisplayValue( badURL ); + linkTextInput = editor.getByDisplayValue( badURL ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, expectedURL ); @@ -645,11 +659,9 @@ describe( 'Embed block', () => { fireEvent( blockSettingsModal, MODAL_DISMISS_EVENT ); // Get Twitter link field. - const twitterLinkField = await waitFor( () => - within( blockSettingsModal ).getByLabelText( - `Twitter link, ${ expectedURL }` - ) - ); + const twitterLinkField = await within( + blockSettingsModal + ).findByLabelText( `Twitter link, ${ expectedURL }` ); expect( twitterLinkField ).toBeDefined(); expect( getEditorHtml() ).toMatchSnapshot(); @@ -665,18 +677,15 @@ describe( 'Embed block', () => { 'Full width', ].forEach( ( alignmentOption ) => it( `sets ${ alignmentOption } option`, async () => { - const { getByLabelText, getByText } = - await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( + RICH_TEXT_EMBED_HTML + ); // Open alignment options. - fireEvent.press( - await waitFor( () => getByLabelText( 'Align' ) ) - ); + fireEvent.press( await editor.findByLabelText( 'Align' ) ); // Select alignment option. - fireEvent.press( - await waitFor( () => getByText( alignmentOption ) ) - ); + fireEvent.press( await editor.findByText( alignmentOption ) ); expect( getEditorHtml() ).toMatchSnapshot(); } ) @@ -690,33 +699,33 @@ describe( 'Embed block', () => { // Return bad response for the first request to oembed endpoint // and success response for the rest of requests. let isFirstEmbedRequest = true; - fetchRequest.mockImplementation( ( { path } ) => { - let response = {}; - const isEmbedRequest = path.startsWith( '/oembed/1.0/proxy' ); - if ( isEmbedRequest ) { + fetchRequest.mockImplementation( async ( req ) => { + if ( req.path.startsWith( '/oembed/1.0/proxy' ) ) { if ( isFirstEmbedRequest ) { isFirstEmbedRequest = false; - response = MOCK_BAD_WORDPRESS_RESPONSE; - } else { - response = RICH_TEXT_EMBED_SUCCESS_RESPONSE; + return MOCK_BAD_WORDPRESS_RESPONSE; } + return RICH_TEXT_EMBED_SUCCESS_RESPONSE; } - if ( path.startsWith( '/wp/v2/block-patterns/categories' ) ) { - response = []; - } - return Promise.resolve( response ); + + return mockOtherResponses( req ); } ); - const { getByTestId, getByText } = await initializeWithEmbedBlock( + const editor = await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + await editor.findByText( 'Unable to embed media' ); + // Retry request. - fireEvent.press( getByText( 'More options' ) ); - fireEvent.press( getByText( 'Retry' ) ); + fireEvent.press( editor.getByText( 'More options' ) ); + fireEvent.press( editor.getByText( 'Retry' ) ); + + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + await editor.findByText( 'Media settings' ); - const blockSettingsModal = await waitFor( () => - getByTestId( 'block-settings-modal' ) + const blockSettingsModal = await editor.findByTestId( + 'block-settings-modal' ); // Get Twitter link field. const twitterLinkField = within( @@ -729,26 +738,25 @@ describe( 'Embed block', () => { it( 'converts to link if preview request failed', async () => { // Return bad response for requests to oembed endpoint. - fetchRequest.mockImplementation( ( { path } ) => { - if ( path.startsWith( '/wp/v2/block-patterns/categories' ) ) { - return Promise.resolve( [] ); + fetchRequest.mockImplementation( async ( req ) => { + if ( req.path.startsWith( '/oembed/1.0/proxy' ) ) { + return MOCK_BAD_WORDPRESS_RESPONSE; } - const isEmbedRequest = path.startsWith( '/oembed/1.0/proxy' ); - return Promise.resolve( - isEmbedRequest ? MOCK_BAD_WORDPRESS_RESPONSE : {} - ); + + return mockOtherResponses( req ); } ); - const { getByLabelText, getByText } = - await initializeWithEmbedBlock( RICH_TEXT_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( + RICH_TEXT_EMBED_HTML + ); // Convert embed to link. - fireEvent.press( getByText( 'More options' ) ); - fireEvent.press( getByText( 'Convert to link' ) ); + fireEvent.press( editor.getByText( 'More options' ) ); + fireEvent.press( editor.getByText( 'Convert to link' ) ); // Get paragraph block where the link is created. - const paragraphBlock = await waitFor( () => - getByLabelText( /Paragraph Block\. Row 1/ ) + const [ paragraphBlock ] = await editor.findAllByLabelText( + /Paragraph Block\. Row 1/ ); expect( paragraphBlock ).toBeDefined(); @@ -760,50 +768,48 @@ describe( 'Embed block', () => { const successURL = 'https://twitter.com/notnownikki'; // Return bad response for WordPress URL and success for Twitter URL. - fetchRequest.mockImplementation( ( { path } ) => { + fetchRequest.mockImplementation( async ( req ) => { const matchesPath = ( url ) => - path === + req.path === `/oembed/1.0/proxy?url=${ encodeURIComponent( url ) }`; - let response = {}; if ( matchesPath( failURL ) ) { - response = MOCK_BAD_WORDPRESS_RESPONSE; - } else if ( matchesPath( successURL ) ) { - response = RICH_TEXT_EMBED_SUCCESS_RESPONSE; - } else if ( - path.startsWith( '/wp/v2/block-patterns/categories' ) - ) { - response = []; + return MOCK_BAD_WORDPRESS_RESPONSE; } - return Promise.resolve( response ); + if ( matchesPath( successURL ) ) { + return RICH_TEXT_EMBED_SUCCESS_RESPONSE; + } + + return mockOtherResponses( req ); } ); - const { - getByLabelText, - getByText, - getByTestId, - getByDisplayValue, - } = await initializeWithEmbedBlock( WP_EMBED_HTML ); + const editor = await initializeWithEmbedBlock( WP_EMBED_HTML ); - fireEvent.press( getByText( 'More options' ) ); - fireEvent.press( getByText( 'Edit link' ) ); + fireEvent.press( editor.getByText( 'More options' ) ); + fireEvent.press( editor.getByText( 'Edit link' ) ); // Start editing link. - fireEvent.press( getByLabelText( `WordPress link, ${ failURL }` ) ); + fireEvent.press( + editor.getByLabelText( `WordPress link, ${ failURL }` ) + ); // Set an URL. - const linkTextInput = getByDisplayValue( failURL ); + const linkTextInput = editor.getByDisplayValue( failURL ); fireEvent( linkTextInput, 'focus' ); fireEvent.changeText( linkTextInput, successURL ); // Dismiss the edit URL modal. - const embedEditURLModal = getByTestId( 'embed-edit-url-modal' ); + const embedEditURLModal = editor.getByTestId( + 'embed-edit-url-modal' + ); fireEvent( embedEditURLModal, 'backdropPress' ); fireEvent( embedEditURLModal, MODAL_DISMISS_EVENT ); - const blockSettingsModal = await waitFor( () => - getByTestId( 'block-settings-modal' ) + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + + const blockSettingsModal = await editor.findByTestId( + 'block-settings-modal' ); // Get Twitter link field. const twitterLinkField = within( @@ -864,17 +870,13 @@ describe( 'Embed block', () => { it( 'creates embed block when pasting URL in paragraph block', async () => { const expectedURL = 'https://www.youtube.com/watch?v=lXMskKTw3Bc'; - const { - getByLabelText, - getByPlaceholderText, - getByTestId, - getByText, - } = await initializeEditor( { + const editor = await initializeEditor( { initialHtml: EMPTY_PARAGRAPH_HTML, } ); // Paste URL in paragraph block. - const paragraphText = getByPlaceholderText( 'Start writing…' ); + const paragraphText = + editor.getByPlaceholderText( 'Start writing…' ); fireEvent( paragraphText, 'focus' ); fireEvent( paragraphText, 'paste', { preventDefault: jest.fn(), @@ -888,36 +890,37 @@ describe( 'Embed block', () => { } ); // Wait for embed handler picker to be visible. - await waitFor( - () => getByTestId( 'embed-handler-picker' ).props.isVisible + const embedHandlerPicker = editor.getByTestId( + 'embed-handler-picker' ); + await waitFor( () => embedHandlerPicker.props.isVisible ); // Select create embed option. - fireEvent.press( getByText( 'Create embed' ) ); + fireEvent.press( editor.getByText( 'Create embed' ) ); // Get the created embed block. - const embedBlock = await waitFor( () => - getByLabelText( /Embed Block\. Row 1/ ) + const [ embedBlock ] = await editor.findAllByLabelText( + /Embed Block\. Row 1/ ); expect( embedBlock ).toBeDefined(); + + await waitFor( () => editor.UNSAFE_getByType( WebView ) ); + await editor.findByText( 'Media settings' ); + expect( getEditorHtml() ).toMatchSnapshot(); } ); it( 'creates link when pasting URL in paragraph block', async () => { const expectedURL = 'https://www.youtube.com/watch?v=lXMskKTw3Bc'; - const { - getByDisplayValue, - getByPlaceholderText, - getByTestId, - getByText, - } = await initializeEditor( { + const editor = await initializeEditor( { initialHtml: EMPTY_PARAGRAPH_HTML, } ); // Paste URL in paragraph block. - const paragraphText = getByPlaceholderText( 'Start writing…' ); + const paragraphText = + editor.getByPlaceholderText( 'Start writing…' ); fireEvent( paragraphText, 'focus' ); fireEvent( paragraphText, 'paste', { preventDefault: jest.fn(), @@ -931,16 +934,17 @@ describe( 'Embed block', () => { } ); // Wait for embed handler picker to be visible. - await waitFor( - () => getByTestId( 'embed-handler-picker' ).props.isVisible + const embedHandlerPicker = editor.getByTestId( + 'embed-handler-picker' ); + await waitFor( () => embedHandlerPicker.props.isVisible ); // Select create link option. - fireEvent.press( getByText( 'Create link' ) ); + fireEvent.press( editor.getByText( 'Create link' ) ); // Get the link text. const linkText = await waitFor( () => - getByDisplayValue( + editor.getByDisplayValue( `

${ expectedURL }

` ) ); @@ -953,10 +957,12 @@ describe( 'Embed block', () => { describe( 'insert via slash inserter', () => { it( 'insert generic embed block', async () => { const embedBlockSlashInserter = '/Embed'; - const { getByPlaceholderText, getByLabelText, getByText } = - await initializeEditor( { initialHtml: EMPTY_PARAGRAPH_HTML } ); + const editor = await initializeEditor( { + initialHtml: EMPTY_PARAGRAPH_HTML, + } ); - const paragraphText = getByPlaceholderText( 'Start writing…' ); + const paragraphText = + editor.getByPlaceholderText( 'Start writing…' ); fireEvent( paragraphText, 'focus' ); // Trigger onSelectionChange to update both the current text and text selection. // This event is required by the autocompleter, as it only displays the slash inserter @@ -977,10 +983,10 @@ describe( 'Embed block', () => { } ); - fireEvent.press( await waitFor( () => getByText( 'Embed' ) ) ); + fireEvent.press( await editor.findByText( 'Embed' ) ); - const block = await waitFor( () => - getByLabelText( /Embed Block\. Row 1/ ) + const [ block ] = await editor.findAllByLabelText( + /Embed Block\. Row 1/ ); const blockName = within( block ).getByText( 'Embed' ); @@ -992,12 +998,12 @@ describe( 'Embed block', () => { MOST_USED_PROVIDERS.forEach( ( { title } ) => it( `inserts ${ title } embed block`, async () => { const embedBlockSlashInserter = `/${ title }`; - const { getByPlaceholderText, getByLabelText, getByText } = - await initializeEditor( { - initialHtml: EMPTY_PARAGRAPH_HTML, - } ); + const editor = await initializeEditor( { + initialHtml: EMPTY_PARAGRAPH_HTML, + } ); - const paragraphText = getByPlaceholderText( 'Start writing…' ); + const paragraphText = + editor.getByPlaceholderText( 'Start writing…' ); fireEvent( paragraphText, 'focus' ); // Trigger onSelectionChange to update both the current text and text selection. // This event is required by the autocompleter, as it only displays the slash inserter @@ -1018,10 +1024,10 @@ describe( 'Embed block', () => { } ); - fireEvent.press( await waitFor( () => getByText( title ) ) ); + fireEvent.press( await editor.findByText( title ) ); - const block = await waitFor( () => - getByLabelText( /Embed Block\. Row 1/ ) + const [ block ] = await editor.findAllByLabelText( + /Embed Block\. Row 1/ ); const blockName = within( block ).getByText( title ); @@ -1060,12 +1066,12 @@ describe( 'Embed block', () => { it( 'displays cannot embed on the placeholder if preview data is null', async () => { // Return null response for requests to oembed endpoint. - fetchRequest.mockImplementation( ( { path } ) => { - if ( path.startsWith( '/wp/v2/block-patterns/categories' ) ) { - return Promise.resolve( [] ); + fetchRequest.mockImplementation( async ( req ) => { + if ( req.path.startsWith( '/oembed/1.0/proxy' ) ) { + return EMBED_NULL_RESPONSE; } - const isEmbedRequest = path.startsWith( '/oembed/1.0/proxy' ); - return Promise.resolve( isEmbedRequest ? EMBED_NULL_RESPONSE : {} ); + + return mockOtherResponses( req ); } ); const { getByText } = await initializeWithEmbedBlock( diff --git a/packages/block-library/src/file/inspector.js b/packages/block-library/src/file/inspector.js index 121f5e0473e76..71e0885eae1c6 100644 --- a/packages/block-library/src/file/inspector.js +++ b/packages/block-library/src/file/inspector.js @@ -56,6 +56,7 @@ export default function FileBlockInspector( { /> { displayPreview && ( - - + /> - - + /> id === imageAttributes.id ) : null; let newClassName; @@ -217,9 +216,19 @@ function GalleryEdit( props ) { } function isValidFileType( file ) { + // It's necessary to retrieve the media type from the raw image data for already-uploaded images on native. + const nativeFileData = + Platform.isNative && file.id + ? imageData.find( ( { id } ) => id === file.id ) + : null; + + const mediaTypeSelector = nativeFileData + ? nativeFileData?.media_type + : file.type; + return ( ALLOWED_MEDIA_TYPES.some( - ( mediaType ) => file.type?.indexOf( mediaType ) === 0 + ( mediaType ) => mediaTypeSelector?.indexOf( mediaType ) === 0 ) || file.url?.indexOf( 'blob:' ) === 0 ); } @@ -323,7 +332,7 @@ function GalleryEdit( props ) { getBlock( clientId ).innerBlocks.forEach( ( block ) => { blocks.push( block.clientId ); const image = block.attributes.id - ? find( imageData, { id: block.attributes.id } ) + ? imageData.find( ( { id } ) => id === block.attributes.id ) : null; changedAttributes[ block.clientId ] = getHrefAndDestination( image, @@ -391,7 +400,7 @@ function GalleryEdit( props ) { getBlock( clientId ).innerBlocks.forEach( ( block ) => { blocks.push( block.clientId ); const image = block.attributes.id - ? find( imageData, { id: block.attributes.id } ) + ? imageData.find( ( { id } ) => id === block.attributes.id ) : null; changedAttributes[ block.clientId ] = getImageSizeAttributes( image, @@ -486,6 +495,7 @@ function GalleryEdit( props ) { { images.length > 1 && ( { isSelected={ isCaptionSelected } accessible={ true } accessibilityLabelCreator={ ( caption ) => - isEmpty( caption ) + ! caption ? /* translators: accessibility text. Empty gallery caption. */ 'Gallery caption. Empty' diff --git a/packages/block-library/src/gallery/shared.js b/packages/block-library/src/gallery/shared.js index 4765b4395994f..f6a614f1e2466 100644 --- a/packages/block-library/src/gallery/shared.js +++ b/packages/block-library/src/gallery/shared.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { get, pick } from 'lodash'; +import { get } from 'lodash'; /** * WordPress dependencies @@ -13,7 +13,12 @@ export function defaultColumnsNumber( imageCount ) { } export const pickRelevantMediaFiles = ( image, sizeSlug = 'large' ) => { - const imageProps = pick( image, [ 'alt', 'id', 'link' ] ); + const imageProps = Object.fromEntries( + Object.entries( image ?? {} ).filter( ( [ key ] ) => + [ 'alt', 'id', 'link' ].includes( key ) + ) + ); + imageProps.url = get( image, [ 'sizes', sizeSlug, 'url' ] ) || get( image, [ 'media_details', 'sizes', sizeSlug, 'source_url' ] ) || @@ -44,18 +49,9 @@ function getGalleryBlockV2Enabled() { * can be removed when minimum supported WP version >=5.9. */ export function isGalleryV2Enabled() { - // The logic for the native version is located in a different if statement - // due to a lint rule that prohibits a single conditional combining - // `process.env.IS_GUTENBERG_PLUGIN` with a native platform check. if ( Platform.isNative ) { return getGalleryBlockV2Enabled(); } - // Only run the Gallery version compat check if the plugin is running, otherwise - // assume we are in 5.9 core and enable by default. - if ( process.env.IS_GUTENBERG_PLUGIN ) { - return getGalleryBlockV2Enabled(); - } - return true; } diff --git a/packages/block-library/src/gallery/transforms.js b/packages/block-library/src/gallery/transforms.js index 3c49c32330144..f499b689b2a0d 100644 --- a/packages/block-library/src/gallery/transforms.js +++ b/packages/block-library/src/gallery/transforms.js @@ -1,8 +1,3 @@ -/** - * External dependencies - */ -import { filter } from 'lodash'; - /** * WordPress dependencies */ @@ -144,7 +139,7 @@ const transforms = { ? sizeSlug : undefined; - const validImages = filter( attributes, ( { url } ) => url ); + const validImages = attributes.filter( ( { url } ) => url ); if ( isGalleryV2Enabled() ) { const innerBlocks = validImages.map( ( image ) => { diff --git a/packages/block-library/src/gallery/use-get-media.js b/packages/block-library/src/gallery/use-get-media.js index 36559447ae291..a6dbe9faabf0b 100644 --- a/packages/block-library/src/gallery/use-get-media.js +++ b/packages/block-library/src/gallery/use-get-media.js @@ -28,7 +28,7 @@ export default function useGetMedia( innerBlockImages ) { return ( select( coreStore ).getMediaItems( { include: imageIds.join( ',' ), - per_page: imageIds.length, + per_page: -1, orderby: 'include', } ) ?? EMPTY_IMAGE_MEDIA ); diff --git a/packages/block-library/src/gallery/use-get-media.native.js b/packages/block-library/src/gallery/use-get-media.native.js index 994a2152f756c..b03defbdb49d5 100644 --- a/packages/block-library/src/gallery/use-get-media.native.js +++ b/packages/block-library/src/gallery/use-get-media.native.js @@ -34,7 +34,8 @@ export default function useGetMedia( innerBlockImages ) { return ( select( coreStore ).getMediaItems( { include: imageIds.join( ',' ), - per_page: imageIds.length, + per_page: + imageIds.length /* 'hard' limit necessary as unbounded queries aren't supported on native */, orderby: 'include', } ) ?? EMPTY_IMAGE_MEDIA ); diff --git a/packages/block-library/src/gallery/v1/edit.js b/packages/block-library/src/gallery/v1/edit.js index 2ac1df11c7279..9c36f02ec028a 100644 --- a/packages/block-library/src/gallery/v1/edit.js +++ b/packages/block-library/src/gallery/v1/edit.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { filter, find, get, isEmpty, map } from 'lodash'; +import { get, isEmpty, map } from 'lodash'; /** * WordPress dependencies @@ -195,7 +195,7 @@ function GalleryEdit( props ) { function onRemoveImage( index ) { return () => { - const newImages = filter( images, ( img, i ) => index !== i ); + const newImages = images.filter( ( img, i ) => index !== i ); setSelectedImage(); setAttributes( { images: newImages, @@ -211,7 +211,7 @@ function GalleryEdit( props ) { // string, so ensure comparison works correctly by converting the // newImage.id to a string. const newImageId = newImage.id.toString(); - const currentImage = find( images, { id: newImageId } ); + const currentImage = images.find( ( { id } ) => id === newImageId ); const currentImageCaption = currentImage ? currentImage.caption : newImage.caption; @@ -220,9 +220,9 @@ function GalleryEdit( props ) { return currentImageCaption; } - const attachment = find( attachmentCaptions, { - id: newImageId, - } ); + const attachment = attachmentCaptions.find( + ( { id } ) => id === newImageId + ); // If the attachment caption is updated. if ( attachment && attachment.caption !== newImage.caption ) { @@ -299,7 +299,7 @@ function GalleryEdit( props ) { function getImagesSizeOptions() { const resizedImageSizes = Object.values( resizedImages ); return map( - filter( imageSizes, ( { slug } ) => + imageSizes.filter( ( { slug } ) => resizedImageSizes.some( ( sizes ) => sizes[ slug ] ) ), ( { name, slug } ) => ( { value: slug, label: name } ) @@ -403,6 +403,7 @@ function GalleryEdit( props ) { { images.length > 1 && ( { isSelected={ isCaptionSelected } accessible={ true } accessibilityLabelCreator={ ( caption ) => - isEmpty( caption ) + ! caption ? /* translators: accessibility text. Empty gallery caption. */ 'Gallery caption. Empty' : sprintf( diff --git a/packages/block-library/src/gallery/v1/shared.js b/packages/block-library/src/gallery/v1/shared.js index 484020cb9d58c..9a0957cc7a120 100644 --- a/packages/block-library/src/gallery/v1/shared.js +++ b/packages/block-library/src/gallery/v1/shared.js @@ -1,10 +1,15 @@ /** * External dependencies */ -import { get, pick } from 'lodash'; +import { get } from 'lodash'; export const pickRelevantMediaFiles = ( image, sizeSlug = 'large' ) => { - const imageProps = pick( image, [ 'alt', 'id', 'link', 'caption' ] ); + const imageProps = Object.fromEntries( + Object.entries( image ?? {} ).filter( ( [ key ] ) => + [ 'alt', 'id', 'link', 'caption' ].includes( key ) + ) + ); + imageProps.url = get( image, [ 'sizes', sizeSlug, 'url' ] ) || get( image, [ 'media_details', 'sizes', sizeSlug, 'source_url' ] ) || diff --git a/packages/block-library/src/group/block.json b/packages/block-library/src/group/block.json index 3f409ba052c95..a3997db0621c5 100644 --- a/packages/block-library/src/group/block.json +++ b/packages/block-library/src/group/block.json @@ -69,7 +69,9 @@ "fontSize": true } }, - "__experimentalLayout": true + "__experimentalLayout": { + "allowSizingOnChildren": true + } }, "editorStyle": "wp-block-group-editor", "style": "wp-block-group" diff --git a/packages/block-library/src/group/edit.js b/packages/block-library/src/group/edit.js index 7c324d76c587d..8962d8a8a3bf1 100644 --- a/packages/block-library/src/group/edit.js +++ b/packages/block-library/src/group/edit.js @@ -12,7 +12,6 @@ import { } from '@wordpress/block-editor'; import { SelectControl } from '@wordpress/components'; import { __ } from '@wordpress/i18n'; -import { useCallback } from '@wordpress/element'; /** * Internal dependencies @@ -123,13 +122,10 @@ function GroupEdit( { ); const { selectBlock } = useDispatch( blockEditorStore ); - const updateSelection = useCallback( - ( newClientId ) => selectBlock( newClientId, -1 ), - [ selectBlock ] - ); + const selectVariation = ( nextVariation ) => { setAttributes( nextVariation.attributes ); - updateSelection( clientId ); + selectBlock( clientId, -1 ); setShowPlaceholder( false ); }; diff --git a/packages/block-library/src/group/placeholder.js b/packages/block-library/src/group/placeholder.js index 0f8fbc253c898..daf535df8bf60 100644 --- a/packages/block-library/src/group/placeholder.js +++ b/packages/block-library/src/group/placeholder.js @@ -24,10 +24,7 @@ const getGroupPlaceholderIcons = ( name = 'group' ) => { height="32" viewBox="0 0 44 32" > - + ), 'group-row': ( @@ -37,10 +34,7 @@ const getGroupPlaceholderIcons = ( name = 'group' ) => { height="32" viewBox="0 0 44 32" > - + ), 'group-stack': ( @@ -50,10 +44,7 @@ const getGroupPlaceholderIcons = ( name = 'group' ) => { height="32" viewBox="0 0 44 32" > - + ), }; @@ -130,23 +121,13 @@ export function useShouldShowPlaceHolder( { * @return {JSX.Element} The placeholder. */ function GroupPlaceHolder( { name, onSelect } ) { - const { defaultVariation, variations } = useSelect( - ( select ) => { - const { getBlockVariations, getDefaultBlockVariation } = - select( blocksStore ); - return { - defaultVariation: getDefaultBlockVariation( name, 'block' ), - variations: getBlockVariations( name, 'block' ) || [], - }; - }, + const variations = useSelect( + ( select ) => select( blocksStore ).getBlockVariations( name, 'block' ), [ name ] ); const blockProps = useBlockProps( { className: 'wp-block-group__placeholder', } ); - const selectVariation = ( nextVariation = defaultVariation ) => - onSelect( nextVariation ); - return (
selectVariation( variation ) } + onClick={ () => onSelect( variation ) } className="wp-block-group-placeholder__variation-button" label={ `${ variation.title }: ${ variation.description }` } /> diff --git a/packages/block-library/src/group/test/__snapshots__/edit.native.js.snap b/packages/block-library/src/group/test/__snapshots__/edit.native.js.snap index ee8488edf8864..7234dc54ee4d5 100644 --- a/packages/block-library/src/group/test/__snapshots__/edit.native.js.snap +++ b/packages/block-library/src/group/test/__snapshots__/edit.native.js.snap @@ -3,7 +3,7 @@ exports[`Group block inserts block and adds a Heading block as an inner block 1`] = ` "
-

+

" `; diff --git a/packages/block-library/src/heading/block.json b/packages/block-library/src/heading/block.json index 89c1514808911..f9d701eace964 100644 --- a/packages/block-library/src/heading/block.json +++ b/packages/block-library/src/heading/block.json @@ -29,7 +29,7 @@ "supports": { "align": [ "wide", "full" ], "anchor": true, - "className": false, + "className": true, "color": { "gradients": true, "link": true, @@ -57,7 +57,6 @@ "textTransform": true } }, - "__experimentalSelector": "h1,h2,h3,h4,h5,h6", "__unstablePasteTextInline": true, "__experimentalSlashInserter": true }, diff --git a/packages/block-library/src/heading/deprecated.js b/packages/block-library/src/heading/deprecated.js index 9eaee48662277..6ea881acbc19f 100644 --- a/packages/block-library/src/heading/deprecated.js +++ b/packages/block-library/src/heading/deprecated.js @@ -63,155 +63,228 @@ const migrateTextAlign = ( attributes ) => { : attributes; }; -const deprecated = [ - { - supports: { - align: [ 'wide', 'full' ], - anchor: true, - className: false, - color: { link: true }, - fontSize: true, - lineHeight: true, - __experimentalSelector: { - 'core/heading/h1': 'h1', - 'core/heading/h2': 'h2', - 'core/heading/h3': 'h3', - 'core/heading/h4': 'h4', - 'core/heading/h5': 'h5', - 'core/heading/h6': 'h6', - }, - __unstablePasteTextInline: true, +const v1 = { + supports: blockSupports, + attributes: { + ...blockAttributes, + customTextColor: { + type: 'string', }, - attributes: blockAttributes, - isEligible: ( { align } ) => TEXT_ALIGN_OPTIONS.includes( align ), - migrate: migrateTextAlign, - save( { attributes } ) { - const { align, content, level } = attributes; - const TagName = 'h' + level; - - const className = classnames( { - [ `has-text-align-${ align }` ]: align, - } ); - - return ( - - - - ); + textColor: { + type: 'string', }, }, - { - supports: blockSupports, - attributes: { - ...blockAttributes, - customTextColor: { - type: 'string', - }, - textColor: { - type: 'string', - }, + migrate: ( attributes ) => + migrateCustomColors( migrateTextAlign( attributes ) ), + save( { attributes } ) { + const { align, level, content, textColor, customTextColor } = + attributes; + const tagName = 'h' + level; + + const textClass = getColorClassName( 'color', textColor ); + + const className = classnames( { + [ textClass ]: textClass, + } ); + + return ( + + ); + }, +}; +const v2 = { + attributes: { + ...blockAttributes, + customTextColor: { + type: 'string', }, - migrate: ( attributes ) => - migrateCustomColors( migrateTextAlign( attributes ) ), - save( { attributes } ) { - const { align, content, customTextColor, level, textColor } = - attributes; - const tagName = 'h' + level; - - const textClass = getColorClassName( 'color', textColor ); - - const className = classnames( { - [ textClass ]: textClass, - 'has-text-color': textColor || customTextColor, - [ `has-text-align-${ align }` ]: align, - } ); - - return ( - - ); + textColor: { + type: 'string', }, }, - { - attributes: { - ...blockAttributes, - customTextColor: { - type: 'string', - }, - textColor: { - type: 'string', - }, + migrate: ( attributes ) => + migrateCustomColors( migrateTextAlign( attributes ) ), + save( { attributes } ) { + const { align, content, customTextColor, level, textColor } = + attributes; + const tagName = 'h' + level; + + const textClass = getColorClassName( 'color', textColor ); + + const className = classnames( { + [ textClass ]: textClass, + [ `has-text-align-${ align }` ]: align, + } ); + + return ( + + ); + }, + supports: blockSupports, +}; +const v3 = { + supports: blockSupports, + attributes: { + ...blockAttributes, + customTextColor: { + type: 'string', }, - migrate: ( attributes ) => - migrateCustomColors( migrateTextAlign( attributes ) ), - save( { attributes } ) { - const { align, content, customTextColor, level, textColor } = - attributes; - const tagName = 'h' + level; - - const textClass = getColorClassName( 'color', textColor ); - - const className = classnames( { - [ textClass ]: textClass, - [ `has-text-align-${ align }` ]: align, - } ); - - return ( - - ); + textColor: { + type: 'string', }, - supports: blockSupports, }, - { - supports: blockSupports, - attributes: { - ...blockAttributes, - customTextColor: { - type: 'string', + migrate: ( attributes ) => + migrateCustomColors( migrateTextAlign( attributes ) ), + save( { attributes } ) { + const { align, content, customTextColor, level, textColor } = + attributes; + const tagName = 'h' + level; + + const textClass = getColorClassName( 'color', textColor ); + + const className = classnames( { + [ textClass ]: textClass, + 'has-text-color': textColor || customTextColor, + [ `has-text-align-${ align }` ]: align, + } ); + + return ( + + ); + }, +}; +const v4 = { + supports: { + align: [ 'wide', 'full' ], + anchor: true, + className: false, + color: { link: true }, + fontSize: true, + lineHeight: true, + __experimentalSelector: { + 'core/heading/h1': 'h1', + 'core/heading/h2': 'h2', + 'core/heading/h3': 'h3', + 'core/heading/h4': 'h4', + 'core/heading/h5': 'h5', + 'core/heading/h6': 'h6', + }, + __unstablePasteTextInline: true, + }, + attributes: blockAttributes, + isEligible: ( { align } ) => TEXT_ALIGN_OPTIONS.includes( align ), + migrate: migrateTextAlign, + save( { attributes } ) { + const { align, content, level } = attributes; + const TagName = 'h' + level; + + const className = classnames( { + [ `has-text-align-${ align }` ]: align, + } ); + + return ( + + + + ); + }, +}; + +// This deprecation covers the serialization of the `wp-block-heading` class +// into the block's markup after className support was enabled. +const v5 = { + supports: { + align: [ 'wide', 'full' ], + anchor: true, + className: false, + color: { + gradients: true, + link: true, + __experimentalDefaultControls: { + background: true, + text: true, }, - textColor: { - type: 'string', + }, + spacing: { + margin: true, + padding: true, + }, + typography: { + fontSize: true, + lineHeight: true, + __experimentalFontFamily: true, + __experimentalFontStyle: true, + __experimentalFontWeight: true, + __experimentalLetterSpacing: true, + __experimentalTextTransform: true, + __experimentalTextDecoration: true, + __experimentalDefaultControls: { + fontSize: true, + fontAppearance: true, + textTransform: true, }, }, - migrate: ( attributes ) => - migrateCustomColors( migrateTextAlign( attributes ) ), - save( { attributes } ) { - const { align, level, content, textColor, customTextColor } = - attributes; - const tagName = 'h' + level; - - const textClass = getColorClassName( 'color', textColor ); - - const className = classnames( { - [ textClass ]: textClass, - } ); - - return ( - - ); + __experimentalSelector: 'h1,h2,h3,h4,h5,h6', + __unstablePasteTextInline: true, + __experimentalSlashInserter: true, + }, + attributes: { + textAlign: { + type: 'string', + }, + content: { + type: 'string', + source: 'html', + selector: 'h1,h2,h3,h4,h5,h6', + default: '', + __experimentalRole: 'content', + }, + level: { + type: 'number', + default: 2, + }, + placeholder: { + type: 'string', }, }, -]; + save( { attributes } ) { + const { textAlign, content, level } = attributes; + const TagName = 'h' + level; + + const className = classnames( { + [ `has-text-align-${ textAlign }` ]: textAlign, + } ); + + return ( + + + + ); + }, +}; + +const deprecated = [ v5, v4, v3, v2, v1 ]; export default deprecated; diff --git a/packages/block-library/src/heading/index.php b/packages/block-library/src/heading/index.php new file mode 100644 index 0000000000000..5a7e8dbaab43c --- /dev/null +++ b/packages/block-library/src/heading/index.php @@ -0,0 +1,52 @@ +Hello World + * + * Would be transformed to: + *

Hello World

+ * + * @param array $attributes Attributes of the block being rendered. + * @param string $content Content of the block being rendered. + * + * @return string The content of the block being rendered. + */ +function block_core_heading_render( $attributes, $content ) { + if ( ! $content ) { + return $content; + } + + $p = new WP_HTML_Tag_Processor( $content ); + + $header_tags = array( 'H1', 'H2', 'H3', 'H4', 'H5', 'H6' ); + while ( $p->next_tag() ) { + if ( in_array( $p->get_tag(), $header_tags, true ) ) { + $p->add_class( 'wp-block-heading' ); + break; + } + } + + return $p->get_updated_html(); +} + +/** + * Registers the `core/heading` block on server. + */ +function register_block_core_heading() { + register_block_type_from_metadata( + __DIR__ . '/heading', + array( + 'render_callback' => 'block_core_heading_render', + ) + ); +} + +add_action( 'init', 'register_block_core_heading' ); diff --git a/packages/block-library/src/heading/test/__snapshots__/index.native.js.snap b/packages/block-library/src/heading/test/__snapshots__/index.native.js.snap new file mode 100644 index 0000000000000..8f3d3518636d3 --- /dev/null +++ b/packages/block-library/src/heading/test/__snapshots__/index.native.js.snap @@ -0,0 +1,7 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`Heading block inserts block 1`] = ` +" +

+" +`; diff --git a/packages/block-library/src/heading/test/index.native.js b/packages/block-library/src/heading/test/index.native.js new file mode 100644 index 0000000000000..fce294cf9c992 --- /dev/null +++ b/packages/block-library/src/heading/test/index.native.js @@ -0,0 +1,44 @@ +/** + * External dependencies + */ +import { + fireEvent, + getEditorHtml, + initializeEditor, + addBlock, + getBlock, +} from 'test/helpers'; + +/** + * WordPress dependencies + */ +import { getBlockTypes, unregisterBlockType } from '@wordpress/blocks'; +import { registerCoreBlocks } from '@wordpress/block-library'; + +beforeAll( () => { + // Register all core blocks + registerCoreBlocks(); +} ); + +afterAll( () => { + // Clean up registered blocks + getBlockTypes().forEach( ( block ) => { + unregisterBlockType( block.name ); + } ); +} ); + +describe( 'Heading block', () => { + it( 'inserts block', async () => { + const screen = await initializeEditor(); + + // Add block + await addBlock( screen, 'Heading' ); + + // Get block + const headingBlock = await getBlock( screen, 'Heading' ); + fireEvent.press( headingBlock ); + expect( headingBlock ).toBeVisible(); + + expect( getEditorHtml() ).toMatchSnapshot(); + } ); +} ); diff --git a/packages/block-library/src/home-link/index.php b/packages/block-library/src/home-link/index.php index 5bf7aeda5505d..33f41057c936b 100644 --- a/packages/block-library/src/home-link/index.php +++ b/packages/block-library/src/home-link/index.php @@ -128,7 +128,7 @@ function render_block_core_home_link( $attributes, $content, $block ) { $aria_current = is_home() || ( is_front_page() && 'page' === get_option( 'show_on_front' ) ) ? ' aria-current="page"' : ''; return sprintf( - '
  • %4$s
  • ', + '
  • %4$s
  • ', block_core_home_link_build_li_wrapper_attributes( $block->context ), esc_url( home_url() ), $aria_current, diff --git a/packages/block-library/src/html/block.json b/packages/block-library/src/html/block.json index e72a674373e14..c1e1e94b87496 100644 --- a/packages/block-library/src/html/block.json +++ b/packages/block-library/src/html/block.json @@ -10,7 +10,7 @@ "attributes": { "content": { "type": "string", - "source": "html" + "source": "raw" } }, "supports": { diff --git a/packages/block-library/src/image/edit.js b/packages/block-library/src/image/edit.js index baba2cee24ec3..4e504896b4709 100644 --- a/packages/block-library/src/image/edit.js +++ b/packages/block-library/src/image/edit.js @@ -2,7 +2,7 @@ * External dependencies */ import classnames from 'classnames'; -import { get, isEmpty, pick } from 'lodash'; +import { get, isEmpty } from 'lodash'; /** * WordPress dependencies @@ -58,7 +58,12 @@ import { } from './constants'; export const pickRelevantMediaFiles = ( image, size ) => { - const imageProps = pick( image, [ 'alt', 'id', 'link', 'caption' ] ); + const imageProps = Object.fromEntries( + Object.entries( image ?? {} ).filter( ( [ key ] ) => + [ 'alt', 'id', 'link', 'caption' ].includes( key ) + ) + ); + imageProps.url = get( image, [ 'sizes', size, 'url' ] ) || get( image, [ 'media_details', 'sizes', size, 'source_url' ] ) || diff --git a/packages/block-library/src/image/edit.native.js b/packages/block-library/src/image/edit.native.js index 8a9881fa95c71..7be5d157374f3 100644 --- a/packages/block-library/src/image/edit.native.js +++ b/packages/block-library/src/image/edit.native.js @@ -61,8 +61,9 @@ import { textColor, } from '@wordpress/icons'; import { store as coreStore } from '@wordpress/core-data'; -import { store as editPostStore } from '@wordpress/edit-post'; import { store as noticesStore } from '@wordpress/notices'; +// eslint-disable-next-line no-restricted-imports +import { store as editPostStore } from '@wordpress/edit-post'; /** * Internal dependencies diff --git a/packages/block-library/src/image/editor.scss b/packages/block-library/src/image/editor.scss index 3e4bb7dd6392f..fb32ab54298bc 100644 --- a/packages/block-library/src/image/editor.scss +++ b/packages/block-library/src/image/editor.scss @@ -161,17 +161,6 @@ figure.wp-block-image:not(.wp-block) { min-width: 260px; overflow: visible !important; } - - .components-range-control { - flex: 1; - } - - .components-base-control__field { - display: flex; - margin-bottom: 0; - flex-direction: column; - align-items: flex-start; - } } .wp-block-image__aspect-ratio { diff --git a/packages/block-library/src/image/image.js b/packages/block-library/src/image/image.js index 7245ce232c64b..5eb511b455864 100644 --- a/packages/block-library/src/image/image.js +++ b/packages/block-library/src/image/image.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { get, filter, isEmpty, map, pick } from 'lodash'; +import { get, isEmpty, map } from 'lodash'; /** * WordPress dependencies @@ -135,12 +135,16 @@ export default function Image( { } = select( blockEditorStore ); const rootClientId = getBlockRootClientId( clientId ); - const settings = pick( getSettings(), [ - 'imageEditing', - 'imageSizes', - 'maxWidth', - 'mediaUpload', - ] ); + const settings = Object.fromEntries( + Object.entries( getSettings() ).filter( ( [ key ] ) => + [ + 'imageEditing', + 'imageSizes', + 'maxWidth', + 'mediaUpload', + ].includes( key ) + ) + ); return { ...settings, @@ -169,7 +173,7 @@ export default function Image( { ! isContentLocked && ! ( isWideAligned && isLargeViewport ); const imageSizeOptions = map( - filter( imageSizes, ( { slug } ) => + imageSizes.filter( ( { slug } ) => get( image, [ 'media_details', 'sizes', slug, 'source_url' ] ) ), ( { name, slug } ) => ( { value: slug, label: name } ) @@ -607,6 +611,7 @@ export default function Image( { height: parseInt( currentHeight + delta.height, 10 ), } ); } } + resizeRatio={ align === 'center' ? 2 : 1 } > { img } diff --git a/packages/block-library/src/image/test/edit.native.js b/packages/block-library/src/image/test/edit.native.js index 2f8aa703ee2b6..73f4ec22a554d 100644 --- a/packages/block-library/src/image/test/edit.native.js +++ b/packages/block-library/src/image/test/edit.native.js @@ -81,7 +81,8 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - fireEvent.press( screen.getByLabelText( /Image Block/ ) ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); + fireEvent.press( imageBlock ); // Awaiting navigation event seemingly required due to React Navigation bug // https://github.com/react-navigation/react-navigation/issues/9701 await act( () => @@ -107,7 +108,8 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - fireEvent.press( screen.getByLabelText( /Image Block/ ) ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); + fireEvent.press( imageBlock ); // Awaiting navigation event seemingly required due to React Navigation bug // https://github.com/react-navigation/react-navigation/issues/9701 await act( () => @@ -133,7 +135,8 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - fireEvent.press( screen.getByLabelText( /Image Block/ ) ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); + fireEvent.press( imageBlock ); // Awaiting navigation event seemingly required due to React Navigation bug // https://github.com/react-navigation/react-navigation/issues/9701 await act( () => @@ -169,7 +172,8 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - fireEvent.press( screen.getByLabelText( /Image Block/ ) ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); + fireEvent.press( imageBlock ); // Awaiting navigation event seemingly required due to React Navigation bug // https://github.com/react-navigation/react-navigation/issues/9701 await act( () => @@ -177,7 +181,7 @@ describe( 'Image Block', () => { ); fireEvent.press( screen.getByText( 'None' ) ); fireEvent.press( screen.getByText( 'Media File' ) ); - await waitFor( () => screen.getByText( 'Custom URL' ) ); + await screen.findByText( 'Custom URL' ); fireEvent.press( screen.getByText( 'Custom URL' ) ); // Await asynchronous fetch of clipboard await act( () => clipboardPromise ); @@ -186,8 +190,7 @@ describe( 'Image Block', () => { 'wordpress.org' ); fireEvent.press( screen.getByLabelText( 'Apply' ) ); - await waitFor( () => screen.getByText( 'Custom URL' ) ); - fireEvent.press( screen.getByText( 'Custom URL' ) ); + fireEvent.press( await screen.findByText( 'Custom URL' ) ); // Await asynchronous fetch of clipboard await act( () => clipboardPromise ); fireEvent.press( screen.getByText( 'Media File' ) ); @@ -211,7 +214,8 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - fireEvent.press( screen.getByLabelText( /Image Block/ ) ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); + fireEvent.press( imageBlock ); // Awaiting navigation event seemingly required due to React Navigation bug // https://github.com/react-navigation/react-navigation/issues/9701 await act( () => @@ -235,7 +239,7 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - const imageBlock = screen.getByLabelText( /Image Block/ ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); fireEvent.press( imageBlock ); const settingsButton = screen.getByLabelText( 'Open Settings' ); @@ -266,7 +270,7 @@ describe( 'Image Block', () => { // We must await the image fetch via `getMedia` await act( () => apiFetchPromise ); - const imageBlock = screen.getByLabelText( /Image Block/ ); + const [ imageBlock ] = screen.getAllByLabelText( /Image Block/ ); fireEvent.press( imageBlock ); const settingsButton = screen.getByLabelText( 'Open Settings' ); diff --git a/packages/block-library/src/index.js b/packages/block-library/src/index.js index 88d2c26f02403..34ba9fd5a2c8f 100644 --- a/packages/block-library/src/index.js +++ b/packages/block-library/src/index.js @@ -67,6 +67,7 @@ import * as navigationSubmenu from './navigation-submenu'; import * as nextpage from './nextpage'; import * as pattern from './pattern'; import * as pageList from './page-list'; +import * as pageListItem from './page-list-item'; import * as paragraph from './paragraph'; import * as postAuthor from './post-author'; import * as postAuthorName from './post-author-name'; @@ -155,6 +156,7 @@ const getAllBlocks = () => more, nextpage, pageList, + pageListItem, pattern, preformatted, pullquote, diff --git a/packages/block-library/src/latest-comments/edit.js b/packages/block-library/src/latest-comments/edit.js index a376ebb94b6de..01eb9e11dc3de 100644 --- a/packages/block-library/src/latest-comments/edit.js +++ b/packages/block-library/src/latest-comments/edit.js @@ -56,6 +56,7 @@ export default function LatestComments( { attributes, setAttributes } ) { } /> diff --git a/packages/block-library/src/latest-posts/edit.js b/packages/block-library/src/latest-posts/edit.js index 3671ec555e600..45ffe1338aaab 100644 --- a/packages/block-library/src/latest-posts/edit.js +++ b/packages/block-library/src/latest-posts/edit.js @@ -234,6 +234,7 @@ export default function LatestPostsEdit( { attributes, setAttributes } ) { { displayPostContent && displayPostContentRadio === 'excerpt' && ( @@ -355,6 +356,7 @@ export default function LatestPostsEdit( { attributes, setAttributes } ) { { postLayout === 'grid' && ( diff --git a/packages/block-library/src/list-item/index.js b/packages/block-library/src/list-item/index.js index 61756019baf92..00adc1c2c4026 100644 --- a/packages/block-library/src/list-item/index.js +++ b/packages/block-library/src/list-item/index.js @@ -10,6 +10,7 @@ import initBlock from '../utils/init-block'; import metadata from './block.json'; import edit from './edit'; import save from './save'; +import transforms from './transforms'; const { name } = metadata; @@ -25,6 +26,7 @@ export const settings = { content: attributes.content + attributesToMerge.content, }; }, + transforms, }; export const init = () => initBlock( { name, metadata, settings } ); diff --git a/packages/block-library/src/list-item/transforms.js b/packages/block-library/src/list-item/transforms.js new file mode 100644 index 0000000000000..6e05f8501b5a3 --- /dev/null +++ b/packages/block-library/src/list-item/transforms.js @@ -0,0 +1,17 @@ +/** + * WordPress dependencies + */ +import { createBlock } from '@wordpress/blocks'; + +const transforms = { + to: [ + { + type: 'block', + blocks: [ 'core/paragraph' ], + transform: ( attributes ) => + createBlock( 'core/paragraph', attributes ), + }, + ], +}; + +export default transforms; diff --git a/packages/block-library/src/list-item/utils.js b/packages/block-library/src/list-item/utils.js index 41f242c84e055..ac7f99ac849e0 100644 --- a/packages/block-library/src/list-item/utils.js +++ b/packages/block-library/src/list-item/utils.js @@ -8,6 +8,7 @@ import { createBlock, switchToBlockType } from '@wordpress/blocks'; */ import { name as listItemName } from './block.json'; import { name as listName } from '../list/block.json'; +import { name as paragraphName } from '../paragraph/block.json'; export function createListItem( listItemAttributes, listAttributes, children ) { return createBlock( @@ -19,6 +20,14 @@ export function createListItem( listItemAttributes, listAttributes, children ) { ); } +function convertBlockToList( block ) { + const list = switchToBlockType( block, listName ); + if ( list ) return list; + const paragraph = switchToBlockType( block, paragraphName ); + if ( paragraph ) return switchToBlockType( paragraph, listName ); + return null; +} + export function convertToListItems( blocks ) { const listItems = []; @@ -27,7 +36,7 @@ export function convertToListItems( blocks ) { listItems.push( block ); } else if ( block.name === listName ) { listItems.push( ...block.innerBlocks ); - } else if ( ( block = switchToBlockType( block, listName ) ) ) { + } else if ( ( block = convertBlockToList( block ) ) ) { for ( const { innerBlocks } of block ) { listItems.push( ...innerBlocks ); } diff --git a/packages/block-library/src/list/test/edit.native.js b/packages/block-library/src/list/test/edit.native.js index 9defa338782db..5b70952925cd9 100644 --- a/packages/block-library/src/list/test/edit.native.js +++ b/packages/block-library/src/list/test/edit.native.js @@ -63,16 +63,18 @@ describe( 'List block', () => {
  • `; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Select List Item block - const listItemBlock = getByLabelText( /List item Block\. Row 1/ ); + const [ listItemBlock ] = screen.getAllByLabelText( + /List item Block\. Row 1/ + ); fireEvent.press( listItemBlock ); const listItemField = @@ -109,25 +111,25 @@ describe( 'List block', () => { `; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Select List Item block - const firstNestedLevelBlock = within( listBlock ).getByLabelText( + const [ firstNestedLevelBlock ] = within( listBlock ).getAllByLabelText( /List item Block\. Row 2/ ); fireEvent.press( firstNestedLevelBlock ); // Select second level list - const secondNestedLevelBlock = within( + const [ secondNestedLevelBlock ] = within( firstNestedLevelBlock - ).getByLabelText( /List Block\. Row 1/ ); + ).getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( secondNestedLevelBlock ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -143,20 +145,22 @@ describe( 'List block', () => { `; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Select Secont List Item block - const listItemBlock = getByLabelText( /List item Block\. Row 2/ ); + const [ listItemBlock ] = screen.getAllByLabelText( + /List item Block\. Row 2/ + ); fireEvent.press( listItemBlock ); // Update indentation - const indentButton = getByLabelText( 'Indent' ); + const indentButton = screen.getByLabelText( 'Indent' ); fireEvent.press( indentButton ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -173,33 +177,33 @@ describe( 'List block', () => { `; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Select List Item block - const firstNestedLevelBlock = within( listBlock ).getByLabelText( + const [ firstNestedLevelBlock ] = within( listBlock ).getAllByLabelText( /List item Block\. Row 1/ ); fireEvent.press( firstNestedLevelBlock ); // Select Inner block List - const innerBlockList = within( firstNestedLevelBlock ).getByLabelText( - /List Block\. Row 1/ - ); + const [ innerBlockList ] = within( + firstNestedLevelBlock + ).getAllByLabelText( /List Block\. Row 1/ ); // Select nested List Item block - const listItemBlock = within( innerBlockList ).getByLabelText( + const [ listItemBlock ] = within( innerBlockList ).getAllByLabelText( /List item Block\. Row 1/ ); fireEvent.press( listItemBlock ); // Update indentation - const outdentButton = getByLabelText( 'Outdent' ); + const outdentButton = screen.getByLabelText( 'Outdent' ); fireEvent.press( outdentButton ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -218,16 +222,16 @@ describe( 'List block', () => { `; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Update to ordered list - const orderedButton = getByLabelText( 'Ordered' ); + const orderedButton = screen.getByLabelText( 'Ordered' ); fireEvent.press( orderedButton ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -246,27 +250,29 @@ describe( 'List block', () => { `; - const { getByLabelText, getByTestId } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Update to ordered list - const orderedButton = getByLabelText( 'Ordered' ); + const orderedButton = screen.getByLabelText( 'Ordered' ); fireEvent.press( orderedButton ); // Set order to reverse // Open block settings - fireEvent.press( getByLabelText( 'Open Settings' ) ); + fireEvent.press( screen.getByLabelText( 'Open Settings' ) ); await waitFor( - () => getByTestId( 'block-settings-modal' ).props.isVisible + () => screen.getByTestId( 'block-settings-modal' ).props.isVisible ); - const reverseButton = getByLabelText( /Reverse list numbering\. Off/ ); + const reverseButton = screen.getByLabelText( + /Reverse list numbering\. Off/ + ); fireEvent.press( reverseButton ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -285,27 +291,27 @@ describe( 'List block', () => { `; - const { getByLabelText, getByTestId } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select List block - const listBlock = getByLabelText( /List Block\. Row 1/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 1/ ); fireEvent.press( listBlock ); // Update to ordered list - const orderedButton = getByLabelText( 'Ordered' ); + const orderedButton = screen.getByLabelText( 'Ordered' ); fireEvent.press( orderedButton ); // Set order to reverse // Open block settings - fireEvent.press( getByLabelText( 'Open Settings' ) ); + fireEvent.press( screen.getByLabelText( 'Open Settings' ) ); await waitFor( - () => getByTestId( 'block-settings-modal' ).props.isVisible + () => screen.getByTestId( 'block-settings-modal' ).props.isVisible ); - const startValueButton = getByLabelText( /Start value\. Empty/ ); + const startValueButton = screen.getByLabelText( /Start value\. Empty/ ); fireEvent.press( startValueButton ); const startValueInput = within( startValueButton ).getByDisplayValue( '' ); @@ -328,11 +334,11 @@ describe( 'List block', () => { } ); // Select List block - const listBlock = screen.getByLabelText( /List Block\. Row 2/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 2/ ); fireEvent.press( listBlock ); // Select List Item block - const listItemBlock = within( listBlock ).getByLabelText( + const [ listItemBlock ] = within( listBlock ).getAllByLabelText( /List item Block\. Row 1/ ); fireEvent.press( listItemBlock ); @@ -361,7 +367,7 @@ describe( 'List block', () => { ` ); } ); - it( 'unwraps list items when attempting to merge with non-list block', async () => { + it( 'unwraps first item when attempting to merge with non-list block', async () => { const initialHtml = `

    A quick brown fox.

    @@ -376,11 +382,11 @@ describe( 'List block', () => { } ); // Select List block - const listBlock = screen.getByLabelText( /List Block\. Row 2/ ); + const [ listBlock ] = screen.getAllByLabelText( /List Block\. Row 2/ ); fireEvent.press( listBlock ); // Select List Item block - const listItemBlock = within( listBlock ).getByLabelText( + const [ listItemBlock ] = within( listBlock ).getAllByLabelText( /List item Block\. Row 1/ ); fireEvent.press( listItemBlock ); @@ -400,14 +406,16 @@ describe( 'List block', () => { "

    A quick brown fox.

    - +

    One

    - - -

    Two

    - " + + +
      +
    • Two
    • +
    + " ` ); } ); } ); diff --git a/packages/block-library/src/list/transforms.js b/packages/block-library/src/list/transforms.js index 2f11119768ef8..a6263d7ad639c 100644 --- a/packages/block-library/src/list/transforms.js +++ b/packages/block-library/src/list/transforms.js @@ -114,17 +114,6 @@ const transforms = { ); }, } ) ), - { - type: 'block', - blocks: [ '*' ], - transform: ( _attributes, childBlocks ) => { - return getListContentFlat( childBlocks ).map( ( content ) => - createBlock( 'core/paragraph', { - content, - } ) - ); - }, - }, ], }; diff --git a/packages/block-library/src/list/utils.js b/packages/block-library/src/list/utils.js index a80ce456e29ab..4fa2a0e0aed94 100644 --- a/packages/block-library/src/list/utils.js +++ b/packages/block-library/src/list/utils.js @@ -56,7 +56,8 @@ export function createListBlockFromDOMElement( listElement ) { } export function migrateToListV2( attributes ) { - const { values, start, reversed, ordered, type } = attributes; + const { values, start, reversed, ordered, type, ...otherAttributes } = + attributes; const list = document.createElement( ordered ? 'ol' : 'ul' ); list.innerHTML = values; @@ -72,5 +73,8 @@ export function migrateToListV2( attributes ) { const [ listBlock ] = rawHandler( { HTML: list.outerHTML } ); - return [ listBlock.attributes, listBlock.innerBlocks ]; + return [ + { ...otherAttributes, ...listBlock.attributes }, + listBlock.innerBlocks, + ]; } diff --git a/packages/block-library/src/media-text/edit.js b/packages/block-library/src/media-text/edit.js index 5a88830d385d1..83b7074e3d155 100644 --- a/packages/block-library/src/media-text/edit.js +++ b/packages/block-library/src/media-text/edit.js @@ -2,7 +2,7 @@ * External dependencies */ import classnames from 'classnames'; -import { map, filter } from 'lodash'; +import { map } from 'lodash'; /** * WordPress dependencies @@ -199,7 +199,7 @@ function MediaTextEdit( { attributes, isSelected, setAttributes, clientId } ) { setAttributes( { mediaWidth: applyWidthConstraints( width ), } ); - setTemporaryMediaWidth( applyWidthConstraints( width ) ); + setTemporaryMediaWidth( null ); }; const classNames = classnames( { @@ -226,7 +226,7 @@ function MediaTextEdit( { attributes, isSelected, setAttributes, clientId } ) { }; const imageSizeOptions = map( - filter( imageSizes, ( { slug } ) => + imageSizes.filter( ( { slug } ) => getImageSourceUrlBySizeSlug( image, slug ) ), ( { name, slug } ) => ( { value: slug, label: name } ) @@ -268,6 +268,7 @@ function MediaTextEdit( { attributes, isSelected, setAttributes, clientId } ) { ) } { imageFill && mediaUrl && mediaType === 'image' && ( { const initialHtml = `
    12
    34
    `; - const { getByLabelText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); - const missingBlock = await waitFor( () => - getByLabelText( /Unsupported Block\. Row 1/ ) + const [ missingBlock ] = await screen.findAllByLabelText( + /Unsupported Block\. Row 1/ ); const translatedTableTitle = @@ -58,24 +58,22 @@ describe( 'Unsupported block', () => { const initialHtml = `
    12
    34
    `; - const { getByLabelText, getByText } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); - const missingBlock = await waitFor( () => - getByLabelText( /Unsupported Block\. Row 1/ ) + const [ missingBlock ] = await screen.findAllByLabelText( + /Unsupported Block\. Row 1/ ); fireEvent.press( missingBlock ); - const helpButton = await waitFor( () => - getByLabelText( 'Help button' ) - ); + const [ helpButton ] = await screen.findAllByLabelText( 'Help button' ); fireEvent.press( helpButton ); - const bottomSheetTitle = await waitFor( () => - getByText( '«Tabla» no es totalmente compatible' ) + const bottomSheetTitle = await screen.findByText( + '«Tabla» no es totalmente compatible' ); expect( bottomSheetTitle ).toBeDefined(); diff --git a/packages/block-library/src/navigation-link/edit.js b/packages/block-library/src/navigation-link/edit.js index ccd0aae57bcbb..7e58527d947f5 100644 --- a/packages/block-library/src/navigation-link/edit.js +++ b/packages/block-library/src/navigation-link/edit.js @@ -2,18 +2,15 @@ * External dependencies */ import classnames from 'classnames'; -import escapeHtml from 'escape-html'; import { unescape } from 'lodash'; /** * WordPress dependencies */ -import { createBlock, switchToBlockType } from '@wordpress/blocks'; +import { createBlock } from '@wordpress/blocks'; import { useSelect, useDispatch } from '@wordpress/data'; import { - Button, PanelBody, - Popover, TextControl, TextareaControl, ToolbarButton, @@ -22,38 +19,35 @@ import { KeyboardShortcuts, } from '@wordpress/components'; import { displayShortcut, isKeyboardEvent, ENTER } from '@wordpress/keycodes'; -import { __, sprintf } from '@wordpress/i18n'; +import { __ } from '@wordpress/i18n'; import { BlockControls, - BlockIcon, InspectorControls, RichText, - __experimentalLinkControl as LinkControl, useBlockProps, store as blockEditorStore, getColorClassName, + useInnerBlocksProps, } from '@wordpress/block-editor'; -import { isURL, prependHTTP, safeDecodeURI } from '@wordpress/url'; +import { isURL, prependHTTP } from '@wordpress/url'; +import { Fragment, useState, useEffect, useRef } from '@wordpress/element'; import { - Fragment, - useState, - useEffect, - useRef, - createInterpolateElement, -} from '@wordpress/element'; -import { placeCaretAtHorizontalEdge } from '@wordpress/dom'; + placeCaretAtHorizontalEdge, + __unstableStripHTML as stripHTML, +} from '@wordpress/dom'; import { link as linkIcon, addSubmenu } from '@wordpress/icons'; import { store as coreStore, useResourcePermissions, } from '@wordpress/core-data'; -import { decodeEntities } from '@wordpress/html-entities'; import { useMergeRefs } from '@wordpress/compose'; /** * Internal dependencies */ import { name } from './block.json'; +import { LinkUI } from './link-ui'; +import { updateAttributes } from './update-attributes'; /** * A React hook to determine if it's dragging within the target element. @@ -106,36 +100,6 @@ const useIsDraggingWithin = ( elementRef ) => { return isDraggingWithin; }; -/** - * Given the Link block's type attribute, return the query params to give to - * /wp/v2/search. - * - * @param {string} type Link block's type attribute. - * @param {string} kind Link block's entity of kind (post-type|taxonomy) - * @return {{ type?: string, subtype?: string }} Search query params. - */ -function getSuggestionsQuery( type, kind ) { - switch ( type ) { - case 'post': - case 'page': - return { type: 'post', subtype: type }; - case 'category': - return { type: 'term', subtype: 'category' }; - case 'tag': - return { type: 'term', subtype: 'post_tag' }; - case 'post_format': - return { type: 'post-format' }; - default: - if ( kind === 'taxonomy' ) { - return { type: 'term', subtype: type }; - } - if ( kind === 'post-type' ) { - return { type: 'post', subtype: type }; - } - return {}; - } -} - /** * Determine the colors for a menu. * @@ -191,102 +155,6 @@ function getColors( context, isSubMenu ) { return colors; } -/** - * @typedef {'post-type'|'custom'|'taxonomy'|'post-type-archive'} WPNavigationLinkKind - */ - -/** - * Navigation Link Block Attributes - * - * @typedef {Object} WPNavigationLinkBlockAttributes - * - * @property {string} [label] Link text. - * @property {WPNavigationLinkKind} [kind] Kind is used to differentiate between term and post ids to check post draft status. - * @property {string} [type] The type such as post, page, tag, category and other custom types. - * @property {string} [rel] The relationship of the linked URL. - * @property {number} [id] A post or term id. - * @property {boolean} [opensInNewTab] Sets link target to _blank when true. - * @property {string} [url] Link href. - * @property {string} [title] Link title attribute. - */ - -/** - * Link Control onChange handler that updates block attributes when a setting is changed. - * - * @param {Object} updatedValue New block attributes to update. - * @param {Function} setAttributes Block attribute update function. - * @param {WPNavigationLinkBlockAttributes} blockAttributes Current block attributes. - * - */ -export const updateNavigationLinkBlockAttributes = ( - updatedValue = {}, - setAttributes, - blockAttributes = {} -) => { - const { - label: originalLabel = '', - kind: originalKind = '', - type: originalType = '', - } = blockAttributes; - - const { - title: newLabel = '', // the title of any provided Post. - url: newUrl = '', - - opensInNewTab, - id, - kind: newKind = originalKind, - type: newType = originalType, - } = updatedValue; - - const newLabelWithoutHttp = newLabel.replace( /http(s?):\/\//gi, '' ); - const newUrlWithoutHttp = newUrl.replace( /http(s?):\/\//gi, '' ); - - const useNewLabel = - newLabel && - newLabel !== originalLabel && - // LinkControl without the title field relies - // on the check below. Specifically, it assumes that - // the URL is the same as a title. - // This logic a) looks suspicious and b) should really - // live in the LinkControl and not here. It's a great - // candidate for future refactoring. - newLabelWithoutHttp !== newUrlWithoutHttp; - - // Unfortunately this causes the escaping model to be inverted. - // The escaped content is stored in the block attributes (and ultimately in the database), - // and then the raw data is "recovered" when outputting into the DOM. - // It would be preferable to store the **raw** data in the block attributes and escape it in JS. - // Why? Because there isn't one way to escape data. Depending on the context, you need to do - // different transforms. It doesn't make sense to me to choose one of them for the purposes of storage. - // See also: - // - https://github.com/WordPress/gutenberg/pull/41063 - // - https://github.com/WordPress/gutenberg/pull/18617. - const label = useNewLabel - ? escapeHtml( newLabel ) - : originalLabel || escapeHtml( newUrlWithoutHttp ); - - // In https://github.com/WordPress/gutenberg/pull/24670 we decided to use "tag" in favor of "post_tag" - const type = newType === 'post_tag' ? 'tag' : newType.replace( '-', '_' ); - - const isBuiltInType = - [ 'post', 'page', 'tag', 'category' ].indexOf( type ) > -1; - - const isCustomLink = - ( ! newKind && ! isBuiltInType ) || newKind === 'custom'; - const kind = isCustomLink ? 'custom' : newKind; - - setAttributes( { - // Passed `url` may already be encoded. To prevent double encoding, decodeURI is executed to revert to the original string. - ...( newUrl && { url: encodeURI( safeDecodeURI( newUrl ) ) } ), - ...( label && { label } ), - ...( undefined !== opensInNewTab && { opensInNewTab } ), - ...( id && Number.isInteger( id ) && { id } ), - ...( kind && { kind } ), - ...( type && type !== 'URL' && { type } ), - } ); -}; - const useIsInvalidLink = ( kind, type, id ) => { const isPostType = kind === 'post-type' || type === 'post' || type === 'page'; @@ -345,90 +213,6 @@ function getMissingText( type ) { return missingText; } -/** - * Removes HTML from a given string. - * Note the does not provide XSS protection or otherwise attempt - * to filter strings with malicious intent. - * - * See also: https://github.com/WordPress/gutenberg/pull/35539 - * - * @param {string} html the string from which HTML should be removed. - * @return {string} the "cleaned" string. - */ -function navStripHTML( html ) { - const doc = document.implementation.createHTMLDocument( '' ); - doc.body.innerHTML = html; - return doc.body.textContent || ''; -} - -/** - * Add transforms to Link Control - */ - -function LinkControlTransforms( { clientId, replace } ) { - const { getBlock, blockTransforms } = useSelect( - ( select ) => { - const { - getBlock: _getBlock, - getBlockRootClientId, - getBlockTransformItems, - } = select( blockEditorStore ); - - return { - getBlock: _getBlock, - blockTransforms: getBlockTransformItems( - _getBlock( clientId ), - getBlockRootClientId( clientId ) - ), - }; - }, - [ clientId ] - ); - - const featuredBlocks = [ - 'core/site-logo', - 'core/social-links', - 'core/search', - ]; - const transforms = blockTransforms.filter( ( item ) => { - return featuredBlocks.includes( item.name ); - } ); - - if ( ! transforms?.length ) { - return null; - } - - return ( -
    -

    - { __( 'Transform' ) } -

    -
    - { transforms.map( ( item, index ) => { - return ( - - ); - } ) } -
    -
    - ); -} - export default function NavigationLinkEdit( { attributes, isSelected, @@ -439,27 +223,11 @@ export default function NavigationLinkEdit( { context, clientId, } ) { - const { - id, - label, - type, - opensInNewTab, - url, - description, - rel, - title, - kind, - } = attributes; + const { id, label, type, url, description, rel, title, kind } = attributes; const [ isInvalid, isDraft ] = useIsInvalidLink( kind, type, id ); const { maxNestingLevel } = context; - const link = { - url, - opensInNewTab, - title: label && navStripHTML( label ), // don't allow HTML to display inside the - }; - const { saveEntityRecord } = useDispatch( coreStore ); const { replaceBlock, __unstableMarkNextChangeAsNotPersistent } = useDispatch( blockEditorStore ); const [ isLinkOpen, setIsLinkOpen ] = useState( false ); @@ -527,7 +295,9 @@ export default function NavigationLinkEdit( { const newSubmenu = createBlock( 'core/navigation-submenu', attributes, - innerBlocks + innerBlocks.length > 0 + ? innerBlocks + : [ createBlock( 'core/navigation-link' ) ] ); replaceBlock( clientId, newSubmenu ); } @@ -540,11 +310,14 @@ export default function NavigationLinkEdit( { if ( ! url ) { setIsLinkOpen( true ); } + }, [ url ] ); + + useEffect( () => { // If block has inner blocks, transform to Submenu. if ( hasChildren ) { transformToSubmenu(); } - }, [] ); + }, [ hasChildren ] ); /** * The hook shouldn't be necessary but due to a focus loss happening @@ -612,33 +385,6 @@ export default function NavigationLinkEdit( { userCanCreate = postsPermissions.canCreate; } - async function handleCreate( pageTitle ) { - const postType = type || 'page'; - - const page = await saveEntityRecord( 'postType', postType, { - title: pageTitle, - status: 'draft', - } ); - - return { - id: page.id, - type: postType, - // Make `title` property consistent with that in `fetchLinkSuggestions` where the `rendered` title (containing HTML entities) - // is also being decoded. By being consistent in both locations we avoid having to branch in the rendering output code. - // Ideally in the future we will update both APIs to utilise the "raw" form of the title which is better suited to edit contexts. - // e.g. - // - title.raw = "Yes & No" - // - title.rendered = "Yes & No" - // - decodeEntities( title.rendered ) = "Yes & No" - // See: - // - https://github.com/WordPress/gutenberg/pull/41063 - // - https://github.com/WordPress/gutenberg/blob/a1e1fdc0e6278457e9f4fc0b31ac6d2095f5450b/packages/core-data/src/fetch/__experimental-fetch-link-suggestions.js#L212-L218 - title: decodeEntities( page.title.rendered ), - url: page.link, - kind: 'post-type', - }; - } - const { textColor, customTextColor, @@ -675,6 +421,21 @@ export default function NavigationLinkEdit( { onKeyDown, } ); + const ALLOWED_BLOCKS = [ + 'core/navigation-link', + 'core/navigation-submenu', + 'core/page-list', + ]; + const DEFAULT_BLOCK = { + name: 'core/navigation-link', + }; + const innerBlocksProps = useInnerBlocksProps( blockProps, { + allowedBlocks: ALLOWED_BLOCKS, + __experimentalDefaultBlock: DEFAULT_BLOCK, + __experimentalDirectInsert: true, + renderAppender: false, + } ); + if ( ! url || isInvalid || isDraft ) { blockProps.onClick = () => setIsLinkOpen( true ); } @@ -714,8 +475,17 @@ export default function NavigationLinkEdit( { ) } + { /* Warning, this duplicated in packages/block-library/src/navigation-submenu/edit.js */ } + { + setAttributes( { label: labelValue } ); + } } + label={ __( 'Label' ) } + autoComplete="off" + /> { @@ -838,7 +608,7 @@ export default function NavigationLinkEdit( { // Ideally they would be stored in a raw, unescaped form. // Unescape is used here to "recover" the escaped characters // so they display without encoding. - // See `updateNavigationLinkBlockAttributes` for more details. + // See `updateAttributes` for more details. `${ unescape( label ) } ${ placeholderText }`.trim() @@ -854,66 +624,25 @@ export default function NavigationLinkEdit( { ) } { isLinkOpen && ( - setIsLinkOpen( false ) } anchor={ popoverAnchor } - shift - > - { - let format; - if ( type === 'post' ) { - /* translators: %s: search term. */ - format = __( - 'Create draft post: %s' - ); - } else { - /* translators: %s: search term. */ - format = __( - 'Create draft page: %s' - ); - } - return createInterpolateElement( - sprintf( format, searchTerm ), - { mark: } - ); - } } - noDirectEntry={ !! type } - noURLSuggestion={ !! type } - suggestionsQuery={ getSuggestionsQuery( - type, - kind - ) } - onChange={ ( updatedValue ) => - updateNavigationLinkBlockAttributes( - updatedValue, - setAttributes, - attributes - ) - } - onRemove={ removeLink } - renderControlBottom={ - ! url - ? () => ( - - ) - : null - } - /> - + hasCreateSuggestion={ userCanCreate } + onRemove={ removeLink } + onChange={ ( updatedValue ) => { + updateAttributes( + updatedValue, + setAttributes, + attributes + ); + } } + /> ) } +
    ); diff --git a/packages/block-library/src/navigation-link/index.php b/packages/block-library/src/navigation-link/index.php index f32437b41a3f8..8853d49f6e2cf 100644 --- a/packages/block-library/src/navigation-link/index.php +++ b/packages/block-library/src/navigation-link/index.php @@ -344,3 +344,35 @@ function register_block_core_navigation_link() { ); } add_action( 'init', 'register_block_core_navigation_link' ); + +/** + * Disables the display of block inspector tabs for the Navigation Link block. + * + * This is only a temporary measure until we have a TabPanel and mechanism that + * will allow the Navigation Link to programmatically select a tab when edited + * via a specific context. + * + * See: + * - https://github.com/WordPress/gutenberg/issues/45951 + * - https://github.com/WordPress/gutenberg/pull/46321 + * - https://github.com/WordPress/gutenberg/pull/46271 + * + * @param array $settings Default editor settings. + * @return array Filtered editor settings. + */ +function gutenberg_disable_tabs_for_navigation_link_block( $settings ) { + $current_tab_settings = _wp_array_get( + $settings, + array( '__experimentalBlockInspectorTabs' ), + array() + ); + + $settings['__experimentalBlockInspectorTabs'] = array_merge( + $current_tab_settings, + array( 'core/navigation-link' => false ) + ); + + return $settings; +} + +add_filter( 'block_editor_settings_all', 'gutenberg_disable_tabs_for_navigation_link_block' ); diff --git a/packages/block-library/src/navigation-link/link-ui.js b/packages/block-library/src/navigation-link/link-ui.js new file mode 100644 index 0000000000000..a2df4f2a29405 --- /dev/null +++ b/packages/block-library/src/navigation-link/link-ui.js @@ -0,0 +1,213 @@ +/** + * WordPress dependencies + */ +import { __unstableStripHTML as stripHTML } from '@wordpress/dom'; +import { Popover, Button } from '@wordpress/components'; +import { __, sprintf } from '@wordpress/i18n'; +import { + __experimentalLinkControl as LinkControl, + BlockIcon, + store as blockEditorStore, +} from '@wordpress/block-editor'; +import { createInterpolateElement } from '@wordpress/element'; +import { store as coreStore } from '@wordpress/core-data'; +import { decodeEntities } from '@wordpress/html-entities'; +import { switchToBlockType } from '@wordpress/blocks'; +import { useSelect, useDispatch } from '@wordpress/data'; + +/** + * Given the Link block's type attribute, return the query params to give to + * /wp/v2/search. + * + * @param {string} type Link block's type attribute. + * @param {string} kind Link block's entity of kind (post-type|taxonomy) + * @return {{ type?: string, subtype?: string }} Search query params. + */ +export function getSuggestionsQuery( type, kind ) { + switch ( type ) { + case 'post': + case 'page': + return { type: 'post', subtype: type }; + case 'category': + return { type: 'term', subtype: 'category' }; + case 'tag': + return { type: 'term', subtype: 'post_tag' }; + case 'post_format': + return { type: 'post-format' }; + default: + if ( kind === 'taxonomy' ) { + return { type: 'term', subtype: type }; + } + if ( kind === 'post-type' ) { + return { type: 'post', subtype: type }; + } + return {}; + } +} + +/** + * Add transforms to Link Control + * + * @param {Object} props Component props. + * @param {string} props.clientId Block client ID. + */ +function LinkControlTransforms( { clientId } ) { + const { getBlock, blockTransforms } = useSelect( + ( select ) => { + const { + getBlock: _getBlock, + getBlockRootClientId, + getBlockTransformItems, + } = select( blockEditorStore ); + + return { + getBlock: _getBlock, + blockTransforms: getBlockTransformItems( + _getBlock( clientId ), + getBlockRootClientId( clientId ) + ), + }; + }, + [ clientId ] + ); + + const { replaceBlock } = useDispatch( blockEditorStore ); + + const featuredBlocks = [ + 'core/page-list', + 'core/site-logo', + 'core/social-links', + 'core/search', + ]; + + const transforms = blockTransforms.filter( ( item ) => { + return featuredBlocks.includes( item.name ); + } ); + + if ( ! transforms?.length ) { + return null; + } + + if ( ! clientId ) { + return null; + } + + return ( +
    +

    + { __( 'Transform' ) } +

    +
    + { transforms.map( ( item, index ) => { + return ( + + ); + } ) } +
    +
    + ); +} + +export function LinkUI( props ) { + const { saveEntityRecord } = useDispatch( coreStore ); + + async function handleCreate( pageTitle ) { + const postType = props.link.type || 'page'; + + const page = await saveEntityRecord( 'postType', postType, { + title: pageTitle, + status: 'draft', + } ); + + return { + id: page.id, + type: postType, + // Make `title` property consistent with that in `fetchLinkSuggestions` where the `rendered` title (containing HTML entities) + // is also being decoded. By being consistent in both locations we avoid having to branch in the rendering output code. + // Ideally in the future we will update both APIs to utilise the "raw" form of the title which is better suited to edit contexts. + // e.g. + // - title.raw = "Yes & No" + // - title.rendered = "Yes & No" + // - decodeEntities( title.rendered ) = "Yes & No" + // See: + // - https://github.com/WordPress/gutenberg/pull/41063 + // - https://github.com/WordPress/gutenberg/blob/a1e1fdc0e6278457e9f4fc0b31ac6d2095f5450b/packages/core-data/src/fetch/__experimental-fetch-link-suggestions.js#L212-L218 + title: decodeEntities( page.title.rendered ), + url: page.link, + kind: 'post-type', + }; + } + + const { label, url, opensInNewTab, type, kind } = props.link; + const link = { + url, + opensInNewTab, + title: label && stripHTML( label ), + }; + + return ( + + { + let format; + + if ( type === 'post' ) { + /* translators: %s: search term. */ + format = __( 'Create draft post: %s' ); + } else { + /* translators: %s: search term. */ + format = __( 'Create draft page: %s' ); + } + + return createInterpolateElement( + sprintf( format, searchTerm ), + { + mark: , + } + ); + } } + noDirectEntry={ !! type } + noURLSuggestion={ !! type } + suggestionsQuery={ getSuggestionsQuery( type, kind ) } + onChange={ props.onChange } + onRemove={ props.onRemove } + renderControlBottom={ + ! url + ? () => ( + + ) + : null + } + /> + + ); +} diff --git a/packages/block-library/src/navigation-link/test/edit.js b/packages/block-library/src/navigation-link/test/edit.js index 8d052a5e3f133..ec396a64b2058 100644 --- a/packages/block-library/src/navigation-link/test/edit.js +++ b/packages/block-library/src/navigation-link/test/edit.js @@ -1,10 +1,10 @@ /** * Internal dependencies */ -import { updateNavigationLinkBlockAttributes } from '../edit'; +import { updateAttributes } from '../update-attributes'; describe( 'edit', () => { - describe( 'updateNavigationLinkBlockAttributes', () => { + describe( 'updateAttributes', () => { // Data shapes are linked to fetchLinkSuggestions from // core-data/src/fetch/__experimental-fetch-link-suggestions.js. it( 'can update a post link', () => { @@ -18,10 +18,7 @@ describe( 'edit', () => { type: 'post', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 1337, label: 'Menu Test', @@ -42,10 +39,7 @@ describe( 'edit', () => { type: 'page', url: 'http://wordpress.local/sample-page/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 2, kind: 'post-type', @@ -66,10 +60,7 @@ describe( 'edit', () => { type: 'post_tag', url: 'http://wordpress.local/tag/bar/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 15, kind: 'taxonomy', @@ -90,10 +81,7 @@ describe( 'edit', () => { type: 'category', url: 'http://wordpress.local/category/cats/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 9, kind: 'taxonomy', @@ -114,10 +102,7 @@ describe( 'edit', () => { type: 'portfolio', url: 'http://wordpress.local/portfolio/fall/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 131, kind: 'post-type', @@ -138,10 +123,7 @@ describe( 'edit', () => { type: 'portfolio_tag', url: 'http://wordpress.local/portfolio_tag/PortfolioTag/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 4, kind: 'taxonomy', @@ -162,10 +144,7 @@ describe( 'edit', () => { type: 'portfolio_category', url: 'http://wordpress.local/portfolio_category/Portfolio-category/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { id: 2, kind: 'taxonomy', @@ -186,10 +165,7 @@ describe( 'edit', () => { type: 'post-format', url: 'http://wordpress.local/type/video/', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); // post_format returns a slug ID value from the Search API // we do not persist this ID since we expect this value to be a post or term ID. expect( setAttributes ).toHaveBeenCalledWith( { @@ -208,10 +184,7 @@ describe( 'edit', () => { opensInNewTab: false, url: 'www.wordpress.org', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, url: 'www.wordpress.org', @@ -229,10 +202,7 @@ describe( 'edit', () => { type: 'URL', url: 'http://www.wordpress.org', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'www.wordpress.org', @@ -250,10 +220,7 @@ describe( 'edit', () => { type: 'mailto', url: 'mailto:foo@example.com', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'mailto:foo@example.com', @@ -272,10 +239,7 @@ describe( 'edit', () => { type: 'internal', url: '#foo', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: '#foo', @@ -294,10 +258,7 @@ describe( 'edit', () => { type: 'tel', url: 'tel:5555555', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'tel:5555555', @@ -319,10 +280,7 @@ describe( 'edit', () => { type: 'URL', url: 'https://www.wordpress.org', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'www.wordpress.org', @@ -339,10 +297,7 @@ describe( 'edit', () => { type: 'URL', url: 'wordpress.org', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'Custom Title', @@ -359,10 +314,7 @@ describe( 'edit', () => { type: 'URL', url: 'https://wordpress.org', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'Custom Title', @@ -380,10 +332,7 @@ describe( 'edit', () => { type: 'URL', url: 'http://wordpress.org/?s=<>', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes - ); + updateAttributes( linkSuggestion, setAttributes ); expect( setAttributes ).toHaveBeenCalledWith( { opensInNewTab: false, label: 'Custom Title', @@ -408,11 +357,7 @@ describe( 'edit', () => { type: 'post', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes, - mockState - ); + updateAttributes( linkSuggestion, setAttributes, mockState ); expect( mockState ).toEqual( { id: 1337, label: 'Menu Test', @@ -422,7 +367,7 @@ describe( 'edit', () => { url: 'https://wordpress.local/menu-test/', } ); // Click on the existing link control, and toggle opens new tab. - updateNavigationLinkBlockAttributes( + updateAttributes( { url: 'https://wordpress.local/menu-test/', opensInNewTab: true, @@ -453,11 +398,7 @@ describe( 'edit', () => { type: 'post', }; - updateNavigationLinkBlockAttributes( - linkSuggestion, - setAttributes, - mockState - ); + updateAttributes( linkSuggestion, setAttributes, mockState ); expect( mockState ).toEqual( { id: 1337, label: 'Menu Test', @@ -467,7 +408,7 @@ describe( 'edit', () => { url: 'https://wordpress.local/menu-test/', } ); // Click on the existing link control, and toggle opens new tab. - updateNavigationLinkBlockAttributes( + updateAttributes( { url: 'https://wordpress.local/foo/', opensInNewTab: false, diff --git a/packages/block-library/src/navigation-link/transforms.js b/packages/block-library/src/navigation-link/transforms.js index 7b213a4805106..fb450a13a02dc 100644 --- a/packages/block-library/src/navigation-link/transforms.js +++ b/packages/block-library/src/navigation-link/transforms.js @@ -40,6 +40,13 @@ const transforms = { return createBlock( 'core/navigation-link' ); }, }, + { + type: 'block', + blocks: [ 'core/page-list' ], + transform: () => { + return createBlock( 'core/navigation-link' ); + }, + }, ], to: [ { @@ -91,6 +98,13 @@ const transforms = { } ); }, }, + { + type: 'block', + blocks: [ 'core/page-list' ], + transform: () => { + return createBlock( 'core/page-list' ); + }, + }, ], }; diff --git a/packages/block-library/src/navigation-link/update-attributes.js b/packages/block-library/src/navigation-link/update-attributes.js new file mode 100644 index 0000000000000..4b7ee45ac9911 --- /dev/null +++ b/packages/block-library/src/navigation-link/update-attributes.js @@ -0,0 +1,99 @@ +/** + * WordPress dependencies + */ +import { safeDecodeURI } from '@wordpress/url'; +import { escapeHTML } from '@wordpress/escape-html'; + +/** + * @typedef {'post-type'|'custom'|'taxonomy'|'post-type-archive'} WPNavigationLinkKind + */ +/** + * Navigation Link Block Attributes + * + * @typedef {Object} WPNavigationLinkBlockAttributes + * + * @property {string} [label] Link text. + * @property {WPNavigationLinkKind} [kind] Kind is used to differentiate between term and post ids to check post draft status. + * @property {string} [type] The type such as post, page, tag, category and other custom types. + * @property {string} [rel] The relationship of the linked URL. + * @property {number} [id] A post or term id. + * @property {boolean} [opensInNewTab] Sets link target to _blank when true. + * @property {string} [url] Link href. + * @property {string} [title] Link title attribute. + */ +/** + * Link Control onChange handler that updates block attributes when a setting is changed. + * + * @param {Object} updatedValue New block attributes to update. + * @param {Function} setAttributes Block attribute update function. + * @param {WPNavigationLinkBlockAttributes} blockAttributes Current block attributes. + * + */ + +export const updateAttributes = ( + updatedValue = {}, + setAttributes, + blockAttributes = {} +) => { + const { + label: originalLabel = '', + kind: originalKind = '', + type: originalType = '', + } = blockAttributes; + + const { + title: newLabel = '', // the title of any provided Post. + url: newUrl = '', + opensInNewTab, + id, + kind: newKind = originalKind, + type: newType = originalType, + } = updatedValue; + + const newLabelWithoutHttp = newLabel.replace( /http(s?):\/\//gi, '' ); + const newUrlWithoutHttp = newUrl.replace( /http(s?):\/\//gi, '' ); + + const useNewLabel = + newLabel && + newLabel !== originalLabel && + // LinkControl without the title field relies + // on the check below. Specifically, it assumes that + // the URL is the same as a title. + // This logic a) looks suspicious and b) should really + // live in the LinkControl and not here. It's a great + // candidate for future refactoring. + newLabelWithoutHttp !== newUrlWithoutHttp; + + // Unfortunately this causes the escaping model to be inverted. + // The escaped content is stored in the block attributes (and ultimately in the database), + // and then the raw data is "recovered" when outputting into the DOM. + // It would be preferable to store the **raw** data in the block attributes and escape it in JS. + // Why? Because there isn't one way to escape data. Depending on the context, you need to do + // different transforms. It doesn't make sense to me to choose one of them for the purposes of storage. + // See also: + // - https://github.com/WordPress/gutenberg/pull/41063 + // - https://github.com/WordPress/gutenberg/pull/18617. + const label = useNewLabel + ? escapeHTML( newLabel ) + : originalLabel || escapeHTML( newUrlWithoutHttp ); + + // In https://github.com/WordPress/gutenberg/pull/24670 we decided to use "tag" in favor of "post_tag" + const type = newType === 'post_tag' ? 'tag' : newType.replace( '-', '_' ); + + const isBuiltInType = + [ 'post', 'page', 'tag', 'category' ].indexOf( type ) > -1; + + const isCustomLink = + ( ! newKind && ! isBuiltInType ) || newKind === 'custom'; + const kind = isCustomLink ? 'custom' : newKind; + + setAttributes( { + // Passed `url` may already be encoded. To prevent double encoding, decodeURI is executed to revert to the original string. + ...( newUrl && { url: encodeURI( safeDecodeURI( newUrl ) ) } ), + ...( label && { label } ), + ...( undefined !== opensInNewTab && { opensInNewTab } ), + ...( id && Number.isInteger( id ) && { id } ), + ...( kind && { kind } ), + ...( type && type !== 'URL' && { type } ), + } ); +}; diff --git a/packages/block-library/src/navigation-submenu/edit.js b/packages/block-library/src/navigation-submenu/edit.js index 0f91313289d0b..47a222bd57bb6 100644 --- a/packages/block-library/src/navigation-submenu/edit.js +++ b/packages/block-library/src/navigation-submenu/edit.js @@ -2,7 +2,6 @@ * External dependencies */ import classnames from 'classnames'; -import escapeHtml from 'escape-html'; /** * WordPress dependencies @@ -10,50 +9,45 @@ import escapeHtml from 'escape-html'; import { useSelect, useDispatch } from '@wordpress/data'; import { PanelBody, - Popover, TextControl, TextareaControl, ToolbarButton, ToolbarGroup, } from '@wordpress/components'; import { displayShortcut, isKeyboardEvent } from '@wordpress/keycodes'; -import { __, sprintf } from '@wordpress/i18n'; +import { __ } from '@wordpress/i18n'; import { BlockControls, InnerBlocks, useInnerBlocksProps, InspectorControls, RichText, - __experimentalLinkControl as LinkControl, useBlockProps, store as blockEditorStore, getColorClassName, } from '@wordpress/block-editor'; -import { isURL, prependHTTP, safeDecodeURI } from '@wordpress/url'; -import { - Fragment, - useState, - useEffect, - useRef, - createInterpolateElement, -} from '@wordpress/element'; +import { isURL, prependHTTP } from '@wordpress/url'; +import { Fragment, useState, useEffect, useRef } from '@wordpress/element'; import { placeCaretAtHorizontalEdge } from '@wordpress/dom'; import { link as linkIcon, removeSubmenu } from '@wordpress/icons'; -import { - useResourcePermissions, - store as coreStore, -} from '@wordpress/core-data'; +import { useResourcePermissions } from '@wordpress/core-data'; import { speak } from '@wordpress/a11y'; import { createBlock } from '@wordpress/blocks'; -import { useMergeRefs } from '@wordpress/compose'; +import { useMergeRefs, usePrevious } from '@wordpress/compose'; /** * Internal dependencies */ import { ItemSubmenuIcon } from './icons'; import { name } from './block.json'; +import { LinkUI } from '../navigation-link/link-ui'; +import { updateAttributes } from '../navigation-link/update-attributes'; -const ALLOWED_BLOCKS = [ 'core/navigation-link', 'core/navigation-submenu' ]; +const ALLOWED_BLOCKS = [ + 'core/navigation-link', + 'core/navigation-submenu', + 'core/page-list', +]; const DEFAULT_BLOCK = { name: 'core/navigation-link', @@ -110,36 +104,6 @@ const useIsDraggingWithin = ( elementRef ) => { return isDraggingWithin; }; -/** - * Given the Link block's type attribute, return the query params to give to - * /wp/v2/search. - * - * @param {string} type Link block's type attribute. - * @param {string} kind Link block's entity of kind (post-type|taxonomy) - * @return {{ type?: string, subtype?: string }} Search query params. - */ -function getSuggestionsQuery( type, kind ) { - switch ( type ) { - case 'post': - case 'page': - return { type: 'post', subtype: type }; - case 'category': - return { type: 'term', subtype: 'category' }; - case 'tag': - return { type: 'term', subtype: 'post_tag' }; - case 'post_format': - return { type: 'post-format' }; - default: - if ( kind === 'taxonomy' ) { - return { type: 'term', subtype: type }; - } - if ( kind === 'post-type' ) { - return { type: 'post', subtype: type }; - } - return {}; - } -} - /** * Determine the colors for a menu. * @@ -214,64 +178,6 @@ function getColors( context, isSubMenu ) { * @property {string} [title] Link title attribute. */ -/** - * Link Control onChange handler that updates block attributes when a setting is changed. - * - * @param {Object} updatedValue New block attributes to update. - * @param {Function} setAttributes Block attribute update function. - * @param {WPNavigationLinkBlockAttributes} blockAttributes Current block attributes. - * - */ -export const updateNavigationLinkBlockAttributes = ( - updatedValue = {}, - setAttributes, - blockAttributes = {} -) => { - const { - label: originalLabel = '', - kind: originalKind = '', - type: originalType = '', - } = blockAttributes; - const { - title = '', - url = '', - opensInNewTab, - id, - kind: newKind = originalKind, - type: newType = originalType, - } = updatedValue; - - const normalizedTitle = title.replace( /http(s?):\/\//gi, '' ); - const normalizedURL = url.replace( /http(s?):\/\//gi, '' ); - const escapeTitle = - title !== '' && - normalizedTitle !== normalizedURL && - originalLabel !== title; - const label = escapeTitle - ? escapeHtml( title ) - : originalLabel || escapeHtml( normalizedURL ); - - // In https://github.com/WordPress/gutenberg/pull/24670 we decided to use "tag" in favor of "post_tag" - const type = newType === 'post_tag' ? 'tag' : newType.replace( '-', '_' ); - - const isBuiltInType = - [ 'post', 'page', 'tag', 'category' ].indexOf( type ) > -1; - - const isCustomLink = - ( ! newKind && ! isBuiltInType ) || newKind === 'custom'; - const kind = isCustomLink ? 'custom' : newKind; - - setAttributes( { - // Passed `url` may already be encoded. To prevent double encoding, decodeURI is executed to revert to the original string. - ...( url && { url: encodeURI( safeDecodeURI( url ) ) } ), - ...( label && { label } ), - ...( undefined !== opensInNewTab && { opensInNewTab } ), - ...( id && Number.isInteger( id ) && { id } ), - ...( kind && { kind } ), - ...( type && type !== 'URL' && { type } ), - } ); -}; - export default function NavigationSubmenuEdit( { attributes, isSelected, @@ -281,14 +187,9 @@ export default function NavigationSubmenuEdit( { context, clientId, } ) { - const { label, type, opensInNewTab, url, description, rel, title, kind } = - attributes; - const link = { - url, - opensInNewTab, - }; + const { label, type, url, description, rel, title } = attributes; + const { showSubmenuIcon, maxNestingLevel, openSubmenusOnClick } = context; - const { saveEntityRecord } = useDispatch( coreStore ); const { __unstableMarkNextChangeAsNotPersistent, replaceBlock } = useDispatch( blockEditorStore ); @@ -362,6 +263,8 @@ export default function NavigationSubmenuEdit( { [ clientId ] ); + const prevHasChildren = usePrevious( hasChildren ); + // Show the LinkControl on mount if the URL is empty // ( When adding a new menu item) // This can't be done in the useState call because it conflicts @@ -431,23 +334,6 @@ export default function NavigationSubmenuEdit( { userCanCreate = postsPermissions.canCreate; } - async function handleCreate( pageTitle ) { - const postType = type || 'page'; - - const page = await saveEntityRecord( 'postType', postType, { - title: pageTitle, - status: 'draft', - } ); - - return { - id: page.id, - type: postType, - title: page.title.rendered, - url: page.link, - kind: 'post-type', - }; - } - const { textColor, customTextColor, @@ -541,6 +427,13 @@ export default function NavigationSubmenuEdit( { replaceBlock( clientId, newLinkBlock ); } + useEffect( () => { + // If block becomes empty, transform to Navigation Link. + if ( ! hasChildren && prevHasChildren ) { + transformToLink(); + } + }, [ hasChildren, prevHasChildren ] ); + const canConvertToLink = ! selectedBlockHasChildren || onlyDescendantIsEmptyLink; @@ -568,8 +461,25 @@ export default function NavigationSubmenuEdit( { /> + { /* Warning, this duplicated in packages/block-library/src/navigation-link/edit.js */ } + { + setAttributes( { label: labelValue } ); + } } + label={ __( 'Label' ) } + autoComplete="off" + /> + { + setAttributes( { url: urlValue } ); + } } + label={ __( 'URL' ) } + autoComplete="off" + /> { @@ -632,55 +542,25 @@ export default function NavigationSubmenuEdit( { /> } { ! openSubmenusOnClick && isLinkOpen && ( - setIsLinkOpen( false ) } anchor={ popoverAnchor } - shift - > - { - let format; - if ( type === 'post' ) { - /* translators: %s: search term. */ - format = __( - 'Create draft post: %s' - ); - } else { - /* translators: %s: search term. */ - format = __( - 'Create draft page: %s' - ); - } - return createInterpolateElement( - sprintf( format, searchTerm ), - { mark: } - ); - } } - noDirectEntry={ !! type } - noURLSuggestion={ !! type } - suggestionsQuery={ getSuggestionsQuery( - type, - kind - ) } - onChange={ ( updatedValue ) => - updateNavigationLinkBlockAttributes( - updatedValue, - setAttributes, - attributes - ) - } - onRemove={ () => { - setAttributes( { url: '' } ); - speak( __( 'Link removed.' ), 'assertive' ); - } } - /> - + hasCreateSuggestion={ userCanCreate } + onRemove={ () => { + setAttributes( { url: '' } ); + speak( __( 'Link removed.' ), 'assertive' ); + } } + onChange={ ( updatedValue ) => { + updateAttributes( + updatedValue, + setAttributes, + attributes + ); + } } + /> ) } { ( showSubmenuIcon || openSubmenusOnClick ) && ( diff --git a/packages/block-library/src/navigation-submenu/index.php b/packages/block-library/src/navigation-submenu/index.php index 87c7c91481cdd..99c86493d5b31 100644 --- a/packages/block-library/src/navigation-submenu/index.php +++ b/packages/block-library/src/navigation-submenu/index.php @@ -255,6 +255,14 @@ function render_block_core_navigation_submenu( $attributes, $content, $block ) { $inner_blocks_html .= $inner_block->render(); } + if ( strpos( $inner_blocks_html, 'current-menu-item' ) ) { + $tag_processor = new WP_HTML_Tag_Processor( $html ); + while ( $tag_processor->next_tag( array( 'class_name' => 'wp-block-navigation-item__content' ) ) ) { + $tag_processor->add_class( 'current-menu-ancestor' ); + } + $html = $tag_processor->get_updated_html(); + } + $html .= sprintf( '
      %s
    ', $inner_blocks_html @@ -281,3 +289,35 @@ function register_block_core_navigation_submenu() { ); } add_action( 'init', 'register_block_core_navigation_submenu' ); + +/** + * Disables display of block inspector tabs for the Navigation Submenu block. + * + * This is only a temporary measure until we have a TabPanel and mechanism that + * will allow the Navigation Submenu to programmatically select a tab when + * edited via a specific context. + * + * See: + * - https://github.com/WordPress/gutenberg/issues/45951 + * - https://github.com/WordPress/gutenberg/pull/46321 + * - https://github.com/WordPress/gutenberg/pull/46271 + * + * @param array $settings Default editor settings. + * @return array Filtered editor settings. + */ +function gutenberg_disable_tabs_for_navigation_submenu_block( $settings ) { + $current_tab_settings = _wp_array_get( + $settings, + array( '__experimentalBlockInspectorTabs' ), + array() + ); + + $settings['__experimentalBlockInspectorTabs'] = array_merge( + $current_tab_settings, + array( 'core/navigation-submenu' => false ) + ); + + return $settings; +} + +add_filter( 'block_editor_settings_all', 'gutenberg_disable_tabs_for_navigation_submenu_block' ); diff --git a/packages/block-library/src/navigation/edit/are-blocks-dirty.js b/packages/block-library/src/navigation/edit/are-blocks-dirty.js new file mode 100644 index 0000000000000..1d7fa8a7286f2 --- /dev/null +++ b/packages/block-library/src/navigation/edit/are-blocks-dirty.js @@ -0,0 +1,51 @@ +export function areBlocksDirty( originalBlocks, blocks ) { + return ! isDeepEqual( originalBlocks, blocks, ( prop, x ) => { + // Skip inner blocks of page list during comparison as they + // are **always** controlled and may be updated async due to + // syncing with entity records. Left unchecked this would + // inadvertently trigger the dirty state. + if ( x?.name === 'core/page-list' && prop === 'innerBlocks' ) { + return true; + } + } ); +} + +/** + * Conditionally compares two candidates for deep equality. + * Provides an option to skip a given property of an object during comparison. + * + * @param {*} x 1st candidate for comparison + * @param {*} y 2nd candidate for comparison + * @param {Function|undefined} shouldSkip a function which can be used to skip a given property of an object. + * @return {boolean} whether the two candidates are deeply equal. + */ +const isDeepEqual = ( x, y, shouldSkip ) => { + if ( x === y ) { + return true; + } else if ( + typeof x === 'object' && + x !== null && + x !== undefined && + typeof y === 'object' && + y !== null && + y !== undefined + ) { + if ( Object.keys( x ).length !== Object.keys( y ).length ) return false; + + for ( const prop in x ) { + if ( y.hasOwnProperty( prop ) ) { + // Afford skipping a given property of an object. + if ( shouldSkip && shouldSkip( prop, x ) ) { + return true; + } + + if ( ! isDeepEqual( x[ prop ], y[ prop ], shouldSkip ) ) + return false; + } else return false; + } + + return true; + } + + return false; +}; diff --git a/packages/block-library/src/navigation/edit/index.js b/packages/block-library/src/navigation/edit/index.js index e15619a2cc74d..b414409791cbf 100644 --- a/packages/block-library/src/navigation/edit/index.js +++ b/packages/block-library/src/navigation/edit/index.js @@ -14,18 +14,18 @@ import { useMemo, } from '@wordpress/element'; import { - __experimentalOffCanvasEditor as OffCanvasEditor, InspectorControls, useBlockProps, __experimentalRecursionProvider as RecursionProvider, __experimentalUseHasRecursion as useHasRecursion, store as blockEditorStore, withColors, - PanelColorSettings, ContrastChecker, getColorClassName, Warning, + __experimentalColorGradientSettingsDropdown as ColorGradientSettingsDropdown, __experimentalUseBlockOverlayActive as useBlockOverlayActive, + __experimentalUseMultipleOriginColorsAndGradients as useMultipleOriginColorsAndGradients, } from '@wordpress/block-editor'; import { EntityProvider, store as coreStore } from '@wordpress/core-data'; @@ -37,8 +37,6 @@ import { __experimentalToggleGroupControlOption as ToggleGroupControlOption, Button, Spinner, - __experimentalHStack as HStack, - __experimentalHeading as Heading, } from '@wordpress/components'; import { __, sprintf } from '@wordpress/i18n'; import { speak } from '@wordpress/a11y'; @@ -53,7 +51,6 @@ import useNavigationEntities from '../use-navigation-entities'; import Placeholder from './placeholder'; import ResponsiveWrapper from './responsive-wrapper'; import NavigationInnerBlocks from './inner-blocks'; -import NavigationMenuSelector from './navigation-menu-selector'; import NavigationMenuNameControl from './navigation-menu-name-control'; import UnsavedInnerBlocks from './unsaved-inner-blocks'; import NavigationMenuDeleteControl from './navigation-menu-delete-control'; @@ -69,6 +66,7 @@ import useCreateNavigationMenu from './use-create-navigation-menu'; import { useInnerBlocks } from './use-inner-blocks'; import { detectColors } from './utils'; import ManageMenusButton from './manage-menus-button'; +import MenuInspectorControls from './menu-inspector-controls'; function Navigation( { attributes, @@ -286,18 +284,37 @@ function Navigation( { ! hasResolvedNavigationMenus || isConvertingClassicMenu || fallbackNavigationMenus?.length > 0 || - classicMenus?.length !== 1 + ! classicMenus?.length ) { return; } // If there's non fallback navigation menus and - // only one classic menu then create a new navigation menu based on it. - convertClassicMenu( - classicMenus[ 0 ].id, - classicMenus[ 0 ].name, - 'publish' + // a classic menu with a `primary` location or slug, + // then create a new navigation menu based on it. + // Otherwise, use the most recently created classic menu. + const primaryMenus = classicMenus.filter( + ( classicMenu ) => + classicMenu.locations.includes( 'primary' ) || + classicMenu.slug === 'primary' ); + + if ( primaryMenus.length ) { + convertClassicMenu( + primaryMenus[ 0 ].id, + primaryMenus[ 0 ].name, + 'publish' + ); + } else { + classicMenus.sort( ( a, b ) => { + return b.id - a.id; + } ); + convertClassicMenu( + classicMenus[ 0 ].id, + classicMenus[ 0 ].name, + 'publish' + ); + } }, [ hasResolvedNavigationMenus ] ); const navRef = useRef(); @@ -396,6 +413,23 @@ function Navigation( { } }; + const onSelectClassicMenu = async ( classicMenu ) => { + const navMenu = await convertClassicMenu( + classicMenu.id, + classicMenu.name, + 'draft' + ); + if ( navMenu ) { + handleUpdateMenu( navMenu.id, { + focusNavigationBlock: true, + } ); + } + }; + + const onSelectNavigationMenu = ( menuId ) => { + handleUpdateMenu( menuId ); + }; + useEffect( () => { hideClassicMenuConversionNotice(); if ( classicMenuConversionStatus === CLASSIC_MENU_CONVERSION_PENDING ) { @@ -515,144 +549,157 @@ function Navigation( { } ); }, [ isDraftNavigationMenu, navigationMenu ] ); + const colorGradientSettings = useMultipleOriginColorsAndGradients(); const stylingInspectorControls = ( - - { hasSubmenuIndicatorSetting && ( - - { isResponsive && ( - <> - + { overlayMenuPreview && ( + ) } - - { overlayMenuPreview && ( - - ) } - - ) } -

    { __( 'Overlay Menu' ) }

    - ) } - onChange={ ( value ) => - setAttributes( { overlayMenu: value } ) - } - isBlock - hideLabelFromVision - > - - - - - { hasSubmenus && ( - <> -

    { __( 'Submenus' ) }

    - { - setAttributes( { - openSubmenusOnClick: value, - ...( value && { - showSubmenuIcon: true, - } ), // Make sure arrows are shown when we toggle this on. - } ); - } } - label={ __( 'Open on click' ) } - /> - - { - setAttributes( { - showSubmenuIcon: value, - } ); - } } - disabled={ attributes.openSubmenusOnClick } - label={ __( 'Show arrow' ) } +

    { __( 'Overlay Menu' ) }

    + + setAttributes( { overlayMenu: value } ) + } + isBlock + hideLabelFromVision + > + - - ) } -
    - ) } - { hasColorSettings && ( - - { enableContrastChecking && ( - <> - - - - ) } - - ) } -
    + + { hasSubmenus && ( + <> +

    { __( 'Submenus' ) }

    + { + setAttributes( { + openSubmenusOnClick: value, + ...( value && { + showSubmenuIcon: true, + } ), // Make sure arrows are shown when we toggle this on. + } ); + } } + label={ __( 'Open on click' ) } + /> + + { + setAttributes( { + showSubmenuIcon: value, + } ); + } } + disabled={ attributes.openSubmenusOnClick } + label={ __( 'Show arrow' ) } + /> + + ) } +
    + ) } +
    + + { hasColorSettings && ( + <> + setTextColor(), + }, + { + colorValue: backgroundColor.color, + label: __( 'Background' ), + onColorChange: setBackgroundColor, + resetAllFilter: () => setBackgroundColor(), + }, + { + colorValue: overlayTextColor.color, + label: __( 'Submenu & overlay text' ), + onColorChange: setOverlayTextColor, + resetAllFilter: () => setOverlayTextColor(), + }, + { + colorValue: overlayBackgroundColor.color, + label: __( 'Submenu & overlay background' ), + onColorChange: setOverlayBackgroundColor, + resetAllFilter: () => + setOverlayBackgroundColor(), + }, + ] } + panelId={ clientId } + { ...colorGradientSettings } + gradients={ [] } + disableCustomGradients={ true } + /> + { enableContrastChecking && ( + <> + + + + ) } + + ) } + + ); // If the block has inner blocks, but no menu id, then these blocks are either: @@ -663,78 +710,25 @@ function Navigation( { // that automatically saves the menu as an entity when changes are made to the inner blocks. const hasUnsavedBlocks = hasUncontrolledInnerBlocks && ! isEntityAvailable; - const WrappedNavigationMenuSelector = ( { currentMenuId } ) => ( - { - handleUpdateMenu( menuId ); - } } - onSelectClassicMenu={ async ( classicMenu ) => { - const navMenu = await convertClassicMenu( - classicMenu.id, - classicMenu.name, - 'draft' - ); - if ( navMenu ) { - handleUpdateMenu( navMenu.id, { - focusNavigationBlock: true, - } ); - } - } } - onCreateNew={ createUntitledEmptyNavigationMenu } - createNavigationMenuIsSuccess={ createNavigationMenuIsSuccess } - createNavigationMenuIsError={ createNavigationMenuIsError } - /* translators: %s: The name of a menu. */ - actionLabel={ __( "Switch to '%s'" ) } - /> - ); - const isManageMenusButtonDisabled = ! hasManagePermissions || ! hasResolvedNavigationMenus; if ( hasUnsavedBlocks && ! isCreatingNavigationMenu ) { return ( - - - { isOffCanvasNavigationEditorEnabled ? ( - <> - - - { __( 'Menu' ) } - - - - - - ) : ( - <> - - - - ) } - - + { stylingInspectorControls } - - - { isOffCanvasNavigationEditorEnabled ? ( - <> - - - { __( 'Menu' ) } - - - -

    Select or create a menu

    - - ) : ( - <> - - - - ) } -
    -
    + { __( 'Navigation menu has been deleted or is unavailable. ' @@ -849,21 +820,8 @@ function Navigation( { isResolvingCanUserCreateNavigationMenu={ isResolvingCanUserCreateNavigationMenu } - onSelectNavigationMenu={ ( menuId ) => { - handleUpdateMenu( menuId ); - } } - onSelectClassicMenu={ async ( classicMenu ) => { - const navMenu = await convertClassicMenu( - classicMenu.id, - classicMenu.name, - 'draft' - ); - if ( navMenu ) { - handleUpdateMenu( navMenu.id, { - focusNavigationBlock: true, - } ); - } - } } + onSelectNavigationMenu={ onSelectNavigationMenu } + onSelectClassicMenu={ onSelectClassicMenu } onCreateEmpty={ createUntitledEmptyNavigationMenu } />
    @@ -873,45 +831,19 @@ function Navigation( { return ( - - - { isOffCanvasNavigationEditorEnabled ? ( - <> - - - { __( 'Menu' ) } - - - - - - ) : ( - <> - - - - ) } - - + { stylingInspectorControls } { isEntityAvailable && ( diff --git a/packages/block-library/src/navigation/edit/menu-inspector-controls.js b/packages/block-library/src/navigation/edit/menu-inspector-controls.js new file mode 100644 index 0000000000000..cefc94a3dcce0 --- /dev/null +++ b/packages/block-library/src/navigation/edit/menu-inspector-controls.js @@ -0,0 +1,78 @@ +/** + * WordPress dependencies + */ +import { + __experimentalOffCanvasEditor as OffCanvasEditor, + InspectorControls, +} from '@wordpress/block-editor'; +import { PanelBody, VisuallyHidden } from '@wordpress/components'; +import { __ } from '@wordpress/i18n'; + +/** + * Internal dependencies + */ +import ManageMenusButton from './manage-menus-button'; +import NavigationMenuSelector from './navigation-menu-selector'; + +const MenuInspectorControls = ( { + createNavigationMenuIsSuccess, + createNavigationMenuIsError, + currentMenuId = null, + innerBlocks, + isManageMenusButtonDisabled, + onCreateNew, + onSelectClassicMenu, + onSelectNavigationMenu, +} ) => { + const isOffCanvasNavigationEditorEnabled = + window?.__experimentalEnableOffCanvasNavigationEditor === true; + const menuControlsSlot = window?.__experimentalEnableBlockInspectorTabs + ? 'list' + : undefined; + /* translators: %s: The name of a menu. */ + const actionLabel = __( "Switch to '%s'" ); + + return ( + + + <> + { isOffCanvasNavigationEditorEnabled && ( + + { __( 'Menu' ) } + + ) } + + { isOffCanvasNavigationEditorEnabled ? ( + + ) : ( + + ) } + + + + ); +}; + +export default MenuInspectorControls; diff --git a/packages/block-library/src/navigation/edit/navigation-menu-selector.js b/packages/block-library/src/navigation/edit/navigation-menu-selector.js index 9817934c3a2b1..2d8594093bf10 100644 --- a/packages/block-library/src/navigation/edit/navigation-menu-selector.js +++ b/packages/block-library/src/navigation/edit/navigation-menu-selector.js @@ -10,7 +10,7 @@ import { VisuallyHidden, } from '@wordpress/components'; import { useEntityProp } from '@wordpress/core-data'; -import { Icon, chevronUp, chevronDown, moreVertical } from '@wordpress/icons'; +import { Icon, chevronUp, chevronDown } from '@wordpress/icons'; import { __, sprintf } from '@wordpress/i18n'; import { decodeEntities } from '@wordpress/html-entities'; import { useEffect, useMemo, useState } from '@wordpress/element'; @@ -31,9 +31,6 @@ function NavigationMenuSelector( { createNavigationMenuIsError, toggleProps = {}, } ) { - const isOffCanvasNavigationEditorEnabled = - window?.__experimentalEnableOffCanvasNavigationEditor === true; - /* translators: %s: The name of a menu. */ const createActionLabel = __( "Create from '%s'" ); @@ -164,23 +161,15 @@ function NavigationMenuSelector( { return ( - { isOffCanvasNavigationEditorEnabled ? '' : selectorLabel } + { selectorLabel } } - icon={ isOffCanvasNavigationEditorEnabled ? moreVertical : null } - toggleProps={ - isOffCanvasNavigationEditorEnabled - ? { isSmall: true } - : toggleProps - } + icon={ null } + toggleProps={ toggleProps } > { ( { onClose } ) => ( <> diff --git a/packages/block-library/src/navigation/edit/test/are-blocks-dirty.js b/packages/block-library/src/navigation/edit/test/are-blocks-dirty.js new file mode 100644 index 0000000000000..dc2b9a3f31f29 --- /dev/null +++ b/packages/block-library/src/navigation/edit/test/are-blocks-dirty.js @@ -0,0 +1,121 @@ +/** + * Internal dependencies + */ +import { areBlocksDirty } from '../are-blocks-dirty'; + +describe( 'areBlocksDirty', () => { + it( 'should be clean if the blocks are the same', () => { + expect( + areBlocksDirty( + [ { name: 'core/paragraph', content: 'I am not dirty.' } ], + [ { name: 'core/paragraph', content: 'I am not dirty.' } ] + ) + ).toBe( false ); + } ); + + it( `should be dirty if the blocks' attributes are different`, () => { + expect( + areBlocksDirty( + [ { name: 'core/paragraph', content: 'I am not dirty.' } ], + [ { name: 'core/paragraph', content: 'I am actually dirty.' } ] + ) + ).toBe( true ); + } ); + + it( `should be dirty if the blocks' attributes don't match`, () => { + expect( + areBlocksDirty( + [ { name: 'core/paragraph' }, { dropCap: false } ], + [ + { name: 'core/paragraph' }, + { content: 'I am actually dirty.' }, + ] + ) + ).toBe( true ); + } ); + + it( `should be dirty if the blocks' inner blocks are dirty`, () => { + expect( + areBlocksDirty( + [ + { + name: 'core/social-links', + innerBlocks: [ + { + name: 'core/social-link', + url: 'www.wordpress.org', + }, + ], + }, + ], + [ + { + name: 'core/social-links', + innerBlocks: [ + { + name: 'core/social-link', + service: 'wordpress', + url: 'www.wordpress.org', + }, + { + name: 'core/social-link', + service: 'wordpress', + url: 'make.wordpress.org', + }, + ], + }, + ] + ) + ).toBe( true ); + } ); + + describe( 'Controlled Page List block specific exceptions', () => { + it( 'should be clean if only page list inner blocks have changed', () => { + expect( + areBlocksDirty( + [ + { name: 'core/paragraph' }, + { + name: 'core/page-list', + innerBlocks: [], + }, + ], + [ + { name: 'core/paragraph' }, + { + name: 'core/page-list', + innerBlocks: [ { name: 'core/page-list-item' } ], + }, + ] + ) + ).toBe( false ); + } ); + + it( 'should be dirty if other blocks have changed alongside page list inner blocks', () => { + expect( + areBlocksDirty( + [ + { + name: 'core/paragraph', + content: 'This is some text', + }, + { + name: 'core/page-list', + innerBlocks: [], + }, + ], + [ + { + name: 'core/paragraph', + content: 'This is some text that has changed', + }, + { + name: 'core/page-list', + innerBlocks: [ { name: 'core/page-list-item' } ], + }, + ] + ) + ).toBe( true ); + } ); + } ); +} ); diff --git a/packages/block-library/src/navigation/edit/unsaved-inner-blocks.js b/packages/block-library/src/navigation/edit/unsaved-inner-blocks.js index 0fa2a119446f3..1017b6594fb4a 100644 --- a/packages/block-library/src/navigation/edit/unsaved-inner-blocks.js +++ b/packages/block-library/src/navigation/edit/unsaved-inner-blocks.js @@ -11,6 +11,7 @@ import { useContext, useEffect, useRef, useMemo } from '@wordpress/element'; * Internal dependencies */ import useNavigationMenu from '../use-navigation-menu'; +import { areBlocksDirty } from './are-blocks-dirty'; const EMPTY_OBJECT = {}; const DRAFT_MENU_PARAMS = [ @@ -51,13 +52,17 @@ export default function UnsavedInnerBlocks( { } }, [ blocks ] ); - // If the current inner blocks object is different in any way - // from the original inner blocks from the post content then the - // user has made changes to the inner blocks. At this point the inner - // blocks can be considered "dirty". - // We also make sure the current innerBlocks had a chance to be set. - const innerBlocksAreDirty = - !! originalBlocks.current && blocks !== originalBlocks.current; + // If the current inner blocks are different from the original inner blocks + // from the post content then the user has made changes to the inner blocks. + // At this point the inner blocks can be considered "dirty". + // Note: referential equality is not sufficient for comparison as the inner blocks + // of the page list are controlled and may be updated async due to syncing with + // entity records. As a result we need to perform a deep equality check skipping + // the page list's inner blocks. + const innerBlocksAreDirty = areBlocksDirty( + originalBlocks?.current, + blocks + ); const shouldDirectInsert = useMemo( () => diff --git a/packages/block-library/src/navigation/index.php b/packages/block-library/src/navigation/index.php index 87ea817270a39..030cab4501eed 100644 --- a/packages/block-library/src/navigation/index.php +++ b/packages/block-library/src/navigation/index.php @@ -257,10 +257,32 @@ function block_core_navigation_get_classic_menu_fallback() { $classic_nav_menus = wp_get_nav_menus(); // If menus exist. - if ( $classic_nav_menus && ! is_wp_error( $classic_nav_menus ) && count( $classic_nav_menus ) === 1 ) { - // Use the first classic menu only. Handles simple use case where user has a single - // classic menu and switches to a block theme. In future this maybe expanded to - // determine the most appropriate classic menu to be used based on location. + if ( $classic_nav_menus && ! is_wp_error( $classic_nav_menus ) ) { + // Handles simple use case where user has a classic menu and switches to a block theme. + + // Returns the menu assigned to location `primary`. + $locations = get_nav_menu_locations(); + if ( isset( $locations['primary'] ) ) { + $primary_menu = wp_get_nav_menu_object( $locations['primary'] ); + if ( $primary_menu ) { + return $primary_menu; + } + } + + // Returns a menu if `primary` is its slug. + foreach ( $classic_nav_menus as $classic_nav_menu ) { + if ( 'primary' === $classic_nav_menu->slug ) { + return $classic_nav_menu; + } + } + + // Otherwise return the most recently created classic menu. + usort( + $classic_nav_menus, + function( $a, $b ) { + return $b->term_id - $a->term_id; + } + ); return $classic_nav_menus[0]; } } @@ -489,6 +511,7 @@ function block_core_navigation_from_block_get_post_ids( $block ) { function render_block_core_navigation( $attributes, $content, $block ) { static $seen_menu_names = array(); + static $seen_ref = array(); // Flag used to indicate whether the rendered output is considered to be // a fallback (i.e. the block has no menu associated with it). @@ -559,6 +582,11 @@ function render_block_core_navigation( $attributes, $content, $block ) { // Load inner blocks from the navigation post. if ( array_key_exists( 'ref', $attributes ) ) { + if ( in_array( $attributes['ref'], $seen_ref, true ) ) { + return ''; + } + $seen_ref[] = $attributes['ref']; + $navigation_post = get_post( $attributes['ref'] ); if ( ! isset( $navigation_post ) ) { return ''; @@ -651,22 +679,41 @@ function render_block_core_navigation( $attributes, $content, $block ) { _prime_post_caches( $post_ids, false, false ); } + $list_item_nav_blocks = array( + 'core/navigation-link', + 'core/home-link', + 'core/site-title', + 'core/site-logo', + 'core/navigation-submenu', + ); + + $needs_list_item_wrapper = array( + 'core/site-title', + 'core/site-logo', + ); + $inner_blocks_html = ''; $is_list_open = false; foreach ( $inner_blocks as $inner_block ) { - if ( ( 'core/navigation-link' === $inner_block->name || 'core/home-link' === $inner_block->name || 'core/site-title' === $inner_block->name || 'core/site-logo' === $inner_block->name || 'core/navigation-submenu' === $inner_block->name ) && ! $is_list_open ) { + $is_list_item = in_array( $inner_block->name, $list_item_nav_blocks, true ); + + if ( $is_list_item && ! $is_list_open ) { $is_list_open = true; $inner_blocks_html .= '
      '; } - if ( 'core/navigation-link' !== $inner_block->name && 'core/home-link' !== $inner_block->name && 'core/site-title' !== $inner_block->name && 'core/site-logo' !== $inner_block->name && 'core/navigation-submenu' !== $inner_block->name && $is_list_open ) { + + if ( ! $is_list_item && $is_list_open ) { $is_list_open = false; $inner_blocks_html .= '
    '; } + $inner_block_content = $inner_block->render(); - if ( 'core/site-title' === $inner_block->name || ( 'core/site-logo' === $inner_block->name && $inner_block_content ) ) { - $inner_blocks_html .= '
  • ' . $inner_block_content . '
  • '; - } else { - $inner_blocks_html .= $inner_block_content; + if ( ! empty( $inner_block_content ) ) { + if ( in_array( $inner_block->name, $needs_list_item_wrapper, true ) ) { + $inner_blocks_html .= '
  • ' . $inner_block_content . '
  • '; + } else { + $inner_blocks_html .= $inner_block_content; + } } } diff --git a/packages/block-library/src/navigation/view-modal.js b/packages/block-library/src/navigation/view-modal.js index 477f05b5d5e51..9477d262816d9 100644 --- a/packages/block-library/src/navigation/view-modal.js +++ b/packages/block-library/src/navigation/view-modal.js @@ -27,6 +27,16 @@ function navigationToggleModal( modal ) { htmlElement.classList.toggle( 'has-modal-open' ); } +function isLinkToAnchorOnCurrentPage( node ) { + return ( + node.hash && + node.protocol === window.location.protocol && + node.host === window.location.host && + node.pathname === window.location.pathname && + node.search === window.location.search + ); +} + window.addEventListener( 'load', () => { MicroModal.init( { onShow: navigationToggleModal, @@ -42,7 +52,7 @@ window.addEventListener( 'load', () => { navigationLinks.forEach( function ( link ) { // Ignore non-anchor links and anchor links which open on a new tab. if ( - ! link.getAttribute( 'href' )?.startsWith( '#' ) || + ! isLinkToAnchorOnCurrentPage( link ) || link.attributes?.target === '_blank' ) { return; diff --git a/packages/block-library/src/page-list-item/block.json b/packages/block-library/src/page-list-item/block.json new file mode 100644 index 0000000000000..97c312d1b7960 --- /dev/null +++ b/packages/block-library/src/page-list-item/block.json @@ -0,0 +1,51 @@ +{ + "$schema": "https://schemas.wp.org/trunk/block.json", + "apiVersion": 2, + "name": "core/page-list-item", + "title": "Page List Item", + "category": "widgets", + "parent": [ "core/page-list" ], + "description": "Displays a page inside a list of all pages.", + "keywords": [ "page", "menu", "navigation" ], + "textdomain": "default", + "attributes": { + "id": { + "type": "number" + }, + "label": { + "type": "string" + }, + "title": { + "type": "string" + }, + "link": { + "type": "string" + }, + "hasChildren": { + "type": "boolean" + } + }, + "usesContext": [ + "textColor", + "customTextColor", + "backgroundColor", + "customBackgroundColor", + "overlayTextColor", + "customOverlayTextColor", + "overlayBackgroundColor", + "customOverlayBackgroundColor", + "fontSize", + "customFontSize", + "showSubmenuIcon", + "style", + "openSubmenusOnClick" + ], + "supports": { + "reusable": false, + "html": false, + "lock": false, + "inserter": false + }, + "editorStyle": "wp-block-page-list-editor", + "style": "wp-block-page-list" +} diff --git a/packages/block-library/src/page-list-item/edit.js b/packages/block-library/src/page-list-item/edit.js new file mode 100644 index 0000000000000..db8e0a5c17373 --- /dev/null +++ b/packages/block-library/src/page-list-item/edit.js @@ -0,0 +1,94 @@ +/** + * External dependencies + */ +import classnames from 'classnames'; + +/** + * WordPress dependencies + */ +import { InnerBlocks } from '@wordpress/block-editor'; +import { useSelect } from '@wordpress/data'; +import { store as coreStore } from '@wordpress/core-data'; + +/** + * Internal dependencies + */ +import { ItemSubmenuIcon } from '../navigation-link/icons'; + +function useFrontPageId() { + return useSelect( ( select ) => { + const canReadSettings = select( coreStore ).canUser( + 'read', + 'settings' + ); + if ( ! canReadSettings ) { + return undefined; + } + + const site = select( coreStore ).getEntityRecord( 'root', 'site' ); + return site?.show_on_front === 'page' && site?.page_on_front; + }, [] ); +} + +export default function PageListItemEdit( { context, attributes } ) { + const { id, label, link, hasChildren } = attributes; + const isNavigationChild = 'showSubmenuIcon' in context; + const frontPageId = useFrontPageId(); + return ( +
  • + { hasChildren && context.openSubmenusOnClick ? ( + <> + + + + + + ) : ( + + { label } + + ) } + { hasChildren && ( + <> + { ! context.openSubmenusOnClick && + context.showSubmenuIcon && ( + + ) } +
      + +
    + + ) } +
  • + ); +} diff --git a/packages/block-library/src/page-list-item/index.js b/packages/block-library/src/page-list-item/index.js new file mode 100644 index 0000000000000..18b7fe2e40e9e --- /dev/null +++ b/packages/block-library/src/page-list-item/index.js @@ -0,0 +1,24 @@ +/** + * WordPress dependencies + */ +import { pages as icon } from '@wordpress/icons'; + +/** + * Internal dependencies + */ +import initBlock from '../utils/init-block'; +import metadata from './block.json'; +import edit from './edit.js'; + +const { name } = metadata; + +export { metadata, name }; + +export const settings = { + __experimentalLabel: ( { label } ) => label, + icon, + example: {}, + edit, +}; + +export const init = () => initBlock( { name, metadata, settings } ); diff --git a/packages/block-library/src/page-list-item/init.js b/packages/block-library/src/page-list-item/init.js new file mode 100644 index 0000000000000..79f0492c2cb2f --- /dev/null +++ b/packages/block-library/src/page-list-item/init.js @@ -0,0 +1,6 @@ +/** + * Internal dependencies + */ +import { init } from './'; + +export default init(); diff --git a/packages/block-library/src/page-list/block.json b/packages/block-library/src/page-list/block.json index 3068a1fb8bc00..2fc6993849d6f 100644 --- a/packages/block-library/src/page-list/block.json +++ b/packages/block-library/src/page-list/block.json @@ -7,7 +7,12 @@ "description": "Display a list of all pages.", "keywords": [ "menu", "navigation" ], "textdomain": "default", - "attributes": {}, + "attributes": { + "parentPageID": { + "type": "integer", + "default": 0 + } + }, "usesContext": [ "textColor", "customTextColor", diff --git a/packages/block-library/src/page-list/constants.js b/packages/block-library/src/page-list/constants.js new file mode 100644 index 0000000000000..9322fa345321c --- /dev/null +++ b/packages/block-library/src/page-list/constants.js @@ -0,0 +1,8 @@ +/** + * WordPress dependencies + */ +import { __ } from '@wordpress/i18n'; + +export const convertDescription = __( + 'This menu is automatically kept in sync with pages on your site. You can manage the menu yourself by clicking customize below.' +); diff --git a/packages/block-library/src/page-list/convert-to-links-modal.js b/packages/block-library/src/page-list/convert-to-links-modal.js index bbb3e1e44e98b..dbc76fc77cdb1 100644 --- a/packages/block-library/src/page-list/convert-to-links-modal.js +++ b/packages/block-library/src/page-list/convert-to-links-modal.js @@ -5,71 +5,20 @@ import { Button, Modal } from '@wordpress/components'; import { __ } from '@wordpress/i18n'; import { useDispatch } from '@wordpress/data'; import { useEntityRecords } from '@wordpress/core-data'; -import { createBlock as create } from '@wordpress/blocks'; import { store as blockEditorStore } from '@wordpress/block-editor'; +/** + * Internal dependencies + */ +import { convertToNavigationLinks } from './convert-to-navigation-links'; + +/** + * Internal dependencies + */ +import { convertDescription } from './constants'; const PAGE_FIELDS = [ 'id', 'title', 'link', 'type', 'parent' ]; const MAX_PAGE_COUNT = 100; -export const convertSelectedBlockToNavigationLinks = - ( { pages, clientId, replaceBlock, createBlock } ) => - () => { - if ( ! pages ) { - return; - } - - const linkMap = {}; - const navigationLinks = []; - pages.forEach( ( { id, title, link: url, type, parent } ) => { - // See if a placeholder exists. This is created if children appear before parents in list. - const innerBlocks = linkMap[ id ]?.innerBlocks ?? []; - linkMap[ id ] = createBlock( - 'core/navigation-link', - { - id, - label: title.rendered, - url, - type, - kind: 'post-type', - }, - innerBlocks - ); - - if ( ! parent ) { - navigationLinks.push( linkMap[ id ] ); - } else { - if ( ! linkMap[ parent ] ) { - // Use a placeholder if the child appears before parent in list. - linkMap[ parent ] = { innerBlocks: [] }; - } - const parentLinkInnerBlocks = linkMap[ parent ].innerBlocks; - parentLinkInnerBlocks.push( linkMap[ id ] ); - } - } ); - - // Transform all links with innerBlocks into Submenus. This can't be done - // sooner because page objects have no information on their children. - - const transformSubmenus = ( listOfLinks ) => { - listOfLinks.forEach( ( block, index, listOfLinksArray ) => { - const { attributes, innerBlocks } = block; - if ( innerBlocks.length !== 0 ) { - transformSubmenus( innerBlocks ); - const transformedBlock = createBlock( - 'core/navigation-submenu', - attributes, - innerBlocks - ); - listOfLinksArray[ index ] = transformedBlock; - } - } ); - }; - - transformSubmenus( navigationLinks ); - - replaceBlock( clientId, navigationLinks ); - }; - export default function ConvertToLinksModal( { onClose, clientId } ) { const { records: pages, hasResolved: pagesFinished } = useEntityRecords( 'postType', @@ -91,19 +40,12 @@ export default function ConvertToLinksModal( { onClose, clientId } ) {

    - { __( - 'To edit this navigation menu, convert it to single page links. This allows you to add, re-order, remove items, or edit their labels.' - ) } -

    -

    - { __( - "Note: if you add new pages to your site, you'll need to add them to your navigation menu." - ) } + { convertDescription }

    diff --git a/packages/block-library/src/page-list/convert-to-navigation-links.js b/packages/block-library/src/page-list/convert-to-navigation-links.js new file mode 100644 index 0000000000000..64b1dbf1c7d4d --- /dev/null +++ b/packages/block-library/src/page-list/convert-to-navigation-links.js @@ -0,0 +1,60 @@ +/** + * WordPress dependencies + */ +import { createBlock } from '@wordpress/blocks'; + +export const convertToNavigationLinks = ( pages ) => { + if ( ! pages ) { + return; + } + + const linkMap = {}; + const navigationLinks = []; + pages.forEach( ( { id, title, link: url, type, parent } ) => { + // See if a placeholder exists. This is created if children appear before parents in list. + const innerBlocks = linkMap[ id ]?.innerBlocks ?? []; + linkMap[ id ] = createBlock( + 'core/navigation-link', + { + id, + label: title.rendered, + url, + type, + kind: 'post-type', + }, + innerBlocks + ); + + if ( ! parent ) { + navigationLinks.push( linkMap[ id ] ); + } else { + if ( ! linkMap[ parent ] ) { + // Use a placeholder if the child appears before parent in list. + linkMap[ parent ] = { innerBlocks: [] }; + } + const parentLinkInnerBlocks = linkMap[ parent ].innerBlocks; + parentLinkInnerBlocks.push( linkMap[ id ] ); + } + } ); + + // Transform all links with innerBlocks into Submenus. This can't be done + // sooner because page objects have no information on their children. + const transformSubmenus = ( listOfLinks ) => { + listOfLinks.forEach( ( block, index, listOfLinksArray ) => { + const { attributes, innerBlocks } = block; + if ( innerBlocks.length !== 0 ) { + transformSubmenus( innerBlocks ); + const transformedBlock = createBlock( + 'core/navigation-submenu', + attributes, + innerBlocks + ); + listOfLinksArray[ index ] = transformedBlock; + } + } ); + }; + + transformSubmenus( navigationLinks ); + + return navigationLinks; +}; diff --git a/packages/block-library/src/page-list/edit.js b/packages/block-library/src/page-list/edit.js index 71cbfa437a1f9..b3d2db31d5e04 100644 --- a/packages/block-library/src/page-list/edit.js +++ b/packages/block-library/src/page-list/edit.js @@ -6,29 +6,51 @@ import classnames from 'classnames'; /** * WordPress dependencies */ +import { createBlock } from '@wordpress/blocks'; import { + InspectorControls, BlockControls, useBlockProps, + useInnerBlocksProps, getColorClassName, + store as blockEditorStore, + Warning, } from '@wordpress/block-editor'; -import { ToolbarButton, Spinner, Notice } from '@wordpress/components'; -import { __ } from '@wordpress/i18n'; -import { useMemo, useState, memo } from '@wordpress/element'; -import { useSelect } from '@wordpress/data'; -import { store as coreStore, useEntityRecords } from '@wordpress/core-data'; +import { + PanelBody, + ToolbarButton, + Spinner, + Notice, + ComboboxControl, + Button, +} from '@wordpress/components'; +import { __, sprintf } from '@wordpress/i18n'; +import { useMemo, useState, useEffect } from '@wordpress/element'; +import { useEntityRecords } from '@wordpress/core-data'; +import { useSelect, useDispatch } from '@wordpress/data'; /** * Internal dependencies */ import ConvertToLinksModal from './convert-to-links-modal'; -import { ItemSubmenuIcon } from '../navigation-link/icons'; +import { convertToNavigationLinks } from './convert-to-navigation-links'; +import { convertDescription } from './constants'; // We only show the edit option when page count is <= MAX_PAGE_COUNT // Performance of Navigation Links is not good past this value. const MAX_PAGE_COUNT = 100; -export default function PageListEdit( { context, clientId } ) { +export default function PageListEdit( { + context, + clientId, + attributes, + setAttributes, +} ) { + const { parentPageID } = attributes; + const [ pages ] = useGetPages(); const { pagesByParentId, totalPages, hasResolvedPages } = usePageData(); + const { replaceInnerBlocks, __unstableMarkNextChangeAsNotPersistent } = + useDispatch( blockEditorStore ); const isNavigationChild = 'showSubmenuIcon' in context; const allowConvertToLinks = @@ -52,6 +74,67 @@ export default function PageListEdit( { context, clientId } ) { style: { ...context.style?.color }, } ); + const getBlockList = ( parentId = parentPageID ) => { + const childPages = pagesByParentId.get( parentId ); + + if ( ! childPages?.length ) { + return []; + } + + return childPages.reduce( ( template, page ) => { + const hasChildren = pagesByParentId.has( page.id ); + const pageProps = { + id: page.id, + label: page.title?.rendered, + title: page.title?.rendered, + link: page.url, + hasChildren, + }; + let item = null; + const children = getBlockList( page.id ); + item = createBlock( 'core/page-list-item', pageProps, children ); + template.push( item ); + + return template; + }, [] ); + }; + + const makePagesTree = ( parentId = 0, level = 0 ) => { + const childPages = pagesByParentId.get( parentId ); + + if ( ! childPages?.length ) { + return []; + } + + return childPages.reduce( ( tree, page ) => { + const hasChildren = pagesByParentId.has( page.id ); + const item = { + value: page.id, + label: '— '.repeat( level ) + page.title.rendered, + rawName: page.title.rendered, + }; + tree.push( item ); + if ( hasChildren ) { + tree.push( ...makePagesTree( page.id, level + 1 ) ); + } + return tree; + }, [] ); + }; + + const pagesTree = useMemo( makePagesTree, [ pagesByParentId ] ); + + const blockList = useMemo( getBlockList, [ + pagesByParentId, + parentPageID, + ] ); + + const innerBlocksProps = useInnerBlocksProps( blockProps, { + allowedBlocks: [ 'core/page-list-item' ], + renderAppender: false, + __unstableDisableDropZone: true, + templateLock: 'all', + } ); + const getBlockContent = () => { if ( ! hasResolvedPages ) { return ( @@ -81,21 +164,93 @@ export default function PageListEdit( { context, clientId } ) { ); } - if ( totalPages > 0 ) { + if ( blockList.length === 0 ) { + const parentPageDetails = + pages && pages.find( ( page ) => page.id === parentPageID ); return ( -
      - -
    +
    + + { sprintf( + // translators: %s: Page title. + __( '"%s" page has no children.' ), + parentPageDetails.title.rendered + ) } + +
    ); } + + if ( totalPages > 0 ) { + return
      ; + } }; + useEffect( () => { + __unstableMarkNextChangeAsNotPersistent(); + if ( blockList ) { + replaceInnerBlocks( clientId, blockList ); + } + }, [ clientId, blockList ] ); + + const { replaceBlock, selectBlock } = useDispatch( blockEditorStore ); + + const { parentNavBlockClientId } = useSelect( ( select ) => { + const { getSelectedBlockClientId, getBlockParentsByBlockName } = + select( blockEditorStore ); + + const _selectedBlockClientId = getSelectedBlockClientId(); + + return { + parentNavBlockClientId: getBlockParentsByBlockName( + _selectedBlockClientId, + 'core/navigation', + true + )[ 0 ], + }; + }, [] ); + return ( <> - { allowConvertToLinks && ( + + { isNavigationChild && pages?.length > 0 && ( + +

      { convertDescription }

      + +
      + ) } + { pagesTree.length > 0 && ( + + + setAttributes( { parentPageID: value ?? 0 } ) + } + help={ __( + 'Choose a page to show only its subpages.' + ) } + /> + + ) } +
      + { allowConvertToLinks && totalPages > 0 && ( { __( 'Edit' ) } @@ -114,22 +269,7 @@ export default function PageListEdit( { context, clientId } ) { ); } -function useFrontPageId() { - return useSelect( ( select ) => { - const canReadSettings = select( coreStore ).canUser( - 'read', - 'settings' - ); - if ( ! canReadSettings ) { - return undefined; - } - - const site = select( coreStore ).getEntityRecord( 'root', 'site' ); - return site?.show_on_front === 'page' && site?.page_on_front; - }, [] ); -} - -function usePageData() { +function useGetPages() { const { records: pages, hasResolved: hasResolvedPages } = useEntityRecords( 'postType', 'page', @@ -142,10 +282,17 @@ function usePageData() { } ); + return [ pages, hasResolvedPages ]; +} + +function usePageData( pageId = 0 ) { + const [ pages, hasResolvedPages ] = useGetPages(); + return useMemo( () => { // TODO: Once the REST API supports passing multiple values to // 'orderby', this can be removed. // https://core.trac.wordpress.org/ticket/39037 + const sortedPages = [ ...( pages ?? [] ) ].sort( ( a, b ) => { if ( a.menu_order === b.menu_order ) { return a.title.rendered.localeCompare( b.title.rendered ); @@ -167,91 +314,5 @@ function usePageData() { hasResolvedPages, totalPages: pages?.length ?? null, }; - }, [ pages, hasResolvedPages ] ); + }, [ pageId, pages, hasResolvedPages ] ); } - -const PageItems = memo( function PageItems( { - context, - pagesByParentId, - parentId = 0, - depth = 0, -} ) { - const pages = pagesByParentId.get( parentId ); - const frontPageId = useFrontPageId(); - - if ( ! pages?.length ) { - return []; - } - - return pages.map( ( page ) => { - const hasChildren = pagesByParentId.has( page.id ); - const isNavigationChild = 'showSubmenuIcon' in context; - return ( -
    • - { hasChildren && context.openSubmenusOnClick ? ( - <> - - - - - - ) : ( - - { page.title?.rendered } - - ) } - { hasChildren && ( - <> - { ! context.openSubmenusOnClick && - context.showSubmenuIcon && ( - - ) } -
        - -
      - - ) } -
    • - ); - } ); -} ); diff --git a/packages/block-library/src/page-list/index.php b/packages/block-library/src/page-list/index.php index 32acdad45ed63..7944f6f3a84ad 100644 --- a/packages/block-library/src/page-list/index.php +++ b/packages/block-library/src/page-list/index.php @@ -252,6 +252,8 @@ function render_block_core_page_list( $attributes, $content, $block ) { static $block_id = 0; ++$block_id; + $parent_page_id = $attributes['parentPageID']; + $all_pages = get_pages( array( 'sort_column' => 'menu_order,post_title', @@ -306,6 +308,13 @@ function render_block_core_page_list( $attributes, $content, $block ) { $nested_pages = block_core_page_list_nest_pages( $top_level_pages, $pages_with_children ); + if ( 0 !== $parent_page_id ) { + $nested_pages = block_core_page_list_nest_pages( + $pages_with_children[ $parent_page_id ], + $pages_with_children + ); + } + $is_navigation_child = array_key_exists( 'showSubmenuIcon', $block->context ); $open_submenus_on_click = array_key_exists( 'openSubmenusOnClick', $block->context ) ? $block->context['openSubmenusOnClick'] : false; diff --git a/packages/block-library/src/page-list/test/convert-to-links-modal.js b/packages/block-library/src/page-list/test/convert-to-links-modal.js index e9908d2b01436..4c83b64f9bb9f 100644 --- a/packages/block-library/src/page-list/test/convert-to-links-modal.js +++ b/packages/block-library/src/page-list/test/convert-to-links-modal.js @@ -1,10 +1,28 @@ /** * Internal dependencies */ -import { convertSelectedBlockToNavigationLinks } from '../convert-to-links-modal'; + +import { convertToNavigationLinks } from '../convert-to-navigation-links'; + +// Mock createBlock to avoid creating the blocks in test environment +// as convertToNavigationLinks calls this method internally. +jest.mock( '@wordpress/blocks', () => { + const blocks = jest.requireActual( '@wordpress/blocks' ); + + return { + ...blocks, + createBlock( name, attributes, innerBlocks ) { + return { + name, + attributes, + innerBlocks, + }; + }, + }; +} ); describe( 'page list convert to links', () => { - describe( 'convertSelectedBlockToNavigationLinks', () => { + describe( 'convertToNavigationLinks', () => { it( 'Can create submenus', () => { const pages = [ { @@ -88,22 +106,10 @@ describe( 'page list convert to links', () => { type: 'page', }, ]; - const replaceBlock = jest.fn(); - const createBlock = jest.fn( - ( name, attributes, innerBlocks ) => ( { - name, - attributes, - innerBlocks, - } ) - ); - const convertLinks = convertSelectedBlockToNavigationLinks( { - pages, - clientId: 'testId', - replaceBlock, - createBlock, - } ); - convertLinks(); - expect( replaceBlock.mock.calls?.[ 0 ]?.[ 1 ] ).toEqual( [ + + const convertLinks = convertToNavigationLinks( pages ); + + expect( convertLinks ).toEqual( [ { attributes: { id: 2, @@ -280,22 +286,10 @@ describe( 'page list convert to links', () => { type: 'page', }, ]; - const replaceBlock = jest.fn(); - const createBlock = jest.fn( - ( name, attributes, innerBlocks ) => ( { - name, - attributes, - innerBlocks, - } ) - ); - const convertLinks = convertSelectedBlockToNavigationLinks( { - pages, - clientId: 'testId', - replaceBlock, - createBlock, - } ); - convertLinks(); - expect( replaceBlock.mock.calls?.[ 0 ]?.[ 1 ] ).toEqual( [ + + const convertLinks = convertToNavigationLinks( pages ); + + expect( convertLinks ).toEqual( [ { attributes: { id: 2, diff --git a/packages/block-library/src/paragraph/index.js b/packages/block-library/src/paragraph/index.js index 70ae87ccf5f36..c9a00ccf4b33f 100644 --- a/packages/block-library/src/paragraph/index.js +++ b/packages/block-library/src/paragraph/index.js @@ -30,12 +30,6 @@ export const settings = { content: __( 'In a village of La Mancha, the name of which I have no desire to call to mind, there lived not long since one of those gentlemen that keep a lance in the lance-rack, an old buckler, a lean hack, and a greyhound for coursing.' ), - style: { - typography: { - fontSize: 28, - }, - }, - dropCap: true, }, }, __experimentalLabel( attributes, { context } ) { diff --git a/packages/block-library/src/post-author/edit.js b/packages/block-library/src/post-author/edit.js index 1fc13ea42add9..b0a2fb2883f79 100644 --- a/packages/block-library/src/post-author/edit.js +++ b/packages/block-library/src/post-author/edit.js @@ -62,7 +62,7 @@ function PostAuthorEdit( { attributes; const avatarSizes = []; const authorName = authorDetails?.name || __( 'Post Author' ); - if ( authorDetails ) { + if ( authorDetails?.avatar_urls ) { Object.keys( authorDetails.avatar_urls ).forEach( ( size ) => { avatarSizes.push( { value: size, @@ -103,6 +103,7 @@ function PostAuthorEdit( { authorOptions.length && ( ( showCombobox && (
      - { showAvatar && authorDetails && ( + { showAvatar && authorDetails?.avatar_urls && (
      context['postId']; + // Check is needed for backward compatibility with third-party plugins + // that might rely on the `in_the_loop` check; calling `the_post` sets it to true. + if ( ! in_the_loop() && have_posts() ) { + the_post(); + } + $is_link = isset( $attributes['isLink'] ) && $attributes['isLink']; $size_slug = isset( $attributes['sizeSlug'] ) ? $attributes['sizeSlug'] : 'post-thumbnail'; - $post_title = trim( strip_tags( get_the_title( $post_ID ) ) ); $attr = get_block_core_post_featured_image_border_attributes( $attributes ); $overlay_markup = get_block_core_post_featured_image_overlay_element_markup( $attributes ); if ( $is_link ) { - $attr['alt'] = $post_title; + $attr['alt'] = trim( strip_tags( get_the_title( $post_ID ) ) ); } if ( ! empty( $attributes['height'] ) ) { diff --git a/packages/block-library/src/post-featured-image/overlay.js b/packages/block-library/src/post-featured-image/overlay.js index f577978b9a3c9..fc885741b4d05 100644 --- a/packages/block-library/src/post-featured-image/overlay.js +++ b/packages/block-library/src/post-featured-image/overlay.js @@ -98,6 +98,7 @@ const Overlay = ( { panelId={ clientId } > diff --git a/packages/block-library/src/post-template/block.json b/packages/block-library/src/post-template/block.json index 380b6d55f71fa..bc9910b47d1dc 100644 --- a/packages/block-library/src/post-template/block.json +++ b/packages/block-library/src/post-template/block.json @@ -22,6 +22,14 @@ "__experimentalLayout": { "allowEditing": false }, + "color": { + "gradients": true, + "link": true, + "__experimentalDefaultControls": { + "background": true, + "text": true + } + }, "typography": { "fontSize": true, "lineHeight": true, diff --git a/packages/block-library/src/preformatted/transforms.js b/packages/block-library/src/preformatted/transforms.js index ef5f332447409..068a08b8bcfc7 100644 --- a/packages/block-library/src/preformatted/transforms.js +++ b/packages/block-library/src/preformatted/transforms.js @@ -34,7 +34,10 @@ const transforms = { type: 'block', blocks: [ 'core/paragraph' ], transform: ( attributes ) => - createBlock( 'core/paragraph', attributes ), + createBlock( 'core/paragraph', { + ...attributes, + content: attributes.content.replace( /\n/g, '
      ' ), + } ), }, { type: 'block', diff --git a/packages/block-library/src/query/block.json b/packages/block-library/src/query/block.json index cd09e22ee57f7..1974761962ec9 100644 --- a/packages/block-library/src/query/block.json +++ b/packages/block-library/src/query/block.json @@ -50,14 +50,6 @@ "supports": { "align": [ "wide", "full" ], "html": false, - "color": { - "gradients": true, - "link": true, - "__experimentalDefaultControls": { - "background": true, - "text": true - } - }, "__experimentalLayout": true }, "editorStyle": "wp-block-query-editor" diff --git a/packages/block-library/src/query/deprecated.js b/packages/block-library/src/query/deprecated.js index 42ac3dc4f1c49..b23455d50489b 100644 --- a/packages/block-library/src/query/deprecated.js +++ b/packages/block-library/src/query/deprecated.js @@ -1,12 +1,18 @@ /** * WordPress dependencies */ +import { createBlock } from '@wordpress/blocks'; import { InnerBlocks, useInnerBlocksProps, useBlockProps, } from '@wordpress/block-editor'; +/** + * Internal dependencies + */ +import cleanEmptyObject from '../utils/clean-empty-object'; + const migrateToTaxQuery = ( attributes ) => { const { query } = attributes; const { categoryIds, tagIds, ...newQuery } = query; @@ -25,106 +31,271 @@ const migrateToTaxQuery = ( attributes ) => { }; }; -const deprecated = [ - // Version with `categoryIds and tagIds`. - { - attributes: { - queryId: { - type: 'number', - }, - query: { - type: 'object', - default: { - perPage: null, - pages: 0, - offset: 0, - postType: 'post', - categoryIds: [], - tagIds: [], - order: 'desc', - orderBy: 'date', - author: '', - search: '', - exclude: [], - sticky: '', - inherit: true, - }, - }, - tagName: { - type: 'string', - default: 'div', +const migrateColors = ( attributes, innerBlocks ) => { + // Remove color style attributes from the Query block. + const { style, backgroundColor, gradient, textColor, ...newAttributes } = + attributes; + + const hasColorStyles = + backgroundColor || + gradient || + textColor || + style?.color || + style?.elements?.link; + + // If the query block doesn't currently have any color styles, + // nothing needs migrating. + if ( ! hasColorStyles ) { + return [ attributes, innerBlocks ]; + } + + // Clean color values from style attribute object. + if ( style ) { + newAttributes.style = cleanEmptyObject( { + ...style, + color: undefined, + elements: { + ...style.elements, + link: undefined, }, - displayLayout: { - type: 'object', - default: { - type: 'list', - }, + } ); + } + + // If the inner blocks are already wrapped in a single group + // block, add the color support styles to that group block. + if ( hasSingleInnerGroupBlock( innerBlocks ) ) { + const groupBlock = innerBlocks[ 0 ]; + + // Create new styles for the group block. + const hasStyles = + style?.color || + style?.elements?.link || + groupBlock.attributes.style; + + const newStyles = hasStyles + ? cleanEmptyObject( { + ...groupBlock.attributes.style, + color: style?.color, + elements: style?.elements?.link + ? { link: style?.elements?.link } + : undefined, + } ) + : undefined; + + // Create a new Group block from the original. + const updatedGroupBlock = createBlock( + 'core/group', + { + ...groupBlock.attributes, + backgroundColor, + gradient, + textColor, + style: newStyles, }, + groupBlock.innerBlocks + ); + + return [ newAttributes, [ updatedGroupBlock ] ]; + } + + // When we don't have a single wrapping group block for the inner + // blocks, wrap the current inner blocks in a group applying the + // color styles to that. + const newGroupBlock = createBlock( + 'core/group', + { + backgroundColor, + gradient, + textColor, + style: cleanEmptyObject( { + color: style?.color, + elements: style?.elements?.link + ? { link: style?.elements?.link } + : undefined, + } ), + }, + innerBlocks + ); + + return [ newAttributes, [ newGroupBlock ] ]; +}; + +const hasSingleInnerGroupBlock = ( innerBlocks = [] ) => + innerBlocks.length === 1 && innerBlocks[ 0 ].name === 'core/group'; + +// Version with NO wrapper `div` element. +const v1 = { + attributes: { + queryId: { + type: 'number', }, - supports: { - align: [ 'wide', 'full' ], - html: false, - color: { - gradients: true, - link: true, + query: { + type: 'object', + default: { + perPage: null, + pages: 0, + offset: 0, + postType: 'post', + categoryIds: [], + tagIds: [], + order: 'desc', + orderBy: 'date', + author: '', + search: '', + exclude: [], + sticky: '', + inherit: true, }, - __experimentalLayout: true, }, - isEligible: ( { query: { categoryIds, tagIds } = {} } ) => - categoryIds || tagIds, - migrate: migrateToTaxQuery, - save( { attributes: { tagName: Tag = 'div' } } ) { - const blockProps = useBlockProps.save(); - const innerBlocksProps = useInnerBlocksProps.save( blockProps ); - return ; + layout: { + type: 'object', + default: { + type: 'list', + }, }, }, - // Version with NO wrapper `div` element. - { - attributes: { - queryId: { - type: 'number', + supports: { + html: false, + }, + migrate( attributes ) { + const withTaxQuery = migrateToTaxQuery( attributes ); + const { layout, ...restWithTaxQuery } = withTaxQuery; + return { + ...restWithTaxQuery, + displayLayout: withTaxQuery.layout, + }; + }, + save() { + return ; + }, +}; + +// Version with `categoryIds and tagIds`. +const v2 = { + attributes: { + queryId: { + type: 'number', + }, + query: { + type: 'object', + default: { + perPage: null, + pages: 0, + offset: 0, + postType: 'post', + categoryIds: [], + tagIds: [], + order: 'desc', + orderBy: 'date', + author: '', + search: '', + exclude: [], + sticky: '', + inherit: true, }, - query: { - type: 'object', - default: { - perPage: null, - pages: 0, - offset: 0, - postType: 'post', - categoryIds: [], - tagIds: [], - order: 'desc', - orderBy: 'date', - author: '', - search: '', - exclude: [], - sticky: '', - inherit: true, - }, + }, + tagName: { + type: 'string', + default: 'div', + }, + displayLayout: { + type: 'object', + default: { + type: 'list', }, - layout: { - type: 'object', - default: { - type: 'list', - }, + }, + }, + supports: { + align: [ 'wide', 'full' ], + html: false, + color: { + gradients: true, + link: true, + }, + __experimentalLayout: true, + }, + isEligible: ( { query: { categoryIds, tagIds } = {} } ) => + categoryIds || tagIds, + migrate( attributes, innerBlocks ) { + const withTaxQuery = migrateToTaxQuery( attributes ); + return migrateColors( withTaxQuery, innerBlocks ); + }, + save( { attributes: { tagName: Tag = 'div' } } ) { + const blockProps = useBlockProps.save(); + const innerBlocksProps = useInnerBlocksProps.save( blockProps ); + return ; + }, +}; + +// Version with color support prior to moving it to the PostTemplate block. +const v3 = { + attributes: { + queryId: { + type: 'number', + }, + query: { + type: 'object', + default: { + perPage: null, + pages: 0, + offset: 0, + postType: 'post', + order: 'desc', + orderBy: 'date', + author: '', + search: '', + exclude: [], + sticky: '', + inherit: true, + taxQuery: null, + parents: [], }, }, - supports: { - html: false, + tagName: { + type: 'string', + default: 'div', + }, + displayLayout: { + type: 'object', + default: { + type: 'list', + }, }, - migrate( attributes ) { - const withTaxQuery = migrateToTaxQuery( attributes ); - const { layout, ...restWithTaxQuery } = withTaxQuery; - return { - ...restWithTaxQuery, - displayLayout: withTaxQuery.layout, - }; + namespace: { + type: 'string', }, - save() { - return ; + }, + supports: { + align: [ 'wide', 'full' ], + html: false, + color: { + gradients: true, + link: true, + __experimentalDefaultControls: { + background: true, + text: true, + }, }, + __experimentalLayout: true, + }, + isEligible( attributes ) { + const { style, backgroundColor, gradient, textColor } = attributes; + return ( + backgroundColor || + gradient || + textColor || + style?.color || + style?.elements?.link + ); }, -]; + migrate: migrateColors, + save( { attributes: { tagName: Tag = 'div' } } ) { + const blockProps = useBlockProps.save(); + const innerBlocksProps = useInnerBlocksProps.save( blockProps ); + return ; + }, +}; + +const deprecated = [ v3, v2, v1 ]; export default deprecated; diff --git a/packages/block-library/src/query/edit/inspector-controls/index.js b/packages/block-library/src/query/edit/inspector-controls/index.js index ffe19060682d3..13cc70e9dc542 100644 --- a/packages/block-library/src/query/edit/inspector-controls/index.js +++ b/packages/block-library/src/query/edit/inspector-controls/index.js @@ -141,6 +141,7 @@ export default function QueryInspectorControls( { { showColumnsControl && ( <> diff --git a/packages/block-library/src/query/edit/query-placeholder.js b/packages/block-library/src/query/edit/query-placeholder.js index 8e1debe5341a6..fa1ef818e2f1f 100644 --- a/packages/block-library/src/query/edit/query-placeholder.js +++ b/packages/block-library/src/query/edit/query-placeholder.js @@ -104,20 +104,8 @@ function QueryVariationPicker( { icon, label, } ) { - const { defaultVariation, scopeVariations } = useSelect( - ( select ) => { - const { - getBlockVariations, - getBlockType, - getDefaultBlockVariation, - } = select( blocksStore ); - - return { - blockType: getBlockType( name ), - defaultVariation: getDefaultBlockVariation( name, 'block' ), - scopeVariations: getBlockVariations( name, 'block' ), - }; - }, + const variations = useSelect( + ( select ) => select( blocksStore ).getBlockVariations( name, 'block' ), [ name ] ); const { replaceInnerBlocks } = useDispatch( blockEditorStore ); @@ -127,8 +115,8 @@ function QueryVariationPicker( { <__experimentalBlockVariationPicker icon={ icon } label={ label } - variations={ scopeVariations } - onSelect={ ( nextVariation = defaultVariation ) => { + variations={ variations } + onSelect={ ( nextVariation ) => { if ( nextVariation.attributes ) { setAttributes( { ...nextVariation.attributes, diff --git a/packages/block-library/src/rss/edit.js b/packages/block-library/src/rss/edit.js index f72bf1c77e674..f7689b8c38ca1 100644 --- a/packages/block-library/src/rss/edit.js +++ b/packages/block-library/src/rss/edit.js @@ -111,6 +111,7 @@ export default function RSSEdit( { attributes, setAttributes } ) { @@ -137,6 +138,7 @@ export default function RSSEdit( { attributes, setAttributes } ) { /> { displayExcerpt && ( @@ -149,6 +151,7 @@ export default function RSSEdit( { attributes, setAttributes } ) { ) } { blockLayout === 'grid' && ( diff --git a/packages/block-library/src/search/edit.native.js b/packages/block-library/src/search/edit.native.js index 8b315bb21f4a9..c5f5fb4cf2354 100644 --- a/packages/block-library/src/search/edit.native.js +++ b/packages/block-library/src/search/edit.native.js @@ -42,6 +42,36 @@ const BUTTON_OPTIONS = [ { value: 'no-button', label: __( 'No button' ) }, ]; +function useIsScreenReaderEnabled() { + const [ isScreenReaderEnabled, setIsScreenReaderEnabled ] = + useState( false ); + + useEffect( () => { + let mounted = true; + + const changeListener = AccessibilityInfo.addEventListener( + 'screenReaderChanged', + ( enabled ) => setIsScreenReaderEnabled( enabled ) + ); + + AccessibilityInfo.isScreenReaderEnabled().then( + ( screenReaderEnabled ) => { + if ( mounted && screenReaderEnabled ) { + setIsScreenReaderEnabled( screenReaderEnabled ); + } + } + ); + + return () => { + mounted = false; + + changeListener.remove(); + }; + }, [] ); + + return isScreenReaderEnabled; +} + export default function SearchEdit( { onFocus, isSelected, @@ -57,8 +87,7 @@ export default function SearchEdit( { useState( false ); const [ isLongButton, setIsLongButton ] = useState( false ); const [ buttonWidth, setButtonWidth ] = useState( MIN_BUTTON_WIDTH ); - const [ isScreenReaderEnabled, setIsScreenReaderEnabled ] = - useState( false ); + const isScreenReaderEnabled = useIsScreenReaderEnabled(); const textInputRef = useRef( null ); @@ -71,31 +100,6 @@ export default function SearchEdit( { buttonText, } = attributes; - /* - * Check if screenreader is enabled and save to state. This is important for - * properly creating accessibilityLabel text. - */ - useEffect( () => { - const a11yInfoChangeSubscription = AccessibilityInfo.addEventListener( - 'screenReaderChanged', - handleScreenReaderToggled - ); - - AccessibilityInfo.isScreenReaderEnabled().then( - ( screenReaderEnabled ) => { - setIsScreenReaderEnabled( screenReaderEnabled ); - } - ); - - return () => { - a11yInfoChangeSubscription.remove(); - }; - }, [] ); - - const handleScreenReaderToggled = ( screenReaderEnabled ) => { - setIsScreenReaderEnabled( screenReaderEnabled ); - }; - /* * Called when the value of isSelected changes. Blurs the PlainText component * used by the placeholder when this block loses focus. diff --git a/packages/block-library/src/search/index.js b/packages/block-library/src/search/index.js index 441336316415b..85770a23268cb 100644 --- a/packages/block-library/src/search/index.js +++ b/packages/block-library/src/search/index.js @@ -1,6 +1,7 @@ /** * WordPress dependencies */ +import { __ } from '@wordpress/i18n'; import { search as icon } from '@wordpress/icons'; /** @@ -17,7 +18,10 @@ export { metadata, name }; export const settings = { icon, - example: {}, + example: { + attributes: { buttonText: __( 'Search' ), label: __( 'Search' ) }, + viewportWidth: 400, + }, variations, edit, }; diff --git a/packages/block-library/src/shortcode/test/edit.native.js b/packages/block-library/src/shortcode/test/edit.native.js index 46c7dbddef2c9..372956a2242a0 100644 --- a/packages/block-library/src/shortcode/test/edit.native.js +++ b/packages/block-library/src/shortcode/test/edit.native.js @@ -1,12 +1,7 @@ /** * External dependencies */ -import { - getEditorHtml, - initializeEditor, - fireEvent, - waitFor, -} from 'test/helpers'; +import { getEditorHtml, initializeEditor, fireEvent } from 'test/helpers'; /** * WordPress dependencies @@ -28,12 +23,11 @@ afterAll( () => { describe( 'Shortcode block', () => { it( 'inserts block', async () => { - const { getByLabelText, getByTestId, getByText } = - await initializeEditor(); + const screen = await initializeEditor(); - fireEvent.press( getByLabelText( 'Add block' ) ); + fireEvent.press( screen.getByLabelText( 'Add block' ) ); - const blockList = getByTestId( 'InserterUI-Blocks' ); + const blockList = screen.getByTestId( 'InserterUI-Blocks' ); // onScroll event used to force the FlatList to render all items fireEvent.scroll( blockList, { nativeEvent: { @@ -43,22 +37,25 @@ describe( 'Shortcode block', () => { }, } ); - fireEvent.press( await waitFor( () => getByText( 'Shortcode' ) ) ); + fireEvent.press( await screen.findByText( 'Shortcode' ) ); - expect( getByLabelText( /Shortcode Block\. Row 1/ ) ).toBeVisible(); + const [ shortcodeBlock ] = screen.getAllByLabelText( + /Shortcode Block\. Row 1/ + ); + expect( shortcodeBlock ).toBeVisible(); expect( getEditorHtml() ).toMatchSnapshot(); } ); it( 'edits content', async () => { - const { getByLabelText, getByPlaceholderText } = await initializeEditor( - { - initialHtml: '', - } + const screen = await initializeEditor( { + initialHtml: '', + } ); + const [ shortcodeBlock ] = screen.getAllByLabelText( + /Shortcode Block\. Row 1/ ); - const shortcodeBlock = getByLabelText( /Shortcode Block\. Row 1/ ); fireEvent.press( shortcodeBlock ); - const textField = getByPlaceholderText( 'Add a shortcode…' ); + const textField = screen.getByPlaceholderText( 'Add a shortcode…' ); fireEvent( textField, 'focus' ); fireEvent( textField, 'onChange', { nativeEvent: { diff --git a/packages/block-library/src/site-logo/edit.js b/packages/block-library/src/site-logo/edit.js index 89b9e3f6eba01..d84b5e6fbd683 100644 --- a/packages/block-library/src/site-logo/edit.js +++ b/packages/block-library/src/site-logo/edit.js @@ -2,7 +2,6 @@ * External dependencies */ import classnames from 'classnames'; -import { pick } from 'lodash'; /** * WordPress dependencies @@ -87,7 +86,11 @@ const SiteLogo = ( { ); return { title: siteEntities?.name, - ...pick( getSettings(), [ 'imageEditing', 'maxWidth' ] ), + ...Object.fromEntries( + Object.entries( getSettings() ).filter( ( [ key ] ) => + [ 'imageEditing', 'maxWidth' ].includes( key ) + ) + ), }; }, [] ); @@ -120,9 +123,10 @@ const SiteLogo = ( { src={ logoUrl } alt={ alt } onLoad={ ( event ) => { - setNaturalSize( - pick( event.target, [ 'naturalWidth', 'naturalHeight' ] ) - ); + setNaturalSize( { + naturalWidth: event.target.naturalWidth, + naturalHeight: event.target.naturalHeight, + } ); } } /> ); @@ -291,6 +295,7 @@ const SiteLogo = ( { setAttributes( { width: newWidth } ) @@ -512,6 +517,9 @@ export default function LogoEdit( { className={ placeholderClassName } preview={ logoImage } withIllustration={ true } + style={ { + width, + } } > { content } diff --git a/packages/block-library/src/site-logo/index.js b/packages/block-library/src/site-logo/index.js index 68d8eb6f8b930..fc10df08e17de 100644 --- a/packages/block-library/src/site-logo/index.js +++ b/packages/block-library/src/site-logo/index.js @@ -16,6 +16,7 @@ export { metadata, name }; export const settings = { icon, + example: {}, edit, transforms, }; diff --git a/packages/block-library/src/site-title/index.js b/packages/block-library/src/site-title/index.js index 01fe15598d6f1..87934888ce438 100644 --- a/packages/block-library/src/site-title/index.js +++ b/packages/block-library/src/site-title/index.js @@ -17,6 +17,7 @@ export { metadata, name }; export const settings = { icon, + example: {}, edit, transforms, deprecated, diff --git a/packages/block-library/src/social-link/social-list.js b/packages/block-library/src/social-link/social-list.js index 9ac527dccd0d3..38c1ef91f9938 100644 --- a/packages/block-library/src/social-link/social-list.js +++ b/packages/block-library/src/social-link/social-list.js @@ -1,8 +1,3 @@ -/** - * External dependencies - */ -import { find } from 'lodash'; - /** * WordPress dependencies */ @@ -22,7 +17,7 @@ import { ChainIcon } from './icons'; * @return {WPComponent} Icon component for social service. */ export const getIconBySite = ( name ) => { - const variation = find( variations, { name } ); + const variation = variations.find( ( v ) => v.name === name ); return variation ? variation.icon : ChainIcon; }; @@ -34,6 +29,6 @@ export const getIconBySite = ( name ) => { * @return {string} Display name for social service */ export const getNameBySite = ( name ) => { - const variation = find( variations, { name } ); + const variation = variations.find( ( v ) => v.name === name ); return variation ? variation.title : __( 'Social Icon' ); }; diff --git a/packages/block-library/src/social-link/test/index.native.js b/packages/block-library/src/social-link/test/index.native.js index 603776e939998..4ad7611e72045 100644 --- a/packages/block-library/src/social-link/test/index.native.js +++ b/packages/block-library/src/social-link/test/index.native.js @@ -42,16 +42,13 @@ describe( '', () => { 'social icons' ); fireEvent.press( - await waitFor( () => - subject.getByLabelText( 'Social Icons block' ) - ) + await subject.findByLabelText( 'Social Icons block' ) + ); + const [ socialIconsBlock ] = subject.getAllByLabelText( + /Social Icons Block. Row 1/ ); fireEvent( - await waitFor( () => - within( - subject.getByLabelText( /Social Icons Block. Row 1/ ) - ).getByTestId( 'block-list-wrapper' ) - ), + within( socialIconsBlock ).getByTestId( 'block-list-wrapper' ), 'layout', { nativeEvent: { layout: { width: 100 } } } ); @@ -99,16 +96,13 @@ describe( '', () => { 'social icons' ); fireEvent.press( - await waitFor( () => - subject.getByLabelText( 'Social Icons block' ) - ) + await subject.findByLabelText( 'Social Icons block' ) + ); + const [ socialIconsBlock ] = subject.getAllByLabelText( + /Social Icons Block. Row 1/ ); fireEvent( - await waitFor( () => - within( - subject.getByLabelText( /Social Icons Block. Row 1/ ) - ).getByTestId( 'block-list-wrapper' ) - ), + within( socialIconsBlock ).getByTestId( 'block-list-wrapper' ), 'layout', { nativeEvent: { layout: { width: 100 } } } ); diff --git a/packages/block-library/src/social-links/test/edit.native.js b/packages/block-library/src/social-links/test/edit.native.js index ade94321278e9..48fefaadaec8c 100644 --- a/packages/block-library/src/social-links/test/edit.native.js +++ b/packages/block-library/src/social-links/test/edit.native.js @@ -104,7 +104,6 @@ describe( 'Social links block', () => { it( 'shows the social links bottom sheet when tapping on the inline appender', async () => { const screen = await initializeEditor(); - const { getByTestId, getByText } = screen; // Add block await addBlock( screen, 'Social Icons' ); @@ -131,7 +130,7 @@ describe( 'Social links block', () => { fireEvent.press( appenderButton ); // Find a social link in the inserter - const blockList = getByTestId( 'InserterUI-Blocks' ); + const blockList = screen.getByTestId( 'InserterUI-Blocks' ); // onScroll event used to force the FlatList to render all items fireEvent.scroll( blockList, { @@ -143,11 +142,13 @@ describe( 'Social links block', () => { } ); // Add the Amazon link - const amazonBlock = await waitFor( () => getByText( 'Amazon' ) ); + const amazonBlock = await screen.findByText( 'Amazon' ); expect( amazonBlock ).toBeVisible(); fireEvent.press( amazonBlock ); + await screen.findByTestId( 'navigation-screen-LinkSettingsScreen' ); + expect( getEditorHtml() ).toMatchSnapshot(); } ); diff --git a/packages/block-library/src/spacer/test/index.native.js b/packages/block-library/src/spacer/test/index.native.js index 7b6ed2ba3f371..5d5229033d19a 100644 --- a/packages/block-library/src/spacer/test/index.native.js +++ b/packages/block-library/src/spacer/test/index.native.js @@ -28,12 +28,11 @@ afterAll( () => { describe( 'Spacer block', () => { it( 'inserts block', async () => { - const { getByLabelText, getByTestId, getByText } = - await initializeEditor(); + const screen = await initializeEditor(); - fireEvent.press( getByLabelText( 'Add block' ) ); + fireEvent.press( screen.getByLabelText( 'Add block' ) ); - const blockList = getByTestId( 'InserterUI-Blocks' ); + const blockList = screen.getByTestId( 'InserterUI-Blocks' ); // onScroll event used to force the FlatList to render all items fireEvent.scroll( blockList, { nativeEvent: { @@ -43,9 +42,11 @@ describe( 'Spacer block', () => { }, } ); - fireEvent.press( await waitFor( () => getByText( 'Spacer' ) ) ); + fireEvent.press( screen.getByText( 'Spacer' ) ); - expect( getByLabelText( /Spacer Block\. Row 1/ ) ).toBeVisible(); + const [ spacerBlock ] = + screen.getAllByLabelText( /Spacer Block\. Row 1/ ); + expect( spacerBlock ).toBeVisible(); expect( getEditorHtml() ).toMatchSnapshot(); } ); @@ -53,24 +54,24 @@ describe( 'Spacer block', () => { const initialHtml = ` `; - const { getByLabelText, getByDisplayValue, getByTestId, getByText } = - await initializeEditor( { - initialHtml, - } ); + const screen = await initializeEditor( { + initialHtml, + } ); // Select Spacer block - const spacerBlock = getByLabelText( /Spacer Block\. Row 1/ ); + const [ spacerBlock ] = + screen.getAllByLabelText( /Spacer Block\. Row 1/ ); fireEvent.press( spacerBlock ); // Open block settings - fireEvent.press( getByLabelText( 'Open Settings' ) ); + fireEvent.press( screen.getByLabelText( 'Open Settings' ) ); await waitFor( - () => getByTestId( 'block-settings-modal' ).props.isVisible + () => screen.getByTestId( 'block-settings-modal' ).props.isVisible ); // Update height attribute - fireEvent.press( getByText( '100' ) ); - const heightTextInput = getByDisplayValue( '100' ); + fireEvent.press( screen.getByText( '100' ) ); + const heightTextInput = screen.getByDisplayValue( '100' ); fireEvent.changeText( heightTextInput, '50' ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -80,28 +81,28 @@ describe( 'Spacer block', () => { const initialHtml = ` `; - const { getByLabelText, getByDisplayValue, getByTestId, getByText } = - await initializeEditor( { - initialHtml, - } ); + const screen = await initializeEditor( { + initialHtml, + } ); // Select Spacer block - const spacerBlock = getByLabelText( /Spacer Block\. Row 1/ ); + const [ spacerBlock ] = + screen.getAllByLabelText( /Spacer Block\. Row 1/ ); fireEvent.press( spacerBlock ); // Open block settings - fireEvent.press( getByLabelText( 'Open Settings' ) ); + fireEvent.press( screen.getByLabelText( 'Open Settings' ) ); await waitFor( - () => getByTestId( 'block-settings-modal' ).props.isVisible + () => screen.getByTestId( 'block-settings-modal' ).props.isVisible ); // Set vh unit - fireEvent.press( getByText( 'px' ) ); - fireEvent.press( getByText( 'Viewport height (vh)' ) ); + fireEvent.press( screen.getByText( 'px' ) ); + fireEvent.press( screen.getByText( 'Viewport height (vh)' ) ); // Update height attribute - fireEvent.press( getByText( '100' ) ); - const heightTextInput = getByDisplayValue( '100' ); + fireEvent.press( screen.getByText( '100' ) ); + const heightTextInput = screen.getByDisplayValue( '100' ); fireEvent.changeText( heightTextInput, '25' ); expect( getEditorHtml() ).toMatchSnapshot(); @@ -111,23 +112,24 @@ describe( 'Spacer block', () => { const initialHtml = ` `; - const { getByLabelText, getByTestId } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select Spacer block - const spacerBlock = getByLabelText( /Spacer Block\. Row 1/ ); + const [ spacerBlock ] = + screen.getAllByLabelText( /Spacer Block\. Row 1/ ); fireEvent.press( spacerBlock ); // Open block settings - fireEvent.press( getByLabelText( 'Open Settings' ) ); + fireEvent.press( screen.getByLabelText( 'Open Settings' ) ); await waitFor( - () => getByTestId( 'block-settings-modal' ).props.isVisible + () => screen.getByTestId( 'block-settings-modal' ).props.isVisible ); // Increment height fireEvent( - getByLabelText( /Height\. Value is 100 Pixels \(px\)/ ), + screen.getByLabelText( /Height\. Value is 100 Pixels \(px\)/ ), 'accessibilityAction', { nativeEvent: { actionName: 'increment' }, @@ -141,23 +143,24 @@ describe( 'Spacer block', () => { const initialHtml = ` `; - const { getByLabelText, getByTestId } = await initializeEditor( { + const screen = await initializeEditor( { initialHtml, } ); // Select Spacer block - const spacerBlock = getByLabelText( /Spacer Block\. Row 1/ ); + const [ spacerBlock ] = + screen.getAllByLabelText( /Spacer Block\. Row 1/ ); fireEvent.press( spacerBlock ); // Open block settings - fireEvent.press( getByLabelText( 'Open Settings' ) ); + fireEvent.press( screen.getByLabelText( 'Open Settings' ) ); await waitFor( - () => getByTestId( 'block-settings-modal' ).props.isVisible + () => screen.getByTestId( 'block-settings-modal' ).props.isVisible ); // Increment height fireEvent( - getByLabelText( /Height\. Value is 100 Pixels \(px\)/ ), + screen.getByLabelText( /Height\. Value is 100 Pixels \(px\)/ ), 'accessibilityAction', { nativeEvent: { actionName: 'decrement' }, diff --git a/packages/block-library/src/table-of-contents/edit.js b/packages/block-library/src/table-of-contents/edit.js index 92f2d39dd98cf..dab35fdd0ce3d 100644 --- a/packages/block-library/src/table-of-contents/edit.js +++ b/packages/block-library/src/table-of-contents/edit.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { isEqual } from 'lodash'; +import fastDeepEqual from 'fast-deep-equal/es6'; /** * WordPress dependencies @@ -204,7 +204,7 @@ export default function TableOfContentsEdit( { } } - if ( isEqual( headings, _latestHeadings ) ) { + if ( fastDeepEqual( headings, _latestHeadings ) ) { return null; } return _latestHeadings; diff --git a/packages/block-library/src/table/block.json b/packages/block-library/src/table/block.json index 3de088e0879b7..5e21c1f07f8ba 100644 --- a/packages/block-library/src/table/block.json +++ b/packages/block-library/src/table/block.json @@ -47,6 +47,11 @@ "type": "string", "source": "attribute", "attribute": "data-align" + }, + "colspan": { + "type": "string", + "source": "attribute", + "attribute": "colspan" } } } @@ -82,6 +87,11 @@ "type": "string", "source": "attribute", "attribute": "data-align" + }, + "colspan": { + "type": "string", + "source": "attribute", + "attribute": "colspan" } } } @@ -117,6 +127,11 @@ "type": "string", "source": "attribute", "attribute": "data-align" + }, + "colspan": { + "type": "string", + "source": "attribute", + "attribute": "colspan" } } } diff --git a/packages/block-library/src/table/edit.js b/packages/block-library/src/table/edit.js index abb6438cb5678..2ca9cb6c42a98 100644 --- a/packages/block-library/src/table/edit.js +++ b/packages/block-library/src/table/edit.js @@ -404,7 +404,7 @@ function TableEdit( { { cells.map( ( - { content, tag: CellTag, scope, align }, + { content, tag: CellTag, scope, align, colspan }, columnIndex ) => ( { diff --git a/packages/block-library/src/table/editor.scss b/packages/block-library/src/table/editor.scss index f1fbae335748c..b019f871c06f6 100644 --- a/packages/block-library/src/table/editor.scss +++ b/packages/block-library/src/table/editor.scss @@ -41,10 +41,6 @@ border-style: double; } - figcaption { - @include caption-style-theme(); - } - // This is only required in the editor to overcome the fact the editor // rewrites border width styles into shorthand. table.has-individual-borders { diff --git a/packages/block-library/src/table/index.js b/packages/block-library/src/table/index.js index 5839abc4c19f2..dea09dd7c9829 100644 --- a/packages/block-library/src/table/index.js +++ b/packages/block-library/src/table/index.js @@ -91,6 +91,7 @@ export const settings = { }, ], }, + viewportWidth: 450, }, transforms, edit, diff --git a/packages/block-library/src/table/save.js b/packages/block-library/src/table/save.js index 4e0c4f1956cc1..d3393267b754d 100644 --- a/packages/block-library/src/table/save.js +++ b/packages/block-library/src/table/save.js @@ -43,7 +43,10 @@ export default function save( { attributes } ) { { rows.map( ( { cells }, rowIndex ) => ( { cells.map( - ( { content, tag, scope, align }, cellIndex ) => { + ( + { content, tag, scope, align, colspan }, + cellIndex + ) => { const cellClasses = classnames( { [ `has-text-align-${ align }` ]: align, } ); @@ -62,6 +65,7 @@ export default function save( { attributes } ) { scope={ tag === 'th' ? scope : undefined } + colSpan={ colspan } /> ); } diff --git a/packages/block-library/src/table/state.js b/packages/block-library/src/table/state.js index 68fee2d0bda9b..a1aafc3145936 100644 --- a/packages/block-library/src/table/state.js +++ b/packages/block-library/src/table/state.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { get, mapValues, pick } from 'lodash'; +import { get, mapValues } from 'lodash'; const INHERITED_COLUMN_ATTRIBUTES = [ 'align' ]; @@ -78,7 +78,11 @@ export function updateSelectedCell( state, selection, updateCell ) { return state; } - const tableSections = pick( state, [ 'head', 'body', 'foot' ] ); + const tableSections = Object.fromEntries( + Object.entries( state ).filter( ( [ key ] ) => + [ 'head', 'body', 'foot' ].includes( key ) + ) + ); const { sectionName: selectionSectionName, rowIndex: selectionRowIndex } = selection; @@ -174,9 +178,12 @@ export function insertRow( state, { sectionName, rowIndex, columnCount } ) { [ 'cells', index ], {} ); - const inheritedAttributes = pick( - firstCellInColumn, - INHERITED_COLUMN_ATTRIBUTES + + const inheritedAttributes = Object.fromEntries( + Object.entries( firstCellInColumn ).filter( + ( [ key ] ) => + INHERITED_COLUMN_ATTRIBUTES.includes( key ) + ) ); return { @@ -220,7 +227,11 @@ export function deleteRow( state, { sectionName, rowIndex } ) { * @return {Object} New table state. */ export function insertColumn( state, { columnIndex } ) { - const tableSections = pick( state, [ 'head', 'body', 'foot' ] ); + const tableSections = Object.fromEntries( + Object.entries( state ).filter( ( [ key ] ) => + [ 'head', 'body', 'foot' ].includes( key ) + ) + ); return mapValues( tableSections, ( section, sectionName ) => { // Bail early if the table section is empty. @@ -259,7 +270,11 @@ export function insertColumn( state, { columnIndex } ) { * @return {Object} New table state. */ export function deleteColumn( state, { columnIndex } ) { - const tableSections = pick( state, [ 'head', 'body', 'foot' ] ); + const tableSections = Object.fromEntries( + Object.entries( state ).filter( ( [ key ] ) => + [ 'head', 'body', 'foot' ].includes( key ) + ) + ); return mapValues( tableSections, ( section ) => { // Bail early if the table section is empty. diff --git a/packages/block-library/src/table/transforms.js b/packages/block-library/src/table/transforms.js index 15ee7dff1178f..0651c3bc64c41 100644 --- a/packages/block-library/src/table/transforms.js +++ b/packages/block-library/src/table/transforms.js @@ -5,11 +5,12 @@ const tableContentPasteSchema = ( { phrasingContentSchema } ) => ( { th: { allowEmpty: true, children: phrasingContentSchema, - attributes: [ 'scope' ], + attributes: [ 'scope', 'colspan' ], }, td: { allowEmpty: true, children: phrasingContentSchema, + attributes: [ 'colspan' ], }, }, }, diff --git a/packages/block-library/src/tag-cloud/edit.js b/packages/block-library/src/tag-cloud/edit.js index 8cb286630d8eb..dd669a7e840b3 100644 --- a/packages/block-library/src/tag-cloud/edit.js +++ b/packages/block-library/src/tag-cloud/edit.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { map, filter } from 'lodash'; +import { map } from 'lodash'; /** * WordPress dependencies @@ -69,7 +69,7 @@ function TagCloudEdit( { attributes, setAttributes, taxonomies } ) { disabled: true, }; const taxonomyOptions = map( - filter( taxonomies, 'show_cloud' ), + taxonomies?.filter( ( tax ) => !! tax.show_cloud ), ( item ) => { return { value: item.slug, @@ -124,6 +124,7 @@ function TagCloudEdit( { attributes, setAttributes, taxonomies } ) { } /> diff --git a/packages/block-library/src/template-part/edit/selection-modal.js b/packages/block-library/src/template-part/edit/selection-modal.js index f979df799c6be..9a83914ddb424 100644 --- a/packages/block-library/src/template-part/edit/selection-modal.js +++ b/packages/block-library/src/template-part/edit/selection-modal.js @@ -1,7 +1,7 @@ /** * WordPress dependencies */ -import { useCallback, useMemo, useState } from '@wordpress/element'; +import { useMemo, useState } from '@wordpress/element'; import { __, sprintf } from '@wordpress/i18n'; import { store as noticesStore } from '@wordpress/notices'; import { useDispatch } from '@wordpress/data'; @@ -57,7 +57,7 @@ export default function TemplatePartSelectionModal( { const { createSuccessNotice } = useDispatch( noticesStore ); - const onTemplatePartSelect = useCallback( ( templatePart ) => { + const onTemplatePartSelect = ( templatePart ) => { setAttributes( { slug: templatePart.slug, theme: templatePart.theme, @@ -74,7 +74,7 @@ export default function TemplatePartSelectionModal( { } ); onClose(); - }, [] ); + }; const createFromBlocks = useCreateTemplatePartFromBlocks( area, @@ -88,6 +88,7 @@ export default function TemplatePartSelectionModal( {
      definedArea.area === area + ); + const defaultArea = definedAreas.find( + ( definedArea ) => definedArea.area === 'uncategorized' + ); return { icon: selectedArea?.icon || defaultArea?.icon, diff --git a/packages/block-library/src/template-part/editor.scss b/packages/block-library/src/template-part/editor.scss index c17d37a6078e2..a847ecec01f94 100644 --- a/packages/block-library/src/template-part/editor.scss +++ b/packages/block-library/src/template-part/editor.scss @@ -5,8 +5,8 @@ // and height are used instead of max-(width/height). .components-modal__frame { @include break-small() { - width: calc(100% - #{ $grid-unit-20 * 2 }); - height: calc(100% - #{ $header-height * 2 }); + width: calc(100% - #{$grid-unit-20 * 2}); + height: calc(100% - #{$header-height * 2}); } @include break-medium() { width: $break-medium - $grid-unit-20 * 2; @@ -24,3 +24,21 @@ padding: $grid-unit-20 0; z-index: z-index(".block-library-template-part__selection-search"); } + +.is-outline-mode .block-editor-block-list__block:not(.remove-outline).wp-block-template-part, +.is-outline-mode .block-editor-block-list__block:not(.remove-outline).is-reusable { + &.is-highlighted, + &.is-selected { + box-shadow: inset 0 0 0 var(--wp-admin-border-width-focus) var(--wp-block-synced-color); + } + + &.block-editor-block-list__block:not([contenteditable]):focus { + &::after { + box-shadow: 0 0 0 var(--wp-admin-border-width-focus) var(--wp-block-synced-color); + // Show a light color for dark themes. + .is-dark-theme & { + box-shadow: 0 0 0 var(--wp-admin-border-width-focus) $dark-theme-focus; + } + } + } +} diff --git a/packages/block-library/src/template-part/index.php b/packages/block-library/src/template-part/index.php index bef49341d7bb5..f320362c32f1d 100644 --- a/packages/block-library/src/template-part/index.php +++ b/packages/block-library/src/template-part/index.php @@ -105,8 +105,7 @@ function render_block_core_template_part( $attributes ) { // WP_DEBUG_DISPLAY must only be honored when WP_DEBUG. This precedent // is set in `wp_debug_mode()`. - $is_debug = defined( 'WP_DEBUG' ) && WP_DEBUG && - defined( 'WP_DEBUG_DISPLAY' ) && WP_DEBUG_DISPLAY; + $is_debug = WP_DEBUG && WP_DEBUG_DISPLAY; if ( is_null( $content ) && $is_debug ) { if ( ! isset( $attributes['slug'] ) ) { diff --git a/packages/block-library/src/text-columns/edit.js b/packages/block-library/src/text-columns/edit.js index 180c786226c80..993e5da0c80e3 100644 --- a/packages/block-library/src/text-columns/edit.js +++ b/packages/block-library/src/text-columns/edit.js @@ -39,6 +39,7 @@ export default function TextColumnsEdit( { attributes, setAttributes } ) { diff --git a/packages/block-library/src/utils/transformation-categories.native.js b/packages/block-library/src/utils/transformation-categories.native.js index 3ddf60d2a4c22..e001f9b67c785 100644 --- a/packages/block-library/src/utils/transformation-categories.native.js +++ b/packages/block-library/src/utils/transformation-categories.native.js @@ -3,6 +3,7 @@ const transformationCategories = { 'core/paragraph', 'core/heading', 'core/list', + 'core/list-item', 'core/quote', 'core/pullquote', 'core/preformatted', diff --git a/packages/block-library/src/video/edit.js b/packages/block-library/src/video/edit.js index e79f8a9774482..f09f7dca49b51 100644 --- a/packages/block-library/src/video/edit.js +++ b/packages/block-library/src/video/edit.js @@ -161,7 +161,7 @@ function VideoEdit( { const embedBlock = createUpgradedEmbedBlock( { attributes: { url: newSrc }, } ); - if ( undefined !== embedBlock ) { + if ( undefined !== embedBlock && onReplace ) { onReplace( embedBlock ); return; } diff --git a/packages/block-library/src/video/edit.native.js b/packages/block-library/src/video/edit.native.js index fb4937ce5da73..99225ba5d5c56 100644 --- a/packages/block-library/src/video/edit.native.js +++ b/packages/block-library/src/video/edit.native.js @@ -2,7 +2,6 @@ * External dependencies */ import { View, TouchableWithoutFeedback, Text } from 'react-native'; -import { isEmpty } from 'lodash'; /** * WordPress dependencies @@ -367,7 +366,7 @@ class VideoEdit extends Component { - isEmpty( caption ) + ! caption ? /* translators: accessibility text. Empty video caption. */ __( 'Video caption. Empty' ) : sprintf( diff --git a/packages/block-library/test/babel-plugin.js b/packages/block-library/test/babel-plugin.js index bb79d12f60f22..7c26b0f7d117c 100644 --- a/packages/block-library/test/babel-plugin.js +++ b/packages/block-library/test/babel-plugin.js @@ -12,56 +12,56 @@ function join( ...strings ) { return strings.join( '\n' ); } -function compare( input, output, isExperimental, options = {} ) { +function transformCode( input, isExperimental, options = {} ) { const blockLibraryPlugin = createBabelPlugin( isExperimental, false ); const { code } = transform( input, { configFile: false, plugins: [ [ blockLibraryPlugin, options ] ], } ); - expect( code ).toEqual( output ); + return code; } describe( 'babel-plugin', () => { it( 'should ignore stable blocks', () => { - compare( - join( - 'import * as experimentalBlock from "./experimental-block";', - 'const blocks = [ experimentalBlock ];' - ), - join( - 'import * as experimentalBlock from "./experimental-block";', - 'const blocks = [experimentalBlock];' - ), - () => false + const input = join( + 'import * as experimentalBlock from "./experimental-block";', + 'const blocks = [ experimentalBlock ];' ); + const expected = join( + 'import * as experimentalBlock from "./experimental-block";', + 'const blocks = [experimentalBlock];' + ); + + expect( transformCode( input, () => false ) ).toEqual( expected ); } ); + it( 'should transform experimental blocks', () => { - compare( - join( - 'import * as experimentalBlock from "./experimental-block";', - 'const blocks = [ experimentalBlock ];' - ), - join( - 'const experimentalBlock = null;', - 'const blocks = [experimentalBlock];' - ), - () => true + const input = join( + 'import * as experimentalBlock from "./experimental-block";', + 'const blocks = [ experimentalBlock ];' + ); + const expected = join( + 'const experimentalBlock = null;', + 'const blocks = [experimentalBlock];' ); + + expect( transformCode( input, () => true ) ).toEqual( expected ); } ); + it( 'should work with mixed imports blocks', () => { - compare( - join( - 'import * as stableBlock from "./stable-block";', - 'import * as experimentalBlock from "./experimental-block";', - 'const blocks = [ stableBlock, experimentalBlock ];' - ), - join( - 'import * as stableBlock from "./stable-block";', - 'const experimentalBlock = null;', - 'const blocks = [stableBlock, experimentalBlock];' - ), - ( path ) => - path.node.specifiers[ 0 ].local.name === 'experimentalBlock' + const input = join( + 'import * as stableBlock from "./stable-block";', + 'import * as experimentalBlock from "./experimental-block";', + 'const blocks = [ stableBlock, experimentalBlock ];' ); + const expected = join( + 'import * as stableBlock from "./stable-block";', + 'const experimentalBlock = null;', + 'const blocks = [stableBlock, experimentalBlock];' + ); + const isExperimental = ( path ) => + path.node.specifiers[ 0 ].local.name === 'experimentalBlock'; + + expect( transformCode( input, isExperimental ) ).toEqual( expected ); } ); } ); diff --git a/packages/blocks/CHANGELOG.md b/packages/blocks/CHANGELOG.md index f1c2317d7791b..5d1f797d7063d 100644 --- a/packages/blocks/CHANGELOG.md +++ b/packages/blocks/CHANGELOG.md @@ -2,6 +2,10 @@ ## Unreleased +### Breaking Changes + +- Updated dependencies to require React 18 ([45235](https://github.com/WordPress/gutenberg/pull/45235)) + ## 11.21.0 (2022-11-16) ## 11.20.0 (2022-11-02) diff --git a/packages/blocks/README.md b/packages/blocks/README.md index f78783f728339..a03fddf71fe1c 100644 --- a/packages/blocks/README.md +++ b/packages/blocks/README.md @@ -529,10 +529,9 @@ _Returns_ - `boolean`: Whether the given block is a template part. -### isUnmodifiedDefaultBlock +### isUnmodifiedBlock -Determines whether the block is a default block -and its attributes are equal to the default attributes +Determines whether the block's attributes are equal to the default attributes which means the block is unmodified. _Parameters_ @@ -541,7 +540,20 @@ _Parameters_ _Returns_ -- `boolean`: Whether the block is an unmodified default block +- `boolean`: Whether the block is an unmodified block. + +### isUnmodifiedDefaultBlock + +Determines whether the block is a default block and its attributes are equal +to the default attributes which means the block is unmodified. + +_Parameters_ + +- _block_ `WPBlock`: Block Object + +_Returns_ + +- `boolean`: Whether the block is an unmodified default block. ### isValidBlockContent diff --git a/packages/blocks/package.json b/packages/blocks/package.json index b67f0e6266e05..028f2eb47af90 100644 --- a/packages/blocks/package.json +++ b/packages/blocks/package.json @@ -44,6 +44,7 @@ "@wordpress/shortcode": "file:../shortcode", "change-case": "^4.1.2", "colord": "^2.7.0", + "fast-deep-equal": "^3.1.3", "hpq": "^1.3.0", "is-plain-object": "^5.0.0", "lodash": "^4.17.21", @@ -55,7 +56,7 @@ "uuid": "^8.3.0" }, "peerDependencies": { - "react": "^17.0.0" + "react": "^18.0.0" }, "publishConfig": { "access": "public" diff --git a/packages/blocks/src/api/index.js b/packages/blocks/src/api/index.js index afa11f82c6b6b..2ddeb3a60f0ab 100644 --- a/packages/blocks/src/api/index.js +++ b/packages/blocks/src/api/index.js @@ -137,6 +137,7 @@ export { unregisterBlockVariation, } from './registration'; export { + isUnmodifiedBlock, isUnmodifiedDefaultBlock, normalizeIconObject, isValidIcon, diff --git a/packages/blocks/src/api/parser/get-block-attributes.js b/packages/blocks/src/api/parser/get-block-attributes.js index b9fd3dcf67508..ffa67c11250c2 100644 --- a/packages/blocks/src/api/parser/get-block-attributes.js +++ b/packages/blocks/src/api/parser/get-block-attributes.js @@ -102,18 +102,20 @@ export function isOfTypes( value, types ) { * commentAttributes returns the attribute value depending on its source * definition of the given attribute key. * - * @param {string} attributeKey Attribute key. - * @param {Object} attributeSchema Attribute's schema. - * @param {string|Node} innerHTML Block's raw content. - * @param {Object} commentAttributes Block's comment attributes. + * @param {string} attributeKey Attribute key. + * @param {Object} attributeSchema Attribute's schema. + * @param {Node} innerDOM Parsed DOM of block's inner HTML. + * @param {Object} commentAttributes Block's comment attributes. + * @param {string} innerHTML Raw HTML from block node's innerHTML property. * * @return {*} Attribute value. */ export function getBlockAttribute( attributeKey, attributeSchema, - innerHTML, - commentAttributes + innerDOM, + commentAttributes, + innerHTML ) { let value; @@ -125,6 +127,10 @@ export function getBlockAttribute( ? commentAttributes[ attributeKey ] : undefined; break; + // raw source means that it's the original raw block content. + case 'raw': + value = innerHTML; + break; case 'attribute': case 'property': case 'html': @@ -133,7 +139,7 @@ export function getBlockAttribute( case 'node': case 'query': case 'tag': - value = parseWithAttributeSchema( innerHTML, attributeSchema ); + value = parseWithAttributeSchema( innerDOM, attributeSchema ); break; } @@ -270,7 +276,7 @@ export function getBlockAttributes( const blockType = normalizeBlockType( blockTypeOrName ); const blockAttributes = mapValues( blockType.attributes, ( schema, key ) => - getBlockAttribute( key, schema, doc, attributes ) + getBlockAttribute( key, schema, doc, attributes, innerHTML ) ); return applyFilters( diff --git a/packages/blocks/src/api/raw-handling/paste-handler.js b/packages/blocks/src/api/raw-handling/paste-handler.js index d0193bc87eb74..c4ad40e0b1f50 100644 --- a/packages/blocks/src/api/raw-handling/paste-handler.js +++ b/packages/blocks/src/api/raw-handling/paste-handler.js @@ -47,6 +47,7 @@ const { console } = window; */ function filterInlineHTML( HTML, preserveWhiteSpace ) { HTML = deepFilterHTML( HTML, [ + headRemover, googleDocsUIDRemover, phrasingContentReducer, commentRemover, diff --git a/packages/blocks/src/api/raw-handling/test/paste-handler.js b/packages/blocks/src/api/raw-handling/test/paste-handler.js new file mode 100644 index 0000000000000..378189b10e0ea --- /dev/null +++ b/packages/blocks/src/api/raw-handling/test/paste-handler.js @@ -0,0 +1,131 @@ +/** + * WordPress dependencies + */ +import { pasteHandler } from '@wordpress/blocks'; +/** + * Internal dependencies + */ +import { init as initAndRegisterTableBlock } from '../../../../../block-library/src/table'; + +const tableWithHeaderFooterAndBodyUsingColspan = ` + + + + + + + + + + + + + + + + + + + +
      Colspan 2Header Cell
      Footer CellFooter Cell
      Colspan 2Cell Data
      `; + +const googleDocsTableWithColspan = ` + +
      + + + + + + + + + + +
      +

      Test colspan +

      +
      +

      +`; + +describe( 'pasteHandler', () => { + beforeAll( () => { + initAndRegisterTableBlock(); + } ); + + it( 'can handle a table with thead, tbody and tfoot using colspan', () => { + const [ result ] = pasteHandler( { + HTML: tableWithHeaderFooterAndBodyUsingColspan, + tagName: 'p', + preserveWhiteSpace: false, + } ); + + expect( console ).toHaveLogged(); + + expect( result.attributes ).toEqual( { + hasFixedLayout: false, + caption: '', + head: [ + { + cells: [ + { content: 'Colspan 2', tag: 'th', colspan: '2' }, + { content: 'Header Cell', tag: 'th' }, + ], + }, + ], + body: [ + { + cells: [ + { content: 'Colspan 2', tag: 'td', colspan: '2' }, + { content: 'Cell Data', tag: 'td' }, + ], + }, + ], + foot: [ + { + cells: [ + { content: 'Footer Cell', tag: 'th', colspan: '2' }, + { content: 'Footer Cell', tag: 'th' }, + ], + }, + ], + } ); + expect( result.name ).toEqual( 'core/table' ); + expect( result.isValid ).toBeTruthy(); + } ); + + it( 'can handle a google docs table with colspan', () => { + const [ result ] = pasteHandler( { + HTML: googleDocsTableWithColspan, + tagName: 'p', + preserveWhiteSpace: false, + } ); + + expect( console ).toHaveLogged(); + + expect( result.attributes ).toEqual( { + body: [ + { + cells: [ + { + align: undefined, + colspan: '2', + content: 'Test colspan', + scope: undefined, + tag: 'td', + }, + ], + }, + ], + caption: '', + foot: [], + hasFixedLayout: false, + head: [], + } ); + expect( result.name ).toEqual( 'core/table' ); + expect( result.isValid ).toBeTruthy(); + } ); +} ); diff --git a/packages/blocks/src/api/registration.js b/packages/blocks/src/api/registration.js index f8f5ac996173b..6c34fea8d95a8 100644 --- a/packages/blocks/src/api/registration.js +++ b/packages/blocks/src/api/registration.js @@ -618,7 +618,7 @@ export function isReusableBlock( blockOrType ) { * @return {boolean} Whether the given block is a template part. */ export function isTemplatePart( blockOrType ) { - return blockOrType.name === 'core/template-part'; + return blockOrType?.name === 'core/template-part'; } /** diff --git a/packages/blocks/src/api/utils.js b/packages/blocks/src/api/utils.js index bc68ccd189537..c43445c627226 100644 --- a/packages/blocks/src/api/utils.js +++ b/packages/blocks/src/api/utils.js @@ -30,37 +30,40 @@ extend( [ namesPlugin, a11yPlugin ] ); const ICON_COLORS = [ '#191e23', '#f8f9f9' ]; /** - * Determines whether the block is a default block - * and its attributes are equal to the default attributes + * Determines whether the block's attributes are equal to the default attributes * which means the block is unmodified. * * @param {WPBlock} block Block Object * - * @return {boolean} Whether the block is an unmodified default block + * @return {boolean} Whether the block is an unmodified block. */ -export function isUnmodifiedDefaultBlock( block ) { - const defaultBlockName = getDefaultBlockName(); - if ( block.name !== defaultBlockName ) { - return false; - } - +export function isUnmodifiedBlock( block ) { // Cache a created default block if no cache exists or the default block // name changed. - if ( - ! isUnmodifiedDefaultBlock.block || - isUnmodifiedDefaultBlock.block.name !== defaultBlockName - ) { - isUnmodifiedDefaultBlock.block = createBlock( defaultBlockName ); + if ( ! isUnmodifiedBlock[ block.name ] ) { + isUnmodifiedBlock[ block.name ] = createBlock( block.name ); } - const newDefaultBlock = isUnmodifiedDefaultBlock.block; - const blockType = getBlockType( defaultBlockName ); + const newBlock = isUnmodifiedBlock[ block.name ]; + const blockType = getBlockType( block.name ); return Object.keys( blockType?.attributes ?? {} ).every( - ( key ) => newDefaultBlock.attributes[ key ] === block.attributes[ key ] + ( key ) => newBlock.attributes[ key ] === block.attributes[ key ] ); } +/** + * Determines whether the block is a default block and its attributes are equal + * to the default attributes which means the block is unmodified. + * + * @param {WPBlock} block Block Object + * + * @return {boolean} Whether the block is an unmodified default block. + */ +export function isUnmodifiedDefaultBlock( block ) { + return block.name === getDefaultBlockName() && isUnmodifiedBlock( block ); +} + /** * Function that checks if the parameter is a valid icon. * diff --git a/packages/blocks/src/api/validation/index.js b/packages/blocks/src/api/validation/index.js index 6f64c5f3923ab..abea733fe4471 100644 --- a/packages/blocks/src/api/validation/index.js +++ b/packages/blocks/src/api/validation/index.js @@ -2,7 +2,7 @@ * External dependencies */ import { Tokenizer } from 'simple-html-tokenizer'; -import { isEqual } from 'lodash'; +import fastDeepEqual from 'fast-deep-equal/es6'; /** * WordPress dependencies @@ -423,7 +423,9 @@ export const isEqualAttributesOfName = { return actualDiff.length === 0 && expectedDiff.length === 0; }, style: ( actual, expected ) => { - return isEqual( ...[ actual, expected ].map( getStyleProperties ) ); + return fastDeepEqual( + ...[ actual, expected ].map( getStyleProperties ) + ); }, // For each boolean attribute, mere presence of attribute in both is enough // to assume equivalence. diff --git a/packages/blocks/src/store/actions.js b/packages/blocks/src/store/actions.js index 76cf631e81448..f8ba8ac83769f 100644 --- a/packages/blocks/src/store/actions.js +++ b/packages/blocks/src/store/actions.js @@ -2,7 +2,6 @@ * External dependencies */ import { isPlainObject } from 'is-plain-object'; -import { pick } from 'lodash'; /** * WordPress dependencies @@ -72,23 +71,24 @@ const processBlockType = ( blockType, { select } ) => { if ( settings.deprecated ) { settings.deprecated = settings.deprecated.map( ( deprecation ) => - pick( - // Only keep valid deprecation keys. - applyFilters( - 'blocks.registerBlockType', - // Merge deprecation keys with pre-filter settings - // so that filters that depend on specific keys being - // present don't fail. - { - // Omit deprecation keys here so that deprecations - // can opt out of specific keys like "supports". - ...omit( blockType, DEPRECATED_ENTRY_KEYS ), - ...deprecation, - }, - name, - deprecation - ), - DEPRECATED_ENTRY_KEYS + Object.fromEntries( + Object.entries( + // Only keep valid deprecation keys. + applyFilters( + 'blocks.registerBlockType', + // Merge deprecation keys with pre-filter settings + // so that filters that depend on specific keys being + // present don't fail. + { + // Omit deprecation keys here so that deprecations + // can opt out of specific keys like "supports". + ...omit( blockType, DEPRECATED_ENTRY_KEYS ), + ...deprecation, + }, + name, + deprecation + ) + ).filter( ( [ key ] ) => DEPRECATED_ENTRY_KEYS.includes( key ) ) ) ); } diff --git a/packages/blocks/src/store/reducer.js b/packages/blocks/src/store/reducer.js index 6770dd6b67f26..16f95127137a5 100644 --- a/packages/blocks/src/store/reducer.js +++ b/packages/blocks/src/store/reducer.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { filter, find, get, isEmpty, map, mapValues } from 'lodash'; +import { get, isEmpty, map, mapValues } from 'lodash'; /** * WordPress dependencies @@ -144,8 +144,11 @@ export function blockStyles( state = {}, action ) { case 'REMOVE_BLOCK_STYLES': return { ...state, - [ action.blockName ]: filter( - get( state, [ action.blockName ], [] ), + [ action.blockName ]: get( + state, + [ action.blockName ], + [] + ).filter( ( style ) => action.styleNames.indexOf( style.name ) === -1 ), }; @@ -195,8 +198,11 @@ export function blockVariations( state = {}, action ) { case 'REMOVE_BLOCK_VARIATIONS': return { ...state, - [ action.blockName ]: filter( - get( state, [ action.blockName ], [] ), + [ action.blockName ]: get( + state, + [ action.blockName ], + [] + ).filter( ( variation ) => action.variationNames.indexOf( variation.name ) === -1 ), @@ -259,7 +265,9 @@ export function categories( state = DEFAULT_CATEGORIES, action ) { if ( ! action.category || isEmpty( action.category ) ) { return state; } - const categoryToChange = find( state, [ 'slug', action.slug ] ); + const categoryToChange = state.find( + ( { slug } ) => slug === action.slug + ); if ( categoryToChange ) { return map( state, ( category ) => { if ( category.slug === action.slug ) { diff --git a/packages/blocks/src/store/selectors.js b/packages/blocks/src/store/selectors.js index 298d7fc12141a..27cacef408582 100644 --- a/packages/blocks/src/store/selectors.js +++ b/packages/blocks/src/store/selectors.js @@ -3,7 +3,7 @@ */ import createSelector from 'rememo'; import removeAccents from 'remove-accents'; -import { filter, get, map } from 'lodash'; +import { get, map } from 'lodash'; /** * WordPress dependencies @@ -554,7 +554,7 @@ export function getGroupingBlockName( state ) { export const getChildBlockNames = createSelector( ( state, blockName ) => { return map( - filter( state.blockTypes, ( blockType ) => { + getBlockTypes( state ).filter( ( blockType ) => { return blockType.parent?.includes( blockName ); } ), ( { name } ) => name diff --git a/packages/blocks/src/store/test/selectors.js b/packages/blocks/src/store/test/selectors.js index 8793e1e354e44..1fda11d72311a 100644 --- a/packages/blocks/src/store/test/selectors.js +++ b/packages/blocks/src/store/test/selectors.js @@ -86,26 +86,28 @@ describe( 'selectors', () => { describe( 'getChildBlockNames', () => { it( 'should return an empty array if state is empty', () => { - const state = {}; + const state = { + blockTypes: {}, + }; expect( getChildBlockNames( state, 'parent1' ) ).toHaveLength( 0 ); } ); it( 'should return an empty array if no children exist', () => { const state = { - blockTypes: [ - { + blockTypes: { + child1: { name: 'child1', parent: [ 'parent1' ], }, - { + child2: { name: 'child2', parent: [ 'parent2' ], }, - { + parent3: { name: 'parent3', }, - ], + }, }; expect( getChildBlockNames( state, 'parent3' ) ).toHaveLength( 0 ); @@ -113,15 +115,15 @@ describe( 'selectors', () => { it( 'should return an empty array if the parent block is not found', () => { const state = { - blockTypes: [ - { + blockTypes: { + child1: { name: 'child1', parent: [ 'parent1' ], }, - { + parent1: { name: 'parent1', }, - ], + }, }; expect( getChildBlockNames( state, 'parent3' ) ).toHaveLength( 0 ); @@ -129,29 +131,29 @@ describe( 'selectors', () => { it( 'should return an array with the child block names', () => { const state = { - blockTypes: [ - { + blockTypes: { + child1: { name: 'child1', parent: [ 'parent1' ], }, - { + child2: { name: 'child2', parent: [ 'parent2' ], }, - { + child3: { name: 'child3', parent: [ 'parent1' ], }, - { + child4: { name: 'child4', }, - { + parent1: { name: 'parent1', }, - { + parent2: { name: 'parent2', }, - ], + }, }; expect( getChildBlockNames( state, 'parent1' ) ).toEqual( [ @@ -162,25 +164,25 @@ describe( 'selectors', () => { it( 'should return an array with the child block names even if only one child exists', () => { const state = { - blockTypes: [ - { + blockTypes: { + child1: { name: 'child1', parent: [ 'parent1' ], }, - { + child2: { name: 'child2', parent: [ 'parent2' ], }, - { + child4: { name: 'child4', }, - { + parent1: { name: 'parent1', }, - { + parent2: { name: 'parent2', }, - ], + }, }; expect( getChildBlockNames( state, 'parent1' ) ).toEqual( [ @@ -190,29 +192,29 @@ describe( 'selectors', () => { it( 'should return an array with the child block names even if children have multiple parents', () => { const state = { - blockTypes: [ - { + blockTypes: { + child1: { name: 'child1', parent: [ 'parent1' ], }, - { + child2: { name: 'child2', parent: [ 'parent1', 'parent2' ], }, - { + child3: { name: 'child3', parent: [ 'parent1' ], }, - { + child4: { name: 'child4', }, - { + parent1: { name: 'parent1', }, - { + parent2: { name: 'parent2', }, - ], + }, }; expect( getChildBlockNames( state, 'parent1' ) ).toEqual( [ diff --git a/packages/components/CHANGELOG.md b/packages/components/CHANGELOG.md index bc95431087c2b..7ed9fc6779bba 100644 --- a/packages/components/CHANGELOG.md +++ b/packages/components/CHANGELOG.md @@ -2,11 +2,63 @@ ## Unreleased +### Breaking Changes + +- Updated dependencies to require React 18 ([45235](https://github.com/WordPress/gutenberg/pull/45235)) + +### New Feature + +- `TabPanel`: support manual tab activation ([#46004](https://github.com/WordPress/gutenberg/pull/46004)). +- `BaseControl`: Add `useBaseControlProps` hook to help generate id-releated props ([#46170](https://github.com/WordPress/gutenberg/pull/46170)). + +### Bug Fix + +- `ColorPalette`: show "Clear" button even when colors array is empty ([#46001](https://github.com/WordPress/gutenberg/pull/46001)). +- `InputControl`: Fix internal `Flex` wrapper usage that could add an unintended `height: 100%` ([#46213](https://github.com/WordPress/gutenberg/pull/46213)). +- `Navigator`: Allow calling `goTo` and `goBack` twice in one render cycle ([#46391](https://github.com/WordPress/gutenberg/pull/46391)). +- `Modal`: Fix unexpected modal closing in IME Composition ([#46453](https://github.com/WordPress/gutenberg/pull/46453)). + +### Enhancements + +- `TabPanel`: Simplify tab-focus style. ([#46276](https://github.com/WordPress/gutenberg/pull/46276)). +- `TabPanel`: Add ability to set icon only tab buttons ([#45005](https://github.com/WordPress/gutenberg/pull/45005)). +- `InputControl`, `NumberControl`, `UnitControl`: Add `help` prop for additional description ([#45931](https://github.com/WordPress/gutenberg/pull/45931)). +- `BorderControl`, `ColorPicker` & `QueryControls`: Replace bottom margin overrides with `__nextHasNoMarginBottom` ([#45985](https://github.com/WordPress/gutenberg/pull/45985)). +- `CustomSelectControl`, `UnitControl`: Add `onFocus` and `onBlur` props ([#46096](https://github.com/WordPress/gutenberg/pull/46096)). +- `ResizableBox`: Prevent unnecessary paint on resize handles ([#46196](https://github.com/WordPress/gutenberg/pull/46196)). +- `Popover`: Prevent unnecessary paint caused by using outline ([#46201](https://github.com/WordPress/gutenberg/pull/46201)). +- `PaletteEdit`: Global styles: add onChange actions to color palette items [#45681](https://github.com/WordPress/gutenberg/pull/45681). +- Lighten the border color on control components ([#46252](https://github.com/WordPress/gutenberg/pull/46252)). +- `Popover`: Prevent unnecessary paint when scrolling by using transform instead of top/left positionning ([#46187](https://github.com/WordPress/gutenberg/pull/46187)). +- `CircularOptionPicker`: Prevent unecessary paint on hover ([#46197](https://github.com/WordPress/gutenberg/pull/46197)). + +### Experimental + +- `TextControl`: Restrict `type` prop to `email`, `number`, `password`, `tel`, `text`, `search` or `url` ([#45433](https://github.com/WordPress/gutenberg/pull/45433/)). + +### Internal + +- `useControlledValue`: let TypeScript infer the return type ([#46164](https://github.com/WordPress/gutenberg/pull/46164)). +- `LinkedButton`: remove unnecessary `span` tag ([#46063](https://github.com/WordPress/gutenberg/pull/46063)). +- NumberControl: refactor styles/tests/stories to TypeScript, replace fireEvent with user-event ([#45990](https://github.com/WordPress/gutenberg/pull/45990)). +- `useBaseField`: Convert to TypeScript ([#45712](https://github.com/WordPress/gutenberg/pull/45712)). +- `Dashicon`: Convert to TypeScript ([#45924](https://github.com/WordPress/gutenberg/pull/45924)). +- `PaletteEdit`: add follow up changelog for #45681 and tests [#46095](https://github.com/WordPress/gutenberg/pull/46095). +- `AlignmentMatrixControl`: Convert to TypeScript ([#46162](https://github.com/WordPress/gutenberg/pull/46162)). + +### Documentation + +- `Tooltip`: Add readme and unit tests for `shortcut` prop ([#46092](https://github.com/WordPress/gutenberg/pull/46092)). + ## 22.1.0 (2022-11-16) ### Enhancements - `ColorPalette`, `BorderBox`, `BorderBoxControl`: polish and DRY prop types, add default values ([#45463](https://github.com/WordPress/gutenberg/pull/45463)). +- `TabPanel`: Add ability to set icon only tab buttons ([#45005](https://github.com/WordPress/gutenberg/pull/45005)). + +### Internal +- `AnglePickerControl`: remove `:focus-visible' outline on `CircleOutlineWrapper` ([#45758](https://github.com/WordPress/gutenberg/pull/45758)) ### Bug Fix @@ -16,6 +68,7 @@ - `Icon`: Making size prop work for icon components using dash icon strings ([#45593](https://github.com/WordPress/gutenberg/pull/45593)) - `ToolsPanelItem`: Prevent unintended calls to onDeselect when parent panel is remounted and item is rendered via SlotFill ([#45673](https://github.com/WordPress/gutenberg/pull/45673)) - `ColorPicker`: Prevent all number fields from becoming "0" when one of them is an empty string ([#45649](https://github.com/WordPress/gutenberg/pull/45649)). +- `ToggleControl`: Fix toggle control label text overflow ([#45962](https://github.com/WordPress/gutenberg/pull/45962)). ### Internal @@ -31,10 +84,12 @@ - `MenuGroup`: Convert to TypeScript ([#45617](https://github.com/WordPress/gutenberg/pull/45617)). - `useCx`: fix story to satisfy the `react-hooks/exhaustive-deps` eslint rule ([#45614](https://github.com/WordPress/gutenberg/pull/45614)) - Activate the `react-hooks/exhuastive-deps` eslint rule for the Components package ([#41166](https://github.com/WordPress/gutenberg/pull/41166)) +- `Snackbar`: Convert to TypeScript ([#45472](https://github.com/WordPress/gutenberg/pull/45472)). ### Experimental - `ToggleGroupControl`: Only show enclosing border when `isBlock` and not `isDeselectable` ([#45492](https://github.com/WordPress/gutenberg/pull/45492)). +- `Theme`: Add support for custom `background` color ([#45466](https://github.com/WordPress/gutenberg/pull/45466)). ## 22.0.0 (2022-11-02) diff --git a/packages/components/package.json b/packages/components/package.json index 5afcbf72040c8..ca77fb07450e7 100644 --- a/packages/components/package.json +++ b/packages/components/package.json @@ -59,7 +59,8 @@ "date-fns": "^2.28.0", "dom-scroll-into-view": "^1.2.1", "downshift": "^6.0.15", - "framer-motion": "^6.2.8", + "fast-deep-equal": "^3.1.3", + "framer-motion": "^7.6.1", "gradient-parser": "^0.1.5", "highlight-words-core": "^1.2.2", "lodash": "^4.17.21", @@ -73,8 +74,8 @@ "valtio": "^1.7.0" }, "peerDependencies": { - "react": "^17.0.0", - "react-dom": "^17.0.0" + "react": "^18.0.0", + "react-dom": "^18.0.0" }, "publishConfig": { "access": "public" diff --git a/packages/components/src/alignment-matrix-control/README.md b/packages/components/src/alignment-matrix-control/README.md index 16b7333c09db7..576fe77c1f25a 100644 --- a/packages/components/src/alignment-matrix-control/README.md +++ b/packages/components/src/alignment-matrix-control/README.md @@ -30,8 +30,7 @@ The component accepts the following props: ### className -The class that will be added to the classes of the wrapper component. - +The class that will be added to the classes of the underlying `grid` widget. - Type: `string` - Required: No @@ -44,7 +43,7 @@ Unique ID for the component. ### label -Accessible label. If provided, sets the `aria-label` attribute of the underlying component. +Accessible label. If provided, sets the `aria-label` attribute of the underlying `grid` widget. - Type: `string` - Required: No @@ -54,26 +53,27 @@ Accessible label. If provided, sets the `aria-label` attribute of the underlying If provided, sets the default alignment value. -- Type: `string` +- Type: `AlignmentMatrixControlValue` - Required: No - Default: `center center` ### value The current alignment value. -- Type: `string` + +- Type: `AlignmentMatrixControlValue` - Required: No ### onChange A function that receives the updated alignment value. -- Type: `( nextValue: string ) => void` +- Type: `( newValue: AlignmentMatrixControlValue ) => void` - Required: No ### width -If provided, sets the width of the wrapper component. +If provided, sets the width of the control. - Type: `number` - Required: No diff --git a/packages/components/src/alignment-matrix-control/cell.js b/packages/components/src/alignment-matrix-control/cell.tsx similarity index 74% rename from packages/components/src/alignment-matrix-control/cell.js rename to packages/components/src/alignment-matrix-control/cell.tsx index 2aeb839831c0c..b6e19c5602291 100644 --- a/packages/components/src/alignment-matrix-control/cell.js +++ b/packages/components/src/alignment-matrix-control/cell.tsx @@ -13,8 +13,14 @@ import { Cell as CellView, Point, } from './styles/alignment-matrix-control-styles'; +import type { AlignmentMatrixControlCellProps } from './types'; +import type { WordPressComponentProps } from '../ui/context'; -export default function Cell( { isActive = false, value, ...props } ) { +export default function Cell( { + isActive = false, + value, + ...props +}: WordPressComponentProps< AlignmentMatrixControlCellProps, 'span', false > ) { const tooltipText = ALIGNMENT_LABEL[ value ]; return ( diff --git a/packages/components/src/alignment-matrix-control/icon.js b/packages/components/src/alignment-matrix-control/icon.tsx similarity index 77% rename from packages/components/src/alignment-matrix-control/icon.js rename to packages/components/src/alignment-matrix-control/icon.tsx index e712b7471ff23..b294bc7551e19 100644 --- a/packages/components/src/alignment-matrix-control/icon.js +++ b/packages/components/src/alignment-matrix-control/icon.tsx @@ -12,17 +12,19 @@ import { Cell, Point, } from './styles/alignment-matrix-control-icon-styles'; +import type { AlignmentMatrixControlIconProps } from './types'; +import type { WordPressComponentProps } from '../ui/context'; const BASE_SIZE = 24; -export default function AlignmentMatrixControlIcon( { +function AlignmentMatrixControlIcon( { className, disablePointerEvents = true, size = BASE_SIZE, style = {}, value = 'center', ...props -} ) { +}: WordPressComponentProps< AlignmentMatrixControlIconProps, 'div', false > ) { const alignIndex = getAlignmentIndex( value ); const scale = ( size / BASE_SIZE ).toFixed( 2 ); @@ -42,7 +44,6 @@ export default function AlignmentMatrixControlIcon( { className={ classes } disablePointerEvents={ disablePointerEvents } role="presentation" - size={ size } style={ styles } > { ALIGNMENTS.map( ( align, index ) => { @@ -57,3 +58,5 @@ export default function AlignmentMatrixControlIcon( { ); } + +export default AlignmentMatrixControlIcon; diff --git a/packages/components/src/alignment-matrix-control/index.js b/packages/components/src/alignment-matrix-control/index.tsx similarity index 70% rename from packages/components/src/alignment-matrix-control/index.js rename to packages/components/src/alignment-matrix-control/index.tsx index 44dbe33d95a94..0e34e4052346d 100644 --- a/packages/components/src/alignment-matrix-control/index.js +++ b/packages/components/src/alignment-matrix-control/index.tsx @@ -18,10 +18,15 @@ import { Composite, CompositeGroup, useCompositeState } from '../composite'; import { Root, Row } from './styles/alignment-matrix-control-styles'; import AlignmentMatrixControlIcon from './icon'; import { GRID, getItemId } from './utils'; +import type { WordPressComponentProps } from '../ui/context'; +import type { + AlignmentMatrixControlProps, + AlignmentMatrixControlValue, +} from './types'; const noop = () => {}; -function useBaseId( id ) { +function useBaseId( id?: string ) { const instanceId = useInstanceId( AlignmentMatrixControl, 'alignment-matrix-control' @@ -30,7 +35,27 @@ function useBaseId( id ) { return id || instanceId; } -export default function AlignmentMatrixControl( { +/** + * + * AlignmentMatrixControl components enable adjustments to horizontal and vertical alignments for UI. + * + * ```jsx + * import { __experimentalAlignmentMatrixControl as AlignmentMatrixControl } from '@wordpress/components'; + * import { useState } from '@wordpress/element'; + * + * const Example = () => { + * const [ alignment, setAlignment ] = useState( 'center center' ); + * + * return ( + * + * ); + * }; + * ``` + */ +export function AlignmentMatrixControl( { className, id, label = __( 'Alignment Matrix Control' ), @@ -39,7 +64,7 @@ export default function AlignmentMatrixControl( { onChange = noop, width = 92, ...props -} ) { +}: WordPressComponentProps< AlignmentMatrixControlProps, 'div', false > ) { const [ immutableDefaultValue ] = useState( value ?? defaultValue ); const baseId = useBaseId( id ); const initialCurrentId = getItemId( baseId, immutableDefaultValue ); @@ -50,7 +75,7 @@ export default function AlignmentMatrixControl( { rtl: isRTL(), } ); - const handleOnChange = ( nextValue ) => { + const handleOnChange = ( nextValue: AlignmentMatrixControlValue ) => { onChange( nextValue ); }; @@ -107,3 +132,5 @@ export default function AlignmentMatrixControl( { } AlignmentMatrixControl.Icon = AlignmentMatrixControlIcon; + +export default AlignmentMatrixControl; diff --git a/packages/components/src/alignment-matrix-control/stories/index.js b/packages/components/src/alignment-matrix-control/stories/index.tsx similarity index 70% rename from packages/components/src/alignment-matrix-control/stories/index.js rename to packages/components/src/alignment-matrix-control/stories/index.tsx index babdd47c1e95f..6401907f92169 100644 --- a/packages/components/src/alignment-matrix-control/stories/index.js +++ b/packages/components/src/alignment-matrix-control/stories/index.tsx @@ -1,3 +1,8 @@ +/** + * External dependencies + */ +import type { ComponentMeta, ComponentStory } from '@storybook/react'; + /** * WordPress dependencies */ @@ -7,21 +12,18 @@ import { Icon } from '@wordpress/icons'; /** * Internal dependencies */ -import AlignmentMatrixControl from '../'; -import { ALIGNMENTS } from '../utils'; +import AlignmentMatrixControl from '..'; import { HStack } from '../../h-stack'; +import type { AlignmentMatrixControlProps } from '../types'; -export default { +const meta: ComponentMeta< typeof AlignmentMatrixControl > = { title: 'Components (Experimental)/AlignmentMatrixControl', component: AlignmentMatrixControl, subcomponents: { 'AlignmentMatrixControl.Icon': AlignmentMatrixControl.Icon, }, argTypes: { - defaultValue: { options: ALIGNMENTS }, onChange: { action: 'onChange', control: { type: null } }, - label: { control: { type: 'text' } }, - width: { control: { type: 'number' } }, value: { control: { type: null } }, }, parameters: { @@ -29,9 +31,15 @@ export default { docs: { source: { state: 'open' } }, }, }; +export default meta; -const Template = ( { defaultValue, onChange, ...props } ) => { - const [ value, setValue ] = useState(); +const Template: ComponentStory< typeof AlignmentMatrixControl > = ( { + defaultValue, + onChange, + ...props +} ) => { + const [ value, setValue ] = + useState< AlignmentMatrixControlProps[ 'value' ] >(); // Convenience handler for Canvas view so changes are reflected useEffect( () => { @@ -43,7 +51,7 @@ const Template = ( { defaultValue, onChange, ...props } ) => { { ...props } onChange={ ( ...changeArgs ) => { setValue( ...changeArgs ); - onChange( ...changeArgs ); + onChange?.( ...changeArgs ); } } value={ value } /> diff --git a/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-icon-styles.js b/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-icon-styles.ts similarity index 72% rename from packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-icon-styles.js rename to packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-icon-styles.ts index 5333e9b775625..a14894633e05d 100644 --- a/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-icon-styles.js +++ b/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-icon-styles.ts @@ -12,6 +12,10 @@ import { pointBase, Cell as CellBase, } from './alignment-matrix-control-styles'; +import type { + AlignmentMatrixControlIconProps, + AlignmentMatrixControlCellProps, +} from '../types'; const rootSize = () => { const padding = 1.5; @@ -25,9 +29,11 @@ const rootSize = () => { } ); }; -const rootPointerEvents = ( { disablePointerEvents } ) => { +const rootPointerEvents = ( { + disablePointerEvents, +}: Pick< AlignmentMatrixControlIconProps, 'disablePointerEvents' > ) => { return css( { - pointerEvents: disablePointerEvents ? 'none' : null, + pointerEvents: disablePointerEvents ? 'none' : undefined, } ); }; @@ -46,7 +52,9 @@ export const Root = styled.div` ${ rootPointerEvents }; `; -const pointActive = ( { isActive } ) => { +const pointActive = ( { + isActive, +}: Pick< AlignmentMatrixControlCellProps, 'isActive' > ) => { const boxShadow = isActive ? `0 0 0 1px currentColor` : null; return css` diff --git a/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-styles.js b/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-styles.ts similarity index 81% rename from packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-styles.js rename to packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-styles.ts index f16ec53827403..5adde6002ba3b 100644 --- a/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-styles.js +++ b/packages/components/src/alignment-matrix-control/styles/alignment-matrix-control-styles.ts @@ -8,6 +8,10 @@ import { css } from '@emotion/react'; * Internal dependencies */ import { COLORS, reduceMotion } from '../../utils'; +import type { + AlignmentMatrixControlProps, + AlignmentMatrixControlCellProps, +} from '../types'; export const rootBase = () => { return css` @@ -27,7 +31,9 @@ const rootSize = ( { size = 92 } ) => { `; }; -export const Root = styled.div` +export const Root = styled.div< { + size: AlignmentMatrixControlProps[ 'width' ]; +} >` ${ rootBase }; border: 1px solid transparent; @@ -43,7 +49,9 @@ export const Row = styled.div` grid-template-columns: repeat( 3, 1fr ); `; -const pointActive = ( { isActive } ) => { +const pointActive = ( { + isActive, +}: Pick< AlignmentMatrixControlCellProps, 'isActive' > ) => { const boxShadow = isActive ? `0 0 0 2px ${ COLORS.gray[ 900 ] }` : null; const pointColor = isActive ? COLORS.gray[ 900 ] : COLORS.gray[ 400 ]; const pointColorHover = isActive ? COLORS.gray[ 900 ] : COLORS.ui.theme; @@ -58,7 +66,9 @@ const pointActive = ( { isActive } ) => { `; }; -export const pointBase = ( props ) => { +export const pointBase = ( + props: Pick< AlignmentMatrixControlCellProps, 'isActive' > +) => { return css` background: currentColor; box-sizing: border-box; diff --git a/packages/components/src/alignment-matrix-control/test/index.js b/packages/components/src/alignment-matrix-control/test/index.tsx similarity index 77% rename from packages/components/src/alignment-matrix-control/test/index.js rename to packages/components/src/alignment-matrix-control/test/index.tsx index 638d8acb96ad5..0e9661fe595b0 100644 --- a/packages/components/src/alignment-matrix-control/test/index.js +++ b/packages/components/src/alignment-matrix-control/test/index.tsx @@ -1,18 +1,18 @@ /** * External dependencies */ -import { render, screen, within } from '@testing-library/react'; +import { act, render, screen, within } from '@testing-library/react'; /** * Internal dependencies */ -import AlignmentMatrixControl from '../'; +import AlignmentMatrixControl from '..'; const getControl = () => { return screen.getByRole( 'grid' ); }; -const getCell = ( name ) => { +const getCell = ( name: string ) => { return within( getControl() ).getByRole( 'gridcell', { name } ); }; @@ -30,14 +30,14 @@ describe( 'AlignmentMatrixControl', () => { it.each( alignments )( 'should change value on %s cell click', - ( alignment ) => { + async ( alignment ) => { const spy = jest.fn(); render( ); - getCell( alignment ).focus(); + await act( () => getCell( alignment ).focus() ); expect( spy ).toHaveBeenCalledWith( alignment ); } diff --git a/packages/components/src/alignment-matrix-control/types.ts b/packages/components/src/alignment-matrix-control/types.ts new file mode 100644 index 0000000000000..892781234e694 --- /dev/null +++ b/packages/components/src/alignment-matrix-control/types.ts @@ -0,0 +1,54 @@ +export type AlignmentMatrixControlValue = + | 'top left' + | 'top center' + | 'top right' + | 'center left' + | 'center' + | 'center center' + | 'center right' + | 'bottom left' + | 'bottom center' + | 'bottom right'; + +export type AlignmentMatrixControlProps = { + /** + * Accessible label. If provided, sets the `aria-label` attribute of the + * underlying `grid` widget. + * + * @default 'Alignment Matrix Control' + */ + label?: string; + /** + * If provided, sets the default alignment value. + * + * @default 'center center' + */ + defaultValue?: AlignmentMatrixControlValue; + /** + * The current alignment value. + */ + value?: AlignmentMatrixControlValue; + /** + * A function that receives the updated alignment value. + */ + onChange?: ( newValue: AlignmentMatrixControlValue ) => void; + /** + * If provided, sets the width of the control. + * + * @default 92 + */ + width?: number; +}; + +export type AlignmentMatrixControlIconProps = Pick< + AlignmentMatrixControlProps, + 'value' +> & { + disablePointerEvents?: boolean; + size?: number; +}; + +export type AlignmentMatrixControlCellProps = { + isActive?: boolean; + value: NonNullable< AlignmentMatrixControlProps[ 'value' ] >; +}; diff --git a/packages/components/src/alignment-matrix-control/utils.js b/packages/components/src/alignment-matrix-control/utils.tsx similarity index 58% rename from packages/components/src/alignment-matrix-control/utils.js rename to packages/components/src/alignment-matrix-control/utils.tsx index 4934e422d196c..a9fa7e57d67db 100644 --- a/packages/components/src/alignment-matrix-control/utils.js +++ b/packages/components/src/alignment-matrix-control/utils.tsx @@ -2,20 +2,25 @@ * WordPress dependencies */ import { __ } from '@wordpress/i18n'; +/** + * Internal dependencies + */ +import type { AlignmentMatrixControlValue } from './types'; -export const GRID = [ +export const GRID: AlignmentMatrixControlValue[][] = [ [ 'top left', 'top center', 'top right' ], [ 'center left', 'center center', 'center right' ], [ 'bottom left', 'bottom center', 'bottom right' ], ]; // Stored as map as i18n __() only accepts strings (not variables) -export const ALIGNMENT_LABEL = { +export const ALIGNMENT_LABEL: Record< AlignmentMatrixControlValue, string > = { 'top left': __( 'Top Left' ), 'top center': __( 'Top Center' ), 'top right': __( 'Top Right' ), 'center left': __( 'Center Left' ), 'center center': __( 'Center Center' ), + center: __( 'Center Center' ), 'center right': __( 'Center Right' ), 'bottom left': __( 'Bottom Left' ), 'bottom center': __( 'Bottom Center' ), @@ -28,25 +33,28 @@ export const ALIGNMENTS = GRID.flat(); /** * Parses and transforms an incoming value to better match the alignment values * - * @param {string} value An alignment value to parse. + * @param value An alignment value to parse. * - * @return {string} The parsed value. + * @return The parsed value. */ -export function transformValue( value ) { +export function transformValue( value: AlignmentMatrixControlValue ) { const nextValue = value === 'center' ? 'center center' : value; - return nextValue.replace( '-', ' ' ); + return nextValue.replace( '-', ' ' ) as AlignmentMatrixControlValue; } /** * Creates an item ID based on a prefix ID and an alignment value. * - * @param {string} prefixId An ID to prefix. - * @param {string} value An alignment value. + * @param prefixId An ID to prefix. + * @param value An alignment value. * - * @return {string} The item id. + * @return The item id. */ -export function getItemId( prefixId, value ) { +export function getItemId( + prefixId: string, + value: AlignmentMatrixControlValue +) { const valueId = transformValue( value ).replace( ' ', '-' ); return `${ prefixId }-${ valueId }`; @@ -55,12 +63,14 @@ export function getItemId( prefixId, value ) { /** * Retrieves the alignment index from a value. * - * @param {string} alignment Value to check. + * @param alignment Value to check. * - * @return {number} The index of a matching alignment. + * @return The index of a matching alignment. */ -export function getAlignmentIndex( alignment = 'center' ) { - const item = transformValue( alignment ).replace( '-', ' ' ); +export function getAlignmentIndex( + alignment: AlignmentMatrixControlValue = 'center' +) { + const item = transformValue( alignment ); const index = ALIGNMENTS.indexOf( item ); return index > -1 ? index : undefined; diff --git a/packages/components/src/angle-picker-control/styles/angle-picker-control-styles.js b/packages/components/src/angle-picker-control/styles/angle-picker-control-styles.js index 1a7ed713d72f1..29b38850c055f 100644 --- a/packages/components/src/angle-picker-control/styles/angle-picker-control-styles.js +++ b/packages/components/src/angle-picker-control/styles/angle-picker-control-styles.js @@ -42,6 +42,10 @@ export const CircleIndicatorWrapper = styled.div` position: relative; width: 100%; height: 100%; + + :focus-visible { + outline: none; + } `; export const CircleIndicator = styled.div` diff --git a/packages/components/src/autocomplete/index.js b/packages/components/src/autocomplete/index.js index f96bd01750c01..45f5415e55de0 100644 --- a/packages/components/src/autocomplete/index.js +++ b/packages/components/src/autocomplete/index.js @@ -415,26 +415,57 @@ function useAutocomplete( { } export function useAutocompleteProps( options ) { + const [ isVisible, setIsVisible ] = useState( false ); const ref = useRef(); + const recordAfterInput = useRef(); const onKeyDownRef = useRef(); const { popover, listBoxId, activeId, onKeyDown } = useAutocomplete( { ...options, contentRef: ref, } ); onKeyDownRef.current = onKeyDown; + + useEffect( () => { + if ( isVisible ) { + if ( ! recordAfterInput.current ) { + recordAfterInput.current = options.record; + } else if ( + recordAfterInput.current.start !== options.record.start || + recordAfterInput.current.end !== options.record.end + ) { + setIsVisible( false ); + recordAfterInput.current = null; + } + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [ options.record ] ); + + const mergedRefs = useMergeRefs( [ + ref, + useRefEffect( ( element ) => { + function _onKeyDown( event ) { + onKeyDownRef.current( event ); + } + function _onInput() { + // Only show auto complete UI if the user is inputting text. + setIsVisible( true ); + recordAfterInput.current = null; + } + element.addEventListener( 'keydown', _onKeyDown ); + element.addEventListener( 'input', _onInput ); + return () => { + element.removeEventListener( 'keydown', _onKeyDown ); + element.removeEventListener( 'input', _onInput ); + }; + }, [] ), + ] ); + + if ( ! isVisible ) { + return { ref: mergedRefs }; + } + return { - ref: useMergeRefs( [ - ref, - useRefEffect( ( element ) => { - function _onKeyDown( event ) { - onKeyDownRef.current( event ); - } - element.addEventListener( 'keydown', _onKeyDown ); - return () => { - element.removeEventListener( 'keydown', _onKeyDown ); - }; - }, [] ), - ] ), + ref: mergedRefs, children: popover, 'aria-autocomplete': listBoxId ? 'list' : undefined, 'aria-owns': listBoxId, diff --git a/packages/components/src/base-control/README.md b/packages/components/src/base-control/README.md index d940a7bc2fdc2..c2f3dc3e0108f 100644 --- a/packages/components/src/base-control/README.md +++ b/packages/components/src/base-control/README.md @@ -4,16 +4,23 @@ ## Usage -Render a `BaseControl` for a textarea input: - ```jsx -import { BaseControl } from '@wordpress/components'; - -// The `id` prop is necessary to accessibly associate the label with the textarea -const MyBaseControl = () => ( - - + + ); ); ``` @@ -23,7 +30,9 @@ The component accepts the following props: ### id -The HTML `id` of the element (passed in as a child to `BaseControl`) to which labels and help text are being generated. This is necessary to accessibly associate the label with that element. +The HTML `id` of the control element (passed in as a child to `BaseControl`) to which labels and help text are being generated. This is necessary to accessibly associate the label with that element. + +The recommended way is to use the `useBaseControlProps` hook, which takes care of generating a unique `id` for you. Otherwise, if you choose to pass an explicit `id` to this prop, you are responsible for ensuring the uniqueness of the `id`. - Type: `String` - Required: No @@ -44,9 +53,9 @@ If true, the label will only be visible to screen readers. ### help -If this property is added, a help text will be generated using help property as the content. +Additional description for the control. It is preferable to use plain text for `help`, as it can be accessibly associated with the control using `aria-describedby`. When the `help` contains links, or otherwise non-plain text content, it will be associated with the control using `aria-details`. -- Type: `String|WPElement` +- Type: `ReactNode` - Required: No ### className diff --git a/packages/components/src/base-control/hooks.ts b/packages/components/src/base-control/hooks.ts new file mode 100644 index 0000000000000..400b15834fead --- /dev/null +++ b/packages/components/src/base-control/hooks.ts @@ -0,0 +1,45 @@ +/** + * WordPress dependencies + */ +import { useInstanceId } from '@wordpress/compose'; + +/** + * Internal dependencies + */ +import BaseControl from '.'; +import type { BaseControlProps } from './types'; + +/** + * Generate props for the `BaseControl` and the inner control itself. + * + * Namely, it takes care of generating a unique `id`, properly associating it with the `label` and `help` elements. + * + * @param props + */ +export function useBaseControlProps( + props: Omit< BaseControlProps, 'children' > +) { + const { help, id: preferredId, ...restProps } = props; + + const uniqueId = useInstanceId( + BaseControl, + 'wp-components-base-control', + preferredId + ); + + // ARIA descriptions can only contain plain text, so fall back to aria-details if not. + const helpPropName = + typeof help === 'string' ? 'aria-describedby' : 'aria-details'; + + return { + baseControlProps: { + id: uniqueId, + help, + ...restProps, + }, + controlProps: { + id: uniqueId, + ...( !! help ? { [ helpPropName ]: `${ uniqueId }__help` } : {} ), + }, + }; +} diff --git a/packages/components/src/base-control/index.tsx b/packages/components/src/base-control/index.tsx index 122577fdeba62..fa2df39bddd45 100644 --- a/packages/components/src/base-control/index.tsx +++ b/packages/components/src/base-control/index.tsx @@ -17,19 +17,30 @@ import { } from './styles/base-control-styles'; import type { WordPressComponentProps } from '../ui/context'; +export { useBaseControlProps } from './hooks'; + /** * `BaseControl` is a component used to generate labels and help text for components handling user inputs. * - * @example + * ```jsx + * import { BaseControl, useBaseControlProps } from '@wordpress/components'; + * * // Render a `BaseControl` for a textarea input - * import { BaseControl } from '@wordpress/components'; + * const MyCustomTextareaControl = ({ children, ...baseProps }) => ( + * // `useBaseControlProps` is a convenience hook to get the props for the `BaseControl` + * // and the inner control itself. Namely, it takes care of generating a unique `id`, + * // properly associating it with the `label` and `help` elements. + * const { baseControlProps, controlProps } = useBaseControlProps( baseProps ); * - * // The `id` prop is necessary to accessibly associate the label with the textarea - * const MyBaseControl = () => ( - * - * + * + * ); * ); + * ``` */ export const BaseControl = ( { __nextHasNoMarginBottom = false, diff --git a/packages/components/src/base-control/stories/index.tsx b/packages/components/src/base-control/stories/index.tsx index edc850f760745..b1b64caec0c3b 100644 --- a/packages/components/src/base-control/stories/index.tsx +++ b/packages/components/src/base-control/stories/index.tsx @@ -6,7 +6,7 @@ import type { ComponentMeta, ComponentStory } from '@storybook/react'; /** * Internal dependencies */ -import BaseControl from '..'; +import BaseControl, { useBaseControlProps } from '..'; import Button from '../../button'; const meta: ComponentMeta< typeof BaseControl > = { @@ -24,13 +24,14 @@ const meta: ComponentMeta< typeof BaseControl > = { }; export default meta; -const BaseControlWithTextarea: ComponentStory< typeof BaseControl > = ( { - id, - ...props -} ) => { +const BaseControlWithTextarea: ComponentStory< typeof BaseControl > = ( + props +) => { + const { baseControlProps, controlProps } = useBaseControlProps( props ); + return ( - -
      ', - 'textarea', - ); - - $examples['Simple title'] = array( - '<span class="d-none d-md-inline">Back to notifications</title</span>
      ', - 'title', - ); - - $examples['Comment opener inside a textarea tag should be ignored'] = array( - '
      -->', - 'textarea', - ); - - $examples['Textarea closer with another textarea tag in closer attributes'] = array( - '
      ', - 'textarea', - ); - - $examples['Textarea closer with attributes'] = array( - '
      ', - 'textarea', - ); - - $examples['Textarea opener with title closer inside'] = array( - '
      ', - 'textarea', - ); - return $examples; - } - - /** - * @ticket 56299 - * - * @covers next_tag - * @covers set_attribute - * @covers __toString - */ - public function test_can_query_and_update_wrongly_nested_tags() { - $p = new WP_HTML_Tag_Processor( - '123

      456789

      ' - ); - $p->next_tag( 'span' ); - $p->set_attribute( 'class', 'span-class' ); - $p->next_tag( 'p' ); - $p->set_attribute( 'class', 'p-class' ); - $this->assertSame( - '123

      456789

      ', - (string) $p - ); - } - - /** - * @ticket 56299 - * - * @covers next_tag - * @covers remove_attribute - * @covers __toString - */ - public function test_removing_attributes_works_even_in_malformed_html() { - $p = new WP_HTML_Tag_Processor( self::HTML_MALFORMED ); - $p->next_tag( 'span' ); - $p->remove_attribute( 'Notifications<' ); - $this->assertSame( - '
      Back to notifications
      ', - (string) $p - ); - } - - /** - * @ticket 56299 - * - * @covers next_Tag - * @covers set_attribute - * @covers __toString - */ - public function test_updating_attributes_works_even_in_malformed_html_1() { - $p = new WP_HTML_Tag_Processor( self::HTML_MALFORMED ); - $p->next_tag( 'span' ); - $p->set_attribute( 'id', 'first' ); - $p->next_tag( 'span' ); - $p->set_attribute( 'id', 'second' ); - $this->assertSame( - '
      Back to notifications
      ', - (string) $p - ); - } - - /** - * @ticket 56299 - * - * @covers next_tag - * @covers set_attribute - * @covers add_class - * @covers __toString - * - * @dataProvider data_malformed_tag - */ - public function test_updating_attributes_works_even_in_malformed_html_2( $html_input, $html_expected ) { - $p = new WP_HTML_Tag_Processor( $html_input ); - $p->next_tag(); - $p->set_attribute( 'foo', 'bar' ); - $p->add_class( 'firstTag' ); - $p->next_tag(); - $p->add_class( 'secondTag' ); - $this->assertSame( - $html_expected, - (string) $p - ); - } - - /** - * Data provider for test_updates_when_malformed_tag(). - * - * @return array { - * @type array { - * @type string $html_input The input HTML snippet. - * @type string $html_expected The expected HTML snippet after processing. - * } - * } - */ - public function data_malformed_tag() { - $null_byte = chr( 0 ); - $examples = array(); - $examples['Invalid entity inside attribute value'] = array( - 'test', - 'test', - ); - - $examples['HTML tag opening inside attribute value'] = array( - '
      This <is> a <strong is="true">thing.
      test', - '
      This <is> a <strong is="true">thing.
      test', - ); - - $examples['HTML tag brackets in attribute values and data markup'] = array( - '
      This <is> a <strong is="true">thing.
      test', - '
      This <is> a <strong is="true">thing.
      test', - ); - - $examples['Single and double quotes in attribute value'] = array( - '

      test', - '

      test', - ); - - $examples['Unquoted attribute values'] = array( - '


      test', - '
      test', - ); - - $examples['Double-quotes escaped in double-quote attribute value'] = array( - '
      test', - '
      test', - ); - - $examples['Unquoted attribute value'] = array( - '
      test', - '
      test', - ); - - $examples['Unquoted attribute value with tag-like value'] = array( - '
      >test', - '
      >test', - ); - - $examples['Unquoted attribute value with tag-like value followed by tag-like data'] = array( - '
      >test', - '
      >test', - ); - - $examples['1'] = array( - '
      test', - '
      test', - ); - - $examples['2'] = array( - '
      test', - '
      test', - ); - - $examples['4'] = array( - '
      test', - '
      test', - ); - - $examples['5'] = array( - '
      code>test', - '
      code>test', - ); - - $examples['6'] = array( - '
      test', - '
      test', - ); - - $examples['7'] = array( - '
      test', - '
      test', - ); - - $examples['8'] = array( - '
      id="test">test', - '
      id="test">test', - ); - - $examples['9'] = array( - '
      test', - '
      test', - ); - - $examples['10'] = array( - 'test', - 'test', - ); - - $examples['11'] = array( - 'The applicative operator <* works well in Haskell; is what?test', - 'The applicative operator <* works well in Haskell; is what?test', - ); - - $examples['12'] = array( - '<3 is a heart but is a tag.test', - '<3 is a heart but is a tag.test', - ); - - $examples['13'] = array( - 'test', - 'test', - ); - - $examples['14'] = array( - 'test', - 'test', - ); - - $examples['15'] = array( - ' a HTML Tag]]>test', - ' a HTML Tag]]>test', - ); - - $examples['16'] = array( - '
      test', - '
      test', - ); - - $examples['17'] = array( - '
      test', - '
      test', - ); - - $examples['18'] = array( - '
      test', - '
      test', - ); - - $examples['19'] = array( - '
      test', - '
      test', - ); - - $examples['20'] = array( - '
      test', - '
      test', - ); - - $examples['21'] = array( - '
      test', - '
      test', - ); - - $examples['22'] = array( - '
      test', - '
      test', - ); - - $examples['23'] = array( - '
      test', - '
      test', - ); - - $examples['24'] = array( - '
      test', - '
      test', - ); - - $examples['25'] = array( - '
      test', - '
      test', - ); - - $examples['Multiple unclosed tags treated as a single tag'] = array( - '
      -test', - '
      -test', - ); - - $examples['27'] = array( - '
      test', - '
      test', - ); - - $examples['28'] = array( - '
      test', - '
      test', - ); - - return $examples; - } -} diff --git a/phpunit/html/wp-html-tag-processor-bookmark-test.php b/phpunit/html/wp-html-tag-processor-bookmark-test.php new file mode 100644 index 0000000000000..e1c4b005ce47c --- /dev/null +++ b/phpunit/html/wp-html-tag-processor-bookmark-test.php @@ -0,0 +1,370 @@ +
    • One
    • Two
    • Three
    • ' ); + $p->next_tag( 'li' ); + $this->assertTrue( $p->set_bookmark( 'first li' ), 'Could not allocate a "first li" bookmark.' ); + $p->next_tag( 'li' ); + $this->assertTrue( $p->set_bookmark( 'second li' ), 'Could not allocate a "second li" bookmark.' ); + $this->assertTrue( $p->set_bookmark( 'first li' ), 'Could not move the "first li" bookmark.' ); + } + + /** + * @ticket 56299 + * + * @covers release_bookmark + */ + public function test_release_bookmark() { + $p = new WP_HTML_Tag_Processor( '
      • One
      • Two
      • Three
      ' ); + $p->next_tag( 'li' ); + $this->assertFalse( $p->release_bookmark( 'first li' ), 'Released a non-existing bookmark.' ); + $p->set_bookmark( 'first li' ); + $this->assertTrue( $p->release_bookmark( 'first li' ), 'Could not release a bookmark.' ); + } + + /** + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + */ + public function test_seek() { + $p = new WP_HTML_Tag_Processor( '
      • One
      • Two
      • Three
      ' ); + $p->next_tag( 'li' ); + $p->set_bookmark( 'first li' ); + + $p->next_tag( 'li' ); + $p->set_attribute( 'foo-2', 'bar-2' ); + + $p->seek( 'first li' ); + $p->set_attribute( 'foo-1', 'bar-1' ); + + $this->assertEquals( + '
      • One
      • Two
      • Three
      ', + $p->get_updated_html() + ); + } + + /** + * WP_HTML_Tag_Processor used to test for the diffs affecting + * the adjusted bookmark position while simultaneously adjusting + * the bookmark in question. As a result, updating the bookmarks + * of a next tag while removing two subsequent attributes in + * a previous tag unfolded like this: + * + * 1. Check if the first removed attribute is before the bookmark: + * + * + * ^-------------------^ ^ + * diff applied here the bookmark is here + * + * (Yes it is) + * + * 2. Move the bookmark to the left by the attribute length: + * + * + * ^ + * the bookmark is here + * + * 3. Check if the second removed attribute is before the bookmark: + * + * + * ^ ^-----^ + * bookmark diff + * + * This time, it isn't! + * + * The fix in the WP_HTML_Tag_Processor involves doing all the checks + * before moving the bookmark. This test is here to guard us from + * the erroneous behavior accidentally returning one day. + * + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + * @covers apply_attributes_updates + */ + public function test_removing_long_attributes_doesnt_break_seek() { + $input = << +HTML; + $p = new WP_HTML_Tag_Processor( $input ); + $p->next_tag( 'button' ); + $p->set_bookmark( 'first' ); + $p->next_tag( 'button' ); + $p->set_bookmark( 'second' ); + + $this->assertTrue( + $p->seek( 'first' ), + 'Seek() to the first button has failed' + ); + $p->remove_attribute( 'twenty_one_characters' ); + $p->remove_attribute( '7_chars' ); + + $this->assertTrue( + $p->seek( 'second' ), + 'Seek() to the second button has failed' + ); + } + + /** + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + */ + public function test_bookmarks_complex_use_case() { + $input = << +
      +
      +
      + + + + + + + +
      +
      +
      +HTML; + $expected_output = << +
      +
      +
      + + + + + + + +
      +
      +
      +HTML; + $p = new WP_HTML_Tag_Processor( $input ); + $p->next_tag( 'div' ); + $p->next_tag( 'div' ); + $p->next_tag( 'div' ); + $p->set_bookmark( 'first div' ); + $p->next_tag( 'button' ); + $p->set_bookmark( 'first button' ); + $p->next_tag( 'button' ); + $p->set_bookmark( 'second button' ); + $p->next_tag( 'button' ); + $p->set_bookmark( 'third button' ); + $p->next_tag( 'button' ); + $p->set_bookmark( 'fourth button' ); + + $p->seek( 'first button' ); + $p->set_attribute( 'type', 'submit' ); + + $this->assertTrue( + $p->seek( 'third button' ), + 'Seek() to the third button failed' + ); + $p->remove_attribute( 'class' ); + $p->remove_attribute( 'type' ); + $p->remove_attribute( 'aria-expanded' ); + $p->set_attribute( 'id', 'rebase-and-merge' ); + $p->remove_attribute( 'data-details-container' ); + + $this->assertTrue( + $p->seek( 'first div' ), + 'Seek() to the first div failed' + ); + $p->set_attribute( 'checked', false ); + + $this->assertTrue( + $p->seek( 'fourth button' ), + 'Seek() to the fourth button failed' + ); + $p->set_attribute( 'id', 'last-button' ); + $p->remove_attribute( 'class' ); + $p->remove_attribute( 'type' ); + $p->remove_attribute( 'checked' ); + $p->remove_attribute( 'aria-label' ); + $p->remove_attribute( 'disabled' ); + $p->remove_attribute( 'data-view-component' ); + + $this->assertTrue( + $p->seek( 'second button' ), + 'Seek() to the second button failed' + ); + $p->remove_attribute( 'type' ); + $p->set_attribute( 'class', 'hx_create-pr-button' ); + + $this->assertEquals( + $expected_output, + $p->get_updated_html() + ); + } + + /** + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + */ + public function test_updates_bookmark_for_additions_after_both_sides() { + $p = new WP_HTML_Tag_Processor( '
      First
      Second
      ' ); + $p->next_tag(); + $p->set_bookmark( 'first' ); + $p->next_tag(); + $p->add_class( 'second' ); + + $p->seek( 'first' ); + $p->add_class( 'first' ); + + $this->assertEquals( + '
      First
      Second
      ', + $p->get_updated_html() + ); + } + + /** + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + */ + public function test_updates_bookmark_for_additions_before_both_sides() { + $p = new WP_HTML_Tag_Processor( '
      First
      Second
      ' ); + $p->next_tag(); + $p->set_bookmark( 'first' ); + $p->next_tag(); + $p->set_bookmark( 'second' ); + + $p->seek( 'first' ); + $p->add_class( 'first' ); + + $p->seek( 'second' ); + $p->add_class( 'second' ); + + $this->assertEquals( + '
      First
      Second
      ', + $p->get_updated_html() + ); + } + + /** + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + */ + public function test_updates_bookmark_for_deletions_after_both_sides() { + $p = new WP_HTML_Tag_Processor( '
      First
      Second
      ' ); + $p->next_tag(); + $p->set_bookmark( 'first' ); + $p->next_tag(); + $p->remove_attribute( 'disabled' ); + + $p->seek( 'first' ); + $p->set_attribute( 'untouched', true ); + + $this->assertEquals( + /** @TODO: we shouldn't have to assert the extra space after removing the attribute. */ + '
      First
      Second
      ', + $p->get_updated_html() + ); + } + + /** + * @ticket 56299 + * + * @covers seek + * @covers set_bookmark + */ + public function test_updates_bookmark_for_deletions_before_both_sides() { + $p = new WP_HTML_Tag_Processor( '
      First
      Second
      ' ); + $p->next_tag(); + $p->set_bookmark( 'first' ); + $p->next_tag(); + $p->set_bookmark( 'second' ); + + $p->seek( 'first' ); + $p->remove_attribute( 'disabled' ); + + $p->seek( 'second' ); + $p->set_attribute( 'safe', true ); + + $this->assertEquals( + /** @TODO: we shouldn't have to assert the extra space after removing the attribute. */ + '
      First
      Second
      ', + $p->get_updated_html() + ); + } + + /** + * @ticket 56299 + * + * @covers set_bookmark + */ + public function test_limits_the_number_of_bookmarks() { + $p = new WP_HTML_Tag_Processor( '
      • One
      • Two
      • Three
      ' ); + $p->next_tag( 'li' ); + + $this->expectException( Exception::class ); + + for ( $i = 0;$i < WP_HTML_Tag_Processor::MAX_BOOKMARKS;$i++ ) { + $this->assertTrue( $p->set_bookmark( "bookmark $i" ), "Could not allocate the bookmark #$i" ); + } + + $this->assertFalse( $p->set_bookmark( 'final bookmark' ), "Allocated $i bookmarks, which is one above the limit." ); + } + + /** + * @ticket 56299 + * + * @covers seek + */ + public function test_limits_the_number_of_seek_calls() { + $p = new WP_HTML_Tag_Processor( '
      • One
      • Two
      • Three
      ' ); + $p->next_tag( 'li' ); + $p->set_bookmark( 'bookmark' ); + + $this->expectException( Exception::class ); + + for ( $i = 0; $i < WP_HTML_Tag_Processor::MAX_SEEK_OPS; $i++ ) { + $this->assertTrue( $p->seek( 'bookmark' ), 'Could not seek to the "bookmark"' ); + } + $this->assertFalse( $p->seek( 'bookmark' ), "$i-th seek() to the bookmark succeeded, even though it should exceed the allowed limit." ); + } +} diff --git a/phpunit/html/wp-html-tag-processor-standalone-test.php b/phpunit/html/wp-html-tag-processor-test.php similarity index 94% rename from phpunit/html/wp-html-tag-processor-standalone-test.php rename to phpunit/html/wp-html-tag-processor-test.php index 8079db28f52be..6a850eb7a4edb 100644 --- a/phpunit/html/wp-html-tag-processor-standalone-test.php +++ b/phpunit/html/wp-html-tag-processor-test.php @@ -1,30 +1,11 @@ Text
      '; const HTML_WITH_CLASSES = '
      Text
      '; const HTML_MALFORMED = '
      Back to notifications
      '; @@ -256,6 +237,35 @@ public function test_next_tag_should_return_false_for_a_non_existing_tag() { $this->assertFalse( $p->next_tag( 'p' ), 'Querying a non-existing tag did not return false' ); } + /** + * @covers next_tag + * @covers is_tag_closer + */ + public function test_next_tag_should_stop_on_closers_only_when_requested() { + $p = new WP_HTML_Tag_Processor( '
      ' ); + $this->assertTrue( $p->next_tag( array( 'tag_name' => 'div' ) ), 'Did not find desired tag opener' ); + $this->assertFalse( $p->next_tag( array( 'tag_name' => 'div' ) ), 'Visited an unwanted tag, a tag closer' ); + + $p = new WP_HTML_Tag_Processor( '
      ' ); + $p->next_tag( + array( + 'tag_name' => 'div', + 'tag_closers' => 'visit', + ) + ); + $this->assertFalse( $p->is_tag_closer(), 'Indicated a tag opener is a tag closer' ); + $this->assertTrue( + $p->next_tag( + array( + 'tag_name' => 'div', + 'tag_closers' => 'visit', + ) + ), + 'Did not stop at desired tag closer' + ); + $this->assertTrue( $p->is_tag_closer(), 'Indicated a tag closer is a tag opener' ); + } + /** * @ticket 56299 * @@ -274,6 +284,33 @@ public function test_set_attribute_on_a_non_existing_tag_does_not_change_the_mar ); } + public function test_attribute_ops_on_tag_closer_do_not_change_the_markup() { + $p = new WP_HTML_Tag_Processor( '
      ' ); + $p->next_tag( + array( + 'tag_name' => 'div', + 'tag_closers' => 'visit', + ) + ); + $this->assertFalse( $p->is_tag_closer(), 'Skipped tag opener' ); + $p->next_tag( + array( + 'tag_name' => 'div', + 'tag_closers' => 'visit', + ) + ); + $this->assertTrue( $p->is_tag_closer(), 'Skipped tag closer' ); + $this->assertFalse( $p->set_attribute( 'id', 'test' ), "Allowed setting an attribute on a tag closer when it shouldn't have" ); + $this->assertFalse( $p->remove_attribute( 'invalid-id' ), "Allowed removing an attribute on a tag closer when it shouldn't have" ); + $this->assertFalse( $p->add_class( 'sneaky' ), "Allowed adding a class on a tag closer when it shouldn't have" ); + $this->assertFalse( $p->remove_class( 'not-appearing-in-this-test' ), "Allowed removing a class on a tag closer when it shouldn't have" ); + $this->assertSame( + '
      ', + $p->get_updated_html(), + 'Calling get_updated_html after updating a non-existing tag returned an HTML that was different from the original HTML' + ); + } + /** * Passing a double quote inside of an attribute values could lead to an XSS attack as follows: * @@ -951,7 +988,7 @@ public function data_script_state() { public function test_next_tag_ignores_the_contents_of_a_rcdata_tag( $rcdata_then_div, $rcdata_tag ) { $p = new WP_HTML_Tag_Processor( $rcdata_then_div ); $p->next_tag(); - $this->assertSame( $rcdata_tag, $p->get_tag(), "The first found tag was not '$rcdata_tag'" ); + $this->assertSame( strtoupper( $rcdata_tag ), $p->get_tag(), "The first found tag was not '$rcdata_tag'" ); $p->next_tag(); $this->assertSame( 'DIV', $p->get_tag(), "The second found tag was not 'div'" ); } @@ -1259,18 +1296,24 @@ public function data_malformed_tag() { ); $examples['Multiple unclosed tags treated as a single tag'] = array( - '
      -test', - '
      -test', + << + test +HTML + , + << + test +HTML + , ); $examples['27'] = array( diff --git a/phpunit/html/wp-html-tag-processor-wp-test.php b/phpunit/html/wp-html-tag-processor-wp-test.php deleted file mode 100644 index 41008800d0b75..0000000000000 --- a/phpunit/html/wp-html-tag-processor-wp-test.php +++ /dev/null @@ -1,91 +0,0 @@ - - * $p = new WP_HTML_Tag_Processor( '
      ' ); - * $p->next_tag(); - * $p->set_attribute('class', '" onclick="alert'); - * echo $p; - * //
      - * - * - * To prevent it, `set_attribute` calls `esc_attr()` on its given values. - * - * - *
      - *
      - * - * @ticket 56299 - * - * @dataProvider data_set_attribute_escapable_values - * @covers set_attribute - */ - public function test_set_attribute_prevents_xss( $value_to_set, $expected_result ) { - $p = new WP_HTML_Tag_Processor( '
      ' ); - $p->next_tag(); - $p->set_attribute( 'test', $value_to_set ); - - /* - * Testing the escaping is hard using tools that properly parse - * HTML because they might interpret the escaped values. It's hard - * with tools that don't understand HTML because they might get - * confused by improperly-escaped values. - * - * For this test, since we control the input HTML we're going to - * do what looks like the opposite of what we want to be doing with - * this library but are only doing so because we have full control - * over the content and because we want to look at the raw values. - */ - $match = null; - preg_match( '~^
      $~', $p->get_updated_html(), $match ); - list( , $actual_value ) = $match; - - $this->assertEquals( $actual_value, '"' . $expected_result . '"' ); - } - - /** - * Data provider with HTML attribute values that might need escaping. - */ - public function data_set_attribute_escapable_values() { - return array( - array( '"', '"' ), - array( '"', '"' ), - array( '&', '&' ), - array( '&', '&' ), - array( '€', '€' ), - array( "'", ''' ), - array( '<>', '<>' ), - array( '"";', '&quot";' ), - array( - '" onclick="alert(\'1\');">', - '" onclick="alert('1');"><span onclick=""></span><script>alert("1")</script>', - ), - ); - } - -} diff --git a/phpunit/style-engine/class-wp-style-engine-css-declarations-test.php b/phpunit/style-engine/class-wp-style-engine-css-declarations-test.php index 5380cc81d5831..011de140d0892 100644 --- a/phpunit/style-engine/class-wp-style-engine-css-declarations-test.php +++ b/phpunit/style-engine/class-wp-style-engine-css-declarations-test.php @@ -9,6 +9,7 @@ /** * Tests registering, storing and generating CSS declarations. * + * @group style-engine * @coversDefaultClass WP_Style_Engine_CSS_Declarations_Gutenberg */ class WP_Style_Engine_CSS_Declarations_Test extends WP_UnitTestCase { diff --git a/phpunit/style-engine/class-wp-style-engine-css-rule-test.php b/phpunit/style-engine/class-wp-style-engine-css-rule-test.php index 26ea41c7ce830..444c83f150f91 100644 --- a/phpunit/style-engine/class-wp-style-engine-css-rule-test.php +++ b/phpunit/style-engine/class-wp-style-engine-css-rule-test.php @@ -9,6 +9,7 @@ /** * Tests for registering, storing and generating CSS rules. * + * @group style-engine * @coversDefaultClass WP_Style_Engine_CSS_Rule_Gutenberg */ class WP_Style_Engine_CSS_Rule_Test extends WP_UnitTestCase { @@ -141,4 +142,31 @@ public function test_should_prettify_css_rule_output() { $this->assertSame( $expected, $css_rule->get_css( true ) ); } + + /** + * Tests that a string of multiple selectors is trimmed. + * + * @covers ::get_css + */ + public function test_should_trim_multiple_selectors() { + $selector = '.poirot, .poirot:active, #miss-marple > .st-mary-mead '; + $input_declarations = array( + 'margin-left' => '0', + 'font-family' => 'Detective Sans', + ); + $css_declarations = new WP_Style_Engine_CSS_Declarations_Gutenberg( $input_declarations ); + $css_rule = new WP_Style_Engine_CSS_Rule_Gutenberg( $selector, $css_declarations ); + $expected = '.poirot, .poirot:active, #miss-marple > .st-mary-mead {margin-left:0;font-family:Detective Sans;}'; + + $this->assertSame( $expected, $css_rule->get_css(), 'Return value should be not prettified.' ); + + $expected_prettified = '.poirot, +.poirot:active, +#miss-marple > .st-mary-mead { + margin-left: 0; + font-family: Detective Sans; +}'; + + $this->assertSame( $expected_prettified, $css_rule->get_css( true ), 'Return value should be prettified with new lines and indents.' ); + } } diff --git a/phpunit/style-engine/class-wp-style-engine-css-rules-store-test.php b/phpunit/style-engine/class-wp-style-engine-css-rules-store-test.php index 700f63c556c25..8529bff78e22c 100644 --- a/phpunit/style-engine/class-wp-style-engine-css-rules-store-test.php +++ b/phpunit/style-engine/class-wp-style-engine-css-rules-store-test.php @@ -9,6 +9,7 @@ /** * Tests for registering, storing and retrieving a collection of CSS Rules (a store). * + * @group style-engine * @coversDefaultClass WP_Style_Engine_CSS_Rules_Store_Gutenberg */ class WP_Style_Engine_CSS_Rules_Store_Test extends WP_UnitTestCase { diff --git a/phpunit/style-engine/class-wp-style-engine-processor-test.php b/phpunit/style-engine/class-wp-style-engine-processor-test.php index cfbde08704bdc..18392f5156fcc 100644 --- a/phpunit/style-engine/class-wp-style-engine-processor-test.php +++ b/phpunit/style-engine/class-wp-style-engine-processor-test.php @@ -9,6 +9,7 @@ /** * Tests for compiling and rendering styles from a store of CSS rules. * + * @group style-engine * @coversDefaultClass WP_Style_Engine_Processor_Gutenberg */ class WP_Style_Engine_Processor_Test extends WP_UnitTestCase { diff --git a/phpunit/style-engine/style-engine-test.php b/phpunit/style-engine/style-engine-test.php index 5f4c57453e8a8..66d8dd6286527 100644 --- a/phpunit/style-engine/style-engine-test.php +++ b/phpunit/style-engine/style-engine-test.php @@ -8,6 +8,8 @@ /** * Tests for registering, storing and generating styles. + * + * @group style-engine */ class WP_Style_Engine_Test extends WP_UnitTestCase { /** diff --git a/schemas/json/block.json b/schemas/json/block.json index 93ed4e5a12b94..3a3726010d0b6 100644 --- a/schemas/json/block.json +++ b/schemas/json/block.json @@ -160,6 +160,7 @@ "attribute", "text", "html", + "raw", "query", "meta" ] @@ -271,7 +272,7 @@ }, "link": { "type": "boolean", - "description": "This property adds block controls which allow the user to set link color in a block, link color is disabled by default.\n\nLink color presets are sourced from the editor-color-palette theme support.\n\nWhen the block declares support for color.link, the attributes definition is extended to include two new attributes: linkColor and style", + "description": "This property adds block controls which allow the user to set link color in a block, link color is disabled by default.\n\nLink color presets are sourced from the editor-color-palette theme support.\n\nWhen the block declares support for color.link, the attributes definition is extended to include the style attribute", "default": false }, "text": { diff --git a/schemas/json/theme.json b/schemas/json/theme.json index d42706bec8df9..1a8aa67b0967b 100644 --- a/schemas/json/theme.json +++ b/schemas/json/theme.json @@ -1169,6 +1169,10 @@ } }, "additionalProperties": false + }, + "css": { + "description": "Sets custom CSS to apply styling not covered by other theme.json properties.", + "type": "string" } } }, @@ -1186,7 +1190,8 @@ "typography": {}, "filter": {}, "shadow": {}, - "outline": {} + "outline": {}, + "css": {} }, "additionalProperties": false } @@ -1210,6 +1215,7 @@ "outline": {}, "spacing": {}, "typography": {}, + "css": {}, ":hover": { "$ref": "#/definitions/stylesPropertiesComplete" }, @@ -1561,6 +1567,7 @@ "filter": {}, "shadow": {}, "outline": {}, + "css": {}, "elements": { "$ref": "#/definitions/stylesElementsPropertiesComplete" } @@ -1629,6 +1636,7 @@ "filter": {}, "shadow": {}, "outline": {}, + "css": {}, "elements": { "description": "Styles defined on a per-element basis using the element's selector.", "$ref": "#/definitions/stylesElementsPropertiesComplete" diff --git a/storybook/decorators/with-theme.js b/storybook/decorators/with-theme.js index 4c0696f7bfe72..9f11dc63dbaed 100644 --- a/storybook/decorators/with-theme.js +++ b/storybook/decorators/with-theme.js @@ -1,28 +1,93 @@ +/** + * External dependencies + */ +import styled from '@emotion/styled'; +import { css } from '@emotion/react'; + /** * Internal dependencies */ import Theme from '../../packages/components/src/theme'; +const themes = { + default: {}, + darkBg: { + accent: '#f7c849', + background: '#1e1e1e', + }, + lightGrayBg: { + accent: '#3858e9', + background: '#f0f0f0', + }, + modern: { + accent: '#3858e9', + }, +}; + +const backgroundStyles = ( { background } ) => { + if ( background ) { + return css` + background: ${ background }; + padding: 20px 20px 8px; + outline: 1px dashed #ccc; + outline-offset: 2px; + `; + } +}; + +const BackgroundColorWrapper = styled.div` + ${ backgroundStyles } +`; + +const BackgroundIndicator = styled.small` + display: block; + opacity: 0.3; + margin-top: 20px; + font-size: 10px; + color: var( --wp-components-color-foreground ); + text-transform: uppercase; + text-align: end; +`; + +const Notice = styled.small` + display: block; + margin-top: 20px; + font-size: 12px; + color: var( --wp-components-color-foreground ); +`; + /** * A Storybook decorator to show a div before and after the story to check for unwanted margins. */ export const WithTheme = ( Story, context ) => { - const themes = { - default: {}, - modern: { - accent: '#3858e9', - }, - sunrise: { - // This color was chosen intentionally, because for sufficient contrast, - // the foreground text should be black when this orange is used as a background color. - accent: '#dd823b', - }, - }; + const selectedTheme = themes[ context.globals.componentsTheme ]; + const selectedBackground = selectedTheme.background; + + if ( context.componentId === 'components-experimental-theme' ) { + return ( + <> + + { context.globals.componentsTheme !== 'default' && ( + + The Theme toolbar addon is disabled for this story. Use + Controls to change the values. + + ) } + + ); + } return ( - - - + + + + { selectedBackground && ( + + Themed background { selectedBackground } + + ) } + + ); }; diff --git a/storybook/main.js b/storybook/main.js index 99c8c2ea84bec..29880d6455700 100644 --- a/storybook/main.js +++ b/storybook/main.js @@ -19,15 +19,17 @@ module.exports = { }, '@storybook/addon-controls', '@storybook/addon-knobs', // Deprecated, new stories should use addon-controls. - '@storybook/addon-storysource', '@storybook/addon-viewport', '@storybook/addon-a11y', '@storybook/addon-toolbars', '@storybook/addon-actions', + 'storybook-source-link', ], + framework: '@storybook/react', features: { babelModeV7: true, emotionAlias: false, + storyStoreV7: true, }, // Workaround: // https://github.com/storybookjs/storybook/issues/12270 diff --git a/storybook/preview.js b/storybook/preview.js index 16e8f71cf5346..01677a768ffb5 100644 --- a/storybook/preview.js +++ b/storybook/preview.js @@ -29,8 +29,9 @@ export const globalTypes = { icon: 'paintbrush', items: [ { value: 'default', title: 'Default' }, - { value: 'modern', title: 'Modern' }, - { value: 'sunrise', title: 'Sunrise' }, + { value: 'darkBg', title: 'Dark (background)' }, + { value: 'lightGrayBg', title: 'Light gray (background)' }, + { value: 'modern', title: 'Modern (accent)' }, ], }, }, @@ -79,11 +80,11 @@ export const globalTypes = { }; export const decorators = [ - WithTheme, WithGlobalCSS, WithMarginChecker, WithRTL, WithMaxWidthWrapper, + WithTheme, ]; export const parameters = { @@ -107,4 +108,5 @@ export const parameters = { ], }, }, + sourceLinkPrefix: 'https://github.com/WordPress/gutenberg/blob/trunk/', }; diff --git a/storybook/stories/docs/inline-icon.js b/storybook/stories/docs/inline-icon.js new file mode 100644 index 0000000000000..2a1056b0069b7 --- /dev/null +++ b/storybook/stories/docs/inline-icon.js @@ -0,0 +1,14 @@ +/** + * External dependencies + */ +import styled from '@emotion/styled'; +import { Icons } from '@storybook/components'; + +const StyledIcons = styled( Icons )` + display: inline-block !important; + width: 14px; +`; + +export const InlineIcon = ( props ) => ( + +); diff --git a/storybook/stories/docs/introduction.story.mdx b/storybook/stories/docs/introduction.story.mdx index bdad5c0336ebf..f353cf7c7e0ab 100644 --- a/storybook/stories/docs/introduction.story.mdx +++ b/storybook/stories/docs/introduction.story.mdx @@ -1,4 +1,5 @@ import { Meta } from '@storybook/addon-docs'; +import { InlineIcon } from './inline-icon'; @@ -17,24 +18,25 @@ Import them from the components root directory like in below example: import { Button } from '@wordpress/components'; export default function MyButton() { - return ; + return ; } -```` +``` ## How this site works -The site shows the individual components in the sidebar and the Canvas on the right. Select the component you’d like to explore, and you’ll see the display on the Canvas tab. If the component also has controls/arguments, you can modify them on the Controls tab on the lower half of the screen. +The site shows the individual components in the sidebar and the Canvas on the right. Select the component you’d like to explore, and you’ll see the display on the **Canvas** tab. If the component also has controls/arguments, you can modify them on the **Controls** tab on the lower half of the screen. To view the documentation for each component use the **Docs** menu item in the top toolbar. +To view the source code for the component and its stories on GitHub, click the View Source Repository button in the top right corner. + To use it in your local development environment run the following command in the top level Gutenberg directory: - ```bash - npm run storybook:dev - ``` +```bash +npm run storybook:dev +``` ## Resources to learn more: -- [Storybook.js.org](https://storybook.js.org/) - Storybook is a frontend workshop for building UI components and pages in isolation. -- [[Package] Components](https://github.com/WordPress/gutenberg/issues?q=is%3Aopen+is%3Aissue+label%3A%22%5BPackage%5D+Components%22) - Open Issue Gutenberg Repo -- [On the known "loading source..." issue](https://github.com/WordPress/gutenberg/issues/45095) at the 'Story' tab for some components +- [Storybook.js.org](https://storybook.js.org/) - Storybook is a frontend workshop for building UI components and pages in isolation. +- [[Package] Components](https://github.com/WordPress/gutenberg/issues?q=is%3Aopen+is%3Aissue+label%3A%22%5BPackage%5D+Components%22) - Open Issue Gutenberg Repo diff --git a/storybook/stories/playground/index.js b/storybook/stories/playground/index.js index a13001f96f28b..92ccd78ae58b7 100644 --- a/storybook/stories/playground/index.js +++ b/storybook/stories/playground/index.js @@ -70,6 +70,9 @@ function App() { export default { title: 'Playground/Block Editor', + parameters: { + sourceLink: 'storybook/stories/playground', + }, }; export const _default = () => { diff --git a/storybook/webpack.config.js b/storybook/webpack.config.js index 000c6930574f9..1747cc2f9caa0 100644 --- a/storybook/webpack.config.js +++ b/storybook/webpack.config.js @@ -44,6 +44,15 @@ module.exports = ( { config } ) => { ), enforce: 'post', }, + { + // Adds a `sourceLink` parameter to the story metadata, based on the file path + test: /\/stories\/.+\.(j|t)sx?$/, + loader: path.resolve( + __dirname, + './webpack/source-link-loader.js' + ), + enforce: 'post', + }, { test: /\.scss$/, exclude: /\.lazy\.scss$/, diff --git a/storybook/webpack/source-link-loader.js b/storybook/webpack/source-link-loader.js new file mode 100644 index 0000000000000..d9501aa737dad --- /dev/null +++ b/storybook/webpack/source-link-loader.js @@ -0,0 +1,67 @@ +/** + * External dependencies + */ +const babel = require( '@babel/core' ); +const path = require( 'path' ); + +const REPO_ROOT = path.resolve( __dirname, '../../' ); + +/** + * Adds a `sourceLink` parameter to the story metadata, based on the file path. + * + * @see https://storybook.js.org/addons/storybook-source-link + */ +function addSourceLinkPlugin() { + return { + visitor: { + ExportDefaultDeclaration( visitorPath, state ) { + const componentPath = getComponentPathFromStoryPath( + state.file.opts.filename + ); + const properties = + // When default export is anonymous, the declaration is an object expression + visitorPath.node.declaration.properties ?? + // When default export is named, the declaration is an identifier, usually the previous node + visitorPath.getPrevSibling().node.declarations[ 0 ].init + .properties; + + alterParameters( properties, componentPath ); + }, + }, + }; +} + +function getComponentPathFromStoryPath( storyPath ) { + const componentRoot = path.resolve( storyPath, '../../' ); + return path.relative( REPO_ROOT, componentRoot ); +} + +function alterParameters( properties, componentPath ) { + const sourceLink = babel.types.objectProperty( + babel.types.identifier( 'sourceLink' ), + babel.types.stringLiteral( componentPath ) + ); + + let parameters = properties.find( ( op ) => op.key.name === 'parameters' ); + + if ( ! parameters ) { + parameters = babel.types.objectProperty( + babel.types.identifier( 'parameters' ) + ); + properties.push( parameters ); + } + + parameters.value.properties = [ + sourceLink, + ...parameters.value.properties, + ]; +} + +module.exports = function ( source, { sources } ) { + const output = babel.transform( source, { + plugins: [ addSourceLinkPlugin ], + filename: sources[ 0 ], + sourceType: 'module', + } ); + return output.code; +}; diff --git a/test/e2e/specs/editor/blocks/buttons.spec.js b/test/e2e/specs/editor/blocks/buttons.spec.js index ece3d06a5e9e8..d88f042ebdf9f 100644 --- a/test/e2e/specs/editor/blocks/buttons.spec.js +++ b/test/e2e/specs/editor/blocks/buttons.spec.js @@ -123,6 +123,136 @@ test.describe( 'Buttons', () => { +` + ); + } ); + + test( 'can resize width', async ( { editor, page } ) => { + await editor.insertBlock( { name: 'core/buttons' } ); + await page.keyboard.type( 'Content' ); + await editor.openDocumentSettingsSidebar(); + await page.click( + 'role=group[name="Button width"i] >> role=button[name="25%"i]' + ); + + // Check the content. + const content = await editor.getEditedPostContent(); + expect( content ).toBe( + ` + +` + ); + } ); + + test( 'can apply named colors', async ( { editor, page } ) => { + await editor.insertBlock( { name: 'core/buttons' } ); + await page.keyboard.type( 'Content' ); + await editor.openDocumentSettingsSidebar(); + + await page.click( + 'role=region[name="Editor settings"i] >> role=button[name="Text"i]' + ); + await page.click( 'role=button[name="Color: Cyan bluish gray"i]' ); + await page.click( + 'role=region[name="Editor settings"i] >> role=button[name="Background"i]' + ); + await page.click( 'role=button[name="Color: Vivid red"i]' ); + + // Check the content. + const content = await editor.getEditedPostContent(); + expect( content ).toBe( + ` + +` + ); + } ); + + test( 'can apply custom colors', async ( { editor, page } ) => { + await editor.insertBlock( { name: 'core/buttons' } ); + await page.keyboard.type( 'Content' ); + await editor.openDocumentSettingsSidebar(); + + await page.click( + 'role=region[name="Editor settings"i] >> role=button[name="Text"i]' + ); + await page.click( 'role=button[name="Custom color picker."i]' ); + await page.fill( 'role=textbox[name="Hex color"i]', 'ff0000' ); + + await page.click( + 'role=region[name="Editor settings"i] >> role=button[name="Background"i]' + ); + await page.click( 'role=button[name="Custom color picker."i]' ); + await page.fill( 'role=textbox[name="Hex color"i]', '00ff00' ); + + // Check the content. + const content = await editor.getEditedPostContent(); + expect( content ).toBe( + ` + +` + ); + } ); + + test( 'can apply named gradient background color', async ( { + editor, + page, + } ) => { + await editor.insertBlock( { name: 'core/buttons' } ); + await page.keyboard.type( 'Content' ); + await editor.openDocumentSettingsSidebar(); + + await page.click( + 'role=region[name="Editor settings"i] >> role=button[name="Background"i]' + ); + await page.click( 'role=tab[name="Gradient"i]' ); + await page.click( 'role=button[name="Gradient: Purple to yellow"i]' ); + + // Check the content. + const content = await editor.getEditedPostContent(); + expect( content ).toBe( + ` + +` + ); + } ); + + test( 'can apply custom gradient background color', async ( { + editor, + page, + } ) => { + await editor.insertBlock( { name: 'core/buttons' } ); + await page.keyboard.type( 'Content' ); + await editor.openDocumentSettingsSidebar(); + + await page.click( + 'role=region[name="Editor settings"i] >> role=button[name="Background"i]' + ); + await page.click( 'role=tab[name="Gradient"i]' ); + await page.click( + 'role=button[name=/^Gradient control point at position 0% with color code/]' + ); + await page.fill( 'role=textbox[name="Hex color"i]', 'ff0000' ); + await page.keyboard.press( 'Escape' ); + await page.click( + 'role=button[name=/^Gradient control point at position 100% with color code/]' + ); + await page.fill( 'role=textbox[name="Hex color"i]', '00ff00' ); + + // Check the content. + const content = await editor.getEditedPostContent(); + expect( content ).toBe( + ` + ` ); } ); diff --git a/test/e2e/specs/editor/blocks/html.spec.js b/test/e2e/specs/editor/blocks/html.spec.js index dedb9197984fc..77e9d2a9186a7 100644 --- a/test/e2e/specs/editor/blocks/html.spec.js +++ b/test/e2e/specs/editor/blocks/html.spec.js @@ -30,4 +30,20 @@ test.describe( 'HTML block', () => { ` ); } ); + + test( 'should not encode <', async ( { editor, page } ) => { + // Create a Custom HTML block with the slash shortcut. + await page.click( 'role=button[name="Add default block"i]' ); + await page.keyboard.type( '/html' ); + await expect( + page.locator( 'role=option[name="Custom HTML"i][selected]' ) + ).toBeVisible(); + await page.keyboard.press( 'Enter' ); + await page.keyboard.type( '1 < 2' ); + await editor.publishPost(); + await page.reload(); + await expect( + page.locator( '[data-type="core/html"] textarea' ) + ).toBeVisible(); + } ); } ); diff --git a/test/e2e/specs/editor/blocks/list.spec.js b/test/e2e/specs/editor/blocks/list.spec.js index daa23241a4e8e..0a1159f6d5d49 100644 --- a/test/e2e/specs/editor/blocks/list.spec.js +++ b/test/e2e/specs/editor/blocks/list.spec.js @@ -1090,9 +1090,11 @@ test.describe( 'List', () => {

      - -

      2

      -` + +
        +
      • 2
      • +
      +` ); } ); diff --git a/test/e2e/specs/editor/blocks/paragraph.spec.js b/test/e2e/specs/editor/blocks/paragraph.spec.js index 1285955b66734..19c386cce457d 100644 --- a/test/e2e/specs/editor/blocks/paragraph.spec.js +++ b/test/e2e/specs/editor/blocks/paragraph.spec.js @@ -127,7 +127,7 @@ test.describe( 'Paragraph', () => { await expect.poll( editor.getEditedPostContent ) .toBe( ` -

      My Heading

      +

      My Heading

      ` ); } ); @@ -138,7 +138,9 @@ test.describe( 'Paragraph', () => { } ) => { await editor.insertBlock( { name: 'core/paragraph' } ); - await draggingUtils.simulateDraggingHTML( '

      My Heading

      ' ); + await draggingUtils.simulateDraggingHTML( + '

      My Heading

      ' + ); const emptyParagraph = page.locator( '[data-type="core/paragraph"][data-empty="true"]' @@ -153,7 +155,7 @@ test.describe( 'Paragraph', () => { await expect.poll( editor.getEditedPostContent ) .toBe( ` -

      My Heading

      +

      My Heading

      ` ); } ); @@ -193,8 +195,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -205,8 +211,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -217,8 +227,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -229,8 +243,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -307,8 +325,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } { @@ -319,8 +341,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } { @@ -331,8 +357,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } { @@ -343,8 +373,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } } ); @@ -385,8 +419,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -397,8 +435,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -409,8 +451,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( firstBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + firstBlockBox + ) + ) + .toBeTruthy(); } { @@ -421,8 +467,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } { @@ -433,8 +483,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } { @@ -445,8 +499,12 @@ test.describe( 'Paragraph', () => { ); await expect( draggingUtils.dropZone ).toBeVisible(); await expect - .poll( () => draggingUtils.dropZone.boundingBox() ) - .toEqual( secondBlockBox ); + .poll( () => + draggingUtils.confirmValidDropZonePosition( + secondBlockBox + ) + ) + .toBeTruthy(); } } ); } ); @@ -506,4 +564,14 @@ class DraggingUtils { await this.page.mouse.move( 0, 0 ); await this.page.mouse.down(); } + + async confirmValidDropZonePosition( element ) { + // Check that both x and y axis of the dropzone + // have a less than 1 difference with a given target element + const box = await this.dropZone.boundingBox(); + return ( + Math.abs( element.x - box.x ) < 1 && + Math.abs( element.y - box.y ) < 1 + ); + } } diff --git a/test/e2e/specs/editor/plugins/__snapshots__/templates-Using-a-CPT-with-a-predefined-templa-7538a--custom-post-types-with-a-predefined-template-1-chromium.txt b/test/e2e/specs/editor/plugins/__snapshots__/Post-type-templates-Using-a-CPT-with-a-predefi-fffe1--custom-post-types-with-a-predefined-template-1-chromium.txt similarity index 100% rename from test/e2e/specs/editor/plugins/__snapshots__/templates-Using-a-CPT-with-a-predefined-templa-7538a--custom-post-types-with-a-predefined-template-1-chromium.txt rename to test/e2e/specs/editor/plugins/__snapshots__/Post-type-templates-Using-a-CPT-with-a-predefi-fffe1--custom-post-types-with-a-predefined-template-1-chromium.txt diff --git a/test/e2e/specs/editor/plugins/format-api.spec.js b/test/e2e/specs/editor/plugins/format-api.spec.js index a294a2034f055..1b1d3b3d4173f 100644 --- a/test/e2e/specs/editor/plugins/format-api.spec.js +++ b/test/e2e/specs/editor/plugins/format-api.spec.js @@ -34,6 +34,28 @@ test.describe( 'Using Format API', () => { expect( content ).toBe( `

      First paragraph

      +` + ); + } ); + + test( 'should show unknow formatting button', async ( { + editor, + page, + } ) => { + await editor.insertBlock( { + name: 'core/paragraph', + attributes: { content: 'test' }, + } ); + expect( await editor.getEditedPostContent() ).toBe( + ` +

      test

      +` + ); + await page.keyboard.press( 'ArrowRight' ); + await editor.clickBlockToolbarButton( 'Clear Unknown Formatting' ); + expect( await editor.getEditedPostContent() ).toBe( + ` +

      test

      ` ); } ); diff --git a/test/e2e/specs/editor/plugins/templates.spec.js b/test/e2e/specs/editor/plugins/post-type-templates.spec.js similarity index 98% rename from test/e2e/specs/editor/plugins/templates.spec.js rename to test/e2e/specs/editor/plugins/post-type-templates.spec.js index bab5cdd739baf..3c5a664259f2d 100644 --- a/test/e2e/specs/editor/plugins/templates.spec.js +++ b/test/e2e/specs/editor/plugins/post-type-templates.spec.js @@ -3,7 +3,7 @@ */ const { test, expect } = require( '@wordpress/e2e-test-utils-playwright' ); -test.describe( 'templates', () => { +test.describe( 'Post type templates', () => { test.describe( 'Using a CPT with a predefined template', () => { test.beforeAll( async ( { requestUtils } ) => { await requestUtils.activatePlugin( diff --git a/test/e2e/specs/editor/various/__snapshots__/Content-only-lock-should-be-able-to-edit-the-content-of-blocks-with-content-only-lock-1-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/Content-only-lock-should-be-able-to-edit-the-content-of-blocks-with-content-only-lock-1-chromium.txt new file mode 100644 index 0000000000000..18f0839c8dc89 --- /dev/null +++ b/test/e2e/specs/editor/various/__snapshots__/Content-only-lock-should-be-able-to-edit-the-content-of-blocks-with-content-only-lock-1-chromium.txt @@ -0,0 +1,5 @@ + +
      +

      Hello World

      +
      + \ No newline at end of file diff --git a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-1-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-1-chromium.txt index aac1718cdb7ff..beef4f8f37b1f 100644 --- a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-1-chromium.txt +++ b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-1-chromium.txt @@ -1,5 +1,5 @@ -

      Heading

      +

      Heading

      diff --git a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-2-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-2-chromium.txt index ba33f4cbc7c75..4423570da521f 100644 --- a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-2-chromium.txt +++ b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-cut-partial-selection-and-merge-like-a-normal-delete---not-forward-2-chromium.txt @@ -1,5 +1,5 @@ -

      ading

      +

      ading

      @@ -7,5 +7,5 @@ -

      Heph

      +

      Heph

      \ No newline at end of file diff --git a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-1-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-1-chromium.txt index aac1718cdb7ff..beef4f8f37b1f 100644 --- a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-1-chromium.txt +++ b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-1-chromium.txt @@ -1,5 +1,5 @@ -

      Heading

      +

      Heading

      diff --git a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-2-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-2-chromium.txt index e16ec5279cca6..2f13cd0939822 100644 --- a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-2-chromium.txt +++ b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-plain-text-in-plain-text-context-when-cross-block-selection-is-copied-2-chromium.txt @@ -1,5 +1,5 @@ -

      Heading

      +

      Heading

      diff --git a/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-preformatted-in-list-1-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-preformatted-in-list-1-chromium.txt new file mode 100644 index 0000000000000..002e7a1920028 --- /dev/null +++ b/test/e2e/specs/editor/various/__snapshots__/Copy-cut-paste-should-paste-preformatted-in-list-1-chromium.txt @@ -0,0 +1,5 @@ + +
        +
      • xy
      • +
      + \ No newline at end of file diff --git a/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-18edb-roduces-more-than-one-block-on-forward-delete-2-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-18edb-roduces-more-than-one-block-on-forward-delete-2-chromium.txt new file mode 100644 index 0000000000000..463d4ecae0e09 --- /dev/null +++ b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-18edb-roduces-more-than-one-block-on-forward-delete-2-chromium.txt @@ -0,0 +1,9 @@ + +

      hi-item 1

      + + + +
        +
      • item 2
      • +
      + \ No newline at end of file diff --git a/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-2a1ee-roduces-more-than-one-block-on-forward-delete-1-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-2a1ee-roduces-more-than-one-block-on-forward-delete-1-chromium.txt index cb5b5fcf148b3..04346aeb9e959 100644 --- a/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-2a1ee-roduces-more-than-one-block-on-forward-delete-1-chromium.txt +++ b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-2a1ee-roduces-more-than-one-block-on-forward-delete-1-chromium.txt @@ -1,3 +1,13 @@ -

      hi-

      - \ No newline at end of file +

      hi

      + + + +

      item 1

      + + + +
        +
      • item 2
      • +
      + \ No newline at end of file diff --git a/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-46dfa-rge-produces-more-than-one-block-on-backspace-2-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-46dfa-rge-produces-more-than-one-block-on-backspace-2-chromium.txt new file mode 100644 index 0000000000000..463d4ecae0e09 --- /dev/null +++ b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-46dfa-rge-produces-more-than-one-block-on-backspace-2-chromium.txt @@ -0,0 +1,9 @@ + +

      hi-item 1

      + + + +
        +
      • item 2
      • +
      + \ No newline at end of file diff --git a/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-92273-rge-produces-more-than-one-block-on-backspace-1-chromium.txt b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-92273-rge-produces-more-than-one-block-on-backspace-1-chromium.txt index 33adf8bef295a..04346aeb9e959 100644 --- a/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-92273-rge-produces-more-than-one-block-on-backspace-1-chromium.txt +++ b/test/e2e/specs/editor/various/__snapshots__/splitting-and-merging-blocks-test-restore-sele-92273-rge-produces-more-than-one-block-on-backspace-1-chromium.txt @@ -3,9 +3,11 @@ -

      -item 1

      +

      item 1

      - -

      item 2

      - \ No newline at end of file + +
        +
      • item 2
      • +
      + \ No newline at end of file diff --git a/test/e2e/specs/editor/various/autocomplete-and-mentions.spec.js b/test/e2e/specs/editor/various/autocomplete-and-mentions.spec.js index 67270fb2cc357..016699ef9236e 100644 --- a/test/e2e/specs/editor/various/autocomplete-and-mentions.spec.js +++ b/test/e2e/specs/editor/various/autocomplete-and-mentions.spec.js @@ -409,4 +409,32 @@ test.describe( 'Autocomplete', () => {

      @ringbearer +thebetterhobbit

      ` ); } ); + + test( 'should hide UI when selection changes (by keyboard)', async ( { + page, + } ) => { + await page.click( 'role=button[name="Add default block"i]' ); + await page.keyboard.type( '@fr' ); + await expect( + page.locator( 'role=option', { hasText: 'Frodo Baggins' } ) + ).toBeVisible(); + await page.keyboard.press( 'ArrowLeft' ); + await expect( + page.locator( 'role=option', { hasText: 'Frodo Baggins' } ) + ).not.toBeVisible(); + } ); + + test( 'should hide UI when selection changes (by mouse)', async ( { + page, + } ) => { + await page.click( 'role=button[name="Add default block"i]' ); + await page.keyboard.type( '@fr' ); + await expect( + page.locator( 'role=option', { hasText: 'Frodo Baggins' } ) + ).toBeVisible(); + await page.click( '[data-type="core/paragraph"]' ); + await expect( + page.locator( 'role=option', { hasText: 'Frodo Baggins' } ) + ).not.toBeVisible(); + } ); } ); diff --git a/test/e2e/specs/editor/various/content-only-lock.spec.js b/test/e2e/specs/editor/various/content-only-lock.spec.js new file mode 100644 index 0000000000000..41626ed986b8f --- /dev/null +++ b/test/e2e/specs/editor/various/content-only-lock.spec.js @@ -0,0 +1,31 @@ +/** + * WordPress dependencies + */ +const { test, expect } = require( '@wordpress/e2e-test-utils-playwright' ); + +test.describe( 'Content-only lock', () => { + test.beforeEach( async ( { admin } ) => { + await admin.createNewPost(); + } ); + + test( 'should be able to edit the content of blocks with content-only lock', async ( { + editor, + page, + pageUtils, + } ) => { + // Add content only locked block in the code editor + await pageUtils.pressKeyWithModifier( 'secondary', 'M' ); // Emulates CTRL+Shift+Alt + M => toggle code editor + await page.click( '.editor-post-text-editor' ); + await page.keyboard + .type( ` +
      +

      Hello

      +
      + ` ); + await pageUtils.pressKeyWithModifier( 'secondary', 'M' ); + + await page.click( 'role=document[name="Paragraph block"i]' ); + await page.keyboard.type( ' World' ); + expect( await editor.getEditedPostContent() ).toMatchSnapshot(); + } ); +} ); diff --git a/test/e2e/specs/editor/various/copy-cut-paste.spec.js b/test/e2e/specs/editor/various/copy-cut-paste.spec.js index c7041a8634e32..4b7fb267f87b7 100644 --- a/test/e2e/specs/editor/various/copy-cut-paste.spec.js +++ b/test/e2e/specs/editor/various/copy-cut-paste.spec.js @@ -433,4 +433,19 @@ test.describe( 'Copy/cut/paste', () => { await page.evaluate( () => document.activeElement.innerHTML ) ).toBe( 'axyb' ); } ); + + test( 'should paste preformatted in list', async ( { + page, + pageUtils, + editor, + } ) => { + await pageUtils.setClipboardData( { + html: '
      x
      ', + } ); + await editor.insertBlock( { name: 'core/list' } ); + await pageUtils.pressKeyWithModifier( 'primary', 'v' ); + // Ensure the selection is correct. + await page.keyboard.type( 'y' ); + expect( await editor.getEditedPostContent() ).toMatchSnapshot(); + } ); } ); diff --git a/test/e2e/specs/editor/various/inner-blocks-templates.spec.js b/test/e2e/specs/editor/various/inner-blocks-templates.spec.js new file mode 100644 index 0000000000000..87ad260428198 --- /dev/null +++ b/test/e2e/specs/editor/various/inner-blocks-templates.spec.js @@ -0,0 +1,56 @@ +/** + * WordPress dependencies + */ +const { test, expect } = require( '@wordpress/e2e-test-utils-playwright' ); + +test.describe( 'Inner blocks templates', () => { + test.beforeAll( async ( { requestUtils } ) => { + await requestUtils.activatePlugin( + 'gutenberg-test-inner-blocks-templates' + ); + } ); + + test.beforeEach( async ( { admin } ) => { + await admin.createNewPost(); + } ); + + test.afterAll( async ( { requestUtils } ) => { + await requestUtils.deactivatePlugin( + 'gutenberg-test-inner-blocks-templates' + ); + } ); + + test( 'applying block templates asynchronously does not create a persistent change in the editor', async ( { + editor, + page, + } ) => { + await editor.insertBlock( { + name: 'test/test-inner-blocks-async-template', + } ); + + const blockWithTemplateContent = page.locator( + 'role=document[name="Block: Test Inner Blocks Async Template"i] >> text=OneTwo' + ); + + // The block template content appears asynchronously, so wait for it. + await expect( blockWithTemplateContent ).toBeVisible(); + + // Publish the post, then reload. + await editor.publishPost(); + await page.reload(); + + // Wait for the block that was inserted to appear with its templated content. + await expect( blockWithTemplateContent ).toBeVisible(); + + // The template resolution shouldn't cause the post to be dirty. + const editorTopBar = page.locator( + 'role=region[name="Editor top bar"i]' + ); + const undoButton = editorTopBar.locator( 'role=button[name="Undo"i]' ); + const updateButton = editorTopBar.locator( + 'role=button[name="Update"i]' + ); + await expect( undoButton ).toHaveAttribute( 'aria-disabled', 'true' ); + await expect( updateButton ).toHaveAttribute( 'aria-disabled', 'true' ); + } ); +} ); diff --git a/test/e2e/specs/editor/various/inserting-blocks.spec.js b/test/e2e/specs/editor/various/inserting-blocks.spec.js index cc75245d48bac..fdc4615f197aa 100644 --- a/test/e2e/specs/editor/various/inserting-blocks.spec.js +++ b/test/e2e/specs/editor/various/inserting-blocks.spec.js @@ -1,3 +1,8 @@ +/** + * External dependencies + */ +const path = require( 'path' ); + /** * WordPress dependencies */ @@ -93,7 +98,7 @@ test.describe( 'Inserting blocks (@firefox, @webkit)', () => { -

      +

      ` ); } ); @@ -286,6 +291,50 @@ test.describe( 'Inserting blocks (@firefox, @webkit)', () => { } ); } ); +test.describe( 'insert media from inserter', () => { + let uploadedMedia; + test.beforeAll( async ( { requestUtils } ) => { + await requestUtils.deleteAllMedia(); + uploadedMedia = await requestUtils.uploadMedia( + path.resolve( + process.cwd(), + 'test/e2e/assets/10x10_e2e_test_image_z9T8jK.png' + ) + ); + } ); + test.afterAll( async ( { requestUtils } ) => { + Promise.all( [ + requestUtils.deleteAllMedia(), + requestUtils.deleteAllPosts(), + ] ); + } ); + test.beforeEach( async ( { admin } ) => { + await admin.createNewPost(); + } ); + test( 'insert media from the global inserter', async ( { + page, + editor, + } ) => { + await page.click( + 'role=region[name="Editor top bar"i] >> role=button[name="Toggle block inserter"i]' + ); + await page.click( + 'role=region[name="Block Library"i] >> role=tab[name="Media"i]' + ); + await page.click( + '[aria-label="Media categories"i] >> role=button[name="Images"i]' + ); + await page.click( + `role=listbox[name="Media List"i] >> role=option[name="${ uploadedMedia.title.raw }"]` + ); + await expect.poll( editor.getEditedPostContent ).toBe( + ` +
      ${ uploadedMedia.alt_text }
      +` + ); + } ); +} ); + class InsertingBlocksUtils { constructor( { page, editor } ) { this.page = page; diff --git a/test/e2e/specs/editor/various/splitting-merging.spec.js b/test/e2e/specs/editor/various/splitting-merging.spec.js index 45f4d9347edcc..510c1cb46cf3b 100644 --- a/test/e2e/specs/editor/various/splitting-merging.spec.js +++ b/test/e2e/specs/editor/various/splitting-merging.spec.js @@ -377,7 +377,11 @@ test.describe( 'splitting and merging blocks', () => { await page.keyboard.type( 'item 1' ); await page.keyboard.press( 'Enter' ); await page.keyboard.type( 'item 2' ); - await pageUtils.pressKeyTimes( 'ArrowUp', 2 ); + await pageUtils.pressKeyTimes( 'ArrowUp', 3 ); + await page.keyboard.press( 'Delete' ); + + expect( await editor.getEditedPostContent() ).toMatchSnapshot(); + await page.keyboard.press( 'Delete' ); // Carret should be in the first block and at the proper position. await page.keyboard.type( '-' ); @@ -395,6 +399,10 @@ test.describe( 'splitting and merging blocks', () => { await page.keyboard.type( 'item 2' ); await page.keyboard.press( 'ArrowUp' ); await pageUtils.pressKeyTimes( 'ArrowLeft', 6 ); + await page.keyboard.press( 'Backspace' ); + + expect( await editor.getEditedPostContent() ).toMatchSnapshot(); + await page.keyboard.press( 'Backspace' ); // Carret should be in the first block and at the proper position. await page.keyboard.type( '-' ); diff --git a/test/e2e/specs/site-editor/block-list-panel-preference.spec.js b/test/e2e/specs/site-editor/block-list-panel-preference.spec.js index af56a2c3d3ee9..e31a06f2c888d 100644 --- a/test/e2e/specs/site-editor/block-list-panel-preference.spec.js +++ b/test/e2e/specs/site-editor/block-list-panel-preference.spec.js @@ -12,12 +12,14 @@ test.describe( 'Block list view', () => { await requestUtils.activateTheme( 'twentytwentyone' ); } ); - test( 'Should open by default', async ( { admin, page } ) => { + test( 'Should open by default', async ( { admin, page, siteEditor } ) => { await admin.visitSiteEditor( { postId: 'emptytheme//index', postType: 'wp_template', } ); + await siteEditor.enterEditMode(); + // Should display the Preview button. await expect( page.locator( 'role=region[name="List View"i]' ) @@ -32,6 +34,8 @@ test.describe( 'Block list view', () => { await page.reload(); + await siteEditor.enterEditMode(); + // Should display the Preview button. await expect( page.locator( 'role=region[name="List View"i]' ) diff --git a/test/e2e/specs/site-editor/site-editor-inserter.spec.js b/test/e2e/specs/site-editor/site-editor-inserter.spec.js index d079176bace94..2dc8002b03089 100644 --- a/test/e2e/specs/site-editor/site-editor-inserter.spec.js +++ b/test/e2e/specs/site-editor/site-editor-inserter.spec.js @@ -16,8 +16,9 @@ test.describe( 'Site Editor Inserter', () => { await requestUtils.activateTheme( 'twentytwentyone' ); } ); - test.beforeEach( async ( { admin } ) => { + test.beforeEach( async ( { admin, siteEditor } ) => { await admin.visitSiteEditor(); + await siteEditor.enterEditMode(); } ); test( 'inserter toggle button should toggle global inserter', async ( { diff --git a/test/e2e/specs/site-editor/style-book.spec.js b/test/e2e/specs/site-editor/style-book.spec.js new file mode 100644 index 0000000000000..4f3d092a5c8d9 --- /dev/null +++ b/test/e2e/specs/site-editor/style-book.spec.js @@ -0,0 +1,149 @@ +/** + * WordPress dependencies + */ +const { test, expect } = require( '@wordpress/e2e-test-utils-playwright' ); + +test.use( { + styleBook: async ( { page }, use ) => { + await use( new StyleBook( { page } ) ); + }, +} ); + +test.describe( 'Style Book', () => { + test.beforeAll( async ( { requestUtils } ) => { + await requestUtils.activateTheme( 'emptytheme' ); + } ); + + test.afterAll( async ( { requestUtils } ) => { + await requestUtils.activateTheme( 'twentytwentyone' ); + } ); + + test.beforeEach( async ( { admin, siteEditor, styleBook, page } ) => { + await admin.visitSiteEditor(); + await siteEditor.enterEditMode(); + await styleBook.open(); + await expect( + page.locator( 'role=region[name="Style Book"i]' ) + ).toBeVisible(); + } ); + + test( 'should disable toolbar butons when open', async ( { page } ) => { + await expect( + page.locator( 'role=button[name="Toggle block inserter"i]' ) + ).not.toBeVisible(); + await expect( + page.locator( 'role=button[name="Tools"i]' ) + ).not.toBeVisible(); + await expect( + page.locator( 'role=button[name="Undo"i]' ) + ).not.toBeVisible(); + await expect( + page.locator( 'role=button[name="Redo"i]' ) + ).not.toBeVisible(); + await expect( + page.locator( 'role=button[name="Show template details"i]' ) + ).not.toBeVisible(); + await expect( + page.locator( 'role=button[name="View"i]' ) + ).not.toBeVisible(); + } ); + + test( 'should have tabs containing block examples', async ( { page } ) => { + await expect( page.locator( 'role=tab[name="Text"i]' ) ).toBeVisible(); + await expect( page.locator( 'role=tab[name="Media"i]' ) ).toBeVisible(); + await expect( + page.locator( 'role=tab[name="Design"i]' ) + ).toBeVisible(); + await expect( + page.locator( 'role=tab[name="Widgets"i]' ) + ).toBeVisible(); + await expect( page.locator( 'role=tab[name="Theme"i]' ) ).toBeVisible(); + + await expect( + page.locator( + 'role=button[name="Open Headings styles in Styles panel"i]' + ) + ).toBeVisible(); + await expect( + page.locator( + 'role=button[name="Open Paragraph styles in Styles panel"i]' + ) + ).toBeVisible(); + + await page.click( 'role=tab[name="Media"i]' ); + + await expect( + page.locator( + 'role=button[name="Open Image styles in Styles panel"i]' + ) + ).toBeVisible(); + await expect( + page.locator( + 'role=button[name="Open Gallery styles in Styles panel"i]' + ) + ).toBeVisible(); + } ); + + test( 'should open correct Global Styles panel when example is clicked', async ( { + page, + } ) => { + await page.click( + 'role=button[name="Open Headings styles in Styles panel"i]' + ); + + await expect( + page.locator( + 'role=region[name="Editor settings"i] >> role=heading[name="Heading"i]' + ) + ).toBeVisible(); + } ); + + test( 'should clear Global Styles navigator history when example is clicked', async ( { + page, + } ) => { + await page.click( 'role=button[name="Blocks styles"]' ); + await page.click( 'role=button[name="Heading block styles"]' ); + await page.click( 'role=button[name="Typography styles"]' ); + + await page.click( + 'role=button[name="Open Quote styles in Styles panel"i]' + ); + + await page.click( 'role=button[name="Navigate to the previous view"]' ); + + await expect( + page.locator( 'role=button[name="Navigate to the previous view"]' ) + ).not.toBeVisible(); + } ); + + test( 'should disappear when closed', async ( { page } ) => { + await page.click( + 'role=region[name="Style Book"i] >> role=button[name="Close Style Book"i]' + ); + + await expect( + page.locator( 'role=region[name="Style Book"i]' ) + ).not.toBeVisible(); + } ); +} ); + +class StyleBook { + constructor( { page } ) { + this.page = page; + } + + async disableWelcomeGuide() { + // Turn off the welcome guide. + await this.page.evaluate( () => { + window.wp.data + .dispatch( 'core/preferences' ) + .set( 'core/edit-site', 'welcomeGuideStyles', false ); + } ); + } + + async open() { + await this.disableWelcomeGuide(); + await this.page.click( 'role=button[name="Styles"i]' ); + await this.page.click( 'role=button[name="Open Style Book"i]' ); + } +} diff --git a/test/e2e/specs/site-editor/style-variations.spec.js b/test/e2e/specs/site-editor/style-variations.spec.js index 56de0ac0de219..925be3780c8fd 100644 --- a/test/e2e/specs/site-editor/style-variations.spec.js +++ b/test/e2e/specs/site-editor/style-variations.spec.js @@ -33,11 +33,13 @@ test.describe( 'Global styles variations', () => { admin, page, siteEditorStyleVariations, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'gutenberg-test-themes/style-variations//index', postType: 'wp_template', } ); + await siteEditor.enterEditMode(); await siteEditorStyleVariations.browseStyles(); @@ -68,11 +70,13 @@ test.describe( 'Global styles variations', () => { admin, page, siteEditorStyleVariations, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'gutenberg-test-themes/style-variations//index', postType: 'wp_template', } ); + await siteEditor.enterEditMode(); await siteEditorStyleVariations.browseStyles(); await page.click( 'role=button[name="pink"i]' ); await page.click( @@ -107,11 +111,13 @@ test.describe( 'Global styles variations', () => { admin, page, siteEditorStyleVariations, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'gutenberg-test-themes/style-variations//index', postType: 'wp_template', } ); + await siteEditor.enterEditMode(); await siteEditorStyleVariations.browseStyles(); await page.click( 'role=button[name="yellow"i]' ); await page.click( @@ -152,11 +158,13 @@ test.describe( 'Global styles variations', () => { admin, page, siteEditorStyleVariations, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'gutenberg-test-themes/style-variations//index', postType: 'wp_template', } ); + await siteEditor.enterEditMode(); await siteEditorStyleVariations.browseStyles(); await page.click( 'role=button[name="pink"i]' ); await page.click( @@ -182,11 +190,13 @@ test.describe( 'Global styles variations', () => { admin, page, siteEditorStyleVariations, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'gutenberg-test-themes/style-variations//index', postType: 'wp_template', } ); + await siteEditor.enterEditMode(); await siteEditorStyleVariations.browseStyles(); await page.click( 'role=button[name="yellow"i]' ); diff --git a/test/e2e/specs/site-editor/template-part.spec.js b/test/e2e/specs/site-editor/template-part.spec.js index a793e97158a11..803ed2e57a8fa 100644 --- a/test/e2e/specs/site-editor/template-part.spec.js +++ b/test/e2e/specs/site-editor/template-part.spec.js @@ -33,11 +33,13 @@ test.describe( 'Template Part', () => { admin, editor, page, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'emptytheme//header', postType: 'wp_template_part', } ); + await siteEditor.enterEditMode(); // Insert a new template block and 'start blank'. await editor.insertBlock( { name: 'core/template-part' } ); @@ -62,10 +64,11 @@ test.describe( 'Template Part', () => { admin, editor, page, + siteEditor, } ) => { // Visit the index. await admin.visitSiteEditor(); - + await siteEditor.enterEditMode(); const headerTemplateParts = editor.canvas.locator( '[data-type="core/template-part"]' ); @@ -88,11 +91,12 @@ test.describe( 'Template Part', () => { admin, editor, page, + siteEditor, } ) => { const paragraphText = 'Test 2'; await admin.visitSiteEditor(); - + await siteEditor.enterEditMode(); // Add a block and select it. await editor.insertBlock( { name: 'core/paragraph', @@ -127,12 +131,13 @@ test.describe( 'Template Part', () => { admin, editor, page, + siteEditor, } ) => { const paragraphText1 = 'Test 3'; const paragraphText2 = 'Test 4'; await admin.visitSiteEditor(); - + await siteEditor.enterEditMode(); // Add a block and select it. await editor.insertBlock( { name: 'core/paragraph', @@ -186,6 +191,7 @@ test.describe( 'Template Part', () => { test( 'can detach blocks from a template part', async ( { admin, editor, + siteEditor, } ) => { const paragraphText = 'Test 3'; @@ -194,7 +200,7 @@ test.describe( 'Template Part', () => { postId: 'emptytheme//header', postType: 'wp_template_part', } ); - + await siteEditor.enterEditMode(); await editor.insertBlock( { name: 'core/paragraph', attributes: { @@ -205,7 +211,7 @@ test.describe( 'Template Part', () => { // Visit the index. await admin.visitSiteEditor(); - + await siteEditor.enterEditMode(); // Check that the header contains the paragraph added earlier. const paragraph = editor.canvas.locator( `p >> text="${ paragraphText }"` @@ -230,6 +236,7 @@ test.describe( 'Template Part', () => { test( 'shows changes in a template when a template part it contains is modified', async ( { admin, editor, + siteEditor, } ) => { const paragraphText = 'Test 1'; @@ -237,7 +244,7 @@ test.describe( 'Template Part', () => { postId: 'emptytheme//header', postType: 'wp_template_part', } ); - + await siteEditor.enterEditMode(); // Edit the header. await editor.insertBlock( { name: 'core/paragraph', @@ -250,7 +257,7 @@ test.describe( 'Template Part', () => { // Visit the index. await admin.visitSiteEditor(); - + await siteEditor.enterEditMode(); const paragraph = editor.canvas.locator( `p >> text="${ paragraphText }"` ); @@ -263,6 +270,7 @@ test.describe( 'Template Part', () => { admin, editor, page, + siteEditor, } ) => { const paragraphText = 'Test 4'; @@ -270,6 +278,7 @@ test.describe( 'Template Part', () => { postId: 'emptytheme//header', postType: 'wp_template_part', } ); + await siteEditor.enterEditMode(); await editor.insertBlock( { name: 'core/paragraph', attributes: { diff --git a/test/e2e/specs/site-editor/template-revert.spec.js b/test/e2e/specs/site-editor/template-revert.spec.js index 6a6ba6353a5e7..dcf8ebedb8b78 100644 --- a/test/e2e/specs/site-editor/template-revert.spec.js +++ b/test/e2e/specs/site-editor/template-revert.spec.js @@ -20,9 +20,10 @@ test.describe( 'Template Revert', () => { await requestUtils.deleteAllTemplates( 'wp_template_part' ); await requestUtils.activateTheme( 'twentytwentyone' ); } ); - test.beforeEach( async ( { admin, requestUtils } ) => { + test.beforeEach( async ( { admin, requestUtils, siteEditor } ) => { await requestUtils.deleteAllTemplates( 'wp_template' ); await admin.visitSiteEditor(); + await siteEditor.enterEditMode(); } ); test( 'should delete the template after saving the reverted template', async ( { @@ -247,6 +248,7 @@ test.describe( 'Template Revert', () => { editor, page, templateRevertUtils, + siteEditor, } ) => { await editor.insertBlock( { name: 'core/paragraph', @@ -265,7 +267,7 @@ test.describe( 'Template Revert', () => { await editor.saveSiteEditorEntities(); await admin.visitSiteEditor(); - + await siteEditor.enterEditMode(); const contentAfter = await templateRevertUtils.getCurrentSiteEditorContent(); expect( contentAfter ).toEqual( contentBefore ); diff --git a/test/e2e/specs/site-editor/title.spec.js b/test/e2e/specs/site-editor/title.spec.js index 6d00c4ad7c3a2..9b1ad4a5098d1 100644 --- a/test/e2e/specs/site-editor/title.spec.js +++ b/test/e2e/specs/site-editor/title.spec.js @@ -15,13 +15,14 @@ test.describe( 'Site editor title', () => { test( 'displays the selected template name in the title for the index template', async ( { admin, page, + siteEditor, } ) => { // Navigate to a template. await admin.visitSiteEditor( { postId: 'emptytheme//index', postType: 'wp_template', } ); - + await siteEditor.enterEditMode(); const title = await page.locator( 'role=region[name="Editor top bar"i] >> role=heading[level=1]' ); @@ -32,13 +33,14 @@ test.describe( 'Site editor title', () => { test( 'displays the selected template name in the title for the header template', async ( { admin, page, + siteEditor, } ) => { // Navigate to a template part. await admin.visitSiteEditor( { postId: 'emptytheme//header', postType: 'wp_template_part', } ); - + await siteEditor.enterEditMode(); const title = await page.locator( 'role=region[name="Editor top bar"i] >> role=heading[level=1]' ); @@ -49,12 +51,13 @@ test.describe( 'Site editor title', () => { test( "displays the selected template part's name in the secondary title when a template part is selected from List View", async ( { admin, page, + siteEditor, } ) => { await admin.visitSiteEditor( { postId: 'emptytheme//index', postType: 'wp_template', } ); - + await siteEditor.enterEditMode(); // Select the header template part via list view. await page.click( 'role=button[name="List View"i]' ); const listView = await page.locator( diff --git a/test/e2e/specs/site-editor/writing-flow.spec.js b/test/e2e/specs/site-editor/writing-flow.spec.js index e854b31cee481..6eb4bf3e471ec 100644 --- a/test/e2e/specs/site-editor/writing-flow.spec.js +++ b/test/e2e/specs/site-editor/writing-flow.spec.js @@ -28,13 +28,14 @@ test.describe( 'Site editor writing flow', () => { editor, page, pageUtils, + siteEditor, } ) => { // Navigate to a template part with only a couple of blocks. await admin.visitSiteEditor( { postId: 'emptytheme//header', postType: 'wp_template_part', } ); - + await siteEditor.enterEditMode(); // Select the first site title block. const siteTitleBlock = editor.canvas.locator( 'role=document[name="Block: Site Title"i]' @@ -56,13 +57,14 @@ test.describe( 'Site editor writing flow', () => { editor, page, pageUtils, + siteEditor, } ) => { // Navigate to a template part with only a couple of blocks. await admin.visitSiteEditor( { postId: 'emptytheme//header', postType: 'wp_template_part', } ); - + await siteEditor.enterEditMode(); // Make sure the sidebar is open. await editor.openDocumentSettingsSidebar(); diff --git a/test/e2e/specs/widgets/customizing-widgets.spec.js b/test/e2e/specs/widgets/customizing-widgets.spec.js index 6b2b06d970e53..776770fecbefe 100644 --- a/test/e2e/specs/widgets/customizing-widgets.spec.js +++ b/test/e2e/specs/widgets/customizing-widgets.spec.js @@ -235,7 +235,7 @@ test.describe( 'Widgets Customizer', () => { 'sidebar-1' ); await requestUtils.addWidgetBlock( - `\n

      First Heading

      \n`, + `\n

      First Heading

      \n`, 'sidebar-1' ); diff --git a/test/emptytheme/block-template-parts/header.html b/test/emptytheme/parts/header.html similarity index 100% rename from test/emptytheme/block-template-parts/header.html rename to test/emptytheme/parts/header.html diff --git a/test/emptytheme/block-templates/category.html b/test/emptytheme/templates/category.html similarity index 100% rename from test/emptytheme/block-templates/category.html rename to test/emptytheme/templates/category.html diff --git a/test/emptytheme/block-templates/index.html b/test/emptytheme/templates/index.html similarity index 100% rename from test/emptytheme/block-templates/index.html rename to test/emptytheme/templates/index.html diff --git a/test/emptytheme/block-templates/singular.html b/test/emptytheme/templates/singular.html similarity index 100% rename from test/emptytheme/block-templates/singular.html rename to test/emptytheme/templates/singular.html diff --git a/test/emptytheme/theme.json b/test/emptytheme/theme.json index ed2d7b8d0946a..b28e6c9f274b2 100644 --- a/test/emptytheme/theme.json +++ b/test/emptytheme/theme.json @@ -1,4 +1,5 @@ { + "$schema": "https://schemas.wp.org/trunk/theme.json", "version": 2, "settings": { "appearanceTools": true, diff --git a/test/integration/__snapshots__/blocks-raw-handling.test.js.snap b/test/integration/__snapshots__/blocks-raw-handling.test.js.snap index 6c341c815e035..19aa5bd262ea3 100644 --- a/test/integration/__snapshots__/blocks-raw-handling.test.js.snap +++ b/test/integration/__snapshots__/blocks-raw-handling.test.js.snap @@ -1,5 +1,41 @@ // Jest Snapshot v1, https://goo.gl/fbAQLP +exports[`Blocks raw handling pasteHandler apple 1`] = `"This is a title


      This is a heading


      This is a paragraph with a link.


      A
      Bulleted
      Indented
      List


      One
      Two
      Three


      One
      Two
      Three
      1
      2
      3
      I
      II
      III


      An image:
      "`; + +exports[`Blocks raw handling pasteHandler classic 1`] = `"First paragraph

      Second paragraph
      Third paragraph
      Fourth paragraph
      Fifth paragraph
      Sixth paragraph"`; + +exports[`Blocks raw handling pasteHandler evernote 1`] = `"This is a paragraph.


      This is a link.


      An
      Unordered
      Indented
      List


      One
      Two
      Indented
      Three






      One
      Two
      Three
      Four
      Five
      Six
      "`; + +exports[`Blocks raw handling pasteHandler google-docs 1`] = `"This is a title

      This is a heading

      Formatting test: bold, italic, link, strikethrough, superscript, subscript, nested.

      A
      Bulleted
      Indented
      List

      One
      Two
      Three




      One
      Two
      Three
      1
      2
      3
      I
      II
      III





      An image:


      "`; + +exports[`Blocks raw handling pasteHandler google-docs-list-only 1`] = `"My first list item
      A sub list item
      A second sub list item
      My second list item
      My third list item"`; + +exports[`Blocks raw handling pasteHandler google-docs-table 1`] = `"



      One
      Two
      Three
      1
      2
      3
      I
      II
      III"`; + +exports[`Blocks raw handling pasteHandler google-docs-table-with-comments 1`] = `"



      One
      Two
      Three
      1
      2
      3
      I
      II
      III"`; + +exports[`Blocks raw handling pasteHandler google-docs-with-comments 1`] = `"This is a title

      This is a heading

      Formatting test: bold, italic, link, strikethrough, superscript, subscript, nested.

      A
      Bulleted
      Indented
      List

      One
      Two
      Three




      One
      Two
      Three
      1
      2
      3
      I
      II
      III





      An image:



      "`; + +exports[`Blocks raw handling pasteHandler gutenberg 1`] = `"Test"`; + +exports[`Blocks raw handling pasteHandler iframe-embed 1`] = `""`; + +exports[`Blocks raw handling pasteHandler markdown 1`] = `"This is a heading with italic
      This is a paragraph with a link, bold, and strikethrough.
      Preserve
      line breaks please.
      Lists
      A
      Bulleted Indented
      List
      One
      Two
      Three
      Table
      First Header
      Second Header
      Content from cell 1
      Content from cell 2
      Content in the first column
      Content in the second column



      Table with empty cells.
      Quote
      First
      Second
      Code
      Inline code tags should work.
      This is a code block."`; + +exports[`Blocks raw handling pasteHandler ms-word 1`] = `"This is a title
       
      This is a subtitle
       
      This is a heading level 1
       
      This is a heading level 2
       
      This is a paragraph with a link.
       
      ·      A
      ·      Bulleted
      o   Indented
      ·      List
       
      1      One
      2      Two
      3      Three
       
      One
      Two
      Three
      1
      2
      3
      I
      II
      III
       
      An image:
       

      This is an anchor link that leads to the next paragraph.
      This is the paragraph with the anchor.
      This is an anchor link that leads nowhere.
      This is a paragraph with an anchor with no link pointing to it.
      This is a reference to a footnote[1].
      This is a reference to an endnote[i].


      [1] This is a footnote.


      [i] This is an endnote."`; + +exports[`Blocks raw handling pasteHandler ms-word-online 1`] = `"This is a heading 
      This is a paragraph with a link

      Bulleted 
      Indented 
      List 
       
      One 
      Two 
      Three 

      One 
      Two 
      Three 




      II 
      III 
       
      An image: 
       "`; + +exports[`Blocks raw handling pasteHandler ms-word-styled 1`] = `"
      Lorem ipsum dolor sit amet, consectetur adipiscing elit 


      Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque aliquet hendrerit auctor. Nam lobortis, est vel lacinia tincidunt, purus tellus vehicula ex, nec pharetra justo dui sed lorem. Nam congue laoreet massa, quis varius est tincidunt ut."`; + +exports[`Blocks raw handling pasteHandler nested-divs 1`] = `"First paragraph

      Second paragraph
      Third paragraph
      Fourth paragraph
      Fifth paragraph
      Sixth paragraph"`; + +exports[`Blocks raw handling pasteHandler one-image 1`] = `"\\"\\""`; + +exports[`Blocks raw handling pasteHandler plain 1`] = `"test
      test

      test"`; + +exports[`Blocks raw handling pasteHandler shortcode-matching 1`] = `"[gallery ids=\\"40,41,42\\"]
      [gallery ids=\\"1000\\"]
      [gallery ids=\\"42\\"]"`; + exports[`Blocks raw handling pasteHandler should remove extra blank lines 1`] = ` "

      1

      @@ -20,7 +56,7 @@ exports[`Blocks raw handling pasteHandler should strip some text-level elements exports[`Blocks raw handling pasteHandler should strip windows data 1`] = ` " -

      Heading Win

      +

      Heading Win

      @@ -28,10 +64,18 @@ exports[`Blocks raw handling pasteHandler should strip windows data 1`] = ` " `; +exports[`Blocks raw handling pasteHandler slack-paragraphs 1`] = `"test with link
      a new linea new paragraph
      another new lineanother paragraph"`; + +exports[`Blocks raw handling pasteHandler slack-quote 1`] = `"Test with link."`; + +exports[`Blocks raw handling pasteHandler two-images 1`] = `"\\"\\" \\"\\""`; + +exports[`Blocks raw handling pasteHandler wordpress 1`] = `"Howdy
      This is a paragraph.
      More tag

      Shortcode
      [gallery ids=\\"1\\"]
      \\"\\"
      \\"\\" test
      \\"\\" test"`; + exports[`Blocks raw handling should correctly handle quotes with mixed content 1`] = ` "
      -

      chicken

      +

      chicken

      @@ -42,7 +86,7 @@ exports[`Blocks raw handling should correctly handle quotes with mixed content 1 exports[`rawHandler should convert HTML post to blocks with minimal content changes 1`] = ` " -

      Howdy

      +

      Howdy

      @@ -58,7 +102,7 @@ exports[`rawHandler should convert HTML post to blocks with minimal content chan -

      More tag

      +

      More tag

      @@ -66,7 +110,7 @@ exports[`rawHandler should convert HTML post to blocks with minimal content chan -

      Shortcode

      +

      Shortcode

      @@ -98,7 +142,7 @@ exports[`rawHandler should convert HTML post to blocks with minimal content chan
      -

      Heading

      +

      Heading

      diff --git a/test/integration/blocks-raw-handling.test.js b/test/integration/blocks-raw-handling.test.js index 2bc08c9a7b35a..eeab475567d27 100644 --- a/test/integration/blocks-raw-handling.test.js +++ b/test/integration/blocks-raw-handling.test.js @@ -249,7 +249,7 @@ describe( 'Blocks raw handling', () => { .join( '' ); expect( filtered ).toBe( - '

      Some heading

      A paragraph.

      ' + '

      Some heading

      A paragraph.

      ' ); expect( console ).toHaveLogged(); } ); @@ -336,7 +336,7 @@ describe( 'Blocks raw handling', () => { it( 'should correctly handle quotes with mixed content', () => { const filtered = serialize( pasteHandler( { - HTML: '

      chicken

      ribs

      ', + HTML: '

      chicken

      ribs

      ', mode: 'AUTO', } ) ); @@ -427,10 +427,14 @@ describe( 'Blocks raw handling', () => { expect( serialized ).toBe( output ); - if ( type !== 'gutenberg' ) { - // eslint-disable-next-line jest/no-conditional-expect - expect( console ).toHaveLogged(); - } + const convertedInline = pasteHandler( { + HTML, + plainText, + mode: 'INLINE', + } ); + + expect( convertedInline ).toMatchSnapshot(); + expect( console ).toHaveLogged(); } ); } ); diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-1.serialized.html b/test/integration/fixtures/blocks/core__heading__deprecated-1.serialized.html index 0a2f890302c3f..a064b6f60cf04 100644 --- a/test/integration/fixtures/blocks/core__heading__deprecated-1.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__deprecated-1.serialized.html @@ -1,3 +1,3 @@ -

      A picture is worth a thousand words, or so the saying goes

      +

      A picture is worth a thousand words, or so the saying goes

      diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-2.serialized.html b/test/integration/fixtures/blocks/core__heading__deprecated-2.serialized.html index 1dbaea2593eb9..989d82a8d7fca 100644 --- a/test/integration/fixtures/blocks/core__heading__deprecated-2.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__deprecated-2.serialized.html @@ -1,3 +1,3 @@ -

      text

      +

      text

      diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-3.serialized.html b/test/integration/fixtures/blocks/core__heading__deprecated-3.serialized.html index c2c4470e500f1..2e04a7e01f977 100644 --- a/test/integration/fixtures/blocks/core__heading__deprecated-3.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__deprecated-3.serialized.html @@ -1,3 +1,3 @@ -

      Text

      +

      Text

      diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-4.serialized.html b/test/integration/fixtures/blocks/core__heading__deprecated-4.serialized.html index 2e4f98cda15ff..cc4218e776a73 100644 --- a/test/integration/fixtures/blocks/core__heading__deprecated-4.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__deprecated-4.serialized.html @@ -1,3 +1,3 @@ -

      Text

      +

      Text

      diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-5.html b/test/integration/fixtures/blocks/core__heading__deprecated-5.html new file mode 100644 index 0000000000000..c66e23d5ef74e --- /dev/null +++ b/test/integration/fixtures/blocks/core__heading__deprecated-5.html @@ -0,0 +1,3 @@ + +

      Text

      + diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-5.json b/test/integration/fixtures/blocks/core__heading__deprecated-5.json new file mode 100644 index 0000000000000..64c89ad0cd979 --- /dev/null +++ b/test/integration/fixtures/blocks/core__heading__deprecated-5.json @@ -0,0 +1,13 @@ +[ + { + "name": "core/heading", + "isValid": true, + "attributes": { + "textAlign": "right", + "content": "Text", + "level": 2, + "textColor": "primary" + }, + "innerBlocks": [] + } +] diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-5.parsed.json b/test/integration/fixtures/blocks/core__heading__deprecated-5.parsed.json new file mode 100644 index 0000000000000..b15597658da02 --- /dev/null +++ b/test/integration/fixtures/blocks/core__heading__deprecated-5.parsed.json @@ -0,0 +1,14 @@ +[ + { + "blockName": "core/heading", + "attrs": { + "textAlign": "right", + "textColor": "primary" + }, + "innerBlocks": [], + "innerHTML": "\n

      Text

      \n", + "innerContent": [ + "\n

      Text

      \n" + ] + } +] diff --git a/test/integration/fixtures/blocks/core__heading__deprecated-5.serialized.html b/test/integration/fixtures/blocks/core__heading__deprecated-5.serialized.html new file mode 100644 index 0000000000000..a81affa226f77 --- /dev/null +++ b/test/integration/fixtures/blocks/core__heading__deprecated-5.serialized.html @@ -0,0 +1,3 @@ + +

      Text

      + diff --git a/test/integration/fixtures/blocks/core__heading__h2-color.html b/test/integration/fixtures/blocks/core__heading__h2-color.html index 23c39f701c765..7d1dee6d70632 100644 --- a/test/integration/fixtures/blocks/core__heading__h2-color.html +++ b/test/integration/fixtures/blocks/core__heading__h2-color.html @@ -1,4 +1,4 @@ -

      Text

      +

      Text

      diff --git a/test/integration/fixtures/blocks/core__heading__h2-color.parsed.json b/test/integration/fixtures/blocks/core__heading__h2-color.parsed.json index 64a2fb552e7fb..763648f6d6112 100644 --- a/test/integration/fixtures/blocks/core__heading__h2-color.parsed.json +++ b/test/integration/fixtures/blocks/core__heading__h2-color.parsed.json @@ -5,9 +5,9 @@ "textColor": "accent" }, "innerBlocks": [], - "innerHTML": "\n

      Text

      \n", + "innerHTML": "\n

      Text

      \n", "innerContent": [ - "\n

      Text

      \n" + "\n

      Text

      \n" ] } ] diff --git a/test/integration/fixtures/blocks/core__heading__h2-color.serialized.html b/test/integration/fixtures/blocks/core__heading__h2-color.serialized.html index 862a62b2d7975..73bb943850272 100644 --- a/test/integration/fixtures/blocks/core__heading__h2-color.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__h2-color.serialized.html @@ -1,3 +1,3 @@ -

      Text

      +

      Text

      diff --git a/test/integration/fixtures/blocks/core__heading__h2.html b/test/integration/fixtures/blocks/core__heading__h2.html index a55b391a25057..d7b487af0b670 100644 --- a/test/integration/fixtures/blocks/core__heading__h2.html +++ b/test/integration/fixtures/blocks/core__heading__h2.html @@ -1,3 +1,3 @@ -

      A picture is worth a thousand words, or so the saying goes

      +

      A picture is worth a thousand words, or so the saying goes

      diff --git a/test/integration/fixtures/blocks/core__heading__h2.parsed.json b/test/integration/fixtures/blocks/core__heading__h2.parsed.json index 7e8932f95e5de..ab763edaea5ec 100644 --- a/test/integration/fixtures/blocks/core__heading__h2.parsed.json +++ b/test/integration/fixtures/blocks/core__heading__h2.parsed.json @@ -3,9 +3,9 @@ "blockName": "core/heading", "attrs": {}, "innerBlocks": [], - "innerHTML": "\n

      A picture is worth a thousand words, or so the saying goes

      \n", + "innerHTML": "\n

      A picture is worth a thousand words, or so the saying goes

      \n", "innerContent": [ - "\n

      A picture is worth a thousand words, or so the saying goes

      \n" + "\n

      A picture is worth a thousand words, or so the saying goes

      \n" ] } ] diff --git a/test/integration/fixtures/blocks/core__heading__h2.serialized.html b/test/integration/fixtures/blocks/core__heading__h2.serialized.html index 1212bf83c3029..96cc3063da105 100644 --- a/test/integration/fixtures/blocks/core__heading__h2.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__h2.serialized.html @@ -1,3 +1,3 @@ -

      A picture is worth a thousand words, or so the saying goes

      +

      A picture is worth a thousand words, or so the saying goes

      diff --git a/test/integration/fixtures/blocks/core__heading__h4-em.html b/test/integration/fixtures/blocks/core__heading__h4-em.html index bd13c30bb968d..0a72af58e24db 100644 --- a/test/integration/fixtures/blocks/core__heading__h4-em.html +++ b/test/integration/fixtures/blocks/core__heading__h4-em.html @@ -1,3 +1,3 @@ -

      The Inserter Tool

      +

      The Inserter Tool

      diff --git a/test/integration/fixtures/blocks/core__heading__h4-em.parsed.json b/test/integration/fixtures/blocks/core__heading__h4-em.parsed.json index 283fd532b3fd2..b8cd44ca0471e 100644 --- a/test/integration/fixtures/blocks/core__heading__h4-em.parsed.json +++ b/test/integration/fixtures/blocks/core__heading__h4-em.parsed.json @@ -5,7 +5,9 @@ "level": 4 }, "innerBlocks": [], - "innerHTML": "\n

      The Inserter Tool

      \n", - "innerContent": [ "\n

      The Inserter Tool

      \n" ] + "innerHTML": "\n

      The Inserter Tool

      \n", + "innerContent": [ + "\n

      The Inserter Tool

      \n" + ] } ] diff --git a/test/integration/fixtures/blocks/core__heading__h4-em.serialized.html b/test/integration/fixtures/blocks/core__heading__h4-em.serialized.html index 051e1c9f39b27..d7a3b904dbd9a 100644 --- a/test/integration/fixtures/blocks/core__heading__h4-em.serialized.html +++ b/test/integration/fixtures/blocks/core__heading__h4-em.serialized.html @@ -1,3 +1,3 @@ -

      The Inserter Tool

      +

      The Inserter Tool

      diff --git a/test/integration/fixtures/blocks/core__heading_align-textalign.html b/test/integration/fixtures/blocks/core__heading_align-textalign.html index 26c8041d1b97a..3f7fb12335b4b 100644 --- a/test/integration/fixtures/blocks/core__heading_align-textalign.html +++ b/test/integration/fixtures/blocks/core__heading_align-textalign.html @@ -1,3 +1,3 @@ -

      Text

      +

      Text

      diff --git a/test/integration/fixtures/blocks/core__heading_align-textalign.parsed.json b/test/integration/fixtures/blocks/core__heading_align-textalign.parsed.json index 918ba0e5c79bb..bd378de8f71dc 100644 --- a/test/integration/fixtures/blocks/core__heading_align-textalign.parsed.json +++ b/test/integration/fixtures/blocks/core__heading_align-textalign.parsed.json @@ -6,9 +6,9 @@ "align": "wide" }, "innerBlocks": [], - "innerHTML": "\n

      Text

      \n", + "innerHTML": "\n

      Text

      \n", "innerContent": [ - "\n

      Text

      \n" + "\n

      Text

      \n" ] } ] diff --git a/test/integration/fixtures/blocks/core__heading_align-textalign.serialized.html b/test/integration/fixtures/blocks/core__heading_align-textalign.serialized.html index 26c8041d1b97a..3f7fb12335b4b 100644 --- a/test/integration/fixtures/blocks/core__heading_align-textalign.serialized.html +++ b/test/integration/fixtures/blocks/core__heading_align-textalign.serialized.html @@ -1,3 +1,3 @@ -

      Text

      +

      Text

      diff --git a/test/integration/fixtures/blocks/core__page-list-item.html b/test/integration/fixtures/blocks/core__page-list-item.html new file mode 100644 index 0000000000000..2b1b33d71231c --- /dev/null +++ b/test/integration/fixtures/blocks/core__page-list-item.html @@ -0,0 +1 @@ + diff --git a/test/integration/fixtures/blocks/core__page-list-item.json b/test/integration/fixtures/blocks/core__page-list-item.json new file mode 100644 index 0000000000000..5e6f09eb971eb --- /dev/null +++ b/test/integration/fixtures/blocks/core__page-list-item.json @@ -0,0 +1,11 @@ +[ + { + "name": "core/page-list-item", + "isValid": true, + "attributes": { + "label": "Page", + "link": "https://example.com" + }, + "innerBlocks": [] + } +] diff --git a/test/integration/fixtures/blocks/core__page-list-item.parsed.json b/test/integration/fixtures/blocks/core__page-list-item.parsed.json new file mode 100644 index 0000000000000..e93cff97dd385 --- /dev/null +++ b/test/integration/fixtures/blocks/core__page-list-item.parsed.json @@ -0,0 +1,14 @@ +[ + { + "blockName": "core/page-list-item", + "attrs": { + "id": "1", + "label": "Page", + "link": "https://example.com", + "hasChildren": "false" + }, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + } +] diff --git a/test/integration/fixtures/blocks/core__page-list-item.serialized.html b/test/integration/fixtures/blocks/core__page-list-item.serialized.html new file mode 100644 index 0000000000000..d61659fe1899a --- /dev/null +++ b/test/integration/fixtures/blocks/core__page-list-item.serialized.html @@ -0,0 +1 @@ + diff --git a/test/integration/fixtures/blocks/core__page-list.json b/test/integration/fixtures/blocks/core__page-list.json index 0bb67d8e44efd..b1992a437b884 100644 --- a/test/integration/fixtures/blocks/core__page-list.json +++ b/test/integration/fixtures/blocks/core__page-list.json @@ -2,7 +2,9 @@ { "name": "core/page-list", "isValid": true, - "attributes": {}, + "attributes": { + "parentPageID": 0 + }, "innerBlocks": [] } ] diff --git a/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.html b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.html new file mode 100644 index 0000000000000..e85e61b5d7b14 --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.html @@ -0,0 +1,6 @@ + + + diff --git a/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.json b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.json new file mode 100644 index 0000000000000..82bc41a40fb1b --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.json @@ -0,0 +1,72 @@ +[ + { + "name": "core/query", + "isValid": true, + "attributes": { + "queryId": 19, + "query": { + "perPage": 3, + "pages": 0, + "offset": 0, + "postType": "post", + "order": "desc", + "orderBy": "date", + "author": "", + "search": "", + "exclude": [], + "sticky": "", + "inherit": false, + "taxQuery": { + "category": [ 3, 5 ], + "post_tag": [ 6 ] + } + }, + "tagName": "div", + "displayLayout": { + "type": "list" + } + }, + "innerBlocks": [ + { + "name": "core/group", + "isValid": true, + "attributes": { + "tagName": "div", + "textColor": "pale-cyan-blue", + "style": { + "color": { + "background": "#284d5f" + }, + "elements": { + "link": { + "color": { + "text": "var:preset|color|luminous-vivid-amber" + } + } + } + } + }, + "innerBlocks": [ + { + "name": "core/post-template", + "isValid": true, + "attributes": {}, + "innerBlocks": [ + { + "name": "core/post-title", + "isValid": true, + "attributes": { + "level": 2, + "isLink": false, + "rel": "", + "linkTarget": "_self" + }, + "innerBlocks": [] + } + ] + } + ] + } + ] + } +] diff --git a/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.parsed.json b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.parsed.json new file mode 100644 index 0000000000000..73e4f9b6c18f4 --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.parsed.json @@ -0,0 +1,59 @@ +[ + { + "blockName": "core/query", + "attrs": { + "queryId": 19, + "query": { + "perPage": 3, + "pages": 0, + "offset": 0, + "postType": "post", + "categoryIds": [ 3, 5 ], + "tagIds": [ 6 ], + "order": "desc", + "orderBy": "date", + "author": "", + "search": "", + "exclude": [], + "sticky": "", + "inherit": false + }, + "style": { + "color": { + "background": "#284d5f" + }, + "elements": { + "link": { + "color": { + "text": "var:preset|color|luminous-vivid-amber" + } + } + } + }, + "textColor": "pale-cyan-blue" + }, + "innerBlocks": [ + { + "blockName": "core/post-template", + "attrs": {}, + "innerBlocks": [ + { + "blockName": "core/post-title", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + } + ], + "innerHTML": "\n\n", + "innerContent": [ "\n", null, "\n" ] + } + ], + "innerHTML": "\n
      \n
      \n", + "innerContent": [ + "\n
      ", + null, + "\n
      \n" + ] + } +] diff --git a/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.serialized.html b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.serialized.html new file mode 100644 index 0000000000000..f86b4f26ecc1d --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-2-with-colors.serialized.html @@ -0,0 +1,7 @@ + +
      + +
      + diff --git a/test/integration/fixtures/blocks/core__query__deprecated-3.html b/test/integration/fixtures/blocks/core__query__deprecated-3.html new file mode 100644 index 0000000000000..9da20c78184e2 --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-3.html @@ -0,0 +1,23 @@ + + + \ No newline at end of file diff --git a/test/integration/fixtures/blocks/core__query__deprecated-3.json b/test/integration/fixtures/blocks/core__query__deprecated-3.json new file mode 100644 index 0000000000000..bb1478cd676f2 --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-3.json @@ -0,0 +1,133 @@ +[ + { + "name": "core/query", + "isValid": true, + "attributes": { + "queryId": 3, + "query": { + "perPage": 3, + "pages": 0, + "offset": 0, + "postType": "post", + "order": "desc", + "orderBy": "date", + "author": "", + "search": "", + "exclude": [], + "sticky": "", + "inherit": false + }, + "tagName": "div", + "displayLayout": { + "type": "flex", + "columns": 3 + }, + "align": "wide" + }, + "innerBlocks": [ + { + "name": "core/group", + "isValid": true, + "attributes": { + "tagName": "div", + "textColor": "pale-cyan-blue", + "style": { + "color": { + "background": "#284d5f" + }, + "elements": { + "link": { + "color": { + "text": "var:preset|color|luminous-vivid-amber" + } + } + } + } + }, + "innerBlocks": [ + { + "name": "core/post-template", + "isValid": true, + "attributes": { + "fontSize": "large" + }, + "innerBlocks": [ + { + "name": "core/post-title", + "isValid": true, + "attributes": { + "level": 2, + "isLink": false, + "rel": "", + "linkTarget": "_self" + }, + "innerBlocks": [] + }, + { + "name": "core/post-date", + "isValid": true, + "attributes": { + "isLink": false, + "displayType": "date" + }, + "innerBlocks": [] + }, + { + "name": "core/post-excerpt", + "isValid": true, + "attributes": { + "showMoreOnNewLine": true + }, + "innerBlocks": [] + } + ] + }, + { + "name": "core/query-pagination", + "isValid": true, + "attributes": { + "paginationArrow": "none" + }, + "innerBlocks": [ + { + "name": "core/query-pagination-previous", + "isValid": true, + "attributes": {}, + "innerBlocks": [] + }, + { + "name": "core/query-pagination-numbers", + "isValid": true, + "attributes": {}, + "innerBlocks": [] + }, + { + "name": "core/query-pagination-next", + "isValid": true, + "attributes": {}, + "innerBlocks": [] + } + ] + }, + { + "name": "core/query-no-results", + "isValid": true, + "attributes": {}, + "innerBlocks": [ + { + "name": "core/paragraph", + "isValid": true, + "attributes": { + "content": "No results found.", + "dropCap": false, + "placeholder": "Add text or blocks that will display when a query returns no results." + }, + "innerBlocks": [] + } + ] + } + ] + } + ] + } +] diff --git a/test/integration/fixtures/blocks/core__query__deprecated-3.parsed.json b/test/integration/fixtures/blocks/core__query__deprecated-3.parsed.json new file mode 100644 index 0000000000000..a26e573bac314 --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-3.parsed.json @@ -0,0 +1,128 @@ +[ + { + "blockName": "core/query", + "attrs": { + "queryId": 3, + "query": { + "perPage": 3, + "pages": 0, + "offset": 0, + "postType": "post", + "order": "desc", + "orderBy": "date", + "author": "", + "search": "", + "exclude": [], + "sticky": "", + "inherit": false + }, + "displayLayout": { + "type": "flex", + "columns": 3 + }, + "align": "wide", + "style": { + "color": { + "background": "#284d5f" + }, + "elements": { + "link": { + "color": { + "text": "var:preset|color|luminous-vivid-amber" + } + } + } + }, + "textColor": "pale-cyan-blue" + }, + "innerBlocks": [ + { + "blockName": "core/post-template", + "attrs": { + "fontSize": "large" + }, + "innerBlocks": [ + { + "blockName": "core/post-title", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + }, + { + "blockName": "core/post-date", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + }, + { + "blockName": "core/post-excerpt", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + } + ], + "innerHTML": "\n\n\n\n\n\n", + "innerContent": [ "\n", null, "\n\n", null, "\n\n", null, "\n" ] + }, + { + "blockName": "core/query-pagination", + "attrs": {}, + "innerBlocks": [ + { + "blockName": "core/query-pagination-previous", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + }, + { + "blockName": "core/query-pagination-numbers", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + }, + { + "blockName": "core/query-pagination-next", + "attrs": {}, + "innerBlocks": [], + "innerHTML": "", + "innerContent": [] + } + ], + "innerHTML": "\n\n\n\n\n\n", + "innerContent": [ "\n", null, "\n\n", null, "\n\n", null, "\n" ] + }, + { + "blockName": "core/query-no-results", + "attrs": {}, + "innerBlocks": [ + { + "blockName": "core/paragraph", + "attrs": { + "placeholder": "Add text or blocks that will display when a query returns no results." + }, + "innerBlocks": [], + "innerHTML": "\n

      No results found.

      \n", + "innerContent": [ "\n

      No results found.

      \n" ] + } + ], + "innerHTML": "\n\n", + "innerContent": [ "\n", null, "\n" ] + } + ], + "innerHTML": "\n
      \n\n\n\n
      \n", + "innerContent": [ + "\n
      ", + null, + "\n\n", + null, + "\n\n", + null, + "
      \n" + ] + } +] diff --git a/test/integration/fixtures/blocks/core__query__deprecated-3.serialized.html b/test/integration/fixtures/blocks/core__query__deprecated-3.serialized.html new file mode 100644 index 0000000000000..edbf5b1a0557b --- /dev/null +++ b/test/integration/fixtures/blocks/core__query__deprecated-3.serialized.html @@ -0,0 +1,25 @@ + +
      + +
      + diff --git a/test/integration/fixtures/documents/google-docs-out.html b/test/integration/fixtures/documents/google-docs-out.html index db8bc81948fcc..e88311fab7b1f 100644 --- a/test/integration/fixtures/documents/google-docs-out.html +++ b/test/integration/fixtures/documents/google-docs-out.html @@ -3,7 +3,7 @@ -

      This is a heading

      +

      This is a heading

      diff --git a/test/integration/fixtures/documents/google-docs-with-comments-out.html b/test/integration/fixtures/documents/google-docs-with-comments-out.html index db8bc81948fcc..e88311fab7b1f 100644 --- a/test/integration/fixtures/documents/google-docs-with-comments-out.html +++ b/test/integration/fixtures/documents/google-docs-with-comments-out.html @@ -3,7 +3,7 @@ -

      This is a heading

      +

      This is a heading

      diff --git a/test/integration/fixtures/documents/markdown-out.html b/test/integration/fixtures/documents/markdown-out.html index c1237925a750b..31fa8dacbebf0 100644 --- a/test/integration/fixtures/documents/markdown-out.html +++ b/test/integration/fixtures/documents/markdown-out.html @@ -1,5 +1,5 @@ -

      This is a heading with italic

      +

      This is a heading with italic

      @@ -11,7 +11,7 @@

      This is a heading with italic

      -

      Lists

      +

      Lists

      @@ -47,7 +47,7 @@

      Lists

      -

      Table

      +

      Table

      @@ -59,7 +59,7 @@

      Table

      -

      Quote

      +

      Quote

      @@ -73,7 +73,7 @@

      Quote

      -

      Code

      +

      Code

      diff --git a/test/integration/fixtures/documents/ms-word-out.html b/test/integration/fixtures/documents/ms-word-out.html index 9a5d5fdf557a4..0b1a3e00fe84d 100644 --- a/test/integration/fixtures/documents/ms-word-out.html +++ b/test/integration/fixtures/documents/ms-word-out.html @@ -7,11 +7,11 @@ -

      This is a heading level 1

      +

      This is a heading level 1

      -

      This is a heading level 2

      +

      This is a heading level 2

      diff --git a/test/integration/fixtures/documents/windows.html b/test/integration/fixtures/documents/windows.html index 022cc8f53b0fa..1d6125160ec08 100644 --- a/test/integration/fixtures/documents/windows.html +++ b/test/integration/fixtures/documents/windows.html @@ -1,6 +1,6 @@ -

      Heading Win

      +

      Heading Win

      diff --git a/test/integration/fixtures/documents/wordpress-out.html b/test/integration/fixtures/documents/wordpress-out.html index 03ab224fa1028..c394fa232081d 100644 --- a/test/integration/fixtures/documents/wordpress-out.html +++ b/test/integration/fixtures/documents/wordpress-out.html @@ -1,5 +1,5 @@ -

      Howdy

      +

      Howdy

      @@ -7,7 +7,7 @@

      Howdy

      -

      More tag

      +

      More tag

      @@ -15,7 +15,7 @@

      More tag

      -

      Shortcode

      +

      Shortcode

      diff --git a/test/native/__mocks__/@wordpress/react-native-aztec/index.js b/test/native/__mocks__/@wordpress/react-native-aztec/index.js index f64e93af00295..1452aeb957c87 100644 --- a/test/native/__mocks__/@wordpress/react-native-aztec/index.js +++ b/test/native/__mocks__/@wordpress/react-native-aztec/index.js @@ -9,9 +9,11 @@ import { omit } from 'lodash'; */ import { forwardRef, useImperativeHandle, useRef } from '@wordpress/element'; -// Preserve the mock of AztecInputState to be exported with the AztecView mock. +// Preserve the mock of AztecInputState and AztecKeyCodes to be exported with the AztecView mock. const AztecInputState = jest.requireActual( '@wordpress/react-native-aztec' ) .default.InputState; +const AztecKeyCodes = jest.requireActual( '@wordpress/react-native-aztec' ) + .default.KeyCodes; const UNSUPPORTED_PROPS = [ 'style' ]; @@ -50,5 +52,6 @@ const RCTAztecView = ( { accessibilityLabel, text, ...rest }, ref ) => { const AztecView = forwardRef( RCTAztecView ); AztecView.InputState = AztecInputState; +AztecView.KeyCodes = AztecKeyCodes; export default AztecView; diff --git a/test/native/integration-test-helpers/get-block.js b/test/native/integration-test-helpers/get-block.js index 5f618ddbefe8e..0770a736affbb 100644 --- a/test/native/integration-test-helpers/get-block.js +++ b/test/native/integration-test-helpers/get-block.js @@ -8,8 +8,7 @@ * @return {import('react-test-renderer').ReactTestInstance} Block instance. */ export const getBlock = ( screen, blockName, { rowIndex = 1 } = {} ) => { - const { getByLabelText } = screen; - return getByLabelText( + return screen.getAllByLabelText( new RegExp( `${ blockName } Block\\. Row ${ rowIndex }` ) - ); + )[ 0 ]; }; diff --git a/test/native/integration-test-helpers/get-inner-block.js b/test/native/integration-test-helpers/get-inner-block.js index 53536c7d2f7eb..1bb1e6938c280 100644 --- a/test/native/integration-test-helpers/get-inner-block.js +++ b/test/native/integration-test-helpers/get-inner-block.js @@ -17,7 +17,7 @@ export const getInnerBlock = ( blockName, { rowIndex = 1 } = {} ) => { - return within( parentBlock ).getByLabelText( + return within( parentBlock ).getAllByLabelText( new RegExp( `${ blockName } Block\\. Row ${ rowIndex }` ) - ); + )[ 0 ]; }; diff --git a/test/native/integration-test-helpers/initialize-editor.js b/test/native/integration-test-helpers/initialize-editor.js index 16d1372d3533f..662c6fc4544d0 100644 --- a/test/native/integration-test-helpers/initialize-editor.js +++ b/test/native/integration-test-helpers/initialize-editor.js @@ -7,8 +7,9 @@ import { v4 as uuid } from 'uuid'; /** * WordPress dependencies */ -import { initializeEditor as internalInitializeEditor } from '@wordpress/edit-post'; import { createElement, cloneElement } from '@wordpress/element'; +// eslint-disable-next-line no-restricted-imports +import { initializeEditor as internalInitializeEditor } from '@wordpress/edit-post'; /** * Internal dependencies diff --git a/test/native/setup.js b/test/native/setup.js index d65865e9da395..5f1801a338985 100644 --- a/test/native/setup.js +++ b/test/native/setup.js @@ -15,6 +15,9 @@ require( '../../packages/react-native-editor/src/globals' ); // Set up Reanimated library for testing require( 'react-native-reanimated/lib/reanimated2/jestUtils' ).setUpTests(); global.__reanimatedWorkletInit = jest.fn(); +global.ReanimatedDataMock = { + now: () => 0, +}; RNNativeModules.UIManager = RNNativeModules.UIManager || {}; RNNativeModules.UIManager.RCTView = RNNativeModules.UIManager.RCTView || {}; diff --git a/test/unit/config/matchers/to-be-positioned-popover.js b/test/unit/config/matchers/to-be-positioned-popover.js new file mode 100644 index 0000000000000..526b8de49a753 --- /dev/null +++ b/test/unit/config/matchers/to-be-positioned-popover.js @@ -0,0 +1,15 @@ +/** + * Asserts that the specified popover has already been positioned. + * Necessary because it will be positioned a bit later after it's displayed. + * + * @param {HTMLElement} element Popover element. + */ +function toBePositionedPopover( element ) { + const pass = element.classList.contains( 'is-positioned' ); + return { + pass, + message: () => `Received element is ${ pass ? '' : 'not ' }positioned`, + }; +} + +expect.extend( { toBePositionedPopover } ); diff --git a/test/unit/config/testing-library.js b/test/unit/config/testing-library.js index dd8c725c6f4ae..8f8c1bab711ad 100644 --- a/test/unit/config/testing-library.js +++ b/test/unit/config/testing-library.js @@ -1,3 +1,4 @@ require( '@testing-library/jest-dom' ); require( 'snapshot-diff/extend-expect' ); require( './matchers/to-match-style-diff-snapshot' ); +require( './matchers/to-be-positioned-popover' );