HtmlDocument.jsx 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. import React from 'react'
  2. import HtmlDocumentComponent from '../component/HtmlDocument.jsx'
  3. import { translate } from 'react-i18next'
  4. import i18n from '../i18n.js'
  5. import {
  6. addAllResourceI18n,
  7. handleFetchResult,
  8. generateAvatarFromPublicName,
  9. PopinFixed,
  10. PopinFixedHeader,
  11. PopinFixedOption,
  12. PopinFixedContent,
  13. Timeline,
  14. NewVersionBtn,
  15. ArchiveDeleteContent,
  16. SelectStatus
  17. } from 'tracim_frontend_lib'
  18. import { MODE, debug } from '../helper.js'
  19. import {
  20. getHtmlDocContent,
  21. getHtmlDocComment,
  22. getHtmlDocRevision,
  23. postHtmlDocNewComment,
  24. putHtmlDocContent,
  25. putHtmlDocStatus,
  26. putHtmlDocIsArchived,
  27. putHtmlDocIsDeleted,
  28. putHtmlDocRestoreArchived,
  29. putHtmlDocRestoreDeleted,
  30. putHtmlDocRead
  31. } from '../action.async.js'
  32. class HtmlDocument extends React.Component {
  33. constructor (props) {
  34. super(props)
  35. this.state = {
  36. appName: 'html-document',
  37. isVisible: true,
  38. config: props.data ? props.data.config : debug.config,
  39. loggedUser: props.data ? props.data.loggedUser : debug.loggedUser,
  40. content: props.data ? props.data.content : debug.content,
  41. rawContentBeforeEdit: '',
  42. timeline: props.data ? [] : [], // debug.timeline,
  43. newComment: '',
  44. timelineWysiwyg: false,
  45. mode: MODE.VIEW
  46. }
  47. // i18n has been init, add resources from frontend
  48. addAllResourceI18n(i18n, this.state.config.translation)
  49. i18n.changeLanguage(this.state.loggedUser.lang)
  50. document.addEventListener('appCustomEvent', this.customEventReducer)
  51. }
  52. customEventReducer = ({ detail: { type, data } }) => { // action: { type: '', data: {} }
  53. switch (type) {
  54. case 'html-document_showApp':
  55. console.log('%c<HtmlDocument> Custom event', 'color: #28a745', type, data)
  56. this.setState({isVisible: true})
  57. break
  58. case 'html-document_hideApp':
  59. console.log('%c<HtmlDocument> Custom event', 'color: #28a745', type, data)
  60. this.setState({isVisible: false})
  61. break
  62. case 'html-document_reloadContent':
  63. console.log('%c<HtmlDocument> Custom event', 'color: #28a745', type, data)
  64. this.setState(prev => ({content: {...prev.content, ...data}, isVisible: true}))
  65. break
  66. case 'allApp_changeLang':
  67. console.log('%c<HtmlDocument> Custom event', 'color: #28a745', type, data)
  68. this.setState(prev => ({
  69. loggedUser: {
  70. ...prev.loggedUser,
  71. lang: data
  72. }
  73. }))
  74. i18n.changeLanguage(data)
  75. break
  76. }
  77. }
  78. componentDidMount () {
  79. console.log('%c<HtmlDocument> did mount', `color: ${this.state.config.hexcolor}`)
  80. this.loadContent()
  81. }
  82. componentDidUpdate (prevProps, prevState) {
  83. const { state } = this
  84. console.log('%c<HtmlDocument> did update', `color: ${this.state.config.hexcolor}`, prevState, state)
  85. if (!prevState.content || !state.content) return
  86. if (prevState.content.content_id !== state.content.content_id) this.loadContent()
  87. if (state.mode === MODE.EDIT && prevState.mode !== state.mode) {
  88. tinymce.remove('#wysiwygNewVersion')
  89. wysiwyg('#wysiwygNewVersion', this.handleChangeText)
  90. }
  91. if (!prevState.timelineWysiwyg && state.timelineWysiwyg) wysiwyg('#wysiwygTimelineComment', this.handleChangeNewComment)
  92. else if (prevState.timelineWysiwyg && !state.timelineWysiwyg) tinymce.remove('#wysiwygTimelineComment')
  93. }
  94. componentWillUnmount () {
  95. console.log('%c<HtmlDocument> will Unmount', `color: ${this.state.config.hexcolor}`)
  96. document.removeEventListener('appCustomEvent', this.customEventReducer)
  97. }
  98. loadContent = async () => {
  99. const { loggedUser, content, config } = this.state
  100. const fetchResultHtmlDocument = getHtmlDocContent(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  101. const fetchResultComment = getHtmlDocComment(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  102. const fetchResultRevision = getHtmlDocRevision(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  103. handleFetchResult(await fetchResultHtmlDocument)
  104. .then(resHtmlDocument => this.setState({content: resHtmlDocument.body}))
  105. .catch(e => console.log('Error loading content.', e))
  106. Promise.all([
  107. handleFetchResult(await fetchResultComment),
  108. handleFetchResult(await fetchResultRevision)
  109. ])
  110. .then(([resComment, resRevision]) => {
  111. const resCommentWithProperDateAndAvatar = resComment.body.map(c => ({
  112. ...c,
  113. created: (new Date(c.created)).toLocaleString(),
  114. author: {
  115. ...c.author,
  116. avatar_url: c.author.avatar_url
  117. ? c.author.avatar_url
  118. : generateAvatarFromPublicName(c.author.public_name)
  119. }
  120. }))
  121. const revisionWithComment = resRevision.body
  122. .map((r, i) => ({
  123. ...r,
  124. created: (new Date(r.created)).toLocaleString(),
  125. timelineType: 'revision',
  126. commentList: r.comment_ids.map(ci => ({
  127. timelineType: 'comment',
  128. ...resCommentWithProperDateAndAvatar.find(c => c.content_id === ci)
  129. })),
  130. number: i + 1
  131. }))
  132. .reduce((acc, rev) => [
  133. ...acc,
  134. rev,
  135. ...rev.commentList.map(comment => ({
  136. ...comment,
  137. customClass: '',
  138. loggedUser: this.state.config.loggedUser
  139. }))
  140. ], [])
  141. this.setState({
  142. timeline: revisionWithComment,
  143. mode: resRevision.body.length === 1 ? MODE.EDIT : MODE.VIEW // first time editing the doc, open in edit mode
  144. })
  145. })
  146. .catch(e => {
  147. console.log('Error loading Timeline.', e)
  148. this.setState({timeline: []})
  149. })
  150. await Promise.all([fetchResultHtmlDocument, fetchResultComment, fetchResultRevision])
  151. putHtmlDocRead(loggedUser, config.apiUrl, content.workspace_id, content.content_id) // mark as read after all requests are finished
  152. }
  153. handleClickBtnCloseApp = () => {
  154. this.setState({ isVisible: false })
  155. GLOBAL_dispatchEvent({type: 'appClosed', data: {}}) // handled by tracim_front::src/container/WorkspaceContent.jsx
  156. }
  157. handleSaveEditTitle = async newTitle => {
  158. const { loggedUser, config, content } = this.state
  159. const fetchResultSaveHtmlDoc = putHtmlDocContent(loggedUser, config.apiUrl, content.workspace_id, content.content_id, newTitle, content.raw_content)
  160. handleFetchResult(await fetchResultSaveHtmlDoc)
  161. .then(resSave => {
  162. if (resSave.apiResponse.status === 200) {
  163. this.loadContent()
  164. GLOBAL_dispatchEvent({ type: 'refreshContentList', data: {} })
  165. } else {
  166. console.warn('Error saving html-document. Result:', resSave, 'content:', content, 'config:', config)
  167. }
  168. })
  169. }
  170. handleClickNewVersion = () => this.setState(prev => ({
  171. rawContentBeforeEdit: prev.content.raw_content,
  172. mode: MODE.EDIT
  173. }))
  174. handleCloseNewVersion = () => {
  175. tinymce.remove('#wysiwygNewVersion')
  176. this.setState(prev => ({
  177. content: {
  178. ...prev.content,
  179. raw_content: prev.rawContentBeforeEdit
  180. },
  181. mode: MODE.VIEW
  182. }))
  183. }
  184. handleSaveHtmlDocument = async () => {
  185. const { loggedUser, content, config } = this.state
  186. const fetchResultSaveHtmlDoc = putHtmlDocContent(loggedUser, config.apiUrl, content.workspace_id, content.content_id, content.label, content.raw_content)
  187. handleFetchResult(await fetchResultSaveHtmlDoc)
  188. .then(resSave => {
  189. if (resSave.apiResponse.status === 200) {
  190. this.handleCloseNewVersion()
  191. this.loadContent()
  192. } else {
  193. console.warn('Error saving html-document. Result:', resSave, 'content:', content, 'config:', config)
  194. }
  195. })
  196. }
  197. handleChangeText = e => {
  198. const newText = e.target.value // because SyntheticEvent is pooled (react specificity)
  199. this.setState(prev => ({content: {...prev.content, raw_content: newText}}))
  200. }
  201. handleChangeNewComment = e => {
  202. const newComment = e.target.value
  203. this.setState({newComment})
  204. }
  205. handleClickValidateNewCommentBtn = async () => {
  206. const { loggedUser, config, content, newComment } = this.state
  207. const fetchResultSaveNewComment = await postHtmlDocNewComment(loggedUser, config.apiUrl, content.workspace_id, content.content_id, newComment)
  208. handleFetchResult(await fetchResultSaveNewComment)
  209. .then(resSave => {
  210. if (resSave.apiResponse.status === 200) {
  211. this.setState({newComment: ''})
  212. if (this.state.timelineWysiwyg) tinymce.get('wysiwygTimelineComment').setContent('')
  213. this.loadContent()
  214. } else {
  215. console.warn('Error saving html-document comment. Result:', resSave, 'content:', content, 'config:', config)
  216. }
  217. })
  218. }
  219. handleToggleWysiwyg = () => this.setState(prev => ({timelineWysiwyg: !prev.timelineWysiwyg}))
  220. handleChangeStatus = async newStatus => {
  221. const { loggedUser, config, content } = this.state
  222. const fetchResultSaveEditStatus = putHtmlDocStatus(loggedUser, config.apiUrl, content.workspace_id, content.content_id, newStatus)
  223. handleFetchResult(await fetchResultSaveEditStatus)
  224. .then(resSave => {
  225. if (resSave.status !== 204) { // 204 no content so dont take status from resSave.apiResponse.status
  226. console.warn('Error saving html-document comment. Result:', resSave, 'content:', content, 'config:', config)
  227. } else {
  228. this.loadContent()
  229. }
  230. })
  231. }
  232. handleClickArchive = async () => {
  233. const { loggedUser, config, content } = this.state
  234. const fetchResultArchive = await putHtmlDocIsArchived(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  235. switch (fetchResultArchive.status) {
  236. case 204: this.setState(prev => ({content: {...prev.content, is_archived: true}})); break
  237. default: GLOBAL_dispatchEvent({
  238. type: 'addFlashMsg',
  239. data: {
  240. msg: this.props.t('Error while archiving document'),
  241. type: 'warning',
  242. delay: undefined
  243. }
  244. })
  245. }
  246. }
  247. handleClickDelete = async () => {
  248. const { loggedUser, config, content } = this.state
  249. const fetchResultArchive = await putHtmlDocIsDeleted(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  250. switch (fetchResultArchive.status) {
  251. case 204: this.setState(prev => ({content: {...prev.content, is_deleted: true}})); break
  252. default: GLOBAL_dispatchEvent({
  253. type: 'addFlashMsg',
  254. data: {
  255. msg: this.props.t('Error while deleting document'),
  256. type: 'warning',
  257. delay: undefined
  258. }
  259. })
  260. }
  261. }
  262. handleClickRestoreArchived = async () => {
  263. const { loggedUser, config, content } = this.state
  264. const fetchResultRestore = await putHtmlDocRestoreArchived(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  265. switch (fetchResultRestore.status) {
  266. case 204: this.setState(prev => ({content: {...prev.content, is_archived: false}})); break
  267. default: GLOBAL_dispatchEvent({
  268. type: 'addFlashMsg',
  269. data: {
  270. msg: this.props.t('Error while restoring document'),
  271. type: 'warning',
  272. delay: undefined
  273. }
  274. })
  275. }
  276. }
  277. handleClickRestoreDeleted = async () => {
  278. const { loggedUser, config, content } = this.state
  279. const fetchResultRestore = await putHtmlDocRestoreDeleted(loggedUser, config.apiUrl, content.workspace_id, content.content_id)
  280. switch (fetchResultRestore.status) {
  281. case 204: this.setState(prev => ({content: {...prev.content, is_deleted: false}})); break
  282. default: GLOBAL_dispatchEvent({
  283. type: 'addFlashMsg',
  284. data: {
  285. msg: this.props.t('Error while restoring document'),
  286. type: 'warning',
  287. delay: undefined
  288. }
  289. })
  290. }
  291. }
  292. handleClickShowRevision = revision => {
  293. const { mode, timeline } = this.state
  294. const revisionArray = timeline.filter(t => t.timelineType === 'revision')
  295. const isLastRevision = revision.revision_id === revisionArray[revisionArray.length - 1].revision_id
  296. if (mode === MODE.REVISION && isLastRevision) {
  297. this.handleClickLastVersion()
  298. return
  299. }
  300. if (mode === MODE.VIEW && isLastRevision) return
  301. this.setState(prev => ({
  302. content: {
  303. ...prev.content,
  304. label: revision.label,
  305. raw_content: revision.raw_content,
  306. number: revision.number,
  307. status: revision.status,
  308. is_archived: prev.is_archived, // archived and delete should always be taken from last version
  309. is_deleted: prev.is_deleted
  310. },
  311. mode: MODE.REVISION
  312. }))
  313. }
  314. handleClickLastVersion = () => {
  315. this.loadContent()
  316. this.setState({mode: MODE.VIEW})
  317. }
  318. render () {
  319. const { isVisible, loggedUser, content, timeline, newComment, timelineWysiwyg, config, mode } = this.state
  320. const { t } = this.props
  321. if (!isVisible) return null
  322. return (
  323. <PopinFixed
  324. customClass={`${config.slug}`}
  325. customColor={config.hexcolor}
  326. >
  327. <PopinFixedHeader
  328. customClass={`${config.slug}`}
  329. customColor={config.hexcolor}
  330. faIcon={config.faIcon}
  331. title={content.label}
  332. idRoleUserWorkspace={loggedUser.idRoleUserWorkspace}
  333. onClickCloseBtn={this.handleClickBtnCloseApp}
  334. onValidateChangeTitle={this.handleSaveEditTitle}
  335. />
  336. <PopinFixedOption
  337. customColor={config.hexcolor}
  338. customClass={`${config.slug}`}
  339. i18n={i18n}
  340. >
  341. <div /* this div in display flex, justify-content space-between */>
  342. <div className='d-flex'>
  343. {loggedUser.idRoleUserWorkspace >= 2 &&
  344. <NewVersionBtn
  345. customColor={config.hexcolor}
  346. onClickNewVersionBtn={this.handleClickNewVersion}
  347. disabled={mode !== MODE.VIEW}
  348. />
  349. }
  350. {mode === MODE.REVISION &&
  351. <button
  352. className='wsContentGeneric__option__menu__lastversion html-document__lastversionbtn btn'
  353. onClick={this.handleClickLastVersion}
  354. style={{backgroundColor: config.hexcolor, color: '#fdfdfd'}}
  355. >
  356. <i className='fa fa-code-fork' />
  357. {t('Last version')}
  358. </button>
  359. }
  360. </div>
  361. <div className='d-flex'>
  362. {loggedUser.idRoleUserWorkspace >= 2 &&
  363. <SelectStatus
  364. selectedStatus={config.availableStatuses.find(s => s.slug === content.status)}
  365. availableStatus={config.availableStatuses}
  366. onChangeStatus={this.handleChangeStatus}
  367. disabled={mode === MODE.REVISION}
  368. />
  369. }
  370. {loggedUser.idRoleUserWorkspace >= 4 &&
  371. <ArchiveDeleteContent
  372. customColor={config.hexcolor}
  373. onClickArchiveBtn={this.handleClickArchive}
  374. onClickDeleteBtn={this.handleClickDelete}
  375. disabled={mode === MODE.REVISION}
  376. />
  377. }
  378. </div>
  379. </div>
  380. </PopinFixedOption>
  381. <PopinFixedContent
  382. customClass={`${config.slug}__contentpage`}
  383. showRightPartOnLoad={mode === MODE.VIEW}
  384. >
  385. <HtmlDocumentComponent
  386. mode={mode}
  387. customColor={config.hexcolor}
  388. wysiwygNewVersion={'wysiwygNewVersion'}
  389. onClickCloseEditMode={this.handleCloseNewVersion}
  390. onClickValidateBtn={this.handleSaveHtmlDocument}
  391. version={content.number}
  392. lastVersion={timeline.filter(t => t.timelineType === 'revision').length}
  393. text={content.raw_content}
  394. onChangeText={this.handleChangeText}
  395. isArchived={content.is_archived}
  396. isDeleted={content.is_deleted}
  397. onClickRestoreArchived={this.handleClickRestoreArchived}
  398. onClickRestoreDeleted={this.handleClickRestoreDeleted}
  399. key={'html-document'}
  400. />
  401. <Timeline
  402. customClass={`${config.slug}__contentpage`}
  403. customColor={config.hexcolor}
  404. loggedUser={loggedUser}
  405. timelineData={timeline}
  406. newComment={newComment}
  407. disableComment={mode === MODE.REVISION}
  408. wysiwyg={timelineWysiwyg}
  409. onChangeNewComment={this.handleChangeNewComment}
  410. onClickValidateNewCommentBtn={this.handleClickValidateNewCommentBtn}
  411. onClickWysiwygBtn={this.handleToggleWysiwyg}
  412. onClickRevisionBtn={this.handleClickShowRevision}
  413. shouldScrollToBottom={mode !== MODE.REVISION}
  414. />
  415. </PopinFixedContent>
  416. </PopinFixed>
  417. )
  418. }
  419. }
  420. export default translate()(HtmlDocument)