/* eslint-disable max-len */
import React from 'react';
import ReactDOM from 'react-dom';
import { Provider } from 'unstated';
import { I18nextProvider } from 'react-i18next';
import * as toastr from 'toastr';
import loggerFactory from '@alias/logger';
import Xss from '@commons/service/xss';
import * as entities from 'entities';
import i18nFactory from './i18n';
import GrowiRenderer from './util/GrowiRenderer';
import HeaderSearchBox from './components/HeaderSearchBox';
import SearchPage from './components/SearchPage';
import TagsList from './components/TagsList';
import PageEditor from './components/PageEditor';
// eslint-disable-next-line import/no-duplicates
import OptionsSelector from './components/PageEditor/OptionsSelector';
// eslint-disable-next-line import/no-duplicates
import { defaultEditorOptions, defaultPreviewOptions } from './components/PageEditor/OptionsSelector';
import SavePageControls from './components/SavePageControls';
import PageEditorByHackmd from './components/PageEditorByHackmd';
import Page from './components/Page';
import PageHistory from './components/PageHistory';
import PageComments from './components/PageComments';
import CommentEditorLazyRenderer from './components/PageComment/CommentEditorLazyRenderer';
import PageAttachment from './components/PageAttachment';
import PageStatusAlert from './components/PageStatusAlert';
import RevisionPath from './components/Page/RevisionPath';
import TagLabels from './components/Page/TagLabels';
import BookmarkButton from './components/BookmarkButton';
import LikeButton from './components/LikeButton';
import PagePathAutoComplete from './components/PagePathAutoComplete';
import RecentCreated from './components/RecentCreated/RecentCreated';
import MyDraftList from './components/MyDraftList/MyDraftList';
import UserPictureList from './components/User/UserPictureList';
import CustomCssEditor from './components/Admin/CustomCssEditor';
import CustomScriptEditor from './components/Admin/CustomScriptEditor';
import CustomHeaderEditor from './components/Admin/CustomHeaderEditor';
import AdminRebuildSearch from './components/Admin/AdminRebuildSearch';
import GroupDeleteModal from './components/GroupDeleteModal/GroupDeleteModal';
import AppContainer from './services/AppContainer';
import PageContainer from './services/PageContainer';
import CommentContainer from './services/CommentContainer';
import EditorContainer from './services/EditorContainer';
import WebsocketContainer from './services/WebsocketContainer';
const logger = loggerFactory('growi:app');
if (!window) {
window = {};
}
const userlang = $('body').data('userlang');
const i18n = i18nFactory(userlang);
// setup xss library
const xss = new Xss();
window.xss = xss;
const mainContent = document.querySelector('#content-main');
let pageId = null;
let pageRevisionId = null;
let pageRevisionCreatedAt = null;
let pageRevisionIdHackmdSynced = null;
let hasDraftOnHackmd = false;
let pageIdOnHackmd = null;
let pagePath;
let pageContent = '';
let markdown = '';
let slackChannels;
let pageTags = [];
let templateTagData = '';
if (mainContent !== null) {
pageId = mainContent.getAttribute('data-page-id') || null;
pageRevisionId = mainContent.getAttribute('data-page-revision-id');
pageRevisionCreatedAt = +mainContent.getAttribute('data-page-revision-created');
pageRevisionIdHackmdSynced = mainContent.getAttribute('data-page-revision-id-hackmd-synced') || null;
pageIdOnHackmd = mainContent.getAttribute('data-page-id-on-hackmd') || null;
hasDraftOnHackmd = !!mainContent.getAttribute('data-page-has-draft-on-hackmd');
pagePath = mainContent.attributes['data-path'].value;
slackChannels = mainContent.getAttribute('data-slack-channels') || '';
templateTagData = mainContent.getAttribute('data-template-tags') || '';
const rawText = document.getElementById('raw-text-original');
if (rawText) {
pageContent = rawText.innerHTML;
}
markdown = entities.decodeHTML(pageContent);
}
const isLoggedin = document.querySelector('.main-container.nologin') == null;
// create unstated container instance
const appContainer = new AppContainer();
const websocketContainer = new WebsocketContainer(appContainer);
const pageContainer = new PageContainer(appContainer);
const commentContainer = new CommentContainer(appContainer);
const editorContainer = new EditorContainer(appContainer, defaultEditorOptions, defaultPreviewOptions);
const injectableContainers = [
appContainer, websocketContainer, pageContainer, commentContainer, editorContainer,
];
window.appContainer = appContainer;
logger.info('unstated containers have been initialized');
// backward compatibility
const crowi = appContainer;
window.crowi = appContainer;
if (isLoggedin) {
appContainer.fetchUsers();
}
const crowiRenderer = new GrowiRenderer(crowi, null, {
mode: 'page',
isAutoSetup: false, // manually setup because plugins may configure it
renderToc: appContainer.getCrowiForJquery().renderTocContent, // function for rendering Table Of Contents
});
window.crowiRenderer = crowiRenderer;
// FIXME
const isEnabledPlugins = $('body').data('plugin-enabled');
if (isEnabledPlugins) {
const crowiPlugin = window.crowiPlugin;
crowiPlugin.installAll(crowi, crowiRenderer);
}
/**
* receive tags from PageTagForm
* @param {Array} tagData new tags
*/
const setTagData = function(tagData) {
pageTags = tagData;
};
/**
* save success handler when reloading is not needed
* @param {object} page Page instance
*/
const saveWithShortcutSuccessHandler = function(page) {
const editorMode = appContainer.getCrowiForJquery().getCurrentEditorMode();
// show toastr
toastr.success(undefined, 'Saved successfully', {
closeButton: true,
progressBar: true,
newestOnTop: false,
showDuration: '100',
hideDuration: '100',
timeOut: '1200',
extendedTimeOut: '150',
});
// update state of PageContainer
const newState = {
pageId: page._id,
revisionId: page.revision._id,
remoteRevisionId: page.revision._id,
revisionIdHackmdSynced: page.revisionHackmdSynced,
hasDraftOnHackmd: page.hasDraftOnHackmd,
markdown: page.revision.body,
};
pageContainer.setState(newState);
// PageEditor component
const pageEditor = appContainer.getComponentInstance('PageEditor');
if (pageEditor != null) {
if (editorMode !== 'builtin') {
pageEditor.updateEditorValue(newState.markdown);
}
}
// PageEditorByHackmd component
const pageEditorByHackmd = appContainer.getComponentInstance('PageEditorByHackmd');
if (pageEditorByHackmd != null) {
// reset
if (editorMode !== 'hackmd') {
pageEditorByHackmd.reset();
}
}
// hidden input
$('input[name="revision_id"]').val(pageRevisionId);
};
const errorHandler = function(error) {
toastr.error(error.message, 'Error occured', {
closeButton: true,
progressBar: true,
newestOnTop: false,
showDuration: '100',
hideDuration: '100',
timeOut: '3000',
});
};
const saveWithShortcut = function(markdown) {
const editorMode = appContainer.getCrowiForJquery().getCurrentEditorMode();
let revisionId = pageRevisionId;
// get options
const options = pageContainer.getCurrentOptionsToSave();
options.socketClientId = websocketContainer.getCocketClientId();
options.pageTags = pageTags;
if (editorMode === 'hackmd') {
// set option to sync
options.isSyncRevisionToHackmd = true;
// use revisionId of PageEditorByHackmd
const pageEditorByHackmd = appContainer.getComponentInstance('PageEditorByHackmd');
revisionId = pageEditorByHackmd.getRevisionIdHackmdSynced();
}
let promise;
if (pageId == null) {
promise = appContainer.createPage(pagePath, markdown, options);
}
else {
promise = appContainer.updatePage(pageId, revisionId, markdown, options);
}
promise
.then(saveWithShortcutSuccessHandler)
.catch(errorHandler);
};
const saveWithSubmitButtonSuccessHandler = function() {
appContainer.clearDraft(pagePath);
window.location.href = pagePath;
};
const saveWithSubmitButton = function(submitOpts) {
const editorMode = appContainer.getCrowiForJquery().getCurrentEditorMode();
if (editorMode == null) {
// do nothing
return;
}
let revisionId = pageRevisionId;
// get options
const options = pageContainer.getCurrentOptionsToSave();
options.socketClientId = websocketContainer.getCocketClientId();
options.pageTags = pageTags;
// set 'submitOpts.overwriteScopesOfDescendants' to options
options.overwriteScopesOfDescendants = submitOpts ? !!submitOpts.overwriteScopesOfDescendants : false;
let promise;
if (editorMode === 'hackmd') {
const pageEditorByHackmd = appContainer.getComponentInstance('PageEditorByHackmd');
// get markdown
promise = pageEditorByHackmd.getMarkdown();
// use revisionId of PageEditorByHackmd
revisionId = pageEditorByHackmd.getRevisionIdHackmdSynced();
// set option to sync
options.isSyncRevisionToHackmd = true;
}
else {
const pageEditor = appContainer.getComponentInstance('PageEditor');
// get markdown
promise = Promise.resolve(pageEditor.getMarkdown());
}
// create or update
if (pageId == null) {
promise = promise.then((markdown) => {
return appContainer.createPage(pagePath, markdown, options);
});
}
else {
promise = promise.then((markdown) => {
return appContainer.updatePage(pageId, revisionId, markdown, options);
});
}
promise
.then(saveWithSubmitButtonSuccessHandler)
.catch(errorHandler);
};
// setup renderer after plugins are installed
crowiRenderer.setup();
// restore draft when the first time to edit
const draft = appContainer.findDraft(pagePath);
if (!pageRevisionId && draft != null) {
markdown = draft;
}
/**
* define components
* key: id of element
* value: React Element
*/
let componentMappings = {
'search-top': ,
'search-sidebar': ,
'search-page': ,
// 'revision-history': ,
'tags-page': ,
'create-page-name-input': ,
'page-editor': ,
'page-editor-options-selector': ,
'page-status-alert': ,
'save-page-controls': ,
};
// additional definitions if data exists
if (pageId) {
componentMappings = Object.assign({
'page-editor-with-hackmd': ,
'page-comments-list': ,
'page-attachment': ,
'page-comment-write': ,
'bookmark-button': ,
'bookmark-button-lg': ,
'rename-page-name-input': ,
'duplicate-page-name-input': ,
}, componentMappings);
}
if (pagePath) {
componentMappings = Object.assign({
// eslint-disable-next-line quote-props
'page': ,
'revision-path': ,
'tag-label': ,
}, componentMappings);
}
Object.keys(componentMappings).forEach((key) => {
const elem = document.getElementById(key);
if (elem) {
ReactDOM.render(
{componentMappings[key]}
,
elem,
);
}
});
// render LikeButton
const likeButtonElem = document.getElementById('like-button');
if (likeButtonElem) {
const isLiked = likeButtonElem.dataset.liked === 'true';
ReactDOM.render(
,
likeButtonElem,
);
}
// render UserPictureList for seen-user-list
const seenUserListElem = document.getElementById('seen-user-list');
if (seenUserListElem) {
const userIdsStr = seenUserListElem.dataset.userIds;
const userIds = userIdsStr.split(',');
ReactDOM.render(
,
seenUserListElem,
);
}
// render UserPictureList for liker-list
const likerListElem = document.getElementById('liker-list');
if (likerListElem) {
const userIdsStr = likerListElem.dataset.userIds;
const userIds = userIdsStr.split(',');
ReactDOM.render(
,
likerListElem,
);
}
const recentCreatedControlsElem = document.getElementById('user-created-list');
if (recentCreatedControlsElem) {
let limit = appContainer.getConfig().recentCreatedLimit;
if (limit == null) {
limit = 10;
}
ReactDOM.render(
, document.getElementById('user-created-list'),
);
}
const myDraftControlsElem = document.getElementById('user-draft-list');
if (myDraftControlsElem) {
let limit = appContainer.getConfig().recentCreatedLimit;
if (limit == null) {
limit = 10;
}
ReactDOM.render(
,
myDraftControlsElem,
);
}
// render for admin
const customCssEditorElem = document.getElementById('custom-css-editor');
if (customCssEditorElem != null) {
// get input[type=hidden] element
const customCssInputElem = document.getElementById('inputCustomCss');
ReactDOM.render(
,
customCssEditorElem,
);
}
const customScriptEditorElem = document.getElementById('custom-script-editor');
if (customScriptEditorElem != null) {
// get input[type=hidden] element
const customScriptInputElem = document.getElementById('inputCustomScript');
ReactDOM.render(
,
customScriptEditorElem,
);
}
const customHeaderEditorElem = document.getElementById('custom-header-editor');
if (customHeaderEditorElem != null) {
// get input[type=hidden] element
const customHeaderInputElem = document.getElementById('inputCustomHeader');
ReactDOM.render(
,
customHeaderEditorElem,
);
}
const adminRebuildSearchElem = document.getElementById('admin-rebuild-search');
if (adminRebuildSearchElem != null) {
ReactDOM.render(
,
adminRebuildSearchElem,
);
}
const adminGrantSelectorElem = document.getElementById('admin-delete-user-group-modal');
if (adminGrantSelectorElem != null) {
ReactDOM.render(
,
adminGrantSelectorElem,
);
}
// うわーもうー (commented by Crowi team -- 2018.03.23 Yuki Takei)
$('a[data-toggle="tab"][href="#revision-history"]').on('show.bs.tab', () => {
ReactDOM.render(
, document.getElementById('revision-history'),
);
});