diff --git a/plugins/qeta-backend/src/service/CatalogMockRouter.ts b/plugins/qeta-backend/src/service/CatalogMockRouter.ts index f69e2960..a852acad 100644 --- a/plugins/qeta-backend/src/service/CatalogMockRouter.ts +++ b/plugins/qeta-backend/src/service/CatalogMockRouter.ts @@ -42,6 +42,10 @@ const entitiesResponse = [ }, ]; +const images: Record = { + thor: '', +}; + export const createCatalogMockRouter = async (): Promise => { const router = Router(); router.use(express.json()); @@ -67,7 +71,12 @@ export const createCatalogMockRouter = async (): Promise => { .map(a => a.charAt(0).toUpperCase() + a.slice(1)) .join(' '), }, - spec: {}, + spec: { + profile: { + picture: + req.params.name in images ? images[req.params.name] : undefined, + }, + }, }); return; } diff --git a/plugins/qeta/src/components/QuestionPage/AuthorBox.tsx b/plugins/qeta/src/components/QuestionPage/AuthorBox.tsx index 66f1d325..c6d3c009 100644 --- a/plugins/qeta/src/components/QuestionPage/AuthorBox.tsx +++ b/plugins/qeta/src/components/QuestionPage/AuthorBox.tsx @@ -1,14 +1,10 @@ import { Avatar, Box, Grid, Typography } from '@material-ui/core'; -import { formatEntityName } from '../../utils/utils'; -import React, { useEffect } from 'react'; -import { useStyles } from '../../utils/hooks'; +import React from 'react'; +import { useEntityAuthor, useStyles } from '../../utils/hooks'; import { AnswerResponse, QuestionResponse, } from '@drodil/backstage-plugin-qeta-common'; -import { identityApiRef, useApi } from '@backstage/core-plugin-api'; -import { catalogApiRef } from '@backstage/plugin-catalog-react'; -import { UserEntity } from '@backstage/catalog-model'; import { RelativeTimeWithTooltip } from '../RelativeTimeWithTooltip/RelativeTimeWithTooltip'; import { AuthorLink, UpdatedByLink } from '../Links/Links'; @@ -16,45 +12,8 @@ export const AuthorBox = (props: { entity: QuestionResponse | AnswerResponse; }) => { const { entity } = props; - const catalogApi = useApi(catalogApiRef); - const identityApi = useApi(identityApiRef); - const [user, setUser] = React.useState(null); - const [currentUser, setCurrentUser] = React.useState(null); const styles = useStyles(); - const anonymous = entity.anonymous ?? false; - useEffect(() => { - if (!anonymous) { - catalogApi - .getEntityByRef(entity.author) - .catch(_ => setUser(null)) - .then(data => (data ? setUser(data as UserEntity) : setUser(null))); - } - }, [catalogApi, entity, anonymous]); - - useEffect(() => { - identityApi.getBackstageIdentity().then(res => { - setCurrentUser(res.userEntityRef ?? 'user:default/guest'); - }); - }, [identityApi]); - - let name = formatEntityName(entity.author); - if (user && user.metadata.title) { - name = user.metadata.title; - } - - if (entity.author === currentUser) { - name = 'You'; - if (anonymous) { - name += ' (anonymous)'; - } - } - - const initials = (name ?? '') - .split(' ') - .map(p => p[0]) - .join('') - .substring(0, 2) - .toUpperCase(); + const { name, initials, user } = useEntityAuthor(entity); return ( diff --git a/plugins/qeta/src/components/QuestionsContainer/QuestionListItem.tsx b/plugins/qeta/src/components/QuestionsContainer/QuestionListItem.tsx index 04a4b240..31db8cd8 100644 --- a/plugins/qeta/src/components/QuestionsContainer/QuestionListItem.tsx +++ b/plugins/qeta/src/components/QuestionsContainer/QuestionListItem.tsx @@ -1,7 +1,8 @@ import { + Avatar, + Box, Card, CardContent, - Grid, Typography, useTheme, } from '@material-ui/core'; @@ -16,8 +17,8 @@ import { userRouteRef, } from '@drodil/backstage-plugin-qeta-react'; import { RelativeTimeWithTooltip } from '../RelativeTimeWithTooltip/RelativeTimeWithTooltip'; -import { useEntityPresentation } from '@backstage/plugin-catalog-react'; import { QuestionResponse } from '@drodil/backstage-plugin-qeta-common'; +import { useEntityAuthor, useStyles } from '../../utils/hooks'; export interface QuestionListItemProps { question: QuestionResponse; @@ -26,96 +27,108 @@ export interface QuestionListItemProps { export const QuestionListItem = (props: QuestionListItemProps) => { const { question, entity } = props; + const questionRoute = useRouteRef(questionRouteRef); const userRoute = useRouteRef(userRouteRef); - const { primaryTitle: userName } = useEntityPresentation(question.author); const theme = useTheme(); + const styles = useStyles(); + const { name, initials, user } = useEntityAuthor(question); return ( - - - - - {question.title} - - - - - - {DOMPurify.sanitize( - truncate(removeMarkdownFormatting(question.content), 150), - )} - - - - - + + {question.score} score + + + {question.answersCount} answers + + + {question.views} views + + + + + - By{' '} - {question.author === 'anonymous' ? ( - 'Anonymous' - ) : ( - {userName} - )}{' '} - - - - Score: {question.score} {' | '} - - + + + {DOMPurify.sanitize( + truncate(removeMarkdownFormatting(question.content), 150), + )} + + + + - Answers: {question.answersCount} - - + {question.author === 'anonymous' ? ( + 'Anonymous' + ) : ( + {name} + )}{' '} + - {' | '} Views: {question.views} - - - - - - + {'asked '} + + + + ); diff --git a/plugins/qeta/src/utils/hooks.ts b/plugins/qeta/src/utils/hooks.ts index 13c5e4c5..cbdb3bea 100644 --- a/plugins/qeta/src/utils/hooks.ts +++ b/plugins/qeta/src/utils/hooks.ts @@ -8,10 +8,18 @@ import { } from '@backstage/core-plugin-api'; import { makeStyles } from '@material-ui/core'; import { CatalogApi } from '@backstage/catalog-client'; -import { catalogApiRef } from '@backstage/plugin-catalog-react'; +import { + catalogApiRef, + useEntityPresentation, +} from '@backstage/plugin-catalog-react'; import { trimEnd } from 'lodash'; import { useSearchParams } from 'react-router-dom'; import React, { useEffect } from 'react'; +import { UserEntity } from '@backstage/catalog-model'; +import { + AnswerResponse, + QuestionResponse, +} from '@drodil/backstage-plugin-qeta-common'; export function useQetaApi( f: (api: QetaApi) => Promise, @@ -136,6 +144,29 @@ export const useStyles = makeStyles(theme => { width: 'calc(100% - 70px)', minHeight: '160px', }, + questionListItemStats: { + width: '80px', + textAlign: 'right', + marginRight: '26px', + display: 'inline-block', + verticalAlign: 'top', + paddingTop: '10px', + }, + questionListItemContent: { + display: 'inline-block', + width: 'calc(100% - 120px)', + }, + questionListItemAuthor: { + display: 'inline', + float: 'right', + }, + questionListItemAvatar: { + display: 'inline-flex', + marginRight: '0.25rem', + fontSize: '1rem', + maxWidth: '1rem', + maxHeight: '1rem', + }, answerCardContent: { display: 'inline-block', width: 'calc(100% - 70px)', @@ -251,3 +282,48 @@ export const useBasePath = () => { const { pathname } = new URL(url, base); return trimEnd(pathname, '/'); }; + +export const useEntityAuthor = (entity: QuestionResponse | AnswerResponse) => { + const catalogApi = useApi(catalogApiRef); + const identityApi = useApi(identityApiRef); + const [user, setUser] = React.useState(null); + const [initials, setInitials] = React.useState(null); + const [currentUser, setCurrentUser] = React.useState(null); + const anonymous = entity.anonymous ?? false; + useEffect(() => { + if (!anonymous) { + catalogApi + .getEntityByRef(entity.author) + .catch(_ => setUser(null)) + .then(data => (data ? setUser(data as UserEntity) : setUser(null))); + } + }, [catalogApi, entity, anonymous]); + + useEffect(() => { + identityApi.getBackstageIdentity().then(res => { + setCurrentUser(res.userEntityRef ?? 'user:default/guest'); + }); + }, [identityApi]); + + const { primaryTitle: userName } = useEntityPresentation(entity.author); + + let name = userName; + if (entity.author === currentUser) { + name = 'You'; + if (anonymous) { + name += ' (anonymous)'; + } + } + + useEffect(() => { + const init = (name ?? '') + .split(' ') + .map(p => p[0]) + .join('') + .substring(0, 2) + .toUpperCase(); + setInitials(init); + }, [name]); + + return { name, initials, user }; +};