mirror of
https://github.com/BrenBroZAYT/gameyfin.git
synced 2026-06-13 16:40:01 +00:00
3a932d953f
* chore: bump version to v2.4.0-preview * Bump actions/cache from 4 to 5 (#865) Bumps [actions/cache](https://github.com/actions/cache) from 4 to 5. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/cache dependency-version: '5' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Increase maximum DB connection pool size (#876) Increase DB connection timeout * Disable length limit for DB field PLUGIN_CONFIG.value (#875) * Bump actions/cache from 4 to 5 (#871) Bumps [actions/cache](https://github.com/actions/cache) from 4 to 5. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/cache dependency-version: '5' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump actions/download-artifact from 7 to 8 (#882) Bumps [actions/download-artifact](https://github.com/actions/download-artifact) from 7 to 8. - [Release notes](https://github.com/actions/download-artifact/releases) - [Commits](https://github.com/actions/download-artifact/compare/v7...v8) --- updated-dependencies: - dependency-name: actions/download-artifact dependency-version: '8' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump actions/upload-artifact from 6 to 7 (#881) Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 6 to 7. - [Release notes](https://github.com/actions/upload-artifact/releases) - [Commits](https://github.com/actions/upload-artifact/compare/v6...v7) --- updated-dependencies: - dependency-name: actions/upload-artifact dependency-version: '7' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump actions/cache from 4 to 5 (#878) Bumps [actions/cache](https://github.com/actions/cache) from 4 to 5. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/cache dependency-version: '5' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Dont perform scans if no metadata plugins are enabled (#877) * Dont perform scans if no metadata plugins are enabled * Fix tests * Add PluginServiceTest.kt * Fix Sonar finding * Fix malformed external links (#886) * Fix external links being treated as internal * chore: bump version to v856-malformed-external-links-preview * Update JVM in Dockerfile to Java 25 * Revert incorrect version update * Allow loading .jar plugins in development mode (#885) * Allow loading .jar plugins in development mode * Remove unnecessary mock * Fix unit test * Add unit tests * Fix/879 add info and reset to config options (#887) * Fix gog.sh script * Add "description" property to ConfigProperties.kt Add InfoPopup.tsx and ResetToDefaultButton.tsx in UI * Bump actions/download-artifact from 7 to 8 (#891) Bumps [actions/download-artifact](https://github.com/actions/download-artifact) from 7 to 8. - [Release notes](https://github.com/actions/download-artifact/releases) - [Commits](https://github.com/actions/download-artifact/compare/v7...v8) --- updated-dependencies: - dependency-name: actions/download-artifact dependency-version: '8' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump actions/cache from 4 to 5 (#890) Bumps [actions/cache](https://github.com/actions/cache) from 4 to 5. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/cache dependency-version: '5' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Bump actions/upload-artifact from 6 to 7 (#889) Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 6 to 7. - [Release notes](https://github.com/actions/upload-artifact/releases) - [Commits](https://github.com/actions/upload-artifact/compare/v6...v7) --- updated-dependencies: - dependency-name: actions/upload-artifact dependency-version: '7' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * Improve memory usage and performance (#888) mprove memory usage and performance by: * Using AOT cache * Using tuned JAVA_OPTIONS * Session timeout * Jetty threadpool * DB batch size * DB pool size * Library scanning * Make scan-concurrency configurable * Log retention * Off-load image processing to disk instead of RAM * Fix bug in PluginState * Update dependency version for ksp * Fix race condition preventing plugins from starting * Show remaining time (estimation) for library scans * Add unit test for plugin loading bugfix * Add unit tests for ImageService calculateBlurHash * Make username claim configurable (#895) Add fallbacks to resolve username * Fix sonar issues (#894) * Add custom "/sonar" command to GH copilot * Add Sonar plugin integration * Fix issues reported by Sonar * Ignore Sonar warning about AES/ECB * Add unit tests for GameyfinPluginManager * Add unit tests for GameService * Add more unit tests for GameService * Improve library card layout (#896) * Fix title not being centered * Add buttons to scan all libraries * Disable AVX for AOT cache training * Improve AOT cache training * Fix tests * Change output type of Docker Build CI action * Increase MAX_WAIT of aot-training to 5min * Optimize Docker CI pipeline * Add Sonar badges to README.md * Add custom metrics (downloads & scans) * Optimize DB connection & add cache for images * Adjusted logging on startup * * Show message on start page when no libraries/games are available * Disable "Scan" buttons when no metadata plugin is enabled * Fix thread pinning causing deadlocks * Pre-populate image cache at startup * Show "Loading" spinner while loading * Optimize static file serving (images) * Switch back to Tomcat (from Jetty) --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
143 lines
6.5 KiB
TypeScript
143 lines
6.5 KiB
TypeScript
import {CoverRow} from "Frontend/components/general/covers/CoverRow";
|
|
import {useSnapshot} from "valtio/react";
|
|
import {libraryState} from "Frontend/state/LibraryState";
|
|
import {gameState} from "Frontend/state/GameState";
|
|
import React, {useMemo} from "react";
|
|
import LibraryDto from "Frontend/generated/org/gameyfin/app/libraries/dto/LibraryDto";
|
|
import {collectionState} from "Frontend/state/CollectionState";
|
|
import CollectionDto from "Frontend/generated/org/gameyfin/app/collections/dto/CollectionDto";
|
|
import {StartPageDisplayCard} from "Frontend/components/general/cards/StartPageDisplayCard";
|
|
import {Link, Spinner} from "@heroui/react";
|
|
import {CaretRightIcon, FolderOpenIcon} from "@phosphor-icons/react";
|
|
import {useAuth} from "Frontend/util/auth";
|
|
import {isAdmin} from "Frontend/util/utils";
|
|
|
|
export default function HomeView() {
|
|
const auth = useAuth();
|
|
const librariesState = useSnapshot(libraryState);
|
|
const collectionsState = useSnapshot(collectionState);
|
|
const gamesState = useSnapshot(gameState);
|
|
const gamesByLibrary = gamesState.gamesByLibraryId;
|
|
const gamesByCollection = gamesState.gamesByCollectionId;
|
|
|
|
const filteredAndSortedLibraries = useMemo(() =>
|
|
librariesState.sorted
|
|
.filter(library => library.metadata!.displayOnHomepage)
|
|
.filter(library =>
|
|
gamesByLibrary[library.id] && gamesByLibrary[library.id].length > 0
|
|
),
|
|
[librariesState.sorted, gamesByLibrary]
|
|
);
|
|
|
|
const filteredAndSortedCollections = useMemo(() =>
|
|
collectionsState.sorted
|
|
.filter(collection => collection.metadata!.displayOnHomepage)
|
|
.filter(collection =>
|
|
gamesByCollection[collection.id] && gamesByCollection[collection.id].length > 0
|
|
),
|
|
[collectionsState.sorted, gamesByCollection]
|
|
);
|
|
|
|
// Sort games by date added (newest first) for libraries
|
|
const getSortedLibraryGames = (libraryId: number) => {
|
|
const games = gamesByLibrary[libraryId] || [];
|
|
return [...games].sort((a, b) => {
|
|
const dateA = new Date(a.createdAt).getTime();
|
|
const dateB = new Date(b.createdAt).getTime();
|
|
return dateB - dateA; // Descending order (newest first)
|
|
});
|
|
};
|
|
|
|
// Sort games by date added (newest first) for collections
|
|
const getSortedCollectionGames = (collection: CollectionDto) => {
|
|
const games = gamesByCollection[collection.id] || [];
|
|
const gamesAddedAt = collection.metadata?.gamesAddedAt || {};
|
|
|
|
return [...games].sort((a, b) => {
|
|
const dateA = gamesAddedAt[a.id.toString()]
|
|
? new Date(gamesAddedAt[a.id.toString()]).getTime()
|
|
: 0;
|
|
const dateB = gamesAddedAt[b.id.toString()]
|
|
? new Date(gamesAddedAt[b.id.toString()]).getTime()
|
|
: 0;
|
|
return dateB - dateA; // Descending order (newest first)
|
|
});
|
|
};
|
|
|
|
const hasNoContent = filteredAndSortedLibraries.length === 0 && filteredAndSortedCollections.length === 0;
|
|
const allStatesLoaded = librariesState.isLoaded && collectionsState.isLoaded && gamesState.isLoaded;
|
|
|
|
if (!allStatesLoaded) {
|
|
return (
|
|
<div className="flex flex-col items-center justify-center h-[70vh] text-center gap-4">
|
|
<Spinner size="lg"/>
|
|
<p className="text-xl font-semibold text-default-600">Loading...</p>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (hasNoContent) {
|
|
return (
|
|
<div className="flex flex-col items-center justify-center h-[70vh] text-center gap-4">
|
|
<FolderOpenIcon size={64} className="text-default-300"/>
|
|
<p className="text-xl font-semibold text-default-600">Nothing here yet</p>
|
|
{isAdmin(auth) ? (
|
|
<>
|
|
<p className="text-default-400 max-w-lg">
|
|
Get started by adding libraries and games in the{" "}
|
|
<Link href="/administration/games" underline="always">
|
|
administration panel
|
|
</Link>.
|
|
</p>
|
|
</>
|
|
) : (
|
|
<>
|
|
<p className="text-default-400 max-w-md">
|
|
There is currently no content available. Check back later!
|
|
</p>
|
|
</>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className="w-full">
|
|
<div className="flex flex-col gap-4">
|
|
{(filteredAndSortedLibraries.length + filteredAndSortedCollections.length > 0) &&
|
|
<div className="flex flex-col gap-2">
|
|
<Link href="/search" className="flex flex-row gap-1 w-fit items-baseline" color="foreground"
|
|
underline="hover">
|
|
<p className="text-2xl font-bold mb-4">Your games</p>
|
|
<CaretRightIcon weight="bold" size={16}/>
|
|
</Link>
|
|
<div className="grid gap-4 grid-cols-[repeat(auto-fill,minmax(353px,1fr))]">
|
|
{filteredAndSortedLibraries.length > 0 &&
|
|
filteredAndSortedLibraries.map((library: LibraryDto) => (
|
|
<StartPageDisplayCard key={library.id} item={library}/>
|
|
))
|
|
}
|
|
{filteredAndSortedCollections.length > 0 &&
|
|
filteredAndSortedCollections.map((collection: CollectionDto) => (
|
|
<StartPageDisplayCard key={collection.id} item={collection}/>
|
|
))
|
|
}
|
|
</div>
|
|
</div>
|
|
}
|
|
{filteredAndSortedLibraries.map((library) => (
|
|
<CoverRow key={library.id} title={library.name}
|
|
games={getSortedLibraryGames(library.id)}
|
|
link={"/library/" + library.id}
|
|
/>
|
|
))}
|
|
{filteredAndSortedCollections.map((collection) => (
|
|
<CoverRow key={collection.id} title={collection.name}
|
|
games={getSortedCollectionGames(collection)}
|
|
link={"/collection/" + collection.id}
|
|
/>
|
|
))}
|
|
</div>
|
|
</div>
|
|
);
|
|
} |