using Infrastructure;
using Infrastructure.Attribute;
using Infrastructure.Extensions;
using Mapster;
using ZR.Common;
using ZR.Model.Content;
using ZR.Model.Content.Dto;
using ZR.Model.Enum;
using ZR.Model.System;
using ZR.Repository;
using ZR.Service.Content.IService;
using ZR.ServiceCore.Services;
namespace ZR.Service.Content
{
///
///
///
[AppService(ServiceType = typeof(IArticleService), ServiceLifetime = LifeTime.Transient)]
public class ArticleService : BaseService, IArticleService
{
private readonly IArticleCategoryService _categoryService;
private readonly IArticleTopicService _topicService;
private readonly ISysUserMsgService _userMsgService;
///
///
///
///
///
///
public ArticleService(
IArticleCategoryService categoryService,
IArticleTopicService topicService,
ISysUserMsgService userMsgService)
{
_categoryService = categoryService;
_topicService = topicService;
_userMsgService = userMsgService;
}
///
/// 查询文章管理列表
///
///
///
public PagedInfo GetList(ArticleQueryDto parm)
{
var predicate = Expressionable.Create();
predicate = predicate.AndIF(!string.IsNullOrEmpty(parm.Title), m => m.Title.Contains(parm.Title));
predicate = predicate.AndIF(!string.IsNullOrEmpty(parm.AbstractText), m => m.AbstractText.Contains(parm.AbstractText));
predicate = predicate.AndIF(!string.IsNullOrEmpty(parm.Status), m => m.Status == parm.Status);
predicate = predicate.AndIF(parm.IsPublic != null, m => m.IsPublic == parm.IsPublic);
predicate = predicate.AndIF(parm.IsTop != null, m => m.IsTop == parm.IsTop);
predicate = predicate.AndIF(parm.ArticleType != null, m => (int)m.ArticleType == parm.ArticleType);
predicate = predicate.AndIF(parm.TopicId != null, m => m.TopicId == parm.TopicId);
predicate = predicate.AndIF(parm.AuditStatus != null, m => m.AuditStatus == parm.AuditStatus);
if (parm.CategoryId != null)
{
var allChildCategory = Context.Queryable()
.ToChildList(m => m.ParentId, parm.CategoryId);
var categoryIdList = allChildCategory.Select(x => x.CategoryId).ToArray();
predicate = predicate.And(m => categoryIdList.Contains(m.CategoryId));
}
var response = Queryable()
.WithCache(60 * 24)
.IgnoreColumns(x => new { x.Content })
.Includes(x => x.CategoryNav) //填充子对象
.Where(predicate.ToExpression())
//.OrderBy(x => x.CreateTime, OrderByType.Desc)
.ToPage(parm);
return response;
}
///
/// 前台查询文章列表
///
///
///
public PagedInfo GetArticleList(ArticleQueryDto parm)
{
var predicate = Expressionable.Create();
predicate = predicate.And(m => m.Status == "1");
predicate = predicate.And(m => m.IsPublic == 1);
predicate = predicate.And(m => m.AuditStatus == AuditStatusEnum.Passed);
predicate = predicate.And(m => m.ArticleType == ArticleTypeEnum.Article);
predicate = predicate.AndIF(parm.SearchText.IsNotEmpty(), m => m.Title.Contains(parm.SearchText) || m.AbstractText.Contains(parm.SearchText));
predicate = predicate.AndIF(parm.TopicId != null, m => m.TopicId == parm.TopicId);
if (parm.CategoryId != null)
{
var allChildCategory = Context.Queryable()
.ToChildList(m => m.ParentId, parm.CategoryId);
var categoryIdList = allChildCategory.Select(x => x.CategoryId).ToArray();
predicate = predicate.And(m => categoryIdList.Contains(m.CategoryId));
}
var response = Queryable()
.WithCache(60 * 30)
.Includes(x => x.CategoryNav)
.LeftJoin((m, u) => m.UserId == u.UserId).Filter(null, true)
.Where(predicate.ToExpression())
.OrderByDescending(m => m.Cid)
.Select((m, u) => new ArticleDto()
{
User = new ArticleUser()
{
Avatar = u.Avatar,
NickName = u.NickName,
Sex = u.Sex,
},
Content = string.Empty,
UserIP = string.Empty
}, true)
.ToPage(parm);
if (parm.UserId > 0)
{
Context.ThenMapper(response.Result, item =>
{
item.IsPraise = Context.Queryable()
.Where(f => f.UserId == parm.UserId && f.IsDelete == 0)
.SetContext(scl => scl.ArticleId, () => item.Cid, item).Any() ? 1 : 0;
});
}
return response;
}
///
/// 查询最新文章列表
///
///
public List GetNewArticleList()
{
var predicate = Expressionable.Create();
predicate = predicate.And(m => m.Status == "1");
predicate = predicate.And(m => m.IsPublic == 1);
predicate = predicate.And(m => m.ArticleType == ArticleTypeEnum.Article);
predicate = predicate.And(m => m.AuditStatus == AuditStatusEnum.Passed);
var response = Queryable()
.Where(predicate.ToExpression())
.Includes(x => x.CategoryNav) //填充子对象
.Take(10)
.OrderBy(f => f.CreateTime, OrderByType.Desc)
.ToList();
return response.Adapt>();
}
///
/// 前台查询动态列表
///
///
///
public PagedInfo GetMonentList(ArticleQueryDto parm)
{
var predicate = Expressionable.Create();
predicate = predicate.And(m => m.Status == "1");
predicate = predicate.And(m => m.IsPublic == 1);
predicate = predicate.And(m => m.ArticleType == ArticleTypeEnum.Monent);
predicate = predicate.And(m => m.AuditStatus == AuditStatusEnum.Passed);
predicate = predicate.AndIF(parm.TopicId != null, m => m.TopicId == parm.TopicId);
predicate = predicate.AndIF(parm.CategoryId != null, m => m.CategoryId == parm.CategoryId);
//获取我的圈子
if (parm.QueryMyJoin)
{
var myJoin = Context.Queryable()
.Where(m => m.UserId == parm.UserId)
.Select(m => m.CategoryId)
.ToList();
predicate = predicate.And(m => myJoin.Contains(m.CategoryId));
}
var response = Queryable()
.Includes(x => x.CategoryNav) //填充子对象
.LeftJoin((m, u) => m.UserId == u.UserId).Filter(null, true)
.Where(predicate.ToExpression())
.OrderByIF(parm.OrderBy == 1, m => new { m.PraiseNum, m.CommentNum }, OrderByType.Desc)
.OrderByIF(parm.OrderBy == 2, m => new { m.Cid }, OrderByType.Desc)
.OrderBy(m => m.Cid, OrderByType.Desc)
.Select((m, u) => new ArticleDto()
{
User = new ArticleUser()
{
Avatar = u.Avatar,
NickName = u.NickName,
Sex = u.Sex,
},
}, true)
.ToPage(parm);
if (parm.UserId > 0)
{
Context.ThenMapper(response.Result, item =>
{
item.IsPraise = Context.Queryable()
.Where(f => f.UserId == parm.UserId && f.IsDelete == 0)
.SetContext(scl => scl.ArticleId, () => item.Cid, item).Any() ? 1 : 0;
});
}
return response;
}
///
/// 查询我的文章列表
///
///
///
public PagedInfo GetMyList(ArticleQueryDto parm)
{
var predicate = Expressionable.Create();
predicate = predicate.AndIF(!string.IsNullOrEmpty(parm.Title), m => m.Title.Contains(parm.Title));
predicate = predicate.AndIF(!string.IsNullOrEmpty(parm.Status), m => m.Status == parm.Status);
predicate = predicate.AndIF(parm.BeginTime != null, m => m.CreateTime >= parm.BeginTime);
predicate = predicate.AndIF(parm.EndTime != null, m => m.CreateTime <= parm.EndTime);
predicate = predicate.And(m => m.UserId == parm.UserId);
predicate = predicate.AndIF(parm.ArticleType != null, m => (int)m.ArticleType == parm.ArticleType);
predicate = predicate.AndIF(parm.TabId == 2, m => m.IsPublic == 0 && m.UserId == parm.UserId);
var response = Queryable()
//.IgnoreColumns(x => new { x.Content })
.Includes(x => x.CategoryNav)
.Where(predicate.ToExpression())
.OrderByIF(parm.TabId == 3, m => new { m.IsTop, m.PraiseNum }, OrderByType.Desc)
.OrderByDescending(m => new { m.IsTop, m.Cid })
.Select((x) => new ArticleDto()
{
Content = x.ArticleType == 0 ? string.Empty : x.Content,
}, true)
.ToPage(parm);
return response;
}
///
/// 修改文章管理
///
///
///
public int UpdateArticle(Article model)
{
var response = Update(w => w.Cid == model.Cid, it => new Article()
{
Title = model.Title,
Content = model.Content,
Status = model.Status,
Tags = model.Tags,
UpdateTime = DateTime.Now,
CoverUrl = model.CoverUrl,
CategoryId = model.CategoryId,
IsPublic = model.IsPublic,
AbstractText = model.AbstractText,
});
return response;
}
///
/// 置顶文章
///
///
///
public int TopArticle(Article model)
{
var response = Update(w => w.Cid == model.Cid, it => new Article()
{
IsTop = model.IsTop,
});
return response;
}
///
/// 评论权限
///
///
///
public int ChangeComment(Article model)
{
var response = Update(w => w.Cid == model.Cid, it => new Article()
{
CommentSwitch = model.CommentSwitch,
});
return response;
}
///
/// 是否公开
///
///
///
public int ChangeArticlePublic(Article model)
{
var response = Update(w => w.Cid == model.Cid, it => new Article()
{
IsPublic = model.IsPublic,
});
return response;
}
///
/// 修改文章访问量
///
///
///
public int UpdateArticleHit(long cid)
{
var response = Update(w => w.Cid == cid, it => new Article() { Hits = it.Hits + 1 });
return response;
}
///
/// 点赞
///
///
///
public int PraiseArticle(long cid)
{
return Update(w => w.Cid == cid, it => new Article() { PraiseNum = it.PraiseNum + 1 });
}
public int CancelPraise(long cid)
{
return Update(w => w.Cid == cid, it => new Article() { PraiseNum = it.PraiseNum - 1 });
}
public Article PublishArticle(Article article)
{
return Publish(article);
}
///
/// 前端-发布动态,文章
///
///
///
public Article Publish(Article article)
{
var httpContext = App.HttpContext;
article.AuthorName = HttpContextExtension.GetName(httpContext);
article.UserId = HttpContextExtension.GetUId(httpContext);
article.UserIP = HttpContextExtension.GetClientUserIp(httpContext);
article.Location = HttpContextExtension.GetIpInfo(article.UserIP);
article.AuditStatus = AuditStatusEnum.Passed;
article = InsertReturnEntity(article);
if (article.Status == "1")
{
//跟新话题加入数
if (article.Cid > 0 && article.TopicId > 0)
{
_topicService.Update(w => w.TopicId == article.TopicId, it => new ArticleTopic() { JoinNum = it.JoinNum + 1 });
}
//更新发布文章数
if (article.Cid > 0 && article.CategoryId > 0)
{
_categoryService.Update(w => w.CategoryId == article.CategoryId, it => new ArticleCategory() { ArticleNum = it.ArticleNum + 1 });
}
}
return article;
}
///
/// 获取文章详情
///
/// 内容id
/// 当前用户id
///
///
public ArticleDto GetArticle(long cid, long userId)
{
var response = GetId(cid);
var model = response.Adapt() ?? throw new CustomException(ResultCode.FAIL, "内容不存在");
if (model.IsPublic == 0 && userId != model.UserId)
{
throw new CustomException(ResultCode.CUSTOM_ERROR, "访问失败");
}
if (model != null)
{
model.CategoryNav = _categoryService.GetById(model.CategoryId).Adapt();
}
var webContext = App.HttpContext;
var CK = "ARTICLE_DETAILS_" + userId + HttpContextExtension.GetClientUserIp(webContext);
if (!CacheHelper.Exists(CK))
{
UpdateArticleHit(cid);
var userIP = HttpContextExtension.GetClientUserIp(webContext);
var location = HttpContextExtension.GetIpInfo(userIP);
itenant.InsertableWithAttr(new ArticleBrowsingLog()
{
ArticleId = cid,
UserId = userId,
Location = location,
UserIP = userIP,
AddTime = DateTime.Now,
}).ExecuteReturnSnowflakeId();
}
CacheHelper.SetCache(CK, 1, 10);
if (userId > 0)
{
model.IsPraise = Context.Queryable()
.Where(f => f.UserId == userId && f.ArticleId == cid && f.IsDelete == 0)
.Any() ? 1 : 0;
}
return model;
}
///
/// 审核通过
///
///
///
public int Passed(long[] idsArr)
{
int result = 0;
List monents = [];
foreach (var item in idsArr)
{
var model = GetFirst(x => x.Cid == item && x.AuditStatus == 0);
if (model == null) continue;
//model.Auditor = HttpContext.GetName();
model.AuditStatus = AuditStatusEnum.Passed;
var response = Update(w => w.Cid == model.Cid, it => new Article()
{
AuditStatus = model.AuditStatus,
//Auditor = model.Auditor,
});
result += response;
monents.Add(model);
}
var data = from a in monents
group a by new { a.UserId } into grp
select new
{
userid = grp.Key.UserId,
num = grp.Count()
};
foreach (var item in data)
{
_userMsgService.AddSysUserMsg(item.userid, "您发布的内容已通过审核", UserMsgType.SYSTEM);
}
return result;
}
///
/// 审核不通过
///
///
///
///
public int Reject(string reason, long[] idsArr)
{
int result = 0;
List monents = new();
foreach (var item in idsArr)
{
var model = GetFirst(x => x.Cid == item && x.AuditStatus == 0);
if (model == null) continue;
//model.Auditor = HttpContext.GetName();
model.AuditStatus = AuditStatusEnum.Reject;
var response = Update(w => w.Cid == model.Cid, it => new Article()
{
AuditStatus = model.AuditStatus,
//Auditor = model.Auditor,
});
result += response;
monents.Add(model);
}
var data = from a in monents
group a by new { a.UserId } into grp
select new
{
userid = grp.Key.UserId,
num = grp.Count()
};
foreach (var item in data)
{
//Console.WriteLine(item.useridx +"," + item.num);
string content = $"您发布的内容未通过审核。";
if (!string.IsNullOrEmpty(reason))
{
content += $"原因:{reason}";
}
_userMsgService.AddSysUserMsg(item.userid, content, UserMsgType.SYSTEM);
}
return result;
}
}
}