]> Untitled Git - lemmy.git/blob - ui/src/components/comment-form.tsx
Merge branch 'dev'
[lemmy.git] / ui / src / components / comment-form.tsx
1 import { Component, linkEvent } from 'inferno';
2 import {
3   CommentNode as CommentNodeI,
4   CommentForm as CommentFormI,
5 } from '../interfaces';
6 import {
7   capitalizeFirstLetter,
8   mdToHtml,
9   randomStr,
10   markdownHelpUrl,
11   toast,
12   setupTribute,
13 } from '../utils';
14 import { WebSocketService, UserService } from '../services';
15 import autosize from 'autosize';
16 import Tribute from 'tributejs/src/Tribute.js';
17 import { i18n } from '../i18next';
18 import { T } from 'inferno-i18next';
19
20 interface CommentFormProps {
21   postId?: number;
22   node?: CommentNodeI;
23   onReplyCancel?(): any;
24   edit?: boolean;
25   disabled?: boolean;
26 }
27
28 interface CommentFormState {
29   commentForm: CommentFormI;
30   buttonTitle: string;
31   previewMode: boolean;
32   imageLoading: boolean;
33 }
34
35 export class CommentForm extends Component<CommentFormProps, CommentFormState> {
36   private id = `comment-form-${randomStr()}`;
37   private tribute: Tribute;
38   private emptyState: CommentFormState = {
39     commentForm: {
40       auth: null,
41       content: null,
42       post_id: this.props.node
43         ? this.props.node.comment.post_id
44         : this.props.postId,
45       creator_id: UserService.Instance.user
46         ? UserService.Instance.user.id
47         : null,
48     },
49     buttonTitle: !this.props.node
50       ? capitalizeFirstLetter(i18n.t('post'))
51       : this.props.edit
52       ? capitalizeFirstLetter(i18n.t('edit'))
53       : capitalizeFirstLetter(i18n.t('reply')),
54     previewMode: false,
55     imageLoading: false,
56   };
57
58   constructor(props: any, context: any) {
59     super(props, context);
60
61     this.tribute = setupTribute();
62     this.state = this.emptyState;
63
64     if (this.props.node) {
65       if (this.props.edit) {
66         this.state.commentForm.edit_id = this.props.node.comment.id;
67         this.state.commentForm.parent_id = this.props.node.comment.parent_id;
68         this.state.commentForm.content = this.props.node.comment.content;
69         this.state.commentForm.creator_id = this.props.node.comment.creator_id;
70       } else {
71         // A reply gets a new parent id
72         this.state.commentForm.parent_id = this.props.node.comment.id;
73       }
74     }
75   }
76
77   componentDidMount() {
78     var textarea: any = document.getElementById(this.id);
79     autosize(textarea);
80     this.tribute.attach(textarea);
81     textarea.addEventListener('tribute-replaced', () => {
82       this.state.commentForm.content = textarea.value;
83       this.setState(this.state);
84       autosize.update(textarea);
85     });
86   }
87
88   render() {
89     return (
90       <div class="mb-3">
91         <form onSubmit={linkEvent(this, this.handleCommentSubmit)}>
92           <div class="form-group row">
93             <div className={`col-sm-12`}>
94               <textarea
95                 id={this.id}
96                 className={`form-control ${this.state.previewMode && 'd-none'}`}
97                 value={this.state.commentForm.content}
98                 onInput={linkEvent(this, this.handleCommentContentChange)}
99                 required
100                 disabled={this.props.disabled}
101                 rows={2}
102                 maxLength={10000}
103               />
104               {this.state.previewMode && (
105                 <div
106                   className="md-div"
107                   dangerouslySetInnerHTML={mdToHtml(
108                     this.state.commentForm.content
109                   )}
110                 />
111               )}
112             </div>
113           </div>
114           <div class="row">
115             <div class="col-sm-12">
116               <button
117                 type="submit"
118                 class="btn btn-sm btn-secondary mr-2"
119                 disabled={this.props.disabled}
120               >
121                 {this.state.buttonTitle}
122               </button>
123               {this.state.commentForm.content && (
124                 <button
125                   className={`btn btn-sm mr-2 btn-secondary ${this.state
126                     .previewMode && 'active'}`}
127                   onClick={linkEvent(this, this.handlePreviewToggle)}
128                 >
129                   <T i18nKey="preview">#</T>
130                 </button>
131               )}
132               {this.props.node && (
133                 <button
134                   type="button"
135                   class="btn btn-sm btn-secondary mr-2"
136                   onClick={linkEvent(this, this.handleReplyCancel)}
137                 >
138                   <T i18nKey="cancel">#</T>
139                 </button>
140               )}
141               <a
142                 href={markdownHelpUrl}
143                 target="_blank"
144                 class="d-inline-block float-right text-muted small font-weight-bold"
145               >
146                 <T i18nKey="formatting_help">#</T>
147               </a>
148               <form class="d-inline-block mr-2 float-right text-muted small font-weight-bold">
149                 <label
150                   htmlFor={`file-upload-${this.id}`}
151                   className={`${UserService.Instance.user && 'pointer'}`}
152                 >
153                   <T i18nKey="upload_image">#</T>
154                 </label>
155                 <input
156                   id={`file-upload-${this.id}`}
157                   type="file"
158                   accept="image/*,video/*"
159                   name="file"
160                   class="d-none"
161                   disabled={!UserService.Instance.user}
162                   onChange={linkEvent(this, this.handleImageUpload)}
163                 />
164               </form>
165               {this.state.imageLoading && (
166                 <svg class="icon icon-spinner spin">
167                   <use xlinkHref="#icon-spinner"></use>
168                 </svg>
169               )}
170             </div>
171           </div>
172         </form>
173       </div>
174     );
175   }
176
177   handleCommentSubmit(i: CommentForm, event: any) {
178     event.preventDefault();
179     if (i.props.edit) {
180       WebSocketService.Instance.editComment(i.state.commentForm);
181     } else {
182       WebSocketService.Instance.createComment(i.state.commentForm);
183     }
184
185     i.state.previewMode = false;
186     i.state.commentForm.content = undefined;
187     event.target.reset();
188     i.setState(i.state);
189     if (i.props.node) {
190       i.props.onReplyCancel();
191     }
192
193     autosize.update(document.querySelector('textarea'));
194   }
195
196   handleCommentContentChange(i: CommentForm, event: any) {
197     i.state.commentForm.content = event.target.value;
198     i.setState(i.state);
199   }
200
201   handlePreviewToggle(i: CommentForm, event: any) {
202     event.preventDefault();
203     i.state.previewMode = !i.state.previewMode;
204     i.setState(i.state);
205   }
206
207   handleReplyCancel(i: CommentForm) {
208     i.props.onReplyCancel();
209   }
210
211   handleImageUpload(i: CommentForm, event: any) {
212     event.preventDefault();
213     let file = event.target.files[0];
214     const imageUploadUrl = `/pictshare/api/upload.php`;
215     const formData = new FormData();
216     formData.append('file', file);
217
218     i.state.imageLoading = true;
219     i.setState(i.state);
220
221     fetch(imageUploadUrl, {
222       method: 'POST',
223       body: formData,
224     })
225       .then(res => res.json())
226       .then(res => {
227         let url = `${window.location.origin}/pictshare/${res.url}`;
228         let imageMarkdown =
229           res.filetype == 'mp4' ? `[vid](${url}/raw)` : `![](${url})`;
230         let content = i.state.commentForm.content;
231         content = content ? `${content}\n${imageMarkdown}` : imageMarkdown;
232         i.state.commentForm.content = content;
233         i.state.imageLoading = false;
234         i.setState(i.state);
235         var textarea: any = document.getElementById(i.id);
236         autosize.update(textarea);
237       })
238       .catch(error => {
239         i.state.imageLoading = false;
240         i.setState(i.state);
241         toast(error, 'danger');
242       });
243   }
244 }