WorkspaceContent.jsx 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. import React from 'react'
  2. import { connect } from 'react-redux'
  3. import { withRouter, Route } from 'react-router-dom'
  4. import appFactory from '../appFactory.js'
  5. import { PAGE } from '../helper.js'
  6. import Folder from '../component/Workspace/Folder.jsx'
  7. import ContentItem from '../component/Workspace/ContentItem.jsx'
  8. import ContentItemHeader from '../component/Workspace/ContentItemHeader.jsx'
  9. import DropdownCreateButton from '../component/common/Input/DropdownCreateButton.jsx'
  10. import OpenContentApp from '../component/Workspace/OpenContentApp.jsx'
  11. import OpenCreateContentApp from '../component/Workspace/OpenCreateContentApp.jsx'
  12. import {
  13. PageWrapper,
  14. PageTitle,
  15. PageContent
  16. } from 'tracim_frontend_lib'
  17. import {
  18. getWorkspaceContentList,
  19. getFolderContent,
  20. putWorkspaceContentArchived,
  21. putWorkspaceContentDeleted
  22. } from '../action-creator.async.js'
  23. import {
  24. newFlashMessage,
  25. setWorkspaceContentList,
  26. setWorkspaceContentArchived,
  27. setWorkspaceContentDeleted
  28. } from '../action-creator.sync.js'
  29. const qs = require('query-string')
  30. class WorkspaceContent extends React.Component {
  31. constructor (props) {
  32. super(props)
  33. this.state = {
  34. workspaceIdInUrl: props.match.params.idws ? parseInt(props.match.params.idws) : null, // this is used to avoid handling the parseInt every time
  35. appOpenedType: false
  36. }
  37. document.addEventListener('appCustomEvent', this.customEventReducer)
  38. }
  39. customEventReducer = async ({ detail: { type, data } }) => {
  40. switch (type) {
  41. case 'refreshContentList':
  42. console.log('%c<WorkspaceContent> Custom event', 'color: #28a745', type, data)
  43. this.loadContentList(this.state.workspaceIdInUrl)
  44. break
  45. case 'openContentUrl':
  46. console.log('%c<WorkspaceContent> Custom event', 'color: #28a745', type, data)
  47. this.props.history.push(PAGE.WORKSPACE.CONTENT(data.idWorkspace, data.contentType, data.idContent))
  48. break
  49. case 'appClosed':
  50. case 'hide_popupCreateContent':
  51. console.log('%c<WorkspaceContent> Custom event', 'color: #28a745', type, data, this.state.workspaceIdInUrl)
  52. this.props.history.push(PAGE.WORKSPACE.CONTENT_LIST(this.state.workspaceIdInUrl))
  53. this.setState({appOpenedType: false})
  54. break
  55. }
  56. }
  57. async componentDidMount () {
  58. const { workspaceList, match } = this.props
  59. console.log('%c<WorkspaceContent> componentDidMount', 'color: #c17838')
  60. let wsToLoad = null
  61. if (match.params.idws === undefined) {
  62. if (workspaceList.length > 0) wsToLoad = workspaceList[0].id
  63. else return
  64. } else wsToLoad = match.params.idws
  65. this.loadContentList(wsToLoad)
  66. }
  67. async componentDidUpdate (prevProps, prevState) {
  68. console.log('%c<WorkspaceContent> componentDidUpdate', 'color: #c17838')
  69. if (this.state.workspaceIdInUrl === null) return
  70. const idWorkspace = parseInt(this.props.match.params.idws)
  71. if (isNaN(idWorkspace)) return
  72. const prevFilter = qs.parse(prevProps.location.search).type
  73. const currentFilter = qs.parse(this.props.location.search).type
  74. if (prevState.workspaceIdInUrl !== idWorkspace || prevFilter !== currentFilter) {
  75. this.setState({workspaceIdInUrl: idWorkspace})
  76. this.loadContentList(idWorkspace)
  77. }
  78. // if (user.user_id !== -1 && prevProps.user.id !== user.id) dispatch(getWorkspaceList(user.user_id, idWorkspace))
  79. }
  80. componentWillUnmount () {
  81. this.props.dispatchCustomEvent('unmount_app')
  82. document.removeEventListener('appCustomEvent', this.customEventReducer)
  83. }
  84. loadContentList = async idWorkspace => {
  85. const { user, location, dispatch } = this.props
  86. const wsContent = await dispatch(getWorkspaceContentList(user, idWorkspace, 0))
  87. if (wsContent.status === 200) dispatch(setWorkspaceContentList(wsContent.json, qs.parse(location.search).type))
  88. else dispatch(newFlashMessage('Error while loading workspace', 'danger'))
  89. }
  90. handleClickContentItem = content => {
  91. console.log('%c<WorkspaceContent> content clicked', 'color: #c17838', content)
  92. this.props.history.push(PAGE.WORKSPACE.CONTENT(content.idWorkspace, content.type, content.id))
  93. }
  94. handleClickEditContentItem = (e, content) => {
  95. e.stopPropagation()
  96. console.log('%c<WorkspaceContent> edit nyi', 'color: #c17838', content)
  97. }
  98. handleClickMoveContentItem = (e, content) => {
  99. e.stopPropagation()
  100. console.log('%c<WorkspaceContent> move nyi', 'color: #c17838', content)
  101. }
  102. handleClickDownloadContentItem = (e, content) => {
  103. e.stopPropagation()
  104. console.log('%c<WorkspaceContent> download nyi', 'color: #c17838', content)
  105. }
  106. handleClickArchiveContentItem = async (e, content) => {
  107. const { props, state } = this
  108. e.stopPropagation()
  109. const fetchPutContentArchived = await props.dispatch(putWorkspaceContentArchived(props.user, content.idWorkspace, content.id))
  110. switch (fetchPutContentArchived.status) {
  111. case 204:
  112. props.dispatch(setWorkspaceContentArchived(content.idWorkspace, content.id))
  113. this.loadContentList(state.workspaceIdInUrl)
  114. break
  115. default: props.dispatch(newFlashMessage(props.t('Error while archiving document')))
  116. }
  117. }
  118. handleClickDeleteContentItem = async (e, content) => {
  119. const { props, state } = this
  120. e.stopPropagation()
  121. const fetchPutContentDeleted = await props.dispatch(putWorkspaceContentDeleted(props.user, content.idWorkspace, content.id))
  122. switch (fetchPutContentDeleted.status) {
  123. case 204:
  124. props.dispatch(setWorkspaceContentDeleted(content.idWorkspace, content.id))
  125. this.loadContentList(state.workspaceIdInUrl)
  126. break
  127. default: props.dispatch(newFlashMessage(props.t('Error while deleting document')))
  128. }
  129. }
  130. handleClickFolder = folderId => {
  131. this.props.dispatch(getFolderContent(this.props.workspace.id, folderId))
  132. }
  133. handleClickCreateContent = (e, idFolder, contentType) => {
  134. e.stopPropagation()
  135. this.props.history.push(`${PAGE.WORKSPACE.NEW(this.state.workspaceIdInUrl, contentType)}?parent_id=${idFolder}`)
  136. }
  137. handleUpdateAppOpenedType = openedAppType => this.setState({appOpenedType: openedAppType})
  138. render () {
  139. const { workspaceContentList, contentType } = this.props
  140. const filterWorkspaceContent = (contentList, filter) => {
  141. return filter.length === 0
  142. ? contentList
  143. : contentList.filter(c => c.type === 'folder' || filter.includes(c.type)) // keep unfiltered files and folders
  144. // @FIXME we need to filter subfolder too, but right now, we dont handle subfolder
  145. // .map(c => c.type !== 'folder' ? c : {...c, content: filterWorkspaceContent(c.content, filter)}) // recursively filter folder content
  146. }
  147. // .filter(c => c.type !== 'folder' || c.content.length > 0) // remove empty folder => 2018/05/21 - since we load only one lvl of content, don't remove empty
  148. const urlFilter = qs.parse(this.props.location.search).type
  149. const filteredWorkspaceContentList = workspaceContentList.length > 0
  150. ? filterWorkspaceContent(workspaceContentList, urlFilter ? [urlFilter] : [])
  151. : []
  152. return (
  153. <div className='WorkspaceContent' style={{width: '100%'}}>
  154. <OpenContentApp
  155. // automatically open the app for the idContent in url
  156. idWorkspace={this.state.workspaceIdInUrl}
  157. appOpenedType={this.state.appOpenedType}
  158. updateAppOpenedType={this.handleUpdateAppOpenedType}
  159. />
  160. <Route path={PAGE.WORKSPACE.NEW(':idws', ':type')} component={() =>
  161. <OpenCreateContentApp
  162. // automatically open the popup create content of the app in url
  163. idWorkspace={this.state.workspaceIdInUrl}
  164. appOpenedType={this.state.appOpenedType}
  165. />
  166. } />
  167. <PageWrapper customeClass='workspace'>
  168. <PageTitle
  169. parentClass='workspace__header'
  170. customClass='justify-content-between'
  171. title='Liste des Contenus'
  172. subtitle={workspaceContentList.label ? workspaceContentList.label : ''}
  173. >
  174. <DropdownCreateButton
  175. parentClass='workspace__header__btnaddcontent'
  176. idFolder={null} // null because it is workspace root content
  177. onClickCreateContent={this.handleClickCreateContent}
  178. availableApp={contentType.filter(ct => ct.slug !== 'comment')} // @FIXME: Côme - 2018/08/21 - should use props.appList
  179. />
  180. </PageTitle>
  181. <PageContent parentClass='workspace__content'>
  182. <div className='workspace__content__fileandfolder folder__content active'>
  183. <ContentItemHeader />
  184. { filteredWorkspaceContentList.map((c, i) => c.type === 'folder'
  185. ? (
  186. <Folder
  187. availableApp={contentType.filter(ct => ct.slug !== 'comment')} // @FIXME: Côme - 2018/08/21 - should use props.appList
  188. folderData={c}
  189. onClickItem={this.handleClickContentItem}
  190. onClickExtendedAction={{
  191. edit: this.handleClickEditContentItem,
  192. move: this.handleClickMoveContentItem,
  193. download: this.handleClickDownloadContentItem,
  194. archive: this.handleClickArchiveContentItem,
  195. delete: this.handleClickDeleteContentItem
  196. }}
  197. onClickFolder={this.handleClickFolder}
  198. onClickCreateContent={this.handleClickCreateContent}
  199. isLast={i === filteredWorkspaceContentList.length - 1}
  200. key={c.id}
  201. />
  202. )
  203. : (
  204. <ContentItem
  205. label={c.label}
  206. type={c.type}
  207. faIcon={contentType.length ? contentType.find(a => a.slug === c.type).faIcon : ''}
  208. statusSlug={c.statusSlug}
  209. contentType={contentType.length ? contentType.find(ct => ct.slug === c.type) : null}
  210. onClickItem={() => this.handleClickContentItem(c)}
  211. onClickExtendedAction={{
  212. edit: e => this.handleClickEditContentItem(e, c),
  213. move: e => this.handleClickMoveContentItem(e, c),
  214. download: e => this.handleClickDownloadContentItem(e, c),
  215. archive: e => this.handleClickArchiveContentItem(e, c),
  216. delete: e => this.handleClickDeleteContentItem(e, c)
  217. }}
  218. onClickCreateContent={this.handleClickCreateContent}
  219. isLast={i === filteredWorkspaceContentList.length - 1}
  220. key={c.id}
  221. />
  222. )
  223. )}
  224. </div>
  225. <DropdownCreateButton
  226. customClass='workspace__content__button'
  227. idFolder={null}
  228. onClickCreateContent={this.handleClickCreateContent}
  229. availableApp={contentType.filter(ct => ct.slug !== 'comment')} // @FIXME: Côme - 2018/08/21 - should use props.appList
  230. />
  231. </PageContent>
  232. </PageWrapper>
  233. </div>
  234. )
  235. }
  236. }
  237. const mapStateToProps = ({ user, workspaceContentList, workspaceList, contentType }) => ({ user, workspaceContentList, workspaceList, contentType })
  238. export default withRouter(connect(mapStateToProps)(appFactory(WorkspaceContent)))