-import { None, Option, Some } from "@sniptt/monads";
import autosize from "autosize";
import { Component, linkEvent } from "inferno";
import { Prompt } from "inferno-router";
import {
- CommunityView,
CreatePost,
EditPost,
- ListingType,
+ Language,
PostResponse,
PostView,
Search,
SearchResponse,
- SearchType,
- SortType,
- toUndefined,
UserOperation,
wsJsonToRes,
wsUserOp,
} from "lemmy-js-client";
import { Subscription } from "rxjs";
-import { pictrsUri } from "../../env";
import { i18n } from "../../i18next";
import { PostFormParams } from "../../interfaces";
import { UserService, WebSocketService } from "../../services";
import {
+ Choice,
archiveTodayUrl,
- auth,
capitalizeFirstLetter,
- choicesConfig,
- communitySelectName,
communityToChoice,
debounce,
fetchCommunities,
+ getIdFromString,
getSiteMetadata,
ghostArchiveUrl,
- isBrowser,
isImage,
+ myAuth,
pictrsDeleteToast,
relTags,
setupTippy,
toast,
trendingFetchLimit,
+ uploadImage,
validTitle,
validURL,
webArchiveUrl,
wsSubscribe,
} from "../../utils";
import { Icon, Spinner } from "../common/icon";
+import { LanguageSelect } from "../common/language-select";
import { MarkdownTextArea } from "../common/markdown-textarea";
+import { SearchableSelect } from "../common/searchable-select";
import { PostListings } from "./post-listings";
-var Choices: any;
-if (isBrowser()) {
- Choices = require("choices.js");
-}
-
const MAX_POST_TITLE_LENGTH = 200;
interface PostFormProps {
- post_view: Option<PostView>; // If a post is given, that means this is an edit
- communities: Option<CommunityView[]>;
- params: Option<PostFormParams>;
+ post_view?: PostView; // If a post is given, that means this is an edit
+ allLanguages: Language[];
+ siteLanguages: number[];
+ params?: PostFormParams;
onCancel?(): any;
onCreate?(post: PostView): any;
onEdit?(post: PostView): any;
enableNsfw?: boolean;
enableDownvotes?: boolean;
+ selectedCommunityChoice?: Choice;
+ onSelectCommunity?: (choice: Choice) => void;
}
interface PostFormState {
- postForm: CreatePost;
- suggestedTitle: Option<string>;
- suggestedPosts: Option<PostView[]>;
- crossPosts: Option<PostView[]>;
+ form: {
+ name?: string;
+ url?: string;
+ body?: string;
+ nsfw?: boolean;
+ language_id?: number;
+ community_id?: number;
+ honeypot?: string;
+ };
+ suggestedTitle?: string;
+ suggestedPosts?: PostView[];
+ crossPosts?: PostView[];
loading: boolean;
imageLoading: boolean;
+ communitySearchLoading: boolean;
+ communitySearchOptions: Choice[];
previewMode: boolean;
}
export class PostForm extends Component<PostFormProps, PostFormState> {
- private subscription: Subscription;
- private choices: any;
- private emptyState: PostFormState = {
- postForm: new CreatePost({
- community_id: undefined,
- name: undefined,
- nsfw: Some(false),
- url: None,
- body: None,
- honeypot: None,
- auth: undefined,
- }),
+ private subscription?: Subscription;
+ state: PostFormState = {
+ form: {},
loading: false,
imageLoading: false,
+ communitySearchLoading: false,
previewMode: false,
- suggestedTitle: None,
- suggestedPosts: None,
- crossPosts: None,
+ communitySearchOptions: [],
};
- constructor(props: any, context: any) {
+ constructor(props: PostFormProps, context: any) {
super(props, context);
this.fetchSimilarPosts = debounce(this.fetchSimilarPosts.bind(this));
this.fetchPageTitle = debounce(this.fetchPageTitle.bind(this));
this.handlePostBodyChange = this.handlePostBodyChange.bind(this);
+ this.handleLanguageChange = this.handleLanguageChange.bind(this);
+ this.handleCommunitySelect = this.handleCommunitySelect.bind(this);
- this.state = this.emptyState;
+ this.parseMessage = this.parseMessage.bind(this);
+ this.subscription = wsSubscribe(this.parseMessage);
// Means its an edit
- this.props.post_view.match({
- some: pv =>
- (this.state.postForm = new CreatePost({
+ const pv = this.props.post_view;
+ if (pv) {
+ this.state = {
+ ...this.state,
+ form: {
body: pv.post.body,
name: pv.post.name,
community_id: pv.community.id,
url: pv.post.url,
- nsfw: Some(pv.post.nsfw),
- honeypot: None,
- auth: auth().unwrap(),
- })),
- none: void 0,
- });
+ nsfw: pv.post.nsfw,
+ language_id: pv.post.language_id,
+ },
+ };
+ }
- this.props.params.match({
- some: params => {
- this.state.postForm.name = toUndefined(params.name);
- this.state.postForm.url = params.url;
- this.state.postForm.body = params.body;
- },
- none: void 0,
- });
+ const selectedCommunityChoice = this.props.selectedCommunityChoice;
- this.parseMessage = this.parseMessage.bind(this);
- this.subscription = wsSubscribe(this.parseMessage);
+ if (selectedCommunityChoice) {
+ this.state = {
+ ...this.state,
+ form: {
+ ...this.state.form,
+ community_id: getIdFromString(selectedCommunityChoice.value),
+ },
+ communitySearchOptions: [selectedCommunityChoice],
+ };
+ }
+
+ const params = this.props.params;
+ if (params) {
+ this.state = {
+ ...this.state,
+ form: {
+ ...this.state.form,
+ ...params,
+ },
+ };
+ }
}
componentDidMount() {
setupTippy();
- this.setupCommunities();
- let textarea: any = document.getElementById("post-title");
+ const textarea: any = document.getElementById("post-title");
+
if (textarea) {
autosize(textarea);
}
componentDidUpdate() {
if (
!this.state.loading &&
- (this.state.postForm.name ||
- this.state.postForm.url.isSome() ||
- this.state.postForm.body.isSome())
+ (this.state.form.name || this.state.form.url || this.state.form.body)
) {
window.onbeforeunload = () => true;
} else {
- window.onbeforeunload = undefined;
+ window.onbeforeunload = null;
}
}
componentWillUnmount() {
- this.subscription.unsubscribe();
+ this.subscription?.unsubscribe();
/* this.choices && this.choices.destroy(); */
window.onbeforeunload = null;
}
+ static getDerivedStateFromProps(
+ { selectedCommunityChoice }: PostFormProps,
+ { form, ...restState }: PostFormState
+ ) {
+ return {
+ ...restState,
+ form: {
+ ...form,
+ community_id: getIdFromString(selectedCommunityChoice?.value),
+ },
+ };
+ }
+
render() {
+ let firstLang = this.state.form.language_id;
+ let selectedLangs = firstLang ? Array.of(firstLang) : undefined;
+
+ let url = this.state.form.url;
return (
<div>
<Prompt
when={
!this.state.loading &&
- (this.state.postForm.name ||
- this.state.postForm.url.isSome() ||
- this.state.postForm.body.isSome())
+ (this.state.form.name ||
+ this.state.form.url ||
+ this.state.form.body)
}
message={i18n.t("block_leaving")}
/>
<form onSubmit={linkEvent(this, this.handlePostSubmit)}>
- <div class="form-group row">
- <label class="col-sm-2 col-form-label" htmlFor="post-url">
+ <div className="form-group row">
+ <label className="col-sm-2 col-form-label" htmlFor="post-url">
{i18n.t("url")}
</label>
- <div class="col-sm-10">
+ <div className="col-sm-10">
<input
type="url"
id="post-url"
- class="form-control"
- value={toUndefined(this.state.postForm.url)}
+ className="form-control"
+ value={this.state.form.url}
onInput={linkEvent(this, this.handlePostUrlChange)}
onPaste={linkEvent(this, this.handleImageUploadPaste)}
/>
- {this.state.suggestedTitle.match({
- some: title => (
- <div
- class="mt-1 text-muted small font-weight-bold pointer"
- role="button"
- onClick={linkEvent(this, this.copySuggestedTitle)}
- >
- {i18n.t("copy_suggested_title", {
- title,
- })}
- </div>
- ),
- none: <></>,
- })}
+ {this.state.suggestedTitle && (
+ <div
+ className="mt-1 text-muted small font-weight-bold pointer"
+ role="button"
+ onClick={linkEvent(this, this.copySuggestedTitle)}
+ >
+ {i18n.t("copy_suggested_title", { title: "" })}{" "}
+ {this.state.suggestedTitle}
+ </div>
+ )}
<form>
<label
htmlFor="file-upload"
className={`${
- UserService.Instance.myUserInfo.isSome() && "pointer"
+ UserService.Instance.myUserInfo && "pointer"
} d-inline-block float-right text-muted font-weight-bold`}
data-tippy-content={i18n.t("upload_image")}
>
type="file"
accept="image/*,video/*"
name="file"
- class="d-none"
- disabled={UserService.Instance.myUserInfo.isNone()}
+ className="d-none"
+ disabled={!UserService.Instance.myUserInfo}
onChange={linkEvent(this, this.handleImageUpload)}
/>
</form>
- {this.state.postForm.url.match({
- some: url =>
- validURL(url) && (
- <div>
- <a
- href={`${webArchiveUrl}/save/${encodeURIComponent(
- url
- )}`}
- class="mr-2 d-inline-block float-right text-muted small font-weight-bold"
- rel={relTags}
- >
- archive.org {i18n.t("archive_link")}
- </a>
- <a
- href={`${ghostArchiveUrl}/search?term=${encodeURIComponent(
- url
- )}`}
- class="mr-2 d-inline-block float-right text-muted small font-weight-bold"
- rel={relTags}
- >
- ghostarchive.org {i18n.t("archive_link")}
- </a>
- <a
- href={`${archiveTodayUrl}/?run=1&url=${encodeURIComponent(
- url
- )}`}
- class="mr-2 d-inline-block float-right text-muted small font-weight-bold"
- rel={relTags}
- >
- archive.today {i18n.t("archive_link")}
- </a>
- </div>
- ),
- none: <></>,
- })}
+ {url && validURL(url) && (
+ <div>
+ <a
+ href={`${webArchiveUrl}/save/${encodeURIComponent(url)}`}
+ className="mr-2 d-inline-block float-right text-muted small font-weight-bold"
+ rel={relTags}
+ >
+ archive.org {i18n.t("archive_link")}
+ </a>
+ <a
+ href={`${ghostArchiveUrl}/search?term=${encodeURIComponent(
+ url
+ )}`}
+ className="mr-2 d-inline-block float-right text-muted small font-weight-bold"
+ rel={relTags}
+ >
+ ghostarchive.org {i18n.t("archive_link")}
+ </a>
+ <a
+ href={`${archiveTodayUrl}/?run=1&url=${encodeURIComponent(
+ url
+ )}`}
+ className="mr-2 d-inline-block float-right text-muted small font-weight-bold"
+ rel={relTags}
+ >
+ archive.today {i18n.t("archive_link")}
+ </a>
+ </div>
+ )}
{this.state.imageLoading && <Spinner />}
- {this.state.postForm.url.match({
- some: url =>
- isImage(url) && <img src={url} class="img-fluid" alt="" />,
- none: <></>,
- })}
- {this.state.crossPosts.match({
- some: xPosts =>
- xPosts.length > 0 && (
- <>
- <div class="my-1 text-muted small font-weight-bold">
- {i18n.t("cross_posts")}
- </div>
- <PostListings
- showCommunity
- posts={xPosts}
- enableDownvotes={this.props.enableDownvotes}
- enableNsfw={this.props.enableNsfw}
- />
- </>
- ),
- none: <></>,
- })}
+ {url && isImage(url) && (
+ <img src={url} className="img-fluid" alt="" />
+ )}
+ {this.state.crossPosts && this.state.crossPosts.length > 0 && (
+ <>
+ <div className="my-1 text-muted small font-weight-bold">
+ {i18n.t("cross_posts")}
+ </div>
+ <PostListings
+ showCommunity
+ posts={this.state.crossPosts}
+ enableDownvotes={this.props.enableDownvotes}
+ enableNsfw={this.props.enableNsfw}
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
+ />
+ </>
+ )}
</div>
</div>
- <div class="form-group row">
- <label class="col-sm-2 col-form-label" htmlFor="post-title">
+ <div className="form-group row">
+ <label className="col-sm-2 col-form-label" htmlFor="post-title">
{i18n.t("title")}
</label>
- <div class="col-sm-10">
+ <div className="col-sm-10">
<textarea
- value={this.state.postForm.name}
+ value={this.state.form.name}
id="post-title"
onInput={linkEvent(this, this.handlePostNameChange)}
- class={`form-control ${
- !validTitle(this.state.postForm.name) && "is-invalid"
+ className={`form-control ${
+ !validTitle(this.state.form.name) && "is-invalid"
}`}
required
rows={1}
minLength={3}
maxLength={MAX_POST_TITLE_LENGTH}
/>
- {!validTitle(this.state.postForm.name) && (
- <div class="invalid-feedback">
+ {!validTitle(this.state.form.name) && (
+ <div className="invalid-feedback">
{i18n.t("invalid_post_title")}
</div>
)}
- {this.state.suggestedPosts.match({
- some: sPosts =>
- sPosts.length > 0 && (
- <>
- <div class="my-1 text-muted small font-weight-bold">
- {i18n.t("related_posts")}
- </div>
- <PostListings
- posts={sPosts}
- enableDownvotes={this.props.enableDownvotes}
- enableNsfw={this.props.enableNsfw}
- />
- </>
- ),
- none: <></>,
- })}
+ {this.state.suggestedPosts &&
+ this.state.suggestedPosts.length > 0 && (
+ <>
+ <div className="my-1 text-muted small font-weight-bold">
+ {i18n.t("related_posts")}
+ </div>
+ <PostListings
+ showCommunity
+ posts={this.state.suggestedPosts}
+ enableDownvotes={this.props.enableDownvotes}
+ enableNsfw={this.props.enableNsfw}
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
+ />
+ </>
+ )}
</div>
</div>
- <div class="form-group row">
- <label class="col-sm-2 col-form-label">{i18n.t("body")}</label>
- <div class="col-sm-10">
+ <div className="form-group row">
+ <label className="col-sm-2 col-form-label">{i18n.t("body")}</label>
+ <div className="col-sm-10">
<MarkdownTextArea
- initialContent={this.state.postForm.body}
+ initialContent={this.state.form.body}
onContentChange={this.handlePostBodyChange}
- placeholder={None}
- buttonTitle={None}
- maxLength={None}
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
/>
</div>
</div>
- {this.props.post_view.isNone() && (
- <div class="form-group row">
- <label class="col-sm-2 col-form-label" htmlFor="post-community">
+ {!this.props.post_view && (
+ <div className="form-group row">
+ <label
+ className="col-sm-2 col-form-label"
+ htmlFor="post-community"
+ >
{i18n.t("community")}
</label>
- <div class="col-sm-10">
- <select
- class="form-control"
+ <div className="col-sm-10">
+ <SearchableSelect
id="post-community"
- value={this.state.postForm.community_id}
- onInput={linkEvent(this, this.handlePostCommunityChange)}
- >
- <option>{i18n.t("select_a_community")}</option>
- {this.props.communities.unwrapOr([]).map(cv => (
- <option value={cv.community.id}>
- {communitySelectName(cv)}
- </option>
- ))}
- </select>
+ value={this.state.form.community_id}
+ options={[
+ {
+ label: i18n.t("select_a_community"),
+ value: "",
+ disabled: true,
+ } as Choice,
+ ].concat(this.state.communitySearchOptions)}
+ loading={this.state.communitySearchLoading}
+ onChange={this.handleCommunitySelect}
+ onSearch={this.handleCommunitySearch}
+ />
</div>
</div>
)}
{this.props.enableNsfw && (
- <div class="form-group row">
- <legend class="col-form-label col-sm-2 pt-0">
+ <div className="form-group row">
+ <legend className="col-form-label col-sm-2 pt-0">
{i18n.t("nsfw")}
</legend>
- <div class="col-sm-10">
- <div class="form-check">
+ <div className="col-sm-10">
+ <div className="form-check">
<input
- class="form-check-input position-static"
+ className="form-check-input position-static"
id="post-nsfw"
type="checkbox"
- checked={toUndefined(this.state.postForm.nsfw)}
+ checked={this.state.form.nsfw}
onChange={linkEvent(this, this.handlePostNsfwChange)}
/>
</div>
</div>
</div>
)}
+ <LanguageSelect
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
+ selectedLanguageIds={selectedLangs}
+ multiple={false}
+ onChange={this.handleLanguageChange}
+ />
<input
tabIndex={-1}
autoComplete="false"
name="a_password"
type="text"
- class="form-control honeypot"
+ className="form-control honeypot"
id="register-honey"
- value={toUndefined(this.state.postForm.honeypot)}
+ value={this.state.form.honeypot}
onInput={linkEvent(this, this.handleHoneyPotChange)}
/>
- <div class="form-group row">
- <div class="col-sm-10">
+ <div className="form-group row">
+ <div className="col-sm-10">
<button
- disabled={
- !this.state.postForm.community_id || this.state.loading
- }
+ disabled={!this.state.form.community_id || this.state.loading}
type="submit"
- class="btn btn-secondary mr-2"
+ className="btn btn-secondary mr-2"
>
{this.state.loading ? (
<Spinner />
- ) : this.props.post_view.isSome() ? (
+ ) : this.props.post_view ? (
capitalizeFirstLetter(i18n.t("save"))
) : (
capitalizeFirstLetter(i18n.t("create"))
)}
</button>
- {this.props.post_view.isSome() && (
+ {this.props.post_view && (
<button
type="button"
- class="btn btn-secondary"
+ className="btn btn-secondary"
onClick={linkEvent(this, this.handleCancel)}
>
{i18n.t("cancel")}
handlePostSubmit(i: PostForm, event: any) {
event.preventDefault();
+ i.setState({ loading: true });
+
// Coerce empty url string to undefined
- if (
- i.state.postForm.url.isSome() &&
- i.state.postForm.url.unwrapOr("blank") === ""
- ) {
- i.state.postForm.url = None;
+ if ((i.state.form.url ?? "blank") === "") {
+ i.setState(s => ((s.form.url = undefined), s));
}
- let pForm = i.state.postForm;
- i.props.post_view.match({
- some: pv => {
- let form = new EditPost({
- name: Some(pForm.name),
+ let pForm = i.state.form;
+ let pv = i.props.post_view;
+ let auth = myAuth();
+ if (auth) {
+ if (pv) {
+ let form: EditPost = {
+ name: pForm.name,
url: pForm.url,
body: pForm.body,
nsfw: pForm.nsfw,
post_id: pv.post.id,
- auth: auth().unwrap(),
- });
+ language_id: pv.post.language_id,
+ auth,
+ };
WebSocketService.Instance.send(wsClient.editPost(form));
- },
- none: () => {
- i.state.postForm.auth = auth().unwrap();
- WebSocketService.Instance.send(wsClient.createPost(i.state.postForm));
- },
- });
- i.state.loading = true;
- i.setState(i.state);
+ } else {
+ if (pForm.name && pForm.community_id) {
+ let form: CreatePost = {
+ name: pForm.name,
+ community_id: pForm.community_id,
+ url: pForm.url,
+ body: pForm.body,
+ nsfw: pForm.nsfw,
+ language_id: pForm.language_id,
+ honeypot: pForm.honeypot,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.createPost(form));
+ }
+ }
+ }
}
copySuggestedTitle(i: PostForm) {
- i.state.suggestedTitle.match({
- some: sTitle => {
- i.state.postForm.name = sTitle.substring(0, MAX_POST_TITLE_LENGTH);
- i.state.suggestedTitle = None;
- setTimeout(() => {
- let textarea: any = document.getElementById("post-title");
- autosize.update(textarea);
- }, 10);
- i.setState(i.state);
- },
- none: void 0,
- });
+ let sTitle = i.state.suggestedTitle;
+ if (sTitle) {
+ i.setState(
+ s => ((s.form.name = sTitle?.substring(0, MAX_POST_TITLE_LENGTH)), s)
+ );
+ i.setState({ suggestedTitle: undefined });
+ setTimeout(() => {
+ let textarea: any = document.getElementById("post-title");
+ autosize.update(textarea);
+ }, 10);
+ }
}
handlePostUrlChange(i: PostForm, event: any) {
- i.state.postForm.url = Some(event.target.value);
- i.setState(i.state);
+ i.setState(s => ((s.form.url = event.target.value), s));
i.fetchPageTitle();
}
fetchPageTitle() {
- this.state.postForm.url.match({
- some: url => {
- if (validURL(url)) {
- let form = new Search({
- q: url,
- community_id: None,
- community_name: None,
- creator_id: None,
- type_: Some(SearchType.Url),
- sort: Some(SortType.TopAll),
- listing_type: Some(ListingType.All),
- page: Some(1),
- limit: Some(trendingFetchLimit),
- auth: auth(false).ok(),
- });
-
- WebSocketService.Instance.send(wsClient.search(form));
-
- // Fetch the page title
- getSiteMetadata(url).then(d => {
- this.state.suggestedTitle = d.metadata.title;
- this.setState(this.state);
- });
- } else {
- this.state.suggestedTitle = None;
- this.state.crossPosts = None;
- }
- },
- none: void 0,
- });
+ let url = this.state.form.url;
+ if (url && validURL(url)) {
+ let form: Search = {
+ q: url,
+ type_: "Url",
+ sort: "TopAll",
+ listing_type: "All",
+ page: 1,
+ limit: trendingFetchLimit,
+ auth: myAuth(false),
+ };
+
+ WebSocketService.Instance.send(wsClient.search(form));
+
+ // Fetch the page title
+ getSiteMetadata(url).then(d => {
+ this.setState({ suggestedTitle: d.metadata.title });
+ });
+ } else {
+ this.setState({ suggestedTitle: undefined, crossPosts: undefined });
+ }
}
handlePostNameChange(i: PostForm, event: any) {
- i.state.postForm.name = event.target.value;
- i.setState(i.state);
+ i.setState(s => ((s.form.name = event.target.value), s));
i.fetchSimilarPosts();
}
fetchSimilarPosts() {
- let form = new Search({
- q: this.state.postForm.name,
- type_: Some(SearchType.Posts),
- sort: Some(SortType.TopAll),
- listing_type: Some(ListingType.All),
- community_id: Some(this.state.postForm.community_id),
- community_name: None,
- creator_id: None,
- page: Some(1),
- limit: Some(trendingFetchLimit),
- auth: auth(false).ok(),
- });
+ let q = this.state.form.name;
+ if (q && q !== "") {
+ let form: Search = {
+ q,
+ type_: "Posts",
+ sort: "TopAll",
+ listing_type: "All",
+ community_id: this.state.form.community_id,
+ page: 1,
+ limit: trendingFetchLimit,
+ auth: myAuth(false),
+ };
- if (this.state.postForm.name !== "") {
WebSocketService.Instance.send(wsClient.search(form));
} else {
- this.state.suggestedPosts = None;
+ this.setState({ suggestedPosts: undefined });
}
-
- this.setState(this.state);
}
handlePostBodyChange(val: string) {
- this.state.postForm.body = Some(val);
- this.setState(this.state);
+ this.setState(s => ((s.form.body = val), s));
}
handlePostCommunityChange(i: PostForm, event: any) {
- i.state.postForm.community_id = Number(event.target.value);
- i.setState(i.state);
+ i.setState(s => ((s.form.community_id = Number(event.target.value)), s));
}
handlePostNsfwChange(i: PostForm, event: any) {
- i.state.postForm.nsfw = Some(event.target.checked);
- i.setState(i.state);
+ i.setState(s => ((s.form.nsfw = event.target.checked), s));
+ }
+
+ handleLanguageChange(val: number[]) {
+ this.setState(s => ((s.form.language_id = val.at(0)), s));
}
handleHoneyPotChange(i: PostForm, event: any) {
- i.state.postForm.honeypot = Some(event.target.value);
- i.setState(i.state);
+ i.setState(s => ((s.form.honeypot = event.target.value), s));
}
handleCancel(i: PostForm) {
- i.props.onCancel();
+ i.props.onCancel?.();
}
handlePreviewToggle(i: PostForm, event: any) {
event.preventDefault();
- i.state.previewMode = !i.state.previewMode;
- i.setState(i.state);
+ i.setState({ previewMode: !i.state.previewMode });
}
handleImageUploadPaste(i: PostForm, event: any) {
file = event;
}
- const formData = new FormData();
- formData.append("images[]", file);
-
- i.state.imageLoading = true;
- i.setState(i.state);
+ i.setState({ imageLoading: true });
- fetch(pictrsUri, {
- method: "POST",
- body: formData,
- })
- .then(res => res.json())
+ uploadImage(file)
.then(res => {
console.log("pictrs upload:");
console.log(res);
- if (res.msg == "ok") {
- let hash = res.files[0].file;
- let url = `${pictrsUri}/${hash}`;
- let deleteToken = res.files[0].delete_token;
- let deleteUrl = `${pictrsUri}/delete/${deleteToken}/${hash}`;
- i.state.postForm.url = Some(url);
- i.state.imageLoading = false;
- i.setState(i.state);
- pictrsDeleteToast(
- i18n.t("click_to_delete_picture"),
- i18n.t("picture_deleted"),
- deleteUrl
- );
+ if (res.msg === "ok") {
+ i.state.form.url = res.url;
+ i.setState({ imageLoading: false });
+ pictrsDeleteToast(file.name, res.delete_url as string);
} else {
- i.state.imageLoading = false;
- i.setState(i.state);
+ i.setState({ imageLoading: false });
toast(JSON.stringify(res), "danger");
}
})
.catch(error => {
- i.state.imageLoading = false;
- i.setState(i.state);
+ i.setState({ imageLoading: false });
console.error(error);
toast(error, "danger");
});
}
- setupCommunities() {
- // Set up select searching
- if (isBrowser()) {
- let selectId: any = document.getElementById("post-community");
- if (selectId) {
- this.choices = new Choices(selectId, choicesConfig);
- this.choices.passedElement.element.addEventListener(
- "choice",
- (e: any) => {
- this.state.postForm.community_id = Number(e.detail.choice.value);
- this.setState(this.state);
- },
- false
- );
- this.choices.passedElement.element.addEventListener(
- "search",
- debounce(async (e: any) => {
- try {
- let communities = (await fetchCommunities(e.detail.value))
- .communities;
- this.choices.setChoices(
- communities.map(cv => communityToChoice(cv)),
- "value",
- "label",
- true
- );
- } catch (err) {
- console.log(err);
- }
- }),
- false
- );
- }
+ handleCommunitySearch = debounce(async (text: string) => {
+ const { selectedCommunityChoice } = this.props;
+ this.setState({ communitySearchLoading: true });
+
+ const newOptions: Choice[] = [];
+
+ if (selectedCommunityChoice) {
+ newOptions.push(selectedCommunityChoice);
}
- this.props.post_view.match({
- some: pv => (this.state.postForm.community_id = pv.community.id),
- none: void 0,
- });
- this.props.params.match({
- some: params =>
- params.nameOrId.match({
- some: nameOrId =>
- nameOrId.match({
- left: name => {
- let foundCommunityId = this.props.communities
- .unwrapOr([])
- .find(r => r.community.name == name).community.id;
- this.state.postForm.community_id = foundCommunityId;
- },
- right: id => (this.state.postForm.community_id = id),
- }),
- none: void 0,
- }),
- none: void 0,
+ if (text.length > 0) {
+ newOptions.push(
+ ...(await fetchCommunities(text)).communities.map(communityToChoice)
+ );
+
+ this.setState({
+ communitySearchOptions: newOptions,
+ });
+ }
+
+ this.setState({
+ communitySearchLoading: false,
});
+ });
- if (isBrowser() && this.state.postForm.community_id) {
- this.choices.setChoiceByValue(
- this.state.postForm.community_id.toString()
- );
+ handleCommunitySelect(choice: Choice) {
+ if (this.props.onSelectCommunity) {
+ this.setState({
+ loading: true,
+ });
+
+ this.props.onSelectCommunity(choice);
+
+ this.setState({ loading: false });
}
- this.setState(this.state);
}
parseMessage(msg: any) {
+ let mui = UserService.Instance.myUserInfo;
let op = wsUserOp(msg);
console.log(msg);
if (msg.error) {
// Errors handled by top level pages
// toast(i18n.t(msg.error), "danger");
- this.state.loading = false;
- this.setState(this.state);
+ this.setState({ loading: false });
return;
} else if (op == UserOperation.CreatePost) {
- let data = wsJsonToRes<PostResponse>(msg, PostResponse);
- UserService.Instance.myUserInfo.match({
- some: mui => {
- if (data.post_view.creator.id == mui.local_user_view.person.id) {
- this.state.loading = false;
- this.props.onCreate(data.post_view);
- }
- },
- none: void 0,
- });
+ let data = wsJsonToRes<PostResponse>(msg);
+ if (data.post_view.creator.id == mui?.local_user_view.person.id) {
+ this.props.onCreate?.(data.post_view);
+ }
} else if (op == UserOperation.EditPost) {
- let data = wsJsonToRes<PostResponse>(msg, PostResponse);
- UserService.Instance.myUserInfo.match({
- some: mui => {
- if (data.post_view.creator.id == mui.local_user_view.person.id) {
- this.state.loading = false;
- this.props.onEdit(data.post_view);
- }
- },
- none: void 0,
- });
+ let data = wsJsonToRes<PostResponse>(msg);
+ if (data.post_view.creator.id == mui?.local_user_view.person.id) {
+ this.setState({ loading: false });
+ this.props.onEdit?.(data.post_view);
+ }
} else if (op == UserOperation.Search) {
- let data = wsJsonToRes<SearchResponse>(msg, SearchResponse);
+ let data = wsJsonToRes<SearchResponse>(msg);
- if (data.type_ == SearchType[SearchType.Posts]) {
- this.state.suggestedPosts = Some(data.posts);
- } else if (data.type_ == SearchType[SearchType.Url]) {
- this.state.crossPosts = Some(data.posts);
+ if (data.type_ == "Posts") {
+ this.setState({ suggestedPosts: data.posts });
+ } else if (data.type_ == "Url") {
+ this.setState({ crossPosts: data.posts });
}
- this.setState(this.state);
}
}
}