Add tagged resources page.

This commit is contained in:
2025-05-13 17:47:15 +08:00
parent 0dd2143664
commit f1345f9a0c
8 changed files with 88 additions and 8 deletions

View File

@@ -8,6 +8,7 @@ import SearchPage from "./pages/search_page.tsx";
import ResourcePage from "./pages/resource_details_page.tsx";
import "./i18n.ts"
import ManagePage from "./pages/manage_page.tsx";
import TaggedResourcesPage from "./pages/tagged_resources_page.tsx";
export default function App() {
return (
@@ -21,6 +22,7 @@ export default function App() {
<Route path={"/search"} element={<SearchPage/>} />
<Route path={"/resources/:id"} element={<ResourcePage/>}/>
<Route path={"/manage"} element={<ManagePage/>}/>
<Route path={"/tag/:tag"} element={<TaggedResourcesPage/>}/>
</Route>
</Routes>
</BrowserRouter>

View File

@@ -17,7 +17,7 @@ export default function ResourcesView({loader}: {loader: (page: number) => Promi
isLoadingRef.current = true
const res = await loader(pageRef.current)
if (!res.success) {
showToast({message: "Error loading resources", type: "error"})
showToast({message: res.message, type: "error"})
} else {
isLoadingRef.current = false
pageRef.current = pageRef.current + 1

View File

@@ -125,3 +125,8 @@ body {
top: -100%;
}
}
.text-md {
font-size: 1rem;
line-height: 1.5rem;
}

View File

@@ -308,6 +308,23 @@ class Network {
}
}
async getResourcesByTag(tag: string, page: number): Promise<PageResponse<Resource>> {
try {
const response = await axios.get(`${this.apiBaseUrl}/resource/tag/${tag}`, {
params: {
page
}
})
return response.data
} catch (e: any) {
console.error(e)
return {
success: false,
message: e.toString(),
}
}
}
async searchResources(keyword: string, page: number): Promise<PageResponse<Resource>> {
try {
const response = await axios.get(`${this.apiBaseUrl}/resource/search`, {

View File

@@ -1,4 +1,4 @@
import {useParams} from "react-router";
import {useNavigate, useParams} from "react-router";
import {createContext, useCallback, useContext, useEffect, useRef, useState} from "react";
import {ResourceDetails, RFile, Storage, Comment} from "../network/models.ts";
import {network} from "../network/network.ts";
@@ -49,6 +49,8 @@ export default function ResourcePage() {
}
}, [id])
const navigate = useNavigate()
if (isNaN(id)) {
return <div className="alert alert-error shadow-lg">
<div>
@@ -84,7 +86,9 @@ export default function ResourcePage() {
<p className={"px-4 pt-2"}>
{
resource.tags.map((e) => {
return <span key={e.id} className="badge badge-primary mr-2 text-sm">{e.name}</span>
return <span key={e.id} className="badge badge-primary mr-2 text-sm cursor-pointer" onClick={() => {
navigate(`/tag/${e.name}`);
}}>{e.name}</span>
})
}
</p>

View File

@@ -0,0 +1,23 @@
import {useParams} from "react-router";
import {ErrorAlert} from "../components/alert.tsx";
import ResourcesView from "../components/resources_view.tsx";
import {network} from "../network/network.ts";
export default function TaggedResourcesPage() {
const {tag} = useParams()
if (!tag) {
return <div>
<ErrorAlert message={"Tag not found"}/>
</div>
}
return <div>
<h1 className={"text-2xl pt-4 pb-2 px-4"}>
{tag}
</h1>
<ResourcesView loader={(page) => {
return network.getResourcesByTag(tag, page)
}}></ResourcesView>
</div>
}

View File

@@ -99,6 +99,34 @@ func handleListResources(c fiber.Ctx) error {
})
}
func handleListResourcesWithTag(c fiber.Ctx) error {
tag := c.Params("tag")
if tag == "" {
return model.NewRequestError("Tag is required")
}
pageStr := c.Query("page")
if pageStr == "" {
pageStr = "1"
}
page, err := strconv.Atoi(pageStr)
if err != nil {
return model.NewRequestError("Invalid page number")
}
resources, totalPages, err := service.GetResourcesWithTag(tag, page)
if err != nil {
return err
}
if resources == nil {
resources = []model.ResourceView{}
}
return c.Status(fiber.StatusOK).JSON(model.PageResponse[model.ResourceView]{
Success: true,
Data: resources,
TotalPages: totalPages,
Message: "Resources retrieved successfully",
})
}
func handleSearchResources(c fiber.Ctx) error {
query := c.Query("keyword")
if query == "" {
@@ -135,5 +163,6 @@ func AddResourceRoutes(api fiber.Router) {
resource.Get("/", handleListResources)
resource.Get("/:id", handleGetResource)
resource.Delete("/:id", handleDeleteResource)
resource.Get("/tag/:tag", handleListResourcesWithTag)
}
}

View File

@@ -41,7 +41,7 @@ func GetResourceList(page, pageSize int) ([]model.Resource, int, error) {
return nil, 0, err
}
totalPages := int(total) / pageSize
totalPages := (total + int64(pageSize) - 1) / int64(pageSize)
return resources, int(totalPages), nil
}
@@ -110,7 +110,7 @@ func Search(query string, page, pageSize int) ([]model.Resource, int, error) {
if endIndex > len(resource) {
endIndex = len(resource)
}
totalPages := len(resource) / pageSize
totalPages := (len(resource) + pageSize - 1) / pageSize
result := make([]model.Resource, 0, endIndex-startIndex)
for i := startIndex; i < endIndex; i++ {
@@ -157,13 +157,13 @@ func GetResourceByTag(tagID uint, page int, pageSize int) ([]model.Resource, int
total = db.Model(&model.Tag{}).Where("id = ?", tagID).Association("Resources").Count()
if err := db.Model(&model.Tag{}).Where("id = ?", tagID).Preload("User").Preload("Resources", func(tx *gorm.DB) *gorm.DB {
return tx.Offset((page - 1) * pageSize).Limit(pageSize)
if err := db.Model(&model.Tag{}).Where("id = ?", tagID).Preload("Resources", func(tx *gorm.DB) *gorm.DB {
return tx.Offset((page - 1) * pageSize).Limit(pageSize).Preload("Tags").Preload("User").Preload("Images").Order("created_at DESC")
}).First(&tag).Error; err != nil {
return nil, 0, err
}
totalPages := int(total) / pageSize
totalPages := (int(total) + pageSize - 1) / pageSize
return tag.Resources, totalPages, nil
}