loadCategories(); // 获取一级分类和对应的书籍 $categoryData = []; foreach ($categories as $category) { if ($category['level'] == 1) { $children = $this->subCategories($categories, $category['id']); $categoryData[] = [ 'category' => $category, 'children' => $children, ]; } } $recentBooks = $this->getRecent(); return view('library.index', compact( 'categoryData', 'categories', 'recentBooks' )); } public function index() { $categories = $this->loadCategories(); // 获取一级分类和对应的书籍 $categoryData = []; foreach ($categories as $category) { if ($category['level'] == 1) { $children = $this->subCategories($categories, $category['id']); $categoryData[] = [ 'category' => $category, 'children' => $children, ]; } } return view('library.index', compact('categoryData', 'categories')); } // app/Http/Controllers/Library/CategoryController.php // category() 方法修改版 // 变更: // 1. $id 改为可选参数,无参数时显示顶级分类(首页复用) // 2. 新增 $filters 过滤参数(type / lang / author / sort) // 3. 新增右边栏数据:$recommended / $activeAuthors // 4. 新增 $filterOptions(过滤器选项 + 计数) // 5. 新增 $totalCount public function category(?int $id = null) { $categories = $this->loadCategories(); // ── 当前分类 ────────────────────────────────────────── if ($id) { $currentCategory = collect($categories)->firstWhere('id', $id); if (!$currentCategory) { abort(404); } $breadcrumbs = $this->getBreadcrumbs($currentCategory, $categories); } else { // 首页:虚拟顶级分类 $currentCategory = ['id' => null, 'name' => '三藏']; $breadcrumbs = []; } // ── 子分类 ───────────────────────────────────────────── $subCategories = array_values(array_filter( $categories, fn($cat) => $cat['parent_id'] == $id )); // ── 过滤参数 ──────────────────────────────────────────── $selectedType = request('type', 'all'); $selectedLang = request('lang', 'all'); $selectedAuthor = request('author', 'all'); $selectedSort = request('sort', 'updated_at'); $selected = [ 'type' => $selectedType, 'lang' => $selectedLang, 'author' => $selectedAuthor, 'sort' => $selectedSort, ]; // ── 书籍列表(过滤+排序,真实实现替换此处) ────────────── $categoryBooks = $this->getBooks($categories, $id, $selected); // TODO: 将 $selected 传入 getBooks() 做实际过滤 $totalCount = count($categoryBooks); // ── 过滤器选项(mock,真实实现从书籍数据聚合) ──────────── $filterOptions = [ 'types' => [ ['value' => 'all', 'label' => '全部', 'count' => $totalCount], ['value' => 'original', 'label' => '原文', 'count' => 0], ['value' => 'translation', 'label' => '译文', 'count' => 0], ['value' => 'nissaya', 'label' => 'Nissaya', 'count' => 0], ], 'languages' => [ ['value' => 'all', 'label' => '全部', 'count' => $totalCount], ['value' => 'zh-Hans', 'label' => '简体中文', 'count' => 0], ['value' => 'zh-Hant', 'label' => '繁体中文', 'count' => 0], ['value' => 'pi', 'label' => '巴利语', 'count' => 0], ['value' => 'en', 'label' => '英语', 'count' => 0], ], 'authors' => $this->getAuthorOptions($categoryBooks), ]; // ── 右边栏:本周推荐(mock) ──────────────────────────── $recommended = [ ['id' => 1, 'title' => '相应部·因缘篇', 'category' => '经藏'], ['id' => 2, 'title' => '法句经', 'category' => '经藏'], ['id' => 3, 'title' => '清净道论', 'category' => '注释'], ['id' => 4, 'title' => '律藏·波罗夷', 'category' => '律藏'], ['id' => 5, 'title' => '长部·梵网经', 'category' => '经藏'], ]; // ── 右边栏:活跃译者(mock) ──────────────────────────── $activeAuthors = [ [ 'name' => 'Bhikkhu Bodhi', 'avatar' => null, 'color' => '#2d5a8e', 'initials' => 'BB', 'count' => 24, ], [ 'name' => 'Bhikkhu Sujato', 'avatar' => null, 'color' => '#5a2d8e', 'initials' => 'BS', 'count' => 18, ], [ 'name' => 'Buddhaghosa', 'avatar' => null, 'color' => '#8e5a2d', 'initials' => 'BG', 'count' => 12, ], [ 'name' => 'Bhikkhu Brahmali', 'avatar' => null, 'color' => '#2d8e5a', 'initials' => 'BR', 'count' => 9, ], ]; $types = $this->types(); return view('library.tipitaka.category', compact( 'currentCategory', 'subCategories', 'categoryBooks', 'breadcrumbs', 'types', 'selected', 'filterOptions', 'totalCount', 'recommended', 'activeAuthors', )); } // ── 辅助:从书籍列表聚合作者选项(mock,真实实现替换) ───────── private function getAuthorOptions(array $books): array { // TODO: 从 $books 聚合真实作者列表 return [ ['value' => 'all', 'label' => '全部作者', 'count' => count($books)], ['value' => 'bhikkhu-bodhi', 'label' => 'Bhikkhu Bodhi', 'count' => 0], ['value' => 'bhikkhu-sujato', 'label' => 'Bhikkhu Sujato', 'count' => 0], ['value' => 'buddhaghosa', 'label' => 'Buddhaghosa', 'count' => 0], ['value' => 'bhikkhu-brahmali', 'label' => 'Bhikkhu Brahmali', 'count' => 0], ]; } private function types() { return [ ['id' => '1', 'name' => 'sutta'], ['id' => '48', 'name' => 'vinaya'], ['id' => '66', 'name' => 'abhidhamma'], ['id' => '82', 'name' => 'añña'] ]; } private function subCategories($categories, int $id) { return array_filter($categories, function ($cat) use ($id) { return $cat['parent_id'] == $id; }); } private function getRecent() { return [ [ 'id' => 'book-001', 'title' => '相应部·因缘篇', 'author' => 'Bhikkhu Bodhi', 'cover' => null, // 无封面时显示渐变 'cover_gradient' => 'linear-gradient(135deg, #2d5a8e 0%, #1a3a5c 100%)', 'updated_at' => '2小时前', 'is_new' => true, // true=新增, false=更新 'category' => '经藏', ], [ 'id' => 'book-002', 'title' => '长部·梵网经', 'author' => 'Bhikkhu Sujato', 'cover' => null, 'cover_gradient' => 'linear-gradient(135deg, #5a2d8e 0%, #3a1a5c 100%)', 'updated_at' => '昨天', 'is_new' => false, 'category' => '经藏', ], [ 'id' => 'book-003', 'title' => '法句经注', 'author' => 'Buddhaghosa', 'cover' => null, 'cover_gradient' => 'linear-gradient(135deg, #8e5a2d 0%, #5c3a1a 100%)', 'updated_at' => '3天前', 'is_new' => false, 'category' => '注释', ], [ 'id' => 'book-004', 'title' => '律藏·波罗夷', 'author' => 'Bhikkhu Brahmali', 'cover' => null, 'cover_gradient' => 'linear-gradient(135deg, #2d8e5a 0%, #1a5c3a 100%)', 'updated_at' => '5天前', 'is_new' => true, 'category' => '律藏', ], [ 'id' => 'book-005', 'title' => '清净道论', 'author' => 'Buddhaghosa', 'cover' => null, 'cover_gradient' => 'linear-gradient(135deg, #8e2d2d 0%, #5c1a1a 100%)', 'updated_at' => '1周前', 'is_new' => false, 'category' => '注释', ], [ 'id' => 'book-006', 'title' => '增支部·一集', 'author' => 'Bhikkhu Bodhi', 'cover' => null, 'cover_gradient' => 'linear-gradient(135deg, #2d7a8e 0%, #1a4a5c 100%)', 'updated_at' => '1周前', 'is_new' => false, 'category' => '经藏', ], ]; } private function getUpdateBooks() { $books = ProgressChapter::with('channel.owner') ->leftJoin('pali_texts', function ($join) { $join->on('progress_chapters.book', '=', 'pali_texts.book') ->on('progress_chapters.para', '=', 'pali_texts.paragraph'); }) ->whereHas('channel', function ($query) { $query->where('status', 30); }) ->where('progress', '>', config('mint.library.list_min_progress')) ->take(10) ->get(); return $this->getBooksInfo($books); } private function getBooks($categories, $id, $filters) { if ($id) { $currentCategory = collect($categories)->firstWhere('id', $id); if (!$currentCategory) { abort(404); } // 标签查章节 $tagNames = $currentCategory['tag']; $tm = (new TagMap)->getTable(); $tg = (new Tag)->getTable(); $pt = (new PaliText)->getTable(); $where1 = " where co = " . count($tagNames); $a = implode(",", array_fill(0, count($tagNames), '?')); $in1 = "and t.name in ({$a})"; $param = $tagNames; $where2 = "where level = 1"; $query = "select uid as id,book,paragraph,level,toc as title,chapter_strlen,parent,path from ( select anchor_id as cid from ( select tm.anchor_id , count(*) as co from $tm as tm left join $tg as t on tm.tag_id = t.id where tm.table_name = 'pali_texts' $in1 group by tm.anchor_id ) T $where1 ) CID left join $pt as pt on CID.cid = pt.uid $where2 order by book,paragraph"; $chapters = DB::select($query, $param); $chaptersParam = []; foreach ($chapters as $key => $chapter) { $chaptersParam[] = [$chapter->book, $chapter->paragraph]; } // 获取该分类下的章节 $books = ProgressChapter::with('channel.owner') ->whereIns(['progress_chapters.book', 'progress_chapters.para'], $chaptersParam) ->whereHas('channel', function ($query) { $query->where('status', 30); }) ->where('progress', '>', config('mint.library.list_min_progress')) ->get(); } else { $booksChapter = PaliText::select(['book', 'paragraph'])->where('level', 1)->get(); $chapters = []; foreach ($booksChapter as $key => $value) { $chapters[] = [$value->book, $value->paragraph]; } $books = ProgressChapter::with('channel.owner') ->whereHas('channel', function ($query) use ($filters) { $filters['type'] === 'all' ? $query->where('status', 30) : $query->where('status', 30)->where('type', $filters['type']); }) ->where('progress', '>', config('mint.library.list_min_progress')) ->whereIns(['book', 'para'], $chapters) ->take(100) ->get(); } return $this->getBooksInfo($books); } private function getBooksInfo($books,) { $pali = PaliText::where('level', 1)->get(); // 获取该分类下的书籍 $categoryBooks = []; $books->each(function ($book) use (&$categoryBooks, $pali) { $title = $book->title; if (empty($title)) { $title = $pali->firstWhere('book', $book->book)->toc; } //Log::debug('getBooksInfo', ['book' => $book->book, 'paragraph' => $book->para]); $pcd_book_id = $pali->first(function ($item) use ($book) { return $item->book == $book->book && $item->paragraph == $book->para; })?->pcd_book_id; $coverFile = "/assets/images/cover/zh-hans/1/{$pcd_book_id}.png"; if (File::exists(public_path($coverFile))) { $coverUrl = $coverFile; } else { $coverUrl = null; } $colorIdx = $this->colorIndex($book->uid); $categoryBooks[] = [ "id" => $book->uid, "title" => $title, "author" => $book->channel->name, "publisher" => $book->channel->owner, "type" => __('labels.' . $book->channel->type), "cover" => $coverUrl, 'cover_gradient' => $this->coverGradients[$colorIdx % count($this->coverGradients)], "description" => $book->summary ?? "比库戒律的详细说明", "language" => __('language.' . $book->channel->lang), ]; }); return $categoryBooks; } private function loadCategories() { $json = file_get_contents(public_path("data/category/default.json")); $tree = json_decode($json, true); $flat = self::flattenWithIds($tree); return $flat; } public static function flattenWithIds(array $tree, int $parentId = 0, int $level = 1): array { $flat = []; foreach ($tree as $node) { $currentId = self::$nextId++; $item = [ 'id' => $currentId, 'parent_id' => $parentId, 'name' => $node['name'] ?? null, 'tag' => $node['tag'] ?? [], "description" => "佛教戒律经典", 'level' => $level, ]; $flat[] = $item; if (isset($node['children']) && is_array($node['children'])) { $childrenLevel = $level + 1; $flat = array_merge($flat, self::flattenWithIds($node['children'], $currentId, $childrenLevel)); } } return $flat; } private function getBreadcrumbs($category, $categories) { $breadcrumbs = []; $current = $category; while ($current) { array_unshift($breadcrumbs, $current); $current = collect($categories)->firstWhere('id', $current['parent_id']); } return $breadcrumbs; } }