diff --git a/frontend/src/metabase/lib/colors.js b/frontend/src/metabase/lib/colors.js index 339a2217527a60d27b9aa33344edd4a55c60a355..dc43b44b322f4b052b1b28bdbc26b6cac4fc3a7e 100644 --- a/frontend/src/metabase/lib/colors.js +++ b/frontend/src/metabase/lib/colors.js @@ -1,3 +1,9 @@ +// @flow + +type ColorName = string; +type Color = string +type ColorFamily = { [name: ColorName]: Color }; + export const normal = { blue: '#509EE3', green: '#9CC177', @@ -57,3 +63,9 @@ export const harmony = [ '#c1a877', '#f95c67', ] + +export const getRandomColor = (family: ColorFamily): Color => { + // $FlowFixMe: Object.values doesn't preserve the type :-/ + const colors: Color[] = Object.values(family) + return colors[Math.floor(Math.random() * colors.length)] +} diff --git a/frontend/src/metabase/lib/colors.spec.js b/frontend/src/metabase/lib/colors.spec.js new file mode 100644 index 0000000000000000000000000000000000000000..1d5d641f775be548f95efc1c00bb7222a433eaff --- /dev/null +++ b/frontend/src/metabase/lib/colors.spec.js @@ -0,0 +1,8 @@ +import { getRandomColor, normal } from 'metabase/lib/colors' + +describe('getRandomColor', () => { + it('should return a color string from the proper family', () => { + const color = getRandomColor(normal) + expect(Object.values(normal)).toContain(color) + }) +}) diff --git a/frontend/src/metabase/questions/containers/CollectionEditorForm.jsx b/frontend/src/metabase/questions/containers/CollectionEditorForm.jsx index 305923466c009198c566d770b6c0b85a10178864..73b18e3d00bba08dbb75e43004f0edc8436fdb57 100644 --- a/frontend/src/metabase/questions/containers/CollectionEditorForm.jsx +++ b/frontend/src/metabase/questions/containers/CollectionEditorForm.jsx @@ -8,9 +8,9 @@ import Modal from "metabase/components/Modal"; import { reduxForm } from "redux-form"; -import { normal } from "metabase/lib/colors"; +import { normal, getRandomColor } from "metabase/lib/colors"; -@reduxForm({ +const formConfig = { form: 'collection', fields: ['id', 'name', 'description', 'color'], validate: (values) => { @@ -29,25 +29,43 @@ import { normal } from "metabase/lib/colors"; name: "", description: "", // pick a random color to start so everything isn't blue all the time - color: normal[Math.floor(Math.random() * normal.length)] + color: getRandomColor(normal) } -}) -export default class CollectionEditorForm extends Component { +} + +export const getFormTitle = ({ id, name }) => + id.value ? name.value : "New collection" + +export const getActionText = ({ id }) => + id.value ? "Update": "Create" + + +export const CollectionEditorFormActions = ({ handleSubmit, invalid, onClose, fields}) => + <div> + <Button className="mr1" onClick={onClose}> + Cancel + </Button> + <Button primary disabled={invalid} onClick={handleSubmit}> + { getActionText(fields) } + </Button> + </div> + +export class CollectionEditorForm extends Component { + props: { + fields: Object, + onClose: Function, + invalid: Boolean, + handleSubmit: Function, + } + render() { - const { fields, handleSubmit, invalid, onClose } = this.props; + const { fields, onClose } = this.props; return ( <Modal inline form - title={fields.id.value != null ? fields.name.value : "New collection"} - footer={[ - <Button className="mr1" onClick={onClose}> - Cancel - </Button>, - <Button primary disabled={invalid} onClick={handleSubmit}> - { fields.id.value != null ? "Update" : "Create" } - </Button> - ]} + title={getFormTitle(fields)} + footer={<CollectionEditorFormActions {...this.props} />} onClose={onClose} > <div className="NewForm ml-auto mr-auto mt4 pt2" style={{ width: 540 }}> @@ -83,3 +101,5 @@ export default class CollectionEditorForm extends Component { ) } } + +export default reduxForm(formConfig)(CollectionEditorForm) diff --git a/frontend/src/metabase/questions/containers/CollectionEditorForm.spec.js b/frontend/src/metabase/questions/containers/CollectionEditorForm.spec.js new file mode 100644 index 0000000000000000000000000000000000000000..cc7b5f07b348f5cb23e188e65997222fbe7eb8bd --- /dev/null +++ b/frontend/src/metabase/questions/containers/CollectionEditorForm.spec.js @@ -0,0 +1,38 @@ +import { + getFormTitle, + getActionText +} from './CollectionEditorForm' + +const FORM_FIELDS = { + id: { value: 4 }, + name: { value: 'Test collection' }, + color: { value: '#409ee3' }, + initialValues: { + color: '#409ee3' + } +} +const NEW_COLLECTION_FIELDS = { ...FORM_FIELDS, id: '', color: '' } + +describe('CollectionEditorForm', () => { + + describe('Title', () => { + it('should have a default title if no collection exists', () => + expect(getFormTitle(NEW_COLLECTION_FIELDS)).toEqual('New collection') + ) + + it('should have the title of the colleciton if one exists', () => + expect(getFormTitle(FORM_FIELDS)).toEqual(FORM_FIELDS.name.value) + ) + }) + + describe('Form actions', () => { + it('should have a "create" primary action if no collection exists', () => + expect(getActionText(NEW_COLLECTION_FIELDS)).toEqual('Create') + ) + + it('should have an "update" primary action if no collection exists', () => + expect(getActionText(FORM_FIELDS)).toEqual('Update') + ) + }) + +})