2021-11-24 12:40:16 +05:00
|
|
|
import { Flex, Text } from "rebass";
|
2021-12-16 12:03:42 +05:00
|
|
|
import { appVersion } from "../../utils/version";
|
2021-11-24 12:40:16 +05:00
|
|
|
import Field from "../field";
|
|
|
|
|
import Dialog from "./dialog";
|
|
|
|
|
import platform from "platform";
|
|
|
|
|
import { useState } from "react";
|
|
|
|
|
import { confirm } from "../../common/dialog-controller";
|
2021-12-02 13:03:10 +05:00
|
|
|
import { isUserPremium } from "../../hooks/use-is-user-premium";
|
|
|
|
|
import * as clipboard from "clipboard-polyfill/text";
|
|
|
|
|
import { store as userstore } from "../../stores/user-store";
|
|
|
|
|
import { db } from "../../common/db";
|
2021-11-24 12:40:16 +05:00
|
|
|
|
|
|
|
|
const PLACEHOLDERS = {
|
|
|
|
|
title: "Briefly describe what happened",
|
|
|
|
|
body: `Tell us more about the issue you are facing.
|
|
|
|
|
|
|
|
|
|
For example things like:
|
|
|
|
|
1. Steps to reproduce the issue
|
|
|
|
|
2. Things you have tried so far
|
|
|
|
|
3. etc.
|
|
|
|
|
|
|
|
|
|
This is all optional, of course.`,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function getDeviceInfo() {
|
2021-12-21 12:59:05 +05:00
|
|
|
const version = appVersion.formatted;
|
2021-11-24 12:40:16 +05:00
|
|
|
const os = platform.os;
|
|
|
|
|
const browser = `${platform.name} ${platform.version}`;
|
|
|
|
|
|
2021-12-21 12:59:05 +05:00
|
|
|
return `App version: ${version}
|
2021-11-24 12:40:16 +05:00
|
|
|
OS: ${os}
|
2021-12-02 13:03:10 +05:00
|
|
|
Browser: ${browser}
|
|
|
|
|
Pro: ${isUserPremium()}`;
|
2021-11-24 12:40:16 +05:00
|
|
|
}
|
|
|
|
|
const BODY_TEMPLATE = (body) => {
|
|
|
|
|
const info = `**Device information:**\n${getDeviceInfo()}`;
|
|
|
|
|
if (!body) return info;
|
|
|
|
|
return `${body}\n\n${info}`;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function IssueDialog(props) {
|
|
|
|
|
const [isSubmitting, setIsSubmitting] = useState(false);
|
|
|
|
|
const [error, setError] = useState();
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<Dialog
|
|
|
|
|
isOpen={true}
|
|
|
|
|
title={"Report an issue"}
|
|
|
|
|
description={
|
|
|
|
|
"Let us know if you are facing an issue. We'll do our best to sort it out for you."
|
|
|
|
|
}
|
|
|
|
|
onClose={props.onClose}
|
|
|
|
|
positiveButton={{
|
|
|
|
|
text: "Report",
|
|
|
|
|
props: {
|
|
|
|
|
form: "issueForm",
|
|
|
|
|
},
|
|
|
|
|
loading: isSubmitting,
|
|
|
|
|
disabled: isSubmitting,
|
|
|
|
|
}}
|
|
|
|
|
negativeButton={{ text: "Cancel", onClick: props.onClose }}
|
|
|
|
|
>
|
|
|
|
|
<Flex
|
|
|
|
|
id="issueForm"
|
|
|
|
|
as="form"
|
|
|
|
|
flexDirection="column"
|
|
|
|
|
onSubmit={async (e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
try {
|
|
|
|
|
setIsSubmitting(true);
|
|
|
|
|
setError();
|
|
|
|
|
|
|
|
|
|
const formData = new FormData(e.target);
|
|
|
|
|
const requestData = Object.fromEntries(formData.entries());
|
|
|
|
|
if (!requestData.title.trim() || !requestData.body.trim()) return;
|
|
|
|
|
requestData.body = BODY_TEMPLATE(requestData.body);
|
2021-12-02 13:03:10 +05:00
|
|
|
const url = await db.debug.report({
|
|
|
|
|
title: requestData.title,
|
|
|
|
|
body: requestData.body,
|
|
|
|
|
userId: userstore.get().user?.id,
|
2021-11-24 12:40:16 +05:00
|
|
|
});
|
2021-12-02 13:03:10 +05:00
|
|
|
await showIssueReportedDialog({ url });
|
2021-11-24 12:40:16 +05:00
|
|
|
} catch (e) {
|
|
|
|
|
setError(e.message);
|
|
|
|
|
} finally {
|
|
|
|
|
setIsSubmitting(false);
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Field
|
|
|
|
|
required
|
|
|
|
|
label="Title"
|
|
|
|
|
id="title"
|
|
|
|
|
name="title"
|
|
|
|
|
placeholder={PLACEHOLDERS.title}
|
|
|
|
|
autoFocus
|
|
|
|
|
/>
|
|
|
|
|
<Field
|
|
|
|
|
as="textarea"
|
|
|
|
|
required
|
|
|
|
|
variant="forms.input"
|
|
|
|
|
label="Description"
|
|
|
|
|
id="body"
|
|
|
|
|
name="body"
|
|
|
|
|
placeholder={PLACEHOLDERS.body}
|
|
|
|
|
sx={{ mt: 1 }}
|
|
|
|
|
styles={{
|
|
|
|
|
input: {
|
|
|
|
|
minHeight: 150,
|
|
|
|
|
},
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
<Text variant="subBody" mt={1}>
|
|
|
|
|
{getDeviceInfo()
|
|
|
|
|
.split("\n")
|
|
|
|
|
.map((t) => (
|
|
|
|
|
<>
|
|
|
|
|
{t}
|
|
|
|
|
<br />
|
|
|
|
|
</>
|
|
|
|
|
))}
|
|
|
|
|
</Text>
|
|
|
|
|
{error && (
|
|
|
|
|
<Text bg="errorBg" variant="error" mt={1} px={1}>
|
|
|
|
|
Error: {error}
|
|
|
|
|
</Text>
|
|
|
|
|
)}
|
|
|
|
|
</Flex>
|
|
|
|
|
</Dialog>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default IssueDialog;
|
2021-12-02 13:03:10 +05:00
|
|
|
|
|
|
|
|
function showIssueReportedDialog({ url }) {
|
|
|
|
|
return confirm({
|
|
|
|
|
title: "Thank you for reporting!",
|
|
|
|
|
yesAction: () => clipboard.writeText(url),
|
|
|
|
|
yesText: "Copy link",
|
|
|
|
|
message: (
|
|
|
|
|
<>
|
|
|
|
|
<p>
|
|
|
|
|
You can track your bug report at{" "}
|
|
|
|
|
<Text as="a" href={url} sx={{ lineBreak: "anywhere" }}>
|
|
|
|
|
{url}
|
|
|
|
|
</Text>
|
|
|
|
|
.
|
|
|
|
|
</p>
|
|
|
|
|
<p>
|
|
|
|
|
Please note that we will respond to your bug report on the link above.{" "}
|
|
|
|
|
<b>
|
|
|
|
|
We recommended that you save the above link for later reference.
|
|
|
|
|
</b>
|
|
|
|
|
</p>
|
|
|
|
|
<p>
|
|
|
|
|
If your issue is critical (e.g. notes not syncing, crashes etc.),
|
|
|
|
|
please{" "}
|
|
|
|
|
<a href="https://discord.com/invite/zQBK97EE22">
|
|
|
|
|
join our Discord community
|
|
|
|
|
</a>{" "}
|
|
|
|
|
for one-to-one support.
|
|
|
|
|
</p>
|
|
|
|
|
</>
|
|
|
|
|
),
|
|
|
|
|
});
|
|
|
|
|
}
|