pzct/app/model/Article.php

542 lines
19 KiB
PHP
Raw Normal View History

2023-07-25 09:22:32 +00:00
<?php
namespace app\model;
use think\Paginator;
class Article extends Base
{
public const STATUS_NORMAL = 1; // 正常
public const STATUS_DISABLE = 0; // 禁用
// 文章属性(默认)
protected static $defaultAttributeList = [
'top' => '置顶',
'hot' => '热门',
'recommend' => '推荐',
];
public static function getAttributeList(array $categoryIds = [])
{
$data = [];
$recommendCategoryList = [];
if(count(array_intersect($categoryIds, $recommendCategoryList)) > 0) {
$data['recommend'] = '推荐';
}
// 新闻动态
$communityCategoryIds = Category::getCategoryWithChildrenIds(Category::CATEGORY_NEWS);
if(count(array_intersect($categoryIds, $communityCategoryIds)) > 0) {
$data['top'] = '置顶';
$data['hot'] = '热门';
$data['recommend'] = '推荐';
}
return $data;
}
/*********************************************
* 分割线
*********************************************/
//获取最高访问的文章列表
public static function getMostVisited($limit = 5)
{
if ($limit <= 0) {
$limit = 5;
}
return self::order('views', 'desc')
->limit($limit)
->select()
->toArray();
}
//获取栏目下最新记录
public static function getLatestByCategory($categoryId, $limit = 5)
{
if (empty($categoryId)) {
return [];
}
if ($limit <= 0) {
$limit = 5;
}
return self::where('category_id', $categoryId)
->order('id', 'desc')
->limit($limit)
->select()
->toArray();
}
//根据文章ID和栏目ID获取下一篇文章
public static function getNextArticleBySortAndCategoryId($sort, $categoryId)
{
return self::alias('a')
->where('a.sort', '<', $sort)
->leftJoin('category c', 'c.id = a.category_id')
->where('a.category_id', $categoryId)
->where('a.status', 1)
->order('a.sort', 'desc')
->field('a.id,a.title,a.category_id, c.title as category_title')
->findOrEmpty()
->toArray();
}
//根据文章ID和栏目ID获取上一篇文章
public static function getPrevArticleBySortAndCategoryId($sort, $categoryId)
{
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->where('a.sort', '>', $sort)
->where('a.category_id', $categoryId)
->where('a.status', 1)
->order('a.sort', 'asc')
->field('a.id,a.title,a.category_id, c.title as category_title')
->findOrEmpty()
->toArray();
}
//根据栏目ID获取文章列表
public static function getListByCategory($categoryId, $limit = 10)
{
return self::where('category_id', $categoryId)
->where('status', 1)
->order("sort", 'desc')
->limit($limit)
->select()
->toArray();
}
//根据栏目ID获取文章分页列表
public static function getListPageByCategory($categoryId, $per = 20, $keyword = '')
{
$where = [
['category_id', '=', $categoryId],
['status', '=', 1],
];
$param['category_id'] = $categoryId;
if ($keyword != '') {
$where[] = ['title', 'like', '%'.$keyword.'%'];
$param['keyword'] = $keyword;
}
$paginate = [
'list_rows' => $per,
'query' => $param
];
return self::where($where)
->order("sort", 'desc')
->paginate($paginate, false);
}
//根据栏目ID获取文章数量状态正常
public static function getNormalListCount($categoryId)
{
$where = [
['category_id', '=', $categoryId],
['status', '=', 1],
];
return self::where($where)
->count();
}
public static function onAfterInsert($article)
{
$article->sort = $article->id;
$article->create_time = $article->update_time = time();
$auth = session('auth');
$article->created = $article->updated = $auth['userName'];
$article->save();
}
/**
* 获取文章列表
* @param int $categoryId 分类ID
* @param int $per 每页数量
* @param string $keyword 关键词
* @param array $param 文章类型:置顶、热门、推荐 ['top','hot','recommend']
* @param int $status 文章状态,-1表示不限制
* @param array $orderList 排序
* @param bool $onlyChild 仅获取下级 默认true false=获取所有后代分类
* @return Paginator
*/
public static function getList($categoryId, $per = 20, $keyword = '', $param = [], $status = -1, $orderList = ['a.sort' => 'desc'], bool $onlyChild = true)
{
$whereMap = [];
$pageParam = [];
if (is_numeric($categoryId) && $categoryId > 0) {
$children = Category::getChildrenByParentId($categoryId, $onlyChild);
if (!empty($children)) {
$categoryIds = [$categoryId];
foreach ($children as $child) {
if ($child['model_id'] == Model::MODEL_ARTICLE) {
$categoryIds[] = $child['id'];
}
}
$whereMap[] = ['a.category_id', 'in', $categoryIds];
} else {
$whereMap[] = ['a.category_id', '=', $categoryId];
}
$pageParam['category_id'] = $categoryId;
}
if (!empty($keyword)) {
$whereMap[] = ['a.title', 'like', '%'.$keyword.'%'];
$pageParam['keyword'] = $keyword;
}
if (is_array($param) && count($param) > 0) {
$pageParam['param'] = $param;
foreach ($param as $vo) {
if (in_array($vo, ['top', 'hot', 'recommend'], true)) {
$whereMap[] = ["a.{$vo}", '=', 1];
}
}
}
$paginate = [
'list_rows' => $per,
'query' => $pageParam
];
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->when(count($whereMap) > 0, function ($query) use ($whereMap) {
$query->where($whereMap);
})
->when($status != -1, function ($query) use ($status) {
$query->where('a.status', $status);
})
->order($orderList)
->field('a.*, c.title as category_title')
->paginate($paginate, false);
}
//获取文章涉及到的图片
public static function getFilesInUse()
{
$items = self::select()->toArray();
$data = [];
foreach ($items as $item) {
$src = trim($item['src']);
if (!empty($src)) {
$key = getKeyByPath($src);
$data[$key] = $src;
}
$imgs = getImageUrlFromText($item['content']);
if (!empty($imgs)) {
$data = array_merge($data, $imgs);
}
$videos = getVideoUrlFromText($item['content']);
if (!empty($videos)) {
$data = array_merge($data, $videos);
}
}
return $data;
}
//推荐列表(其他推荐类)
public static function getRecommendList($categoryId, $recommend = '', $limit = 3, $excludeIds = [], $orderList = ['a.id' => 'desc'])
{
if (empty($categoryId) || empty($recommend)) {
return [];
}
$whereMap = [];
if (is_numeric($categoryId) && $categoryId > 0) {
$children = Category::getChildrenByParentId($categoryId);
if (!empty($children)) {
$categoryIds = [$categoryId];
foreach ($children as $child) {
if ($child['model_id'] == Model::MODEL_ARTICLE) {
$categoryIds[] = $child['id'];
}
}
$whereMap[] = ['a.category_id', 'in', $categoryIds];
} else {
$whereMap[] = ['a.category_id', '=', $categoryId];
}
}
if (!empty($excludeIds)) {
$whereMap[] = ['a.id', 'not in', $excludeIds];
}
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->when(count($whereMap) > 0, function ($query) use ($whereMap) {
$query->where($whereMap);
})
->whereRaw("FIND_IN_SET(:recommend, a.`recommend_other`)", ['recommend' => $recommend])
->where('a.status', 1)
->order($orderList)
->limit($limit)
->field('a.*, c.title as category_title')
->select()
->toArray();
}
// 转换文章的推荐设置
public static function convertRecommendOther($categoryIds = [], $articles, $isMulti = true)
{
if (empty($articles) || count($articles) == 0) {
return $articles;
}
$attributeList = self::getAttributeList($categoryIds);
if ($isMulti) {
foreach ($articles as &$article) {
$recommendOtherList = [];
$recommendOtherStrList = [];
if (isset($article['recommend_other']) && !empty($article['recommend_other'])) {
$recommendOtherList = explode(',', $article['recommend_other']);
foreach ($recommendOtherList as $recommendKey) {
if (isset($attributeList[$recommendKey]) && !empty($attributeList[$recommendKey])) {
$recommendOtherStrList[] = $attributeList[$recommendKey];
}
}
}
$article['recommend_other_list'] = $recommendOtherList;
$article['recommend_other_str'] = implode(',', $recommendOtherStrList);
}
unset($article);
} else {
$recommendOtherList = [];
$recommendOtherStrList = [];
if (isset($articles['recommend_other']) && !empty($articles['recommend_other'])) {
$recommendOtherList = explode(',', $articles['recommend_other']);
foreach ($recommendOtherList as $recommendKey) {
if (isset($attributeList[$recommendKey]) && !empty($attributeList[$recommendKey])) {
$recommendOtherStrList[] = $attributeList[$recommendKey];
}
}
}
$articles['recommend_other_list'] = $recommendOtherList;
$articles['recommend_other_str'] = implode(',', $recommendOtherStrList);
}
return $articles;
}
// 获取最新动态
public static function getLastDynamicsList($categoryId, $withChild = false, $limit = 10)
{
$categoryIds = [$categoryId];
if ($withChild) {
$childCategories = Category::getChildrenByParentId($categoryId);
if (!empty($childCategories)) {
foreach ($childCategories as $category) {
if ($category['model_id'] == Model::MODEL_ARTICLE) {
$categoryIds[] = $category['id'];
}
}
}
}
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->whereIn('a.category_id', $categoryIds)
->where('a.status', 1)
->order('a.id', 'desc')
->limit($limit)
->field('a.*, c.title as category_title')
->select()
->toArray();
}
//根据文章ID和栏目IDs默认按创建时间获取下一篇文章
public static function getNextArticleByTimeAndCategoryIds(array $where, $sortOrder = ['a.id' => 'desc'])
{
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->where($where)
->where('a.status', 1)
->order($sortOrder)
->field('a.*, c.title as category_title')
->findOrEmpty()
->toArray();
}
//根据文章ID和栏目IDs默认按创建时间获取上一篇文章
public static function getPrevArticleByTimeAndCategoryIds(array $where, $sortOrder = ['a.id' => 'asc'])
{
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
// ->where('a.id','>',$curId)
// ->whereIn('a.category_id', $categoryIds)
->where($where)
->where('a.status', 1)
->order($sortOrder)
->field('a.*, c.title as category_title')
->findOrEmpty()
->toArray();
}
public static function getLastTopList($categoryId, $limit = 3, $excludeIds = [], $orderList = ['a.recommend_other' => 'desc', 'a.id' => 'desc'])
{
if (empty($categoryId)) {
return [];
}
$whereMap = [];
if (is_numeric($categoryId) && $categoryId > 0) {
$children = Category::getChildrenByParentId($categoryId);
if (!empty($children)) {
$categoryIds = [$categoryId];
foreach ($children as $child) {
if ($child['model_id'] == Model::MODEL_ARTICLE) {
$categoryIds[] = $child['id'];
}
}
$whereMap[] = ['a.category_id', 'in', $categoryIds];
} else {
$whereMap[] = ['a.category_id', '=', $categoryId];
}
}
if (!empty($excludeIds)) {
$whereMap[] = ['a.id', 'not in', $excludeIds];
}
$items = self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->when(count($whereMap) > 0, function ($query) use ($whereMap) {
$query->where($whereMap);
})
->where('a.status', 1)
->order($orderList);
if ($limit > 0) {
$items = $items->limit($limit);
}
$items = $items->limit($limit)
->field('a.*, c.title as category_title')
->select()
->toArray();
return $items;
}
// 分页获取最新动态
public static function getLastDynamicsPageList($categoryId, $withChild = true, $per = 20)
{
$categoryIds = [$categoryId];
if ($withChild) {
$childCategories = Category::getChildrenByParentId($categoryId);
if (!empty($childCategories)) {
foreach ($childCategories as $category) {
if ($category['model_id'] == Model::MODEL_ARTICLE) {
$categoryIds[] = $category['id'];
}
}
}
}
$paginate = [
'list_rows' => $per,
'query' => [
'category_id' => $categoryId
]
];
return self::alias('a')
->leftJoin('category c', 'c.id = a.category_id')
->whereIn('a.category_id', $categoryIds)
->where('a.status', 1)
->order('a.id', 'desc')
->field('a.*, c.title as category_title')
->paginate($paginate, false);
}
// 根据栏目ID进行分组按sort倒序排列每组最多获取n条(默认10条)数据
public static function getGroupListByCategoryIds(array $categoryIds, int $size = 10)
{
return self::alias('ca')
->where($size, '>', function ($q) {
$q->table('bee_article cb')
->whereRaw('ca.category_id = cb.category_id and ca.sort < cb.sort')->field('count(*)');
})
->whereIn('ca.category_id', $categoryIds)
->field('ca.*')
->order(['ca.sort' => 'desc'])
->select();
}
public static function parseList($items)
{
try {
$tagNameList = [];
foreach ($items as $ki => $item) {
$tagStr = trim($item['tag'] ?? '');
$tagNames = empty($tagStr) ? [] : explode(',', $tagStr);
$tagNameList = array_merge($tagNameList, $tagNames);
}
$tagNameList = array_unique($tagNameList);
$tagKVs = ArticleTags::findByNames($tagNameList)->column('id', 'name');
foreach ($items as $ki => $item) {
$tagStr = trim($item['tag'] ?? '');
$tagNames = empty($tagStr) ? [] : explode(',', $tagStr);
$tagList = [];
foreach ($tagNames as $tagName) {
$tagList[] = [
'name' => $tagName,
'id' => $tagKVs[$tagName] ?? 0
];
}
$items[$ki]['tag_list'] = $tagList;
$createTime = is_numeric($item['create_time']) ? $item['create_time'] : strtotime($item['create_time']);
$items[$ki]['create_date'] = date('Y-m-d', $createTime);
$item['create_dateTime'] = date('Y-m-d H:i:s', $createTime);
}
} catch (\Exception $e) {
}
return $items;
}
public static function parseInfo($item)
{
if ($item) {
$tagStr = trim($item['tag'] ?? '');
$tagNameList = empty($tagStr) ? [] : explode(',', $tagStr);
$tagKVs = ArticleTags::findByNames($tagNameList)->column('id', 'name');
$tagList = [];
foreach ($tagNameList as $tagName) {
$tagList[] = [
'name' => $tagName,
'id' => $tagKVs[$tagName] ?? 0
];
}
$item['tag_list'] = $tagList;
$createTime = is_numeric($item['create_time']) ? $item['create_time'] : strtotime($item['create_time']);
$item['create_date'] = date('Y-m-d', $createTime);
$item['create_dateTime'] = date('Y-m-d H:i:s', $createTime);
}
return $item;
}
public static function countList(array $where, callable $call = null)
{
$q = new static();
$q = $q->where($where);
if ($call) {
$q = $call($q);
}
return $q->count();
}
public static function findListByWhere(array $where, int $page = 1, int $size = 10, callable $call = null, array $sortList = [])
{
$q = new static();
$q = $q->where($where);
if ($call) {
$q = $call($q);
}
if ($size > 0) {
$q = $q->page($page, $size);
}
if (count($sortList)) {
$q = $q->order($sortList);
}
return $q->select();
}
}