diff --git a/.eslintrc.js b/.eslintrc.js index 254e9b9bf7..eae41e4059 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -5,22 +5,47 @@ module.exports = { rules: { camelcase: 'off', 'class-methods-use-this': 'off', - 'flowtype/sort-keys': 'off', //fixme + 'consistent-return': 'off', // fixme + 'flowtype/sort-keys': 'off', // fixme + 'jsx-a11y/anchor-is-valid': 'off', // fixme + 'jsx-a11y/click-events-have-key-events': 'off', // fixme 'jsx-a11y/label-has-associated-control': 'off', - 'no-shadow': 'off', //fixme - 'react/default-props-match-prop-types': 'off', //fixme - 'react/destructuring-assignment': 'off', //fixme + 'jsx-a11y/label-has-for': 'off', // fixme + 'jsx-a11y/no-noninteractive-tabindex': 'off', // fixme + 'jsx-a11y/no-static-element-interactions': 'off', // fixme + 'jsx-a11y/no-onchange': 'off', // fixme + 'import/no-extraneous-dependencies': 'off', // fixme + 'import/prefer-default-export': 'off', // fixme + 'no-plusplus': 'off', + 'no-restricted-globals': 'off', // fixme + 'no-shadow': 'off', // fixme + 'no-underscore-dangle': 'off', // fixme + 'no-unused-vars': 'off', // fixme + 'one-var': 'off', // fixme + 'prefer-destructuring': 'off', // fixme + 'prefer-promise-reject-errors': 'off', // fixme + 'react/button-has-type': 'off', // fixme + 'react/default-props-match-prop-types': 'off', // fixme + 'react/destructuring-assignment': 'off', // fixme 'react/display-name': 'off', - 'react/jsx-sort-props': 'off', //fixme - 'react/sort-comp': 'off', //fixme - 'react/no-unused-prop-types': 'off', //fixme - 'react/no-access-state-in-setstate': 'off', //fixme + 'react/forbid-prop-types': 'off', // fixme + 'react/jsx-sort-props': 'off', // fixme + 'react/jsx-no-bind': 'off', // fixme + 'react/sort-comp': 'off', // fixme + 'react/no-unused-prop-types': 'off', // fixme + 'react/no-access-state-in-setstate': 'off', // fixme + 'react/no-array-index-key': 'off', // fixme 'react/no-this-in-sfc': 'off', - 'react/no-unused-state': 'off', //fixme + 'react/no-unused-state': 'off', // fixme + 'react/prefer-stateless-function': 'off', // fixme }, settings: { 'import/resolver': { - 'babel-module': {} - } - } + 'babel-module': {}, + }, + }, + globals: { + shallow: true, + mount: true, + }, }; diff --git a/.flowconfig b/.flowconfig index 3eade2b5d7..a638401143 100644 --- a/.flowconfig +++ b/.flowconfig @@ -13,6 +13,7 @@ .*/node_modules/draft-js* .*/node_modules/findup* .*/node_modules/config-chain* +.*/__tests__/.* [untyped] .*/node_modules/box-react-ui/* @@ -23,6 +24,9 @@ [libs] [options] +esproposal.export_star_as=enable +module.file_ext=.js +module.file_ext=.scss module.system.node.resolve_dirname=node_modules module.system.node.resolve_dirname=./src module.name_mapper.extension='scss' -> '/flow/SCSSFlowStub.js.flow' @@ -35,5 +39,7 @@ module.name_mapper='react-virtualized/dist/es/AutoSizer' -> '/flow module.name_mapper='react-virtualized/dist/es/CellMeasurer' -> '/flow/ReactVirtualizedStub.js.flow' module.name_mapper='draft-js' -> '/flow/DraftJSFlowStub.js.flow' module.name_mapper='draft-js/EditorState' -> '/flow/DraftJSFlowStub.js.flow' +module.name_mapper='draft-js/CompositeDecorator' -> '/flow/DraftJSFlowStub.js.flow' +module.name_mapper='draft-js/Modifier' -> '/flow/DraftJSFlowStub.js.flow' module.name_mapper='react-measure' -> '/flow/ReactMeasure.js.flow' suppress_comment= \\(.\\|\n\\)*\\$FlowFixMe diff --git a/.travis.yml b/.travis.yml index 5ad1e3dd5b..6e6090f727 100644 --- a/.travis.yml +++ b/.travis.yml @@ -9,6 +9,7 @@ cache: before_install: - curl -o- -L https://yarnpkg.com/install.sh | bash -s -- --version 1.13.0 - export PATH=$HOME/.yarn/bin:$PATH + - export TZ=America/Los_Angeles jobs: include: - name: "Commit Lint" @@ -17,8 +18,10 @@ jobs: script: npm-run-all clean build:i18n lint - name: "Flow Types" script: npm-run-all clean build:i18n && yarn flow - - name: "Tests" - script: yarn test && yarn test:e2e + - name: "Unit Tests" + script: yarn test + - name: "E2E Tests" + script: yarn test:e2e - name: "Build Babel" script: npm-run-all clean build:i18n build:ci:es - name: "Build Webpack" diff --git a/conf/jest/enzyme-adapter.js b/conf/jest/enzyme-adapter.js index cbebdd8812..4781788004 100644 --- a/conf/jest/enzyme-adapter.js +++ b/conf/jest/enzyme-adapter.js @@ -2,7 +2,11 @@ import 'core-js/es6/map'; import 'core-js/es6/set'; import 'raf/polyfill'; -import Enzyme from 'enzyme'; +import Enzyme, { mount, shallow } from 'enzyme'; import Adapter from 'enzyme-adapter-react-16'; Enzyme.configure({ adapter: new Adapter() }); + +// make Enzyme functions available in all test files without importing +global.shallow = shallow; +global.mount = mount; diff --git a/conf/prepush.sh b/conf/prepush.sh index e52bbbe855..bb2b75e289 100755 --- a/conf/prepush.sh +++ b/conf/prepush.sh @@ -3,55 +3,70 @@ # translation properties file PROPERTIES="i18n/en-US.properties" +# Styling variables +red=$"\e[1;31m" +green=$"\e[1;32m" +blue=$"\e[1;34m" +end=$"\e[0m\n" + # commit updated translations if any check_and_commit_updated_translations() { if ! git diff --quiet HEAD $PROPERTIES; then - echo "--------------------------------------------------------" - echo "Committing updated translations" - echo "--------------------------------------------------------" + printf "${red}--------------------------------------------------------${end}" + printf "${red}Committing updated translations${end}" + printf "${red}--------------------------------------------------------${end}" git add $PROPERTIES || exit 1 git commit --amend --no-edit --no-verify || exit 1 - echo "--------------------------------------------------------" - echo "Amended commit with translations, please push again with --no-verify" - echo "--------------------------------------------------------" + printf "${red}--------------------------------------------------------${end}" + printf "${red}Amended commit with translations, please push again with --no-verify${end}" + printf "${red}--------------------------------------------------------${end}" exit 1 + else + printf "${green}--------------------------------------------------------${end}" + printf "${green}en-US.properties is already up to date${end}" + printf "${green}--------------------------------------------------------${end}" fi } # lint, test, and build assets to update translations prepush() { - echo "--------------------------------------------------------" - echo "Build bundles" - echo "--------------------------------------------------------" + printf "${blue}-------------------------------------------------------------${end}" + printf "${blue}Building bundles${end}" + printf "${blue}-------------------------------------------------------------${end}" yarn build:i18n || exit 1 - echo "--------------------------------------------------------" - echo "Linting" - echo "--------------------------------------------------------" + printf "${blue}-------------------------------------------------------------${end}" + printf "${blue}Linting${end}" + printf "${blue}-------------------------------------------------------------${end}" yarn lint || exit 1 - echo "--------------------------------------------------------" - echo "Checking flow types" - echo "--------------------------------------------------------" + printf "${blue}-------------------------------------------------------------${end}" + printf "${blue}Checking flow types${end}" + printf "${blue}-------------------------------------------------------------${end}" yarn flow check || exit 1 - echo "--------------------------------------------------------" - echo "Testing" - echo "--------------------------------------------------------" + printf "${blue}-------------------------------------------------------------${end}" + printf "${blue}Testing${end}" + printf "${blue}-------------------------------------------------------------${end}" yarn test || exit 1 - echo "--------------------------------------------------------" - echo "Building" - echo "--------------------------------------------------------" - yarn build:dev:dist || exit 1 + printf "${blue}-------------------------------------------------------------${end}" + printf "${blue}Building all sources, this will update i18n/json${end}" + printf "${blue}-------------------------------------------------------------${end}" + yarn build:prod:es || exit 1 + + printf "${blue}-------------------------------------------------------------${end}" + printf "${blue}Building bundles again, this will update en-US.properties${end}" + printf "${blue}-------------------------------------------------------------${end}" + yarn build:i18n || exit 1 check_and_commit_updated_translations } # Execute this script if ! prepush; then - echo "----------------------------------------------------" - echo "Error: failure in prepush script" - echo "----------------------------------------------------" + printf "${red}---------------------------------------------------------${end}" + printf "${red}Error: failure in prepush script${end}" + printf "${red}---------------------------------------------------------${end}" exit 1 fi diff --git a/conf/styleguide.config.js b/conf/styleguide.config.js index d69cd59195..012f5b4ee2 100644 --- a/conf/styleguide.config.js +++ b/conf/styleguide.config.js @@ -1,22 +1,244 @@ const path = require('path'); const webpackConf = require('./webpack.config.js'); +const webpackConfig = Array.isArray(webpackConf) ? webpackConf[0] : webpackConf; + +const allSections = [ + { + name: 'Elements', + components: () => [ + '../src/elements/content-picker/ContentPicker.js', + '../src/elements/content-explorer/ContentExplorer.js', + '../src/elements/content-uploader/ContentUploader.js', + '../src/elements/content-sidebar/ContentSidebar.js', + '../src/elements/content-preview/ContentPreview.js', + '../src/elements/content-open-with/ContentOpenWith.js', + ], + }, + { + name: 'Colors', + content: '../examples/colors.md', + }, + { + name: 'Typography', + content: '../examples/typography.md', + }, + { + name: 'Components', + components: () => [ + '../src/components/avatar/Avatar.js', + '../src/components/badge/Badge.js', + '../src/components/badgeable/Badgeable.js', + '../src/components/breadcrumb/Breadcrumb.js', + '../src/components/button/Button.js', + '../src/components/button-group/ButtonGroup.js', + '../src/components/checkbox/Checkbox.js', + '../src/components/collapsible/Collapsible.js', + '../src/components/context-menu/ContextMenu.js', + '../src/components/count-badge/CountBadge.js', + '../src/components/datalist-item/DatalistItem.js', + '../src/components/date-picker/DatePicker.js', + '../src/components/draggable-list/DraggableList.js', + '../src/components/dropdown-menu/DropdownMenu.js', + '../src/components/error-mask/ErrorMask.js', + '../src/components/flyout/Flyout.js', + '../src/components/flyout/Overlay.js', + '../src/components/focus-trap/FocusTrap.js', + '../src/components/header/Header.js', + '../src/components/hotkeys/HotkeyFriendlyModal.js', + '../src/components/hotkeys/HotkeyLayer.js', + '../src/components/hotkeys/Hotkeys.js', + '../src/components/infinite-scroll/InfiniteScroll.js', + '../src/components/inline-error/InlineError.js', + '../src/components/inline-notice/InlineNotice.js', + '../src/components/label/Label.js', + '../src/components/link/Link.js', + '../src/components/link/LinkButton.js', + '../src/components/link/LinkGroup.js', + '../src/components/link/LinkPrimaryButton.js', + '../src/components/loading-indicator/LoadingIndicator.js', + '../src/components/loading-indicator/LoadingIndicatorWrapper.js', + '../src/components/logo/Logo.js', + '../src/components/menu/Menu.js', + '../src/components/menu/SelectMenuLinkItem.js', + '../src/components/modal/Modal.js', + '../src/components/modal/ModalActions.js', + '../src/components/modal/ModalDialog.js', + '../src/components/nav-sidebar/NavSidebar.js', + '../src/components/notification/Notification.js', + '../src/components/notification/NotificationsWrapper.js', + '../src/components/pill-cloud/PillCloud.js', + '../src/components/pill-selector-dropdown/PillSelectorDropdown.js', + '../src/components/plain-button/PlainButton.js', + '../src/components/primary-button/PrimaryButton.js', + '../src/components/progress-bar/ProgressBar.js', + '../src/components/radar/RadarAnimation.js', + '../src/components/radio/RadioButton.js', + '../src/components/radio/RadioGroup.js', + '../src/components/time/ReadableTime.js', + '../src/components/scroll-wrapper/ScrollWrapper.js', + '../src/components/search-form/SearchForm.js', + '../src/components/section/Section.js', + '../src/components/select/Select.js', + '../src/components/select-field/MultiSelectField.js', + '../src/components/select-field/SingleSelectField.js', + '../src/components/selector-dropdown/SelectorDropdown.js', + '../src/components/slide-carousel/SlideCarousel.js', + '../src/components/tab-view/TabView.js', + '../src/components/table/Table.js', + '../src/components/text-area/TextArea.js', + '../src/components/text-input/TextInput.js', + '../src/components/text-input-with-copy-button/TextInputWithCopyButton.js', + '../src/components/thumbnail-card/ThumbnailCard.js', + '../src/components/toggle/Toggle.js', + '../src/components/tooltip/Tooltip.js', + ], + }, + { + name: 'Icons', + components: () => [ + '../src/icons/item-icon/ItemIcon.js', + '../src/icons/folder-icon/FolderIcon.js', + '../src/icons/file-icon/FileIcon.js', + '../src/icons/bookmark-icon/BookmarkIcon.js', + '../src/icons/adobe-sign/IconAdobeSign.js', + '../src/icons/google-docs/GoogleDocsIcon.js', + ], + sections: [ + { + name: 'Annotations', + components: '../src/icons/annotations/[A-Z]*.js', + }, + { + name: 'Avatar', + components: '../src/icons/avatars/[A-Z]*.js', + }, + { + name: 'Badges', + components: '../src/icons/badges/[A-Z]*.js', + }, + { + name: 'Box Tools', + components: '../src/icons/box-tools/[A-Z]*.js', + }, + { + name: 'General', + components: '../src/icons/general/[A-Z]*.js', + }, + { + name: 'File', + components: '../src/icons/file/[A-Z]*.js', + }, + { + name: 'Folder', + components: '../src/icons/folder/[A-Z]*.js', + }, + { + name: 'Illustrations', + components: '../src/icons/illustrations/[A-Z]*.js', + }, + { + name: 'iWork', + components: () => [ + '../src/icons/iwork/IconIWorkTrio.js', + '../src/icons/iwork/IWorkIcon.js', + '../src/icons/iwork/IWorkDesktopIcon.js', + ], + }, + { + name: 'Microsoft Office', + components: () => [ + '../src/icons/microsoft-office/IconOfficeWordmark.js', + '../src/icons/microsoft-office/OfficeOnlineIcon.js', + '../src/icons/microsoft-office/OfficeDesktopIcon.js', + ], + }, + { + name: 'Placeholders', + components: '../src/icons/placeholders/[A-Z]*.js', + }, + { + name: 'States', + components: '../src/icons/states/[A-Z]*.js', + }, + { + name: 'Metadata View', + components: '../src/icons/metadata-view/[A-Z]*.js', + }, + ], + }, + { + name: 'Form Elements', + components: () => [ + '../src/components/form-elements/draft-js-mention-selector/DraftJSMentionSelector.js', + '../src/components/form-elements/form/Form.js', + '../src/components/form-elements/text-area/TextArea.js', + '../src/components/form-elements/text-input/TextInput.js', + ], + }, + { + name: 'Features', + components: () => [ + '../src/features/activity-feed/activity-feed/ActivityFeed.js', + '../src/features/in-app-message/create-message-form/CreateMessageForm.js', + '../src/features/in-app-message/targeting-expression-editor/TargetingExpressionEditor.js', + '../src/features/invite-collaborators-modal/InviteCollaboratorsModal.js', + '../src/features/left-sidebar/LeftSidebar.js', + '../src/features/header-flyout/HeaderFlyout.js', + '../src/features/presence/Presence.js', + '../src/features/presence/PresenceLink.js', + '../src/features/security-cloud-game/SecurityCloudGame.js', + '../src/features/share/ShareMenu.js', + '../src/features/shared-link-modal/SharedLink.js', + '../src/features/shared-link-modal/SharedLinkModal.js', + '../src/features/shared-link-settings-modal/SharedLinkSettingsModal.js', + '../src/features/unified-share-modal/UnifiedShareModal.js', + '../src/features/version-history-modal/VersionHistoryModal.js', + ], + sections: [ + { + name: 'Metadata', + components: () => [ + '../src/features/metadata-instance-editor/MetadataInstanceEditor.js', + '../src/features/metadata-view/MetadataView.js', + ], + }, + { + name: 'Content Explorer', + components: () => [ + '../src/features/content-explorer/content-explorer-modal-container/ContentExplorerModalContainer.js', + '../src/features/content-explorer/content-explorer/ContentExplorer.js', + '../src/features/content-explorer/new-folder-modal/NewFolderModal.js', + ], + }, + { + name: 'Item Details', + components: () => [ + '../src/features/item-details/VersionHistoryLink.js', + '../src/features/item-details/ItemExpirationNotice.js', + '../src/features/item-details/SharedLinkExpirationNotice.js', + '../src/features/item-details/ItemProperties.js', + ], + }, + { + name: 'Access Stats', + components: () => ['../src/features/access-stats/AccessStats.js'], + }, + { + name: 'Quick Search', + components: () => [ + '../src/features/quick-search/QuickSearch.js', + '../src/features/quick-search/QuickSearchItem.js', + ], + }, + ], + }, +]; + module.exports = { - webpackConfig: Array.isArray(webpackConf) ? webpackConf[0] : webpackConf, + webpackConfig, styleguideDir: path.join(__dirname, '../styleguide'), - sections: [ - { - name: 'Elements', - components: () => [ - '../src/elements/content-picker/ContentPicker.js', - '../src/elements/content-explorer/ContentExplorer.js', - '../src/elements/content-uploader/ContentUploader.js', - '../src/elements/content-sidebar/ContentSidebar.js', - '../src/elements/content-preview/ContentPreview.js', - '../src/elements/content-open-with/ContentOpenWith.js', - ], - }, - ], + sections: allSections, title: 'Box UI Elements', theme: { color: { diff --git a/conf/webpack.config.js b/conf/webpack.config.js index 569e5e1283..1cacc63975 100644 --- a/conf/webpack.config.js +++ b/conf/webpack.config.js @@ -15,6 +15,7 @@ const isRelease = process.env.NODE_ENV === 'production'; const isDev = process.env.NODE_ENV === 'dev'; const language = process.env.LANGUAGE; const react = process.env.REACT === 'true'; +const examples = process.env.EXAMPLES === 'true'; const token = process.env.TOKEN; // used for examples only const folderId = process.env.FOLDERID; // used for examples only const fileId = process.env.FILEID; // used for examples only @@ -50,11 +51,13 @@ function getConfig(isReactExternalized) { resolve: { modules: ['src', 'node_modules'], alias: { - examples: path.join(__dirname, '../examples/src'), + 'box-ui-elements/es': path.join(__dirname, '../src'), // for examples only + examples: path.join(__dirname, '../examples/src'), // for examples only 'react-intl-locale-data': path.resolve(`node_modules/react-intl/locale-data/${locale}`), 'box-ui-elements-locale-data': path.resolve(`i18n/${language}`), 'box-react-ui-locale-data': path.resolve(`node_modules/box-react-ui/i18n/${language}`), moment: path.resolve('src/utils/MomentShim'), // Hack to leverage Intl instead + 'rsg-components/Wrapper': path.join(__dirname, '../examples/Wrapper'), // for examples only }, }, devServer: { @@ -114,12 +117,14 @@ function getConfig(isReactExternalized) { if (isDev) { config.devtool = 'source-map'; config.plugins.push(Translations); - config.plugins.push( - new CircularDependencyPlugin({ - exclude: /node_modules/, - failOnError: true, - }), - ); + if (!examples) { + config.plugins.push( + new CircularDependencyPlugin({ + exclude: /node_modules/, + failOnError: true, + }), + ); + } } if (isRelease && language === 'en-US') { diff --git a/examples/Wrapper.js b/examples/Wrapper.js new file mode 100644 index 0000000000..bb810753fe --- /dev/null +++ b/examples/Wrapper.js @@ -0,0 +1,22 @@ +// @flow +/** + * Wrapper component for styleguidist examples + */ +// $FlowFixMe +import 'core-js'; // For IE11 +import * as React from 'react'; +import { IntlProvider } from 'react-intl'; + +import './styles/styles.scss'; + +type Props = { + children: React.Node, +}; + +const Wrapper = ({ children }: Props) => ( + + {children} + +); + +export default Wrapper; diff --git a/examples/colors.md b/examples/colors.md new file mode 100644 index 0000000000..a768df3d1a --- /dev/null +++ b/examples/colors.md @@ -0,0 +1,111 @@ +**Box brand blues** + +``` +
+
+
+ +
+
+
+ +
+
+``` + +**Secondary Colors** + +``` +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+``` + +**Tinted Grays** + +``` +
+
+
+ +
+
+
+ +
+
+
+ +
+
+
+ +
+
+``` + +**Neutral Grays** + +``` +
+
+
+ +
+
+
+ +
+
+
+ +
+
+``` diff --git a/examples/src/AvatarImageExample.js b/examples/src/AvatarImageExample.js new file mode 100644 index 0000000000..5e402727c5 --- /dev/null +++ b/examples/src/AvatarImageExample.js @@ -0,0 +1,10 @@ +import React from 'react'; + +import Avatar from 'components/avatar'; + +const LevieFace = + ''; + +const AvatarImageExample = () => ; + +export default AvatarImageExample; diff --git a/examples/src/ContentExplorerExamples.js b/examples/src/ContentExplorerExamples.js new file mode 100644 index 0000000000..5f02133ddc --- /dev/null +++ b/examples/src/ContentExplorerExamples.js @@ -0,0 +1,133 @@ +import React, { Component } from 'react'; + +import { ContentExplorer } from 'features/content-explorer'; + +const INITIAL_FOLDERS_PATH = [ + { + id: '0', + name: 'All Files', + }, +]; + +const ITEMS = { + 0: [ + { + id: '1', + name: "Can't choose but can drill in", + type: 'folder', + hasCollaborations: true, + isActionDisabled: true, + }, + { + id: '2', + name: 'Choose me or drill in', + type: 'folder', + }, + { + id: '3', + name: 'File.boxnote', + type: 'file', + extension: 'boxnote', + isDisabled: true, + }, + { + id: '4', + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: true, + }, + ], + 1: [ + { + id: '11', + name: 'Choose me or drill in', + type: 'folder', + }, + { + id: '12', + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: true, + }, + { + id: '13', + name: 'File.ppt', + type: 'file', + extension: 'ppt', + isDisabled: true, + }, + ], + search: [ + { + id: '1', + name: 'Choose me or drill in', + type: 'folder', + }, + ], +}; + +class ContentExplorerExamples extends Component { + constructor(props) { + super(props); + this.state = { + exampleMessage: 'Content Explorer is configured to select a folder.', + folderId: '0', + }; + } + + handleEnterFolder = (enteredFolder, newFoldersPath) => { + this.setState({ + exampleMessage: `Enter folder { id: ${enteredFolder.id}, name: ${enteredFolder.name} }`, + folderId: enteredFolder.id, + }); + }; + + handleChooseItems = chosenItems => { + this.setState({ + exampleMessage: chosenItems.reduce( + (message, chosenItem) => `${message}, Choose { id: ${chosenItem.id}, name: ${chosenItem.name} }`, + '', + ), + }); + }; + + handleSearchSubmit = searchQuery => { + this.setState({ + exampleMessage: `Searched for '${searchQuery}'`, + folderId: 'search', + }); + }; + + handleExitSearch = ({ id }) => { + this.setState({ + exampleMessage: 'Exited search', + folderId: id, + }); + }; + + render() { + const { exampleMessage, folderId } = this.state; + const items = ITEMS[folderId] || []; + + return ( +
+ +
{exampleMessage}
+
+ ); + } +} + +export default ContentExplorerExamples; diff --git a/examples/src/ContentExplorerModalContainerExamples.js b/examples/src/ContentExplorerModalContainerExamples.js new file mode 100644 index 0000000000..496f91abea --- /dev/null +++ b/examples/src/ContentExplorerModalContainerExamples.js @@ -0,0 +1,175 @@ +import React, { Component } from 'react'; + +import Button from 'components/button'; +import { ContentExplorerModalContainer } from 'features/content-explorer'; + +const INITIAL_FOLDERS_PATH = [ + { + id: '0', + name: 'All Files', + }, +]; + +const ITEMS = { + 0: [ + { + id: '1', + name: "Can't choose but can drill in", + type: 'folder', + hasCollaborations: true, + isActionDisabled: true, + }, + { + id: '2', + name: 'Choose me or drill in', + type: 'folder', + }, + { + name: 'File.boxnote', + type: 'file', + extension: 'boxnote', + isDisabled: true, + }, + { + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: true, + }, + ], + 1: [ + { + id: '3', + name: 'Choose me or drill in', + type: 'folder', + }, + { + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: true, + }, + { + name: 'File.ppt', + type: 'file', + extension: 'ppt', + isDisabled: true, + }, + ], + search: [ + { + id: '4', + name: 'Choose me or drill in', + type: 'folder', + }, + ], +}; + +class ContentExplorerModalContainerExamples extends Component { + constructor(props) { + super(props); + this.state = { + exampleMessage: 'Content Explorer is configured to move/copy a file.', + folderId: '0', + foldersPath: INITIAL_FOLDERS_PATH, + isModalOpen: false, + }; + } + + openModal = () => { + this.setState({ isModalOpen: true }); + }; + + closeModal = () => { + this.setState({ + folderId: '0', + foldersPath: INITIAL_FOLDERS_PATH, + isModalOpen: false, + }); + }; + + handleCreateFolderSubmit = folderName => { + const { folderId: currentFolderId, foldersPath } = this.state; + + // Add folder to list of items + const folderId = `${Date.now()}`; + const newFolder = { + id: folderId, + name: folderName, + type: 'folder', + }; + ITEMS[currentFolderId] = ITEMS[currentFolderId] || []; + ITEMS[currentFolderId].push(newFolder); + + // Drill into that folder + this.setState({ + folderId, + foldersPath: foldersPath.concat([newFolder]), + }); + }; + + handleEnterFolder = (enteredFolder, newFoldersPath) => { + this.setState({ folderId: enteredFolder.id }); + }; + + handleMoveItem = destFolder => { + this.setState({ + exampleMessage: `Moved item to { id: ${destFolder.id}, name: ${destFolder.name} }`, + }); + this.closeModal(); + }; + + handleCopyItem = destFolder => { + this.setState({ + exampleMessage: `Copied item to { id: ${destFolder.id}, name: ${destFolder.name} }`, + }); + this.closeModal(); + }; + + handleSearchSubmit = searchQuery => { + this.setState({ + folderId: 'search', + }); + }; + + handleExitSearch = ({ id }) => { + this.setState({ + folderId: id, + }); + }; + + render() { + const { exampleMessage, folderId, foldersPath, isModalOpen } = this.state; + const items = ITEMS[folderId] || []; + + return ( +
+ +
{exampleMessage}
+ {isModalOpen && ( + {}} + listWidth={600} + listHeight={350} + /> + )} +
+ ); + } +} + +export default ContentExplorerModalContainerExamples; diff --git a/examples/src/ContentExplorerMultiSelectExamples.js b/examples/src/ContentExplorerMultiSelectExamples.js new file mode 100644 index 0000000000..7fdb3d368a --- /dev/null +++ b/examples/src/ContentExplorerMultiSelectExamples.js @@ -0,0 +1,189 @@ +import React, { Component } from 'react'; + +import { ContentExplorer } from 'features/content-explorer'; + +import './ContentExplorerMultiSelectExamples.scss'; + +const INITIAL_FOLDERS_PATH = [ + { + id: '0', + name: 'All Files', + }, +]; + +const ITEMS = { + 0: [ + { + id: '1', + name: 'Choose me or drill in', + type: 'folder', + hasCollaborations: true, + isActionDisabled: true, + }, + { + id: '2', + name: 'Choose me or drill in 2', + type: 'folder', + }, + { + id: '3', + name: 'File.boxnote', + type: 'file', + extension: 'boxnote', + isDisabled: false, + }, + { + id: '4', + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: false, + }, + { + id: '5', + name: 'Choose me or drill with loading items', + type: 'folder', + }, + { + id: '6', + name: 'File 2.docx', + type: 'file', + extension: 'docx', + isDisabled: true, + }, + ], + 1: [ + { + id: '11', + name: 'Choose me or drill in', + type: 'folder', + }, + { + id: '12', + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: false, + }, + { + id: '13', + name: 'File.ppt', + type: 'file', + extension: 'ppt', + isDisabled: false, + }, + ], + 5: [ + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + ], + search: [ + { + id: '2', + name: 'Choose me or drill in', + type: 'folder', + }, + ], + userFilter: [ + { + id: '30', + name: 'Folder owned by the filtered user', + type: 'folder', + }, + { + id: '31', + name: 'File owned by the filtered user', + type: 'file', + }, + ], +}; + +class ContentExplorerMultiSelectExamples extends Component { + constructor(props) { + super(props); + this.state = { + exampleMessage: 'Content Explorer is configured for multi select.', + folderId: '0', + user: null, + }; + } + + handleEnterFolder = (enteredFolder, newFoldersPath) => { + this.setState({ + exampleMessage: `Enter folder { id: ${enteredFolder.id}, name: ${enteredFolder.name} }`, + folderId: enteredFolder.id, + }); + }; + + handleChooseItems = chosenItems => { + this.setState({ + exampleMessage: chosenItems.reduce( + (message, chosenItem) => `${message}, Choose { id: ${chosenItem.id}, name: ${chosenItem.name} }`, + '', + ), + }); + }; + + handleSearchSubmit = searchQuery => { + this.setState({ + exampleMessage: `Searched for '${searchQuery}'`, + folderId: 'search', + }); + }; + + handleExitSearch = ({ id }) => { + this.setState({ + exampleMessage: 'Exited search', + folderId: id, + }); + }; + + handleOnItemAdd = newItem => { + this.setState({ + exampleMessage: `User Filtered: ${newItem.name}`, + folderId: 'userFilter', + }); + }; + + handleOnItemRemove = (item, index) => { + this.setState({ + exampleMessage: 'User Filtered Removed', + folderId: '0', + }); + }; + + renderHeaderActionsAccessory = () => { + return
Accessory
; + }; + + render() { + const { exampleMessage, folderId } = this.state; + const items = ITEMS[folderId] || []; + + return ( +
+ +
{exampleMessage}
+
+ ); + } +} + +export default ContentExplorerMultiSelectExamples; diff --git a/examples/src/ContentExplorerMultiSelectExamples.scss b/examples/src/ContentExplorerMultiSelectExamples.scss new file mode 100644 index 0000000000..9903470d2b --- /dev/null +++ b/examples/src/ContentExplorerMultiSelectExamples.scss @@ -0,0 +1,9 @@ +.content-explorer-example { + .selector-dropdown-wrapper { + top: 7px; + } + + .pill-selector-wrapper div.pill-selector-input-wrapper { + width: 160px; + } +} diff --git a/examples/src/ContentExplorerMultiSelectModalContainerExamples.js b/examples/src/ContentExplorerMultiSelectModalContainerExamples.js new file mode 100644 index 0000000000..8c83280cb4 --- /dev/null +++ b/examples/src/ContentExplorerMultiSelectModalContainerExamples.js @@ -0,0 +1,199 @@ +import React, { Component } from 'react'; + +import Button from 'components/button'; +import { ContentExplorerModalContainer } from 'features/content-explorer'; + +import './ContentExplorerMultiSelectModalContainerExamples.scss'; + +const INITIAL_FOLDERS_PATH = [ + { + id: '0', + name: 'All Files', + }, +]; + +const ITEMS = { + 0: [ + { + id: '1', + name: 'Choose me or drill in', + type: 'folder', + hasCollaborations: true, + isActionDisabled: true, + }, + { + id: '2', + name: 'Choose me or drill in 2', + type: 'folder', + }, + { + id: '3', + name: 'File.boxnote', + type: 'file', + extension: 'boxnote', + isDisabled: false, + }, + { + id: '4', + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: false, + }, + { + id: '5', + name: 'Choose me or drill with loading items', + type: 'folder', + }, + { + id: '6', + name: 'File 2.docx', + type: 'file', + extension: 'docx', + isDisabled: true, + }, + ], + 1: [ + { + id: '11', + name: 'Choose me or drill in', + type: 'folder', + }, + { + id: '12', + name: 'File.docx', + type: 'file', + extension: 'docx', + isDisabled: false, + }, + { + id: '13', + name: 'File.ppt', + type: 'file', + extension: 'ppt', + isDisabled: false, + }, + ], + 5: [ + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + { isLoading: true }, + ], + search: [ + { + id: '2', + name: 'Choose me or drill in', + type: 'folder', + }, + ], + userFilter: [ + { + id: '30', + name: 'Folder owned by the filtered user', + type: 'folder', + }, + { + id: '31', + name: 'File owned by the filtered user', + type: 'file', + }, + ], +}; + +class ContentExplorerMultiSelectModalContainerExamples extends Component { + constructor(props) { + super(props); + this.state = { + exampleMessage: 'Content Explorer is configured to selevt multiple file / folder.', + folderId: '0', + foldersPath: INITIAL_FOLDERS_PATH, + isModalOpen: false, + }; + } + + openModal = () => { + this.setState({ isModalOpen: true }); + }; + + closeModal = () => { + this.setState({ + folderId: '0', + foldersPath: INITIAL_FOLDERS_PATH, + isModalOpen: false, + }); + }; + + handleEnterFolder = (enteredFolder, newFoldersPath) => { + this.setState({ folderId: enteredFolder.id }); + }; + + handleSearchSubmit = searchQuery => { + this.setState({ + folderId: 'search', + }); + }; + + handleExitSearch = ({ id }) => { + this.setState({ + folderId: id, + }); + }; + + handleOnItemAdd = newItem => { + this.setState({ + exampleMessage: `User Filtered: ${newItem.name}`, + folderId: 'userFilter', + }); + }; + + handleOnItemRemove = (item, index) => { + this.setState({ + exampleMessage: 'User Filtered Removed', + folderId: '0', + }); + }; + + renderHeaderActionsAccessory = () => { + return
Accessory
; + }; + + render() { + const { exampleMessage, folderId, foldersPath, isModalOpen } = this.state; + const items = ITEMS[folderId] || []; + + return ( +
+ +
{exampleMessage}
+ {isModalOpen && ( + {}} + listWidth={600} + listHeight={350} + /> + )} +
+ ); + } +} + +export default ContentExplorerMultiSelectModalContainerExamples; diff --git a/examples/src/ContentExplorerMultiSelectModalContainerExamples.scss b/examples/src/ContentExplorerMultiSelectModalContainerExamples.scss new file mode 100644 index 0000000000..38a7e03285 --- /dev/null +++ b/examples/src/ContentExplorerMultiSelectModalContainerExamples.scss @@ -0,0 +1,9 @@ +.content-explorer-modal { + .selector-dropdown-wrapper { + top: 7px; + } + + .pill-selector-wrapper div.pill-selector-input-wrapper { + width: 160px; + } +} diff --git a/examples/src/ContextMenuWithSubmenuWithBoundariesElementExample.js b/examples/src/ContextMenuWithSubmenuWithBoundariesElementExample.js new file mode 100644 index 0000000000..e21a14c925 --- /dev/null +++ b/examples/src/ContextMenuWithSubmenuWithBoundariesElementExample.js @@ -0,0 +1,60 @@ +/* eslint-disable no-unused-expressions */ +import * as React from 'react'; + +import { Menu, SubmenuItem, MenuItem } from 'components/menu'; +import ContextMenu from 'components/context-menu'; + +class ContextMenuWithSubmenuWithBoundariesElementExample extends React.Component { + state = { + rightBoundaryElement: null, + bottomBoundaryElement: null, + }; + + render() { + return ( + +
{ + !this.state.rightBoundaryElement && + this.setState({ + rightBoundaryElement: ref, + }); + }} + > + Target Component - right click me +
+ { + !this.state.bottomBoundaryElement && + this.setState({ + bottomBoundaryElement: ref, + }); + }} + > + View Profile + View Profile + {this.state.rightBoundaryElement && ( + + Submenu + + View Profile + View Profile + View Profile + + + )} + Help + +
+ ); + } +} + +export default ContextMenuWithSubmenuWithBoundariesElementExample; diff --git a/examples/src/DraftJSMentionSelectorExamples.js b/examples/src/DraftJSMentionSelectorExamples.js new file mode 100644 index 0000000000..499abb57a2 --- /dev/null +++ b/examples/src/DraftJSMentionSelectorExamples.js @@ -0,0 +1,123 @@ +import PropTypes from 'prop-types'; +import React, { Component } from 'react'; +import { ContentState, EditorState } from 'draft-js'; + +import Avatar from 'components/avatar'; +import Section from 'components/section'; +import PrimaryButton from 'components/primary-button'; +import { DraftMentionDecorator } from 'components/form-elements/draft-js-mention-selector'; + +import MentionSelectorContainer from './MentionSelectorContainer'; + +const CustomSelectorRow = ({ item }) => ( +
+
+ +
+ {item.name} +
+); + +CustomSelectorRow.propTypes = { + item: PropTypes.object, +}; + +class DraftJSMentionSelectorExamples extends Component { + constructor(props) { + super(props); + + this.state = { + exampleExternalEditorState: this.initialEditorState, + }; + } + + onExternalEditorStateChange = newEditorState => { + this.setState({ exampleExternalEditorState: newEditorState }); + }; + + getMentionReplacement(mention) { + return { + component: ({ id, name, ...rest }) => { + const style = { + display: 'inline-block', + verticalAlign: 'middle', + }; + return ( +
+ +
+ ); + }, + data: { + ...mention, + }, + serializedText: () => { + return `@[${mention.id}:${mention.name}]`; + }, + text: '', + }; + } + + initialEditorState = EditorState.createWithContent( + ContentState.createFromText('Is this thing on?'), + DraftMentionDecorator, + ); + + initializeEditorState = () => { + this.setState({ exampleExternalEditorState: this.initialEditorState }); + }; + + render() { + const { exampleExternalEditorState } = this.state; + + return ( +
+

Mention Selector

+
+ + Reset State +
+
+ +
+
+ !event.shiftKey} + /> +
+
+ +
+
+ { + console.log('a change: ', x.toJS()); // eslint-disable-line + }} + onFocus={() => { + console.log('the draft mention selector was focused'); // eslint-disable-line + }} + /> +
+
+ ); + } +} + +export default DraftJSMentionSelectorExamples; diff --git a/examples/src/DraggableListExamples.js b/examples/src/DraggableListExamples.js new file mode 100644 index 0000000000..aec34abfb0 --- /dev/null +++ b/examples/src/DraggableListExamples.js @@ -0,0 +1,75 @@ +// @flow +import React, { Component } from 'react'; +import uniqueId from 'lodash/uniqueId'; + +import Checkbox from '../../src/components/checkbox/Checkbox'; +import DraggableList from '../../src/components/draggable-list'; +import DraggableListItem from '../../src/components/draggable-list/DraggableListItem'; + +import reorder from '../../src/components/draggable-list/draggable-list-utils/reorder'; + +import '../styles/DraggableListExamples.scss'; + +type Props = { + isDraggableViaHandle?: boolean, +}; + +type State = { + items: Array, + listId: string, +}; + +class DraggableListExamples extends Component { + state = { + items: [], + listId: '', + }; + + componentDidMount() { + this.setState({ + items: this.getItems(10), + listId: uniqueId(), + }); + } + + getItems = (count: number): Array => { + return Array.from({ length: count }, (v, k) => k).map(k => ({ + id: uniqueId('item_'), + label: `item ${k}`, + })); + }; + + onDragEnd = (sourceIndex: number, destinationIndex: number) => { + if (!destinationIndex) { + return; + } + + const items = reorder(this.state.items, sourceIndex, destinationIndex); + + this.setState({ + items, + }); + }; + + render() { + const { isDraggableViaHandle } = this.props; + const { items, listId } = this.state; + + return ( + + {items.map((item, index) => ( + + + + ))} + + ); + } +} + +export default DraggableListExamples; diff --git a/examples/src/MentionSelectorContainer.js b/examples/src/MentionSelectorContainer.js new file mode 100644 index 0000000000..98a5ce1547 --- /dev/null +++ b/examples/src/MentionSelectorContainer.js @@ -0,0 +1,63 @@ +import React, { Component } from 'react'; + +import DraftJSMentionSelector from 'components/form-elements/draft-js-mention-selector'; + +const contactData = [ + { email: 'ken@streetfighter.com', id: '1', name: 'Ken' }, + { email: 'ryu@streetfighter.com', id: '2', name: 'Ryu' }, + { email: 'guile@streetfighter.com', id: '3', name: 'Guile' }, + { email: 'm.bison@streetfighter.com', id: '4', name: 'M. Bison' }, + { email: 'vega@streetfighter.com', id: '5', name: 'Vega' }, + { email: 'blanka@streetfighter.com', id: '6', name: 'Blanka' }, + { email: 'akuma@streetfighter.com', id: '7', name: 'Akuma' }, + { email: 'りゅう@streetfighter.com', id: '8', name: 'りゅう' }, + { email: 'リュー@streetfighter.com', id: '9', name: 'リュー' }, +]; + +class MentionSelectorContainer extends Component { + constructor(props) { + super(props); + this.state = { + contacts: [], + }; + } + + isMatchingContact(mentionString, contact) { + const { email, name } = contact; + return email.toLowerCase().indexOf(mentionString) > -1 || name.toLowerCase().indexOf(mentionString) > -1; + } + + handleMention = mentionString => { + if (!mentionString.length) { + return; + } + const matchingContacts = contactData.reduce((prev, contact) => { + return this.isMatchingContact(mentionString, contact) ? prev.concat([contact]) : prev; + }, []); + + this.setState({ + contacts: matchingContacts, + }); + }; + + render() { + const { ...props } = this.props; + const { contacts } = this.state; + + return ( + ({ + id: contact.id, + name: contact.name, + item: contact, + }))} + label="Comment" + onMention={this.handleMention} + {...props} + /> + ); + } +} + +export default MentionSelectorContainer; diff --git a/examples/src/MetadataExamples.js b/examples/src/MetadataExamples.js new file mode 100644 index 0000000000..1e7255ff9b --- /dev/null +++ b/examples/src/MetadataExamples.js @@ -0,0 +1,1108 @@ +import * as React from 'react'; +import uniqueId from 'lodash/uniqueId'; + +import MetadataInstanceEditor from 'features/metadata-instance-editor/MetadataInstanceEditor'; +import LoadingIndicator from 'components/loading-indicator'; + +// Templates +const template0 = { + id: 'template0', + templateKey: 'template0', + displayName: 'template0 simple', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'somefieldkey', + displayName: 'Description of the field', + }, + ], +}; + +const template1 = { + id: 'template1', + templateKey: 'template1', + displayName: 'template1 title', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + { + id: 'field9', + type: 'multiSelect', + key: 'multiselectfield', + displayName: 'Multi-Select Field', + description: 'example of a multi-select field', + options: [ + { key: 'yes' }, + { key: 'no' }, + { key: 'maybe' }, + { key: 'idk?' }, + { key: 'oh well' }, + { key: 'whatever' }, + ], + }, + { + id: 'field10', + type: 'multiSelect', + key: 'emptymultiselectfield', + displayName: 'Empty Multi-Select Field', + description: 'example of an empty multi-select field', + options: [ + { key: 'yes' }, + { key: 'no' }, + { key: 'maybe' }, + { key: 'idk?' }, + { key: 'oh well' }, + { key: 'whatever' }, + ], + }, + { + id: 'field11', + type: 'integer', + key: 'integerfield', + displayName: 'Integer Field', + description: 'example of an integer field', + }, + { + id: 'field12', + type: 'integer', + key: 'emptyintegerfield', + displayName: 'Empty Integer Field', + description: 'example of an empty integer field', + }, + { + id: 'field13', + type: 'string', + key: + 'longlonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + displayName: + 'longlonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + description: + 'longlonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + }, + { + id: 'field14', + type: 'string', + key: + 'long long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + displayName: + 'long long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + description: + 'long long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + }, + { + id: 'field15', + type: 'enum', + key: + 'enumlonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + displayName: + 'enumlonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + description: + 'enumlonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + options: [ + { key: 'yes' }, + { key: 'no' }, + { key: 'maybe' }, + { key: 'idk?' }, + { key: 'oh well' }, + { key: 'whatever' }, + ], + }, + { + id: 'field16', + type: 'multiSelect', + key: + 'multiselectlonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + displayName: + 'multiSelectlonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + description: + 'multiSelectlonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglonglongverylonglonglonglonglonglonglonglonglonglonglonglonglonglonglong', + options: [ + { key: 'yes' }, + { key: 'no' }, + { key: 'maybe' }, + { key: 'idk?' }, + { key: 'oh well' }, + { key: 'whatever' }, + ], + }, + { + id: 'field17', + type: 'enum', + key: + 'enumlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + displayName: + 'enumlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + description: + 'enumlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + options: [ + { key: 'yes' }, + { key: 'no' }, + { key: 'maybe' }, + { key: 'idk?' }, + { key: 'oh well' }, + { key: 'whatever' }, + ], + }, + { + id: 'field18', + type: 'multiSelect', + key: + 'multiSelectlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + displayName: + 'multiSelectlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + description: + 'multiSelectlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long longlong long long', + options: [ + { key: 'yes' }, + { key: 'no' }, + { key: 'maybe' }, + { key: 'idk?' }, + { key: 'oh well' }, + { key: 'whatever' }, + ], + }, + { + id: 'field19', + type: 'badfieldtype', + key: 'badfieldtype', + displayName: 'badfieldtype', + description: 'bad field type', + }, + ], +}; + +const template2 = { + id: 'template2', + templateKey: 'template2', + displayName: 'another template2 title', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + ], +}; + +const template3 = { + id: 'template3', + templateKey: 'template3', + displayName: 'yet another template3 title', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + ], +}; + +const template4 = { + id: 'template4', + templateKey: 'template4', + displayName: 'best template4 title', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + ], +}; + +const template5 = { + id: 'template5', + templateKey: 'template5', + displayName: 'good template5 title', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + ], +}; + +const properties = { + id: 'properties', + scope: 'global', + templateKey: 'properties', +}; + +const template6 = { + id: 'template6', + templateKey: 'template6', + displayName: 'template with some error', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + ], +}; + +const template7 = { + id: 'template7', + templateKey: 'template7', + displayName: 'non editable template', + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + { + id: 'field1', + type: 'string', + key: 'stringfield', + displayName: 'String Field', + description: 'example of a string field', + }, + { + id: 'field2', + type: 'string', + key: 'emptystring', + displayName: 'Empty String Field', + description: 'example of an empty string field', + }, + { + id: 'field3', + type: 'float', + key: 'floatfield', + displayName: 'Float Field', + description: 'example of a float field', + }, + { + id: 'field4', + type: 'float', + key: 'emptyfloat', + displayName: 'Empty Float Field', + description: 'example of an empty float field', + }, + { + id: 'field5', + type: 'enum', + key: 'enumfield', + displayName: 'Enum Field', + description: 'example of a enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field6', + type: 'enum', + key: 'emptyenumfield', + displayName: 'Empty Enum Field', + description: 'example of an empty enum field', + options: [{ key: 'yes' }, { key: 'no' }], + }, + { + id: 'field7', + type: 'date', + key: 'datefield', + displayName: 'Date Field', + description: 'example of a date field', + }, + { + id: 'field8', + type: 'date', + key: 'emptydatefield', + displayName: 'Empty Date Field', + description: 'example of an empty date field', + }, + ], +}; + +// Instances with templates metadata + +const editor0 = { + instance: { + canEdit: true, + id: 'editor0', + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + datefield: '2018-06-20T00:00:00.000Z', + }, + cascadePolicy: { + canEdit: true, + isEnabled: false, + id: 'some cascading policy id', + }, + }, + template: template0, +}; + +const editor1 = { + instance: { + canEdit: true, + id: 'editor1', + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + integerfield: 3, + multiselectfield: ['no', 'maybe'], + datefield: '2018-06-20T00:00:00.000Z', + }, + cascadePolicy: { + canEdit: true, + isEnabled: false, + id: 'some cascading policy id', + }, + }, + template: template1, +}; + +const editor2 = { + instance: { + canEdit: true, + id: 'editor2', + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + datefield: '2018-06-20T00:00:00.000Z', + }, + cascadePolicy: { + canEdit: true, + isEnabled: true, + id: 'some cascading policy id', + }, + }, + template: template2, +}; + +const editor3 = { + instance: { + canEdit: true, + id: 'editor3', + data: {}, + cascadePolicy: { + canEdit: true, + isEnabled: true, + id: 'some cascading policy id', + }, + }, + template: template3, +}; + +const editor7 = { + instance: { + id: 'editor7', + canEdit: false, + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + datefield: '2018-06-20T00:00:00.000Z', + }, + cascadePolicy: { + canEdit: true, + isEnabled: true, + id: 'some cascading policy id', + }, + }, + template: template1, +}; + +const editor8 = { + instance: { + canEdit: false, + id: 'editor8', + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + datefield: '2018-06-20T00:00:00.000Z', + }, + }, + template: template7, +}; + +const editor9 = { + instance: { + id: 'editor9', + canEdit: true, + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + datefield: '2018-06-20T00:00:00.000Z', + }, + }, + template: template6, + hasError: true, +}; + +const editor10 = { + instance: { + canEdit: true, + id: 'editor10', + data: { + stringfield: 'some string', + floatfield: 1, + enumfield: 'yes', + datefield: '2018-06-20T00:00:00.000Z', + }, + }, + template: template2, +}; + +// Instances with custom metadata + +const editor4 = { + instance: { + canEdit: true, + id: 'editor4', + data: { + string: 'string', + number: '1', + bool: 'true', + }, + }, + template: properties, +}; + +const editor5 = { + instance: { + canEdit: true, + id: 'editor5', + data: {}, + }, + template: properties, + hasError: true, +}; + +const editor6 = { + instance: { + id: 'editor6', + canEdit: false, + data: { + string: 'string', + number: '1', + bool: 'true', + }, + }, + template: properties, +}; + +const getExtraTemplates = num => + new Array(num).fill(true).map((_, index) => ({ + id: `generatedtemplate-${index}`, + templateKey: `generatedtemplate-${index}`, + displayName: `generated template-${index}`, + scope: 'enterprise_123', + fields: [ + { + id: 'field0', + type: 'string', + key: 'nodescfield', + displayName: 'No Description Field', + }, + ], + })); + +// State of templates from server +const templatesOnServer = [ + template0, + template1, + template2, + template3, + template4, + template5, + properties, + template6, + template7, + ...getExtraTemplates(1), +]; + +// State of instnaces from server +const editorsOnServer = [ + editor0, + editor1, + editor2, + editor3, + editor4, + editor5, + editor6, + editor7, + editor8, + editor9, + editor10, +]; + +class MetadataExamples extends React.PureComponent { + state = {}; + + componentDidMount() { + // Timeout to simulate network call + setTimeout(() => { + this.setState({ + editors: editorsOnServer.map(editor => ({ + hasError: editor.hasError, + template: editor.template, + instance: { + id: editor.instance.id, + canEdit: editor.instance.canEdit, + data: { ...editor.instance.data }, + cascadePolicy: { ...editor.instance.cascadePolicy }, + }, + })), + }); + }, 1000); + + // Timeout to simulate network call + setTimeout(() => { + this.setState({ templates: templatesOnServer.slice(0) }); + }, 1000); + } + + onRemove = id => { + // Timeout to simulate a network call + setTimeout(() => { + // Get rid of the editor whose instance id matches + const editors = this.state.editors.filter(editor => editor.instance.id !== id); + this.setState({ editors: editors.slice(0) }); // slice for react change detection + }, 1000); + }; + + onAdd = template => { + // Timeout to simulate network call + setTimeout(() => { + // The server applies the new instance (or dummy instance) + const newEditor = { + instance: { + canEdit: true, + id: uniqueId('i_'), + data: {}, + }, + template: this.state.templates.find(t => t.templateKey === template.templateKey), + }; + editorsOnServer.push(newEditor); + const editors = this.state.editors.slice(0); + editors.push({ ...newEditor }); + this.setState({ editors }); + }, 1000); + }; + + onSave = (id, ops, cascadePolicy = {}, rawData) => { + // Whats being sent to the server to show in the examples page + document.querySelector('.metadata-operations').textContent = JSON.stringify(ops, null, 2); + document.querySelector('.metadata-cascading-policy').textContent = JSON.stringify(cascadePolicy, null, 2); + + // Timeout to simulate network call + setTimeout(() => { + // Find the editor that needs saving + const editorOnServer = editorsOnServer.find(editor => editor.instance.id === id); + + // Iterate over all ops and apply them + ops.forEach(op => { + const key = op.path.substr(1); + if (op.op === 'replace' || op.op === 'add') { + editorOnServer.instance.data[key] = op.value; + } else if (op.op === 'remove') { + delete editorOnServer.instance.data[key]; + } + }); + + // Apply the cascading policy + if (cascadePolicy) { + if (cascadePolicy.isEnabled && !editorOnServer.instance.cascadePolicy) { + editorOnServer.instance.cascadePolicy = { + id: uniqueId('cascade_policy_id_'), + }; + } else if (!cascadePolicy.isEnabled && editorOnServer.instance.cascadePolicy) { + editorOnServer.instance.cascadePolicy = { + canEdit: true, + }; + } + } + + // Server has saved the new data and will return an updated editor instance + const editorOnClientIndex = this.state.editors.findIndex(editor => editor.instance.id === id); + const editors = this.state.editors.slice(0); // clone for react changes + editors.splice(editorOnClientIndex, 1, { ...editorOnServer }); + this.setState({ editors }); + }, 1000); + }; + + onModification = (id, isDirty) => { + const editorOnClientIndex = this.state.editors.findIndex(editor => editor.instance.id === id); + const editorOnClient = { ...this.state.editors[editorOnClientIndex] }; + editorOnClient.isDirty = isDirty; + const editors = this.state.editors.slice(0); // clone for react changes + editors.splice(editorOnClientIndex, 1, editorOnClient); + this.setState({ editors }); + + // To show on the examples page what all instances are dirty + document.querySelector('.metadata-dirty').textContent = JSON.stringify( + this.state.editors.filter(editor => editor.isDirty), + null, + 2, + ); + }; + + render() { + if (!this.state.templates || !this.state.editors) { + return ; + } + + return ( +
+
+

Instances that are dirty

+
+                    

Data saved

+
+                    

Cascading policy data saved

+
+                
+
+
+
+
+

Editable Metadata with Cascade Policy applicable

+
+ +
+
+
+
+
+
+

Editable Metadata with Cascade Policy not applicable

+
+ Metadata} + /> +
+
+
+

Empty State

+
+ +
+
+
+
+

Add Template Dropdown with No Templates on Server Message

+
+ + {this.state.isBusy && ( +
+ +
+ )} +
+
+
+
+

Data to be saved

+
+                
+
+ ); + } +} + +export default MetadataExamples; diff --git a/examples/src/MetadataViewExamples.js b/examples/src/MetadataViewExamples.js new file mode 100644 index 0000000000..e972a7029c --- /dev/null +++ b/examples/src/MetadataViewExamples.js @@ -0,0 +1,184 @@ +// @flow +import * as React from 'react'; + +import MetadataView from '../../src/features/metadata-view/MetadataView'; + +import type { Template } from '../../src/features/metadata-instance-editor/flowTypes'; + +const totalWidth = 700; +const tableHeight = 300; +const tableHeaderHeight = 40; +const tableRowHeight = 50; +const widths = { + icon: 0.11, + name: 0.3, + lastModified: 0.6, + size: 0.3, + contractValue: 0.9, +}; + +const instances = [ + { + canEdit: true, + id: 'editor1', + data: { + lastModified: '2013-05-20T00:00:00.000Z', + contractValue: 3, + fileType: 'pdf', + name: 'Google', + size: 1, + }, + cascadePolicy: { + canEdit: true, + isEnabled: false, + id: 'some cascading policy id', + }, + }, + { + canEdit: true, + id: 'editor2', + data: { + lastModified: '2017-04-21T00:00:00.000Z', + contractValue: 34588, + fileType: 'powerpoint-presentation', + name: 'Facebook', + size: 127834, + }, + cascadePolicy: { + canEdit: true, + isEnabled: false, + id: 'some cascading policy id', + }, + }, + { + canEdit: true, + id: 'editor3', + data: { + lastModified: '2016-02-22T00:00:00.000Z', + contractValue: 548949829823984983498398, + fileType: 'video', + name: + 'Long name - KJNASDKJNSDKnskdfjnskldjfnskldjnfkjsdngkjdfngamazonKJNASDKJNSDKnskdfjnskldjfnskldjnfkjsdngkjdfngamazonKJNASDKJNSDKnskdfjnskldjfnskldjnfkjsdngkjdfngamazonKJNASDKJNSDKnskdfjnskldjfnskldjnfkjsdngkjdfngamazonKJNASDKJNSDKnskdfjnskldjfnskldjnfkjsdngkjdfng', + size: 14594600037438723487, + }, + cascadePolicy: { + canEdit: true, + isEnabled: false, + id: 'some cascading policy id', + }, + }, +]; + +const fieldTypes = { + FIELD_TYPE_STRING: 'string', + FIELD_TYPE_DATE: 'date', + FIELD_TYPE_FLOAT: 'float', + FIELD_TYPE_ENUM: 'enum', +}; + +export const template1 = { + id: 'template1', + templateKey: 'template1', + displayName: 'template1 title that is super long', + scope: 'enterprise_123', + 'Vendor Name': { + operators: ['is', 'is greater than', 'is less than', 'is not', 'is blank', 'matches any'], + values: ['Google', 'Apple', 'Facebook'], + }, + 'Expiration Month': { + operators: ['is', 'is greater than', 'is less than', 'is not'], + values: ['August 2018', 'September 2018', 'October 2018'], + }, + 'File Type': { + operators: ['is', 'is not'], + values: ['.docx', '.mp3', 'mp4'], + }, + fields: [ + { + id: 'field1', + type: fieldTypes.FIELD_TYPE_STRING, + key: 'name', + displayName: 'Name', + }, + { + id: 'field7', + type: fieldTypes.FIELD_TYPE_DATE, + key: 'lastModified', + displayName: 'Last Modified', + }, + { + id: 'field11', + type: fieldTypes.FIELD_TYPE_FLOAT, + key: 'size', + displayName: 'Size', + description: 'example of an integer field', + }, + { + id: 'field5', + type: fieldTypes.FIELD_TYPE_ENUM, + key: 'contractValue', + displayName: 'Contract Value', + options: [{ key: '$100' }, { key: '$2000' }, { key: '$10000' }, { key: '$200000' }], + }, + ], +}; + +export const template2 = { + id: 'template1', + templateKey: 'template1', + displayName: 'template1 title', + scope: 'enterprise_123', + 'Vendor Name': { + operators: ['is', 'is greater than', 'is less than', 'is not', 'is blank', 'matches any'], + values: ['Google', 'Apple', 'Facebook'], + }, + 'Expiration Month': { + operators: ['is', 'is greater than', 'is less than', 'is not'], + values: ['August 2018', 'September 2018', 'October 2018'], + }, + 'File Type': { + operators: ['is', 'is not'], + values: ['.docx', '.mp3', 'mp4'], + }, + fields: [ + { + id: 'field1', + type: fieldTypes.FIELD_TYPE_STRING, + key: 'name', + displayName: 'Name', + }, + ], +}; + +type Props = { + currentMessage?: string, + template: Template, + templates?: Array