@drodil/backstage-plugin-qeta 2.3.0 → 2.3.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.
Files changed (103) hide show
  1. package/dist/api/QetaClient.esm.js +434 -0
  2. package/dist/api/QetaClient.esm.js.map +1 -0
  3. package/dist/components/AskAnonymouslyCheckbox/AskAnonymouslyCheckbox.esm.js +32 -0
  4. package/dist/components/AskAnonymouslyCheckbox/AskAnonymouslyCheckbox.esm.js.map +1 -0
  5. package/dist/components/AskForm/AskForm.esm.js +218 -0
  6. package/dist/components/AskForm/AskForm.esm.js.map +1 -0
  7. package/dist/components/AskForm/EntitiesInput.esm.js +98 -0
  8. package/dist/components/AskForm/EntitiesInput.esm.js.map +1 -0
  9. package/dist/components/AskForm/TagInput.esm.js +86 -0
  10. package/dist/components/AskForm/TagInput.esm.js.map +1 -0
  11. package/dist/components/AskPage/AskPage.esm.js +38 -0
  12. package/dist/components/AskPage/AskPage.esm.js.map +1 -0
  13. package/dist/components/Buttons/AskQuestionButton.esm.js +43 -0
  14. package/dist/components/Buttons/AskQuestionButton.esm.js.map +1 -0
  15. package/dist/components/Buttons/BackToQuestionsButton.esm.js +38 -0
  16. package/dist/components/Buttons/BackToQuestionsButton.esm.js.map +1 -0
  17. package/dist/components/CommentSection/CommentList.esm.js +47 -0
  18. package/dist/components/CommentSection/CommentList.esm.js.map +1 -0
  19. package/dist/components/CommentSection/CommentSection.esm.js +97 -0
  20. package/dist/components/CommentSection/CommentSection.esm.js.map +1 -0
  21. package/dist/components/DeleteModal/DeleteModal.esm.js +77 -0
  22. package/dist/components/DeleteModal/DeleteModal.esm.js.map +1 -0
  23. package/dist/components/FavoritePage/FavoritePage.esm.js +13 -0
  24. package/dist/components/FavoritePage/FavoritePage.esm.js.map +1 -0
  25. package/dist/components/HomePage/HomePage.esm.js +144 -0
  26. package/dist/components/HomePage/HomePage.esm.js.map +1 -0
  27. package/dist/components/HomePage/index.esm.js +2 -0
  28. package/dist/components/HomePage/index.esm.js.map +1 -0
  29. package/dist/components/Links/Links.esm.js +31 -0
  30. package/dist/components/Links/Links.esm.js.map +1 -0
  31. package/dist/components/MarkdownEditor/MarkdownEditor.esm.js +80 -0
  32. package/dist/components/MarkdownEditor/MarkdownEditor.esm.js.map +1 -0
  33. package/dist/components/QuestionHighlightList/QuestionHighlightList.esm.js +61 -0
  34. package/dist/components/QuestionHighlightList/QuestionHighlightList.esm.js.map +1 -0
  35. package/dist/components/QuestionPage/AnswerCard.esm.js +101 -0
  36. package/dist/components/QuestionPage/AnswerCard.esm.js.map +1 -0
  37. package/dist/components/QuestionPage/AnswerForm.esm.js +130 -0
  38. package/dist/components/QuestionPage/AnswerForm.esm.js.map +1 -0
  39. package/dist/components/QuestionPage/AuthorBox.esm.js +25 -0
  40. package/dist/components/QuestionPage/AuthorBox.esm.js.map +1 -0
  41. package/dist/components/QuestionPage/EntityChip.esm.js +26 -0
  42. package/dist/components/QuestionPage/EntityChip.esm.js.map +1 -0
  43. package/dist/components/QuestionPage/FavoriteButton.esm.js +43 -0
  44. package/dist/components/QuestionPage/FavoriteButton.esm.js.map +1 -0
  45. package/dist/components/QuestionPage/LinkButton.esm.js +32 -0
  46. package/dist/components/QuestionPage/LinkButton.esm.js.map +1 -0
  47. package/dist/components/QuestionPage/QuestionCard.esm.js +103 -0
  48. package/dist/components/QuestionPage/QuestionCard.esm.js.map +1 -0
  49. package/dist/components/QuestionPage/QuestionPage.esm.js +131 -0
  50. package/dist/components/QuestionPage/QuestionPage.esm.js.map +1 -0
  51. package/dist/components/QuestionPage/TagsAndEntities.esm.js +44 -0
  52. package/dist/components/QuestionPage/TagsAndEntities.esm.js.map +1 -0
  53. package/dist/components/QuestionPage/VoteButtons.esm.js +146 -0
  54. package/dist/components/QuestionPage/VoteButtons.esm.js.map +1 -0
  55. package/dist/components/QuestionTableCard/Content.esm.js +9 -0
  56. package/dist/components/QuestionTableCard/Content.esm.js.map +1 -0
  57. package/dist/components/QuestionTableCard/QuestionTableRow.esm.js +21 -0
  58. package/dist/components/QuestionTableCard/QuestionTableRow.esm.js.map +1 -0
  59. package/dist/components/QuestionTableCard/QuestionsTable.esm.js +130 -0
  60. package/dist/components/QuestionTableCard/QuestionsTable.esm.js.map +1 -0
  61. package/dist/components/QuestionTableCard/index.esm.js +3 -0
  62. package/dist/components/QuestionTableCard/index.esm.js.map +1 -0
  63. package/dist/components/QuestionsContainer/FilterPanel.esm.js +230 -0
  64. package/dist/components/QuestionsContainer/FilterPanel.esm.js.map +1 -0
  65. package/dist/components/QuestionsContainer/NoQuestionsCard.esm.js +46 -0
  66. package/dist/components/QuestionsContainer/NoQuestionsCard.esm.js.map +1 -0
  67. package/dist/components/QuestionsContainer/QuestionList.esm.js +102 -0
  68. package/dist/components/QuestionsContainer/QuestionList.esm.js.map +1 -0
  69. package/dist/components/QuestionsContainer/QuestionListItem.esm.js +119 -0
  70. package/dist/components/QuestionsContainer/QuestionListItem.esm.js.map +1 -0
  71. package/dist/components/QuestionsContainer/QuestionsContainer.esm.js +231 -0
  72. package/dist/components/QuestionsContainer/QuestionsContainer.esm.js.map +1 -0
  73. package/dist/components/RelativeTimeWithTooltip/RelativeTimeWithTooltip.esm.js +22 -0
  74. package/dist/components/RelativeTimeWithTooltip/RelativeTimeWithTooltip.esm.js.map +1 -0
  75. package/dist/components/Statistics/StatisticsPage.esm.js +13 -0
  76. package/dist/components/Statistics/StatisticsPage.esm.js.map +1 -0
  77. package/dist/components/Statistics/TopRankingUsersCard.esm.js +163 -0
  78. package/dist/components/Statistics/TopRankingUsersCard.esm.js.map +1 -0
  79. package/dist/components/Statistics/TrophyIcon.esm.js +19 -0
  80. package/dist/components/Statistics/TrophyIcon.esm.js.map +1 -0
  81. package/dist/components/Statistics/styles.esm.js +23 -0
  82. package/dist/components/Statistics/styles.esm.js.map +1 -0
  83. package/dist/components/TagPage/TagPage.esm.js +16 -0
  84. package/dist/components/TagPage/TagPage.esm.js.map +1 -0
  85. package/dist/components/TagPage/TagsContainer.esm.js +63 -0
  86. package/dist/components/TagPage/TagsContainer.esm.js.map +1 -0
  87. package/dist/components/UserPage/UserPage.esm.js +24 -0
  88. package/dist/components/UserPage/UserPage.esm.js.map +1 -0
  89. package/dist/index.esm.js +9 -32
  90. package/dist/index.esm.js.map +1 -1
  91. package/dist/plugin.esm.js +60 -0
  92. package/dist/plugin.esm.js.map +1 -0
  93. package/dist/utils/hooks.esm.js +283 -0
  94. package/dist/utils/hooks.esm.js.map +1 -0
  95. package/dist/utils/utils.esm.js +31 -0
  96. package/dist/utils/utils.esm.js.map +1 -0
  97. package/package.json +15 -15
  98. package/dist/esm/index-C7A2tOkT.esm.js +0 -33
  99. package/dist/esm/index-C7A2tOkT.esm.js.map +0 -1
  100. package/dist/esm/index-CFNWBJNR.esm.js +0 -2412
  101. package/dist/esm/index-CFNWBJNR.esm.js.map +0 -1
  102. package/dist/esm/index-rSPV8dxK.esm.js +0 -1120
  103. package/dist/esm/index-rSPV8dxK.esm.js.map +0 -1
