mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-01 15:29:19 +03:00
2734a88ca9
closes https://linear.app/tryghost/issue/MOM-160 - return `undefined` early from `searchLinks` when the underlying task gets cancelled - bump `@tryghost/koenig-lexical` so it properly handles cancelled search promises
678 lines
24 KiB
JavaScript
678 lines
24 KiB
JavaScript
import * as Sentry from '@sentry/ember';
|
|
import Component from '@glimmer/component';
|
|
import React, {Suspense} from 'react';
|
|
import fetch from 'fetch';
|
|
import ghostPaths from 'ghost-admin/utils/ghost-paths';
|
|
import moment from 'moment-timezone';
|
|
import {action} from '@ember/object';
|
|
import {didCancel, task} from 'ember-concurrency';
|
|
import {inject} from 'ghost-admin/decorators/inject';
|
|
import {inject as service} from '@ember/service';
|
|
|
|
export const fileTypes = {
|
|
image: {
|
|
mimeTypes: ['image/gif', 'image/jpg', 'image/jpeg', 'image/png', 'image/svg+xml', 'image/webp'],
|
|
extensions: ['gif', 'jpg', 'jpeg', 'png', 'svg', 'svgz', 'webp'],
|
|
endpoint: '/images/upload/',
|
|
resourceName: 'images'
|
|
},
|
|
video: {
|
|
mimeTypes: ['video/mp4', 'video/webm', 'video/ogg'],
|
|
extensions: ['mp4', 'webm', 'ogv'],
|
|
endpoint: '/media/upload/',
|
|
resourceName: 'media'
|
|
},
|
|
audio: {
|
|
mimeTypes: ['audio/mp3', 'audio/mpeg', 'audio/ogg', 'audio/wav', 'audio/vnd.wav', 'audio/wave', 'audio/x-wav', 'audio/mp4', 'audio/x-m4a'],
|
|
extensions: ['mp3', 'wav', 'ogg', 'm4a'],
|
|
endpoint: '/media/upload/',
|
|
resourceName: 'media'
|
|
},
|
|
mediaThumbnail: {
|
|
mimeTypes: ['image/gif', 'image/jpg', 'image/jpeg', 'image/png', 'image/webp'],
|
|
extensions: ['gif', 'jpg', 'jpeg', 'png', 'webp'],
|
|
endpoint: '/media/thumbnail/upload/',
|
|
requestMethod: 'put',
|
|
resourceName: 'media'
|
|
},
|
|
file: {
|
|
endpoint: '/files/upload/',
|
|
resourceName: 'files'
|
|
}
|
|
};
|
|
|
|
function LockIcon({...props}) {
|
|
return (
|
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" {...props}>
|
|
<g transform="matrix(0.6666666666666666,0,0,0.6666666666666666,0,0)">
|
|
<path fill="currentColor" d="M19.5,9.5h-.75V6.75a6.75,6.75,0,0,0-13.5,0V9.5H4.5a2,2,0,0,0-2,2V22a2,2,0,0,0,2,2h15a2,2,0,0,0,2-2V11.5A2,2,0,0,0,19.5,9.5Zm-7.5,9a2,2,0,1,1,2-2A2,2,0,0,1,12,18.5ZM16.25,9a.5.5,0,0,1-.5.5H8.25a.5.5,0,0,1-.5-.5V6.75a4.25,4.25,0,0,1,8.5,0Z"></path>
|
|
</g>
|
|
</svg>
|
|
);
|
|
}
|
|
|
|
function DollarIcon({...props}) {
|
|
return (
|
|
<svg xmlns="http://www.w3.org/2000/svg" height="24" width="24" viewBox="0 0 24 24" {...props}>
|
|
<g fill="currentColor" class="nc-icon-wrapper">
|
|
<path
|
|
d="M13,10.265V5.013a9.722,9.722,0,0,1,2.6.722l1.342.662,1.327-2.69-1.345-.663A12.4,12.4,0,0,0,13,1.989V0H11V1.983c-3.537.306-5.773,2.3-5.773,5.264,0,3.726,3.174,4.85,5.773,5.577V18.09a15.77,15.77,0,0,1-4.24-.819l-1.411-.509L4.33,19.583l1.411.51A18.577,18.577,0,0,0,11,21.1V24h2V21.087c5.125-.431,5.708-3.776,5.708-5.264C18.708,12.129,15.587,10.993,13,10.265ZM8.227,7.247c0-1.6,1.6-2.1,2.773-2.249V9.69C9.1,9.092,8.227,8.523,8.227,7.247ZM13,18.072V13.4c1.857.591,2.708,1.161,2.708,2.422C15.708,16.382,15.7,17.769,13,18.072Z"
|
|
fill="currentColor"
|
|
></path>
|
|
</g>
|
|
</svg>
|
|
);
|
|
}
|
|
|
|
export function decoratePostSearchResult(item, settings) {
|
|
const date = moment.utc(item.publishedAt).tz(settings.timezone).format('D MMM YYYY');
|
|
|
|
item.metaText = date;
|
|
|
|
if (settings.membersEnabled && item.visibility) {
|
|
if (item.visibility === 'members') {
|
|
item.MetaIcon = LockIcon;
|
|
item.metaIconTitle = 'Members only';
|
|
} else if (item.visibility === 'paid') {
|
|
item.MetaIcon = DollarIcon;
|
|
item.metaIconTitle = 'Paid-members only';
|
|
} else if (item.visibility === 'tiers') {
|
|
item.MetaIcon = DollarIcon;
|
|
item.metaIconTitle = 'Specific tiers only';
|
|
}
|
|
}
|
|
}
|
|
|
|
class ErrorHandler extends React.Component {
|
|
state = {
|
|
hasError: false
|
|
};
|
|
|
|
static getDerivedStateFromError() {
|
|
return {hasError: true};
|
|
}
|
|
|
|
componentDidCatch(error) {
|
|
if (this.props.config.sentry_dsn) {
|
|
Sentry.captureException(error, {
|
|
tags: {
|
|
lexical: true
|
|
}
|
|
});
|
|
}
|
|
|
|
console.error(error); // eslint-disable-line
|
|
}
|
|
|
|
render() {
|
|
if (this.state.hasError) {
|
|
return (
|
|
<p className="koenig-react-editor-error">Loading has failed. Try refreshing the browser!</p>
|
|
);
|
|
}
|
|
|
|
return this.props.children;
|
|
}
|
|
}
|
|
|
|
const KoenigComposer = ({editorResource, ...props}) => {
|
|
const {KoenigComposer: _KoenigComposer} = editorResource.read();
|
|
return <_KoenigComposer {...props} />;
|
|
};
|
|
|
|
const KoenigEditor = ({editorResource, ...props}) => {
|
|
const {KoenigEditor: _KoenigEditor} = editorResource.read();
|
|
return <_KoenigEditor {...props} />;
|
|
};
|
|
|
|
const WordCountPlugin = ({editorResource, ...props}) => {
|
|
const {WordCountPlugin: _WordCountPlugin} = editorResource.read();
|
|
return <_WordCountPlugin {...props} />;
|
|
};
|
|
|
|
const TKCountPlugin = ({editorResource, ...props}) => {
|
|
const {TKCountPlugin: _TKCountPlugin} = editorResource.read();
|
|
return <_TKCountPlugin {...props} />;
|
|
};
|
|
|
|
export default class KoenigLexicalEditor extends Component {
|
|
@service ajax;
|
|
@service feature;
|
|
@service ghostPaths;
|
|
@service koenig;
|
|
@service membersUtils;
|
|
@service search;
|
|
@service session;
|
|
@service settings;
|
|
@service store;
|
|
|
|
@inject config;
|
|
|
|
offers = null;
|
|
contentKey = null;
|
|
defaultLinks = null;
|
|
|
|
editorResource = this.koenig.resource;
|
|
|
|
get pinturaJsUrl() {
|
|
if (!this.settings.pintura) {
|
|
return null;
|
|
}
|
|
return this.config.pintura?.js || this.settings.pinturaJsUrl;
|
|
}
|
|
|
|
get pinturaCSSUrl() {
|
|
if (!this.settings.pintura) {
|
|
return null;
|
|
}
|
|
return this.config.pintura?.css || this.settings.pinturaCssUrl;
|
|
}
|
|
|
|
get pinturaConfig() {
|
|
const jsUrl = this.getImageEditorJSUrl();
|
|
const cssUrl = this.getImageEditorCSSUrl();
|
|
if (!jsUrl || !cssUrl) {
|
|
return null;
|
|
}
|
|
return {
|
|
jsUrl,
|
|
cssUrl
|
|
};
|
|
}
|
|
|
|
getImageEditorJSUrl() {
|
|
let importUrl = this.pinturaJsUrl;
|
|
|
|
if (!importUrl) {
|
|
return null;
|
|
}
|
|
|
|
// load the script from admin root if relative
|
|
if (importUrl.startsWith('/')) {
|
|
importUrl = window.location.origin + this.ghostPaths.adminRoot.replace(/\/$/, '') + importUrl;
|
|
}
|
|
return importUrl;
|
|
}
|
|
|
|
getImageEditorCSSUrl() {
|
|
let cssImportUrl = this.pinturaCSSUrl;
|
|
|
|
if (!cssImportUrl) {
|
|
return null;
|
|
}
|
|
|
|
// load the css from admin root if relative
|
|
if (cssImportUrl.startsWith('/')) {
|
|
cssImportUrl = window.location.origin + this.ghostPaths.adminRoot.replace(/\/$/, '') + cssImportUrl;
|
|
}
|
|
return cssImportUrl;
|
|
}
|
|
|
|
@action
|
|
onError(error) {
|
|
// ensure we're still showing errors in development
|
|
console.error(error); // eslint-disable-line
|
|
|
|
if (this.config.sentry_dsn) {
|
|
Sentry.captureException(error, {
|
|
tags: {lexical: true},
|
|
contexts: {
|
|
koenig: {
|
|
version: window['@tryghost/koenig-lexical']?.version
|
|
}
|
|
}
|
|
});
|
|
}
|
|
// don't rethrow, Lexical will attempt to gracefully recover
|
|
}
|
|
|
|
@task({restartable: true})
|
|
*fetchOffersTask() {
|
|
if (this.offers) {
|
|
return this.offers;
|
|
}
|
|
this.offers = yield this.store.query('offer', {limit: 'all', filter: 'status:active'});
|
|
return this.offers;
|
|
}
|
|
|
|
@task({restartable: true})
|
|
*fetchLabelsTask() {
|
|
if (this.labels) {
|
|
return this.labels;
|
|
}
|
|
|
|
this.labels = yield this.store.query('label', {limit: 'all', fields: 'id, name'});
|
|
return this.labels;
|
|
}
|
|
|
|
ReactComponent = (props) => {
|
|
const fetchEmbed = async (url, {type}) => {
|
|
let oembedEndpoint = this.ghostPaths.url.api('oembed');
|
|
let response = await this.ajax.request(oembedEndpoint, {
|
|
data: {url, type}
|
|
});
|
|
return response;
|
|
};
|
|
|
|
const fetchCollectionPosts = async (collectionSlug) => {
|
|
if (!this.contentKey) {
|
|
const integrations = await this.store.findAll('integration');
|
|
const contentIntegration = integrations.findBy('slug', 'ghost-core-content');
|
|
this.contentKey = contentIntegration?.contentKey.secret;
|
|
}
|
|
|
|
const postsUrl = new URL(this.ghostPaths.url.admin('/api/content/posts/'), window.location.origin);
|
|
postsUrl.searchParams.append('key', this.contentKey);
|
|
postsUrl.searchParams.append('collection', collectionSlug);
|
|
postsUrl.searchParams.append('limit', 12);
|
|
|
|
const response = await fetch(postsUrl.toString());
|
|
const {posts} = await response.json();
|
|
|
|
return posts;
|
|
};
|
|
|
|
const fetchAutocompleteLinks = async () => {
|
|
const offers = await this.fetchOffersTask.perform();
|
|
|
|
const defaults = [
|
|
{label: 'Homepage', value: window.location.origin + '/'},
|
|
{label: 'Free signup', value: '#/portal/signup/free'}
|
|
];
|
|
|
|
const memberLinks = () => {
|
|
let links = [];
|
|
if (this.membersUtils.paidMembersEnabled) {
|
|
links = [
|
|
{
|
|
label: 'Paid signup',
|
|
value: '#/portal/signup'
|
|
},
|
|
{
|
|
label: 'Upgrade or change plan',
|
|
value: '#/portal/account/plans'
|
|
}];
|
|
}
|
|
|
|
return links;
|
|
};
|
|
|
|
const donationLink = () => {
|
|
if (this.feature.tipsAndDonations && this.settings.donationsEnabled) {
|
|
return [{
|
|
label: 'Tip or donation',
|
|
value: '#/portal/support'
|
|
}];
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
const recommendationLink = () => {
|
|
if (this.settings.recommendationsEnabled) {
|
|
return [{
|
|
label: 'Recommendations',
|
|
value: '#/portal/recommendations'
|
|
}];
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
const offersLinks = offers.toArray().map((offer) => {
|
|
return {
|
|
label: `Offer - ${offer.name}`,
|
|
value: this.config.getSiteUrl(offer.code)
|
|
};
|
|
});
|
|
|
|
return [...defaults, ...memberLinks(), ...donationLink(), ...recommendationLink(), ...offersLinks];
|
|
};
|
|
|
|
const fetchLabels = async () => {
|
|
const labels = await this.fetchLabelsTask.perform();
|
|
|
|
return labels.map(label => label.name);
|
|
};
|
|
|
|
const searchLinks = async (term) => {
|
|
// when no term is present we should show latest 5 posts
|
|
if (!term) {
|
|
// we cache the default links to avoid fetching them every time
|
|
if (this.defaultLinks) {
|
|
return this.defaultLinks;
|
|
}
|
|
|
|
const posts = await this.store.query('post', {filter: 'status:published', fields: 'id,url,title,visibility,published_at', order: 'published_at desc', limit: 5});
|
|
// NOTE: these posts are Ember Data models, not plain objects like the search results
|
|
const results = posts.toArray().map(post => ({
|
|
groupName: 'Latest posts',
|
|
id: post.id,
|
|
title: post.title,
|
|
url: post.url,
|
|
visibility: post.visibility,
|
|
publishedAt: post.publishedAtUTC.toISOString()
|
|
}));
|
|
|
|
results.forEach(item => decoratePostSearchResult(item, this.settings));
|
|
|
|
this.defaultLinks = [{
|
|
label: 'Latest posts',
|
|
items: results
|
|
}];
|
|
return this.defaultLinks;
|
|
}
|
|
|
|
let results = [];
|
|
|
|
try {
|
|
results = await this.search.searchTask.perform(term);
|
|
} catch (error) {
|
|
// don't surface task cancellation errors
|
|
if (!didCancel(error)) {
|
|
throw error;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// only published posts/pages and staff with posts have URLs
|
|
const filteredResults = [];
|
|
results.forEach((group) => {
|
|
let items = group.options;
|
|
|
|
if (group.groupName === 'Posts' || group.groupName === 'Pages') {
|
|
items = items.filter(i => i.status === 'published');
|
|
}
|
|
|
|
if (group.groupName === 'Staff') {
|
|
items = items.filter(i => !/\/404\//.test(i.url));
|
|
}
|
|
|
|
if (items.length === 0) {
|
|
return;
|
|
}
|
|
|
|
// update the group items with metadata
|
|
if (group.groupName === 'Posts' || group.groupName === 'Pages') {
|
|
items.forEach(item => decoratePostSearchResult(item, this.settings));
|
|
}
|
|
|
|
filteredResults.push({
|
|
label: group.groupName,
|
|
items
|
|
});
|
|
});
|
|
|
|
return filteredResults;
|
|
};
|
|
|
|
const unsplashConfig = {
|
|
defaultHeaders: {
|
|
Authorization: `Client-ID 8672af113b0a8573edae3aa3713886265d9bb741d707f6c01a486cde8c278980`,
|
|
'Accept-Version': 'v1',
|
|
'Content-Type': 'application/json',
|
|
'App-Pragma': 'no-cache',
|
|
'X-Unsplash-Cache': true
|
|
}
|
|
};
|
|
|
|
const defaultCardConfig = {
|
|
unsplash: this.settings.unsplash ? unsplashConfig.defaultHeaders : null,
|
|
tenor: this.config.tenor?.googleApiKey ? this.config.tenor : null,
|
|
fetchAutocompleteLinks,
|
|
fetchCollectionPosts,
|
|
fetchEmbed,
|
|
fetchLabels,
|
|
feature: {
|
|
collectionsCard: this.feature.collectionsCard,
|
|
collections: this.feature.collections,
|
|
internalLinking: this.feature.internalLinking,
|
|
internalLinkingAtLinks: this.feature.internalLinkingAtLinks
|
|
},
|
|
deprecated: {
|
|
headerV1: true // if false, shows header v1 in the menu
|
|
},
|
|
membersEnabled: this.settings.membersSignupAccess === 'all',
|
|
searchLinks,
|
|
siteTitle: this.settings.title,
|
|
siteDescription: this.settings.description
|
|
};
|
|
const cardConfig = Object.assign({}, defaultCardConfig, props.cardConfig, {pinturaConfig: this.pinturaConfig});
|
|
|
|
const useFileUpload = (type = 'image') => {
|
|
const [progress, setProgress] = React.useState(0);
|
|
const [isLoading, setLoading] = React.useState(false);
|
|
const [errors, setErrors] = React.useState([]);
|
|
const [filesNumber, setFilesNumber] = React.useState(0);
|
|
|
|
const progressTracker = React.useRef(new Map());
|
|
|
|
function updateProgress() {
|
|
if (progressTracker.current.size === 0) {
|
|
setProgress(0);
|
|
return;
|
|
}
|
|
|
|
let totalProgress = 0;
|
|
|
|
progressTracker.current.forEach(value => totalProgress += value);
|
|
|
|
setProgress(Math.round(totalProgress / progressTracker.current.size));
|
|
}
|
|
|
|
// we only check the file extension by default because IE doesn't always
|
|
// expose the mime-type, we'll rely on the API for final validation
|
|
function defaultValidator(file) {
|
|
// if type is file we don't need to validate since the card can accept any file type
|
|
if (type === 'file') {
|
|
return true;
|
|
}
|
|
let extensions = fileTypes[type].extensions;
|
|
let [, extension] = (/(?:\.([^.]+))?$/).exec(file.name);
|
|
|
|
// if extensions is falsy exit early and accept all files
|
|
if (!extensions) {
|
|
return true;
|
|
}
|
|
|
|
if (!Array.isArray(extensions)) {
|
|
extensions = extensions.split(',');
|
|
}
|
|
|
|
if (!extension || extensions.indexOf(extension.toLowerCase()) === -1) {
|
|
let validExtensions = `.${extensions.join(', .').toUpperCase()}`;
|
|
return `The file type you uploaded is not supported. Please use ${validExtensions}`;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
const validate = (files = []) => {
|
|
const validationResult = [];
|
|
|
|
for (let i = 0; i < files.length; i += 1) {
|
|
let file = files[i];
|
|
let result = defaultValidator(file);
|
|
if (result === true) {
|
|
continue;
|
|
}
|
|
|
|
validationResult.push({fileName: file.name, message: result});
|
|
}
|
|
|
|
return validationResult;
|
|
};
|
|
|
|
const _uploadFile = async (file, {formData = {}} = {}) => {
|
|
progressTracker.current[file] = 0;
|
|
|
|
const fileFormData = new FormData();
|
|
fileFormData.append('file', file, file.name);
|
|
|
|
Object.keys(formData || {}).forEach((key) => {
|
|
fileFormData.append(key, formData[key]);
|
|
});
|
|
|
|
const url = `${ghostPaths().apiRoot}${fileTypes[type].endpoint}`;
|
|
|
|
try {
|
|
const requestMethod = fileTypes[type].requestMethod || 'post';
|
|
const response = await this.ajax[requestMethod](url, {
|
|
data: fileFormData,
|
|
processData: false,
|
|
contentType: false,
|
|
dataType: 'text',
|
|
xhr: () => {
|
|
const xhr = new window.XMLHttpRequest();
|
|
|
|
xhr.upload.addEventListener('progress', (event) => {
|
|
if (event.lengthComputable) {
|
|
progressTracker.current.set(file, (event.loaded / event.total) * 100);
|
|
updateProgress();
|
|
}
|
|
}, false);
|
|
|
|
return xhr;
|
|
}
|
|
});
|
|
|
|
// force tracker progress to 100% in case we didn't get a final event
|
|
progressTracker.current.set(file, 100);
|
|
updateProgress();
|
|
|
|
let uploadResponse;
|
|
let responseUrl;
|
|
|
|
try {
|
|
uploadResponse = JSON.parse(response);
|
|
} catch (error) {
|
|
if (!(error instanceof SyntaxError)) {
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
if (uploadResponse) {
|
|
const resource = uploadResponse[fileTypes[type].resourceName];
|
|
if (resource && Array.isArray(resource) && resource[0]) {
|
|
responseUrl = resource[0].url;
|
|
}
|
|
}
|
|
|
|
return {
|
|
url: responseUrl,
|
|
fileName: file.name
|
|
};
|
|
} catch (error) {
|
|
console.error(error); // eslint-disable-line
|
|
|
|
// grab custom error message if present
|
|
let message = error.payload?.errors?.[0]?.message || '';
|
|
let context = error.payload?.errors?.[0]?.context || '';
|
|
|
|
// fall back to EmberData/ember-ajax default message for error type
|
|
if (!message) {
|
|
message = error.message;
|
|
}
|
|
|
|
// TODO: check for or expose known error types?
|
|
const errorResult = {
|
|
message,
|
|
context,
|
|
fileName: file.name
|
|
};
|
|
|
|
throw errorResult;
|
|
}
|
|
};
|
|
|
|
const upload = async (files = [], options = {}) => {
|
|
setFilesNumber(files.length);
|
|
setLoading(true);
|
|
|
|
const validationResult = validate(files);
|
|
|
|
if (validationResult.length) {
|
|
setErrors(validationResult);
|
|
setLoading(false);
|
|
setProgress(100);
|
|
|
|
return null;
|
|
}
|
|
|
|
const uploadPromises = [];
|
|
|
|
for (let i = 0; i < files.length; i += 1) {
|
|
const file = files[i];
|
|
uploadPromises.push(_uploadFile(file, options));
|
|
}
|
|
|
|
try {
|
|
const uploadResult = await Promise.all(uploadPromises);
|
|
setProgress(100);
|
|
progressTracker.current.clear();
|
|
|
|
setLoading(false);
|
|
|
|
setErrors([]); // components expect array of objects: { fileName: string, message: string }[]
|
|
|
|
return uploadResult;
|
|
} catch (error) {
|
|
console.error(error); // eslint-disable-line no-console
|
|
|
|
setErrors([...errors, error]);
|
|
setLoading(false);
|
|
setProgress(100);
|
|
progressTracker.current.clear();
|
|
|
|
return null;
|
|
}
|
|
};
|
|
|
|
return {progress, isLoading, upload, errors, filesNumber};
|
|
};
|
|
|
|
// TODO: react component isn't re-rendered when its props are changed meaning we don't transition
|
|
// to enabling multiplayer when a new post is saved and it gets an ID we can use for a YJS doc
|
|
// - figure out how to re-render the component when its props change
|
|
// - figure out some other mechanism for handling posts that don't really exist yet with multiplayer
|
|
const enableMultiplayer = this.feature.lexicalMultiplayer && !cardConfig.post.isNew;
|
|
const multiplayerWsProtocol = window.location.protocol === 'https:' ? `wss:` : `ws:`;
|
|
const multiplayerEndpoint = multiplayerWsProtocol + window.location.host + this.ghostPaths.url.api('posts', 'multiplayer');
|
|
const multiplayerDocId = cardConfig.post.id;
|
|
const multiplayerUsername = this.session.user.name;
|
|
|
|
return (
|
|
<div className={['koenig-react-editor', 'koenig-lexical', this.args.className].filter(Boolean).join(' ')}>
|
|
<ErrorHandler config={this.config}>
|
|
<Suspense fallback={<p className="koenig-react-editor-loading">Loading editor...</p>}>
|
|
<KoenigComposer
|
|
editorResource={this.editorResource}
|
|
cardConfig={cardConfig}
|
|
enableMultiplayer={enableMultiplayer}
|
|
fileUploader={{useFileUpload, fileTypes}}
|
|
initialEditorState={this.args.lexical}
|
|
multiplayerUsername={multiplayerUsername}
|
|
multiplayerDocId={multiplayerDocId}
|
|
multiplayerEndpoint={multiplayerEndpoint}
|
|
onError={this.onError}
|
|
darkMode={this.feature.nightShift}
|
|
isTKEnabled={true}
|
|
>
|
|
<KoenigEditor
|
|
editorResource={this.editorResource}
|
|
cursorDidExitAtTop={this.args.cursorDidExitAtTop}
|
|
placeholderText={this.args.placeholder}
|
|
darkMode={this.feature.nightShift}
|
|
onChange={this.args.onChange}
|
|
registerAPI={this.args.registerAPI}
|
|
/>
|
|
<WordCountPlugin editorResource={this.editorResource} onChange={this.args.updateWordCount} />
|
|
<TKCountPlugin editorResource={this.editorResource} onChange={this.args.updatePostTkCount} />
|
|
</KoenigComposer>
|
|
</Suspense>
|
|
</ErrorHandler>
|
|
</div>
|
|
);
|
|
};
|
|
}
|