1 import { Component, linkEvent } from 'inferno';
2 import { Prompt } from 'inferno-router';
3 import { PostListings } from './post-listings';
4 import { Subscription } from 'rxjs';
5 import { retryWhen, delay, take } from 'rxjs/operators';
13 ListCommunitiesResponse,
20 WebSocketJsonResponse,
21 } from '../interfaces';
22 import { WebSocketService, UserService } from '../services';
27 capitalizeFirstLetter,
40 import autosize from 'autosize';
41 import Tribute from 'tributejs/src/Tribute.js';
42 import emojiShortName from 'emoji-short-name';
43 import Selectr from 'mobius1-selectr';
44 import { i18n } from '../i18next';
46 const MAX_POST_TITLE_LENGTH = 200;
48 interface PostFormProps {
49 post?: Post; // If a post is given, that means this is an edit
50 params?: PostFormParams;
52 onCreate?(id: number): any;
53 onEdit?(post: Post): any;
56 interface PostFormState {
58 communities: Array<Community>;
60 imageLoading: boolean;
62 suggestedTitle: string;
63 suggestedPosts: Array<Post>;
64 crossPosts: Array<Post>;
68 export class PostForm extends Component<PostFormProps, PostFormState> {
69 private id = `post-form-${randomStr()}`;
70 private tribute: Tribute;
71 private subscription: Subscription;
72 private emptyState: PostFormState = {
78 creator_id: UserService.Instance.user
79 ? UserService.Instance.user.id
86 suggestedTitle: undefined,
89 enable_nsfw: undefined,
92 constructor(props: any, context: any) {
93 super(props, context);
94 this.fetchSimilarPosts = debounce(this.fetchSimilarPosts).bind(this);
95 this.fetchPageTitle = debounce(this.fetchPageTitle).bind(this);
97 this.tribute = setupTribute();
98 this.setupEmojiPicker();
100 this.state = this.emptyState;
102 if (this.props.post) {
103 this.state.postForm = {
104 body: this.props.post.body,
105 // NOTE: debouncing breaks both these for some reason, unless you use defaultValue
106 name: this.props.post.name,
107 community_id: this.props.post.community_id,
108 edit_id: this.props.post.id,
109 creator_id: this.props.post.creator_id,
110 url: this.props.post.url,
111 nsfw: this.props.post.nsfw,
116 if (this.props.params) {
117 this.state.postForm.name = this.props.params.name;
118 if (this.props.params.url) {
119 this.state.postForm.url = this.props.params.url;
121 if (this.props.params.body) {
122 this.state.postForm.body = this.props.params.body;
126 this.subscription = WebSocketService.Instance.subject
127 .pipe(retryWhen(errors => errors.pipe(delay(3000), take(10))))
129 msg => this.parseMessage(msg),
130 err => console.error(err),
131 () => console.log('complete')
134 let listCommunitiesForm: ListCommunitiesForm = {
135 sort: SortType[SortType.TopAll],
139 WebSocketService.Instance.listCommunities(listCommunitiesForm);
140 WebSocketService.Instance.getSite();
143 componentDidMount() {
144 var textarea: any = document.getElementById(this.id);
146 this.tribute.attach(textarea);
147 textarea.addEventListener('tribute-replaced', () => {
148 this.state.postForm.body = textarea.value;
149 this.setState(this.state);
150 autosize.update(textarea);
155 componentWillUnmount() {
156 this.subscription.unsubscribe();
164 !this.state.loading &&
165 (this.state.postForm.name ||
166 this.state.postForm.url ||
167 this.state.postForm.body)
169 message={i18n.t('block_leaving')}
171 <form onSubmit={linkEvent(this, this.handlePostSubmit)}>
172 <div class="form-group row">
173 <label class="col-sm-2 col-form-label" htmlFor="post-url">
176 <div class="col-sm-10">
181 value={this.state.postForm.url}
182 onInput={linkEvent(this, this.handlePostUrlChange)}
183 onPaste={linkEvent(this, this.handleImageUploadPaste)}
185 {this.state.suggestedTitle && (
187 class="mt-1 text-muted small font-weight-bold pointer"
188 onClick={linkEvent(this, this.copySuggestedTitle)}
190 {i18n.t('copy_suggested_title', {
191 title: this.state.suggestedTitle,
197 htmlFor="file-upload"
199 UserService.Instance.user && 'pointer'
200 } d-inline-block float-right text-muted font-weight-bold`}
201 data-tippy-content={i18n.t('upload_image')}
203 <svg class="icon icon-inline">
204 <use xlinkHref="#icon-image"></use>
210 accept="image/*,video/*"
213 disabled={!UserService.Instance.user}
214 onChange={linkEvent(this, this.handleImageUpload)}
217 {validURL(this.state.postForm.url) && (
219 href={`${archiveUrl}/?run=1&url=${encodeURIComponent(
220 this.state.postForm.url
223 class="mr-2 d-inline-block float-right text-muted small font-weight-bold"
225 {i18n.t('archive_link')}
228 {this.state.imageLoading && (
229 <svg class="icon icon-spinner spin">
230 <use xlinkHref="#icon-spinner"></use>
233 {isImage(this.state.postForm.url) && (
234 <img src={this.state.postForm.url} class="img-fluid" />
236 {this.state.crossPosts.length > 0 && (
238 <div class="my-1 text-muted small font-weight-bold">
239 {i18n.t('cross_posts')}
241 <PostListings showCommunity posts={this.state.crossPosts} />
246 <div class="form-group row">
247 <label class="col-sm-2 col-form-label" htmlFor="post-title">
250 <div class="col-sm-10">
252 value={this.state.postForm.name}
254 onInput={linkEvent(this, this.handlePostNameChange)}
259 maxLength={MAX_POST_TITLE_LENGTH}
261 {this.state.suggestedPosts.length > 0 && (
263 <div class="my-1 text-muted small font-weight-bold">
264 {i18n.t('related_posts')}
266 <PostListings posts={this.state.suggestedPosts} />
272 <div class="form-group row">
273 <label class="col-sm-2 col-form-label" htmlFor={this.id}>
276 <div class="col-sm-10">
279 value={this.state.postForm.body}
280 onInput={linkEvent(this, this.handlePostBodyChange)}
281 className={`form-control ${this.state.previewMode && 'd-none'}`}
285 {this.state.previewMode && (
288 dangerouslySetInnerHTML={mdToHtml(this.state.postForm.body)}
291 {this.state.postForm.body && (
293 className={`mt-1 mr-2 btn btn-sm btn-secondary ${
294 this.state.previewMode && 'active'
296 onClick={linkEvent(this, this.handlePreviewToggle)}
302 href={markdownHelpUrl}
304 class="d-inline-block float-right text-muted font-weight-bold"
305 title={i18n.t('formatting_help')}
307 <svg class="icon icon-inline">
308 <use xlinkHref="#icon-help-circle"></use>
312 onClick={linkEvent(this, this.handleEmojiPickerClick)}
313 class="pointer unselectable d-inline-block mr-3 float-right text-muted font-weight-bold"
314 data-tippy-content={i18n.t('emoji_picker')}
316 <svg class="icon icon-inline">
317 <use xlinkHref="#icon-smile"></use>
322 {!this.props.post && (
323 <div class="form-group row">
324 <label class="col-sm-2 col-form-label" htmlFor="post-community">
325 {i18n.t('community')}
327 <div class="col-sm-10">
331 value={this.state.postForm.community_id}
332 onInput={linkEvent(this, this.handlePostCommunityChange)}
334 <option>{i18n.t('select_a_community')}</option>
335 {this.state.communities.map(community => (
336 <option value={community.id}>
339 : `${hostname(community.actor_id)}/${community.name}`}
346 {this.state.enable_nsfw && (
347 <div class="form-group row">
348 <div class="col-sm-10">
349 <div class="form-check">
351 class="form-check-input"
354 checked={this.state.postForm.nsfw}
355 onChange={linkEvent(this, this.handlePostNsfwChange)}
357 <label class="form-check-label" htmlFor="post-nsfw">
364 <div class="form-group row">
365 <div class="col-sm-10">
367 disabled={!this.state.postForm.community_id}
369 class="btn btn-secondary mr-2"
371 {this.state.loading ? (
372 <svg class="icon icon-spinner spin">
373 <use xlinkHref="#icon-spinner"></use>
375 ) : this.props.post ? (
376 capitalizeFirstLetter(i18n.t('save'))
378 capitalizeFirstLetter(i18n.t('create'))
381 {this.props.post && (
384 class="btn btn-secondary"
385 onClick={linkEvent(this, this.handleCancel)}
398 emojiPicker.on('emoji', twemojiHtmlStr => {
399 if (this.state.postForm.body == null) {
400 this.state.postForm.body = '';
402 var el = document.createElement('div');
403 el.innerHTML = twemojiHtmlStr;
404 let nativeUnicode = (el.childNodes[0] as HTMLElement).getAttribute('alt');
405 let shortName = `:${emojiShortName[nativeUnicode]}:`;
406 this.state.postForm.body += shortName;
407 this.setState(this.state);
411 handlePostSubmit(i: PostForm, event: any) {
412 event.preventDefault();
414 WebSocketService.Instance.editPost(i.state.postForm);
416 WebSocketService.Instance.createPost(i.state.postForm);
418 i.state.loading = true;
422 copySuggestedTitle(i: PostForm) {
423 i.state.postForm.name = i.state.suggestedTitle.substring(
425 MAX_POST_TITLE_LENGTH
427 i.state.suggestedTitle = undefined;
431 handlePostUrlChange(i: PostForm, event: any) {
432 i.state.postForm.url = event.target.value;
438 if (validURL(this.state.postForm.url)) {
439 let form: SearchForm = {
440 q: this.state.postForm.url,
441 type_: SearchType[SearchType.Url],
442 sort: SortType[SortType.TopAll],
447 WebSocketService.Instance.search(form);
449 // Fetch the page title
450 getPageTitle(this.state.postForm.url).then(d => {
451 this.state.suggestedTitle = d;
452 this.setState(this.state);
455 this.state.suggestedTitle = undefined;
456 this.state.crossPosts = [];
460 handlePostNameChange(i: PostForm, event: any) {
461 i.state.postForm.name = event.target.value;
463 i.fetchSimilarPosts();
466 fetchSimilarPosts() {
467 let form: SearchForm = {
468 q: this.state.postForm.name,
469 type_: SearchType[SearchType.Posts],
470 sort: SortType[SortType.TopAll],
471 community_id: this.state.postForm.community_id,
476 if (this.state.postForm.name !== '') {
477 WebSocketService.Instance.search(form);
479 this.state.suggestedPosts = [];
482 this.setState(this.state);
485 handlePostBodyChange(i: PostForm, event: any) {
486 i.state.postForm.body = event.target.value;
490 handlePostCommunityChange(i: PostForm, event: any) {
491 i.state.postForm.community_id = Number(event.target.value);
495 handlePostNsfwChange(i: PostForm, event: any) {
496 i.state.postForm.nsfw = event.target.checked;
500 handleCancel(i: PostForm) {
504 handlePreviewToggle(i: PostForm, event: any) {
505 event.preventDefault();
506 i.state.previewMode = !i.state.previewMode;
510 handleImageUploadPaste(i: PostForm, event: any) {
511 let image = event.clipboardData.files[0];
513 i.handleImageUpload(i, image);
517 handleImageUpload(i: PostForm, event: any) {
520 event.preventDefault();
521 file = event.target.files[0];
526 const imageUploadUrl = `/pictshare/api/upload.php`;
527 const formData = new FormData();
528 formData.append('file', file);
530 i.state.imageLoading = true;
533 fetch(imageUploadUrl, {
537 .then(res => res.json())
539 let url = `${window.location.origin}/pictshare/${encodeURI(res.url)}`;
540 if (res.filetype == 'mp4') {
543 i.state.postForm.url = url;
544 i.state.imageLoading = false;
548 i.state.imageLoading = false;
550 toast(error, 'danger');
554 handleEmojiPickerClick(_i: PostForm, event: any) {
555 emojiPicker.togglePicker(event.target);
558 parseMessage(msg: WebSocketJsonResponse) {
559 let res = wsJsonToRes(msg);
561 toast(i18n.t(msg.error), 'danger');
562 this.state.loading = false;
563 this.setState(this.state);
565 } else if (res.op == UserOperation.ListCommunities) {
566 let data = res.data as ListCommunitiesResponse;
567 this.state.communities = data.communities;
568 if (this.props.post) {
569 this.state.postForm.community_id = this.props.post.community_id;
570 } else if (this.props.params && this.props.params.community) {
571 let foundCommunityId = data.communities.find(
572 r => r.name == this.props.params.community
574 this.state.postForm.community_id = foundCommunityId;
576 // By default, the null valued 'Select a Community'
578 this.setState(this.state);
580 // Set up select searching
581 let selectId: any = document.getElementById('post-community');
583 let selector = new Selectr(selectId, { nativeDropdown: false });
584 selector.on('selectr.select', option => {
585 this.state.postForm.community_id = Number(option.value);
586 this.setState(this.state);
589 } else if (res.op == UserOperation.CreatePost) {
590 let data = res.data as PostResponse;
591 if (data.post.creator_id == UserService.Instance.user.id) {
592 this.state.loading = false;
593 this.props.onCreate(data.post.id);
595 } else if (res.op == UserOperation.EditPost) {
596 let data = res.data as PostResponse;
597 if (data.post.creator_id == UserService.Instance.user.id) {
598 this.state.loading = false;
599 this.props.onEdit(data.post);
601 } else if (res.op == UserOperation.Search) {
602 let data = res.data as SearchResponse;
604 if (data.type_ == SearchType[SearchType.Posts]) {
605 this.state.suggestedPosts = data.posts;
606 } else if (data.type_ == SearchType[SearchType.Url]) {
607 this.state.crossPosts = data.posts;
609 this.setState(this.state);
610 } else if (res.op == UserOperation.GetSite) {
611 let data = res.data as GetSiteResponse;
612 this.state.enable_nsfw = data.site.enable_nsfw;
613 this.setState(this.state);