@@ -0,0 +1,230 @@
1
+ import React, { useEffect } from 'react';
2
+ import { Box, Grid, FormGroup, FormControlLabel, Checkbox, FormControl, FormLabel, RadioGroup, TextField, Radio } from '@material-ui/core';
3
+ import { useStyles, useQetaApi } 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
+
10
+ const radioSelect = (value, label) => {
11
+ return /* @__PURE__ */ React.createElement(
12
+ FormControlLabel,
13
+ {
14
+ value,
15
+ control: /* @__PURE__ */ React.createElement(Radio, { size: "small" }),
16
+ label
17
+ }
18
+ );
19
+ };
20
+ const filterKeys = [
21
+ "orderBy",
22
+ "order",
23
+ "noAnswers",
24
+ "noCorrectAnswer",
25
+ "noVotes",
26
+ "entity",
27
+ "tags"
28
+ ];
29
+ const FilterPanel = (props) => {
30
+ const {
31
+ onChange,
32
+ filters,
33
+ showEntityFilter = true,
34
+ showTagFilter = true
35
+ } = props;
36
+ const styles = useStyles();
37
+ const { value: refs } = useQetaApi((api) => api.getEntities(), []);
38
+ const { value: tags } = useQetaApi((api) => api.getTags(), []);
39
+ const catalogApi = useApi(catalogApiRef);
40
+ const [availableEntities, setAvailableEntities] = React.useState(null);
41
+ const [selectedEntity, setSelectedEntity] = React.useState(void 0);
42
+ const [availableTags, setAvailableTags] = React.useState(
43
+ null
44
+ );
45
+ useEffect(() => {
46
+ if (tags && tags.length > 0 || filters.tags) {
47
+ const ts = (tags != null ? tags : []).map((t) => t.tag);
48
+ if (filters.tags) {
49
+ ts.push(...filters.tags);
50
+ }
51
+ setAvailableTags([...new Set(ts)]);
52
+ }
53
+ }, [tags, filters.tags]);
54
+ useEffect(() => {
55
+ const entityRefs = [];
56
+ if (filters.entity && !Array.isArray(filters.entity)) {
57
+ entityRefs.push(filters.entity);
58
+ }
59
+ if (refs && (refs == null ? void 0 : refs.length) > 0) {
60
+ refs == null ? void 0 : refs.forEach((ref) => {
61
+ if (ref.entityRef !== filters.entity) {
62
+ entityRefs.push(ref.entityRef);
63
+ }
64
+ });
65
+ }
66
+ if (entityRefs.length > 0) {
67
+ catalogApi.getEntitiesByRefs({
68
+ entityRefs,
69
+ fields: [
70
+ "kind",
71
+ "metadata.name",
72
+ "metadata.namespace",
73
+ "metadata.title"
74
+ ]
75
+ }).then((resp) => {
76
+ const filtered = resp.items.filter((i) => i !== void 0);
77
+ setAvailableEntities(filtered);
78
+ });
79
+ }
80
+ }, [filters.entity, catalogApi, refs]);
81
+ useEffect(() => {
82
+ if (filters.entity && availableEntities) {
83
+ const value = availableEntities.find(
84
+ (e) => stringifyEntityRef(e) === filters.entity
85
+ );
86
+ setSelectedEntity(value);
87
+ if (!value) {
88
+ onChange("entity", "");
89
+ }
90
+ } else {
91
+ setSelectedEntity(void 0);
92
+ }
93
+ }, [availableEntities, filters.entity, onChange]);
94
+ const handleChange = (event) => {
95
+ let value = event.target.value;
96
+ if (event.target.type === "checkbox") {
97
+ value = event.target.checked ? "true" : "false";
98
+ }
99
+ onChange(event.target.name, value);
100
+ };
101
+ 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, /* @__PURE__ */ React.createElement(
102
+ FormControlLabel,
103
+ {
104
+ control: /* @__PURE__ */ React.createElement(
105
+ Checkbox,
106
+ {
107
+ size: "small",
108
+ name: "noAnswers",
109
+ onChange: handleChange,
110
+ checked: filters.noAnswers === "true"
111
+ }
112
+ ),
113
+ label: "No answers"
114
+ }
115
+ ), /* @__PURE__ */ React.createElement(
116
+ FormControlLabel,
117
+ {
118
+ control: /* @__PURE__ */ React.createElement(
119
+ Checkbox,
120
+ {
121
+ size: "small",
122
+ name: "noCorrectAnswer",
123
+ checked: filters.noCorrectAnswer === "true",
124
+ onChange: handleChange
125
+ }
126
+ ),
127
+ label: "No correct answers"
128
+ }
129
+ ), /* @__PURE__ */ React.createElement(
130
+ FormControlLabel,
131
+ {
132
+ control: /* @__PURE__ */ React.createElement(
133
+ Checkbox,
134
+ {
135
+ size: "small",
136
+ name: "noVotes",
137
+ checked: filters.noVotes === "true",
138
+ onChange: handleChange
139
+ }
140
+ ),
141
+ label: "No votes"
142
+ }
143
+ ))), /* @__PURE__ */ React.createElement(Grid, { item: true, md: 2, xs: 4 }, /* @__PURE__ */ React.createElement(FormControl, null, /* @__PURE__ */ React.createElement(FormLabel, { id: "qeta-filter-order-by" }, "Order by"), /* @__PURE__ */ React.createElement(
144
+ RadioGroup,
145
+ {
146
+ "aria-labelledby": "qeta-filter-order-by",
147
+ name: "orderBy",
148
+ value: filters.orderBy,
149
+ onChange: handleChange
150
+ },
151
+ radioSelect("created", "Created"),
152
+ radioSelect("views", "Views"),
153
+ radioSelect("score", "Score"),
154
+ radioSelect("answersCount", "Answers")
155
+ ))), /* @__PURE__ */ React.createElement(Grid, { item: true, md: 2, xs: 4 }, /* @__PURE__ */ React.createElement(FormControl, null, /* @__PURE__ */ React.createElement(FormLabel, { id: "qeta-filter-order" }, "Order"), /* @__PURE__ */ React.createElement(
156
+ RadioGroup,
157
+ {
158
+ "aria-labelledby": "qeta-filter-order",
159
+ name: "order",
160
+ value: filters.order,
161
+ onChange: handleChange
162
+ },
163
+ radioSelect("desc", "Descending"),
164
+ radioSelect("asc", "Ascending")
165
+ ))), (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" }, "Filters"), showEntityFilter && availableEntities && availableEntities.length > 0 && /* @__PURE__ */ React.createElement(
166
+ Autocomplete,
167
+ {
168
+ multiple: false,
169
+ className: "qetaEntityFilter",
170
+ value: selectedEntity != null ? selectedEntity : null,
171
+ id: "entities-select",
172
+ options: availableEntities,
173
+ getOptionLabel: getEntityTitle,
174
+ getOptionSelected: (o, v) => {
175
+ if (!o || !v) {
176
+ return false;
177
+ }
178
+ return stringifyEntityRef(o) === stringifyEntityRef(v);
179
+ },
180
+ onChange: (_e, newValue) => {
181
+ handleChange({
182
+ target: {
183
+ name: "entity",
184
+ value: newValue ? stringifyEntityRef(newValue) : ""
185
+ }
186
+ });
187
+ },
188
+ renderInput: (params) => /* @__PURE__ */ React.createElement(
189
+ TextField,
190
+ {
191
+ ...params,
192
+ variant: "outlined",
193
+ margin: "normal",
194
+ label: "Entity",
195
+ placeholder: "Type or select entity"
196
+ }
197
+ )
198
+ }
199
+ ), showTagFilter && availableTags && availableTags.length > 0 && /* @__PURE__ */ React.createElement(
200
+ Autocomplete,
201
+ {
202
+ multiple: true,
203
+ className: "qetaTagFilter",
204
+ value: filters.tags,
205
+ id: "tags-select",
206
+ options: availableTags,
207
+ onChange: (_e, newValue) => {
208
+ handleChange({
209
+ target: {
210
+ name: "tags",
211
+ value: newValue
212
+ }
213
+ });
214
+ },
215
+ renderInput: (params) => /* @__PURE__ */ React.createElement(
216
+ TextField,
217
+ {
218
+ ...params,
219
+ variant: "outlined",
220
+ margin: "normal",
221
+ label: "Tag",
222
+ placeholder: "Type or select tag"
223
+ }
224
+ )
225
+ }
226
+ ))));
227
+ };
228
+
229
+ export { FilterPanel, filterKeys };
230
+ //# sourceMappingURL=FilterPanel.esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FilterPanel.esm.js","sources":["../../../src/components/QuestionsContainer/FilterPanel.tsx"],"sourcesContent":["import React, { useEffect } from 'react';\nimport {\n Box,\n Checkbox,\n FormControl,\n FormControlLabel,\n FormGroup,\n FormLabel,\n Grid,\n Radio,\n RadioGroup,\n TextField,\n} from '@material-ui/core';\nimport { useQetaApi, useStyles } from '../../utils/hooks';\nimport { Autocomplete } from '@material-ui/lab';\nimport { Entity, stringifyEntityRef } from '@backstage/catalog-model';\nimport { useApi } from '@backstage/core-plugin-api';\nimport { catalogApiRef } from '@backstage/plugin-catalog-react';\nimport { getEntityTitle } from '../../utils/utils';\n\nconst radioSelect = (value: string, label: string) => {\n return (\n <FormControlLabel\n value={value}\n control={<Radio size=\"small\" />}\n label={label}\n />\n );\n};\n\nexport const filterKeys = [\n 'orderBy',\n 'order',\n 'noAnswers',\n 'noCorrectAnswer',\n 'noVotes',\n 'entity',\n 'tags',\n] as const;\nexport type FilterKey = (typeof filterKeys)[number];\n\nexport type Filters = {\n order: string;\n orderBy: string;\n noAnswers: string;\n noCorrectAnswer: string;\n noVotes: string;\n searchQuery: string;\n entity: string;\n tags: string[];\n};\n\nexport interface FilterPanelProps {\n onChange: (key: FilterKey, value: string | string[]) => void;\n filters: Filters;\n showEntityFilter?: boolean;\n showTagFilter?: boolean;\n}\n\nexport const FilterPanel = (props: FilterPanelProps) => {\n const {\n onChange,\n filters,\n showEntityFilter = true,\n showTagFilter = true,\n } = props;\n const styles = useStyles();\n const { value: refs } = useQetaApi(api => api.getEntities(), []);\n const { value: tags } = useQetaApi(api => api.getTags(), []);\n const catalogApi = useApi(catalogApiRef);\n const [availableEntities, setAvailableEntities] = React.useState<\n Entity[] | null\n >(null);\n const [selectedEntity, setSelectedEntity] = React.useState<\n Entity | undefined\n >(undefined);\n const [availableTags, setAvailableTags] = React.useState<string[] | null>(\n null,\n );\n\n useEffect(() => {\n if ((tags && tags.length > 0) || filters.tags) {\n const ts = (tags ?? []).map(t => t.tag);\n if (filters.tags) {\n ts.push(...filters.tags);\n }\n setAvailableTags([...new Set(ts)]);\n }\n }, [tags, filters.tags]);\n\n useEffect(() => {\n const entityRefs: string[] = [];\n if (filters.entity && !Array.isArray(filters.entity)) {\n entityRefs.push(filters.entity);\n }\n if (refs && refs?.length > 0) {\n refs?.forEach(ref => {\n // ignore currently selected entity if exist in refs\n if (ref.entityRef !== filters.entity) {\n entityRefs.push(ref.entityRef);\n }\n });\n }\n if (entityRefs.length > 0) {\n catalogApi\n .getEntitiesByRefs({\n entityRefs,\n fields: [\n 'kind',\n 'metadata.name',\n 'metadata.namespace',\n 'metadata.title',\n ],\n })\n .then(resp => {\n const filtered = resp.items.filter(i => i !== undefined) as Entity[];\n setAvailableEntities(filtered);\n });\n }\n }, [filters.entity, catalogApi, refs]);\n\n useEffect(() => {\n if (filters.entity && availableEntities) {\n const value = availableEntities.find(\n e => stringifyEntityRef(e) === filters.entity,\n );\n setSelectedEntity(value);\n if (!value) {\n onChange('entity', '');\n }\n } else {\n setSelectedEntity(undefined);\n }\n }, [availableEntities, filters.entity, onChange]);\n\n const handleChange = (event: {\n target: {\n value: string | string[];\n type?: string;\n name: string;\n checked?: boolean;\n };\n }) => {\n let value = event.target.value;\n if (event.target.type === 'checkbox') {\n value = event.target.checked ? 'true' : 'false';\n }\n onChange(event.target.name as FilterKey, value);\n };\n\n return (\n <Box className={`qetaFilterPanel ${styles.filterPanel}`}>\n <Grid container spacing={4}>\n <Grid item md={3} xs={4}>\n <FormGroup>\n <FormControlLabel\n control={\n <Checkbox\n size=\"small\"\n name=\"noAnswers\"\n onChange={handleChange}\n checked={filters.noAnswers === 'true'}\n />\n }\n label=\"No answers\"\n />\n <FormControlLabel\n control={\n <Checkbox\n size=\"small\"\n name=\"noCorrectAnswer\"\n checked={filters.noCorrectAnswer === 'true'}\n onChange={handleChange}\n />\n }\n label=\"No correct answers\"\n />\n <FormControlLabel\n control={\n <Checkbox\n size=\"small\"\n name=\"noVotes\"\n checked={filters.noVotes === 'true'}\n onChange={handleChange}\n />\n }\n label=\"No votes\"\n />\n </FormGroup>\n </Grid>\n <Grid item md={2} xs={4}>\n <FormControl>\n <FormLabel id=\"qeta-filter-order-by\">Order by</FormLabel>\n <RadioGroup\n aria-labelledby=\"qeta-filter-order-by\"\n name=\"orderBy\"\n value={filters.orderBy}\n onChange={handleChange}\n >\n {radioSelect('created', 'Created')}\n {radioSelect('views', 'Views')}\n {radioSelect('score', 'Score')}\n {radioSelect('answersCount', 'Answers')}\n </RadioGroup>\n </FormControl>\n </Grid>\n <Grid item md={2} xs={4}>\n <FormControl>\n <FormLabel id=\"qeta-filter-order\">Order</FormLabel>\n <RadioGroup\n aria-labelledby=\"qeta-filter-order\"\n name=\"order\"\n value={filters.order}\n onChange={handleChange}\n >\n {radioSelect('desc', 'Descending')}\n {radioSelect('asc', 'Ascending')}\n </RadioGroup>\n </FormControl>\n </Grid>\n {((availableEntities && availableEntities.length > 0) ||\n (availableTags && availableTags.length > 0)) &&\n (showEntityFilter || showTagFilter) && (\n <Grid item md={4} xs={8}>\n <FormLabel id=\"qeta-filter-entity\">Filters</FormLabel>\n {showEntityFilter &&\n availableEntities &&\n availableEntities.length > 0 && (\n <Autocomplete\n multiple={false}\n className=\"qetaEntityFilter\"\n value={selectedEntity ?? null}\n id=\"entities-select\"\n options={availableEntities}\n getOptionLabel={getEntityTitle}\n getOptionSelected={(o, v) => {\n if (!o || !v) {\n return false;\n }\n return stringifyEntityRef(o) === stringifyEntityRef(v);\n }}\n onChange={(_e, newValue) => {\n handleChange({\n target: {\n name: 'entity',\n value: newValue ? stringifyEntityRef(newValue) : '',\n },\n });\n }}\n renderInput={params => (\n <TextField\n {...params}\n variant=\"outlined\"\n margin=\"normal\"\n label=\"Entity\"\n placeholder=\"Type or select entity\"\n />\n )}\n />\n )}\n {showTagFilter && availableTags && availableTags.length > 0 && (\n <Autocomplete\n multiple\n className=\"qetaTagFilter\"\n value={filters.tags}\n id=\"tags-select\"\n options={availableTags}\n onChange={(_e, newValue) => {\n handleChange({\n target: {\n name: 'tags',\n value: newValue,\n },\n });\n }}\n renderInput={params => (\n <TextField\n {...params}\n variant=\"outlined\"\n margin=\"normal\"\n label=\"Tag\"\n placeholder=\"Type or select tag\"\n />\n )}\n />\n )}\n </Grid>\n )}\n </Grid>\n </Box>\n );\n};\n"],"names":[],"mappings":"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"}
@@ -0,0 +1,46 @@
1
+ import { Card, CardContent, Grid, Typography } from '@material-ui/core';
2
+ import { LinkButton } from '@backstage/core-components';
3
+ import HelpOutline from '@material-ui/icons/HelpOutline';
4
+ import React from 'react';
5
+ import { useRouteRef } from '@backstage/core-plugin-api';
6
+ import { askRouteRef } from '@drodil/backstage-plugin-qeta-react';
7
+ import { useEntityQueryParameter } from '../../utils/hooks.esm.js';
8
+
9
+ const NoQuestionsCard = (props) => {
10
+ const { showNoQuestionsBtn, entity, entityPage, tags } = props;
11
+ const askRoute = useRouteRef(askRouteRef);
12
+ const entityRef = useEntityQueryParameter(entity);
13
+ const queryParams = new URLSearchParams();
14
+ if (entityRef) {
15
+ queryParams.set("entity", entityRef);
16
+ }
17
+ if (entityPage) {
18
+ queryParams.set("entityPage", "true");
19
+ }
20
+ if (tags && tags.length > 0) {
21
+ queryParams.set("tags", tags.join(","));
22
+ }
23
+ return /* @__PURE__ */ React.createElement(Card, { style: { marginTop: "2rem" } }, /* @__PURE__ */ React.createElement(CardContent, null, /* @__PURE__ */ React.createElement(
24
+ Grid,
25
+ {
26
+ container: true,
27
+ justifyContent: "center",
28
+ alignItems: "center",
29
+ direction: "column"
30
+ },
31
+ /* @__PURE__ */ React.createElement(Grid, { item: true }, /* @__PURE__ */ React.createElement(Typography, { variant: "h6" }, "No questions found")),
32
+ showNoQuestionsBtn && /* @__PURE__ */ React.createElement(Grid, { item: true }, /* @__PURE__ */ React.createElement(
33
+ LinkButton,
34
+ {
35
+ to: entityRef || tags ? `${askRoute()}?${queryParams.toString()}` : `${askRoute()}`,
36
+ startIcon: /* @__PURE__ */ React.createElement(HelpOutline, null),
37
+ color: "primary",
38
+ variant: "outlined"
39
+ },
40
+ "Go ahead and ask one!"
41
+ ))
42
+ )));
43
+ };
44
+
45
+ export { NoQuestionsCard };
46
+ //# sourceMappingURL=NoQuestionsCard.esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"NoQuestionsCard.esm.js","sources":["../../../src/components/QuestionsContainer/NoQuestionsCard.tsx"],"sourcesContent":["import { Card, CardContent, Grid, Typography } from '@material-ui/core';\nimport { LinkButton } from '@backstage/core-components';\nimport HelpOutline from '@material-ui/icons/HelpOutline';\nimport React from 'react';\nimport { useRouteRef } from '@backstage/core-plugin-api';\nimport { askRouteRef } from '@drodil/backstage-plugin-qeta-react';\nimport { useEntityQueryParameter } from '../../utils/hooks';\n\nexport const NoQuestionsCard = (props: {\n showNoQuestionsBtn?: boolean;\n entity?: string;\n entityPage?: boolean;\n tags?: string[];\n}) => {\n const { showNoQuestionsBtn, entity, entityPage, tags } = props;\n const askRoute = useRouteRef(askRouteRef);\n const entityRef = useEntityQueryParameter(entity);\n\n const queryParams = new URLSearchParams();\n if (entityRef) {\n queryParams.set('entity', entityRef);\n }\n if (entityPage) {\n queryParams.set('entityPage', 'true');\n }\n if (tags && tags.length > 0) {\n queryParams.set('tags', tags.join(','));\n }\n\n return (\n <Card style={{ marginTop: '2rem' }}>\n <CardContent>\n <Grid\n container\n justifyContent=\"center\"\n alignItems=\"center\"\n direction=\"column\"\n >\n <Grid item>\n <Typography variant=\"h6\">No questions found</Typography>\n </Grid>\n {showNoQuestionsBtn && (\n <Grid item>\n <LinkButton\n to={\n entityRef || tags\n ? `${askRoute()}?${queryParams.toString()}`\n : `${askRoute()}`\n }\n startIcon={<HelpOutline />}\n color=\"primary\"\n variant=\"outlined\"\n >\n Go ahead and ask one!\n </LinkButton>\n </Grid>\n )}\n </Grid>\n </CardContent>\n </Card>\n );\n};\n"],"names":[],"mappings":";;;;;;;;AAQa,MAAA,eAAA,GAAkB,CAAC,KAK1B,KAAA;AACJ,EAAA,MAAM,EAAE,kBAAA,EAAoB,MAAQ,EAAA,UAAA,EAAY,MAAS,GAAA,KAAA,CAAA;AACzD,EAAM,MAAA,QAAA,GAAW,YAAY,WAAW,CAAA,CAAA;AACxC,EAAM,MAAA,SAAA,GAAY,wBAAwB,MAAM,CAAA,CAAA;AAEhD,EAAM,MAAA,WAAA,GAAc,IAAI,eAAgB,EAAA,CAAA;AACxC,EAAA,IAAI,SAAW,EAAA;AACb,IAAY,WAAA,CAAA,GAAA,CAAI,UAAU,SAAS,CAAA,CAAA;AAAA,GACrC;AACA,EAAA,IAAI,UAAY,EAAA;AACd,IAAY,WAAA,CAAA,GAAA,CAAI,cAAc,MAAM,CAAA,CAAA;AAAA,GACtC;AACA,EAAI,IAAA,IAAA,IAAQ,IAAK,CAAA,MAAA,GAAS,CAAG,EAAA;AAC3B,IAAA,WAAA,CAAY,GAAI,CAAA,MAAA,EAAQ,IAAK,CAAA,IAAA,CAAK,GAAG,CAAC,CAAA,CAAA;AAAA,GACxC;AAEA,EACE,uBAAA,KAAA,CAAA,aAAA,CAAC,QAAK,KAAO,EAAA,EAAE,WAAW,MAAO,EAAA,EAAA,sCAC9B,WACC,EAAA,IAAA,kBAAA,KAAA,CAAA,aAAA;AAAA,IAAC,IAAA;AAAA,IAAA;AAAA,MACC,SAAS,EAAA,IAAA;AAAA,MACT,cAAe,EAAA,QAAA;AAAA,MACf,UAAW,EAAA,QAAA;AAAA,MACX,SAAU,EAAA,QAAA;AAAA,KAAA;AAAA,oBAEV,KAAA,CAAA,aAAA,CAAC,QAAK,IAAI,EAAA,IAAA,EAAA,sCACP,UAAW,EAAA,EAAA,OAAA,EAAQ,IAAK,EAAA,EAAA,oBAAkB,CAC7C,CAAA;AAAA,IACC,kBACC,oBAAA,KAAA,CAAA,aAAA,CAAC,IAAK,EAAA,EAAA,IAAA,EAAI,IACR,EAAA,kBAAA,KAAA,CAAA,aAAA;AAAA,MAAC,UAAA;AAAA,MAAA;AAAA,QACC,EACE,EAAA,SAAA,IAAa,IACT,GAAA,CAAA,EAAG,QAAS,EAAC,CAAI,CAAA,EAAA,WAAA,CAAY,QAAS,EAAC,CACvC,CAAA,GAAA,CAAA,EAAG,UAAU,CAAA,CAAA;AAAA,QAEnB,SAAA,sCAAY,WAAY,EAAA,IAAA,CAAA;AAAA,QACxB,KAAM,EAAA,SAAA;AAAA,QACN,OAAQ,EAAA,UAAA;AAAA,OAAA;AAAA,MACT,uBAAA;AAAA,KAGH,CAAA;AAAA,GAGN,CACF,CAAA,CAAA;AAEJ;;;;"}
@@ -0,0 +1,102 @@
1
+ import { useStyles } from '../../utils/hooks.esm.js';
2
+ import { Progress, WarningPanel } from '@backstage/core-components';
3
+ import { Box, Grid, Divider, Tooltip, FormControl, Select, MenuItem } from '@material-ui/core';
4
+ import React, { useRef, useState, useEffect } from 'react';
5
+ import { QuestionListItem } from './QuestionListItem.esm.js';
6
+ import { Pagination } from '@material-ui/lab';
7
+ import { NoQuestionsCard } from './NoQuestionsCard.esm.js';
8
+
9
+ const QuestionList = (props) => {
10
+ const {
11
+ loading,
12
+ error,
13
+ response,
14
+ onPageChange,
15
+ entity,
16
+ page,
17
+ onPageSizeChange,
18
+ showNoQuestionsBtn = true,
19
+ entityPage,
20
+ tags
21
+ } = props;
22
+ const styles = useStyles();
23
+ const listRef = useRef(null);
24
+ const [initialLoad, setInitialLoad] = useState(true);
25
+ useEffect(() => {
26
+ if (!initialLoad) {
27
+ setInitialLoad(false);
28
+ }
29
+ }, [initialLoad, loading]);
30
+ const handlePageChange = (_event, value) => {
31
+ if (listRef.current) {
32
+ listRef.current.scrollIntoView();
33
+ }
34
+ onPageChange(value);
35
+ };
36
+ const handlePageSizeChange = (event) => {
37
+ if (listRef.current) {
38
+ listRef.current.scrollIntoView();
39
+ }
40
+ onPageSizeChange(Number.parseInt(event.target.value, 10));
41
+ };
42
+ if (loading && initialLoad) {
43
+ return /* @__PURE__ */ React.createElement(Progress, null);
44
+ }
45
+ if (error || response === void 0) {
46
+ return /* @__PURE__ */ React.createElement(WarningPanel, { severity: "error", title: "Could not load questions." }, error == null ? void 0 : error.message);
47
+ }
48
+ if (initialLoad && (!response.questions || response.questions.length === 0)) {
49
+ return /* @__PURE__ */ React.createElement(
50
+ NoQuestionsCard,
51
+ {
52
+ showNoQuestionsBtn,
53
+ entity,
54
+ entityPage,
55
+ tags
56
+ }
57
+ );
58
+ }
59
+ const pageCount = response.total < props.pageSize ? 1 : Math.ceil(response.total / props.pageSize);
60
+ return /* @__PURE__ */ React.createElement("div", { ref: listRef }, /* @__PURE__ */ React.createElement(Box, { sx: { mt: 2 }, className: "qetaQuestionList" }, /* @__PURE__ */ React.createElement(Grid, { container: true, spacing: 2, className: "qetaQuestionListGrid" }, response.questions.map((question) => {
61
+ return /* @__PURE__ */ React.createElement(Grid, { item: true, xs: 12, key: question.id }, /* @__PURE__ */ React.createElement(QuestionListItem, { question, entity }), /* @__PURE__ */ React.createElement(Divider, null));
62
+ })), /* @__PURE__ */ React.createElement(
63
+ Grid,
64
+ {
65
+ container: true,
66
+ spacing: 0,
67
+ className: `qetaQuestionListPaginationGrid ${styles.questionListPagination}`,
68
+ alignItems: "center",
69
+ justifyContent: "space-between"
70
+ },
71
+ /* @__PURE__ */ React.createElement(Tooltip, { title: "Questions per page", arrow: true }, /* @__PURE__ */ React.createElement(FormControl, { variant: "filled" }, /* @__PURE__ */ React.createElement(
72
+ Select,
73
+ {
74
+ value: props.pageSize,
75
+ onChange: handlePageSizeChange,
76
+ className: `qetaQuestionListPaginationSizeSelect ${styles.questionsPerPage}`,
77
+ inputProps: { className: styles.questionsPerPageInput }
78
+ },
79
+ /* @__PURE__ */ React.createElement(MenuItem, { value: 5 }, "5"),
80
+ /* @__PURE__ */ React.createElement(MenuItem, { value: 10 }, "10"),
81
+ /* @__PURE__ */ React.createElement(MenuItem, { value: 25 }, "25"),
82
+ /* @__PURE__ */ React.createElement(MenuItem, { value: 50 }, "50"),
83
+ /* @__PURE__ */ React.createElement(MenuItem, { value: 100 }, "100")
84
+ ))),
85
+ /* @__PURE__ */ React.createElement(
86
+ Pagination,
87
+ {
88
+ page,
89
+ onChange: handlePageChange,
90
+ count: pageCount,
91
+ size: "large",
92
+ variant: "outlined",
93
+ className: "qetaQuestionListPagination",
94
+ showFirstButton: true,
95
+ showLastButton: true
96
+ }
97
+ )
98
+ )));
99
+ };
100
+
101
+ export { QuestionList };
102
+ //# sourceMappingURL=QuestionList.esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"QuestionList.esm.js","sources":["../../../src/components/QuestionsContainer/QuestionList.tsx"],"sourcesContent":["import { useStyles } from '../../utils/hooks';\nimport { Progress, WarningPanel } from '@backstage/core-components';\nimport {\n Box,\n Divider,\n FormControl,\n Grid,\n MenuItem,\n Select,\n Tooltip,\n} from '@material-ui/core';\nimport React, { useEffect, useRef, useState } from 'react';\nimport { QuestionListItem } from './QuestionListItem';\nimport { Pagination } from '@material-ui/lab';\nimport { QuestionsResponse } from '@drodil/backstage-plugin-qeta-common';\nimport { NoQuestionsCard } from './NoQuestionsCard';\n\nexport const QuestionList = (props: {\n loading: boolean;\n error: any;\n response?: QuestionsResponse;\n onPageChange: (page: number) => void;\n onPageSizeChange: (size: number) => void;\n page: number;\n pageSize: number;\n entity?: string;\n tags?: string[];\n showNoQuestionsBtn?: boolean;\n entityPage?: boolean;\n}) => {\n const {\n loading,\n error,\n response,\n onPageChange,\n entity,\n page,\n onPageSizeChange,\n showNoQuestionsBtn = true,\n entityPage,\n tags,\n } = props;\n const styles = useStyles();\n const listRef = useRef<HTMLDivElement | null>(null);\n const [initialLoad, setInitialLoad] = useState(true);\n\n useEffect(() => {\n if (!initialLoad) {\n setInitialLoad(false);\n }\n }, [initialLoad, loading]);\n\n const handlePageChange = (\n _event: React.ChangeEvent<unknown>,\n value: number,\n ) => {\n if (listRef.current) {\n listRef.current.scrollIntoView();\n }\n onPageChange(value);\n };\n\n const handlePageSizeChange = (\n event: React.ChangeEvent<{ value: unknown }>,\n ) => {\n if (listRef.current) {\n listRef.current.scrollIntoView();\n }\n onPageSizeChange(Number.parseInt(event.target.value as string, 10));\n };\n\n if (loading && initialLoad) {\n return <Progress />;\n }\n\n if (error || response === undefined) {\n return (\n <WarningPanel severity=\"error\" title=\"Could not load questions.\">\n {error?.message}\n </WarningPanel>\n );\n }\n\n if (initialLoad && (!response.questions || response.questions.length === 0)) {\n return (\n <NoQuestionsCard\n showNoQuestionsBtn={showNoQuestionsBtn}\n entity={entity}\n entityPage={entityPage}\n tags={tags}\n />\n );\n }\n\n const pageCount =\n response.total < props.pageSize\n ? 1\n : Math.ceil(response.total / props.pageSize);\n\n return (\n <div ref={listRef}>\n <Box sx={{ mt: 2 }} className=\"qetaQuestionList\">\n <Grid container spacing={2} className=\"qetaQuestionListGrid\">\n {response.questions.map(question => {\n return (\n <Grid item xs={12} key={question.id}>\n <QuestionListItem question={question} entity={entity} />\n <Divider />\n </Grid>\n );\n })}\n </Grid>\n <Grid\n container\n spacing={0}\n className={`qetaQuestionListPaginationGrid ${styles.questionListPagination}`}\n alignItems=\"center\"\n justifyContent=\"space-between\"\n >\n <Tooltip title=\"Questions per page\" arrow>\n <FormControl variant=\"filled\">\n <Select\n value={props.pageSize}\n onChange={handlePageSizeChange}\n className={`qetaQuestionListPaginationSizeSelect ${styles.questionsPerPage}`}\n inputProps={{ className: styles.questionsPerPageInput }}\n >\n <MenuItem value={5}>5</MenuItem>\n <MenuItem value={10}>10</MenuItem>\n <MenuItem value={25}>25</MenuItem>\n <MenuItem value={50}>50</MenuItem>\n <MenuItem value={100}>100</MenuItem>\n </Select>\n </FormControl>\n </Tooltip>\n <Pagination\n page={page}\n onChange={handlePageChange}\n count={pageCount}\n size=\"large\"\n variant=\"outlined\"\n className=\"qetaQuestionListPagination\"\n showFirstButton\n showLastButton\n />\n </Grid>\n </Box>\n </div>\n );\n};\n"],"names":[],"mappings":"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"}
@@ -0,0 +1,119 @@
1
+ import { useTheme, Card, CardContent, Box, Typography, Avatar } from '@material-ui/core';
2
+ import { Link } from '@backstage/core-components';
3
+ import React, { useState, useEffect } from 'react';
4
+ import DOMPurify from 'dompurify';
5
+ import { truncate, removeMarkdownFormatting } from '../../utils/utils.esm.js';
6
+ import { TagsAndEntities } from '../QuestionPage/TagsAndEntities.esm.js';
7
+ import { useRouteRef } from '@backstage/core-plugin-api';
8
+ import { questionRouteRef, userRouteRef } from '@drodil/backstage-plugin-qeta-react';
9
+ import { RelativeTimeWithTooltip } from '../RelativeTimeWithTooltip/RelativeTimeWithTooltip.esm.js';
10
+ import { useStyles, useEntityAuthor } from '../../utils/hooks.esm.js';
11
+ import { useSignal } from '@backstage/plugin-signals-react';
12
+
13
+ const QuestionListItem = (props) => {
14
+ var _a, _b;
15
+ const { question, entity } = props;
16
+ const [correctAnswer, setCorrectAnswer] = useState(question.correctAnswer);
17
+ const [answersCount, setAnswersCount] = useState(question.answersCount);
18
+ const [score, setScore] = useState(question.score);
19
+ const [views, setViews] = useState(question.views);
20
+ const { lastSignal } = useSignal(`qeta:question_${question.id}`);
21
+ useEffect(() => {
22
+ if ((lastSignal == null ? void 0 : lastSignal.type) === "question_stats") {
23
+ setCorrectAnswer(lastSignal.correctAnswer);
24
+ setAnswersCount(lastSignal.answersCount);
25
+ setScore(lastSignal.score);
26
+ setViews(lastSignal.views);
27
+ }
28
+ }, [lastSignal]);
29
+ const questionRoute = useRouteRef(questionRouteRef);
30
+ const userRoute = useRouteRef(userRouteRef);
31
+ const theme = useTheme();
32
+ const styles = useStyles();
33
+ const { name, initials, user } = useEntityAuthor(question);
34
+ return /* @__PURE__ */ React.createElement(Card, { className: "qetaQuestionListItem" }, /* @__PURE__ */ React.createElement(CardContent, null, /* @__PURE__ */ React.createElement(Box, { className: styles.questionListItemStats }, /* @__PURE__ */ React.createElement(
35
+ Typography,
36
+ {
37
+ display: "block",
38
+ variant: "caption",
39
+ className: "qetaQuestionListItemScore"
40
+ },
41
+ score,
42
+ " score"
43
+ ), /* @__PURE__ */ React.createElement(
44
+ Typography,
45
+ {
46
+ variant: "caption",
47
+ display: "block",
48
+ className: `qetaQuestionListItemAnswers ${correctAnswer ? "qetaQuestionListItemCorrectAnswer" : "quetaQuestionListItemNoCorrectAnswer"}`,
49
+ style: {
50
+ color: correctAnswer ? theme.palette.success.main : void 0
51
+ }
52
+ },
53
+ answersCount,
54
+ " answers"
55
+ ), /* @__PURE__ */ React.createElement(
56
+ Typography,
57
+ {
58
+ display: "block",
59
+ variant: "caption",
60
+ className: "qetaQuestionListItemViews"
61
+ },
62
+ views,
63
+ " views"
64
+ )), /* @__PURE__ */ React.createElement(Box, { className: styles.questionListItemContent }, /* @__PURE__ */ React.createElement(Typography, { variant: "h5", component: "div" }, /* @__PURE__ */ React.createElement(
65
+ Link,
66
+ {
67
+ to: entity ? `${questionRoute({
68
+ id: question.id.toString(10)
69
+ })}?entity=${entity}` : questionRoute({ id: question.id.toString(10) }),
70
+ className: "qetaQuestionListItemQuestionBtn"
71
+ },
72
+ question.title
73
+ )), /* @__PURE__ */ React.createElement(
74
+ Typography,
75
+ {
76
+ variant: "caption",
77
+ noWrap: true,
78
+ component: "div",
79
+ className: "qetaQuestionListItemContent",
80
+ style: { marginBottom: "5px" }
81
+ },
82
+ DOMPurify.sanitize(
83
+ truncate(removeMarkdownFormatting(question.content), 150)
84
+ )
85
+ ), /* @__PURE__ */ React.createElement(TagsAndEntities, { question }), /* @__PURE__ */ React.createElement(
86
+ Typography,
87
+ {
88
+ variant: "caption",
89
+ display: "inline",
90
+ className: `${styles.questionListItemAuthor} qetaQuestionListItemAuthor`
91
+ },
92
+ /* @__PURE__ */ React.createElement(
93
+ Avatar,
94
+ {
95
+ src: (_b = (_a = user == null ? void 0 : user.spec) == null ? void 0 : _a.profile) == null ? void 0 : _b.picture,
96
+ className: styles.questionListItemAvatar,
97
+ alt: name,
98
+ variant: "rounded"
99
+ },
100
+ initials
101
+ ),
102
+ question.author === "anonymous" ? "Anonymous" : /* @__PURE__ */ React.createElement(Link, { to: `${userRoute()}/${question.author}` }, name),
103
+ " ",
104
+ /* @__PURE__ */ React.createElement(
105
+ Link,
106
+ {
107
+ to: entity ? `${questionRoute({
108
+ id: question.id.toString(10)
109
+ })}?entity=${entity}` : questionRoute({ id: question.id.toString(10) }),
110
+ className: "qetaQuestionListItemQuestionBtn"
111
+ },
112
+ "asked ",
113
+ /* @__PURE__ */ React.createElement(RelativeTimeWithTooltip, { value: question.created })
114
+ )
115
+ ))));
116
+ };
117
+
118
+ export { QuestionListItem };
119
+ //# sourceMappingURL=QuestionListItem.esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"QuestionListItem.esm.js","sources":["../../../src/components/QuestionsContainer/QuestionListItem.tsx"],"sourcesContent":["import {\n Avatar,\n Box,\n Card,\n CardContent,\n Typography,\n useTheme,\n} from '@material-ui/core';\nimport { Link } from '@backstage/core-components';\nimport React, { useEffect, useState } from 'react';\nimport DOMPurify from 'dompurify';\nimport { removeMarkdownFormatting, truncate } from '../../utils/utils';\nimport { TagsAndEntities } from '../QuestionPage/TagsAndEntities';\nimport { useRouteRef } from '@backstage/core-plugin-api';\nimport {\n questionRouteRef,\n userRouteRef,\n} from '@drodil/backstage-plugin-qeta-react';\nimport { RelativeTimeWithTooltip } from '../RelativeTimeWithTooltip/RelativeTimeWithTooltip';\nimport {\n QetaSignal,\n QuestionResponse,\n} from '@drodil/backstage-plugin-qeta-common';\nimport { useEntityAuthor, useStyles } from '../../utils/hooks';\nimport { useSignal } from '@backstage/plugin-signals-react';\n\nexport interface QuestionListItemProps {\n question: QuestionResponse;\n entity?: string;\n}\n\nexport const QuestionListItem = (props: QuestionListItemProps) => {\n const { question, entity } = props;\n\n const [correctAnswer, setCorrectAnswer] = useState(question.correctAnswer);\n const [answersCount, setAnswersCount] = useState(question.answersCount);\n const [score, setScore] = useState(question.score);\n const [views, setViews] = useState(question.views);\n\n const { lastSignal } = useSignal<QetaSignal>(`qeta:question_${question.id}`);\n\n useEffect(() => {\n if (lastSignal?.type === 'question_stats') {\n setCorrectAnswer(lastSignal.correctAnswer);\n setAnswersCount(lastSignal.answersCount);\n setScore(lastSignal.score);\n setViews(lastSignal.views);\n }\n }, [lastSignal]);\n\n const questionRoute = useRouteRef(questionRouteRef);\n const userRoute = useRouteRef(userRouteRef);\n const theme = useTheme();\n const styles = useStyles();\n const { name, initials, user } = useEntityAuthor(question);\n\n return (\n <Card className=\"qetaQuestionListItem\">\n <CardContent>\n <Box className={styles.questionListItemStats}>\n <Typography\n display=\"block\"\n variant=\"caption\"\n className=\"qetaQuestionListItemScore\"\n >\n {score} score\n </Typography>\n <Typography\n variant=\"caption\"\n display=\"block\"\n className={`qetaQuestionListItemAnswers ${\n correctAnswer\n ? 'qetaQuestionListItemCorrectAnswer'\n : 'quetaQuestionListItemNoCorrectAnswer'\n }`}\n style={{\n color: correctAnswer ? theme.palette.success.main : undefined,\n }}\n >\n {answersCount} answers\n </Typography>\n <Typography\n display=\"block\"\n variant=\"caption\"\n className=\"qetaQuestionListItemViews\"\n >\n {views} views\n </Typography>\n </Box>\n <Box className={styles.questionListItemContent}>\n <Typography variant=\"h5\" component=\"div\">\n <Link\n to={\n entity\n ? `${questionRoute({\n id: question.id.toString(10),\n })}?entity=${entity}`\n : questionRoute({ id: question.id.toString(10) })\n }\n className=\"qetaQuestionListItemQuestionBtn\"\n >\n {question.title}\n </Link>\n </Typography>\n <Typography\n variant=\"caption\"\n noWrap\n component=\"div\"\n className=\"qetaQuestionListItemContent\"\n style={{ marginBottom: '5px' }}\n >\n {DOMPurify.sanitize(\n truncate(removeMarkdownFormatting(question.content), 150),\n )}\n </Typography>\n <TagsAndEntities question={question} />\n <Typography\n variant=\"caption\"\n display=\"inline\"\n className={`${styles.questionListItemAuthor} qetaQuestionListItemAuthor`}\n >\n <Avatar\n src={user?.spec?.profile?.picture}\n className={styles.questionListItemAvatar}\n alt={name}\n variant=\"rounded\"\n >\n {initials}\n </Avatar>\n {question.author === 'anonymous' ? (\n 'Anonymous'\n ) : (\n <Link to={`${userRoute()}/${question.author}`}>{name}</Link>\n )}{' '}\n <Link\n to={\n entity\n ? `${questionRoute({\n id: question.id.toString(10),\n })}?entity=${entity}`\n : questionRoute({ id: question.id.toString(10) })\n }\n className=\"qetaQuestionListItemQuestionBtn\"\n >\n {'asked '}\n <RelativeTimeWithTooltip value={question.created} />\n </Link>\n </Typography>\n </Box>\n </CardContent>\n </Card>\n );\n};\n"],"names":[],"mappings":"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"}