548 lines
22 KiB
PHP
Executable File
548 lines
22 KiB
PHP
Executable File
<?php
|
||
|
||
namespace app\controller\manager;
|
||
|
||
use app\model\{Article as MArticle, ArticleTags, Category, SpecialRoute, System, Log};
|
||
use app\validate\Article as VArticle;
|
||
use think\exception\ValidateException;
|
||
use app\service\Tool;
|
||
use think\facade\Db;
|
||
|
||
/**
|
||
* 内容管理 - 文章管理
|
||
*/
|
||
class Article extends Base
|
||
{
|
||
// 允许设置组图的模版列表
|
||
protected $allowImgTemplate = [
|
||
Category::TEMPLATE_CASES,
|
||
];
|
||
|
||
|
||
//批量修改属性
|
||
public function attribute()
|
||
{
|
||
if ($this->request->isPost()) {
|
||
$ids = input('post.id/a');
|
||
if (empty($ids) || !is_array($ids)) {
|
||
return $this->json(2, '参数错误,请核对之后再操作!');
|
||
}
|
||
$data = [];
|
||
foreach (['top', 'hot', 'recommend'] as $key) {
|
||
$val = input('post.'.$key, 0);
|
||
if (in_array($val, [1, 2])) {
|
||
if ($val == 1) {
|
||
$data[$key] = 1;
|
||
} else {
|
||
$data[$key] = 0;
|
||
}
|
||
}
|
||
}
|
||
if (!empty($data)) {
|
||
MArticle::whereIn('id', $ids)->update($data);
|
||
Log::write('article', 'attribute', '批量修改了文章属性,涉及到的文章ID为:'.implode(',', $ids));
|
||
}
|
||
return $this->json();
|
||
}
|
||
return $this->json(1, '非法请求!');
|
||
}
|
||
|
||
//批量删除
|
||
public function batchDel()
|
||
{
|
||
if ($this->request->isPost()) {
|
||
$ids = input('post.ids/a');
|
||
if (empty($ids) || !is_array($ids)) {
|
||
return $this->json(2, '参数错误,请核对之后再操作!');
|
||
}
|
||
$items = MArticle::getListByIds($ids);
|
||
if (empty($items)) {
|
||
return $this->json(3, '待删除文章列表为空');
|
||
}
|
||
|
||
Db::startTrans();
|
||
try {
|
||
$delIds = [];
|
||
$cateId = $items[0]['category_id'];
|
||
foreach ($items as $item) {
|
||
$delIds[] = $item['id'];
|
||
}
|
||
|
||
MArticle::destroy($delIds);
|
||
SpecialRoute::deleteByTypeIds($delIds, SpecialRoute::type_archives);
|
||
Log::write('article', 'betchDel', '批量删除了文章,涉及到的文章ID为:'.implode(',', $delIds));
|
||
Db::commit();
|
||
} catch (\Exception $e) {
|
||
Db::rollback();
|
||
return $this->json(5000, '文章删除失败!');
|
||
}
|
||
|
||
return $this->json();
|
||
}
|
||
return $this->json(1, '非法请求!');
|
||
}
|
||
|
||
//删除
|
||
public function del()
|
||
{
|
||
if ($this->request->isPost()) {
|
||
$id = input('post.id/d');
|
||
if (is_numeric($id) && $id > 0) {
|
||
$item = MArticle::getById($id);
|
||
if (empty($item)) {
|
||
return $this->json(3, '待删除文章不存在');
|
||
}
|
||
|
||
Db::startTrans();
|
||
try {
|
||
MArticle::destroy($id);
|
||
Log::write('article', 'del', '删除文章,ID:'.$id.',标题:'.$item['title']);
|
||
SpecialRoute::deleteByTypeIds([$id], SpecialRoute::type_archives);
|
||
Db::commit();
|
||
} catch (\Exception $e) {
|
||
Db::rollback();
|
||
return $this->json(5000, '文章删除失败!');
|
||
}
|
||
return $this->json();
|
||
}
|
||
|
||
return $this->json(2, '参数错误,请核对之后再操作!');
|
||
}
|
||
return $this->json(1, '非法请求!');
|
||
}
|
||
|
||
//排序
|
||
public function sort()
|
||
{
|
||
if ($this->request->isPost()) {
|
||
$id = input('post.id/d');
|
||
$sort = input('post.sort');
|
||
$num = input('post.num/d', 1);
|
||
$categoryId = $this->request->param('category_id/d', 0); // get | post
|
||
if ($num <= 0) {
|
||
$num = 1;
|
||
}
|
||
if (!in_array($sort, ['up', 'down'], true)) {
|
||
return $this->json(2, '参数错误');
|
||
}
|
||
$item = MArticle::getById($id);
|
||
if (empty($item)) {
|
||
return $this->json(3, '该文章信息不存在');
|
||
}
|
||
$whereMap = [];
|
||
if (is_numeric($categoryId) && $categoryId > 0) {
|
||
$children = Category::getChildrenByParentId($categoryId);
|
||
if (!empty($children)) {
|
||
$childrenIds = [];
|
||
foreach ($children as $child) {
|
||
if ($child['model_id'] == 31) {
|
||
$childrenIds[] = $child['id'];
|
||
}
|
||
}
|
||
$whereMap[] = ['category_id', 'in', $childrenIds];
|
||
} else {
|
||
$whereMap[] = ['category_id', '=', $categoryId];
|
||
}
|
||
} else {
|
||
$whereMap[] = ['category_id', '=', $item['category_id']];
|
||
}
|
||
|
||
if ($sort == 'up') {
|
||
$whereMap[] = ['sort', '>', $item['sort']];
|
||
$order = "sort asc";
|
||
} else {
|
||
$whereMap[] = ['sort', '<', $item['sort']];
|
||
$order = "sort desc";
|
||
}
|
||
$forSortItems = MArticle::getListByWhereAndOrder($whereMap, $order, $num);
|
||
if (!empty($forSortItems)) {
|
||
$updateData = [];
|
||
$forSortCount = count($forSortItems);
|
||
for ($i = 0; $i < $forSortCount; $i++) {
|
||
if ($i == 0) {
|
||
$updateData[] = [
|
||
'id' => $forSortItems[$i]['id'],
|
||
'sort' => $item['sort']
|
||
];
|
||
} else {
|
||
$updateData[] = [
|
||
'id' => $forSortItems[$i]['id'],
|
||
'sort' => $forSortItems[$i - 1]['sort']
|
||
];
|
||
}
|
||
}
|
||
$updateData[] = [
|
||
'id' => $item['id'],
|
||
'sort' => $forSortItems[$i - 1]['sort']
|
||
];
|
||
if (!empty($updateData)) {
|
||
$model = new MArticle();
|
||
$model->saveAll($updateData);
|
||
$sortStr = $sort == 'up' ? '上移' : '下调';
|
||
Log::write('article', 'sort', "文章排序,ID:{$id} ,标题:{$item['title']},{$sortStr}了{$num}位");
|
||
return $this->json();
|
||
}
|
||
}
|
||
return $this->json(4, '无须调整排序!');
|
||
}
|
||
return $this->json(1, '非法请求!');
|
||
}
|
||
|
||
//编辑
|
||
public function edit()
|
||
{
|
||
if ($this->request->isPost()) {
|
||
$item = input('post.item/a');
|
||
$img = input('post.img', '');
|
||
$imgMobile = input('post.img_mobile', '');
|
||
$imgs = input('post.img_imgs', []);
|
||
$imgsMobile = input('post.img_imgs_mobile', []);
|
||
$resourceImgs = input('post.img_resource', []);
|
||
$resourceImgsMobile = input('post.img_resource_mobile', []);
|
||
|
||
$imgs = json_encode($imgs, true);
|
||
$imgsMobile = json_encode($imgsMobile, true);
|
||
|
||
$item['imgs'] = $imgs;
|
||
$item['imgs_mobile'] = $imgsMobile;
|
||
|
||
$resourceImgs = json_encode($resourceImgs, true);
|
||
$resourceImgsMobile = json_encode($resourceImgsMobile, true);
|
||
|
||
$item['other_text1'] = $resourceImgs;
|
||
$item['other_text2'] = $resourceImgsMobile;
|
||
|
||
$logo = input('post.img_logo', '');
|
||
$banner = input('post.img_banner', '');
|
||
$video = input('post.video', '');
|
||
$id = input('post.id/d');
|
||
$article = MArticle::getById($id);
|
||
if (empty($article)) {
|
||
return $this->json(1, '该文章不存在!');
|
||
}
|
||
$item['src'] = $img;
|
||
$item['src_mobile'] = $imgMobile;
|
||
if (!empty($logo)) {
|
||
$item['logo'] = $logo;
|
||
}
|
||
if (!empty($banner)) {
|
||
$item['banner'] = $banner;
|
||
}
|
||
$item['video'] = $video;
|
||
|
||
try {
|
||
validate(VArticle::class)->scene("edit")->check($item);
|
||
$auth = session('auth');
|
||
$item['update_time'] = time();
|
||
$item['updated'] = $auth['userName'];
|
||
|
||
cache('articlesForRoute', null); //清理缓存
|
||
|
||
// 默认属性
|
||
$defaultAttributes = ['recommend', 'top', 'hot'];
|
||
$item['recommend'] = $item['recommend'] ?? 0;
|
||
$item['top'] = $item['top'] ?? 0;
|
||
$item['hot'] = $item['hot'] ?? 0;
|
||
|
||
// 自定义属性配置(包含默认属性)
|
||
$recommendOtherNameStr = input('post.recommend_other_str/s', '');
|
||
$recommendOtherList = [];
|
||
if (!empty($recommendOtherNameStr)) {
|
||
$recommendOtherNameList = explode(',', $recommendOtherNameStr);
|
||
$attributeList = MArticle::getAttributeList([$article['category_id']]);
|
||
$attributeKeyList = array_flip($attributeList);
|
||
|
||
foreach ($recommendOtherNameList as $recommendOtherName) {
|
||
$attributeKey = $attributeKeyList[$recommendOtherName] ?? '';
|
||
if (!empty($attributeKey)) {
|
||
$recommendOtherList[] = $attributeKey;
|
||
|
||
// 默认属性配置,适用于排序
|
||
if (in_array($attributeKey, $defaultAttributes)) {
|
||
$item[$attributeKey] = 1;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
$item['recommend_other'] = empty($recommendOtherList) ? '' : implode(',', $recommendOtherList);
|
||
|
||
// 标签
|
||
$item['tag'] = trim($item['tag'] ?? '');
|
||
$item['tag'] = str_replace(',', ',', $item['tag']);
|
||
if (!empty($tagSaveCheck) && ($article['status'] == MArticle::STATUS_NORMAL || $item['status'] == MArticle::STATUS_NORMAL)) {
|
||
$oldTagList = empty($article['tag']) ? [] : explode(',', $article['tag']);
|
||
$curTagList = empty($item['tag']) ? [] : explode(',', $item['tag']);
|
||
/**
|
||
* $oldTagList 差集:total - 1
|
||
* $curTagList 差集:不存在则新增,存在则total + 1
|
||
* [normal->disable] total - 1
|
||
* [disable->normal] total + 1
|
||
*/
|
||
if ($article['status'] != $item['status']) {
|
||
if ($article['status'] == MArticle::STATUS_NORMAL) {
|
||
$curTagList = [];
|
||
} else {
|
||
$oldTagList = [];
|
||
}
|
||
}
|
||
|
||
$oldTagsDiff = array_diff($oldTagList, $curTagList);
|
||
$curTagsDiff = array_diff($curTagList, $oldTagList);
|
||
if (count($oldTagsDiff) > 0) {
|
||
ArticleTags::whereIn('name', $oldTagsDiff)
|
||
->where($tagSaveCheck, '>', 0)
|
||
->dec($tagSaveCheck, 1)
|
||
->update();
|
||
}
|
||
if (count($curTagsDiff) > 0) {
|
||
$hadTagItems = ArticleTags::findByNames($curTagsDiff);
|
||
$hadTags = $hadTagItems->column('name');
|
||
$newTags = array_diff($curTagsDiff, $hadTags);
|
||
|
||
$tagsInsert = [];
|
||
$tagsUpdate = [];
|
||
foreach ($newTags as $tagName) {
|
||
$tagsInsert[] = [
|
||
'name' => $tagName,
|
||
$tagSaveCheck => 1
|
||
];
|
||
}
|
||
foreach ($hadTagItems as $tagItem) {
|
||
$tagsUpdate[] = [
|
||
'id' => $tagItem['id'],
|
||
$tagSaveCheck => $tagItem[$tagSaveCheck] + 1
|
||
];
|
||
}
|
||
|
||
if (count($tagsInsert) > 0) {
|
||
ArticleTags::insertAll($tagsInsert);
|
||
}
|
||
if (count($tagsUpdate) > 0) {
|
||
(new ArticleTags())->saveAll($tagsUpdate);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
MArticle::updateById($id, $item);
|
||
|
||
//处理特殊路由
|
||
if (array_key_exists("route", $item) && !empty($item['route'])) {
|
||
$specialRoute = SpecialRoute::findByTypeRelaTioneId($id, SpecialRoute::type_archives);
|
||
if (empty($specialRoute)) {
|
||
$specialRouteData = [
|
||
"route" => $item["route"] ?? '',
|
||
"type" => SpecialRoute::type_archives,
|
||
"relation_id" => $id,
|
||
];
|
||
SpecialRoute::create($specialRouteData);
|
||
} else {
|
||
$specialRoute->save(["route" => $item["route"] ?? '']);
|
||
}
|
||
} else {
|
||
SpecialRoute::deleteByTypeIds([$id], SpecialRoute::type_archives);
|
||
}
|
||
|
||
Db::commit();
|
||
Log::write('article', 'edit', "文章编辑,ID:{$id} ,标题:{$item['title']}");
|
||
return $this->json();
|
||
} catch (ValidateException $e) {
|
||
Db::rollback();
|
||
return $this->json(2, $e->getError());
|
||
} catch (\Exception $e) {
|
||
Db::rollback();
|
||
return $this->json(3, $e->getMessage());
|
||
}
|
||
} else {
|
||
$id = input('param.id');
|
||
$article = MArticle::getById($id);
|
||
$category = [];
|
||
$categoryId = $article['category_id'] ?? 0;
|
||
$imgSize = '';
|
||
$attributeList = [];
|
||
|
||
if ($article) {
|
||
$article = MArticle::convertRecommendOther([$categoryId], $article, false);
|
||
$attributeList = MArticle::getAttributeList([$categoryId]);
|
||
|
||
$category = Category::getById($categoryId);
|
||
if ($category['img_width'] && $category['img_height']) {
|
||
$imgSize = $category['img_width'].'像素 X '.$category['img_height'].'像素';
|
||
} else {
|
||
$imgSize = System::getArticleImageSize();
|
||
}
|
||
}
|
||
|
||
$this->getImgSize($categoryId);
|
||
|
||
$this->data['item'] = $article;
|
||
$this->data['category'] = $category;
|
||
$this->data['imgSize'] = $imgSize;
|
||
$this->data['attributeList'] = $attributeList;
|
||
$this->data['contentRaw'] = in_array($categoryId, Category::$contentRawCategoryList);
|
||
$this->data['allowTag'] = in_array($categoryId, Category::$allowTagCategoryList);
|
||
$this->data['allowImgs'] = in_array($category['template_list'], $this->allowImgTemplate);
|
||
|
||
return $this->view();
|
||
}
|
||
}
|
||
|
||
//添加
|
||
public function add()
|
||
{
|
||
if ($this->request->isPost()) {
|
||
$item = input('post.item/a');
|
||
$img = input('post.img', '');
|
||
$imgMobile = input('post.img_mobile', '');
|
||
$imgs = input('post.img_imgs', []);
|
||
$imgsMobile = input('post.img_imgs_mobile', []);
|
||
$logo = input('post.img_logo', '');
|
||
$banner = input('post.img_banner', '');
|
||
$video = input('post.video', '');
|
||
|
||
$resourceImgs = input('post.img_resource', []);
|
||
$resourceImgsMobile = input('post.img_resource_mobile', []);
|
||
|
||
$imgs = json_encode($imgs, true);
|
||
$imgsMobile = json_encode($imgsMobile, true);
|
||
|
||
$item['imgs'] = $imgs;
|
||
$item['imgs_mobile'] = $imgsMobile;
|
||
|
||
$resourceImgs = json_encode($resourceImgs, true);
|
||
$resourceImgsMobile = json_encode($resourceImgsMobile, true);
|
||
|
||
$item['other_text1'] = $resourceImgs;
|
||
$item['other_text2'] = $resourceImgsMobile;
|
||
|
||
$item['src'] = $img;
|
||
$item['src_mobile'] = $imgMobile;
|
||
if (!empty($logo)) {
|
||
$item['logo'] = $logo;
|
||
}
|
||
|
||
if (!empty($banner)) {
|
||
$item['banner'] = $banner;
|
||
}
|
||
$item['video'] = $video;
|
||
Db::startTrans();
|
||
try {
|
||
validate(VArticle::class)->scene("add")->check($item);
|
||
$content = $item['content'] ?? '';
|
||
if (isset($item['content'])) {
|
||
unset($item['content']);
|
||
}
|
||
$item['content'] = $content;
|
||
if (!empty($item['alias'])) {
|
||
if (Tool::hasAlias($item['alias'])) {
|
||
throw new ValidateException('别名已存在');
|
||
}
|
||
cache('articlesForRoute', null); //有别名时,清理缓存
|
||
}
|
||
|
||
// 默认属性
|
||
$defaultAttributes = ['recommend', 'top', 'hot'];
|
||
$item['recommend'] = $item['recommend'] ?? 0;
|
||
$item['top'] = $item['top'] ?? 0;
|
||
$item['hot'] = $item['hot'] ?? 0;
|
||
|
||
// 自定义属性配置
|
||
$recommendOtherNameStr = input('post.recommend_other_str', '');
|
||
$recommendOtherList = [];
|
||
if (!empty($recommendOtherNameStr)) {
|
||
$recommendOtherNameList = explode(',', $recommendOtherNameStr);
|
||
$attributeList = MArticle::getAttributeList([$item['category_id']]);
|
||
$attributeKeyList = array_flip($attributeList);
|
||
|
||
foreach ($recommendOtherNameList as $recommendOtherName) {
|
||
$attributeKey = $attributeKeyList[$recommendOtherName] ?? '';
|
||
if (!empty($attributeKey)) {
|
||
$recommendOtherList[] = $attributeKey;
|
||
|
||
// 默认属性配置,适用于排序
|
||
if (in_array($attributeKey, $defaultAttributes)) {
|
||
$item[$attributeKey] = 1;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
$item['recommend_other'] = empty($recommendOtherList) ? '' : implode(',', $recommendOtherList);
|
||
|
||
// 标签
|
||
$item['tag'] = trim($item['tag'] ?? '');
|
||
$item['tag'] = str_replace(',', ',', $item['tag']);
|
||
if (!empty($item['tag']) && !empty($tagSaveCheck) && $item['status'] == MArticle::STATUS_NORMAL) {
|
||
$curTags = explode(',', $item['tag']);
|
||
$tagItems = ArticleTags::findByNames($curTags);
|
||
$hadTags = $tagItems->column('name');
|
||
$newTags = array_diff($curTags, $hadTags);
|
||
if (count($newTags) > 0) {
|
||
$tagsInsert = [];
|
||
$tagsUpdate = [];
|
||
foreach ($newTags as $tag) {
|
||
$tagsInsert[] = [
|
||
'name' => $tag,
|
||
$tagSaveCheck => 1
|
||
];
|
||
}
|
||
foreach ($tagItems as $tagItem) {
|
||
$tagsUpdate[] = [
|
||
'id' => $tagItem['id'],
|
||
$tagSaveCheck => $tagItem[$tagSaveCheck] + 1,
|
||
];
|
||
}
|
||
|
||
if (count($tagsInsert) > 0) {
|
||
ArticleTags::insertAll($tagsInsert);
|
||
}
|
||
if (count($tagsUpdate)) {
|
||
(new ArticleTags())->saveAll($tagsUpdate);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
$article = MArticle::create($item);
|
||
//处理特殊路由
|
||
if (array_key_exists("route", $item) && !empty($item['route'])) {
|
||
$specialRouteData = [
|
||
"route" => $item["route"] ?? '',
|
||
"type" => SpecialRoute::type_archives,
|
||
"relation_id" => $article->id,
|
||
];
|
||
SpecialRoute::create($specialRouteData);
|
||
}
|
||
Log::write('article', 'add', "文章新增,ID:{$article->id} ,标题:{$item['title']}");
|
||
Db::commit();
|
||
return $this->json();
|
||
} catch (ValidateException $e) {
|
||
Db::rollback();
|
||
return $this->json(2, $e->getError());
|
||
} catch (\Exception $e) {
|
||
Db::rollback();
|
||
return $this->json(2, 'request fail! '.$e->getMessage());
|
||
}
|
||
} else {
|
||
$categoryId = input('param.category_id');
|
||
$category = Category::getById($categoryId);
|
||
if (count($category) > 0 && $category['img_width'] && $category['img_height']) {
|
||
$imgSize = $category['img_width'].'像素 X '.$category['img_height'].'像素';
|
||
} else {
|
||
$imgSize = System::getArticleImageSize();
|
||
}
|
||
|
||
$this->getImgSize($categoryId);
|
||
|
||
$this->data['category'] = $category;
|
||
$this->data['imgSize'] = $imgSize;
|
||
$this->data['attributeList'] = MArticle::getAttributeList([$categoryId]);
|
||
$this->data['contentRaw'] = in_array($categoryId, Category::$contentRawCategoryList);
|
||
$this->data['allowTag'] = in_array($categoryId, Category::$allowTagCategoryList);
|
||
$this->data['allowImgs'] = in_array($category['template_list'], $this->allowImgTemplate);
|
||
|
||
return $this->view();
|
||
}
|
||
}
|
||
}
|