Skip to content

Commit

Permalink
feat: Add QualificationModal component
Browse files Browse the repository at this point in the history
  • Loading branch information
JF-Cozy committed Dec 12, 2024
1 parent 4ca7867 commit e2ebcc0
Show file tree
Hide file tree
Showing 7 changed files with 142 additions and 0 deletions.
1 change: 1 addition & 0 deletions docs/styleguide.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,7 @@ module.exports = {
'../react/QualificationGrid',
'../react/QualificationIconStack',
'../react/QualificationItem',
'../react/QualificationModal',
'../react/UploadQueue'
]
},
Expand Down
28 changes: 28 additions & 0 deletions react/QualificationModal/Readme.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
```jsx
import DemoProvider from 'cozy-ui/docs/components/DemoProvider'
import QualificationModal from 'cozy-ui/transpiled/react/QualificationModal'
import Typography from 'cozy-ui/transpiled/react/Typography'
import CloudIcon from 'cozy-ui/transpiled/react/Icons/Cloud'
import FormControlLabel from 'cozy-ui/transpiled/react/FormControlLabel'
import RadioGroup from 'cozy-ui/transpiled/react/RadioGroup'
import Radio from 'cozy-ui/transpiled/react/Radios'
import FormControl from 'cozy-ui/transpiled/react/FormControl'
import Button from 'cozy-ui/transpiled/react/Buttons'

initialState = { show: isTesting() ? true : false }

const show = () => setState({show: true})
const hide = () => setState({show: false})

;

<DemoProvider client={{ collection: () => ({ updateMetadataAttribute: () => {}}) }}>
<Button label={state.show ? "Close modal" : "Open modal"} variant="ghost" onClick={show} />
{state.show &&
<QualificationModal
file={{ name: "toto.txt", metadata: { qualification: { label: 'isp_invoice' } } }}
onClose={hide}
/>
}
</DemoProvider>
```
94 changes: 94 additions & 0 deletions react/QualificationModal/index.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
import PropTypes from 'prop-types'
import React, { useMemo } from 'react'

import { useClient } from 'cozy-client'
import { themesList } from 'cozy-client/dist/models/document/documentTypeData'
import { isQualificationNote } from 'cozy-client/dist/models/document/documentTypeDataHelpers'
import { getBoundT } from 'cozy-client/dist/models/document/locales'
import { getQualification } from 'cozy-client/dist/models/document/qualification'

import { locales } from './locales'
import Icon from '../Icon'
import FileTypeNoteIcon from '../Icons/FileTypeNote'
import NestedSelectResponsive from '../NestedSelect/NestedSelectResponsive'
import QualificationIconStack from '../QualificationIconStack'
import { useI18n, useExtendI18n } from '../providers/I18n'

const makeOptions = lang => {
const qualifT = getBoundT(lang)

return {
children: [
{
id: 'none',
title: qualifT('Scan.themes.none'),
icon: <QualificationIconStack />
},
...themesList.map(theme => ({
id: theme.id,
title: qualifT(`Scan.themes.${theme.label}`),
icon: <QualificationIconStack theme={theme.label} />,
children: theme.items.map(item => ({
id: item.label,
item,
title: qualifT(`Scan.items.${item.label}`),
icon: isQualificationNote(item) ? (
<Icon icon={FileTypeNoteIcon} size={64} />
) : (
<QualificationIconStack qualification={item.label} />
)
}))
}))
]
}
}

const QualificationModal = ({ file, title, onClose }) => {
useExtendI18n(locales)
const client = useClient()
const { t, lang } = useI18n()

const qualificationLabel = getQualification(file)?.label
const options = useMemo(() => makeOptions(lang), [lang])

const isSelected = ({ id, item }) => {
return qualificationLabel
? qualificationLabel === item?.label
: id === 'none'
}

const handleClick = async ({ id, item }) => {
const fileCollection = client.collection('io.cozy.files')
const removeQualification = qualificationLabel && id === 'none'

/*
In the case where we remove the qualification it's necessary to define the attribute to `null` and not `undefined`, with `undefined` the stack does not return the attribute and today the Redux store is not updated for a missing attribute.
As a result, the UI is not updated and continues to display the qualification on the document, even though it has been deleted in CouchDB.
*/
await fileCollection.updateMetadataAttribute(file._id, {
qualification: removeQualification ? null : item
})

onClose()
}

return (
<NestedSelectResponsive
title={title || t('QualificationModal.title')}
options={options}
noDivider
document={file}
isSelected={isSelected}
onSelect={handleClick}
onClose={onClose}
/>
)
}

QualificationModal.propTypes = {
file: PropTypes.object,
title: PropTypes.string,
onClose: PropTypes.func
}

export default QualificationModal
5 changes: 5 additions & 0 deletions react/QualificationModal/locales/en.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"QualificationModal": {
"title": "Document type"
}
}
5 changes: 5 additions & 0 deletions react/QualificationModal/locales/fr.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"QualificationModal": {
"title": "Type de document"
}
}
7 changes: 7 additions & 0 deletions react/QualificationModal/locales/index.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import en from './en.json'
import fr from './fr.json'

export const locales = {
en,
fr
}
2 changes: 2 additions & 0 deletions react/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,8 @@ export { default as Thumbnail } from './Thumbnail'
export { default as ButtonBase } from './ButtonBase'
export { default as QualificationGrid } from './QualificationGrid'
export { default as QualificationItem } from './QualificationItem'
export { default as QualificationIconStack } from './QualificationIconStack'
export { default as QualificationModal } from './QualificationModal'
export { default as Timeline } from './Timeline'
export { default as TimelineConnector } from './TimelineConnector'
export { default as TimelineContent } from './TimelineContent'
Expand Down

0 comments on commit e2ebcc0

Please sign in to comment.