@@ -53,41 +54,3 @@ export function DatasetCitation({ thumbnail, version }: DatasetCitationProps) {
>
)
}
-
-function CitationDescription({
- citation,
- publishingStatus
-}: {
- citation: string
- publishingStatus: DatasetPublishingStatus
-}) {
- const citationAsReactElement = parse(citation)
-
- return (
-
- {citationAsReactElement}
-
-
- )
-}
-
-interface CitationDatasetStatusProps {
- status: DatasetPublishingStatus
-}
-
-function CitationTooltip({ status }: CitationDatasetStatusProps) {
- const { t } = useTranslation('dataset')
-
- if (status !== DatasetPublishingStatus.RELEASED) {
- return (
- <>
- {' '}
-
- >
- )
- }
- return <>>
-}
diff --git a/src/sections/dataset/dataset-citation/DatasetCitationTooltip.tsx b/src/sections/dataset/dataset-citation/DatasetCitationTooltip.tsx
new file mode 100644
index 000000000..0ae1676b1
--- /dev/null
+++ b/src/sections/dataset/dataset-citation/DatasetCitationTooltip.tsx
@@ -0,0 +1,24 @@
+import { DatasetPublishingStatus } from '../../../dataset/domain/models/Dataset'
+import { useTranslation } from 'react-i18next'
+import { QuestionMarkTooltip } from '@iqss/dataverse-design-system'
+
+interface DatasetCitationTooltipProps {
+ status: DatasetPublishingStatus
+}
+
+export function DatasetCitationTooltip({ status }: DatasetCitationTooltipProps) {
+ const { t } = useTranslation('dataset')
+
+ if (status !== DatasetPublishingStatus.RELEASED) {
+ return (
+ <>
+ {' '}
+
+ >
+ )
+ }
+ return <>>
+}
diff --git a/src/sections/dataset/dataset-files/DatasetFiles.tsx b/src/sections/dataset/dataset-files/DatasetFiles.tsx
index cccd061a5..afb96ec8b 100644
--- a/src/sections/dataset/dataset-files/DatasetFiles.tsx
+++ b/src/sections/dataset/dataset-files/DatasetFiles.tsx
@@ -4,9 +4,9 @@ import { FilesTable } from './files-table/FilesTable'
import { FileCriteriaForm } from './file-criteria-form/FileCriteriaForm'
import { FileCriteria } from '../../../files/domain/models/FileCriteria'
import { useFiles } from './useFiles'
-import { FilePaginationInfo } from '../../../files/domain/models/FilePaginationInfo'
-import { FilesPagination } from './files-pagination/FilesPagination'
+import { PaginationControls } from '../../shared/pagination/PaginationControls'
import { DatasetVersion } from '../../../dataset/domain/models/Dataset'
+import { FilePaginationInfo } from '../../../files/domain/models/FilePaginationInfo'
interface DatasetFilesProps {
filesRepository: FileRepository
@@ -44,10 +44,8 @@ export function DatasetFiles({
filesTotalDownloadSize={filesTotalDownloadSize}
criteria={criteria}
/>
-
>
diff --git a/src/sections/dataset/dataset-files/files-table/FilesTable.tsx b/src/sections/dataset/dataset-files/files-table/FilesTable.tsx
index 4ff8e8c4f..8bf0b8874 100644
--- a/src/sections/dataset/dataset-files/files-table/FilesTable.tsx
+++ b/src/sections/dataset/dataset-files/files-table/FilesTable.tsx
@@ -6,10 +6,10 @@ import { FilePreview } from '../../../../files/domain/models/FilePreview'
import { RowSelectionMessage } from './row-selection/RowSelectionMessage'
import { ZipDownloadLimitMessage } from './zip-download-limit-message/ZipDownloadLimitMessage'
import { SpinnerSymbol } from './spinner-symbol/SpinnerSymbol'
-import { FilePaginationInfo } from '../../../../files/domain/models/FilePaginationInfo'
import { useEffect, useState } from 'react'
import { FileSelection } from './row-selection/useFileSelection'
import { FileCriteria } from '../../../../files/domain/models/FileCriteria'
+import { FilePaginationInfo } from '../../../../files/domain/models/FilePaginationInfo'
interface FilesTableProps {
files: FilePreview[]
@@ -56,7 +56,7 @@ export function FilesTable({
>
}
return (
- {`${startIndex} to ${endIndex} of ${fileCount} Files`}
+
)
}
diff --git a/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileDate.tsx b/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileDate.tsx
index 597ec8a87..59076096b 100644
--- a/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileDate.tsx
+++ b/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileDate.tsx
@@ -1,17 +1,13 @@
import { FileDate as FileDateModel } from '../../../../../../../files/domain/models/FilePreview'
import { useTranslation } from 'react-i18next'
+import { DateHelper } from '../../../../../../../shared/domain/helpers/DateHelper'
export function FileDate({ date }: { date: FileDateModel }) {
const { t } = useTranslation('files')
return (
- {t(`table.date.${date.type}`)}{' '}
- {date.date.toLocaleDateString(Intl.DateTimeFormat().resolvedOptions().locale, {
- year: 'numeric',
- month: 'short',
- day: 'numeric'
- })}
+ {t(`table.date.${date.type}`)} {DateHelper.toDisplayFormat(date.date)}
)
diff --git a/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileEmbargoDate.tsx b/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileEmbargoDate.tsx
index b05471d8e..4f95ccc4b 100644
--- a/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileEmbargoDate.tsx
+++ b/src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileEmbargoDate.tsx
@@ -3,6 +3,7 @@ import {
FilePublishingStatus
} from '../../../../../../../files/domain/models/FilePreview'
import { useTranslation } from 'react-i18next'
+import { DateHelper } from '../../../../../../../shared/domain/helpers/DateHelper'
interface FileEmbargoDateProps {
embargo: FileEmbargo | undefined
@@ -20,11 +21,7 @@ export function FileEmbargoDate({ embargo, publishingStatus }: FileEmbargoDatePr
{t(embargoTypeOfDate(embargo.isActive, publishingStatus))}{' '}
- {embargo.dateAvailable.toLocaleDateString(Intl.DateTimeFormat().resolvedOptions().locale, {
- year: 'numeric',
- month: 'short',
- day: 'numeric'
- })}
+ {DateHelper.toDisplayFormat(embargo.dateAvailable)}
)
diff --git a/src/sections/dataset/dataset-files/files-table/row-selection/useFileSelection.ts b/src/sections/dataset/dataset-files/files-table/row-selection/useFileSelection.ts
index 8d6b540ea..fb921b418 100644
--- a/src/sections/dataset/dataset-files/files-table/row-selection/useFileSelection.ts
+++ b/src/sections/dataset/dataset-files/files-table/row-selection/useFileSelection.ts
@@ -1,8 +1,8 @@
import { useEffect, useState } from 'react'
-import { FilePaginationInfo } from '../../../../../files/domain/models/FilePaginationInfo'
import { FilePreview } from '../../../../../files/domain/models/FilePreview'
import { Row } from '@tanstack/react-table'
import { RowSelection } from '../useFilesTable'
+import { FilePaginationInfo } from '../../../../../files/domain/models/FilePaginationInfo'
export type FileSelection = {
[key: string]: FilePreview | undefined
@@ -61,7 +61,7 @@ export function useFileSelection(
const selectAllFiles = () => {
setCurrentPageRowSelection(createRowSelection(paginationInfo.pageSize))
- const totalFilesFileSelection = createFileSelection(paginationInfo.totalFiles)
+ const totalFilesFileSelection = createFileSelection(paginationInfo.totalItems)
const newFileSelection = { ...totalFilesFileSelection, ...fileSelection }
setFileSelection(newFileSelection)
}
diff --git a/src/sections/dataset/dataset-files/files-table/useFilesTable.tsx b/src/sections/dataset/dataset-files/files-table/useFilesTable.tsx
index 3016c2f0e..1ffa88043 100644
--- a/src/sections/dataset/dataset-files/files-table/useFilesTable.tsx
+++ b/src/sections/dataset/dataset-files/files-table/useFilesTable.tsx
@@ -2,8 +2,8 @@ import { useEffect, useState } from 'react'
import { FilePreview } from '../../../../files/domain/models/FilePreview'
import { getCoreRowModel, Row, useReactTable } from '@tanstack/react-table'
import { createColumnsDefinition } from './FilesTableColumnsDefinition'
-import { FilePaginationInfo } from '../../../../files/domain/models/FilePaginationInfo'
import { useFileSelection } from './row-selection/useFileSelection'
+import { FilePaginationInfo } from '../../../../files/domain/models/FilePaginationInfo'
export type RowSelection = {
[key: string]: boolean
diff --git a/src/sections/dataset/dataset-files/useFiles.tsx b/src/sections/dataset/dataset-files/useFiles.tsx
index 40208a177..6e0fcf7e2 100644
--- a/src/sections/dataset/dataset-files/useFiles.tsx
+++ b/src/sections/dataset/dataset-files/useFiles.tsx
@@ -5,11 +5,11 @@ import { getFilesByDatasetPersistentId } from '../../../files/domain/useCases/ge
import { FileCriteria } from '../../../files/domain/models/FileCriteria'
import { FilesCountInfo } from '../../../files/domain/models/FilesCountInfo'
import { getFilesCountInfoByDatasetPersistentId } from '../../../files/domain/useCases/getFilesCountInfoByDatasetPersistentId'
-import { FilePaginationInfo } from '../../../files/domain/models/FilePaginationInfo'
import { useFilePermissions } from '../../file/file-permissions/FilePermissionsContext'
import { FilePermission } from '../../../files/domain/models/FileUserPermissions'
import { DatasetVersion } from '../../../dataset/domain/models/Dataset'
import { getFilesTotalDownloadSize } from '../../../files/domain/useCases/getFilesTotalDownloadSize'
+import { FilePaginationInfo } from '../../../files/domain/models/FilePaginationInfo'
export function useFiles(
filesRepository: FileRepository,
@@ -33,7 +33,7 @@ export function useFiles(
)
.then((filesCountInfo: FilesCountInfo) => {
setFilesCountInfo(filesCountInfo)
- if (filesCountInfo.total !== paginationInfo.totalFiles) {
+ if (filesCountInfo.total !== paginationInfo.totalItems) {
onPaginationInfoChange(paginationInfo.withTotal(filesCountInfo.total))
}
return filesCountInfo
diff --git a/src/sections/dataset/dataset-icon/DatasetIcon.module.scss b/src/sections/dataset/dataset-icon/DatasetIcon.module.scss
new file mode 100644
index 000000000..4025c0bbe
--- /dev/null
+++ b/src/sections/dataset/dataset-icon/DatasetIcon.module.scss
@@ -0,0 +1,6 @@
+@import "node_modules/@iqss/dataverse-design-system/src/lib/assets/styles/design-tokens/colors.module";
+
+.icon {
+ color: $dv-info-border-color;
+ line-height: 1.1;
+}
\ No newline at end of file
diff --git a/src/sections/dataset/dataset-icon/DatasetIcon.tsx b/src/sections/dataset/dataset-icon/DatasetIcon.tsx
new file mode 100644
index 000000000..ac68ba92b
--- /dev/null
+++ b/src/sections/dataset/dataset-icon/DatasetIcon.tsx
@@ -0,0 +1,10 @@
+import styles from './DatasetIcon.module.scss'
+import { Icon, IconName } from '@iqss/dataverse-design-system'
+
+export function DatasetIcon() {
+ return (
+
+
+
+ )
+}
diff --git a/src/sections/dataset/dataset-thumbnail/DatasetThumbnail.module.scss b/src/sections/dataset/dataset-thumbnail/DatasetThumbnail.module.scss
new file mode 100644
index 000000000..a464f4dc4
--- /dev/null
+++ b/src/sections/dataset/dataset-thumbnail/DatasetThumbnail.module.scss
@@ -0,0 +1,6 @@
+.preview-image {
+ width: 100%;
+ max-width: 140px;
+ height: auto;
+ max-height: 140px;
+}
\ No newline at end of file
diff --git a/src/sections/dataset/dataset-thumbnail/DatasetThumbnail.tsx b/src/sections/dataset/dataset-thumbnail/DatasetThumbnail.tsx
new file mode 100644
index 000000000..628cdd8ac
--- /dev/null
+++ b/src/sections/dataset/dataset-thumbnail/DatasetThumbnail.tsx
@@ -0,0 +1,16 @@
+import styles from './DatasetThumbnail.module.scss'
+import { DatasetIcon } from '../dataset-icon/DatasetIcon'
+
+interface DatasetThumbnailProps {
+ thumbnail?: string
+ title: string
+ isDeaccessioned?: boolean
+}
+
+export function DatasetThumbnail({ thumbnail, title, isDeaccessioned }: DatasetThumbnailProps) {
+ if (thumbnail && !isDeaccessioned) {
+ return
+ }
+
+ return
+}
diff --git a/src/sections/hello-dataverse/HelloDataverse.module.scss b/src/sections/hello-dataverse/HelloDataverse.module.scss
deleted file mode 100644
index 505b1cb9c..000000000
--- a/src/sections/hello-dataverse/HelloDataverse.module.scss
+++ /dev/null
@@ -1,26 +0,0 @@
-@import "node_modules/@iqss/dataverse-design-system/src/lib/assets/styles/design-tokens/colors.module";
-@import "src/sections/assets/variables";
-
-.container {
- display: flex;
- flex-direction: column;
- align-items: center;
- justify-content: center;
- min-height: $body-available-height;
- font-size: calc(10px + 2vmin);
- text-align: center;
-}
-
-.title {
- color: $dv-brand-color;
-}
-
-.logo {
- height: 40vmin;
- pointer-events: none;
-}
-
-.link {
- color: $dv-link-color;
-}
-
diff --git a/src/sections/hello-dataverse/HelloDataverse.tsx b/src/sections/hello-dataverse/HelloDataverse.tsx
deleted file mode 100644
index e955d0e66..000000000
--- a/src/sections/hello-dataverse/HelloDataverse.tsx
+++ /dev/null
@@ -1,24 +0,0 @@
-import logo from '../assets/logo.svg'
-import styles from './HelloDataverse.module.scss'
-import { Trans, useTranslation } from 'react-i18next'
-
-export function HelloDataverse() {
- const { t } = useTranslation('helloDataverse')
-
- return (
-
- )
-}
diff --git a/src/sections/home/Home.tsx b/src/sections/home/Home.tsx
new file mode 100644
index 000000000..803e2a1d6
--- /dev/null
+++ b/src/sections/home/Home.tsx
@@ -0,0 +1,20 @@
+import { Row } from '@iqss/dataverse-design-system'
+import { useTranslation } from 'react-i18next'
+import { DatasetRepository } from '../../dataset/domain/repositories/DatasetRepository'
+import { DatasetsList } from './datasets-list/DatasetsList'
+interface HomeProps {
+ datasetRepository: DatasetRepository
+}
+
+export function Home({ datasetRepository }: HomeProps) {
+ const { t } = useTranslation('home')
+
+ return (
+
+
+
+
+ )
+}
diff --git a/src/sections/home/HomeFactory.tsx b/src/sections/home/HomeFactory.tsx
new file mode 100644
index 000000000..f37a69caa
--- /dev/null
+++ b/src/sections/home/HomeFactory.tsx
@@ -0,0 +1,10 @@
+import { ReactElement } from 'react'
+import { Home } from './Home'
+import { DatasetJSDataverseRepository } from '../../dataset/infrastructure/repositories/DatasetJSDataverseRepository'
+
+const datasetRepository = new DatasetJSDataverseRepository()
+export class HomeFactory {
+ static create(): ReactElement {
+ return
+ }
+}
diff --git a/src/sections/home/datasets-list/DatasetsList.module.scss b/src/sections/home/datasets-list/DatasetsList.module.scss
new file mode 100644
index 000000000..15aff10a5
--- /dev/null
+++ b/src/sections/home/datasets-list/DatasetsList.module.scss
@@ -0,0 +1,13 @@
+@import "node_modules/@iqss/dataverse-design-system/src/lib/assets/styles/design-tokens/colors.module";
+
+.container {
+ min-height: calc(100vh + 100px);
+ padding:15px;
+ border: 1px solid #ddd;
+ border-radius: 4px;
+}
+
+.empty-message-container {
+ padding: .5em 1em;
+ background: $dv-warning-box-color;
+}
diff --git a/src/sections/home/datasets-list/DatasetsList.tsx b/src/sections/home/datasets-list/DatasetsList.tsx
new file mode 100644
index 000000000..da13f06d3
--- /dev/null
+++ b/src/sections/home/datasets-list/DatasetsList.tsx
@@ -0,0 +1,53 @@
+import { useDatasets } from './useDatasets'
+import styles from './DatasetsList.module.scss'
+import { DatasetRepository } from '../../../dataset/domain/repositories/DatasetRepository'
+import { useEffect, useState } from 'react'
+import { PaginationResultsInfo } from '../../shared/pagination/PaginationResultsInfo'
+import { PaginationControls } from '../../shared/pagination/PaginationControls'
+import { DatasetPaginationInfo } from '../../../dataset/domain/models/DatasetPaginationInfo'
+import { useLoading } from '../../loading/LoadingContext'
+import { DatasetsListSkeleton } from './DatasetsListSkeleton'
+import { NoDatasetsMessage } from './NoDatasetsMessage'
+import { DatasetCard } from './dataset-card/DatasetCard'
+
+interface DatasetsListProps {
+ datasetRepository: DatasetRepository
+}
+const NO_DATASETS = 0
+export function DatasetsList({ datasetRepository }: DatasetsListProps) {
+ const { setIsLoading } = useLoading()
+ const [paginationInfo, setPaginationInfo] = useState(
+ new DatasetPaginationInfo()
+ )
+ const { datasets, isLoading } = useDatasets(datasetRepository, setPaginationInfo, paginationInfo)
+
+ useEffect(() => {
+ setIsLoading(isLoading)
+ }, [isLoading])
+
+ if (isLoading) {
+ return
+ }
+
+ return (
+
+ {datasets.length === NO_DATASETS ? (
+
+ ) : (
+ <>
+
+ {datasets.map((dataset) => (
+
+ ))}
+
+ >
+ )}
+
+ )
+}
diff --git a/src/sections/home/datasets-list/DatasetsListSkeleton.tsx b/src/sections/home/datasets-list/DatasetsListSkeleton.tsx
new file mode 100644
index 000000000..b07bc39d3
--- /dev/null
+++ b/src/sections/home/datasets-list/DatasetsListSkeleton.tsx
@@ -0,0 +1,26 @@
+import styles from './DatasetsList.module.scss'
+import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'
+
+export function DatasetsListSkeleton() {
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/sections/home/datasets-list/NoDatasetsMessage.tsx b/src/sections/home/datasets-list/NoDatasetsMessage.tsx
new file mode 100644
index 000000000..a7ad3cfc0
--- /dev/null
+++ b/src/sections/home/datasets-list/NoDatasetsMessage.tsx
@@ -0,0 +1,24 @@
+import styles from './DatasetsList.module.scss'
+import { Trans, useTranslation } from 'react-i18next'
+import { useSession } from '../../session/SessionContext'
+import { Route } from '../../Route.enum'
+
+export function NoDatasetsMessage() {
+ const { t } = useTranslation('home')
+ const { user } = useSession()
+
+ return (
+
+ {user ? (
+
{t('noDatasetsMessage.authenticated')}
+ ) : (
+
+
+ This dataverse currently has no datasets. Please log in to
+ see if you are able to add to it.
+
+
+ )}
+
+ )
+}
diff --git a/src/sections/home/datasets-list/dataset-card/DatasetCard.module.scss b/src/sections/home/datasets-list/dataset-card/DatasetCard.module.scss
new file mode 100644
index 000000000..92f7696b2
--- /dev/null
+++ b/src/sections/home/datasets-list/dataset-card/DatasetCard.module.scss
@@ -0,0 +1,71 @@
+@use 'sass:color';
+@import "node_modules/@iqss/dataverse-design-system/src/lib/assets/styles/design-tokens/colors.module";
+@import "node_modules/@iqss/dataverse-design-system/src/lib/assets/styles/design-tokens/typography.module";
+
+.container {
+ margin: 6px 0;
+ padding: 4px 10px;
+ border: 1px solid $dv-info-border-color;
+}
+
+.header {
+ display: flex;
+ justify-content: space-between;
+}
+
+.title {
+ display: flex;
+
+ > * {
+ margin-right: .5em;
+ }
+}
+
+.icon {
+ margin-top: 2px;
+ font-size: 1.3em;
+
+ > div >span {
+ margin-right: 0;
+ }
+}
+
+.thumbnail {
+ width: 48px;
+ margin: 8px 12px 6px 0;
+ font-size: 2.8em;
+
+ img {
+ vertical-align: top;
+ }
+}
+
+.info {
+ display: flex;
+}
+
+.description {
+ display: flex;
+ flex-direction: column;
+ width: 100%;
+ font-size: $dv-font-size-sm;
+}
+
+.date {
+ color: $dv-subtext-color;
+
+}
+
+.citation-box {
+ margin-top: 4px;
+ margin-bottom: .5em;
+ padding: 4px;
+ background-color: color.adjust($dv-primary-color, $lightness: 51%) ;
+}
+
+.citation-box-deaccessioned {
+ margin-top: 4px;
+ margin-bottom: .5em;
+ padding: 4px;
+ background-color: color.adjust($dv-danger-box-color, $lightness: 6%);
+}
\ No newline at end of file
diff --git a/src/sections/home/datasets-list/dataset-card/DatasetCard.tsx b/src/sections/home/datasets-list/dataset-card/DatasetCard.tsx
new file mode 100644
index 000000000..6fe1f807e
--- /dev/null
+++ b/src/sections/home/datasets-list/dataset-card/DatasetCard.tsx
@@ -0,0 +1,21 @@
+import { DatasetPreview } from '../../../../dataset/domain/models/DatasetPreview'
+import styles from './DatasetCard.module.scss'
+import { DatasetCardHeader } from './DatasetCardHeader'
+import { DatasetCardThumbnail } from './DatasetCardThumbnail'
+import { DatasetCardInfo } from './DatasetCardInfo'
+
+interface DatasetCardProps {
+ dataset: DatasetPreview
+}
+
+export function DatasetCard({ dataset }: DatasetCardProps) {
+ return (
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/sections/home/datasets-list/dataset-card/DatasetCardHeader.tsx b/src/sections/home/datasets-list/dataset-card/DatasetCardHeader.tsx
new file mode 100644
index 000000000..9e927b2fc
--- /dev/null
+++ b/src/sections/home/datasets-list/dataset-card/DatasetCardHeader.tsx
@@ -0,0 +1,25 @@
+import styles from './DatasetCard.module.scss'
+import { LinkToPage } from '../../../shared/link-to-page/LinkToPage'
+import { Route } from '../../../Route.enum'
+import { DatasetLabels } from '../../../dataset/dataset-labels/DatasetLabels'
+import { DatasetPreview } from '../../../../dataset/domain/models/DatasetPreview'
+import { DatasetIcon } from '../../../dataset/dataset-icon/DatasetIcon'
+
+interface DatasetCardHeaderProps {
+ dataset: DatasetPreview
+}
+export function DatasetCardHeader({ dataset }: DatasetCardHeaderProps) {
+ return (
+
+
+
+ {dataset.title}
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/sections/home/datasets-list/dataset-card/DatasetCardInfo.tsx b/src/sections/home/datasets-list/dataset-card/DatasetCardInfo.tsx
new file mode 100644
index 000000000..76a03a645
--- /dev/null
+++ b/src/sections/home/datasets-list/dataset-card/DatasetCardInfo.tsx
@@ -0,0 +1,23 @@
+import styles from './DatasetCard.module.scss'
+import { DateHelper } from '../../../../shared/domain/helpers/DateHelper'
+import { DatasetPreview } from '../../../../dataset/domain/models/DatasetPreview'
+import { CitationDescription } from '../../../shared/citation/CitationDescription'
+
+interface DatasetCardInfoProps {
+ dataset: DatasetPreview
+}
+
+export function DatasetCardInfo({ dataset }: DatasetCardInfoProps) {
+ return (
+
+ {DateHelper.toDisplayFormat(dataset.releaseOrCreateDate)}
+
+
+
+ {dataset.abbreviatedDescription}
+
+ )
+}
diff --git a/src/sections/home/datasets-list/dataset-card/DatasetCardThumbnail.tsx b/src/sections/home/datasets-list/dataset-card/DatasetCardThumbnail.tsx
new file mode 100644
index 000000000..cba65b215
--- /dev/null
+++ b/src/sections/home/datasets-list/dataset-card/DatasetCardThumbnail.tsx
@@ -0,0 +1,23 @@
+import styles from './DatasetCard.module.scss'
+import { DatasetPreview } from '../../../../dataset/domain/models/DatasetPreview'
+import { LinkToPage } from '../../../shared/link-to-page/LinkToPage'
+import { Route } from '../../../Route.enum'
+import { DatasetThumbnail } from '../../../dataset/dataset-thumbnail/DatasetThumbnail'
+
+interface DatasetCardThumbnailProps {
+ dataset: DatasetPreview
+}
+
+export function DatasetCardThumbnail({ dataset }: DatasetCardThumbnailProps) {
+ return (
+
+
+
+
+
+ )
+}
diff --git a/src/sections/home/datasets-list/useDatasets.tsx b/src/sections/home/datasets-list/useDatasets.tsx
new file mode 100644
index 000000000..8bb900423
--- /dev/null
+++ b/src/sections/home/datasets-list/useDatasets.tsx
@@ -0,0 +1,65 @@
+import { useEffect, useState } from 'react'
+import { DatasetRepository } from '../../../dataset/domain/repositories/DatasetRepository'
+import { getDatasets } from '../../../dataset/domain/useCases/getDatasets'
+import { getTotalDatasetsCount } from '../../../dataset/domain/useCases/getTotalDatasetsCount'
+import { TotalDatasetsCount } from '../../../dataset/domain/models/TotalDatasetsCount'
+import { DatasetPaginationInfo } from '../../../dataset/domain/models/DatasetPaginationInfo'
+import { DatasetPreview } from '../../../dataset/domain/models/DatasetPreview'
+
+export function useDatasets(
+ datasetRepository: DatasetRepository,
+ onPaginationInfoChange: (paginationInfo: DatasetPaginationInfo) => void,
+ paginationInfo: DatasetPaginationInfo
+) {
+ const [datasets, setDatasets] = useState([])
+ const [isLoading, setIsLoading] = useState(true)
+ const [totalDatasetsCount, setTotalDatasetsCount] = useState()
+
+ const fetchTotalDatasetsCount: () => Promise = () => {
+ return getTotalDatasetsCount(datasetRepository)
+ .then((totalDatasetsCount: TotalDatasetsCount) => {
+ setTotalDatasetsCount(totalDatasetsCount)
+ if (totalDatasetsCount !== paginationInfo.totalItems) {
+ onPaginationInfoChange(paginationInfo.withTotal(totalDatasetsCount))
+ }
+ return totalDatasetsCount
+ })
+ .catch(() => {
+ throw new Error('There was an error getting the datasets count info')
+ })
+ }
+ const fetchDatasets = (totalDatasetsCount: TotalDatasetsCount) => {
+ if (typeof totalDatasetsCount !== 'undefined') {
+ if (totalDatasetsCount === 0) {
+ setIsLoading(false)
+ return
+ }
+ return getDatasets(datasetRepository, paginationInfo.withTotal(totalDatasetsCount))
+ .then((datasets: DatasetPreview[]) => {
+ setDatasets(datasets)
+ setIsLoading(false)
+ return datasets
+ })
+ .catch(() => {
+ throw new Error('There was an error getting the datasets')
+ })
+ }
+ }
+
+ useEffect(() => {
+ setIsLoading(true)
+
+ fetchTotalDatasetsCount()
+ .then((totalDatasetsCount) => fetchDatasets(totalDatasetsCount))
+ .catch(() => {
+ console.error('There was an error getting the datasets')
+ setIsLoading(false)
+ })
+ }, [datasetRepository, paginationInfo.page])
+
+ return {
+ datasets,
+ totalDatasetsCount,
+ isLoading
+ }
+}
diff --git a/src/sections/layout/header/Header.tsx b/src/sections/layout/header/Header.tsx
index e83b454be..7121bd1f1 100644
--- a/src/sections/layout/header/Header.tsx
+++ b/src/sections/layout/header/Header.tsx
@@ -3,22 +3,31 @@ import { useTranslation } from 'react-i18next'
import { Navbar } from '@iqss/dataverse-design-system'
import { Route } from '../../Route.enum'
import { useSession } from '../../session/SessionContext'
+import { useNavigate } from 'react-router-dom'
+const currentPage = 0
export function Header() {
const { t } = useTranslation('header')
const { user, logout } = useSession()
+ const navigate = useNavigate()
const baseRemoteUrl = import.meta.env.VITE_DATAVERSE_BACKEND_URL as string
+ const onLogoutClick = () => {
+ void logout().then(() => {
+ navigate(currentPage)
+ })
+ }
+
return (
{user ? (
-
+
{t('logOut')}
diff --git a/src/sections/session/SessionContext.ts b/src/sections/session/SessionContext.ts
index 4840d510a..5bafb77cf 100644
--- a/src/sections/session/SessionContext.ts
+++ b/src/sections/session/SessionContext.ts
@@ -4,12 +4,12 @@ import { User } from '../../users/domain/models/User'
interface SessionContextProps {
user: User | null
setUser: (user: User) => void
- logout: () => void
+ logout: () => Promise
}
export const SessionContext = createContext({
user: null,
setUser: () => {},
- logout: () => {}
+ logout: () => Promise.resolve()
})
export const useSession = () => useContext(SessionContext)
diff --git a/src/sections/session/SessionProvider.tsx b/src/sections/session/SessionProvider.tsx
index 56fa07d81..4fb95b74b 100644
--- a/src/sections/session/SessionProvider.tsx
+++ b/src/sections/session/SessionProvider.tsx
@@ -20,7 +20,7 @@ export function SessionProvider({ repository, children }: PropsWithChildren {
- logOut(repository)
+ return logOut(repository)
.then(() => {
setUser(null)
})
diff --git a/src/sections/shared/citation/CitationDescription.tsx b/src/sections/shared/citation/CitationDescription.tsx
new file mode 100644
index 000000000..eb2730bd2
--- /dev/null
+++ b/src/sections/shared/citation/CitationDescription.tsx
@@ -0,0 +1,11 @@
+import parse from 'html-react-parser'
+
+interface CitationDescriptionProps {
+ citation: string
+}
+
+export function CitationDescription({ citation }: CitationDescriptionProps) {
+ const citationAsReactElement = parse(citation)
+
+ return {citationAsReactElement}
+}
diff --git a/src/sections/dataset/dataset-files/files-pagination/PageNumbersButtons.tsx b/src/sections/shared/pagination/PageNumbersButtons.tsx
similarity index 100%
rename from src/sections/dataset/dataset-files/files-pagination/PageNumbersButtons.tsx
rename to src/sections/shared/pagination/PageNumbersButtons.tsx
diff --git a/src/sections/dataset/dataset-files/files-pagination/PageNumbersButtonsWithEllipsis.tsx b/src/sections/shared/pagination/PageNumbersButtonsWithEllipsis.tsx
similarity index 100%
rename from src/sections/dataset/dataset-files/files-pagination/PageNumbersButtonsWithEllipsis.tsx
rename to src/sections/shared/pagination/PageNumbersButtonsWithEllipsis.tsx
diff --git a/src/sections/dataset/dataset-files/files-pagination/PageSizeSelector.tsx b/src/sections/shared/pagination/PageSizeSelector.tsx
similarity index 80%
rename from src/sections/dataset/dataset-files/files-pagination/PageSizeSelector.tsx
rename to src/sections/shared/pagination/PageSizeSelector.tsx
index a5acbf945..2378453df 100644
--- a/src/sections/dataset/dataset-files/files-pagination/PageSizeSelector.tsx
+++ b/src/sections/shared/pagination/PageSizeSelector.tsx
@@ -1,20 +1,24 @@
-import styles from './FilesPagination.module.scss'
+import styles from './Pagination.module.scss'
import { useTranslation } from 'react-i18next'
export function PageSizeSelector({
+ itemName,
pageSize,
setPageSize
}: {
+ itemName: string
pageSize: number
setPageSize: (pageSize: number) => void
}) {
- const { t } = useTranslation('files')
+ const { t } = useTranslation('pagination')
const availableSizes = [10, 25, 50]
return (
diff --git a/src/sections/shared/pagination/PaginationResultsInfo.tsx b/src/sections/shared/pagination/PaginationResultsInfo.tsx
new file mode 100644
index 000000000..df57ad701
--- /dev/null
+++ b/src/sections/shared/pagination/PaginationResultsInfo.tsx
@@ -0,0 +1,23 @@
+import styles from './Pagination.module.scss'
+import { PaginationInfo } from '../../../shared/domain/models/PaginationInfo'
+import { useTranslation } from 'react-i18next'
+import { DatasetPaginationInfo } from '../../../dataset/domain/models/DatasetPaginationInfo'
+import { FilePaginationInfo } from '../../../files/domain/models/FilePaginationInfo'
+
+interface PaginationResultsInfoProps {
+ paginationInfo: PaginationInfo
+}
+
+export function PaginationResultsInfo({ paginationInfo }: PaginationResultsInfoProps) {
+ const { t } = useTranslation('pagination')
+ return (
+
+ {t('results', {
+ start: paginationInfo.pageStartItem,
+ end: paginationInfo.pageEndItem,
+ total: paginationInfo.totalItems,
+ item: paginationInfo.itemName
+ })}
+
+ )
+}
diff --git a/src/shared/domain/helpers/DateHelper.ts b/src/shared/domain/helpers/DateHelper.ts
new file mode 100644
index 000000000..021ec2a68
--- /dev/null
+++ b/src/shared/domain/helpers/DateHelper.ts
@@ -0,0 +1,9 @@
+export class DateHelper {
+ static toDisplayFormat(date: Date): string {
+ return date.toLocaleDateString(Intl.DateTimeFormat().resolvedOptions().locale, {
+ year: 'numeric',
+ month: 'short',
+ day: 'numeric'
+ })
+ }
+}
diff --git a/src/shared/domain/models/PaginationInfo.ts b/src/shared/domain/models/PaginationInfo.ts
new file mode 100644
index 000000000..c21822b7c
--- /dev/null
+++ b/src/shared/domain/models/PaginationInfo.ts
@@ -0,0 +1,72 @@
+export class PaginationInfo> {
+ constructor(
+ public readonly page: number = 1,
+ public readonly pageSize: number = 10,
+ public readonly totalItems: number = 0,
+ public readonly itemName: string = 'Item'
+ ) {}
+
+ get offset(): number {
+ return (this.page - 1) * this.pageSize
+ }
+
+ get pageStartItem(): number {
+ return (this.page - 1) * this.pageSize + 1
+ }
+
+ get pageEndItem(): number {
+ return Math.min(this.pageStartItem + this.pageSize - 1, this.totalItems)
+ }
+
+ withTotal(total: number): T {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ return new (this.constructor as new (...args: any[]) => T)(this.page, this.pageSize, total)
+ }
+ goToPage(page: number): T {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ return new (this.constructor as new (...args: any[]) => T)(page, this.pageSize, this.totalItems)
+ }
+
+ goToPreviousPage(): T {
+ if (!this.previousPage) throw new Error('No previous page')
+ return this.goToPage(this.previousPage)
+ }
+
+ goToNextPage(): T {
+ if (!this.nextPage) throw new Error('No next page')
+ return this.goToPage(this.nextPage)
+ }
+
+ withPageSize(pageSize: number): T {
+ const getNewPage = (oldPageSize: number, newPageSize: number) => {
+ const newPage = Math.ceil(((this.page - 1) * oldPageSize + 1) / newPageSize)
+ return newPage > 0 ? newPage : 1
+ }
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ return new (this.constructor as new (...args: any[]) => T)(
+ getNewPage(this.pageSize, pageSize),
+ pageSize,
+ this.totalItems
+ )
+ }
+
+ get totalPages(): number {
+ return Math.ceil(this.totalItems / this.pageSize)
+ }
+
+ get hasPreviousPage(): boolean {
+ return this.page > 1
+ }
+
+ get hasNextPage(): boolean {
+ return this.page < this.totalPages
+ }
+
+ get previousPage(): number | undefined {
+ return this.hasPreviousPage ? this.page - 1 : undefined
+ }
+
+ get nextPage(): number | undefined {
+ return this.hasNextPage ? this.page + 1 : undefined
+ }
+}
diff --git a/src/stories/WithLoggedInUser.tsx b/src/stories/WithLoggedInUser.tsx
index 75a5878a4..233c46e6f 100644
--- a/src/stories/WithLoggedInUser.tsx
+++ b/src/stories/WithLoggedInUser.tsx
@@ -5,7 +5,7 @@ import { UserMother } from '../../tests/component/users/domain/models/UserMother
export const WithLoggedInUser = (Story: StoryFn) => {
return (
{}, setUser: () => {} }}>
+ value={{ user: UserMother.create(), logout: () => Promise.resolve(), setUser: () => {} }}>
)
diff --git a/src/stories/dataset/DatasetLoadingMockRepository.ts b/src/stories/dataset/DatasetLoadingMockRepository.ts
new file mode 100644
index 000000000..b7051afc7
--- /dev/null
+++ b/src/stories/dataset/DatasetLoadingMockRepository.ts
@@ -0,0 +1,10 @@
+import { DatasetMockRepository } from './DatasetMockRepository'
+import { DatasetPaginationInfo } from '../../dataset/domain/models/DatasetPaginationInfo'
+import { DatasetPreview } from '../../dataset/domain/models/DatasetPreview'
+
+export class DatasetLoadingMockRepository extends DatasetMockRepository {
+ // eslint-disable-next-line unused-imports/no-unused-vars
+ getAll(paginationInfo: DatasetPaginationInfo): Promise {
+ return new Promise(() => {})
+ }
+}
diff --git a/src/stories/dataset/DatasetMockRepository.ts b/src/stories/dataset/DatasetMockRepository.ts
new file mode 100644
index 000000000..9ffa083ed
--- /dev/null
+++ b/src/stories/dataset/DatasetMockRepository.ts
@@ -0,0 +1,49 @@
+import { Dataset } from '../../dataset/domain/models/Dataset'
+import { DatasetRepository } from '../../dataset/domain/repositories/DatasetRepository'
+import { DatasetMother } from '../../../tests/component/dataset/domain/models/DatasetMother'
+import { TotalDatasetsCount } from '../../dataset/domain/models/TotalDatasetsCount'
+import { DatasetPaginationInfo } from '../../dataset/domain/models/DatasetPaginationInfo'
+import { DatasetPreview } from '../../dataset/domain/models/DatasetPreview'
+import { DatasetPreviewMother } from '../../../tests/component/dataset/domain/models/DatasetPreviewMother'
+
+export class DatasetMockRepository implements DatasetRepository {
+ // eslint-disable-next-line unused-imports/no-unused-vars
+ getAll(paginationInfo: DatasetPaginationInfo): Promise {
+ return new Promise((resolve) => {
+ setTimeout(() => {
+ resolve(DatasetPreviewMother.createManyRealistic(paginationInfo.pageSize))
+ }, 1000)
+ })
+ }
+
+ getTotalDatasetsCount(): Promise {
+ return new Promise((resolve) => {
+ setTimeout(() => {
+ resolve(200)
+ }, 1000)
+ })
+ }
+
+ getByPersistentId(
+ // eslint-disable-next-line unused-imports/no-unused-vars
+ persistentId: string,
+ // eslint-disable-next-line unused-imports/no-unused-vars
+ version?: string | undefined
+ ): Promise {
+ return new Promise((resolve) => {
+ setTimeout(() => {
+ resolve(DatasetMother.createRealistic())
+ }, 1000)
+ })
+ }
+ getByPrivateUrlToken(
+ // eslint-disable-next-line unused-imports/no-unused-vars
+ privateUrlToken: string
+ ): Promise {
+ return new Promise((resolve) => {
+ setTimeout(() => {
+ resolve(DatasetMother.createRealistic())
+ }, 1000)
+ })
+ }
+}
diff --git a/src/stories/dataset/NoDatasetsMockRepository.ts b/src/stories/dataset/NoDatasetsMockRepository.ts
new file mode 100644
index 000000000..b5fa7b41c
--- /dev/null
+++ b/src/stories/dataset/NoDatasetsMockRepository.ts
@@ -0,0 +1,22 @@
+import { DatasetMockRepository } from './DatasetMockRepository'
+import { DatasetPaginationInfo } from '../../dataset/domain/models/DatasetPaginationInfo'
+import { DatasetPreview } from '../../dataset/domain/models/DatasetPreview'
+import { TotalDatasetsCount } from '../../dataset/domain/models/TotalDatasetsCount'
+export class NoDatasetsMockRepository extends DatasetMockRepository {
+ // eslint-disable-next-line unused-imports/no-unused-vars
+ getAll(paginationInfo: DatasetPaginationInfo): Promise {
+ return new Promise((resolve) => {
+ setTimeout(() => {
+ resolve([])
+ }, 1000)
+ })
+ }
+
+ getTotalDatasetsCount(): Promise {
+ return new Promise((resolve) => {
+ setTimeout(() => {
+ resolve(0)
+ }, 1000)
+ })
+ }
+}
diff --git a/src/stories/dataset/WithDataset.tsx b/src/stories/dataset/WithDataset.tsx
index 6c1df0ddf..b559c9532 100644
--- a/src/stories/dataset/WithDataset.tsx
+++ b/src/stories/dataset/WithDataset.tsx
@@ -1,23 +1,10 @@
import { StoryFn } from '@storybook/react'
import { DatasetProvider } from '../../sections/dataset/DatasetProvider'
import { DatasetRepository } from '../../dataset/domain/repositories/DatasetRepository'
-import { Dataset } from '../../dataset/domain/models/Dataset'
-import { DatasetMother } from '../../../tests/component/dataset/domain/models/DatasetMother'
+import { DatasetMockRepository } from './DatasetMockRepository'
export const WithDataset = (Story: StoryFn) => {
- const datasetRepository = {} as DatasetRepository
- datasetRepository.getByPersistentId = (
- // eslint-disable-next-line unused-imports/no-unused-vars
- persistentId: string,
- // eslint-disable-next-line unused-imports/no-unused-vars
- version?: string | undefined
- ): Promise => {
- return new Promise((resolve) => {
- setTimeout(() => {
- resolve(DatasetMother.createRealistic())
- }, 1000)
- })
- }
+ const datasetRepository: DatasetRepository = new DatasetMockRepository()
return (
= {
- title: 'Sections/Dataset Page/DatasetFiles/FilesPagination',
- component: FilesPagination,
- decorators: [WithI18next]
-}
-
-export default meta
-type Story = StoryObj
-
-const emptyFunction = () => {}
-export const Default: Story = {
- render: () => (
-
- )
-}
-
-export const NoEllipsis: Story = {
- render: () => (
-
- )
-}
diff --git a/src/stories/files/FileMockRepository.ts b/src/stories/files/FileMockRepository.ts
index fb03cfc96..cbb518ecf 100644
--- a/src/stories/files/FileMockRepository.ts
+++ b/src/stories/files/FileMockRepository.ts
@@ -3,12 +3,12 @@ import { FilesMockData } from './FileMockData'
import { FilePreview, FileDownloadMode } from '../../files/domain/models/FilePreview'
import { FilesCountInfo } from '../../files/domain/models/FilesCountInfo'
import { FilesCountInfoMother } from '../../../tests/component/files/domain/models/FilesCountInfoMother'
-import { FilePaginationInfo } from '../../files/domain/models/FilePaginationInfo'
import { FileUserPermissionsMother } from '../../../tests/component/files/domain/models/FileUserPermissionsMother'
import { FileUserPermissions } from '../../files/domain/models/FileUserPermissions'
import { DatasetVersion, DatasetVersionNumber } from '../../dataset/domain/models/Dataset'
import { FileCriteria } from '../../files/domain/models/FileCriteria'
import { FilePreviewMother } from '../../../tests/component/files/domain/models/FilePreviewMother'
+import { FilePaginationInfo } from '../../files/domain/models/FilePaginationInfo'
import { FileMother } from '../../../tests/component/files/domain/models/FileMother'
import { File } from '../../files/domain/models/File'
diff --git a/src/stories/hello-dataverse/HelloDataverse.stories.tsx b/src/stories/hello-dataverse/HelloDataverse.stories.tsx
deleted file mode 100644
index f95572055..000000000
--- a/src/stories/hello-dataverse/HelloDataverse.stories.tsx
+++ /dev/null
@@ -1,17 +0,0 @@
-import type { Meta, StoryObj } from '@storybook/react'
-import { HelloDataverse } from '../../sections/hello-dataverse/HelloDataverse'
-import { WithI18next } from '../WithI18next'
-import { WithLayout } from '../WithLayout'
-
-const meta: Meta = {
- title: 'Pages/Hello Dataverse',
- component: HelloDataverse,
- decorators: [WithI18next, WithLayout]
-}
-
-export default meta
-type Story = StoryObj
-
-export const Default: Story = {
- render: () =>
-}
diff --git a/src/stories/home/Home.stories.tsx b/src/stories/home/Home.stories.tsx
new file mode 100644
index 000000000..800bb977e
--- /dev/null
+++ b/src/stories/home/Home.stories.tsx
@@ -0,0 +1,28 @@
+import type { Meta, StoryObj } from '@storybook/react'
+import { Home } from '../../sections/home/Home'
+import { WithI18next } from '../WithI18next'
+import { WithLayout } from '../WithLayout'
+import { DatasetMockRepository } from '../dataset/DatasetMockRepository'
+import { DatasetLoadingMockRepository } from '../dataset/DatasetLoadingMockRepository'
+import { NoDatasetsMockRepository } from '../dataset/NoDatasetsMockRepository'
+
+const meta: Meta = {
+ title: 'Pages/Home',
+ component: Home,
+ decorators: [WithI18next, WithLayout]
+}
+
+export default meta
+type Story = StoryObj
+
+export const Default: Story = {
+ render: () =>
+}
+
+export const Loading: Story = {
+ render: () =>
+}
+
+export const NoResults: Story = {
+ render: () =>
+}
diff --git a/src/stories/home/datasets-list/DatasetCard.stories.tsx b/src/stories/home/datasets-list/DatasetCard.stories.tsx
new file mode 100644
index 000000000..26f6af0da
--- /dev/null
+++ b/src/stories/home/datasets-list/DatasetCard.stories.tsx
@@ -0,0 +1,21 @@
+import { Meta, StoryObj } from '@storybook/react'
+import { WithI18next } from '../../WithI18next'
+import { DatasetCard } from '../../../sections/home/datasets-list/dataset-card/DatasetCard'
+import { DatasetPreviewMother } from '../../../../tests/component/dataset/domain/models/DatasetPreviewMother'
+
+const meta: Meta = {
+ title: 'Sections/Home/DatasetCard',
+ component: DatasetCard,
+ decorators: [WithI18next]
+}
+
+export default meta
+type Story = StoryObj
+
+export const Default: Story = {
+ render: () =>
+}
+
+export const Deaccessioned: Story = {
+ render: () =>
+}
diff --git a/src/stories/home/datasets-list/DatasetsList.stories.tsx b/src/stories/home/datasets-list/DatasetsList.stories.tsx
new file mode 100644
index 000000000..51287336f
--- /dev/null
+++ b/src/stories/home/datasets-list/DatasetsList.stories.tsx
@@ -0,0 +1,27 @@
+import { Meta, StoryObj } from '@storybook/react'
+import { WithI18next } from '../../WithI18next'
+import { DatasetMockRepository } from '../../dataset/DatasetMockRepository'
+import { DatasetsList } from '../../../sections/home/datasets-list/DatasetsList'
+import { DatasetLoadingMockRepository } from '../../dataset/DatasetLoadingMockRepository'
+import { NoDatasetsMockRepository } from '../../dataset/NoDatasetsMockRepository'
+
+const meta: Meta = {
+ title: 'Sections/Home/DatasetsList',
+ component: DatasetsList,
+ decorators: [WithI18next]
+}
+
+export default meta
+type Story = StoryObj
+
+export const Default: Story = {
+ render: () =>
+}
+
+export const Loading: Story = {
+ render: () =>
+}
+
+export const NoResults: Story = {
+ render: () =>
+}
diff --git a/src/stories/home/datasets-list/NoDatasetsMessage.stories.tsx b/src/stories/home/datasets-list/NoDatasetsMessage.stories.tsx
new file mode 100644
index 000000000..d81a48e90
--- /dev/null
+++ b/src/stories/home/datasets-list/NoDatasetsMessage.stories.tsx
@@ -0,0 +1,23 @@
+import { Meta, StoryObj } from '@storybook/react'
+import { Home } from '../../../sections/home/Home'
+import { WithI18next } from '../../WithI18next'
+import { NoDatasetsMessage } from '../../../sections/home/datasets-list/NoDatasetsMessage'
+import { WithLoggedInUser } from '../../WithLoggedInUser'
+
+const meta: Meta = {
+ title: 'Sections/Home/NoDatasetsMessage',
+ component: Home,
+ decorators: [WithI18next]
+}
+
+export default meta
+type Story = StoryObj
+
+export const AnonymousUser: Story = {
+ render: () =>
+}
+
+export const AuthenticatedUser: Story = {
+ decorators: [WithLoggedInUser],
+ render: () =>
+}
diff --git a/src/stories/shared/pagination/PaginationControls.stories.tsx b/src/stories/shared/pagination/PaginationControls.stories.tsx
new file mode 100644
index 000000000..a482f80d2
--- /dev/null
+++ b/src/stories/shared/pagination/PaginationControls.stories.tsx
@@ -0,0 +1,38 @@
+import { Meta, StoryObj } from '@storybook/react'
+import { WithI18next } from '../../WithI18next'
+import { PaginationControls } from '../../../sections/shared/pagination/PaginationControls'
+import { PaginationInfo } from '../../../shared/domain/models/PaginationInfo'
+import { DatasetPaginationInfo } from '../../../dataset/domain/models/DatasetPaginationInfo'
+import { FilePaginationInfo } from '../../../files/domain/models/FilePaginationInfo'
+
+const meta: Meta = {
+ title: 'Sections/Shared/Pagination/PaginationControls',
+ component: PaginationControls,
+ decorators: [WithI18next]
+}
+
+export default meta
+type Story = StoryObj
+
+const emptyFunction = () => {}
+export const Default: Story = {
+ render: () => (
+ (10, 10, 200)
+ }
+ onPaginationInfoChange={emptyFunction}
+ />
+ )
+}
+
+export const NoEllipsis: Story = {
+ render: () => (
+ (1, 10, 100)
+ }
+ onPaginationInfoChange={emptyFunction}
+ />
+ )
+}
diff --git a/tests/component/dataset/domain/models/DatasetMother.ts b/tests/component/dataset/domain/models/DatasetMother.ts
index 76a69dcab..81a4f0628 100644
--- a/tests/component/dataset/domain/models/DatasetMother.ts
+++ b/tests/component/dataset/domain/models/DatasetMother.ts
@@ -2,6 +2,7 @@ import { faker } from '@faker-js/faker'
import {
ANONYMIZED_FIELD_VALUE,
Dataset,
+ DatasetLabel,
DatasetDownloadUrls,
DatasetLabelSemanticMeaning,
DatasetLabelValue,
@@ -244,6 +245,45 @@ export class DatasetLockMother {
}
}
+export class DatasetLabelsMother {
+ static create(): DatasetLabel[] {
+ return [{ value: 'Version 1.0', semanticMeaning: DatasetLabelSemanticMeaning.FILE }]
+ }
+
+ static createDraft(): DatasetLabel[] {
+ return [
+ {
+ value: DatasetLabelValue.UNPUBLISHED,
+ semanticMeaning: DatasetLabelSemanticMeaning.WARNING
+ },
+ { value: DatasetLabelValue.DRAFT, semanticMeaning: DatasetLabelSemanticMeaning.DATASET }
+ ]
+ }
+
+ static createDeaccessioned(): DatasetLabel[] {
+ return [
+ {
+ value: DatasetLabelValue.DEACCESSIONED,
+ semanticMeaning: DatasetLabelSemanticMeaning.DANGER
+ }
+ ]
+ }
+}
+
+export class DatasetCitationMother {
+ static create(): string {
+ return 'Finch, Fiona, 2023, "Darwin\'s Finches", https://doi.org/10.5072/FK2/0YFWKL, Root, V1'
+ }
+
+ static createDraft(): string {
+ return 'Finch, Fiona, 2023, "Darwin\'s Finches", https://doi.org/10.5072/FK2/0YFWKL, Root, DRAFT VERSION'
+ }
+
+ static createDeaccessioned(): string {
+ return 'Finch, Fiona, 2023, "Darwin\'s Finches", https://doi.org/10.5072/FK2/0YFWKL, Root, V1, DEACCESSIONED VERSION'
+ }
+}
+
export class DatasetFileDownloadSizeMother {
static create(props?: Partial): FileDownloadSize {
return new FileDownloadSize(
@@ -284,6 +324,10 @@ export class DatasetMother {
return undefined
}
+ static createMany(count: number): Dataset[] {
+ return Array.from({ length: count }, () => this.create())
+ }
+
static create(props?: Partial): Dataset {
const dataset = {
persistentId: faker.datatype.uuid(),
@@ -293,24 +337,7 @@ export class DatasetMother {
uri: 'https://creativecommons.org/publicdomain/zero/1.0/',
iconUri: 'https://licensebuttons.net/p/zero/1.0/88x31.png'
},
- labels: [
- {
- value: DatasetLabelValue.IN_REVIEW,
- semanticMeaning: faker.helpers.arrayElement(Object.values(DatasetLabelSemanticMeaning))
- },
- {
- value: DatasetLabelValue.EMBARGOED,
- semanticMeaning: faker.helpers.arrayElement(Object.values(DatasetLabelSemanticMeaning))
- },
- {
- value: DatasetLabelValue.UNPUBLISHED,
- semanticMeaning: faker.helpers.arrayElement(Object.values(DatasetLabelSemanticMeaning))
- },
- {
- value: `Version ${faker.lorem.word()}`,
- semanticMeaning: faker.helpers.arrayElement(Object.values(DatasetLabelSemanticMeaning))
- }
- ],
+ labels: DatasetLabelsMother.create(),
summaryFields: [
{
name: MetadataBlockName.CITATION,
diff --git a/tests/component/dataset/domain/models/DatasetPreviewMother.ts b/tests/component/dataset/domain/models/DatasetPreviewMother.ts
new file mode 100644
index 000000000..83e080ff9
--- /dev/null
+++ b/tests/component/dataset/domain/models/DatasetPreviewMother.ts
@@ -0,0 +1,68 @@
+import { faker } from '@faker-js/faker'
+import { DatasetPreview } from '../../../../../src/dataset/domain/models/DatasetPreview'
+import { DatasetCitationMother, DatasetLabelsMother, DatasetVersionMother } from './DatasetMother'
+
+export class DatasetPreviewMother {
+ static createMany(count: number): DatasetPreview[] {
+ return Array.from({ length: count }, () => this.create())
+ }
+
+ static createManyRealistic(count: number): DatasetPreview[] {
+ return Array.from({ length: count }, () => this.createRealistic())
+ }
+
+ static create(props?: Partial): DatasetPreview {
+ const datasetPreview = {
+ persistentId: faker.datatype.uuid(),
+ title: faker.lorem.sentence(),
+ labels: DatasetLabelsMother.create(),
+ isDeaccessioned: faker.datatype.boolean(),
+ thumbnail: faker.datatype.boolean() ? faker.image.imageUrl() : undefined,
+ releaseOrCreateDate: faker.date.past(),
+ version: DatasetVersionMother.create(),
+ citation: DatasetCitationMother.create(),
+ description: faker.lorem.paragraph(),
+ ...props
+ }
+
+ return new DatasetPreview(
+ datasetPreview.persistentId,
+ datasetPreview.title,
+ datasetPreview.version,
+ datasetPreview.citation,
+ datasetPreview.labels,
+ datasetPreview.isDeaccessioned,
+ datasetPreview.releaseOrCreateDate,
+ datasetPreview.description,
+ datasetPreview.thumbnail
+ )
+ }
+
+ static createRealistic(): DatasetPreview {
+ return faker.datatype.boolean() ? this.createDraft() : this.createDeaccessioned()
+ }
+
+ static createDraft(): DatasetPreview {
+ return this.create({
+ isDeaccessioned: false,
+ labels: DatasetLabelsMother.createDraft(),
+ citation: DatasetCitationMother.createDraft()
+ })
+ }
+
+ static createWithThumbnail(): DatasetPreview {
+ return this.create({ thumbnail: faker.image.imageUrl(), isDeaccessioned: false })
+ }
+
+ static createWithNoThumbnail(): DatasetPreview {
+ return this.create({ thumbnail: undefined, isDeaccessioned: false })
+ }
+
+ static createDeaccessioned(): DatasetPreview {
+ return this.create({
+ isDeaccessioned: true,
+ labels: DatasetLabelsMother.createDeaccessioned(),
+ citation: DatasetCitationMother.createDeaccessioned()
+ })
+ }
+}
diff --git a/tests/component/sections/dataset/dataset-citation/CitationThumbnail.spec.tsx b/tests/component/sections/dataset/dataset-citation/CitationThumbnail.spec.tsx
deleted file mode 100644
index c73ab2359..000000000
--- a/tests/component/sections/dataset/dataset-citation/CitationThumbnail.spec.tsx
+++ /dev/null
@@ -1,39 +0,0 @@
-import { DatasetPublishingStatus } from '../../../../../src/dataset/domain/models/Dataset'
-import { CitationThumbnail } from '../../../../../src/sections/dataset/dataset-citation/CitationThumbnail'
-
-describe('CitationThumbnail', () => {
- it('renders the dataset icon when there is no thumbnail', () => {
- cy.customMount(
-
- )
-
- cy.findByLabelText('icon-dataset').should('exist')
- })
-
- it('renders the dataset icon when the dataset is deaccessioned', () => {
- cy.customMount(
-
- )
-
- cy.findByLabelText('icon-dataset').should('exist')
- })
-
- it('renders the dataset thumbnail when there is one and the dataset is not deaccessioned', () => {
- cy.customMount(
-
- )
-
- cy.findByRole('img', { name: 'Dataset title' }).should('exist')
- })
-})
diff --git a/tests/component/sections/dataset/dataset-files/DatasetFiles.spec.tsx b/tests/component/sections/dataset/dataset-files/DatasetFiles.spec.tsx
index 238fde4c4..39b32b584 100644
--- a/tests/component/sections/dataset/dataset-files/DatasetFiles.spec.tsx
+++ b/tests/component/sections/dataset/dataset-files/DatasetFiles.spec.tsx
@@ -8,7 +8,6 @@ import {
FileTag
} from '../../../../../src/files/domain/models/FileCriteria'
import { FilesCountInfoMother } from '../../../files/domain/models/FilesCountInfoMother'
-import { FilePaginationInfo } from '../../../../../src/files/domain/models/FilePaginationInfo'
import {
FileSize,
FileSizeUnit,
@@ -20,6 +19,7 @@ import { SettingMother } from '../../../settings/domain/models/SettingMother'
import { ZipDownloadLimit } from '../../../../../src/settings/domain/models/ZipDownloadLimit'
import { SettingsProvider } from '../../../../../src/sections/settings/SettingsProvider'
import { SettingRepository } from '../../../../../src/settings/domain/repositories/SettingRepository'
+import { FilePaginationInfo } from '../../../../../src/files/domain/models/FilePaginationInfo'
const testFiles = FilePreviewMother.createMany(10)
const datasetPersistentId = 'test-dataset-persistent-id'
@@ -46,7 +46,7 @@ const testFilesCountInfo = FilesCountInfoMother.create({
{ tag: new FileTag('code'), count: 10 }
]
})
-const filePaginationInfo = new FilePaginationInfo(1, 10, 200)
+const paginationInfo: FilePaginationInfo = new FilePaginationInfo(1, 10, 200)
const settingsRepository = {} as SettingRepository
describe('DatasetFiles', () => {
beforeEach(() => {
@@ -421,7 +421,7 @@ describe('DatasetFiles', () => {
'be.calledWith',
datasetPersistentId,
datasetVersion,
- filePaginationInfo,
+ paginationInfo,
new FileCriteria().withSortBy(FileSortByOption.NAME_AZ)
)
})
@@ -441,7 +441,7 @@ describe('DatasetFiles', () => {
'be.calledWith',
datasetPersistentId,
datasetVersion,
- filePaginationInfo,
+ paginationInfo,
new FileCriteria().withFilterByType('image/png')
)
})
@@ -461,7 +461,7 @@ describe('DatasetFiles', () => {
'be.calledWith',
datasetPersistentId,
datasetVersion,
- filePaginationInfo,
+ paginationInfo,
new FileCriteria().withFilterByAccess(FileAccessOption.PUBLIC)
)
})
@@ -481,7 +481,7 @@ describe('DatasetFiles', () => {
'be.calledWith',
datasetPersistentId,
datasetVersion,
- filePaginationInfo,
+ paginationInfo,
new FileCriteria().withFilterByTag('document')
)
})
@@ -500,7 +500,7 @@ describe('DatasetFiles', () => {
'be.calledWith',
datasetPersistentId,
datasetVersion,
- filePaginationInfo,
+ paginationInfo,
new FileCriteria().withSearchText('test')
)
})
diff --git a/tests/component/sections/dataset/dataset-files/files-table/FilesTable.spec.tsx b/tests/component/sections/dataset/dataset-files/files-table/FilesTable.spec.tsx
index 4ff5fda66..e9d10fc6e 100644
--- a/tests/component/sections/dataset/dataset-files/files-table/FilesTable.spec.tsx
+++ b/tests/component/sections/dataset/dataset-files/files-table/FilesTable.spec.tsx
@@ -8,8 +8,8 @@ import { SettingMother } from '../../../../settings/domain/models/SettingMother'
import { ZipDownloadLimit } from '../../../../../../src/settings/domain/models/ZipDownloadLimit'
import { SettingsContext } from '../../../../../../src/sections/settings/SettingsContext'
import styles from '../../../../../../src/sections/dataset/dataset-files/files-table/FilesTable.module.scss'
-import { FilePaginationInfo } from '../../../../../../src/files/domain/models/FilePaginationInfo'
import { FileCriteria } from '../../../../../../src/files/domain/models/FileCriteria'
+import { FilePaginationInfo } from '../../../../../../src/files/domain/models/FilePaginationInfo'
const testFiles = FilePreviewMother.createMany(10)
const paginationInfo = new FilePaginationInfo(1, 10, 200)
diff --git a/tests/component/sections/dataset/dataset-files/files-table/files-info/file-info-cell/file-info-data/FileDate.spec.tsx b/tests/component/sections/dataset/dataset-files/files-table/files-info/file-info-cell/file-info-data/FileDate.spec.tsx
index feb5659ac..a8192fdde 100644
--- a/tests/component/sections/dataset/dataset-files/files-table/files-info/file-info-cell/file-info-data/FileDate.spec.tsx
+++ b/tests/component/sections/dataset/dataset-files/files-table/files-info/file-info-cell/file-info-data/FileDate.spec.tsx
@@ -1,16 +1,13 @@
import { FileDate } from '../../../../../../../../../src/sections/dataset/dataset-files/files-table/file-info/file-info-cell/file-info-data/FileDate'
import { FileDateType } from '../../../../../../../../../src/files/domain/models/FilePreview'
+import { DateHelper } from '../../../../../../../../../src/shared/domain/helpers/DateHelper'
describe('FileDate', () => {
it('renders the date', () => {
const fileDate = new Date('2023-09-18')
const date = { type: FileDateType.PUBLISHED, date: fileDate }
cy.customMount()
- const dateString = fileDate.toLocaleDateString(Intl.DateTimeFormat().resolvedOptions().locale, {
- year: 'numeric',
- month: 'short',
- day: 'numeric'
- })
+ const dateString = DateHelper.toDisplayFormat(fileDate)
cy.findByText(`Published ` + dateString).should('exist')
})
})
diff --git a/tests/component/sections/dataset/dataset-files/useFiles.spec.tsx b/tests/component/sections/dataset/dataset-files/useFiles.spec.tsx
index 498f1fa44..3ed044b5f 100644
--- a/tests/component/sections/dataset/dataset-files/useFiles.spec.tsx
+++ b/tests/component/sections/dataset/dataset-files/useFiles.spec.tsx
@@ -37,7 +37,7 @@ const FilesTableTestComponent = ({ datasetPersistentId }: { datasetPersistentId:
}}>
Sort by name Z-A
- Files count: {paginationInfo.totalFiles}
+ Files count: {paginationInfo.totalItems}
Files total download size: {filesTotalDownloadSize}
diff --git a/tests/component/sections/dataset/dataset-icon/DatasetIcon.spec.tsx b/tests/component/sections/dataset/dataset-icon/DatasetIcon.spec.tsx
new file mode 100644
index 000000000..163759a50
--- /dev/null
+++ b/tests/component/sections/dataset/dataset-icon/DatasetIcon.spec.tsx
@@ -0,0 +1,9 @@
+import { DatasetIcon } from '../../../../../src/sections/dataset/dataset-icon/DatasetIcon'
+
+describe('DatasetIcon', () => {
+ it('renders the dataset icon', () => {
+ cy.customMount()
+
+ cy.findByLabelText('icon-dataset').should('exist')
+ })
+})
diff --git a/tests/component/sections/dataset/dataset-thumbanil/DatasetThumbnail.spec.tsx b/tests/component/sections/dataset/dataset-thumbanil/DatasetThumbnail.spec.tsx
new file mode 100644
index 000000000..1bb47523e
--- /dev/null
+++ b/tests/component/sections/dataset/dataset-thumbanil/DatasetThumbnail.spec.tsx
@@ -0,0 +1,29 @@
+import { DatasetThumbnail } from '../../../../../src/sections/dataset/dataset-thumbnail/DatasetThumbnail'
+
+describe('DatasetThumbnail', () => {
+ it('renders the dataset icon when there is no thumbnail', () => {
+ cy.customMount()
+
+ cy.findByLabelText('icon-dataset').should('exist')
+ })
+
+ it('renders the dataset icon when the dataset is deaccessioned', () => {
+ cy.customMount(
+
+ )
+
+ cy.findByLabelText('icon-dataset').should('exist')
+ })
+
+ it('renders the dataset thumbnail when there is one and the dataset is not deaccessioned', () => {
+ cy.customMount(
+
+ )
+
+ cy.findByRole('img', { name: 'Dataset title' }).should('exist')
+ })
+})
diff --git a/tests/component/sections/hello-dataverse/HelloDataverse.spec.tsx b/tests/component/sections/hello-dataverse/HelloDataverse.spec.tsx
deleted file mode 100644
index 7e1668691..000000000
--- a/tests/component/sections/hello-dataverse/HelloDataverse.spec.tsx
+++ /dev/null
@@ -1,8 +0,0 @@
-import { HelloDataverse } from '../../../../src/sections/hello-dataverse/HelloDataverse'
-
-describe('HelloDataverse page', () => {
- it('renders hello dataverse title', () => {
- cy.customMount()
- cy.findByRole('heading').should('contain.text', 'Hello Dataverse')
- })
-})
diff --git a/tests/component/sections/home/Home.spec.tsx b/tests/component/sections/home/Home.spec.tsx
new file mode 100644
index 000000000..52fe1d34a
--- /dev/null
+++ b/tests/component/sections/home/Home.spec.tsx
@@ -0,0 +1,28 @@
+import { Home } from '../../../../src/sections/home/Home'
+import { DatasetRepository } from '../../../../src/dataset/domain/repositories/DatasetRepository'
+import { DatasetPreviewMother } from '../../dataset/domain/models/DatasetPreviewMother'
+
+const datasetRepository: DatasetRepository = {} as DatasetRepository
+const totalDatasetsCount = 10
+const datasets = DatasetPreviewMother.createMany(totalDatasetsCount)
+describe('Home page', () => {
+ beforeEach(() => {
+ datasetRepository.getAll = cy.stub().resolves(datasets)
+ datasetRepository.getTotalDatasetsCount = cy.stub().resolves(totalDatasetsCount)
+ })
+
+ it('renders Root title', () => {
+ cy.customMount()
+ cy.findByRole('heading').should('contain.text', 'Root')
+ })
+
+ it('renders the datasets list', () => {
+ cy.customMount()
+
+ cy.wrap(datasetRepository.getAll).should('be.calledOnce')
+
+ datasets.forEach((dataset) => {
+ cy.findByText(dataset.title).should('exist')
+ })
+ })
+})
diff --git a/tests/component/sections/home/datasets-list/DatasetsList.spec.tsx b/tests/component/sections/home/datasets-list/DatasetsList.spec.tsx
new file mode 100644
index 000000000..e126ce245
--- /dev/null
+++ b/tests/component/sections/home/datasets-list/DatasetsList.spec.tsx
@@ -0,0 +1,58 @@
+import { DatasetRepository } from '../../../../../src/dataset/domain/repositories/DatasetRepository'
+import { DatasetsList } from '../../../../../src/sections/home/datasets-list/DatasetsList'
+import { DatasetPaginationInfo } from '../../../../../src/dataset/domain/models/DatasetPaginationInfo'
+import { DatasetPreviewMother } from '../../../dataset/domain/models/DatasetPreviewMother'
+
+const datasetRepository: DatasetRepository = {} as DatasetRepository
+const totalDatasetsCount = 200
+const datasets = DatasetPreviewMother.createMany(totalDatasetsCount)
+describe('Datasets List', () => {
+ beforeEach(() => {
+ datasetRepository.getAll = cy.stub().resolves(datasets)
+ datasetRepository.getTotalDatasetsCount = cy.stub().resolves(totalDatasetsCount)
+ })
+
+ it('renders skeleton while loading', () => {
+ cy.customMount()
+
+ cy.findByTestId('datasets-list-skeleton').should('exist')
+ datasets.forEach((dataset) => {
+ cy.findByRole('link', { name: dataset.title }).should('not.exist')
+ })
+ })
+
+ it('renders no datasets message when there are no datasets', () => {
+ datasetRepository.getAll = cy.stub().resolves([])
+ cy.customMount()
+
+ cy.findByText(/This dataverse currently has no datasets./).should('exist')
+ })
+
+ it('renders the datasets list', () => {
+ cy.customMount()
+
+ cy.wrap(datasetRepository.getAll).should(
+ 'be.calledOnceWith',
+ new DatasetPaginationInfo(1, 10, totalDatasetsCount)
+ )
+
+ cy.findByText('1 to 10 of 200 Datasets').should('exist')
+ datasets.forEach((dataset) => {
+ cy.findByText(dataset.title)
+ .should('exist')
+ .should('have.attr', 'href', `/datasets?persistentId=${dataset.persistentId}`)
+ })
+ })
+
+ it('renders the datasets list with the correct header on a page different than the first one ', () => {
+ cy.customMount()
+
+ cy.findByRole('button', { name: '6' }).click()
+
+ cy.wrap(datasetRepository.getAll).should(
+ 'be.calledWith',
+ new DatasetPaginationInfo(1, 10, totalDatasetsCount).goToPage(6)
+ )
+ cy.findByText('51 to 60 of 200 Datasets').should('exist')
+ })
+})
diff --git a/tests/component/sections/home/datasets-list/NoDatasetsMessage.spec.tsx b/tests/component/sections/home/datasets-list/NoDatasetsMessage.spec.tsx
new file mode 100644
index 000000000..c8b580bea
--- /dev/null
+++ b/tests/component/sections/home/datasets-list/NoDatasetsMessage.spec.tsx
@@ -0,0 +1,20 @@
+import { NoDatasetsMessage } from '../../../../../src/sections/home/datasets-list/NoDatasetsMessage'
+
+describe('No Datasets Message', () => {
+ it('renders the message for anonymous user', () => {
+ cy.customMount()
+ cy.findByText(/This dataverse currently has no datasets. Please /).should('exist')
+ cy.findByRole('link', { name: 'log in' }).should(
+ 'have.attr',
+ 'href',
+ '/loginpage.xhtml?redirectPage=%2Fdataverse.xhtml'
+ )
+ })
+
+ it('renders the message for authenticated user', () => {
+ cy.mountAuthenticated()
+ cy.findByText(
+ 'This dataverse currently has no datasets. You can add to it by using the Add Data button on this page.'
+ ).should('exist')
+ })
+})
diff --git a/tests/component/sections/home/datasets-list/dataset-card/DatasetCard.spec.tsx b/tests/component/sections/home/datasets-list/dataset-card/DatasetCard.spec.tsx
new file mode 100644
index 000000000..8b66c37d1
--- /dev/null
+++ b/tests/component/sections/home/datasets-list/dataset-card/DatasetCard.spec.tsx
@@ -0,0 +1,21 @@
+import { DatasetPreviewMother } from '../../../../dataset/domain/models/DatasetPreviewMother'
+import { DatasetCard } from '../../../../../../src/sections/home/datasets-list/dataset-card/DatasetCard'
+import { DateHelper } from '../../../../../../src/shared/domain/helpers/DateHelper'
+import styles from '../../../../../../src/sections/home/datasets-list/dataset-card/DatasetCard.module.scss'
+
+describe('DatasetCard', () => {
+ it('should render the card', () => {
+ const dataset = DatasetPreviewMother.createWithThumbnail()
+ cy.customMount()
+
+ cy.findByText(dataset.title).should('exist')
+
+ cy.findByRole('img', { name: dataset.title }).should('exist')
+ cy.findByText(DateHelper.toDisplayFormat(dataset.releaseOrCreateDate)).should('exist')
+ cy.findByText(/Finch, Fiona, 2023, "Darwin's Finches"/)
+ .should('exist')
+ .parent()
+ .should('have.class', styles['citation-box'])
+ cy.findByText(dataset.abbreviatedDescription).should('exist')
+ })
+})
diff --git a/tests/component/sections/home/datasets-list/dataset-card/DatasetCardHeader.spec.tsx b/tests/component/sections/home/datasets-list/dataset-card/DatasetCardHeader.spec.tsx
new file mode 100644
index 000000000..1e7773dfc
--- /dev/null
+++ b/tests/component/sections/home/datasets-list/dataset-card/DatasetCardHeader.spec.tsx
@@ -0,0 +1,17 @@
+import { DatasetCardHeader } from '../../../../../../src/sections/home/datasets-list/dataset-card/DatasetCardHeader'
+import { DatasetPreviewMother } from '../../../../dataset/domain/models/DatasetPreviewMother'
+
+describe('DatasetCardHeader', () => {
+ it('should render the header', () => {
+ const dataset = DatasetPreviewMother.create()
+ cy.customMount()
+
+ cy.findByText(dataset.title)
+ .should('exist')
+ .should('have.attr', 'href', `/datasets?persistentId=${dataset.persistentId}`)
+ dataset.labels.forEach((label) => {
+ cy.findByText(label.value).should('exist')
+ })
+ cy.findByLabelText('icon-dataset').should('exist')
+ })
+})
diff --git a/tests/component/sections/home/datasets-list/dataset-card/DatasetCardInfo.spec.tsx b/tests/component/sections/home/datasets-list/dataset-card/DatasetCardInfo.spec.tsx
new file mode 100644
index 000000000..5563fe412
--- /dev/null
+++ b/tests/component/sections/home/datasets-list/dataset-card/DatasetCardInfo.spec.tsx
@@ -0,0 +1,28 @@
+import { DatasetPreviewMother } from '../../../../dataset/domain/models/DatasetPreviewMother'
+import { DateHelper } from '../../../../../../src/shared/domain/helpers/DateHelper'
+import styles from '../../../../../../src/sections/home/datasets-list/dataset-card/DatasetCard.module.scss'
+import { DatasetCardInfo } from '../../../../../../src/sections/home/datasets-list/dataset-card/DatasetCardInfo'
+
+describe('DatasetCardInfo', () => {
+ it('should render the dataset info', () => {
+ const dataset = DatasetPreviewMother.createDraft()
+ cy.customMount()
+
+ cy.findByText(DateHelper.toDisplayFormat(dataset.releaseOrCreateDate)).should('exist')
+ cy.findByText(/Finch, Fiona, 2023, "Darwin's Finches"/)
+ .should('exist')
+ .parent()
+ .should('have.class', styles['citation-box'])
+ cy.findByText(dataset.abbreviatedDescription).should('exist')
+ })
+
+ it('should render the citation with the deaccessioned background if the dataset is deaccessioned', () => {
+ const dataset = DatasetPreviewMother.createDeaccessioned()
+ cy.customMount()
+
+ cy.findByText(/Finch, Fiona, 2023, "Darwin's Finches"/)
+ .should('exist')
+ .parent()
+ .should('have.class', styles['citation-box-deaccessioned'])
+ })
+})
diff --git a/tests/component/sections/home/datasets-list/dataset-card/DatasetCardThumbnail.spec.tsx b/tests/component/sections/home/datasets-list/dataset-card/DatasetCardThumbnail.spec.tsx
new file mode 100644
index 000000000..e261e32db
--- /dev/null
+++ b/tests/component/sections/home/datasets-list/dataset-card/DatasetCardThumbnail.spec.tsx
@@ -0,0 +1,25 @@
+import { DatasetPreviewMother } from '../../../../dataset/domain/models/DatasetPreviewMother'
+import { DatasetCardThumbnail } from '../../../../../../src/sections/home/datasets-list/dataset-card/DatasetCardThumbnail'
+
+describe('DatasetCardThumbnail', () => {
+ it('should render the thumbnail', () => {
+ const dataset = DatasetPreviewMother.createWithThumbnail()
+ cy.customMount()
+
+ cy.findByRole('img', { name: dataset.title })
+ .should('exist')
+ .parent('a')
+ .should('have.attr', 'href', `/datasets?persistentId=${dataset.persistentId}`)
+ })
+
+ it('should render the placeholder if the dataset has no thumbnail', () => {
+ const dataset = DatasetPreviewMother.createWithNoThumbnail()
+ cy.customMount()
+
+ cy.findByRole('img', { name: 'icon-dataset' })
+ .should('exist')
+ .parent()
+ .parent('a')
+ .should('have.attr', 'href', `/datasets?persistentId=${dataset.persistentId}`)
+ })
+})
diff --git a/tests/component/sections/layout/header/Header.spec.tsx b/tests/component/sections/layout/header/Header.spec.tsx
index a3c208624..9db242389 100644
--- a/tests/component/sections/layout/header/Header.spec.tsx
+++ b/tests/component/sections/layout/header/Header.spec.tsx
@@ -11,6 +11,17 @@ describe('Header component', () => {
userRepository.removeAuthenticated = cy.stub().resolves()
})
+ it('displays the brand', () => {
+ cy.customMount(
+
+
+
+ )
+
+ cy.findByRole('link', { name: /Dataverse/ }).should('exist')
+ cy.findByRole('link').should('have.attr', 'href', '/spa/')
+ })
+
it('displays the user name when the user is logged in', () => {
cy.customMount(
diff --git a/tests/component/sections/session/useSession.spec.tsx b/tests/component/sections/session/useSession.spec.tsx
index bd5f678b4..a1f15f1d0 100644
--- a/tests/component/sections/session/useSession.spec.tsx
+++ b/tests/component/sections/session/useSession.spec.tsx
@@ -32,11 +32,14 @@ describe('useSession', () => {
it('should unset user after calling logOut on repository', () => {
function TestComponent() {
const { user, logout } = useSession()
+ const onLogoutClick = () => {
+ void logout()
+ }
return (
{user ? {user.name} : <>>}
-
+
)
}
diff --git a/tests/component/sections/shared/citation/CitationDescription.spec.tsx b/tests/component/sections/shared/citation/CitationDescription.spec.tsx
new file mode 100644
index 000000000..ce5b1a57e
--- /dev/null
+++ b/tests/component/sections/shared/citation/CitationDescription.spec.tsx
@@ -0,0 +1,12 @@
+import { CitationDescription } from '../../../../../src/sections/shared/citation/CitationDescription'
+
+describe('CitationDescription', () => {
+ it('renders the citation', () => {
+ const citation =
+ 'Finch, Fiona, 2023, "Darwin\'s Finches", https://doi.org/10.5072/FK2/0YFWKL, Root, V1'
+ cy.customMount()
+
+ cy.findByText(/Finch, Fiona, 2023, "Darwin's Finches",/).should('exist')
+ cy.findByRole('link', { name: 'https://doi.org/10.5072/FK2/0YFWKL' }).should('exist')
+ })
+})
diff --git a/tests/component/sections/dataset/dataset-files/files-pagination/PageNumbersButtonsWithEllipsis.spec.tsx b/tests/component/sections/shared/pagination/PageNumbersButtonsWithEllipsis.spec.tsx
similarity index 95%
rename from tests/component/sections/dataset/dataset-files/files-pagination/PageNumbersButtonsWithEllipsis.spec.tsx
rename to tests/component/sections/shared/pagination/PageNumbersButtonsWithEllipsis.spec.tsx
index 0b4fffc39..eb644b64b 100644
--- a/tests/component/sections/dataset/dataset-files/files-pagination/PageNumbersButtonsWithEllipsis.spec.tsx
+++ b/tests/component/sections/shared/pagination/PageNumbersButtonsWithEllipsis.spec.tsx
@@ -1,4 +1,4 @@
-import { PageNumbersButtonsWithEllipsis } from '../../../../../../src/sections/dataset/dataset-files/files-pagination/PageNumbersButtonsWithEllipsis'
+import { PageNumbersButtonsWithEllipsis } from '../../../../../src/sections/shared/pagination/PageNumbersButtonsWithEllipsis'
let goToPage: (pageIndex: number) => void
const selectedPageIndex = 3
diff --git a/tests/component/sections/dataset/dataset-files/files-pagination/FilesPagination.spec.tsx b/tests/component/sections/shared/pagination/PaginationControls.spec.tsx
similarity index 63%
rename from tests/component/sections/dataset/dataset-files/files-pagination/FilesPagination.spec.tsx
rename to tests/component/sections/shared/pagination/PaginationControls.spec.tsx
index df2a50ce6..4ca16340c 100644
--- a/tests/component/sections/dataset/dataset-files/files-pagination/FilesPagination.spec.tsx
+++ b/tests/component/sections/shared/pagination/PaginationControls.spec.tsx
@@ -1,23 +1,27 @@
-import { FilesPagination } from '../../../../../../src/sections/dataset/dataset-files/files-pagination/FilesPagination'
-import { FilePaginationInfo } from '../../../../../../src/files/domain/models/FilePaginationInfo'
+import { PaginationControls } from '../../../../../src/sections/shared/pagination/PaginationControls'
+import { PaginationInfo } from '../../../../../src/shared/domain/models/PaginationInfo'
+import { FilePaginationInfo } from '../../../../../src/files/domain/models/FilePaginationInfo'
+import { DatasetPaginationInfo } from '../../../../../src/dataset/domain/models/DatasetPaginationInfo'
-let paginationInfo: FilePaginationInfo
+let paginationInfo: PaginationInfo
const page = 3
const pageSize = 10
const total = 200
-describe('FilesPagination', () => {
+describe('PaginationControls', () => {
beforeEach(() => {
cy.viewport(1000, 1000)
- paginationInfo = new FilePaginationInfo(page, pageSize, total)
+ paginationInfo = new PaginationInfo(
+ page,
+ pageSize,
+ total
+ )
})
it('clicking on the first page button calls goToPage 1', () => {
const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
cy.customMount(
-
)
@@ -29,10 +33,8 @@ describe('FilesPagination', () => {
it('clicking on the previous page button calls goToPreviousPage', () => {
const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
cy.customMount(
-
)
@@ -47,10 +49,8 @@ describe('FilesPagination', () => {
it('clicking on a page button calls goToPage with the correct number', () => {
const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
cy.customMount(
-
)
@@ -62,10 +62,8 @@ describe('FilesPagination', () => {
it('clicking on the next page button calls goToNextPage', () => {
const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
cy.customMount(
-
)
@@ -77,10 +75,8 @@ describe('FilesPagination', () => {
it('clicking on the last page button calls setPageIndex with the last index', () => {
const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
cy.customMount(
-
)
@@ -92,15 +88,13 @@ describe('FilesPagination', () => {
it('selecting a page size calls setPageSize with the selected value', () => {
const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
cy.customMount(
-
)
- cy.findByLabelText('Files per page').select('50')
+ cy.findByLabelText('Items per page').select('50')
cy.wrap(onPaginationInfoChange).should('have.been.calledWith', paginationInfo.withPageSize(50))
cy.findByRole('button', { name: 'Last' }).click()
@@ -109,4 +103,17 @@ describe('FilesPagination', () => {
paginationInfo.withPageSize(50).goToPage(4)
)
})
+
+ it('does not show the page size selector if the prop is false', () => {
+ const onPaginationInfoChange = cy.stub().as('onPaginationInfoChange')
+ cy.customMount(
+
+ )
+
+ cy.findByLabelText('Items per page').should('not.exist')
+ })
})
diff --git a/tests/e2e-integration/e2e/sections/hello-dataverse/HelloDataverse.spec.ts b/tests/e2e-integration/e2e/sections/home/Home.spec.ts
similarity index 59%
rename from tests/e2e-integration/e2e/sections/hello-dataverse/HelloDataverse.spec.ts
rename to tests/e2e-integration/e2e/sections/home/Home.spec.ts
index 4e364a2ea..1f85bc181 100644
--- a/tests/e2e-integration/e2e/sections/hello-dataverse/HelloDataverse.spec.ts
+++ b/tests/e2e-integration/e2e/sections/home/Home.spec.ts
@@ -1,23 +1,23 @@
-describe('Hello Dataverse', () => {
+describe('Home Page', () => {
it('successfully loads', () => {
cy.visit('/spa')
- cy.findAllByText(/Hello Dataverse/i).should('exist')
+ cy.findAllByText(/Root/i).should('exist')
})
it('log in Dataverse Admin user', () => {
cy.loginAsAdmin('/spa')
- cy.findAllByText(/Hello Dataverse/i).should('exist')
+ cy.findAllByText(/Root/i).should('exist')
cy.findByText(/Dataverse Admin/i).should('exist')
})
it('log out Dataverse Admin user', () => {
cy.loginAsAdmin('/spa')
- cy.findAllByText(/Hello Dataverse/i).should('exist')
+ cy.findAllByText(/Root/i).should('exist')
cy.findByText(/Dataverse Admin/i).click()
- cy.findByRole('link', { name: /Log Out/i }).click()
+ cy.findByRole('button', { name: /Log Out/i }).click()
cy.findByText(/Dataverse Admin/i).should('not.exist')
})
})
diff --git a/tests/e2e-integration/integration/files/FileJSDataverseRepository.spec.ts b/tests/e2e-integration/integration/files/FileJSDataverseRepository.spec.ts
index 774bea907..92825b822 100644
--- a/tests/e2e-integration/integration/files/FileJSDataverseRepository.spec.ts
+++ b/tests/e2e-integration/integration/files/FileJSDataverseRepository.spec.ts
@@ -14,7 +14,6 @@ import {
import chai from 'chai'
import chaiAsPromised from 'chai-as-promised'
import { DatasetJSDataverseRepository } from '../../../../src/dataset/infrastructure/repositories/DatasetJSDataverseRepository'
-import { FilePaginationInfo } from '../../../../src/files/domain/models/FilePaginationInfo'
import {
FileAccessOption,
FileCriteria,
@@ -25,6 +24,7 @@ import { DatasetHelper } from '../../shared/datasets/DatasetHelper'
import { FileData, FileHelper } from '../../shared/files/FileHelper'
import { FilesCountInfo } from '../../../../src/files/domain/models/FilesCountInfo'
import { DatasetVersionMother } from '../../../component/dataset/domain/models/DatasetMother'
+import { FilePaginationInfo } from '../../../../src/files/domain/models/FilePaginationInfo'
chai.use(chaiAsPromised)
const expect = chai.expect