| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338 |
- <?php
- namespace App\Http\Controllers;
- use Illuminate\Http\Request;
- use Illuminate\Support\Facades\File;
- use Illuminate\Support\Facades\DB;
- use App\Models\PaliText;
- use App\Models\ProgressChapter;
- use App\Models\Tag;
- use App\Models\TagMap;
- class CategoryController extends Controller
- {
- // 封面渐变色池:uid 首字节取余循环,保证同一文集颜色稳定
- private array $coverGradients = [
- 'linear-gradient(160deg, #2d1020, #ae6b8b)',
- 'linear-gradient(160deg, #1a2d10,rgba(75, 114, 36, 0.61))',
- 'linear-gradient(160deg, #0d1f3c,rgb(55, 98, 150))',
- 'linear-gradient(160deg, #2d1020,rgb(151, 69, 94))',
- 'linear-gradient(160deg, #1a1a2d,rgb(76, 68, 146))',
- 'linear-gradient(160deg, #1a2820,rgb(55, 124, 99))',
- ];
- // -------------------------------------------------------------------------
- // 从 uid / id 字符串中提取一个稳定的整数,用于色池取余
- // -------------------------------------------------------------------------
- private function colorIndex(string $uid): int
- {
- return hexdec(substr(str_replace('-', '', $uid), 0, 4)) % 255;
- }
- protected static int $nextId = 1;
- public function home()
- {
- $categories = $this->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'));
- }
- public function category(int $id)
- {
- $categories = $this->loadCategories();
- $currentCategory = collect($categories)->firstWhere('id', $id);
- if (!$currentCategory) {
- abort(404);
- }
- // 获取子分类
- $subCategories = array_filter($categories, function ($cat) use ($id) {
- return $cat['parent_id'] == $id;
- });
- // 获取该分类下的书籍
- $categoryBooks = $this->getBooks($categories, $id);
- // 获取面包屑
- $breadcrumbs = $this->getBreadcrumbs($currentCategory, $categories);
- $types = $this->types();
- return view('library.tipitaka.category', compact(
- 'currentCategory',
- 'subCategories',
- 'categoryBooks',
- 'breadcrumbs',
- 'types'
- ));
- }
- private function types()
- {
- return ['translation', 'original', 'nissaya'];
- }
- 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)
- {
- $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();
- 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;
- }
- }
|