From d770f26f3a3160add1c725135cce36f6a4784576 Mon Sep 17 00:00:00 2001 From: Chris Williams Date: Wed, 4 Apr 2018 17:34:11 -0700 Subject: [PATCH] [dashboard builder] static layout + toasts (#4763) * [dashboard-builder] remove spacer component * [dashboard-builder] better transparent indicator, better grid gutter logic, no dragging top-level tabs, headers are multiples of grid unit, fix row height granularity, update redux state key dashboard => dashboardLayout * [dashboard-builder] don't blast column child dimensions on resize * [dashboard-builder] ResizableContainer min size can't be smaller than size, fix row style, role=none on WithPopoverMenu container * [edit mode] add edit mode to redux and propogate to all s * [toasts] add Toast component, ToastPresenter container and component, and toast redux actions + reducers * [dashboard-builder] add info toast when dropResult overflows parent --- .../javascripts/components/EditableTitle.jsx | 16 ++- .../assets/javascripts/dashboard/index.jsx | 5 +- .../actions/{index.js => dashboardLayout.js} | 35 +++-- .../dashboard/v2/actions/editMode.js | 9 ++ .../dashboard/v2/actions/messageToasts.js | 49 +++++++ .../v2/components/BuilderComponentPane.jsx | 2 - .../v2/components/DashboardBuilder.jsx | 28 ++-- .../dashboard/v2/components/DashboardGrid.jsx | 10 +- .../v2/components/DashboardHeader.jsx | 14 +- .../dashboard/v2/components/Toast.jsx | 87 ++++++++++++ .../v2/components/ToastPresenter.jsx | 39 ++++++ .../v2/components/dnd/DragDroppable.jsx | 4 + .../dashboard/v2/components/dnd/handleDrop.js | 2 +- .../v2/components/gridComponents/Chart.jsx | 17 ++- .../v2/components/gridComponents/Column.jsx | 75 +++++------ .../v2/components/gridComponents/Divider.jsx | 10 +- .../v2/components/gridComponents/Header.jsx | 13 +- .../v2/components/gridComponents/Row.jsx | 69 ++++------ .../v2/components/gridComponents/Spacer.jsx | 106 --------------- .../v2/components/gridComponents/Tab.jsx | 18 ++- .../v2/components/gridComponents/Tabs.jsx | 39 +++--- .../v2/components/gridComponents/index.js | 4 - .../new/DraggableNewComponent.jsx | 1 + .../gridComponents/new/NewSpacer.jsx | 24 ---- .../v2/components/menu/WithPopoverMenu.jsx | 40 +++--- .../resizable/ResizableContainer.jsx | 32 +++-- .../v2/containers/DashboardBuilder.jsx | 7 +- .../v2/containers/DashboardComponent.jsx | 17 +-- .../dashboard/v2/containers/DashboardGrid.jsx | 4 +- .../v2/containers/DashboardHeader.jsx | 16 ++- .../v2/containers/ToastPresenter.jsx | 10 ++ .../{dashboard.js => dashboardLayout.js} | 5 +- .../dashboard/v2/reducers/editMode.js | 11 ++ .../dashboard/v2/reducers/index.js | 12 +- .../dashboard/v2/reducers/messageToasts.js | 18 +++ .../dashboard/v2/stylesheets/builder.less | 14 +- .../components/DashboardBuilder.jsx | 127 ------------------ .../v2/stylesheets/components/chart.less | 4 +- .../v2/stylesheets/components/column.less | 30 ++++- .../v2/stylesheets/components/divider.less | 2 +- .../v2/stylesheets/components/header.less | 28 +++- .../v2/stylesheets/components/index.less | 1 - .../stylesheets/components/new-component.less | 12 -- .../v2/stylesheets/components/row.less | 23 +++- .../v2/stylesheets/components/spacer.less | 13 -- .../dashboard/v2/stylesheets/grid.less | 24 +--- .../dashboard/v2/stylesheets/index.less | 1 + .../v2/stylesheets/popover-menu.less | 4 +- .../dashboard/v2/stylesheets/resizable.less | 16 +-- .../dashboard/v2/stylesheets/toast.less | 59 ++++++++ .../dashboard/v2/stylesheets/variables.less | 8 ++ .../dashboard/v2/util/componentIsResizable.js | 2 - .../dashboard/v2/util/componentTypes.js | 2 - .../dashboard/v2/util/constants.js | 8 +- .../dashboard/v2/util/dropOverflowsParent.js | 24 ++++ .../dashboard/v2/util/getChildWidth.js | 5 +- .../dashboard/v2/util/getDropPosition.js | 2 + .../dashboard/v2/util/isValidChild.js | 11 +- .../dashboard/v2/util/newComponentFactory.js | 6 +- .../dashboard/v2/util/newComponentIdToType.js | 35 ----- .../dashboard/v2/util/propShapes.jsx | 7 + superset/assets/stylesheets/superset.less | 29 ++-- 62 files changed, 715 insertions(+), 630 deletions(-) rename superset/assets/javascripts/dashboard/v2/actions/{index.js => dashboardLayout.js} (79%) create mode 100644 superset/assets/javascripts/dashboard/v2/actions/editMode.js create mode 100644 superset/assets/javascripts/dashboard/v2/actions/messageToasts.js create mode 100644 superset/assets/javascripts/dashboard/v2/components/Toast.jsx create mode 100644 superset/assets/javascripts/dashboard/v2/components/ToastPresenter.jsx delete mode 100644 superset/assets/javascripts/dashboard/v2/components/gridComponents/Spacer.jsx delete mode 100644 superset/assets/javascripts/dashboard/v2/components/gridComponents/new/NewSpacer.jsx create mode 100644 superset/assets/javascripts/dashboard/v2/containers/ToastPresenter.jsx rename superset/assets/javascripts/dashboard/v2/reducers/{dashboard.js => dashboardLayout.js} (98%) create mode 100644 superset/assets/javascripts/dashboard/v2/reducers/editMode.js create mode 100644 superset/assets/javascripts/dashboard/v2/reducers/messageToasts.js delete mode 100644 superset/assets/javascripts/dashboard/v2/stylesheets/components/DashboardBuilder.jsx delete mode 100644 superset/assets/javascripts/dashboard/v2/stylesheets/components/spacer.less create mode 100644 superset/assets/javascripts/dashboard/v2/stylesheets/toast.less create mode 100644 superset/assets/javascripts/dashboard/v2/util/dropOverflowsParent.js delete mode 100644 superset/assets/javascripts/dashboard/v2/util/newComponentIdToType.js diff --git a/superset/assets/javascripts/components/EditableTitle.jsx b/superset/assets/javascripts/components/EditableTitle.jsx index a7e3f17f1e35b..45fea1dcb030d 100644 --- a/superset/assets/javascripts/components/EditableTitle.jsx +++ b/superset/assets/javascripts/components/EditableTitle.jsx @@ -116,7 +116,7 @@ class EditableTitle extends React.PureComponent { } render() { - let input = ( + let content = ( ); if (this.props.showTooltip) { - input = ( + content = ( - {input} + {content} ); } return ( - - {input} + + {content} ); } diff --git a/superset/assets/javascripts/dashboard/index.jsx b/superset/assets/javascripts/dashboard/index.jsx index bb21a4303b3c6..1aadc5812690e 100644 --- a/superset/assets/javascripts/dashboard/index.jsx +++ b/superset/assets/javascripts/dashboard/index.jsx @@ -19,12 +19,15 @@ initJQueryAjax(); const appContainer = document.getElementById('app'); // const bootstrapData = JSON.parse(appContainer.getAttribute('data-bootstrap')); // const initState = Object.assign({}, getInitialState(bootstrapData)); + const initState = { - dashboard: { + dashboardLayout: { past: [], present: emptyDashboardLayout, future: [], }, + editMode: true, + messageToasts: [], }; const store = createStore( diff --git a/superset/assets/javascripts/dashboard/v2/actions/index.js b/superset/assets/javascripts/dashboard/v2/actions/dashboardLayout.js similarity index 79% rename from superset/assets/javascripts/dashboard/v2/actions/index.js rename to superset/assets/javascripts/dashboard/v2/actions/dashboardLayout.js index a6c7b77223945..b6d41c44b87d7 100644 --- a/superset/assets/javascripts/dashboard/v2/actions/index.js +++ b/superset/assets/javascripts/dashboard/v2/actions/dashboardLayout.js @@ -1,10 +1,8 @@ +import { addInfoToast } from './messageToasts'; +import { CHART_TYPE, MARKDOWN_TYPE, TABS_TYPE } from '../util/componentTypes'; import { DASHBOARD_ROOT_ID, NEW_COMPONENTS_SOURCE_ID } from '../util/constants'; +import dropOverflowsParent from '../util/dropOverflowsParent'; import findParentId from '../util/findParentId'; -import { - CHART_TYPE, - MARKDOWN_TYPE, - TABS_TYPE, -} from '../util/componentTypes'; // Component CRUD ------------------------------------------------------------- export const UPDATE_COMPONENTS = 'UPDATE_COMPONENTS'; @@ -61,8 +59,8 @@ export function deleteTopLevelTabs() { export const RESIZE_COMPONENT = 'RESIZE_COMPONENT'; export function resizeComponent({ id, width, height }) { return (dispatch, getState) => { - const { dashboard: undoableDashboard } = getState(); - const { present: dashboard } = undoableDashboard; + const { dashboardLayout: undoableLayout } = getState(); + const { present: dashboard } = undoableLayout; const component = dashboard[id]; if ( @@ -88,8 +86,8 @@ export function resizeComponent({ id, width, height }) { ...child, meta: { ...child.meta, - width: width || component.meta.width, - height: height || component.meta.height, + width: width || child.meta.width, + height: height || child.meta.height, }, }; } @@ -114,6 +112,15 @@ export function moveComponent(dropResult) { export const HANDLE_COMPONENT_DROP = 'HANDLE_COMPONENT_DROP'; export function handleComponentDrop(dropResult) { return (dispatch, getState) => { + const overflowsParent = dropOverflowsParent(dropResult, getState().dashboardLayout.present); + + if (overflowsParent) { + return dispatch(addInfoToast( + `Parent does not have enough space for this component. + Try decreasing its width or add it to a new row.`, + )); + } + const { source, destination } = dropResult; const droppedOnRoot = destination && destination.id === DASHBOARD_ROOT_ID; const isNewComponent = source.id === NEW_COMPONENTS_SOURCE_ID; @@ -133,14 +140,14 @@ export function handleComponentDrop(dropResult) { dispatch(moveComponent(dropResult)); } - // if we moved a tab and the parent tabs no longer has children, delete it. + // if we moved a Tab and the parent Tabs no longer has children, delete it. if (!isNewComponent) { - const { dashboard: undoableDashboard } = getState(); - const { present: dashboard } = undoableDashboard; - const sourceComponent = dashboard[source.id]; + const { dashboardLayout: undoableLayout } = getState(); + const { present: layout } = undoableLayout; + const sourceComponent = layout[source.id]; if (sourceComponent.type === TABS_TYPE && sourceComponent.children.length === 0) { - const parentId = findParentId({ childId: source.id, components: dashboard }); + const parentId = findParentId({ childId: source.id, components: layout }); dispatch(deleteComponent(source.id, parentId)); } } diff --git a/superset/assets/javascripts/dashboard/v2/actions/editMode.js b/superset/assets/javascripts/dashboard/v2/actions/editMode.js new file mode 100644 index 0000000000000..0a849ea190688 --- /dev/null +++ b/superset/assets/javascripts/dashboard/v2/actions/editMode.js @@ -0,0 +1,9 @@ +export const SET_EDIT_MODE = 'SET_EDIT_MODE'; +export function setEditMode(editMode) { + return { + type: SET_EDIT_MODE, + payload: { + editMode, + }, + }; +} diff --git a/superset/assets/javascripts/dashboard/v2/actions/messageToasts.js b/superset/assets/javascripts/dashboard/v2/actions/messageToasts.js new file mode 100644 index 0000000000000..af10eadf04412 --- /dev/null +++ b/superset/assets/javascripts/dashboard/v2/actions/messageToasts.js @@ -0,0 +1,49 @@ +import { INFO_TOAST, SUCCESS_TOAST, WARNING_TOAST, DANGER_TOAST } from '../util/constants'; + +function getToastUuid(type) { + return `${Math.random().toString(16).slice(2)}-${type}-${Math.random().toString(16).slice(2)}`; +} + +export const ADD_TOAST = 'ADD_TOAST'; +export function addToast({ toastType, text }) { + debugger; + return { + type: ADD_TOAST, + payload: { + id: getToastUuid(toastType), + toastType, + text, + }, + }; +} + +export const REMOVE_TOAST = 'REMOVE_TOAST'; +export function removeToast(id) { + return { + type: REMOVE_TOAST, + payload: { + id, + }, + }; +} + +// Different types of toasts +export const ADD_INFO_TOAST = 'ADD_INFO_TOAST'; +export function addInfoToast(text) { + return dispatch => dispatch(addToast({ text, toastType: INFO_TOAST })); +} + +export const ADD_SUCCESS_TOAST = 'ADD_SUCCESS_TOAST'; +export function addSuccessToast(text) { + return dispatch => dispatch(addToast({ text, toastType: SUCCESS_TOAST })); +} + +export const ADD_WARNING_TOAST = 'ADD_WARNING_TOAST'; +export function addWarningToast(text) { + return dispatch => dispatch(addToast({ text, toastType: WARNING_TOAST })); +} + +export const ADD_DANGER_TOAST = 'ADD_DANGER_TOAST'; +export function addDangerToast(text) { + return dispatch => dispatch(addToast({ text, toastType: DANGER_TOAST })); +} diff --git a/superset/assets/javascripts/dashboard/v2/components/BuilderComponentPane.jsx b/superset/assets/javascripts/dashboard/v2/components/BuilderComponentPane.jsx index 86f3788bae6ae..efef5a59a1fea 100644 --- a/superset/assets/javascripts/dashboard/v2/components/BuilderComponentPane.jsx +++ b/superset/assets/javascripts/dashboard/v2/components/BuilderComponentPane.jsx @@ -6,7 +6,6 @@ import NewColumn from './gridComponents/new/NewColumn'; import NewDivider from './gridComponents/new/NewDivider'; import NewHeader from './gridComponents/new/NewHeader'; import NewRow from './gridComponents/new/NewRow'; -import NewSpacer from './gridComponents/new/NewSpacer'; import NewTabs from './gridComponents/new/NewTabs'; const propTypes = { @@ -24,7 +23,6 @@ class BuilderComponentPane extends React.PureComponent { - diff --git a/superset/assets/javascripts/dashboard/v2/components/DashboardBuilder.jsx b/superset/assets/javascripts/dashboard/v2/components/DashboardBuilder.jsx index f3717187c13dd..8e2d985861543 100644 --- a/superset/assets/javascripts/dashboard/v2/components/DashboardBuilder.jsx +++ b/superset/assets/javascripts/dashboard/v2/components/DashboardBuilder.jsx @@ -1,3 +1,4 @@ +import cx from 'classnames'; import React from 'react'; import PropTypes from 'prop-types'; import HTML5Backend from 'react-dnd-html5-backend'; @@ -9,6 +10,7 @@ import DashboardGrid from '../containers/DashboardGrid'; import IconButton from './IconButton'; import DragDroppable from './dnd/DragDroppable'; import DashboardComponent from '../containers/DashboardComponent'; +import ToastPresenter from '../containers/ToastPresenter'; import WithPopoverMenu from './menu/WithPopoverMenu'; import { @@ -18,11 +20,10 @@ import { } from '../util/constants'; const propTypes = { - editMode: PropTypes.bool, - // redux - dashboard: PropTypes.object.isRequired, + dashboardLayout: PropTypes.object.isRequired, deleteTopLevelTabs: PropTypes.func.isRequired, + editMode: PropTypes.bool.isRequired, handleComponentDrop: PropTypes.func.isRequired, }; @@ -52,20 +53,20 @@ class DashboardBuilder extends React.Component { render() { const { tabIndex } = this.state; - const { handleComponentDrop, dashboard, deleteTopLevelTabs } = this.props; - const dashboardRoot = dashboard[DASHBOARD_ROOT_ID]; + const { handleComponentDrop, dashboardLayout, deleteTopLevelTabs, editMode } = this.props; + const dashboardRoot = dashboardLayout[DASHBOARD_ROOT_ID]; const rootChildId = dashboardRoot.children[0]; - const topLevelTabs = rootChildId !== DASHBOARD_GRID_ID && dashboard[rootChildId]; + const topLevelTabs = rootChildId !== DASHBOARD_GRID_ID && dashboardLayout[rootChildId]; const gridComponentId = topLevelTabs ? topLevelTabs.children[Math.min(topLevelTabs.children.length - 1, tabIndex)] : DASHBOARD_GRID_ID; - const gridComponent = dashboard[gridComponentId]; + const gridComponent = dashboardLayout[gridComponentId]; return ( -
- {topLevelTabs ? ( // you cannot drop on/displace tabs if they already exist +
+ {topLevelTabs || !editMode ? ( // you cannot drop on/displace tabs if they already exist ) : ( {({ dropIndicatorProps }) => (
@@ -94,6 +96,7 @@ class DashboardBuilder extends React.Component { onClick={deleteTopLevelTabs} />, ]} + editMode={editMode} > } -
+
- + {editMode && }
+
); } diff --git a/superset/assets/javascripts/dashboard/v2/components/DashboardGrid.jsx b/superset/assets/javascripts/dashboard/v2/components/DashboardGrid.jsx index cfe99c71a26af..9f4cb9317fe72 100644 --- a/superset/assets/javascripts/dashboard/v2/components/DashboardGrid.jsx +++ b/superset/assets/javascripts/dashboard/v2/components/DashboardGrid.jsx @@ -13,6 +13,7 @@ import { const propTypes = { depth: PropTypes.number.isRequired, + editMode: PropTypes.bool.isRequired, gridComponent: componentShape.isRequired, handleComponentDrop: PropTypes.func.isRequired, resizeComponent: PropTypes.func.isRequired, @@ -70,7 +71,7 @@ class DashboardGrid extends React.PureComponent { } render() { - const { gridComponent, handleComponentDrop, depth } = this.props; + const { gridComponent, handleComponentDrop, depth, editMode } = this.props; const { isResizing, rowGuideTop } = this.state; return ( @@ -99,18 +100,19 @@ class DashboardGrid extends React.PureComponent { ))} {/* render an empty drop target */} - {gridComponent.children.length === 0 && + {editMode && {({ dropIndicatorProps }) => dropIndicatorProps && -
} +
} } {isResizing && Array(GRID_COLUMN_COUNT).fill(null).map((_, i) => ( diff --git a/superset/assets/javascripts/dashboard/v2/components/DashboardHeader.jsx b/superset/assets/javascripts/dashboard/v2/components/DashboardHeader.jsx index e0d14c4712131..ca204e5b7166d 100644 --- a/superset/assets/javascripts/dashboard/v2/components/DashboardHeader.jsx +++ b/superset/assets/javascripts/dashboard/v2/components/DashboardHeader.jsx @@ -7,8 +7,7 @@ import { componentShape } from '../util/propShapes'; import EditableTitle from '../../../components/EditableTitle'; const propTypes = { - // editMode: PropTypes.bool.isRequired, - // setEditMode: PropTypes.func.isRequired, + editMode: PropTypes.bool.isRequired, component: componentShape.isRequired, // redux @@ -17,6 +16,7 @@ const propTypes = { onRedo: PropTypes.func.isRequired, canUndo: PropTypes.bool.isRequired, canRedo: PropTypes.bool.isRequired, + setEditMode: PropTypes.func.isRequired, }; class DashboardHeader extends React.Component { @@ -27,8 +27,7 @@ class DashboardHeader extends React.Component { } toggleEditMode() { - console.log('@TODO toggleEditMode'); - // this.props.setEditMode(!this.props.editMode); + this.props.setEditMode(!this.props.editMode); } handleChangeText(nextText) { @@ -47,19 +46,18 @@ class DashboardHeader extends React.Component { } render() { - const { component, onUndo, onRedo, canUndo, canRedo } = this.props; - const editMode = true; + const { component, onUndo, onRedo, canUndo, canRedo, editMode } = this.props; return (
-

+
-

+