Source code of filesite.io.
https://filesite.io
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
233 lines
8.3 KiB
233 lines
8.3 KiB
<?php |
|
/** |
|
* List Controller |
|
*/ |
|
require_once __DIR__ . '/../../../lib/DirScanner.php'; |
|
require_once __DIR__ . '/../../../plugins/Parsedown.php'; |
|
require_once __DIR__ . '/../../../plugins/Common.php'; |
|
|
|
Class ListController extends Controller { |
|
|
|
public function actionIndex() { |
|
$cateId = $this->get('id', ''); |
|
$cacheParentDataId = $this->get('cid', ''); |
|
if (empty($cateId) || empty($cacheParentDataId)) { |
|
throw new Exception("参数缺失!", 403); |
|
} |
|
|
|
|
|
//获取数据 |
|
$menus = array(); //菜单,一级目录 |
|
$htmlReadme = ''; //Readme.md 内容,底部网站详细介绍 |
|
$htmlCateReadme = ''; //当前目录下的Readme.md 内容 |
|
$menus_sorted = array(); //Readme_sort.txt 说明文件内容,一级目录菜单从上到下的排序 |
|
|
|
$scanner = new DirScanner(); |
|
|
|
//TODO: 根据参数cid获取id对应的目录realpath,从而只扫描这个目录 |
|
$cacheSeconds = 3600; |
|
$cachedParentData = Common::getCacheFromFile($cacheParentDataId, $cacheSeconds); |
|
if (empty($cachedParentData)) { |
|
throw new Exception("缓存已过期,请返回上一页重新进入!", 404); |
|
} |
|
|
|
$currentDir = $cachedParentData[$cateId]; |
|
if (empty($currentDir)) { |
|
throw new Exception("缓存数据中找不到当前目录,请返回上一页重新进入!", 404); |
|
} |
|
|
|
//$scanner->setWebRoot(FSC::$app['config']['content_directory']); |
|
$scanner->setWebRoot($this->getCurrentWebroot($currentDir)); |
|
$scanner->setRootDir($currentDir['realpath']); |
|
//header('Content-Type: text/html; charset=utf-8'); |
|
//echo $this->getCurrentWebroot($currentDir) . "\n"; |
|
//echo $currentDir['realpath'] . "\n"; |
|
//exit; |
|
|
|
//优先从缓存读取数据 |
|
$prefix = FSC::$app['config']['theme'] . '_v1.0'; |
|
$maxScanDeep = 2; //最大扫描目录级数 |
|
$cacheKey = "{$prefix}_ltree_{$maxScanDeep}_{$cateId}"; |
|
$cachedData = Common::getCacheFromFile($cacheKey); |
|
if (!empty($cachedData)) { |
|
$dirTree = $cachedData; |
|
$scanner->setTreeData($cachedData); |
|
}else { |
|
$dirTree = $scanner->scan($currentDir['realpath'], $maxScanDeep); |
|
Common::saveCacheToFile($cacheKey, $dirTree); |
|
} |
|
|
|
//优先从缓存读取数据 |
|
$cacheKey = $cacheDataId = "{$prefix}_ldata_{$maxScanDeep}_{$cateId}"; |
|
$cachedData = Common::getCacheFromFile($cacheKey); |
|
if (!empty($cachedData)) { |
|
$scanResults = $cachedData; |
|
$scanner->setScanResults($cachedData); |
|
}else { |
|
$scanResults = $scanner->getScanResults(); |
|
Common::saveCacheToFile($cacheKey, $scanResults); |
|
} |
|
|
|
//按照scanResults格式把当前目录扫描结果中的目录数据拼接到当前目录数据里: currentDir |
|
if (!empty($currentDir['directories'])) { |
|
foreach ($currentDir['directories'] as $id => $item) { |
|
if (empty($scanResults[$id]['directories'])) { |
|
continue; |
|
} |
|
|
|
$currentDir['directories'][$id] = $scanResults[$id]; |
|
} |
|
} |
|
$scanResults = array($cateId => $currentDir); //重新组装数据 |
|
|
|
//header('Content-Type: text/html; charset=utf-8'); |
|
//print_r($currentDir); |
|
//print_r($scanResults); |
|
//exit; |
|
|
|
|
|
//获取目录 |
|
$menus = $scanner->getMenus(); |
|
|
|
$titles = array(); |
|
$readmeFile = $scanner->getDefaultReadme(); |
|
if (!empty($readmeFile)) { |
|
if (!empty($readmeFile['sort'])) { |
|
$menus_sorted = explode("\n", $readmeFile['sort']); |
|
} |
|
|
|
$titles = $scanner->getMDTitles($readmeFile['id']); |
|
|
|
$Parsedown = new Parsedown(); |
|
$content = file_get_contents($readmeFile['realpath']); |
|
$htmlReadme = $Parsedown->text($content); |
|
$htmlReadme = $scanner->fixMDUrls($readmeFile['realpath'], $htmlReadme); |
|
} |
|
|
|
//排序 |
|
$sortedTree = $this->sortMenusAndDirTree($menus_sorted, $menus, $dirTree); |
|
if (!empty($sortedTree)) { |
|
$menus = $sortedTree['menus']; |
|
$dirTree = $sortedTree['dirTree']; |
|
} |
|
|
|
//获取目录面包屑 |
|
//TODO: 重写此方法,根据scanner->getId()方法来逐级目录生成 |
|
$subcate = $scanResults[$cateId]; |
|
$breadcrumbs = $this->getBreadcrumbs($currentDir, $cachedParentData, $scanner); |
|
|
|
//获取当前目录下的readme |
|
$cateReadmeFile = $scanner->getDefaultReadme($cateId); |
|
if (!empty($cateReadmeFile)) { |
|
$Parsedown = new Parsedown(); |
|
$content = file_get_contents($cateReadmeFile['realpath']); |
|
$htmlCateReadme = $Parsedown->text($content); |
|
$htmlCateReadme = $scanner->fixMDUrls($cateReadmeFile['realpath'], $htmlCateReadme); |
|
} |
|
|
|
//获取默认mp3文件 |
|
$rootCateId = $this->get('id', ''); |
|
$mp3File = $scanner->getDefaultFile('mp3', $rootCateId); |
|
if (empty($mp3File)) { |
|
$mp3File = $scanner->getDefaultFile('mp3'); |
|
} |
|
|
|
//翻页支持 |
|
$page = $this->get('page', 1); |
|
$pageSize = $this->get('limit', 24); |
|
|
|
$pageTitle = !empty($titles) ? $titles[0]['name'] : "FileSite.io - 无数据库、基于文件和目录的Markdown文档、网址导航、图书、图片、视频网站PHP开源系统"; |
|
if (!empty($subcate)) { |
|
$pageTitle = "{$subcate['directory']}的照片,来自{$pageTitle}"; |
|
if (!empty($subcate['title'])) { |
|
$pageTitle = $subcate['title']; |
|
} |
|
} |
|
$viewName = '//site/index'; //共享视图 |
|
$params = compact( |
|
'cateId', 'dirTree', 'scanResults', 'menus', 'htmlReadme', 'breadcrumbs', 'htmlCateReadme', |
|
'mp3File', 'page', 'pageSize', 'cacheDataId', 'cacheParentDataId' |
|
); |
|
return $this->render($viewName, $params, $pageTitle); |
|
} |
|
|
|
//实现php 5.5开始支持的array_column方法 |
|
protected function array_column($arr, $col) { |
|
$out = array(); |
|
|
|
if (!empty($arr) && is_array($arr) && !empty($col)) { |
|
foreach ($arr as $index => $item) { |
|
if (!empty($item[$col])) { |
|
array_push($out, $item[$col]); |
|
} |
|
} |
|
} |
|
|
|
return $out; |
|
} |
|
|
|
//根据pid在目录数组里找出对应的父目录数据 |
|
protected function getParentCateByPid($pid, $cates) { |
|
$parent = array(); |
|
|
|
foreach ($cates as $index => $item) { |
|
if ($item['id'] == $pid) { |
|
$parent = $item; |
|
break; |
|
}else if (!empty($item['directories'])) { |
|
$parent = $this->getParentCateByPid($pid, $item['directories']); |
|
if (!empty($parent)) {break;} |
|
} |
|
} |
|
|
|
|
|
return $parent; |
|
} |
|
|
|
protected function getCurrentWebroot($currentDir) { |
|
$webroot = FSC::$app['config']['content_directory']; |
|
$arr = explode($webroot, $currentDir['realpath']); |
|
if (count($arr) < 2) { |
|
return $webroot; |
|
} |
|
|
|
return "{$webroot}{$arr[1]}/"; |
|
} |
|
|
|
//根据目录结构以及当前目录获取面包屑 |
|
protected function getBreadcrumbs($currentDir, $scanResults, $scanner) { |
|
$webroot = FSC::$app['config']['content_directory']; |
|
$arr = explode($webroot, $currentDir['realpath']); |
|
$breads = array(); |
|
|
|
if (count($arr) < 2) { |
|
return $breads; |
|
} |
|
|
|
$cates = explode('/', $arr[1]); |
|
$parentCate = "{$arr[0]}{$webroot}"; |
|
foreach ($cates as $cate) { |
|
if ($cate == $currentDir['directory']) {break;} |
|
|
|
$subcate = "{$parentCate}{$cate}"; |
|
$cateId = $scanner->getId($subcate); |
|
array_push($breads, [ |
|
'id' => $cateId, |
|
'name' => $cate, |
|
'url' => $scanResults[$cateId]['path'], |
|
]); |
|
|
|
$parentCate = "{$subcate}/"; //记录上一级目录 |
|
} |
|
|
|
//最后一级 |
|
array_push($breads, [ |
|
'id' => $currentDir['id'], |
|
'name' => $currentDir['directory'], |
|
'url' => $currentDir['path'], |
|
]); |
|
|
|
return $breads; |
|
} |
|
|
|
}
|
|
|