WorkspaceContent.jsx 10KB

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