WorkspaceContent.jsx 10.0KB

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