zzwy2/app/controller/manager/Article.php

548 lines
22 KiB
PHP
Executable File
Raw Blame History

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

<?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();
}
}
}