diff --git a/services/backend/src/routes/changelog.ts b/services/backend/src/routes/changelog.ts index a48ef1dda0..01b268e8eb 100644 --- a/services/backend/src/routes/changelog.ts +++ b/services/backend/src/routes/changelog.ts @@ -2,39 +2,44 @@ import axios from 'axios' import { Request, Response, Router } from 'express' import { isDev } from '../config' +import { Release } from '../shared/types' const router = Router() -type ChangeLogData = { - description: string - time: Date - title: string -} - -const changelog: { data?: ChangeLogData[] } = {} +const changelog: { data?: Release[] } = {} router.get('/', async (_req: Request, res: Response) => { if (changelog.data) { return res.status(200).send(changelog.data) } if (isDev) { - const fakeChangeLogData: ChangeLogData[] = [ + const fakeRelease: Release[] = [ + { + description: '**Feature 1**\n- Added a fancy new feature \n\n**Feature 2**\n- Fixed a bug\n- Fixed another bug', + title: 'Release 1', + time: new Date().toISOString(), + }, + { + description: "Let's not spam the GitHub API in development!", + title: 'Release 2', + time: new Date().toISOString(), + }, { - description: "### Fake release\nLet's not spam the GitHub API in development!", - title: 'Fake title for fake release', - time: new Date(), + description: 'This release should not be visible on the frontpage', + title: 'Release 3', + time: new Date().toISOString(), }, ] - return res.status(200).json(fakeChangeLogData) + return res.status(200).json(fakeRelease) } const response = await axios.get('https://api.github.com/repos/UniversityOfHelsinkiCS/oodikone/releases') - const newChangelogData = response.data.map((release: Record) => ({ + const releasesFromAPI: Release[] = response.data.map((release: Record) => ({ description: release.body, time: release.created_at, title: release.name, })) - changelog.data = newChangelogData - res.status(200).send(newChangelogData) + changelog.data = releasesFromAPI + res.status(200).json(releasesFromAPI) }) export default router diff --git a/services/frontend/src/common/index.js b/services/frontend/src/common/index.js index 2fc3782fa1..bf5f5b5bbd 100644 --- a/services/frontend/src/common/index.js +++ b/services/frontend/src/common/index.js @@ -434,7 +434,10 @@ export const getEnrollmentTypeTextForExcel = (type, statutoryAbsence) => { } export const isDefaultServiceProvider = () => { - return serviceProvider && serviceProvider === 'toska' + if (!serviceProvider) { + return false + } + return serviceProvider === 'toska' } export const formatContent = content => content.replace(/\n +/g, '\n') @@ -447,3 +450,14 @@ export const getCalendarYears = years => { return all.concat(Number(year.slice(0, 4))) }, []) } + +export const filterInternalReleases = release => !release.title.startsWith('Internal:') + +export const getDescription = description => { + const lines = description.split('\n') + const internalIndex = lines.findIndex(line => line.toLowerCase().includes('internal')) + if (internalIndex === -1 || internalIndex === 0) { + return description + } + return lines.slice(0, internalIndex).join('\n') +} diff --git a/services/frontend/src/components/Frontpage/Changelog.jsx b/services/frontend/src/components/Frontpage/Changelog.jsx deleted file mode 100644 index 44e980fd38..0000000000 --- a/services/frontend/src/components/Frontpage/Changelog.jsx +++ /dev/null @@ -1,72 +0,0 @@ -import { useEffect, useState } from 'react' -import ReactMarkdown from 'react-markdown' -import { Divider, Header, Loader } from 'semantic-ui-react' - -import { builtAt } from '@/conf' -import { useGetChangelogQuery } from '@/redux/changelog' - -export const Changelog = ({ showFullChangelog }) => { - const [itemsToShow, setItemsToShow] = useState([]) - const { data, isLoading } = useGetChangelogQuery() - - const filterInternalReleases = release => !release.title.startsWith('Internal:') - - useEffect(() => { - if (!data) return - setItemsToShow( - showFullChangelog - ? [...data.filter(filterInternalReleases).slice(0, 20)] - : [...data.filter(filterInternalReleases).slice(0, 2)] - ) - }, [data, showFullChangelog]) - - const formatDate = dateString => { - const date = new Date(dateString) - const dateFormatter = new Intl.DateTimeFormat(undefined, { dateStyle: 'medium', timeStyle: 'medium' }) - return dateFormatter.format(date) - } - - /** - * Strips the internal part of the changelog description. It doesn't matter how - * the keyword is spelled in the changelog, but using **Internal** is recommended. - * - * @param {string} string - The description to parse. - * @returns string - The description without the internal part. - */ - const getDescription = string => { - const lines = string.split('\n') - const internalIndex = lines.findIndex(line => line.toLowerCase().includes('internal')) - if (internalIndex === -1 || internalIndex === 0) { - return string - } - return lines.slice(0, internalIndex).join('\n') - } - - const getReleaseString = release => { - const date = formatDate(release.time) - const description = getDescription(release.description) - const releaseString = showFullChangelog - ? `## ${release.title}\n${date}\n\n${description}` - : `#### ${release.title}\n${description}` - return releaseString - } - - if (isLoading || itemsToShow.length === 0) return - - return ( -
- {!showFullChangelog && ( - <> -
Updates
-

Last update on: {builtAt ? formatDate(builtAt) : formatDate(itemsToShow[0].time)}

- - )} - {itemsToShow.map(release => ( -
- - {getReleaseString(release)} -
- ))} -
- ) -} diff --git a/services/frontend/src/components/Frontpage/index.jsx b/services/frontend/src/components/Frontpage/index.jsx deleted file mode 100644 index 9cd72f0ef9..0000000000 --- a/services/frontend/src/components/Frontpage/index.jsx +++ /dev/null @@ -1,123 +0,0 @@ -import { Fragment, useState } from 'react' -import { Button, Container, Divider, Grid, Header, Image, Icon, List } from 'semantic-ui-react' -import { checkUserAccess, getFullStudyProgrammeRights, images, isDefaultServiceProvider } from '@/common' -import { useTitle } from '@/common/hooks' -import { useGetAuthorizedUserQuery } from '@/redux/auth' -import { Changelog } from './Changelog' - -const FrontPageItem = ({ title, content }) => { - return ( - <> -
{title}
- {content} - - ) -} - -export const FrontPage = () => { - const { roles, programmeRights } = useGetAuthorizedUserQuery() - const fullStudyProgrammeRights = getFullStudyProgrammeRights(programmeRights) - - const [showFullChangelog, setShowFullChangelog] = useState(false) - - useTitle() - - const items = [ - { - show: true, - title: 'University', - content:

View tables and diagrams about study progress of different faculties

, - }, - { - show: checkUserAccess(['admin', 'fullSisuAccess'], roles) || programmeRights.length > 0, - title: 'Programmes', - content: ( - - - Class statistics: View details of a specific year of a study programme - - - Overview: View statistics of a programme across all years - - - ), - }, - { - show: checkUserAccess(['courseStatistics', 'admin'], roles) || fullStudyProgrammeRights.length > 0, - title: 'Courses', - content:

View statistics about course attempts, completions and grades

, - }, - { - show: checkUserAccess(['studyGuidanceGroups', 'admin'], roles) || fullStudyProgrammeRights.length > 0, - title: 'Students', - content:

View detailed information for a given student

, - }, - { - show: isDefaultServiceProvider(), - title: 'Feedback', - content: ( -

- For questions and suggestions, please use the{' '} - feedback form or shoot an email to{' '} - grp-toska@helsinki.fi. -

- ), - }, - ] - - return ( -
- - {showFullChangelog ? ( - <> -
- Latest updates -
- - - - - ) : ( - <> -
- Oodikone -
-
- Exploratory Research on Study Data -
- - - - {items.map( - (item, index) => - item.show && ( - - - {index !== items.length - 1 ? : null} - - ) - )} - - - - - - - - - )} -
- - Logo of Toska - -
- ) -} diff --git a/services/frontend/src/components/Routes/index.jsx b/services/frontend/src/components/Routes/index.jsx index 328cc7a78a..43ef3f5f1b 100644 --- a/services/frontend/src/components/Routes/index.jsx +++ b/services/frontend/src/components/Routes/index.jsx @@ -10,7 +10,6 @@ import { CustomPopulation } from '@/components/CustomPopulation' import { EvaluationOverview } from '@/components/EvaluationOverview' import { UniversityViewPage } from '@/components/EvaluationOverview/UniversityView' import { FacultyStatistics } from '@/components/FacultyStatistics' -import { FrontPage } from '@/components/Frontpage' import { LanguageCenterView } from '@/components/LanguageCenterView' import { PopulationStatistics } from '@/components/PopulationStatistics' import { SegmentDimmer } from '@/components/SegmentDimmer' @@ -21,7 +20,9 @@ import { Teachers } from '@/components/Teachers' import { Updater } from '@/components/Updater' import { Users } from '@/components/Users' import { languageCenterViewEnabled } from '@/conf' +import { Changelog } from '@/pages/Changelog' import { Feedback } from '@/pages/Feedback' +import { FrontPage } from '@/pages/FrontPage' import { ProtectedRoute } from './ProtectedRoute' const routes = { @@ -43,12 +44,14 @@ const routes = { closeToGraduation: '/close-to-graduation', populations: '/populations', studyProgramme: '/study-programme/:studyProgrammeId?', + changelog: '/changelog', } export const Routes = () => ( }> + {isDefaultServiceProvider() && } { +export const PageTitle = ({ subtitle, title }: { subtitle?: string; title: string }) => { return ( - + {title} - + {subtitle && ( + + {subtitle} + + )} + ) } diff --git a/services/frontend/src/conf.js b/services/frontend/src/conf.js index 158eb5a41e..0ee4009369 100644 --- a/services/frontend/src/conf.js +++ b/services/frontend/src/conf.js @@ -13,9 +13,6 @@ export const runningInCypress = typeof window !== 'undefined' && !!window.Cypres export const basePath = process.env.PUBLIC_URL || '' export const apiBasePath = `${basePath}/api` -// Update time for frontpage -export const builtAt = process.env.REACT_APP_BUILT_AT || '' - // Service provider depending this hiding some not needed features default value toska export const serviceProvider = process.env.REACT_APP_SERVICE_PROVIDER ? process.env.REACT_APP_SERVICE_PROVIDER.toLowerCase() diff --git a/services/frontend/src/pages/Changelog/ReleaseCard.tsx b/services/frontend/src/pages/Changelog/ReleaseCard.tsx new file mode 100644 index 0000000000..740aa55c7d --- /dev/null +++ b/services/frontend/src/pages/Changelog/ReleaseCard.tsx @@ -0,0 +1,21 @@ +import { Card, Typography } from '@mui/material' +import ReactMarkdown from 'react-markdown' + +import { getDescription } from '@/common' +import { DISPLAY_DATE_FORMAT } from '@/constants/date' +import { Release } from '@/shared/types' +import { reformatDate } from '@/util/timeAndDate' + +export const ReleaseCard = ({ isLoading, release }: { isLoading: boolean; release: Release }) => { + return ( + + + {isLoading ? 'Loading title...' : release.title} + + + {isLoading ? 'Loading date...' : `Released on ${reformatDate(release.time, DISPLAY_DATE_FORMAT)}`} + + {isLoading ? 'Loading description...' : getDescription(release.description)} + + ) +} diff --git a/services/frontend/src/pages/Changelog/index.tsx b/services/frontend/src/pages/Changelog/index.tsx new file mode 100644 index 0000000000..e2de61f0ce --- /dev/null +++ b/services/frontend/src/pages/Changelog/index.tsx @@ -0,0 +1,34 @@ +import { Container, Divider, Stack } from '@mui/material' +import { useEffect, useState } from 'react' + +import { filterInternalReleases } from '@/common' +import { useTitle } from '@/common/hooks' +import { PageTitle } from '@/components/material/PageTitle' +import { useGetChangelogQuery } from '@/redux/changelog' +import { Release } from '@/shared/types' +import { ReleaseCard } from './ReleaseCard' + +export const Changelog = () => { + useTitle('Changelog') + + const { data: releaseData, isLoading } = useGetChangelogQuery() + const [visibleReleases, setVisibleReleases] = useState([]) + + useEffect(() => { + if (!releaseData) { + return + } + setVisibleReleases([...releaseData.filter(filterInternalReleases).slice(0, 10)]) + }, [releaseData]) + + return ( + + + } gap={1}> + {visibleReleases.map(release => ( + + ))} + + + ) +} diff --git a/services/frontend/src/pages/Feedback/index.tsx b/services/frontend/src/pages/Feedback/index.tsx index ac2276702c..6fa9799963 100644 --- a/services/frontend/src/pages/Feedback/index.tsx +++ b/services/frontend/src/pages/Feedback/index.tsx @@ -60,8 +60,8 @@ export const Feedback = () => { open={showError} severity="error" /> + - We are constantly improving Oodikone. Please share your thoughts using the form below, or contact us at
diff --git a/services/frontend/src/pages/FrontPage/FeatureItem.tsx b/services/frontend/src/pages/FrontPage/FeatureItem.tsx new file mode 100644 index 0000000000..758c0d70d6 --- /dev/null +++ b/services/frontend/src/pages/FrontPage/FeatureItem.tsx @@ -0,0 +1,14 @@ +import { Box, Typography } from '@mui/material' + +export const FeatureItem = ({ content, title }: { content: JSX.Element | string; title: string }) => { + return ( + + + {title} + + + {content} + + + ) +} diff --git a/services/frontend/src/pages/FrontPage/ReleaseItem.tsx b/services/frontend/src/pages/FrontPage/ReleaseItem.tsx new file mode 100644 index 0000000000..395aec0e0c --- /dev/null +++ b/services/frontend/src/pages/FrontPage/ReleaseItem.tsx @@ -0,0 +1,23 @@ +import { Box, Typography } from '@mui/material' +import ReactMarkdown from 'react-markdown' + +import { getDescription } from '@/common' +import { DISPLAY_DATE_FORMAT } from '@/constants/date' +import { Release } from '@/shared/types' +import { reformatDate } from '@/util/timeAndDate' + +export const ReleaseItem = ({ isLoading, release }: { isLoading: boolean; release: Release }) => { + return ( + + + {isLoading ? 'Loading title...' : release.title} + + + {isLoading ? 'Loading date...' : reformatDate(release.time, DISPLAY_DATE_FORMAT)} + + + {isLoading ? 'Loading description...' : getDescription(release.description)} + + + ) +} diff --git a/services/frontend/src/pages/FrontPage/SectionTitle.tsx b/services/frontend/src/pages/FrontPage/SectionTitle.tsx new file mode 100644 index 0000000000..6519d13b58 --- /dev/null +++ b/services/frontend/src/pages/FrontPage/SectionTitle.tsx @@ -0,0 +1,11 @@ +import { Box, Typography } from '@mui/material' + +export const SectionTitle = ({ title }: { title: string }) => { + return ( + + + {title} + + + ) +} diff --git a/services/frontend/src/pages/FrontPage/index.tsx b/services/frontend/src/pages/FrontPage/index.tsx new file mode 100644 index 0000000000..6373c359ed --- /dev/null +++ b/services/frontend/src/pages/FrontPage/index.tsx @@ -0,0 +1,103 @@ +import { Box, Button, Container, Divider, Stack } from '@mui/material' +import { useEffect, useState } from 'react' +import { Link } from 'react-router-dom' + +import { + checkUserAccess, + filterInternalReleases, + getFullStudyProgrammeRights, + isDefaultServiceProvider, +} from '@/common' +import { useTitle } from '@/common/hooks' +import { PageTitle } from '@/components/material/PageTitle' +import { useGetAuthorizedUserQuery } from '@/redux/auth' +import { useGetChangelogQuery } from '@/redux/changelog' +import { Release } from '@/shared/types' +import { FeatureItem } from './FeatureItem' +import { ReleaseItem } from './ReleaseItem' +import { SectionTitle } from './SectionTitle' + +export const FrontPage = () => { + useTitle() + + const { data: releaseData, isLoading } = useGetChangelogQuery() + const { roles, programmeRights } = useGetAuthorizedUserQuery() + const fullStudyProgrammeRights = getFullStudyProgrammeRights(programmeRights) + const [visibleReleases, setVisibleReleases] = useState([]) + + // TODO: Add missing features and extract access right checking + const featureItems = [ + { + show: true, + title: 'University', + content: 'View tables and diagrams about study progress of different faculties', + }, + { + show: checkUserAccess(['admin', 'fullSisuAccess'], roles) || programmeRights.length > 0, + title: 'Programmes', + content: ( +
    +
  • Class statistics: View details of a specific year of a study programme
  • +
  • Overview: View statistics of a programme across all years
  • +
+ ), + }, + { + show: checkUserAccess(['courseStatistics', 'admin'], roles) || fullStudyProgrammeRights.length > 0, + title: 'Courses', + content: 'View statistics about course attempts, completions and grades', + }, + { + show: checkUserAccess(['studyGuidanceGroups', 'admin'], roles) || fullStudyProgrammeRights.length > 0, + title: 'Students', + content: 'View detailed information for a given student', + }, + { + show: isDefaultServiceProvider(), + title: 'Feedback', + content: ( +

+ For questions and suggestions, please use the{' '} + feedback form or shoot an email to{' '} + grp-toska@helsinki.fi. +

+ ), + }, + ] + + useEffect(() => { + if (!releaseData) { + return + } + setVisibleReleases([...releaseData.filter(filterInternalReleases).slice(0, 2)]) + }, [releaseData]) + + return ( + + + } gap={3}> + + + } gap={2}> + {featureItems.map( + item => item.show && + )} + + + + + } gap={2}> + {visibleReleases.map(release => ( + + ))} + + + + + + + + ) +} diff --git a/services/frontend/src/redux/changelog.js b/services/frontend/src/redux/changelog.ts similarity index 70% rename from services/frontend/src/redux/changelog.js rename to services/frontend/src/redux/changelog.ts index bd726e5ce8..c2f7802e4c 100644 --- a/services/frontend/src/redux/changelog.js +++ b/services/frontend/src/redux/changelog.ts @@ -1,8 +1,9 @@ import { RTKApi } from '@/apiConnection' +import { Release } from '@/shared/types' const changelogApi = RTKApi.injectEndpoints({ endpoints: builder => ({ - getChangelog: builder.query({ + getChangelog: builder.query({ query: () => 'changelog', }), }), diff --git a/services/shared/types/index.ts b/services/shared/types/index.ts index 32b3dd8a22..80438c9cb0 100644 --- a/services/shared/types/index.ts +++ b/services/shared/types/index.ts @@ -1 +1,2 @@ export type { Name } from './name' +export type { Release } from './release' diff --git a/services/shared/types/release.ts b/services/shared/types/release.ts new file mode 100644 index 0000000000..2e34a7e520 --- /dev/null +++ b/services/shared/types/release.ts @@ -0,0 +1,5 @@ +export type Release = { + description: string + time: string + title: string +}