@drodil/backstage-plugin-qeta-react 2.15.0 → 3.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api.esm.js +8 -0
- package/dist/api.esm.js.map +1 -0
- package/dist/components/AnswerCard/AnswerCard.esm.js +113 -0
- package/dist/components/AnswerCard/AnswerCard.esm.js.map +1 -0
- package/dist/components/AnswerForm/AnswerForm.esm.js +153 -0
- package/dist/components/AnswerForm/AnswerForm.esm.js.map +1 -0
- package/dist/components/AnswersContainer/AnswerList.esm.js +100 -0
- package/dist/components/AnswersContainer/AnswerList.esm.js.map +1 -0
- package/dist/components/AnswersContainer/AnswerListItem.esm.js +90 -0
- package/dist/components/AnswersContainer/AnswerListItem.esm.js.map +1 -0
- package/dist/components/AnswersContainer/AnswersContainer.esm.js +210 -0
- package/dist/components/AnswersContainer/AnswersContainer.esm.js.map +1 -0
- package/dist/components/ArticleContent/ArticleButtons.esm.js +91 -0
- package/dist/components/ArticleContent/ArticleButtons.esm.js.map +1 -0
- package/dist/components/ArticleContent/ArticleContent.esm.js +82 -0
- package/dist/components/ArticleContent/ArticleContent.esm.js.map +1 -0
- package/dist/components/AuthorBox/AuthorBox.esm.js +25 -0
- package/dist/components/AuthorBox/AuthorBox.esm.js.map +1 -0
- package/dist/components/Buttons/AddToCollectionButton.esm.js +68 -0
- package/dist/components/Buttons/AddToCollectionButton.esm.js.map +1 -0
- package/dist/components/Buttons/AskQuestionButton.esm.js +40 -0
- package/dist/components/Buttons/AskQuestionButton.esm.js.map +1 -0
- package/dist/components/Buttons/BackToArticlesButton.esm.js +40 -0
- package/dist/components/Buttons/BackToArticlesButton.esm.js.map +1 -0
- package/dist/components/Buttons/BackToCollectionsButton.esm.js +25 -0
- package/dist/components/Buttons/BackToCollectionsButton.esm.js.map +1 -0
- package/dist/components/Buttons/BackToQuestionsButton.esm.js +40 -0
- package/dist/components/Buttons/BackToQuestionsButton.esm.js.map +1 -0
- package/dist/components/Buttons/CreateCollectionButton.esm.js +29 -0
- package/dist/components/Buttons/CreateCollectionButton.esm.js.map +1 -0
- package/dist/components/Buttons/EntityFollowButton.esm.js +34 -0
- package/dist/components/Buttons/EntityFollowButton.esm.js.map +1 -0
- package/dist/components/Buttons/FavoriteButton.esm.js +45 -0
- package/dist/components/Buttons/FavoriteButton.esm.js.map +1 -0
- package/dist/components/Buttons/LinkButton.esm.js +29 -0
- package/dist/components/Buttons/LinkButton.esm.js.map +1 -0
- package/dist/components/Buttons/TagFollowButton.esm.js +34 -0
- package/dist/components/Buttons/TagFollowButton.esm.js.map +1 -0
- package/dist/components/Buttons/VoteButtons.esm.js +75 -0
- package/dist/components/Buttons/VoteButtons.esm.js.map +1 -0
- package/dist/components/Buttons/WriteArticleButton.esm.js +40 -0
- package/dist/components/Buttons/WriteArticleButton.esm.js.map +1 -0
- package/dist/components/CollectionCard/CollectionCard.esm.js +63 -0
- package/dist/components/CollectionCard/CollectionCard.esm.js.map +1 -0
- package/dist/components/CollectionForm/CollectionForm.esm.js +231 -0
- package/dist/components/CollectionForm/CollectionForm.esm.js.map +1 -0
- package/dist/components/CollectionsGrid/CollectionsGrid.esm.js +40 -0
- package/dist/components/CollectionsGrid/CollectionsGrid.esm.js.map +1 -0
- package/dist/components/CollectionsGrid/CollectionsGridContent.esm.js +67 -0
- package/dist/components/CollectionsGrid/CollectionsGridContent.esm.js.map +1 -0
- package/dist/components/CollectionsGrid/CollectionsGridItem.esm.js +48 -0
- package/dist/components/CollectionsGrid/CollectionsGridItem.esm.js.map +1 -0
- package/dist/components/CommentSection/CommentList.esm.js +47 -0
- package/dist/components/CommentSection/CommentList.esm.js.map +1 -0
- package/dist/components/CommentSection/CommentSection.esm.js +126 -0
- package/dist/components/CommentSection/CommentSection.esm.js.map +1 -0
- package/dist/components/DeleteModal/DeleteModal.esm.js +88 -0
- package/dist/components/DeleteModal/DeleteModal.esm.js.map +1 -0
- package/dist/components/EntitiesGrid/EntitiesGrid.esm.js +45 -0
- package/dist/components/EntitiesGrid/EntitiesGrid.esm.js.map +1 -0
- package/dist/components/EntitiesGrid/EntitiesGridItem.esm.js +61 -0
- package/dist/components/EntitiesGrid/EntitiesGridItem.esm.js.map +1 -0
- package/dist/components/FilterPanel/DateRangeFilter.esm.js +110 -0
- package/dist/components/FilterPanel/DateRangeFilter.esm.js.map +1 -0
- package/dist/components/FilterPanel/FilterPanel.esm.js +238 -0
- package/dist/components/FilterPanel/FilterPanel.esm.js.map +1 -0
- package/dist/components/FollowedLists/FollowedEntitiesList.esm.js +43 -0
- package/dist/components/FollowedLists/FollowedEntitiesList.esm.js.map +1 -0
- package/dist/components/FollowedLists/FollowedTagsList.esm.js +43 -0
- package/dist/components/FollowedLists/FollowedTagsList.esm.js.map +1 -0
- package/dist/components/HomePageCards/ImpactCard.esm.js +22 -0
- package/dist/components/HomePageCards/ImpactCard.esm.js.map +1 -0
- package/dist/components/HomePageCards/PostsCard.esm.js +42 -0
- package/dist/components/HomePageCards/PostsCard.esm.js.map +1 -0
- package/dist/components/Links/Links.esm.js +33 -0
- package/dist/components/Links/Links.esm.js.map +1 -0
- package/dist/components/MarkdownEditor/MarkdownEditor.esm.js +63 -0
- package/dist/components/MarkdownEditor/MarkdownEditor.esm.js.map +1 -0
- package/dist/components/MarkdownRenderer/MarkdownRenderer.esm.js +97 -0
- package/dist/components/MarkdownRenderer/MarkdownRenderer.esm.js.map +1 -0
- package/dist/components/PostAnonymouslyCheckbox/PostAnonymouslyCheckbox.esm.js +34 -0
- package/dist/components/PostAnonymouslyCheckbox/PostAnonymouslyCheckbox.esm.js.map +1 -0
- package/dist/components/PostForm/EntitiesInput.esm.js +100 -0
- package/dist/components/PostForm/EntitiesInput.esm.js.map +1 -0
- package/dist/components/PostForm/PostForm.esm.js +283 -0
- package/dist/components/PostForm/PostForm.esm.js.map +1 -0
- package/dist/components/PostForm/TagInput.esm.js +81 -0
- package/dist/components/PostForm/TagInput.esm.js.map +1 -0
- package/dist/components/PostHighlightList/PostHighlightList.esm.js +72 -0
- package/dist/components/PostHighlightList/PostHighlightList.esm.js.map +1 -0
- package/dist/components/PostsContainer/NoPostsCard.esm.js +50 -0
- package/dist/components/PostsContainer/NoPostsCard.esm.js.map +1 -0
- package/dist/components/PostsContainer/PostList.esm.js +113 -0
- package/dist/components/PostsContainer/PostList.esm.js.map +1 -0
- package/dist/components/PostsContainer/PostListItem.esm.js +138 -0
- package/dist/components/PostsContainer/PostListItem.esm.js.map +1 -0
- package/dist/components/PostsContainer/PostsContainer.esm.js +152 -0
- package/dist/components/PostsContainer/PostsContainer.esm.js.map +1 -0
- package/dist/components/PostsGrid/PostsGrid.esm.js +146 -0
- package/dist/components/PostsGrid/PostsGrid.esm.js.map +1 -0
- package/dist/components/PostsGrid/PostsGridContent.esm.js +90 -0
- package/dist/components/PostsGrid/PostsGridContent.esm.js.map +1 -0
- package/dist/components/PostsGrid/PostsGridItem.esm.js +57 -0
- package/dist/components/PostsGrid/PostsGridItem.esm.js.map +1 -0
- package/dist/components/QuestionCard/QuestionCard.esm.js +107 -0
- package/dist/components/QuestionCard/QuestionCard.esm.js.map +1 -0
- package/dist/components/QuestionsTable/QuestionTableRow.esm.js +21 -0
- package/dist/components/QuestionsTable/QuestionTableRow.esm.js.map +1 -0
- package/dist/components/QuestionsTable/QuestionsTable.esm.js +131 -0
- package/dist/components/QuestionsTable/QuestionsTable.esm.js.map +1 -0
- package/dist/components/RelativeTimeWithTooltip/RelativeTimeWithTooltip.esm.js +22 -0
- package/dist/components/RelativeTimeWithTooltip/RelativeTimeWithTooltip.esm.js.map +1 -0
- package/dist/components/StatsChart/StatsChart.esm.js +245 -0
- package/dist/components/StatsChart/StatsChart.esm.js.map +1 -0
- package/dist/components/SummaryStatsGrid/SummaryStatsGrid.esm.js +53 -0
- package/dist/components/SummaryStatsGrid/SummaryStatsGrid.esm.js.map +1 -0
- package/dist/components/TagsAndEntities/EntityChip.esm.js +80 -0
- package/dist/components/TagsAndEntities/EntityChip.esm.js.map +1 -0
- package/dist/components/TagsAndEntities/TagChip.esm.js +78 -0
- package/dist/components/TagsAndEntities/TagChip.esm.js.map +1 -0
- package/dist/components/TagsAndEntities/TagsAndEntities.esm.js +32 -0
- package/dist/components/TagsAndEntities/TagsAndEntities.esm.js.map +1 -0
- package/dist/components/TagsGrid/EditTagModal.esm.js +73 -0
- package/dist/components/TagsGrid/EditTagModal.esm.js.map +1 -0
- package/dist/components/TagsGrid/TagGridItem.esm.js +67 -0
- package/dist/components/TagsGrid/TagGridItem.esm.js.map +1 -0
- package/dist/components/TagsGrid/TagsGrid.esm.js +49 -0
- package/dist/components/TagsGrid/TagsGrid.esm.js.map +1 -0
- package/dist/components/TopRankingUsersCard/TopRankingUsersCard.esm.js +161 -0
- package/dist/components/TopRankingUsersCard/TopRankingUsersCard.esm.js.map +1 -0
- package/dist/components/TopRankingUsersCard/TrophyIcon.esm.js +19 -0
- package/dist/components/TopRankingUsersCard/TrophyIcon.esm.js.map +1 -0
- package/dist/components/TopRankingUsersCard/styles.esm.js +23 -0
- package/dist/components/TopRankingUsersCard/styles.esm.js.map +1 -0
- package/dist/components/UsersGrid/UsersGrid.esm.js +43 -0
- package/dist/components/UsersGrid/UsersGrid.esm.js.map +1 -0
- package/dist/components/UsersGrid/UsersGridItem.esm.js +66 -0
- package/dist/components/UsersGrid/UsersGridItem.esm.js.map +1 -0
- package/dist/index.d.ts +568 -1
- package/dist/index.esm.js +45 -1
- package/dist/index.esm.js.map +1 -1
- package/dist/routes.esm.js +56 -1
- package/dist/routes.esm.js.map +1 -1
- package/dist/translation.esm.js +470 -0
- package/dist/translation.esm.js.map +1 -0
- package/dist/utils/hooks.esm.js +735 -0
- package/dist/utils/hooks.esm.js.map +1 -0
- package/dist/utils/utils.esm.js +93 -0
- package/dist/utils/utils.esm.js.map +1 -0
- package/package.json +32 -5
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"CommentSection.esm.js","sources":["../../../src/components/CommentSection/CommentSection.tsx"],"sourcesContent":["import React, { useEffect, useState } from 'react';\nimport { Box, Button, Grid, TextField } from '@material-ui/core';\nimport { Link } from '@backstage/core-components';\nimport {\n AnswerResponse,\n PostResponse,\n qetaCreateCommentPermission,\n} from '@drodil/backstage-plugin-qeta-common';\nimport { Controller, useForm } from 'react-hook-form';\nimport { useAnalytics, useApi } from '@backstage/core-plugin-api';\nimport { CommentList } from './CommentList';\nimport { qetaApiRef } from '../../api';\nimport { useTranslation } from '../../utils/hooks';\nimport { confirmNavigationIfEdited } from '../../utils/utils';\nimport { RequirePermission } from '@backstage/plugin-permission-react';\n\nexport const CommentSection = (props: {\n onCommentPost: (question: PostResponse, answer?: AnswerResponse) => void;\n onCommentDelete: (question: PostResponse, answer?: AnswerResponse) => void;\n post: PostResponse;\n answer?: AnswerResponse;\n className?: string;\n}) => {\n const { answer, post, onCommentPost, onCommentDelete } = props;\n const analytics = useAnalytics();\n const qetaApi = useApi(qetaApiRef);\n const [posting, setPosting] = React.useState(false);\n const [formVisible, setFormVisible] = useState(false);\n const [edited, setEdited] = React.useState(false);\n const { t } = useTranslation();\n const {\n handleSubmit,\n control,\n formState: { errors },\n reset,\n } = useForm<{ content: string }>({});\n\n const postComment = (data: { content: string }) => {\n setPosting(true);\n if (answer) {\n qetaApi.commentAnswer(post.id, answer.id, data.content).then(a => {\n setFormVisible(false);\n analytics.captureEvent('comment', 'answer');\n reset();\n setPosting(false);\n setEdited(false);\n onCommentPost(post, a);\n });\n return;\n }\n\n qetaApi.commentPost(post.id, data.content).then(q => {\n setFormVisible(false);\n analytics.captureEvent('comment', post.type);\n reset();\n setPosting(false);\n setEdited(false);\n onCommentPost(q);\n });\n };\n\n useEffect(() => {\n return confirmNavigationIfEdited(edited);\n }, [edited]);\n\n return (\n <Box marginLeft={9} className={`${props.className} qetaCommentSection`}>\n <CommentList\n question={post}\n answer={answer}\n onCommentDelete={onCommentDelete}\n />\n <RequirePermission\n permission={qetaCreateCommentPermission}\n errorPage={<></>}\n >\n {!formVisible && (\n <Link\n underline=\"none\"\n to=\"#\"\n className=\"qetaAddCommentBtn\"\n onClick={() => setFormVisible(true)}\n >\n {t('commentSection.addComment')}\n </Link>\n )}\n {formVisible && (\n <form\n onSubmit={handleSubmit(postComment)}\n onChange={() => {\n setEdited(true);\n }}\n className=\"qetaCommentForm\"\n >\n <Grid container>\n <Grid item xs={11}>\n <Controller\n control={control}\n defaultValue=\"\"\n rules={{\n required: true,\n }}\n render={({ field: { onChange, value } }) => (\n <TextField\n id=\"comment\"\n multiline\n minRows={2}\n fullWidth\n className=\"qetaCommentInput\"\n value={value}\n placeholder={t('commentSection.input.placeholder')}\n onChange={onChange}\n variant=\"outlined\"\n error={'content' in errors}\n />\n )}\n name=\"content\"\n />\n </Grid>\n <Grid item xs={1}>\n <Button\n variant=\"contained\"\n size=\"small\"\n className=\"qetaCommentBtn\"\n type=\"submit\"\n color=\"primary\"\n disabled={posting}\n >\n {t('commentSection.post')}\n </Button>\n </Grid>\n </Grid>\n </form>\n )}\n </RequirePermission>\n </Box>\n );\n};\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import { Modal, Backdrop, Box, Typography, Button } from '@material-ui/core';
|
|
2
|
+
import { Alert } from '@material-ui/lab';
|
|
3
|
+
import DeleteIcon from '@material-ui/icons/Delete';
|
|
4
|
+
import React from 'react';
|
|
5
|
+
import { useBasePath, useStyles, useTranslation } from '../../utils/hooks.esm.js';
|
|
6
|
+
import { useApi } from '@backstage/core-plugin-api';
|
|
7
|
+
import { useNavigate } from 'react-router-dom';
|
|
8
|
+
import { qetaApiRef } from '../../api.esm.js';
|
|
9
|
+
|
|
10
|
+
const DeleteModal = (props) => {
|
|
11
|
+
const qetaApi = useApi(qetaApiRef);
|
|
12
|
+
const base_path = useBasePath();
|
|
13
|
+
const navigate = useNavigate();
|
|
14
|
+
const { entity, open, question, onClose } = props;
|
|
15
|
+
const styles = useStyles();
|
|
16
|
+
const [error, setError] = React.useState(false);
|
|
17
|
+
const { t } = useTranslation();
|
|
18
|
+
const isQuestion = "title" in entity;
|
|
19
|
+
const isCollection = "owner" in entity;
|
|
20
|
+
const title = isCollection ? t("deleteModal.title.collection") : isQuestion ? t("deleteModal.title.question") : t("deleteModal.title.answer");
|
|
21
|
+
const handleDelete = () => {
|
|
22
|
+
if (isCollection) {
|
|
23
|
+
qetaApi.deleteCollection(entity.id).catch((_) => setError(true)).then((ret) => {
|
|
24
|
+
if (ret) {
|
|
25
|
+
onClose();
|
|
26
|
+
navigate(`${base_path}/qeta`);
|
|
27
|
+
} else {
|
|
28
|
+
setError(true);
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
} else if (isQuestion) {
|
|
32
|
+
qetaApi.deletePost(entity.id).catch((_) => setError(true)).then((ret) => {
|
|
33
|
+
if (ret) {
|
|
34
|
+
onClose();
|
|
35
|
+
navigate(`${base_path}/qeta`);
|
|
36
|
+
} else {
|
|
37
|
+
setError(true);
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
} else if (question) {
|
|
41
|
+
qetaApi.deleteAnswer(question.id, entity.id).catch((_) => setError(true)).then((ret) => {
|
|
42
|
+
if (ret) {
|
|
43
|
+
onClose();
|
|
44
|
+
window.location.reload();
|
|
45
|
+
} else {
|
|
46
|
+
setError(true);
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
};
|
|
51
|
+
return /* @__PURE__ */ React.createElement(
|
|
52
|
+
Modal,
|
|
53
|
+
{
|
|
54
|
+
open,
|
|
55
|
+
onClose,
|
|
56
|
+
className: "qetaDeleteModal",
|
|
57
|
+
"aria-labelledby": "modal-modal-title",
|
|
58
|
+
"aria-describedby": "modal-modal-description",
|
|
59
|
+
closeAfterTransition: true,
|
|
60
|
+
BackdropComponent: Backdrop,
|
|
61
|
+
BackdropProps: {
|
|
62
|
+
timeout: 500
|
|
63
|
+
}
|
|
64
|
+
},
|
|
65
|
+
/* @__PURE__ */ React.createElement(Box, { className: `qetaDeleteModalContent ${styles.deleteModal}` }, error && /* @__PURE__ */ React.createElement(Alert, { severity: "error" }, t("deleteModal.errorDeleting")), /* @__PURE__ */ React.createElement(
|
|
66
|
+
Typography,
|
|
67
|
+
{
|
|
68
|
+
id: "modal-modal-title",
|
|
69
|
+
className: "qetaDeleteModalTitle",
|
|
70
|
+
variant: "h6",
|
|
71
|
+
component: "h2"
|
|
72
|
+
},
|
|
73
|
+
title
|
|
74
|
+
), /* @__PURE__ */ React.createElement(
|
|
75
|
+
Button,
|
|
76
|
+
{
|
|
77
|
+
onClick: handleDelete,
|
|
78
|
+
className: "qetaDeleteModalDeleteBtn",
|
|
79
|
+
startIcon: /* @__PURE__ */ React.createElement(DeleteIcon, null),
|
|
80
|
+
color: "secondary"
|
|
81
|
+
},
|
|
82
|
+
t("deleteModal.deleteButton")
|
|
83
|
+
), /* @__PURE__ */ React.createElement(Button, { onClick: onClose, className: "qetaDeleteModalCancelBtn" }, t("deleteModal.cancelButton")))
|
|
84
|
+
);
|
|
85
|
+
};
|
|
86
|
+
|
|
87
|
+
export { DeleteModal };
|
|
88
|
+
//# sourceMappingURL=DeleteModal.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DeleteModal.esm.js","sources":["../../../src/components/DeleteModal/DeleteModal.tsx"],"sourcesContent":["import {\n AnswerResponse,\n CollectionResponse,\n PostResponse,\n} from '@drodil/backstage-plugin-qeta-common';\nimport { Backdrop, Box, Button, Modal, Typography } from '@material-ui/core';\nimport { Alert } from '@material-ui/lab';\nimport Delete from '@material-ui/icons/Delete';\nimport React from 'react';\nimport { useBasePath, useStyles, useTranslation } from '../../utils/hooks';\nimport { useApi } from '@backstage/core-plugin-api';\nimport { useNavigate } from 'react-router-dom';\nimport { qetaApiRef } from '../../api';\n\nexport const DeleteModal = (props: {\n entity: PostResponse | AnswerResponse | CollectionResponse;\n open: boolean;\n onClose: () => void;\n question?: PostResponse;\n}) => {\n const qetaApi = useApi(qetaApiRef);\n const base_path = useBasePath();\n const navigate = useNavigate();\n const { entity, open, question, onClose } = props;\n const styles = useStyles();\n const [error, setError] = React.useState(false);\n const { t } = useTranslation();\n const isQuestion = 'title' in entity;\n const isCollection = 'owner' in entity;\n\n // eslint-disable-next-line no-nested-ternary\n const title = isCollection\n ? t('deleteModal.title.collection')\n : isQuestion\n ? t('deleteModal.title.question')\n : t('deleteModal.title.answer');\n\n const handleDelete = () => {\n if (isCollection) {\n qetaApi\n .deleteCollection(entity.id)\n .catch(_ => setError(true))\n .then(ret => {\n if (ret) {\n onClose();\n navigate(`${base_path}/qeta`);\n } else {\n setError(true);\n }\n });\n } else if (isQuestion) {\n qetaApi\n .deletePost(entity.id)\n .catch(_ => setError(true))\n .then(ret => {\n if (ret) {\n onClose();\n navigate(`${base_path}/qeta`);\n } else {\n setError(true);\n }\n });\n } else if (question) {\n qetaApi\n .deleteAnswer(question.id, entity.id)\n .catch(_ => setError(true))\n .then(ret => {\n if (ret) {\n onClose();\n window.location.reload();\n } else {\n setError(true);\n }\n });\n }\n };\n\n return (\n <Modal\n open={open}\n onClose={onClose}\n className=\"qetaDeleteModal\"\n aria-labelledby=\"modal-modal-title\"\n aria-describedby=\"modal-modal-description\"\n closeAfterTransition\n BackdropComponent={Backdrop}\n BackdropProps={{\n timeout: 500,\n }}\n >\n <Box className={`qetaDeleteModalContent ${styles.deleteModal}`}>\n {error && (\n <Alert severity=\"error\">{t('deleteModal.errorDeleting')}</Alert>\n )}\n <Typography\n id=\"modal-modal-title\"\n className=\"qetaDeleteModalTitle\"\n variant=\"h6\"\n component=\"h2\"\n >\n {title}\n </Typography>\n <Button\n onClick={handleDelete}\n className=\"qetaDeleteModalDeleteBtn\"\n startIcon={<Delete />}\n color=\"secondary\"\n >\n {t('deleteModal.deleteButton')}\n </Button>\n <Button onClick={onClose} className=\"qetaDeleteModalCancelBtn\">\n {t('deleteModal.cancelButton')}\n </Button>\n </Box>\n </Modal>\n );\n};\n"],"names":["Delete"],"mappings":"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"}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { Grid, TextField, IconButton, Typography } from '@material-ui/core';
|
|
2
|
+
import React from 'react';
|
|
3
|
+
import { useTranslation, useQetaApi } from '../../utils/hooks.esm.js';
|
|
4
|
+
import { Progress, WarningPanel } from '@backstage/core-components';
|
|
5
|
+
import { EntitiesGridItem } from './EntitiesGridItem.esm.js';
|
|
6
|
+
|
|
7
|
+
const EntitiesGrid = () => {
|
|
8
|
+
const [searchQuery, setSearchQuery] = React.useState("");
|
|
9
|
+
const { t } = useTranslation();
|
|
10
|
+
const {
|
|
11
|
+
value: response,
|
|
12
|
+
loading,
|
|
13
|
+
error
|
|
14
|
+
} = useQetaApi((api) => api.getEntities(), []);
|
|
15
|
+
if (loading) {
|
|
16
|
+
return /* @__PURE__ */ React.createElement(Progress, null);
|
|
17
|
+
}
|
|
18
|
+
if (error || response === void 0) {
|
|
19
|
+
return /* @__PURE__ */ React.createElement(WarningPanel, { severity: "error", title: t("entitiesPage.errorLoading") }, error?.message);
|
|
20
|
+
}
|
|
21
|
+
const filterData = (query, data) => {
|
|
22
|
+
if (!query) {
|
|
23
|
+
return data;
|
|
24
|
+
}
|
|
25
|
+
return data.filter(
|
|
26
|
+
(entity) => entity.entityRef.toLowerCase().includes(query)
|
|
27
|
+
);
|
|
28
|
+
};
|
|
29
|
+
const entities = filterData(searchQuery, response);
|
|
30
|
+
return /* @__PURE__ */ React.createElement(Grid, { container: true, className: "qetaEntitiesContainer" }, /* @__PURE__ */ React.createElement(Grid, { item: true, xs: 12 }, /* @__PURE__ */ React.createElement(
|
|
31
|
+
TextField,
|
|
32
|
+
{
|
|
33
|
+
id: "search-bar",
|
|
34
|
+
className: "text qetaEntitiesContainerSearchInput",
|
|
35
|
+
onChange: (event) => setSearchQuery(event.target.value),
|
|
36
|
+
label: t("entitiesPage.search.label"),
|
|
37
|
+
variant: "outlined",
|
|
38
|
+
placeholder: t("entitiesPage.search.placeholder"),
|
|
39
|
+
size: "small"
|
|
40
|
+
}
|
|
41
|
+
), /* @__PURE__ */ React.createElement(IconButton, { type: "submit", "aria-label": "search" })), /* @__PURE__ */ React.createElement(Grid, { item: true, xs: 12 }, /* @__PURE__ */ React.createElement(Typography, { variant: "h6", className: "qetaEntitiesContainerTitle" }, t("entitiesPage.entities", { count: entities.length }))), /* @__PURE__ */ React.createElement(Grid, { container: true, item: true, xs: 12, alignItems: "stretch" }, entities.map((entity) => /* @__PURE__ */ React.createElement(EntitiesGridItem, { entity, key: entity.entityRef }))));
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
export { EntitiesGrid };
|
|
45
|
+
//# sourceMappingURL=EntitiesGrid.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"EntitiesGrid.esm.js","sources":["../../../src/components/EntitiesGrid/EntitiesGrid.tsx"],"sourcesContent":["import { Grid, IconButton, TextField, Typography } from '@material-ui/core';\nimport React from 'react';\nimport { useQetaApi, useTranslation } from '../../utils/hooks';\nimport { Progress, WarningPanel } from '@backstage/core-components';\nimport { EntityResponse } from '@drodil/backstage-plugin-qeta-common';\nimport { EntitiesGridItem } from './EntitiesGridItem';\n\nexport const EntitiesGrid = () => {\n const [searchQuery, setSearchQuery] = React.useState('');\n const { t } = useTranslation();\n\n const {\n value: response,\n loading,\n error,\n } = useQetaApi(api => api.getEntities(), []);\n\n if (loading) {\n return <Progress />;\n }\n\n if (error || response === undefined) {\n return (\n <WarningPanel severity=\"error\" title={t('entitiesPage.errorLoading')}>\n {error?.message}\n </WarningPanel>\n );\n }\n\n const filterData = (query: string, data: EntityResponse[]) => {\n if (!query) {\n return data;\n }\n return data.filter(entity =>\n entity.entityRef.toLowerCase().includes(query),\n );\n };\n\n const entities = filterData(searchQuery, response);\n\n return (\n <Grid container className=\"qetaEntitiesContainer\">\n <Grid item xs={12}>\n <TextField\n id=\"search-bar\"\n className=\"text qetaEntitiesContainerSearchInput\"\n onChange={(\n event: React.ChangeEvent<HTMLTextAreaElement | HTMLInputElement>,\n ) => setSearchQuery(event.target.value)}\n label={t('entitiesPage.search.label')}\n variant=\"outlined\"\n placeholder={t('entitiesPage.search.placeholder')}\n size=\"small\"\n />\n <IconButton type=\"submit\" aria-label=\"search\" />\n </Grid>\n <Grid item xs={12}>\n <Typography variant=\"h6\" className=\"qetaEntitiesContainerTitle\">\n {t('entitiesPage.entities', { count: entities.length })}\n </Typography>\n </Grid>\n <Grid container item xs={12} alignItems=\"stretch\">\n {entities.map(entity => (\n <EntitiesGridItem entity={entity} key={entity.entityRef} />\n ))}\n </Grid>\n </Grid>\n );\n};\n"],"names":[],"mappings":";;;;;;AAOO,MAAM,eAAe,MAAM;AAChC,EAAA,MAAM,CAAC,WAAa,EAAA,cAAc,CAAI,GAAA,KAAA,CAAM,SAAS,EAAE,CAAA,CAAA;AACvD,EAAM,MAAA,EAAE,CAAE,EAAA,GAAI,cAAe,EAAA,CAAA;AAE7B,EAAM,MAAA;AAAA,IACJ,KAAO,EAAA,QAAA;AAAA,IACP,OAAA;AAAA,IACA,KAAA;AAAA,MACE,UAAW,CAAA,CAAA,GAAA,KAAO,IAAI,WAAY,EAAA,EAAG,EAAE,CAAA,CAAA;AAE3C,EAAA,IAAI,OAAS,EAAA;AACX,IAAA,2CAAQ,QAAS,EAAA,IAAA,CAAA,CAAA;AAAA,GACnB;AAEA,EAAI,IAAA,KAAA,IAAS,aAAa,KAAW,CAAA,EAAA;AACnC,IACE,uBAAA,KAAA,CAAA,aAAA,CAAC,gBAAa,QAAS,EAAA,OAAA,EAAQ,OAAO,CAAE,CAAA,2BAA2B,CAChE,EAAA,EAAA,KAAA,EAAO,OACV,CAAA,CAAA;AAAA,GAEJ;AAEA,EAAM,MAAA,UAAA,GAAa,CAAC,KAAA,EAAe,IAA2B,KAAA;AAC5D,IAAA,IAAI,CAAC,KAAO,EAAA;AACV,MAAO,OAAA,IAAA,CAAA;AAAA,KACT;AACA,IAAA,OAAO,IAAK,CAAA,MAAA;AAAA,MAAO,YACjB,MAAO,CAAA,SAAA,CAAU,WAAY,EAAA,CAAE,SAAS,KAAK,CAAA;AAAA,KAC/C,CAAA;AAAA,GACF,CAAA;AAEA,EAAM,MAAA,QAAA,GAAW,UAAW,CAAA,WAAA,EAAa,QAAQ,CAAA,CAAA;AAEjD,EACE,uBAAA,KAAA,CAAA,aAAA,CAAC,IAAK,EAAA,EAAA,SAAA,EAAS,IAAC,EAAA,SAAA,EAAU,uBACxB,EAAA,kBAAA,KAAA,CAAA,aAAA,CAAC,IAAK,EAAA,EAAA,IAAA,EAAI,IAAC,EAAA,EAAA,EAAI,EACb,EAAA,kBAAA,KAAA,CAAA,aAAA;AAAA,IAAC,SAAA;AAAA,IAAA;AAAA,MACC,EAAG,EAAA,YAAA;AAAA,MACH,SAAU,EAAA,uCAAA;AAAA,MACV,UAAU,CACR,KAAA,KACG,cAAe,CAAA,KAAA,CAAM,OAAO,KAAK,CAAA;AAAA,MACtC,KAAA,EAAO,EAAE,2BAA2B,CAAA;AAAA,MACpC,OAAQ,EAAA,UAAA;AAAA,MACR,WAAA,EAAa,EAAE,iCAAiC,CAAA;AAAA,MAChD,IAAK,EAAA,OAAA;AAAA,KAAA;AAAA,GACP,kBACC,KAAA,CAAA,aAAA,CAAA,UAAA,EAAA,EAAW,IAAK,EAAA,QAAA,EAAS,cAAW,QAAS,EAAA,CAChD,CACA,kBAAA,KAAA,CAAA,aAAA,CAAC,IAAK,EAAA,EAAA,IAAA,EAAI,MAAC,EAAI,EAAA,EAAA,EAAA,kBACZ,KAAA,CAAA,aAAA,CAAA,UAAA,EAAA,EAAW,OAAQ,EAAA,IAAA,EAAK,WAAU,4BAChC,EAAA,EAAA,CAAA,CAAE,uBAAyB,EAAA,EAAE,KAAO,EAAA,QAAA,CAAS,QAAQ,CACxD,CACF,CAAA,kBACC,KAAA,CAAA,aAAA,CAAA,IAAA,EAAA,EAAK,WAAS,IAAC,EAAA,IAAA,EAAI,IAAC,EAAA,EAAA,EAAI,EAAI,EAAA,UAAA,EAAW,aACrC,QAAS,CAAA,GAAA,CAAI,CACZ,MAAA,qBAAA,KAAA,CAAA,aAAA,CAAC,gBAAiB,EAAA,EAAA,MAAA,EAAgB,KAAK,MAAO,CAAA,SAAA,EAAW,CAC1D,CACH,CACF,CAAA,CAAA;AAEJ;;;;"}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { Grid, Card, CardActionArea, CardHeader, Avatar, CardContent, Typography, CardActions, Button } from '@material-ui/core';
|
|
2
|
+
import React from 'react';
|
|
3
|
+
import { useRouteRef } from '@backstage/core-plugin-api';
|
|
4
|
+
import { useNavigate } from 'react-router-dom';
|
|
5
|
+
import { useTranslation, useEntityFollow } from '../../utils/hooks.esm.js';
|
|
6
|
+
import { useEntityPresentation } from '@backstage/plugin-catalog-react';
|
|
7
|
+
import { entityRouteRef } from '../../routes.esm.js';
|
|
8
|
+
import { parseEntityRef } from '@backstage/catalog-model';
|
|
9
|
+
|
|
10
|
+
const EntitiesGridItem = (props) => {
|
|
11
|
+
const { entity } = props;
|
|
12
|
+
const entityRoute = useRouteRef(entityRouteRef);
|
|
13
|
+
const navigate = useNavigate();
|
|
14
|
+
const { t } = useTranslation();
|
|
15
|
+
const entityFollow = useEntityFollow();
|
|
16
|
+
const compound = parseEntityRef(entity.entityRef);
|
|
17
|
+
const { primaryTitle, Icon } = useEntityPresentation(compound);
|
|
18
|
+
return /* @__PURE__ */ React.createElement(Grid, { item: true, xs: 3 }, /* @__PURE__ */ React.createElement(
|
|
19
|
+
Card,
|
|
20
|
+
{
|
|
21
|
+
variant: "outlined",
|
|
22
|
+
style: { height: "100%", display: "flex", flexDirection: "column" }
|
|
23
|
+
},
|
|
24
|
+
/* @__PURE__ */ React.createElement(
|
|
25
|
+
CardActionArea,
|
|
26
|
+
{
|
|
27
|
+
onClick: () => navigate(entityRoute({ entityRef: entity.entityRef }))
|
|
28
|
+
},
|
|
29
|
+
/* @__PURE__ */ React.createElement(
|
|
30
|
+
CardHeader,
|
|
31
|
+
{
|
|
32
|
+
title: primaryTitle,
|
|
33
|
+
avatar: Icon ? /* @__PURE__ */ React.createElement(Avatar, null, /* @__PURE__ */ React.createElement(Icon, null)) : null
|
|
34
|
+
}
|
|
35
|
+
),
|
|
36
|
+
/* @__PURE__ */ React.createElement(CardContent, null, /* @__PURE__ */ React.createElement(Typography, { variant: "caption" }, t("common.posts", {
|
|
37
|
+
count: entity.postsCount,
|
|
38
|
+
itemType: "post"
|
|
39
|
+
}), " \xB7 ", t("common.followers", { count: entity.followerCount })))
|
|
40
|
+
),
|
|
41
|
+
/* @__PURE__ */ React.createElement(CardActions, { style: { marginTop: "auto" } }, /* @__PURE__ */ React.createElement(Grid, { container: true, justifyContent: "center" }, /* @__PURE__ */ React.createElement(Grid, { item: true }, /* @__PURE__ */ React.createElement(
|
|
42
|
+
Button,
|
|
43
|
+
{
|
|
44
|
+
size: "small",
|
|
45
|
+
variant: "outlined",
|
|
46
|
+
color: entityFollow.isFollowingEntity(entity.entityRef) ? "secondary" : "primary",
|
|
47
|
+
onClick: () => {
|
|
48
|
+
if (entityFollow.isFollowingEntity(entity.entityRef)) {
|
|
49
|
+
entityFollow.unfollowEntity(entity.entityRef);
|
|
50
|
+
} else {
|
|
51
|
+
entityFollow.followEntity(entity.entityRef);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
},
|
|
55
|
+
entityFollow.isFollowingEntity(entity.entityRef) ? t("entityButton.unfollow") : t("entityButton.follow")
|
|
56
|
+
))))
|
|
57
|
+
));
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
export { EntitiesGridItem };
|
|
61
|
+
//# sourceMappingURL=EntitiesGridItem.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"EntitiesGridItem.esm.js","sources":["../../../src/components/EntitiesGrid/EntitiesGridItem.tsx"],"sourcesContent":["import { EntityResponse } from '@drodil/backstage-plugin-qeta-common';\nimport {\n Avatar,\n Button,\n Card,\n CardActionArea,\n CardActions,\n CardContent,\n CardHeader,\n Grid,\n Typography,\n} from '@material-ui/core';\nimport React from 'react';\nimport { useRouteRef } from '@backstage/core-plugin-api';\nimport { useNavigate } from 'react-router-dom';\nimport { useTranslation } from '../../utils';\nimport { useEntityFollow } from '../../utils/hooks';\nimport { useEntityPresentation } from '@backstage/plugin-catalog-react';\nimport { entityRouteRef } from '../../routes';\nimport { parseEntityRef } from '@backstage/catalog-model';\n\nexport const EntitiesGridItem = (props: { entity: EntityResponse }) => {\n const { entity } = props;\n const entityRoute = useRouteRef(entityRouteRef);\n const navigate = useNavigate();\n const { t } = useTranslation();\n const entityFollow = useEntityFollow();\n const compound = parseEntityRef(entity.entityRef);\n const { primaryTitle, Icon } = useEntityPresentation(compound);\n\n return (\n <Grid item xs={3}>\n <Card\n variant=\"outlined\"\n style={{ height: '100%', display: 'flex', flexDirection: 'column' }}\n >\n <CardActionArea\n onClick={() => navigate(entityRoute({ entityRef: entity.entityRef }))}\n >\n <CardHeader\n title={primaryTitle}\n avatar={\n Icon ? (\n <Avatar>\n <Icon />\n </Avatar>\n ) : null\n }\n />\n <CardContent>\n <Typography variant=\"caption\">\n {t('common.posts', {\n count: entity.postsCount,\n itemType: 'post',\n })}\n {' · '}\n {t('common.followers', { count: entity.followerCount })}\n </Typography>\n </CardContent>\n </CardActionArea>\n <CardActions style={{ marginTop: 'auto' }}>\n <Grid container justifyContent=\"center\">\n <Grid item>\n <Button\n size=\"small\"\n variant=\"outlined\"\n color={\n entityFollow.isFollowingEntity(entity.entityRef)\n ? 'secondary'\n : 'primary'\n }\n onClick={() => {\n if (entityFollow.isFollowingEntity(entity.entityRef)) {\n entityFollow.unfollowEntity(entity.entityRef);\n } else {\n entityFollow.followEntity(entity.entityRef);\n }\n }}\n >\n {entityFollow.isFollowingEntity(entity.entityRef)\n ? t('entityButton.unfollow')\n : t('entityButton.follow')}\n </Button>\n </Grid>\n </Grid>\n </CardActions>\n </Card>\n </Grid>\n );\n};\n"],"names":[],"mappings":";;;;;;;;;AAqBa,MAAA,gBAAA,GAAmB,CAAC,KAAsC,KAAA;AACrE,EAAM,MAAA,EAAE,QAAW,GAAA,KAAA,CAAA;AACnB,EAAM,MAAA,WAAA,GAAc,YAAY,cAAc,CAAA,CAAA;AAC9C,EAAA,MAAM,WAAW,WAAY,EAAA,CAAA;AAC7B,EAAM,MAAA,EAAE,CAAE,EAAA,GAAI,cAAe,EAAA,CAAA;AAC7B,EAAA,MAAM,eAAe,eAAgB,EAAA,CAAA;AACrC,EAAM,MAAA,QAAA,GAAW,cAAe,CAAA,MAAA,CAAO,SAAS,CAAA,CAAA;AAChD,EAAA,MAAM,EAAE,YAAA,EAAc,IAAK,EAAA,GAAI,sBAAsB,QAAQ,CAAA,CAAA;AAE7D,EAAA,uBACG,KAAA,CAAA,aAAA,CAAA,IAAA,EAAA,EAAK,IAAI,EAAA,IAAA,EAAC,IAAI,CACb,EAAA,kBAAA,KAAA,CAAA,aAAA;AAAA,IAAC,IAAA;AAAA,IAAA;AAAA,MACC,OAAQ,EAAA,UAAA;AAAA,MACR,OAAO,EAAE,MAAA,EAAQ,QAAQ,OAAS,EAAA,MAAA,EAAQ,eAAe,QAAS,EAAA;AAAA,KAAA;AAAA,oBAElE,KAAA,CAAA,aAAA;AAAA,MAAC,cAAA;AAAA,MAAA;AAAA,QACC,OAAA,EAAS,MAAM,QAAS,CAAA,WAAA,CAAY,EAAE,SAAW,EAAA,MAAA,CAAO,SAAU,EAAC,CAAC,CAAA;AAAA,OAAA;AAAA,sBAEpE,KAAA,CAAA,aAAA;AAAA,QAAC,UAAA;AAAA,QAAA;AAAA,UACC,KAAO,EAAA,YAAA;AAAA,UACP,QACE,IACE,mBAAA,KAAA,CAAA,aAAA,CAAC,8BACE,KAAA,CAAA,aAAA,CAAA,IAAA,EAAA,IAAK,CACR,CACE,GAAA,IAAA;AAAA,SAAA;AAAA,OAER;AAAA,0CACC,WACC,EAAA,IAAA,kBAAA,KAAA,CAAA,aAAA,CAAC,cAAW,OAAQ,EAAA,SAAA,EAAA,EACjB,EAAE,cAAgB,EAAA;AAAA,QACjB,OAAO,MAAO,CAAA,UAAA;AAAA,QACd,QAAU,EAAA,MAAA;AAAA,OACX,CACA,EAAA,QAAA,EACA,CAAE,CAAA,kBAAA,EAAoB,EAAE,KAAA,EAAO,MAAO,CAAA,aAAA,EAAe,CACxD,CACF,CAAA;AAAA,KACF;AAAA,wCACC,WAAY,EAAA,EAAA,KAAA,EAAO,EAAE,SAAA,EAAW,QAC/B,EAAA,kBAAA,KAAA,CAAA,aAAA,CAAC,IAAK,EAAA,EAAA,SAAA,EAAS,MAAC,cAAe,EAAA,QAAA,EAAA,kBAC5B,KAAA,CAAA,aAAA,CAAA,IAAA,EAAA,EAAK,MAAI,IACR,EAAA,kBAAA,KAAA,CAAA,aAAA;AAAA,MAAC,MAAA;AAAA,MAAA;AAAA,QACC,IAAK,EAAA,OAAA;AAAA,QACL,OAAQ,EAAA,UAAA;AAAA,QACR,OACE,YAAa,CAAA,iBAAA,CAAkB,MAAO,CAAA,SAAS,IAC3C,WACA,GAAA,SAAA;AAAA,QAEN,SAAS,MAAM;AACb,UAAA,IAAI,YAAa,CAAA,iBAAA,CAAkB,MAAO,CAAA,SAAS,CAAG,EAAA;AACpD,YAAa,YAAA,CAAA,cAAA,CAAe,OAAO,SAAS,CAAA,CAAA;AAAA,WACvC,MAAA;AACL,YAAa,YAAA,CAAA,YAAA,CAAa,OAAO,SAAS,CAAA,CAAA;AAAA,WAC5C;AAAA,SACF;AAAA,OAAA;AAAA,MAEC,YAAA,CAAa,kBAAkB,MAAO,CAAA,SAAS,IAC5C,CAAE,CAAA,uBAAuB,CACzB,GAAA,CAAA,CAAE,qBAAqB,CAAA;AAAA,KAE/B,CACF,CACF,CAAA;AAAA,GAEJ,CAAA,CAAA;AAEJ;;;;"}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import React, { useState, useEffect } from 'react';
|
|
2
|
+
import { useMediaQuery, Box, TextField, MenuItem, Typography } from '@material-ui/core';
|
|
3
|
+
import { useStyles, useTranslation } from '../../utils/hooks.esm.js';
|
|
4
|
+
import { formatDate } from '../../utils/utils.esm.js';
|
|
5
|
+
|
|
6
|
+
const DateRangeFilter = (props) => {
|
|
7
|
+
const { value, onChange } = props;
|
|
8
|
+
const styles = useStyles();
|
|
9
|
+
const [dateRangeOption, setDateRangeOption] = useState(
|
|
10
|
+
value
|
|
11
|
+
);
|
|
12
|
+
const { t } = useTranslation();
|
|
13
|
+
const [fromDate, setFromDate] = useState("");
|
|
14
|
+
const [toDate, setToDate] = useState("");
|
|
15
|
+
const [validation, setValidation] = useState({
|
|
16
|
+
isValid: true
|
|
17
|
+
});
|
|
18
|
+
const isSmallScreen = useMediaQuery(
|
|
19
|
+
(theme) => theme.breakpoints.down("sm")
|
|
20
|
+
);
|
|
21
|
+
const localDate = formatDate(/* @__PURE__ */ new Date());
|
|
22
|
+
useEffect(() => {
|
|
23
|
+
setDateRangeOption(value || "");
|
|
24
|
+
if (value && value.indexOf("--") >= 0) {
|
|
25
|
+
setDateRangeOption("custom");
|
|
26
|
+
setFromDate(value.split("--")[0] || "");
|
|
27
|
+
setToDate(value.split("--")[1] || "");
|
|
28
|
+
}
|
|
29
|
+
}, [value]);
|
|
30
|
+
useEffect(() => {
|
|
31
|
+
if (dateRangeOption && dateRangeOption !== "custom") {
|
|
32
|
+
setFromDate("");
|
|
33
|
+
setToDate("");
|
|
34
|
+
onChange(
|
|
35
|
+
"dateRange",
|
|
36
|
+
!dateRangeOption || dateRangeOption === "select" ? "" : dateRangeOption
|
|
37
|
+
);
|
|
38
|
+
}
|
|
39
|
+
}, [dateRangeOption, onChange]);
|
|
40
|
+
useEffect(() => {
|
|
41
|
+
if (fromDate && toDate) {
|
|
42
|
+
const startDate = new Date(fromDate);
|
|
43
|
+
const endDate = new Date(toDate);
|
|
44
|
+
if (startDate <= endDate) {
|
|
45
|
+
setValidation({ isValid: true });
|
|
46
|
+
onChange("dateRange", `${fromDate}--${toDate}`);
|
|
47
|
+
} else {
|
|
48
|
+
setValidation({
|
|
49
|
+
isValid: false,
|
|
50
|
+
message: t("datePicker.invalidRange")
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}, [t, fromDate, toDate, onChange]);
|
|
55
|
+
return /* @__PURE__ */ React.createElement(Box, { display: isSmallScreen ? "block" : "flex", gridGap: "16px" }, /* @__PURE__ */ React.createElement(
|
|
56
|
+
TextField,
|
|
57
|
+
{
|
|
58
|
+
id: "outlined-select-currency",
|
|
59
|
+
select: true,
|
|
60
|
+
label: t("datePicker.range.label"),
|
|
61
|
+
value: dateRangeOption || "select",
|
|
62
|
+
className: styles.dateFilter,
|
|
63
|
+
onChange: (_e) => {
|
|
64
|
+
setDateRangeOption(_e.target.value);
|
|
65
|
+
},
|
|
66
|
+
variant: "outlined",
|
|
67
|
+
defaultValue: "None"
|
|
68
|
+
},
|
|
69
|
+
/* @__PURE__ */ React.createElement(MenuItem, { value: "select" }, t("datePicker.range.default")),
|
|
70
|
+
/* @__PURE__ */ React.createElement(MenuItem, { value: "7-days" }, t("datePicker.range.last7days")),
|
|
71
|
+
/* @__PURE__ */ React.createElement(MenuItem, { value: "30-days" }, t("datePicker.range.last30days")),
|
|
72
|
+
/* @__PURE__ */ React.createElement(MenuItem, { value: "custom" }, t("datePicker.range.custom"))
|
|
73
|
+
), dateRangeOption === "custom" && /* @__PURE__ */ React.createElement(Box, { display: "flex", flexDirection: "column" }, /* @__PURE__ */ React.createElement(Box, { display: isSmallScreen ? "block" : "flex", gridGap: "12px" }, /* @__PURE__ */ React.createElement(
|
|
74
|
+
TextField,
|
|
75
|
+
{
|
|
76
|
+
variant: "outlined",
|
|
77
|
+
label: t("datePicker.from"),
|
|
78
|
+
id: "from-date",
|
|
79
|
+
type: "date",
|
|
80
|
+
value: fromDate,
|
|
81
|
+
className: styles.dateFilter,
|
|
82
|
+
InputLabelProps: { shrink: true },
|
|
83
|
+
error: !validation.isValid,
|
|
84
|
+
onChange: (_e) => setFromDate(_e.target.value),
|
|
85
|
+
inputProps: {
|
|
86
|
+
max: toDate || localDate
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
), /* @__PURE__ */ React.createElement(
|
|
90
|
+
TextField,
|
|
91
|
+
{
|
|
92
|
+
variant: "outlined",
|
|
93
|
+
label: t("datePicker.to"),
|
|
94
|
+
id: "to-date",
|
|
95
|
+
type: "date",
|
|
96
|
+
value: toDate,
|
|
97
|
+
className: styles.dateFilter,
|
|
98
|
+
InputLabelProps: { shrink: true },
|
|
99
|
+
error: !validation.isValid,
|
|
100
|
+
onChange: (_e) => setToDate(_e.target.value),
|
|
101
|
+
inputProps: {
|
|
102
|
+
min: fromDate,
|
|
103
|
+
max: localDate
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
)), validation.message && /* @__PURE__ */ React.createElement(Typography, { color: "error", variant: "body2" }, validation.message)));
|
|
107
|
+
};
|
|
108
|
+
|
|
109
|
+
export { DateRangeFilter };
|
|
110
|
+
//# sourceMappingURL=DateRangeFilter.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DateRangeFilter.esm.js","sources":["../../../src/components/FilterPanel/DateRangeFilter.tsx"],"sourcesContent":["import React, { useEffect, useState } from 'react';\nimport {\n Box,\n MenuItem,\n TextField,\n Theme,\n Typography,\n useMediaQuery,\n} from '@material-ui/core';\nimport { useStyles, useTranslation } from '../../utils/hooks';\nimport { formatDate } from '../../utils/utils';\n\nexport interface DateRangeFilterProps {\n value?: string;\n onChange: (dateRange: 'dateRange', value: string | string[]) => void;\n}\n\ntype DateRangeValidation = {\n isValid: boolean;\n message?: string;\n};\n\nexport const DateRangeFilter = (props: DateRangeFilterProps) => {\n const { value, onChange } = props;\n const styles = useStyles();\n const [dateRangeOption, setDateRangeOption] = useState<string | undefined>(\n value,\n );\n const { t } = useTranslation();\n const [fromDate, setFromDate] = useState('');\n const [toDate, setToDate] = useState('');\n const [validation, setValidation] = useState<DateRangeValidation>({\n isValid: true,\n });\n const isSmallScreen = useMediaQuery<Theme>(theme =>\n theme.breakpoints.down('sm'),\n );\n\n const localDate = formatDate(new Date());\n\n useEffect(() => {\n setDateRangeOption(value || '');\n if (value && value.indexOf('--') >= 0) {\n setDateRangeOption('custom');\n setFromDate(value.split('--')[0] || '');\n setToDate(value.split('--')[1] || '');\n }\n }, [value]);\n\n useEffect(() => {\n if (dateRangeOption && dateRangeOption !== 'custom') {\n setFromDate('');\n setToDate('');\n onChange(\n 'dateRange',\n !dateRangeOption || dateRangeOption === 'select' ? '' : dateRangeOption,\n );\n }\n }, [dateRangeOption, onChange]);\n\n useEffect(() => {\n if (fromDate && toDate) {\n const startDate = new Date(fromDate);\n const endDate = new Date(toDate);\n if (startDate <= endDate) {\n setValidation({ isValid: true });\n onChange('dateRange', `${fromDate}--${toDate}`);\n } else {\n setValidation({\n isValid: false,\n message: t('datePicker.invalidRange'),\n });\n }\n }\n }, [t, fromDate, toDate, onChange]);\n\n return (\n <Box display={isSmallScreen ? 'block' : 'flex'} gridGap=\"16px\">\n <TextField\n id=\"outlined-select-currency\"\n select\n label={t('datePicker.range.label')}\n value={dateRangeOption || 'select'}\n className={styles.dateFilter}\n onChange={_e => {\n setDateRangeOption(_e.target.value);\n }}\n variant=\"outlined\"\n defaultValue=\"None\"\n >\n <MenuItem value=\"select\">{t('datePicker.range.default')}</MenuItem>\n <MenuItem value=\"7-days\">{t('datePicker.range.last7days')}</MenuItem>\n <MenuItem value=\"30-days\">{t('datePicker.range.last30days')}</MenuItem>\n <MenuItem value=\"custom\">{t('datePicker.range.custom')}</MenuItem>\n </TextField>\n {dateRangeOption === 'custom' && (\n <Box display=\"flex\" flexDirection=\"column\">\n <Box display={isSmallScreen ? 'block' : 'flex'} gridGap=\"12px\">\n <TextField\n variant=\"outlined\"\n label={t('datePicker.from')}\n id=\"from-date\"\n type=\"date\"\n value={fromDate}\n className={styles.dateFilter}\n InputLabelProps={{ shrink: true }}\n error={!validation.isValid}\n onChange={_e => setFromDate(_e.target.value)}\n inputProps={{\n max: toDate || localDate,\n }}\n />\n <TextField\n variant=\"outlined\"\n label={t('datePicker.to')}\n id=\"to-date\"\n type=\"date\"\n value={toDate}\n className={styles.dateFilter}\n InputLabelProps={{ shrink: true }}\n error={!validation.isValid}\n onChange={_e => setToDate(_e.target.value)}\n inputProps={{\n min: fromDate,\n max: localDate,\n }}\n />\n </Box>\n {validation.message && (\n <Typography color=\"error\" variant=\"body2\">\n {validation.message}\n </Typography>\n )}\n </Box>\n )}\n </Box>\n );\n};\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
import React, { useEffect } from 'react';
|
|
2
|
+
import { Box, Grid, FormGroup, FormControlLabel, Checkbox, FormControl, FormLabel, RadioGroup, TextField, Divider, Radio } from '@material-ui/core';
|
|
3
|
+
import { useStyles, useQetaApi, useTranslation } from '../../utils/hooks.esm.js';
|
|
4
|
+
import { Autocomplete } from '@material-ui/lab';
|
|
5
|
+
import { stringifyEntityRef } from '@backstage/catalog-model';
|
|
6
|
+
import { useApi } from '@backstage/core-plugin-api';
|
|
7
|
+
import { catalogApiRef } from '@backstage/plugin-catalog-react';
|
|
8
|
+
import { getEntityTitle } from '../../utils/utils.esm.js';
|
|
9
|
+
import { DateRangeFilter } from './DateRangeFilter.esm.js';
|
|
10
|
+
|
|
11
|
+
const radioSelect = (value, label) => {
|
|
12
|
+
return /* @__PURE__ */ React.createElement(
|
|
13
|
+
FormControlLabel,
|
|
14
|
+
{
|
|
15
|
+
value,
|
|
16
|
+
control: /* @__PURE__ */ React.createElement(Radio, { size: "small" }),
|
|
17
|
+
label
|
|
18
|
+
}
|
|
19
|
+
);
|
|
20
|
+
};
|
|
21
|
+
const filterKeys = [
|
|
22
|
+
"orderBy",
|
|
23
|
+
"order",
|
|
24
|
+
"noAnswers",
|
|
25
|
+
"noCorrectAnswer",
|
|
26
|
+
"noVotes",
|
|
27
|
+
"entity",
|
|
28
|
+
"tags",
|
|
29
|
+
"dateRange"
|
|
30
|
+
];
|
|
31
|
+
const FilterPanel = (props) => {
|
|
32
|
+
const {
|
|
33
|
+
onChange,
|
|
34
|
+
filters,
|
|
35
|
+
showEntityFilter = true,
|
|
36
|
+
showTagFilter = true,
|
|
37
|
+
answerFilters = false,
|
|
38
|
+
type
|
|
39
|
+
} = props;
|
|
40
|
+
const styles = useStyles();
|
|
41
|
+
const { value: refs } = useQetaApi((api) => api.getEntities(), []);
|
|
42
|
+
const { value: tags } = useQetaApi((api) => api.getTags(), []);
|
|
43
|
+
const catalogApi = useApi(catalogApiRef);
|
|
44
|
+
const { t } = useTranslation();
|
|
45
|
+
const [availableEntities, setAvailableEntities] = React.useState(null);
|
|
46
|
+
const [selectedEntity, setSelectedEntity] = React.useState(void 0);
|
|
47
|
+
const [availableTags, setAvailableTags] = React.useState(
|
|
48
|
+
null
|
|
49
|
+
);
|
|
50
|
+
useEffect(() => {
|
|
51
|
+
if (tags && tags.length > 0 || filters.tags) {
|
|
52
|
+
const ts = (tags ?? []).map((tag) => tag.tag);
|
|
53
|
+
if (filters.tags) {
|
|
54
|
+
ts.push(...filters.tags);
|
|
55
|
+
}
|
|
56
|
+
setAvailableTags([...new Set(ts)]);
|
|
57
|
+
}
|
|
58
|
+
}, [tags, filters.tags]);
|
|
59
|
+
useEffect(() => {
|
|
60
|
+
const entityRefs = [];
|
|
61
|
+
if (filters.entity && !Array.isArray(filters.entity)) {
|
|
62
|
+
entityRefs.push(filters.entity);
|
|
63
|
+
}
|
|
64
|
+
if (refs && refs?.length > 0) {
|
|
65
|
+
refs?.forEach((ref) => {
|
|
66
|
+
if (ref.entityRef !== filters.entity) {
|
|
67
|
+
entityRefs.push(ref.entityRef);
|
|
68
|
+
}
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
if (entityRefs.length > 0) {
|
|
72
|
+
catalogApi.getEntitiesByRefs({
|
|
73
|
+
entityRefs,
|
|
74
|
+
fields: [
|
|
75
|
+
"kind",
|
|
76
|
+
"metadata.name",
|
|
77
|
+
"metadata.namespace",
|
|
78
|
+
"metadata.title"
|
|
79
|
+
]
|
|
80
|
+
}).then((resp) => {
|
|
81
|
+
const filtered = resp.items.filter((i) => i !== void 0);
|
|
82
|
+
setAvailableEntities(filtered);
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
}, [filters.entity, catalogApi, refs]);
|
|
86
|
+
useEffect(() => {
|
|
87
|
+
if (filters.entity && availableEntities) {
|
|
88
|
+
const value = availableEntities.find(
|
|
89
|
+
(e) => stringifyEntityRef(e) === filters.entity
|
|
90
|
+
);
|
|
91
|
+
setSelectedEntity(value);
|
|
92
|
+
if (!value) {
|
|
93
|
+
onChange("entity", "");
|
|
94
|
+
}
|
|
95
|
+
} else {
|
|
96
|
+
setSelectedEntity(void 0);
|
|
97
|
+
}
|
|
98
|
+
}, [availableEntities, filters.entity, onChange]);
|
|
99
|
+
const handleChange = (event) => {
|
|
100
|
+
let value = event.target.value;
|
|
101
|
+
if (event.target.type === "checkbox") {
|
|
102
|
+
value = event.target.checked ? "true" : "false";
|
|
103
|
+
}
|
|
104
|
+
onChange(event.target.name, value);
|
|
105
|
+
};
|
|
106
|
+
return /* @__PURE__ */ React.createElement(Box, { className: `qetaFilterPanel ${styles.filterPanel}` }, /* @__PURE__ */ React.createElement(Grid, { container: true, spacing: 4 }, /* @__PURE__ */ React.createElement(Grid, { item: true, md: 3, xs: 4 }, /* @__PURE__ */ React.createElement(FormGroup, null, !answerFilters && type !== "article" && /* @__PURE__ */ React.createElement(
|
|
107
|
+
FormControlLabel,
|
|
108
|
+
{
|
|
109
|
+
control: /* @__PURE__ */ React.createElement(
|
|
110
|
+
Checkbox,
|
|
111
|
+
{
|
|
112
|
+
size: "small",
|
|
113
|
+
name: "noAnswers",
|
|
114
|
+
onChange: handleChange,
|
|
115
|
+
checked: filters.noAnswers === "true"
|
|
116
|
+
}
|
|
117
|
+
),
|
|
118
|
+
label: t("filterPanel.noAnswers.label")
|
|
119
|
+
}
|
|
120
|
+
), type !== "article" && /* @__PURE__ */ React.createElement(
|
|
121
|
+
FormControlLabel,
|
|
122
|
+
{
|
|
123
|
+
control: /* @__PURE__ */ React.createElement(
|
|
124
|
+
Checkbox,
|
|
125
|
+
{
|
|
126
|
+
size: "small",
|
|
127
|
+
name: "noCorrectAnswer",
|
|
128
|
+
checked: filters.noCorrectAnswer === "true",
|
|
129
|
+
onChange: handleChange
|
|
130
|
+
}
|
|
131
|
+
),
|
|
132
|
+
label: t("filterPanel.noCorrectAnswers.label")
|
|
133
|
+
}
|
|
134
|
+
), /* @__PURE__ */ React.createElement(
|
|
135
|
+
FormControlLabel,
|
|
136
|
+
{
|
|
137
|
+
control: /* @__PURE__ */ React.createElement(
|
|
138
|
+
Checkbox,
|
|
139
|
+
{
|
|
140
|
+
size: "small",
|
|
141
|
+
name: "noVotes",
|
|
142
|
+
checked: filters.noVotes === "true",
|
|
143
|
+
onChange: handleChange
|
|
144
|
+
}
|
|
145
|
+
),
|
|
146
|
+
label: t("filterPanel.noVotes.label")
|
|
147
|
+
}
|
|
148
|
+
))), /* @__PURE__ */ React.createElement(Grid, { item: true, md: 2, xs: 4 }, /* @__PURE__ */ React.createElement(FormControl, null, /* @__PURE__ */ React.createElement(FormLabel, { id: "qeta-filter-order-by" }, t("filterPanel.orderBy.label")), /* @__PURE__ */ React.createElement(
|
|
149
|
+
RadioGroup,
|
|
150
|
+
{
|
|
151
|
+
"aria-labelledby": "qeta-filter-order-by",
|
|
152
|
+
name: "orderBy",
|
|
153
|
+
value: filters.orderBy,
|
|
154
|
+
onChange: handleChange
|
|
155
|
+
},
|
|
156
|
+
radioSelect("created", t("filterPanel.orderBy.created")),
|
|
157
|
+
!answerFilters && radioSelect("views", t("filterPanel.orderBy.views")),
|
|
158
|
+
radioSelect("score", t("filterPanel.orderBy.score")),
|
|
159
|
+
!answerFilters && radioSelect("answersCount", t("filterPanel.orderBy.answers")),
|
|
160
|
+
radioSelect("updated", t("filterPanel.orderBy.updated"))
|
|
161
|
+
))), /* @__PURE__ */ React.createElement(Grid, { item: true, md: 2, xs: 4 }, /* @__PURE__ */ React.createElement(FormControl, null, /* @__PURE__ */ React.createElement(FormLabel, { id: "qeta-filter-order" }, t("filterPanel.order.label")), /* @__PURE__ */ React.createElement(
|
|
162
|
+
RadioGroup,
|
|
163
|
+
{
|
|
164
|
+
"aria-labelledby": "qeta-filter-order",
|
|
165
|
+
name: "order",
|
|
166
|
+
value: filters.order,
|
|
167
|
+
onChange: handleChange
|
|
168
|
+
},
|
|
169
|
+
radioSelect("desc", t("filterPanel.order.desc")),
|
|
170
|
+
radioSelect("asc", t("filterPanel.order.asc"))
|
|
171
|
+
))), (availableEntities && availableEntities.length > 0 || availableTags && availableTags.length > 0) && (showEntityFilter || showTagFilter) && /* @__PURE__ */ React.createElement(Grid, { item: true, md: 4, xs: 8 }, /* @__PURE__ */ React.createElement(FormLabel, { id: "qeta-filter-entity" }, t("filterPanel.filters.label")), showEntityFilter && availableEntities && availableEntities.length > 0 && /* @__PURE__ */ React.createElement(
|
|
172
|
+
Autocomplete,
|
|
173
|
+
{
|
|
174
|
+
multiple: false,
|
|
175
|
+
className: "qetaEntityFilter",
|
|
176
|
+
value: selectedEntity ?? null,
|
|
177
|
+
id: "entities-select",
|
|
178
|
+
options: availableEntities,
|
|
179
|
+
getOptionLabel: getEntityTitle,
|
|
180
|
+
getOptionSelected: (o, v) => {
|
|
181
|
+
if (!o || !v) {
|
|
182
|
+
return false;
|
|
183
|
+
}
|
|
184
|
+
return stringifyEntityRef(o) === stringifyEntityRef(v);
|
|
185
|
+
},
|
|
186
|
+
onChange: (_e, newValue) => {
|
|
187
|
+
handleChange({
|
|
188
|
+
target: {
|
|
189
|
+
name: "entity",
|
|
190
|
+
value: newValue ? stringifyEntityRef(newValue) : ""
|
|
191
|
+
}
|
|
192
|
+
});
|
|
193
|
+
},
|
|
194
|
+
renderInput: (params) => /* @__PURE__ */ React.createElement(
|
|
195
|
+
TextField,
|
|
196
|
+
{
|
|
197
|
+
...params,
|
|
198
|
+
variant: "outlined",
|
|
199
|
+
margin: "normal",
|
|
200
|
+
label: t("filterPanel.filters.entity.label"),
|
|
201
|
+
placeholder: t(
|
|
202
|
+
"filterPanel.filters.entity.placeholder"
|
|
203
|
+
)
|
|
204
|
+
}
|
|
205
|
+
)
|
|
206
|
+
}
|
|
207
|
+
), showTagFilter && availableTags && availableTags.length > 0 && /* @__PURE__ */ React.createElement(
|
|
208
|
+
Autocomplete,
|
|
209
|
+
{
|
|
210
|
+
multiple: true,
|
|
211
|
+
className: "qetaTagFilter",
|
|
212
|
+
value: filters.tags,
|
|
213
|
+
id: "tags-select",
|
|
214
|
+
options: availableTags,
|
|
215
|
+
onChange: (_e, newValue) => {
|
|
216
|
+
handleChange({
|
|
217
|
+
target: {
|
|
218
|
+
name: "tags",
|
|
219
|
+
value: newValue
|
|
220
|
+
}
|
|
221
|
+
});
|
|
222
|
+
},
|
|
223
|
+
renderInput: (params) => /* @__PURE__ */ React.createElement(
|
|
224
|
+
TextField,
|
|
225
|
+
{
|
|
226
|
+
...params,
|
|
227
|
+
variant: "outlined",
|
|
228
|
+
margin: "normal",
|
|
229
|
+
label: t("filterPanel.filters.tag.label"),
|
|
230
|
+
placeholder: t("filterPanel.filters.tag.placeholder")
|
|
231
|
+
}
|
|
232
|
+
)
|
|
233
|
+
}
|
|
234
|
+
))), /* @__PURE__ */ React.createElement(Box, { marginY: "24px" }, /* @__PURE__ */ React.createElement(Divider, null)), /* @__PURE__ */ React.createElement(DateRangeFilter, { value: filters.dateRange, onChange }));
|
|
235
|
+
};
|
|
236
|
+
|
|
237
|
+
export { FilterPanel, filterKeys };
|
|
238
|
+
//# sourceMappingURL=FilterPanel.esm.js.map
|