SysLangTextService.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. namespace Admin.NET.Core.Service;
  7. /// <summary>
  8. /// 翻译服务 🧩
  9. /// </summary>
  10. [ApiDescriptionSettings(Order = 100, Description = "翻译服务")]
  11. public partial class SysLangTextService : IDynamicApiController, ITransient
  12. {
  13. private readonly SqlSugarRepository<SysLangText> _sysLangTextRep;
  14. private readonly ISqlSugarClient _sqlSugarClient;
  15. public SysLangTextService(SqlSugarRepository<SysLangText> sysLangTextRep, ISqlSugarClient sqlSugarClient)
  16. {
  17. _sysLangTextRep = sysLangTextRep;
  18. _sqlSugarClient = sqlSugarClient;
  19. }
  20. /// <summary>
  21. /// 分页查询翻译表 🔖
  22. /// </summary>
  23. /// <param name="input"></param>
  24. /// <returns></returns>
  25. [DisplayName("分页查询翻译表")]
  26. [ApiDescriptionSettings(Name = "Page"), HttpPost]
  27. public async Task<SqlSugarPagedList<SysLangTextOutput>> Page(PageSysLangTextInput input)
  28. {
  29. input.Keyword = input.Keyword?.Trim();
  30. var query = _sysLangTextRep.AsQueryable()
  31. .WhereIF(!string.IsNullOrWhiteSpace(input.Keyword), u => u.EntityName.Contains(input.Keyword) || u.FieldName.Contains(input.Keyword) || u.LangCode.Contains(input.Keyword) || u.Content.Contains(input.Keyword))
  32. .WhereIF(!string.IsNullOrWhiteSpace(input.EntityName), u => u.EntityName.Contains(input.EntityName.Trim()))
  33. .WhereIF(!string.IsNullOrWhiteSpace(input.FieldName), u => u.FieldName.Contains(input.FieldName.Trim()))
  34. .WhereIF(!string.IsNullOrWhiteSpace(input.LangCode), u => u.LangCode.Contains(input.LangCode.Trim()))
  35. .WhereIF(!string.IsNullOrWhiteSpace(input.Content), u => u.Content.Contains(input.Content.Trim()))
  36. .WhereIF(input.EntityId != null, u => u.EntityId == input.EntityId)
  37. .Select<SysLangTextOutput>();
  38. return await query.OrderBuilder(input).ToPagedListAsync(input.Page, input.PageSize);
  39. }
  40. [DisplayName("获取翻译表")]
  41. [ApiDescriptionSettings(Name = "List"), HttpPost]
  42. public async Task<List<SysLangTextOutput>> List(ListSysLangTextInput input)
  43. {
  44. var query = _sysLangTextRep.AsQueryable()
  45. .Where(u => u.EntityName == input.EntityName.Trim() && u.FieldName == input.FieldName.Trim() && u.EntityId == input.EntityId)
  46. .WhereIF(!string.IsNullOrWhiteSpace(input.LangCode), u => u.LangCode == input.LangCode.Trim())
  47. .Select<SysLangTextOutput>();
  48. return await query.ToListAsync();
  49. }
  50. /// <summary>
  51. /// 获取翻译表详情 ℹ️
  52. /// </summary>
  53. /// <param name="input"></param>
  54. /// <returns></returns>
  55. [DisplayName("获取翻译表详情")]
  56. [ApiDescriptionSettings(Name = "Detail"), HttpGet]
  57. public async Task<SysLangText> Detail([FromQuery] QueryByIdSysLangTextInput input)
  58. {
  59. return await _sysLangTextRep.GetFirstAsync(u => u.Id == input.Id);
  60. }
  61. /// <summary>
  62. /// 增加翻译表 ➕
  63. /// </summary>
  64. /// <param name="input"></param>
  65. /// <returns></returns>
  66. [DisplayName("增加翻译表")]
  67. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  68. public async Task<long> Add(AddSysLangTextInput input)
  69. {
  70. var entity = input.Adapt<SysLangText>();
  71. return await _sysLangTextRep.InsertAsync(entity) ? entity.Id : 0;
  72. }
  73. /// <summary>
  74. /// 更新翻译表 ✏️
  75. /// </summary>
  76. /// <param name="input"></param>
  77. /// <returns></returns>
  78. [DisplayName("更新翻译表")]
  79. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  80. public async Task Update(UpdateSysLangTextInput input)
  81. {
  82. var entity = input.Adapt<SysLangText>();
  83. await _sysLangTextRep.AsUpdateable(entity)
  84. .ExecuteCommandAsync();
  85. }
  86. /// <summary>
  87. /// 删除翻译表 ❌
  88. /// </summary>
  89. /// <param name="input"></param>
  90. /// <returns></returns>
  91. [DisplayName("删除翻译表")]
  92. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  93. public async Task Delete(DeleteSysLangTextInput input)
  94. {
  95. var entity = await _sysLangTextRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  96. await _sysLangTextRep.DeleteAsync(entity); //真删除
  97. }
  98. /// <summary>
  99. /// 批量删除翻译表 ❌
  100. /// </summary>
  101. /// <param name="input"></param>
  102. /// <returns></returns>
  103. [DisplayName("批量删除翻译表")]
  104. [ApiDescriptionSettings(Name = "BatchDelete"), HttpPost]
  105. public async Task BatchDelete([Required(ErrorMessage = "主键列表不能为空")] List<DeleteSysLangTextInput> input)
  106. {
  107. var exp = Expressionable.Create<SysLangText>();
  108. foreach (var row in input) exp = exp.Or(it => it.Id == row.Id);
  109. var list = await _sysLangTextRep.AsQueryable().Where(exp.ToExpression()).ToListAsync();
  110. await _sysLangTextRep.DeleteAsync(list); //真删除
  111. }
  112. private static readonly object _sysLangTextBatchSaveLock = new object();
  113. /// <summary>
  114. /// 批量保存翻译表 ✏️
  115. /// </summary>
  116. /// <param name="input"></param>
  117. /// <returns></returns>
  118. [DisplayName("批量保存翻译表")]
  119. [ApiDescriptionSettings(Name = "BatchSave"), HttpPost]
  120. public void BatchSave([Required(ErrorMessage = "列表不能为空")] List<ImportSysLangTextInput> input)
  121. {
  122. lock (_sysLangTextBatchSaveLock)
  123. {
  124. // 校验并过滤必填基本类型为null的字段
  125. var rows = input.Where(x =>
  126. {
  127. if (!string.IsNullOrWhiteSpace(x.Error)) return false;
  128. if (x.EntityId == null)
  129. {
  130. x.Error = "所属实体ID不能为空";
  131. return false;
  132. }
  133. return true;
  134. }).Adapt<List<SysLangText>>();
  135. var storageable = _sysLangTextRep.Context.Storageable(rows)
  136. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.EntityName), "所属实体名不能为空")
  137. .SplitError(it => it.Item.EntityName?.Length > 255, "所属实体名长度不能超过255个字符")
  138. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.FieldName), "字段名不能为空")
  139. .SplitError(it => it.Item.FieldName?.Length > 255, "字段名长度不能超过255个字符")
  140. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.LangCode), "语言代码不能为空")
  141. .SplitError(it => it.Item.LangCode?.Length > 255, "语言代码长度不能超过255个字符")
  142. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.Content), "翻译内容不能为空")
  143. .WhereColumns(it => new { it.EntityId, it.EntityName, it.FieldName, it.LangCode })
  144. .SplitInsert(it => it.NotAny())
  145. .SplitUpdate(it => it.Any())
  146. .ToStorage();
  147. storageable.AsInsertable.ExecuteCommand();// 不存在插入
  148. storageable.AsUpdateable.UpdateColumns(it => new
  149. {
  150. it.EntityName,
  151. it.EntityId,
  152. it.FieldName,
  153. it.LangCode,
  154. it.Content,
  155. }).ExecuteCommand();// 存在更新
  156. if (storageable.ErrorList.Any())
  157. {
  158. throw Oops.Oh($"处理过程中出现以下错误:{string.Join(";", storageable.ErrorList.Distinct())}");
  159. }
  160. }
  161. }
  162. /// <summary>
  163. /// 导出翻译表记录 🔖
  164. /// </summary>
  165. /// <param name="input"></param>
  166. /// <returns></returns>
  167. [DisplayName("导出翻译表记录")]
  168. [ApiDescriptionSettings(Name = "Export"), HttpPost, NonUnify]
  169. public async Task<IActionResult> Export(PageSysLangTextInput input)
  170. {
  171. var list = (await Page(input)).Items?.Adapt<List<ExportSysLangTextOutput>>() ?? new();
  172. if (input.SelectKeyList?.Count > 0) list = list.Where(x => input.SelectKeyList.Contains(x.Id)).ToList();
  173. return ExcelHelper.ExportTemplate(list, "翻译表导出记录");
  174. }
  175. /// <summary>
  176. /// 下载翻译表数据导入模板 ⬇️
  177. /// </summary>
  178. /// <returns></returns>
  179. [DisplayName("下载翻译表数据导入模板")]
  180. [ApiDescriptionSettings(Name = "Import"), HttpGet, NonUnify]
  181. public IActionResult DownloadTemplate()
  182. {
  183. return ExcelHelper.ExportTemplate(new List<ExportSysLangTextOutput>(), "翻译表导入模板");
  184. }
  185. private static readonly object _sysLangTextImportLock = new object();
  186. /// <summary>
  187. /// 导入翻译表记录 💾
  188. /// </summary>
  189. /// <returns></returns>
  190. [DisplayName("导入翻译表记录")]
  191. [ApiDescriptionSettings(Name = "Import"), HttpPost, NonUnify, UnitOfWork]
  192. public IActionResult ImportData([Required] IFormFile file)
  193. {
  194. lock (_sysLangTextImportLock)
  195. {
  196. var stream = ExcelHelper.ImportData<ImportSysLangTextInput, SysLangText>(file, (list, markerErrorAction) =>
  197. {
  198. _sqlSugarClient.Utilities.PageEach(list, 2048, pageItems =>
  199. {
  200. // 校验并过滤必填基本类型为null的字段
  201. var rows = pageItems.Where(x =>
  202. {
  203. if (!string.IsNullOrWhiteSpace(x.Error)) return false;
  204. if (x.EntityId == null)
  205. {
  206. x.Error = "所属实体ID不能为空";
  207. return false;
  208. }
  209. return true;
  210. }).Adapt<List<SysLangText>>();
  211. var storageable = _sysLangTextRep.Context.Storageable(rows)
  212. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.EntityName), "所属实体名不能为空")
  213. .SplitError(it => it.Item.EntityName?.Length > 255, "所属实体名长度不能超过255个字符")
  214. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.FieldName), "字段名不能为空")
  215. .SplitError(it => it.Item.FieldName?.Length > 255, "字段名长度不能超过255个字符")
  216. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.LangCode), "语言代码不能为空")
  217. .SplitError(it => it.Item.LangCode?.Length > 255, "语言代码长度不能超过255个字符")
  218. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.Content), "翻译内容不能为空")
  219. .SplitError(it => it.Item.Content?.Length > 255, "翻译内容长度不能超过255个字符")
  220. .WhereColumns(it => new { it.EntityId, it.EntityName, it.FieldName, it.LangCode })
  221. .SplitInsert(it => it.NotAny())
  222. .SplitUpdate(it => it.Any())
  223. .ToStorage();
  224. storageable.AsInsertable.ExecuteCommand();// 不存在插入
  225. storageable.AsUpdateable.UpdateColumns(it => new
  226. {
  227. it.EntityName,
  228. it.EntityId,
  229. it.FieldName,
  230. it.LangCode,
  231. it.Content,
  232. }).ExecuteCommand();// 存在更新
  233. // 标记错误信息
  234. markerErrorAction.Invoke(storageable, pageItems, rows);
  235. });
  236. });
  237. return stream;
  238. }
  239. }
  240. }