ExcelHelper.cs 88 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254
  1. using NPOI.HSSF.UserModel;
  2. using System;
  3. using System.Collections;
  4. using System.Collections.Generic;
  5. using System.Globalization;
  6. using System.IO;
  7. using System.Linq;
  8. using System.Linq.Expressions;
  9. using System.Reflection;
  10. using System.Runtime.CompilerServices;
  11. using System.Security.Cryptography;
  12. using System.Text;
  13. using System.Text.RegularExpressions;
  14. using NPOI.HSSF.Util;
  15. using NPOI.SS.Formula.Functions;
  16. using NPOI.SS.UserModel;
  17. using NPOI.SS.Util;
  18. using NPOI.XSSF.UserModel;
  19. namespace ShwasherSys
  20. {
  21. public static class ExcelHelper
  22. {
  23. #region Excel导入
  24. /// <summary>
  25. /// 从Excel取数据并记录到List集合里
  26. /// </summary>
  27. /// <param name="cellHeader">单元头的值和名称:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  28. /// <param name="filePath">保存文件绝对路径</param>
  29. /// <param name="errorMsg">错误信息</param>
  30. /// <param name="startIndex">数据行开始序列,默认为1(即第二列,从0开始)</param>
  31. /// <returns>转换后的List对象集合</returns>
  32. public static List<T> ExcelToEntityList<T>(this Dictionary<string, string> cellHeader, string filePath, out StringBuilder errorMsg, int startIndex = 1) where T : new()
  33. {
  34. List<T> enlist = new List<T>();
  35. errorMsg = new StringBuilder();
  36. try
  37. {
  38. if (Regex.IsMatch(filePath, ".xls$")) // 2003
  39. {
  40. enlist = Excel2003ToEntityList<T>(cellHeader, filePath, out errorMsg, startIndex);
  41. }
  42. else if (Regex.IsMatch(filePath, ".xlsx$")) // 2007
  43. {
  44. enlist = Excel2007ToEntityList<T>(cellHeader, filePath, out errorMsg, startIndex);
  45. }
  46. return enlist;
  47. }
  48. catch (Exception ex)
  49. {
  50. //typeof(ExcelHelper).LogError(ex);
  51. return default(List<T>);
  52. }
  53. }
  54. /// <summary>
  55. /// 从Excel2003取数据并记录到List集合里
  56. /// </summary>
  57. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  58. /// <param name="filePath">保存文件绝对路径</param>
  59. /// <param name="errorMsg">错误信息</param>
  60. /// <param name="startIndex"></param>
  61. /// <returns>转换好的List对象集合</returns>
  62. private static List<T> Excel2003ToEntityList<T>(this Dictionary<string, string> cellHeader, string filePath, out StringBuilder errorMsg, int startIndex = 1) where T : new()
  63. {
  64. errorMsg = new StringBuilder(); // 错误信息,Excel转换到实体对象时,会有格式的错误信息
  65. List<T> enlist = new List<T>(); // 转换后的集合
  66. try
  67. {
  68. using (FileStream fs = File.OpenRead(filePath))
  69. {
  70. HSSFWorkbook workbook = new HSSFWorkbook(fs);
  71. HSSFSheet sheet = (HSSFSheet)workbook.GetSheetAt(0); // 获取此文件第一个Sheet页
  72. for (int rowIndex = startIndex; rowIndex <= sheet.LastRowNum; rowIndex++)
  73. {
  74. // 1.判断当前行是否空行,若空行就不在进行读取下一行操作,结束Excel读取操作
  75. IRow row = sheet.GetRow(rowIndex);
  76. if (row == null)
  77. {
  78. break;
  79. }
  80. // 2.每一个Excel row转换为一个实体对象
  81. T en = new T();
  82. ExcelRowToEntity(cellHeader, row, rowIndex, en, ref errorMsg);
  83. enlist.Add(en);
  84. }
  85. }
  86. return enlist;
  87. }
  88. catch (Exception ex)
  89. {
  90. //typeof(ExcelHelper).LogError(ex);
  91. return default(List<T>);
  92. }
  93. }
  94. /// <summary>
  95. /// 从Excel2007取数据并记录到List集合里
  96. /// </summary>
  97. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  98. /// <param name="filePath">保存文件绝对路径</param>
  99. /// <param name="errorMsg">错误信息</param>
  100. /// <param name="startIndex">数据行开始序列,默认为1(即第二列,从0开始)</param>
  101. /// <returns>转换好的List对象集合</returns>
  102. private static List<T> Excel2007ToEntityList<T>(this Dictionary<string, string> cellHeader, string filePath, out StringBuilder errorMsg, int startIndex = 1) where T : new()
  103. {
  104. errorMsg = new StringBuilder(); // 错误信息,Excel转换到实体对象时,会有格式的错误信息
  105. List<T> enlist = new List<T>(); // 转换后的集合
  106. try
  107. {
  108. using (FileStream fs = File.OpenRead(filePath))
  109. {
  110. XSSFWorkbook workbook = new XSSFWorkbook(fs);
  111. XSSFSheet sheet = (XSSFSheet)workbook.GetSheetAt(0); // 获取此文件第一个Sheet页
  112. for (int rowIndex = startIndex; rowIndex <= sheet.LastRowNum; rowIndex++)
  113. {
  114. // 1.判断当前行是否空行,若空行就不在进行读取下一行操作,结束Excel读取操作
  115. IRow row = sheet.GetRow(rowIndex);
  116. if (row == null)
  117. {
  118. break;
  119. }
  120. // 2.每一个Excel row转换为一个实体对象
  121. T en = new T();
  122. ExcelRowToEntity(cellHeader, row, rowIndex, en, ref errorMsg);
  123. enlist.Add(en);
  124. }
  125. }
  126. return enlist;
  127. }
  128. catch (Exception ex)
  129. {
  130. //typeof(ExcelHelper).LogError(ex);
  131. return default(List<T>);
  132. }
  133. }
  134. #endregion Excel导入
  135. #region Excel导出
  136. public static string ToExcel2003(List<ToExcelObj> cellHeader, IList enList,
  137. string sheetName, string filePath)
  138. {
  139. var lcRetVal = "";
  140. try
  141. {
  142. string fileName = "D-" + sheetName + "-" + DateTime.Now.ToString("yyyyMMddHHmmssfff") + ".xls"; // 文件名称
  143. if (!Directory.Exists(filePath))
  144. {
  145. Directory.CreateDirectory(filePath);
  146. }
  147. filePath = Path.Combine(filePath, fileName);
  148. // 2.解析单元格头部,设置单元头的中文名称
  149. HSSFWorkbook workbook = new HSSFWorkbook(); // 工作簿
  150. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  151. IRow row = sheet.CreateRow(0);//创建
  152. string headcss = $"bgc:{ColorType.LightBlue.ToString()};" +
  153. $"warp:{HorizontalAlignment.Center.ToString()};" +
  154. $"align:{HorizontalAlignment.Center.ToString()};" +
  155. $"v-align:{VerticalAlignment.Center.ToString()};" +
  156. $"b:{BorderStyle.Medium.ToString()};" +
  157. $"bc:{ColorType.Black.ToString()};" +
  158. "inden:0;" +
  159. "df:@;"+
  160. $"fc:{ColorType.White.ToString()};" +//font-color
  161. "fn:宋体;" +//font-name
  162. "fs:15;" +//font-size
  163. "fw:normal;" +//font-weight
  164. "fu:none;" +//font-underline
  165. "fi:false;" +//font-italic
  166. "fst:false;" +//font-strikeout
  167. "fss:none;";//font-superscript;
  168. for (int i = 0; i < cellHeader.Count; i++)
  169. {
  170. var cellhead = row.CreateCell(i);
  171. cellhead.SetCellValue(cellHeader[i].ShowColumn); // 列名为Key的值
  172. CellStyleCss.Instants.Css(cellhead, headcss);
  173. sheet.AutoSizeColumn(i);
  174. }
  175. // 3.List对象的值赋值到Excel的单元格里
  176. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  177. foreach (var en in enList)
  178. {
  179. IRow rowTmp = sheet.CreateRow(rowIndex);
  180. for (int i = 0; i < cellHeader.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  181. {
  182. string cellValue = ""; // 单元格的值
  183. object properotyValue = null; // 属性的值
  184. PropertyInfo properotyInfo = en.GetType().GetProperty(cellHeader[i].MapColumn);
  185. if (properotyInfo != null)
  186. {
  187. properotyValue = properotyInfo.GetValue(en, null);
  188. }
  189. // 3.3 属性值经过转换赋值给单元格值
  190. if (properotyValue != null)
  191. {
  192. cellValue = properotyValue.ToString();
  193. // 3.3.1 对时间初始值赋值为空
  194. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  195. {
  196. cellValue = "";
  197. }
  198. }
  199. // 3.4 填充到Excel的单元格里
  200. var cl= rowTmp.CreateCell(i);
  201. cl.SetCellValue(cellValue);
  202. //CellStyleCss.Instants.Css(cl, cellHeader[i].StyleStr);
  203. //sheet.AutoSizeColumn(i);
  204. }
  205. rowIndex++;
  206. }
  207. // 4.生成文件
  208. FileStream file = new FileStream(filePath, FileMode.Create);
  209. workbook.Write(file);
  210. file.Close();
  211. lcRetVal = fileName;
  212. }
  213. catch (Exception ex)
  214. {
  215. LogHelper.LogError("excel导出",ex);
  216. }
  217. return lcRetVal;
  218. }
  219. /// <summary>
  220. /// 实体类集合导出到EXCLE2003
  221. /// </summary>
  222. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  223. /// <param name="enList">数据源</param>
  224. /// <param name="sheetName">工作表名称</param>
  225. /// <param name="filePath">文件的下载地址</param>
  226. /// <returns></returns>
  227. public static string EntityListToExcel2003(this Dictionary<string, string> cellHeader, IList enList, string sheetName, string filePath)
  228. {
  229. var lcRetVal = "";
  230. try
  231. {
  232. string fileName = "D-" + sheetName + "-" + DateTime.Now.ToString("yyyyMMddHHmmssfff") + ".xls"; // 文件名称
  233. //string urlPath = "UpFiles/ExcelFiles/" + fileName; // 文件下载的URL地址,供给前台下载
  234. //string filePath = HttpContext.Current.Server.MapPath("\\" + urlPath); // 文件路径
  235. // 1.检测是否存在文件夹,若不存在就建立个文件夹
  236. //string directoryName = Path.GetDirectoryName(filePath);
  237. if (!Directory.Exists(filePath))
  238. {
  239. Directory.CreateDirectory(filePath);
  240. }
  241. filePath = Path.Combine(filePath, fileName);
  242. // 2.解析单元格头部,设置单元头的中文名称
  243. HSSFWorkbook workbook = new HSSFWorkbook(); // 工作簿
  244. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  245. IRow row = sheet.CreateRow(0);
  246. List<string> keys = cellHeader.Keys.ToList();
  247. for (int i = 0; i < keys.Count; i++)
  248. {
  249. row.CreateCell(i).SetCellValue(cellHeader[keys[i]]); // 列名为Key的值
  250. }
  251. // 3.List对象的值赋值到Excel的单元格里
  252. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  253. foreach (var en in enList)
  254. {
  255. IRow rowTmp = sheet.CreateRow(rowIndex);
  256. for (int i = 0; i < keys.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  257. {
  258. string cellValue = ""; // 单元格的值
  259. object properotyValue = null; // 属性的值
  260. System.Reflection.PropertyInfo properotyInfo; // 属性的信息
  261. // 3.1 若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.UserName
  262. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  263. {
  264. // 3.1.1 解析子类属性(这里只解析1层子类,多层子类未处理)
  265. string[] properotyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  266. string subClassName = properotyArray[0]; // '.'前面的为子类的名称
  267. string subClassProperotyName = properotyArray[1]; // '.'后面的为子类的属性名称
  268. System.Reflection.PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  269. if (subClassInfo != null)
  270. {
  271. // 3.1.2 获取子类的实例
  272. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  273. // 3.1.3 根据属性名称获取子类里的属性类型
  274. properotyInfo = subClassInfo.PropertyType.GetProperty(subClassProperotyName);
  275. if (properotyInfo != null)
  276. {
  277. properotyValue = properotyInfo.GetValue(subClassEn, null); // 获取子类属性的值
  278. }
  279. }
  280. }
  281. else
  282. {
  283. // 3.2 若不是子类的属性,直接根据属性名称获取对象对应的属性
  284. properotyInfo = en.GetType().GetProperty(keys[i]);
  285. if (properotyInfo != null)
  286. {
  287. properotyValue = properotyInfo.GetValue(en, null);
  288. }
  289. }
  290. // 3.3 属性值经过转换赋值给单元格值
  291. if (properotyValue != null)
  292. {
  293. cellValue = properotyValue.ToString();
  294. // 3.3.1 对时间初始值赋值为空
  295. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  296. {
  297. cellValue = "";
  298. }
  299. }
  300. // 3.4 填充到Excel的单元格里
  301. rowTmp.CreateCell(i).SetCellValue(cellValue);
  302. }
  303. rowIndex++;
  304. }
  305. // 4.生成文件
  306. FileStream file = new FileStream(filePath, FileMode.Create);
  307. workbook.Write(file);
  308. file.Close();
  309. lcRetVal = fileName;
  310. }
  311. catch (Exception ex)
  312. {
  313. //typeof(ExcelHelper).LogError(ex);
  314. }
  315. return lcRetVal;
  316. }
  317. public static string EntityListToExcel2003(this HSSFWorkbook workbook,
  318. string sheetName, string filePath)
  319. {
  320. var lcRetVal = "";
  321. try
  322. {
  323. string fileName = "D-" + sheetName + "-" + DateTime.Now.ToString("yyyyMMddHHmmssfff") + ".xls"; // 文件名称
  324. if (!Directory.Exists(filePath))
  325. {
  326. Directory.CreateDirectory(filePath);
  327. }
  328. filePath = Path.Combine(filePath, fileName);
  329. FileStream file = new FileStream(filePath, FileMode.Create);
  330. workbook.Write(file);
  331. file.Close();
  332. lcRetVal = fileName;
  333. }
  334. catch (Exception ex)
  335. {
  336. //typeof(ExcelHelper).LogError(ex);
  337. }
  338. return lcRetVal;
  339. }
  340. public static HSSFWorkbook EntityListToExcel2003book(this Dictionary<string, string> cellHeader, IList enList,
  341. string sheetName)
  342. {
  343. HSSFWorkbook workbook = new HSSFWorkbook(); // 工作簿
  344. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  345. IRow row = sheet.CreateRow(0);
  346. List<string> keys = cellHeader.Keys.ToList();
  347. for (int i = 0; i < keys.Count; i++)
  348. {
  349. row.CreateCell(i).SetCellValue(cellHeader[keys[i]]); // 列名为Key的值
  350. }
  351. // 3.List对象的值赋值到Excel的单元格里
  352. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  353. foreach (var en in enList)
  354. {
  355. IRow rowTmp = sheet.CreateRow(rowIndex);
  356. for (int i = 0; i < keys.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  357. {
  358. string cellValue = ""; // 单元格的值
  359. object properotyValue = null; // 属性的值
  360. System.Reflection.PropertyInfo properotyInfo; // 属性的信息
  361. // 3.1 若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.UserName
  362. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  363. {
  364. // 3.1.1 解析子类属性(这里只解析1层子类,多层子类未处理)
  365. string[] properotyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  366. string subClassName = properotyArray[0]; // '.'前面的为子类的名称
  367. string subClassProperotyName = properotyArray[1]; // '.'后面的为子类的属性名称
  368. System.Reflection.PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  369. if (subClassInfo != null)
  370. {
  371. // 3.1.2 获取子类的实例
  372. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  373. // 3.1.3 根据属性名称获取子类里的属性类型
  374. properotyInfo = subClassInfo.PropertyType.GetProperty(subClassProperotyName);
  375. if (properotyInfo != null)
  376. {
  377. properotyValue = properotyInfo.GetValue(subClassEn, null); // 获取子类属性的值
  378. }
  379. }
  380. }
  381. else
  382. {
  383. // 3.2 若不是子类的属性,直接根据属性名称获取对象对应的属性
  384. properotyInfo = en.GetType().GetProperty(keys[i]);
  385. if (properotyInfo != null)
  386. {
  387. properotyValue = properotyInfo.GetValue(en, null);
  388. }
  389. }
  390. // 3.3 属性值经过转换赋值给单元格值
  391. if (properotyValue != null)
  392. {
  393. cellValue = properotyValue.ToString();
  394. // 3.3.1 对时间初始值赋值为空
  395. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  396. {
  397. cellValue = "";
  398. }
  399. }
  400. // 3.4 填充到Excel的单元格里
  401. rowTmp.CreateCell(i).SetCellValue(cellValue);
  402. }
  403. rowIndex++;
  404. }
  405. return workbook;
  406. }
  407. /// <summary>
  408. /// 实体类集合导出到EXCLE2007
  409. /// </summary>
  410. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  411. /// <param name="enList">数据源</param>
  412. /// <param name="sheetName">工作表名称</param>
  413. /// <param name="filePath">文件的下载地址</param>
  414. /// <returns></returns>
  415. public static string EntityListToExcel2007(this Dictionary<string, string> cellHeader, IList enList, string sheetName, string filePath)
  416. {
  417. var lcRetVal = "";
  418. try
  419. {
  420. string fileName = "D-" + sheetName + "-" + DateTime.Now.ToString("yyyyMMddHHmmssfff") + ".xlsx"; // 文件名称
  421. // 1.检测是否存在文件夹,若不存在就建立个文件夹
  422. if (!Directory.Exists(filePath))
  423. {
  424. Directory.CreateDirectory(filePath);
  425. }
  426. filePath = Path.Combine(filePath, fileName);
  427. // 2.解析单元格头部,设置单元头的中文名称
  428. XSSFWorkbook workbook = new XSSFWorkbook(); // 工作簿
  429. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  430. IRow row = sheet.CreateRow(0);
  431. List<string> keys = cellHeader.Keys.ToList();
  432. for (int i = 0; i < keys.Count; i++)
  433. {
  434. row.CreateCell(i).SetCellValue(cellHeader[keys[i]]); // 列名为Key的值
  435. }
  436. // 3.List对象的值赋值到Excel的单元格里
  437. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  438. foreach (var en in enList)
  439. {
  440. IRow rowTmp = sheet.CreateRow(rowIndex);
  441. for (int i = 0; i < keys.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  442. {
  443. string cellValue = ""; // 单元格的值
  444. object properotyValue = null; // 属性的值
  445. System.Reflection.PropertyInfo properotyInfo; // 属性的信息
  446. // 3.1 若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.UserName
  447. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  448. {
  449. // 3.1.1 解析子类属性(这里只解析1层子类,多层子类未处理)
  450. string[] properotyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  451. string subClassName = properotyArray[0]; // '.'前面的为子类的名称
  452. string subClassProperotyName = properotyArray[1]; // '.'后面的为子类的属性名称
  453. System.Reflection.PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  454. if (subClassInfo != null)
  455. {
  456. // 3.1.2 获取子类的实例
  457. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  458. // 3.1.3 根据属性名称获取子类里的属性类型
  459. properotyInfo = subClassInfo.PropertyType.GetProperty(subClassProperotyName);
  460. if (properotyInfo != null)
  461. {
  462. properotyValue = properotyInfo.GetValue(subClassEn, null); // 获取子类属性的值
  463. }
  464. }
  465. }
  466. else
  467. {
  468. // 3.2 若不是子类的属性,直接根据属性名称获取对象对应的属性
  469. properotyInfo = en.GetType().GetProperty(keys[i]);
  470. if (properotyInfo != null)
  471. {
  472. properotyValue = properotyInfo.GetValue(en, null);
  473. }
  474. }
  475. // 3.3 属性值经过转换赋值给单元格值
  476. if (properotyValue != null)
  477. {
  478. cellValue = properotyValue.ToString();
  479. // 3.3.1 对时间初始值赋值为空
  480. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  481. {
  482. cellValue = "";
  483. }
  484. }
  485. // 3.4 填充到Excel的单元格里
  486. rowTmp.CreateCell(i).SetCellValue(cellValue);
  487. }
  488. rowIndex++;
  489. }
  490. // 4.生成文件
  491. FileStream file = new FileStream(filePath, FileMode.Create);
  492. workbook.Write(file);
  493. file.Close();
  494. lcRetVal = fileName;
  495. }
  496. catch (Exception ex)
  497. {
  498. //typeof(ExcelHelper).LogError(ex);
  499. }
  500. return lcRetVal;
  501. }
  502. #endregion Excel导出
  503. #region Common
  504. /// <summary>
  505. /// Excel row转换为实体对象
  506. /// </summary>
  507. /// <typeparam name="T"></typeparam>
  508. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  509. /// <param name="row">Excel row</param>
  510. /// <param name="rowIndex">row index</param>
  511. /// <param name="en">实体</param>
  512. /// <param name="errorMsg">错误信息</param>
  513. private static void ExcelRowToEntity<T>(Dictionary<string, string> cellHeader, IRow row, int rowIndex, T en, ref StringBuilder errorMsg)
  514. {
  515. List<string> keys = cellHeader.Keys.ToList(); // 要赋值的实体对象属性名称
  516. string errStr = ""; // 当前行转换时,是否有错误信息,格式为:第1行数据转换异常:XXX列;
  517. for (int i = 0; i < keys.Count; i++)
  518. {
  519. // 1.若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.TrueName
  520. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  521. {
  522. // 1)解析子类属性
  523. string[] properotyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  524. string subClassName = properotyArray[0]; // '.'前面的为子类的名称
  525. string subClassProperotyName = properotyArray[1]; // '.'后面的为子类的属性名称
  526. System.Reflection.PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  527. if (subClassInfo != null)
  528. {
  529. // 2)获取子类的实例
  530. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  531. // 3)根据属性名称获取子类里的属性信息
  532. System.Reflection.PropertyInfo properotyInfo = subClassInfo.PropertyType.GetProperty(subClassProperotyName);
  533. if (properotyInfo != null)
  534. {
  535. try
  536. {
  537. // Excel单元格的值转换为对象属性的值,若类型不对,记录出错信息
  538. properotyInfo.SetValue(subClassEn, GetExcelCellToProperty(properotyInfo.PropertyType, row.GetCell(i)), null);
  539. }
  540. catch (Exception e)
  541. {
  542. //typeof(ExcelHelper).LogError(ex);
  543. if (errStr.Length == 0)
  544. {
  545. errStr = "第" + rowIndex + "行数据转换异常:";
  546. }
  547. errStr += cellHeader[keys[i]] + "列;";
  548. }
  549. }
  550. }
  551. }
  552. else
  553. {
  554. // 2.给指定的属性赋值
  555. System.Reflection.PropertyInfo properotyInfo = en.GetType().GetProperty(keys[i]);
  556. if (properotyInfo != null)
  557. {
  558. try
  559. {
  560. // Excel单元格的值转换为对象属性的值,若类型不对,记录出错信息
  561. properotyInfo.SetValue(en, GetExcelCellToProperty(properotyInfo.PropertyType, row.GetCell(i)), null);
  562. }
  563. catch (Exception e)
  564. {
  565. //typeof(ExcelHelper).LogError(ex);
  566. if (errStr.Length == 0)
  567. {
  568. errStr = "第" + rowIndex + "行数据转换异常:";
  569. }
  570. errStr += cellHeader[keys[i]] + "列;";
  571. }
  572. }
  573. }
  574. }
  575. // 若有错误信息,就添加到错误信息里
  576. if (errStr.Length > 0)
  577. {
  578. errorMsg.AppendLine(errStr);
  579. }
  580. }
  581. /// <summary>
  582. /// Excel Cell转换为实体的属性值
  583. /// </summary>
  584. /// <param name="distanceType">目标对象类型</param>
  585. /// <param name="sourceCell">对象属性的值</param>
  586. private static object GetExcelCellToProperty(Type distanceType, ICell sourceCell)
  587. {
  588. object rs = distanceType.IsValueType ? Activator.CreateInstance(distanceType) : null;
  589. // 1.判断传递的单元格是否为空
  590. if (sourceCell == null || string.IsNullOrEmpty(sourceCell.ToString()))
  591. {
  592. return rs;
  593. }
  594. // 2.Excel文本和数字单元格转换,在Excel里文本和数字是不能进行转换,所以这里预先存值
  595. object sourceValue = null;
  596. switch (sourceCell.CellType)
  597. {
  598. case CellType.Blank:
  599. break;
  600. case CellType.Boolean:
  601. break;
  602. case CellType.Error:
  603. break;
  604. case CellType.Formula:
  605. break;
  606. case CellType.Numeric:
  607. sourceValue = sourceCell.NumericCellValue;
  608. break;
  609. case CellType.String:
  610. sourceValue = sourceCell.StringCellValue;
  611. break;
  612. case CellType.Unknown:
  613. break;
  614. }
  615. string valueDataType = distanceType.Name;
  616. // 在这里进行特定类型的处理
  617. switch (valueDataType.ToLower()) // 以防出错,全部小写
  618. {
  619. case "string":
  620. rs = sourceValue?.ToString();
  621. break;
  622. case "int":
  623. case "int16":
  624. case "int32":
  625. rs = (int)Convert.ChangeType(sourceCell.NumericCellValue.ToString(CultureInfo.InvariantCulture), distanceType);
  626. break;
  627. case "float":
  628. case "single":
  629. rs = (float)Convert.ChangeType(sourceCell.NumericCellValue.ToString(CultureInfo.InvariantCulture), distanceType);
  630. break;
  631. case "datetime":
  632. rs = sourceCell.DateCellValue;
  633. break;
  634. case "guid":
  635. rs = (Guid)Convert.ChangeType(sourceCell.NumericCellValue.ToString(CultureInfo.InvariantCulture), distanceType);
  636. return rs;
  637. }
  638. return rs;
  639. }
  640. #endregion
  641. public static HSSFWorkbook CreateWorkBook03(string filePath = null)
  642. {
  643. if (string.IsNullOrEmpty(filePath))
  644. {
  645. return new HSSFWorkbook();
  646. }
  647. FileStream file = new FileStream(filePath, FileMode.Open);
  648. return new HSSFWorkbook(file);
  649. }
  650. public static XSSFWorkbook CreateWorkBook07(string filePath = null)
  651. {
  652. if (string.IsNullOrEmpty(filePath))
  653. {
  654. return new XSSFWorkbook();
  655. }
  656. return new XSSFWorkbook(filePath);
  657. }
  658. /// <summary>
  659. /// 创建Sheet
  660. /// </summary>
  661. /// <param name="sheetName"></param>
  662. /// <param name="defaultWidth"></param>
  663. /// <param name="defaultHeight"></param>
  664. /// <param name="is07"></param>
  665. /// <param name="workbook"></param>
  666. /// <returns></returns>
  667. public static ISheet GenerateSheet(this string sheetName, int defaultWidth = 20, int defaultHeight = 20, bool is07 = true, IWorkbook workbook = null)
  668. {
  669. workbook = is07 ? workbook ?? CreateWorkBook07() : workbook ?? CreateWorkBook03();
  670. var sheet = workbook.CreateSheet(sheetName);
  671. sheet.DefaultColumnWidth = defaultWidth;
  672. sheet.DefaultRowHeight = (short)(defaultHeight * 20);
  673. return sheet;
  674. }
  675. /// <summary>
  676. /// 创建行
  677. /// </summary>
  678. /// <param name="sheet">表</param>
  679. /// <param name="rowIndex">第几行(从1开始计数)</param>
  680. /// <returns></returns>
  681. public static IRow GenerateRow(this ISheet sheet, int rowIndex)
  682. {
  683. rowIndex = rowIndex <= 0 ? 1 : rowIndex;
  684. var row = sheet.GetRow(rowIndex - 1) ?? sheet.CreateRow(rowIndex - 1);
  685. return row;
  686. }
  687. //插入
  688. public static void InsertRows(this ISheet sheet, int insertRowIndex, List<IRow> formatRows)
  689. {
  690. foreach (var row in formatRows)
  691. {
  692. var r = row.Sheet.GenerateRow(insertRowIndex);
  693. if (r != null)
  694. {
  695. row.Sheet.RemoveRow(r);
  696. }
  697. row.CopyRowTo(insertRowIndex - 1);
  698. insertRowIndex++;
  699. }
  700. }
  701. //取
  702. public static List<IRow> GetRows(this ISheet sheet, int startRowIndex,int endRowIndex)
  703. {
  704. List<IRow> rows = new List<IRow>();
  705. for (int i = 0; i <= endRowIndex-startRowIndex; i++)
  706. {
  707. rows.Add(sheet.GenerateRow(startRowIndex+i));
  708. }
  709. return rows;
  710. }
  711. /// <summary>
  712. /// 创建单元格
  713. /// </summary>
  714. /// <param name="sheet">表</param>
  715. /// <param name="rowIndex">第几行(从1开始计数)</param>
  716. /// <param name="columnIndex">第几列(从1开始计数)</param>
  717. /// <param name="val">值</param>
  718. /// <param name="cellType">单元格类型</param>
  719. /// <returns></returns>
  720. public static ICell GenerateCell(this ISheet sheet, int rowIndex, int columnIndex, string val = null, CellType cellType = CellType.String)
  721. {
  722. var row = sheet.GenerateRow(rowIndex);
  723. var cell = GenerateCell(row, columnIndex, val, cellType);
  724. return cell;
  725. }
  726. /// <summary>
  727. /// 创建单元格
  728. /// </summary>
  729. /// <param name="row">行</param>
  730. /// <param name="columnIndex">第几列(从1开始计数)</param>
  731. /// <param name="val">值</param>
  732. /// <param name="cellType">单元格类型</param>
  733. /// <returns></returns>
  734. public static ICell GenerateCell(this IRow row, int columnIndex, string val = null, CellType cellType = CellType.String)
  735. {
  736. columnIndex = columnIndex <= 0 ? 1 : columnIndex;
  737. var cell = row.GetCell(columnIndex - 1) ?? row.CreateCell(columnIndex - 1);
  738. if (!string.IsNullOrEmpty(val))
  739. {
  740. cell.SetCellValue(val);
  741. cell.SetCellType(cellType);
  742. }
  743. return cell;
  744. }
  745. /// <summary>
  746. /// 设置列宽度
  747. /// </summary>
  748. /// <param name="sheet">表</param>
  749. /// <param name="columnIndex">第几列(从1开始计数)</param>
  750. /// <param name="width">宽度</param>
  751. /// <returns></returns>
  752. public static ISheet SetCellWidth(this ISheet sheet, int columnIndex, int width)
  753. {
  754. sheet.SetColumnWidth(columnIndex - 1, width * 256);
  755. return sheet;
  756. }
  757. /// <summary>
  758. /// 设置列宽度
  759. /// </summary>
  760. /// <param name="row">行</param>
  761. /// <param name="columnIndex">第几列(从1开始计数)</param>
  762. /// <param name="width">宽度</param>
  763. /// <returns></returns>
  764. public static IRow SetCellWidth(this IRow row, int columnIndex, int width)
  765. {
  766. row.Sheet.SetColumnWidth(columnIndex - 1, width * 256);
  767. return row;
  768. }
  769. /// <summary>
  770. /// 设置列宽度
  771. /// </summary>
  772. /// <param name="cell">单元格</param>
  773. /// <param name="columnIndex">第几列(从1开始计数)</param>
  774. /// <param name="width">宽度</param>
  775. /// <returns></returns>
  776. public static ICell SetCellWidth(this ICell cell, int columnIndex, int width)
  777. {
  778. cell.Sheet.SetColumnWidth(columnIndex - 1, width * 256);
  779. return cell;
  780. }
  781. /// <summary>
  782. /// 设置行高度
  783. /// </summary>
  784. /// <param name="sheet">表</param>
  785. /// <param name="rowIndex">第几行(从1开始计数)</param>
  786. /// <param name="height">宽度</param>
  787. /// <returns></returns>
  788. public static ISheet SetRowHeight(this ISheet sheet, int rowIndex, int height)
  789. {
  790. sheet.GetRow(rowIndex - 1).SetRowHeight(height);
  791. return sheet;
  792. }
  793. /// <summary>
  794. /// 设置行高度
  795. /// </summary>
  796. /// <param name="row">行</param>
  797. /// <param name="height">宽度</param>
  798. /// <returns></returns>
  799. public static IRow SetRowHeight(this IRow row, int height)
  800. {
  801. row.Height = (short)(height * 20);
  802. return row;
  803. }
  804. /// <summary>
  805. /// 设置行高度
  806. /// </summary>
  807. /// <param name="cell">单元格</param>
  808. /// <param name="height">宽度</param>
  809. /// <returns></returns>
  810. public static ICell SetRowHeight(this ICell cell, int height)
  811. {
  812. cell.Row.SetRowHeight(height);
  813. return cell;
  814. }
  815. /// <summary>
  816. /// 设置单元格样式
  817. /// </summary>
  818. /// <param name="cell"></param>
  819. /// <param name="styleStr">样式字符串</param>
  820. /// <param name="instants">样式转换器</param>
  821. /// <returns></returns>
  822. public static ICell SetCellCss(this ICell cell, CellStyleCss instants = null, string styleStr = null)
  823. {
  824. instants = instants ?? CellStyleCss.Instants;
  825. return instants.Css(cell, styleStr);
  826. }
  827. public static ICell SetValue<T>(this ICell cell, object obj,ICellStyle cellStyle)
  828. {
  829. cell.CellStyle = cellStyle;
  830. return SetValue<T>(cell, obj);
  831. }
  832. public static ICell SetValue(this ICell cell, string obj, ICellStyle cellStyle)
  833. {
  834. cell.CellStyle = cellStyle;
  835. return SetValue<string>(cell, obj);
  836. }
  837. /// <summary>
  838. /// 设置单元格的值
  839. /// </summary>
  840. /// <param name="cell"></param>
  841. /// <param name="obj"></param>
  842. /// <returns></returns>
  843. public static ICell SetValue(this ICell cell, string obj)
  844. {
  845. obj = obj ?? "";
  846. return SetValue<string>(cell, obj);
  847. }
  848. /// <summary>
  849. /// 设置单元格的值
  850. /// </summary>
  851. /// <typeparam name="T">值类型(sting,bool,double,DateTime)</typeparam>
  852. /// <param name="cell"></param>
  853. /// <param name="obj"></param>
  854. /// <returns></returns>
  855. public static ICell SetValue<T>(this ICell cell, object obj)
  856. {
  857. Type type = typeof(T);
  858. if (type == typeof(string))
  859. {
  860. cell.SetCellValue(obj.ToString());
  861. }
  862. else if (type == typeof(bool) && bool.TryParse(obj.ToString(), out var bValue))
  863. {
  864. cell.SetCellValue(bValue);
  865. }
  866. else if ((type == typeof(int) || type == typeof(double) || type == typeof(decimal)) && double.TryParse(obj.ToString(), out var iValue))
  867. {
  868. cell.SetCellValue(iValue);
  869. }
  870. else if (type == typeof(DateTime) && DateTime.TryParse(obj.ToString(), out var dValue))
  871. {
  872. cell.SetCellValue(dValue);
  873. }
  874. return cell;
  875. }
  876. #region DATA-FORMAT
  877. public static ICellStyle SetCellDateTime(this ICell cell, DateTime value, string formatStr = "yyyy-MM-dd HH:mm:ss")
  878. {
  879. cell.SetCellValue(value);
  880. cell.SetDataFormat(formatStr);
  881. return cell.CellStyle;
  882. }
  883. public static ICellStyle SetCellString(this ICell cell, double value, string formatStr = "@")
  884. {
  885. cell.SetCellValue(value);
  886. cell.SetDataFormat(formatStr);
  887. return cell.CellStyle;
  888. }
  889. public static ICellStyle SetCellDouble(this ICell cell, double value, string formatStr = "0.00")
  890. {
  891. cell.SetCellValue(value);
  892. cell.SetDataFormat(formatStr);
  893. return cell.CellStyle;
  894. }
  895. public static ICellStyle SetCellCurrency(this ICell cell, double value, string formatStr = "¥#,###.##")
  896. {
  897. cell.SetCellValue(value);
  898. cell.SetDataFormat(formatStr);
  899. return cell.CellStyle;
  900. }
  901. public static ICellStyle SetCellPercent(this ICell cell, double value, string formatStr = "0.00%")
  902. {
  903. cell.SetCellValue(value);
  904. cell.SetDataFormat(formatStr);
  905. return cell.CellStyle;
  906. }
  907. public static void SetDataFormat(this ICell cell, string formatStr = "¥#,##0.00")
  908. {
  909. var cellStyle = cell.CellStyle ?? GetCellStyle(cell);
  910. var format = cell.Sheet.Workbook.CreateDataFormat();
  911. cellStyle.DataFormat = format.GetFormat(formatStr);
  912. cell.SetCellStyle(cellStyle);
  913. }
  914. #endregion
  915. #region 合并单元格
  916. /// <summary>
  917. /// 合并单元格
  918. /// </summary>
  919. /// <param name="sheet"></param>
  920. /// <param name="firstRow">开始行数(从1开始计数)</param>
  921. /// <param name="lastRow">结束行数(从1开始计数)</param>
  922. /// <param name="firstColumn">开始列数(从1开始计数)</param>
  923. /// <param name="lastColumn">结束列数(从1开始计数)</param>
  924. /// <param name="borderTypeStr">边框样式(默认细线)</param>
  925. /// <param name="borderColorStr">边框颜色(默认黑色)</param>
  926. /// <returns></returns>
  927. public static ISheet MergedRegion(this ISheet sheet, int firstRow, int lastRow, int firstColumn, int lastColumn)
  928. {
  929. var region = GetCellRegion(firstRow, lastRow, firstColumn, lastColumn);
  930. sheet.MergedRegion(region);
  931. return sheet;
  932. }
  933. public static ISheet MergedRegion(this ISheet sheet, CellRangeAddress region)
  934. {
  935. sheet.AddMergedRegion(region);
  936. //SetRegionBorder(sheet, region, borderTypeStr, borderColorStr);
  937. return sheet;
  938. }
  939. public static CellRangeAddress GetCellRegion(this int firstRow, int lastRow, int firstColumn, int lastColumn)
  940. {
  941. firstRow = firstRow <= 0 ? 1 : firstRow;
  942. firstColumn = firstColumn <= 0 ? 1 : firstColumn;
  943. lastRow = lastRow <= 0 ? 1 : lastRow;
  944. lastColumn = lastColumn <= 0 ? 1 : lastColumn;
  945. return new CellRangeAddress(firstRow - 1, lastRow - 1, firstColumn - 1, lastColumn - 1);
  946. }
  947. public static ISheet SetRegionBorder(this CellRangeAddress region, ISheet sheet, string borderTypeStr, string borderColorStr)
  948. {
  949. ICellStyle style = sheet.Workbook.CreateCellStyle();
  950. borderTypeStr = borderTypeStr ?? "Thin";
  951. borderColorStr = borderColorStr ?? "Black";
  952. style.BorderColors(borderColorStr);
  953. style.BorderTypes(borderTypeStr);
  954. for (int i = region.FirstRow; i <= region.LastRow; i++)
  955. {
  956. IRow row = sheet.GenerateRow(i + 1);
  957. if (i == region.FirstRow)
  958. {
  959. }
  960. else if (i == region.LastRow)
  961. {
  962. }
  963. var leftCell = row.GenerateCell(region.FirstColumn + 1);
  964. leftCell.CellStyle.BorderLeft = style.BorderLeft;
  965. leftCell.CellStyle.LeftBorderColor = style.LeftBorderColor;
  966. var rightCell = row.GenerateCell(region.LastColumn + 1);
  967. rightCell.CellStyle.BorderRight = style.BorderRight;
  968. rightCell.CellStyle.RightBorderColor = style.RightBorderColor;
  969. //for (int j = region.FirstColumn; j <= region.LastColumn; j++)
  970. //{
  971. // ICell singleCell = row.GenerateCell((short)j);
  972. //}
  973. }
  974. return sheet;
  975. }
  976. #endregion
  977. public static ICellStyle GetCellStyle(this ICell cell)
  978. {
  979. return cell.CellStyle ?? cell.Sheet.Workbook.CreateCellStyle();
  980. }
  981. public static void SetCellStyle(this ICell cell, ICellStyle cellStyle)
  982. {
  983. cell.CellStyle = cellStyle;
  984. }
  985. public static ICell SetCellStyleAndValue(this ICell cell, ICellStyle cellStyle, string obj)
  986. {
  987. cell.CellStyle = cellStyle;
  988. return SetValue<string>(cell, obj);
  989. }
  990. public static ICell SetCellStyleAndValue<T>(this ICell cell, ICellStyle cellStyle,object obj)
  991. {
  992. cell.CellStyle = cellStyle;
  993. return SetValue<T>(cell, obj);
  994. }
  995. /// <summary>
  996. /// 插入行
  997. /// </summary>
  998. /// <param name="sheet">表</param>
  999. /// <param name="startRow">第几行后开始插入(源行)</param>
  1000. /// <param name="count">插入的行数</param>
  1001. /// <returns></returns>
  1002. public static void InsertRows(this ISheet sheet, int startRow, int count)
  1003. {
  1004. var rowSource = sheet.GenerateRow(startRow);
  1005. sheet.ShiftRows(startRow, sheet.LastRowNum, count, true, false);
  1006. if (rowSource == null)
  1007. return;
  1008. var rowStyle = rowSource.RowStyle;
  1009. for (int i = startRow + 1; i <= startRow + count; i++)
  1010. {
  1011. var rowInsert = sheet.GenerateRow(i);
  1012. rowInsert.Height = rowSource.Height;
  1013. if (rowStyle!=null)
  1014. {
  1015. rowInsert.RowStyle = rowStyle;
  1016. }
  1017. for (int col = 1; col <= rowSource.LastCellNum; col++)
  1018. {
  1019. var cellSource = rowSource.GenerateCell(col);
  1020. var cellInsert = rowInsert.GenerateCell(col);
  1021. var cellStyle = cellSource.CellStyle;
  1022. //设置单元格样式    
  1023. if (cellStyle != null)
  1024. cellInsert.CellStyle = cellSource.CellStyle;
  1025. }
  1026. }
  1027. }
  1028. /// <summary>
  1029. /// 保存工作簿
  1030. /// </summary>
  1031. /// <param name="sheet"></param>
  1032. /// <param name="filePath"></param>
  1033. /// <param name="fileName"></param>
  1034. /// <returns></returns>
  1035. public static string SaveWorkBook(this ISheet sheet, string filePath, string fileName)
  1036. {
  1037. try
  1038. {
  1039. if (!Directory.Exists(filePath))
  1040. {
  1041. Directory.CreateDirectory(filePath);
  1042. }
  1043. filePath = filePath.EndsWith("\\") ? filePath : filePath + "\\";
  1044. FileStream file = new FileStream(filePath + fileName, FileMode.Create);
  1045. sheet.Workbook.Write(file);
  1046. file.Close();
  1047. return "";
  1048. }
  1049. catch (Exception e)
  1050. {
  1051. return e.Message;
  1052. }
  1053. }
  1054. /// <summary>
  1055. /// 保存工作簿
  1056. /// </summary>
  1057. /// <param name="workbook"></param>
  1058. /// <param name="filePath"></param>
  1059. /// <param name="fileName"></param>
  1060. /// <returns></returns>
  1061. public static string SaveWorkBook(this IWorkbook workbook, string filePath, string fileName)
  1062. {
  1063. try
  1064. {
  1065. if (!Directory.Exists(filePath))
  1066. {
  1067. Directory.CreateDirectory(filePath);
  1068. }
  1069. filePath = filePath.EndsWith("\\") ? filePath : filePath + "\\";
  1070. FileStream file = new FileStream(filePath + fileName, FileMode.Create);
  1071. workbook.Write(file);
  1072. file.Close();
  1073. return "";
  1074. }
  1075. catch (Exception e)
  1076. {
  1077. return e.Message;
  1078. }
  1079. }
  1080. }
  1081. public class CellStyleCss
  1082. {
  1083. public static CellStyleCss Instants => new CellStyleCss()
  1084. {
  1085. //缩写
  1086. DefaultStyle = $"bgc:{ColorType.White.ToString()};" +
  1087. $"warp:{HorizontalAlignment.Center.ToString()};" +
  1088. $"align:{HorizontalAlignment.Center.ToString()};" +
  1089. $"v-align:{VerticalAlignment.Center.ToString()};" +
  1090. $"b:{BorderStyle.Dotted.ToString()};" +
  1091. $"bc:{ColorType.Black.ToString()};" +
  1092. "inden:0;" +
  1093. "df:@;"
  1094. };
  1095. public void SetDefaultStyle(string styleStr)
  1096. {
  1097. var dic = new SortedDictionary<string, string>();
  1098. DefaultStyle = dic.GetCleanStyle("");
  1099. DefaultStyle = dic.GetCleanStyle(styleStr);
  1100. }
  1101. /// <summary>
  1102. /// 默认样式css
  1103. /// </summary>
  1104. private string DefaultStyle { get; set; }
  1105. //标准写法
  1106. //private static readonly string DefaultFontStyle = "font-color:black;" +
  1107. // "font-name:Arial;" +
  1108. // "font-size:10;" +
  1109. // "font-weight:normal;" +
  1110. // "font-underline:none;" +
  1111. // "font-italic:false;" +
  1112. // "font-strikeout:false;" +
  1113. // "font-superscript:none;"+
  1114. // "background-color:white;"+
  1115. // "text-align:none;"+
  1116. // "vertical-align:none;"+
  1117. // "data-format:none;border-type:Thin";
  1118. /// <summary>
  1119. /// 把css样式设置给单元格
  1120. /// </summary>
  1121. /// <param name="cell"></param>
  1122. /// <param name="styleStr"></param>
  1123. /// <returns></returns>
  1124. public ICell Css(ICell cell, string styleStr = null)
  1125. {
  1126. var dic = new SortedDictionary<string, string>();
  1127. var sortedCss = dic.GetCleanStyle(DefaultStyle);
  1128. if (!string.IsNullOrEmpty(styleStr))
  1129. sortedCss = dic.GetCleanStyle(styleStr);
  1130. var cssKey = $"CellStyle_{sortedCss.Md5()}";
  1131. var workbook = cell.Sheet.Workbook;
  1132. ICellStyle cellStyle = workbook.GetCellStyle(cssKey);
  1133. if (cellStyle == null)
  1134. {
  1135. cellStyle = workbook.GetCellStyle(dic, cell);
  1136. workbook.AttachedCellStyle(cssKey, cellStyle);
  1137. }
  1138. cell.CellStyle = cellStyle;
  1139. //cell.CellStyle = workbook.GetCellStyle(dic, cell);
  1140. return cell;//返回单元格方便流水式编程
  1141. }
  1142. public ICellStyle GetCssStyle(IWorkbook workbook, string styleStr)
  1143. {
  1144. var dic = new SortedDictionary<string, string>();
  1145. var sortedCss = dic.GetCleanStyle(DefaultStyle);
  1146. if (!string.IsNullOrEmpty(styleStr))
  1147. sortedCss = dic.GetCleanStyle(styleStr);
  1148. var cssKey = $"CellStyle_{sortedCss.Md5()}";
  1149. var cellStyle = workbook.GetCellStyle(cssKey);
  1150. if (cellStyle == null)
  1151. {
  1152. cellStyle = workbook.GetCellStyle(dic);
  1153. workbook.AttachedCellStyle(cssKey, cellStyle);
  1154. }
  1155. return cellStyle;
  1156. }
  1157. }
  1158. internal static class CellStyleRender
  1159. {
  1160. #region 解析css样式
  1161. /// <summary>
  1162. /// 默认字体样式
  1163. /// </summary>
  1164. private static string DefaultFontStyle { get; } = $"fc:{ColorType.Black.ToString()};" +//font-color
  1165. "fn:宋体;" +//font-name
  1166. "fs:12;" +//font-size
  1167. "fw:normal;" +//font-weight
  1168. "fu:none;" +//font-underline
  1169. "fi:false;" +//font-italic
  1170. "fst:false;" +//font-strikeout
  1171. "fss:none;";//font-superscript
  1172. #region 设置样式
  1173. /// <summary>
  1174. /// 缓存
  1175. /// </summary>
  1176. private static readonly ConditionalWeakTable<IWorkbook, Dictionary<string, ICellStyle>> Table =
  1177. new ConditionalWeakTable<IWorkbook, Dictionary<string, ICellStyle>>();
  1178. /// <summary>
  1179. /// 获取CellStyle
  1180. /// </summary>
  1181. /// <param name="workbook"></param>
  1182. /// <param name="dic"></param>
  1183. /// <param name="cell"></param>
  1184. /// <returns></returns>
  1185. public static ICellStyle GetCellStyle(this IWorkbook workbook, SortedDictionary<string, string> dic, ICell cell = null)
  1186. {
  1187. ICellStyle cellStyle = workbook.CreateCellStyle();
  1188. //if (cell != null)
  1189. //{
  1190. // cellStyle.CloneStyleFrom(cell.CellStyle);
  1191. //}
  1192. var fontStyles = dic.Where(w => w.Key.StartsWith("font-")).ToArray();
  1193. var fontDic = new SortedDictionary<string, string>();
  1194. foreach (var kv in fontStyles)
  1195. {
  1196. fontDic.Add(kv.Key, kv.Value);
  1197. }
  1198. var font = workbook.GetFont(fontDic);
  1199. cellStyle.SetFont(font);//TODO 在基于style.xls基础的样式上增加css时,会造成原字体设置的丢失
  1200. var xdic = dic.Where(w => !w.Key.StartsWith("font-")).ToArray();
  1201. foreach (var kvp in xdic)
  1202. {
  1203. FireCssAccess(cellStyle, workbook, kvp);
  1204. }
  1205. return cellStyle;
  1206. }
  1207. /// <summary>
  1208. /// 从缓存读取CellStyle
  1209. /// </summary>
  1210. /// <param name="workbook"></param>
  1211. /// <param name="propertyName"></param>
  1212. /// <returns></returns>
  1213. public static ICellStyle GetCellStyle(this IWorkbook workbook, string propertyName)
  1214. {
  1215. if (!Table.TryGetValue(workbook, out var values)) return null;
  1216. if (values.TryGetValue(propertyName, out var temp))
  1217. return temp;
  1218. return null;
  1219. }
  1220. /// <summary>
  1221. /// 缓存CellStyle
  1222. /// </summary>
  1223. /// <param name="workbook"></param>
  1224. /// <param name="propertyName"></param>
  1225. /// <param name="value"></param>
  1226. public static void AttachedCellStyle(this IWorkbook workbook, string propertyName, ICellStyle value)
  1227. {
  1228. if (!Table.TryGetValue(workbook, out var values))
  1229. {
  1230. values = new Dictionary<string, ICellStyle>();
  1231. Table.Add(workbook, values);
  1232. }
  1233. values[propertyName] = value;
  1234. }
  1235. /// <summary>
  1236. /// Md5 key
  1237. /// </summary>
  1238. /// <param name="input"></param>
  1239. /// <returns></returns>
  1240. public static string Md5(this string input)
  1241. {
  1242. if (input == null)
  1243. input = string.Empty;
  1244. byte[] data = Encoding.UTF8.GetBytes(input.Trim().ToLowerInvariant());
  1245. using (var md5 = new MD5CryptoServiceProvider())
  1246. {
  1247. data = md5.ComputeHash(data);
  1248. }
  1249. var ret = new StringBuilder();
  1250. foreach (byte b in data)
  1251. {
  1252. ret.Append(b.ToString("x2").ToLowerInvariant());
  1253. }
  1254. return ret.ToString();
  1255. }
  1256. /// <summary>
  1257. /// 设置不是字体的样式
  1258. /// </summary>
  1259. /// <param name="style"></param>
  1260. /// <param name="workbook"></param>
  1261. /// <param name="kvp"></param>
  1262. private static void FireCssAccess(ICellStyle style, IWorkbook workbook, KeyValuePair<string, string> kvp)
  1263. {
  1264. switch (kvp.Key)
  1265. {
  1266. case "WrapText":
  1267. style.TextWrap(kvp.Value);
  1268. break;
  1269. case "Indention":
  1270. style.TextIndention(kvp.Value);
  1271. break;
  1272. case "text-align":
  1273. style.TextAlign(kvp.Value);
  1274. break;
  1275. case "vertical-align":
  1276. style.VerticalAlign(kvp.Value);
  1277. break;
  1278. case "background-color":
  1279. style.BackgroundColor(kvp.Value);
  1280. break;
  1281. case "border-type":
  1282. style.BorderTypes(kvp.Value);
  1283. break;
  1284. case "top-border-type":
  1285. style.BorderTopTypes(kvp.Value);
  1286. break;
  1287. case "right-border-type":
  1288. style.BorderRightTypes(kvp.Value);
  1289. break;
  1290. case "bottom-border-type":
  1291. style.BorderBottomTypes(kvp.Value);
  1292. break;
  1293. case "left-border-type":
  1294. style.BorderLeftTypes(kvp.Value);
  1295. break;
  1296. case "border-color":
  1297. style.BorderColors(kvp.Value);
  1298. break;
  1299. case "top-border-color":
  1300. style.BorderTopColors(kvp.Value);
  1301. break;
  1302. case "right-border-color":
  1303. style.BorderRightColors(kvp.Value);
  1304. break;
  1305. case "bottom-border-color":
  1306. style.BorderBottomColors(kvp.Value);
  1307. break;
  1308. case "left-border-color":
  1309. style.BorderLeftColors(kvp.Value);
  1310. break;
  1311. case "data-format":
  1312. style.DataFormat(workbook, kvp.Value);
  1313. break;
  1314. }
  1315. }
  1316. /// <summary>
  1317. /// 获取字体样式
  1318. /// </summary>
  1319. /// <param name="workbook"></param>
  1320. /// <param name="fontdic"></param>
  1321. /// <returns></returns>
  1322. private static IFont GetFont(this IWorkbook workbook, SortedDictionary<string, string> fontdic)
  1323. {
  1324. var weight = fontdic.FontWeight();
  1325. var color = fontdic.FontColor();
  1326. var size = fontdic.FontSize();
  1327. var name = fontdic.FontName();
  1328. var underline = fontdic.FontUnderline();
  1329. var italic = fontdic.FontItalic();
  1330. var strikeout = fontdic.FontStrikeout();
  1331. var offset = fontdic.ConvertToSuperScript();
  1332. var findHeight = (short)(size * 20);
  1333. var font = workbook.FindFont(weight, color, findHeight, name, italic, strikeout, offset, underline);
  1334. if (font == null)
  1335. {
  1336. font = workbook.CreateFont();
  1337. font.Boldweight = weight;
  1338. font.Color = color;
  1339. font.FontHeightInPoints = size;
  1340. font.FontName = name;
  1341. font.Underline = underline;
  1342. font.IsItalic = italic;
  1343. font.IsStrikeout = strikeout;
  1344. font.TypeOffset = offset;
  1345. }
  1346. return font;
  1347. }
  1348. #endregion
  1349. #region 获取样式
  1350. /// <summary>
  1351. /// 默认设置
  1352. /// </summary>
  1353. /// <param name="dic"></param>
  1354. public static void InitStyleDic(this SortedDictionary<string, string> dic)
  1355. {
  1356. var cssItems = GetCssItems(DefaultFontStyle);
  1357. foreach (var cssitem in cssItems)
  1358. {
  1359. var kvp = GetCssKeyValue(cssitem);
  1360. if (dic.ContainsKey(kvp.Key))
  1361. dic[kvp.Key] = kvp.Value; //覆盖相同key的值
  1362. else
  1363. dic.Add(kvp.Key, kvp.Value);
  1364. }
  1365. }
  1366. /// <summary>
  1367. /// 获取样式简洁字符串
  1368. /// </summary>
  1369. /// <param name="dic"></param>
  1370. /// <param name="style"></param>
  1371. /// <returns></returns>
  1372. public static string GetCleanStyle(this SortedDictionary<string, string> dic, string style)
  1373. {
  1374. style = Regex.Replace(style.Trim(), "\\s+", " ");
  1375. style = Regex.Replace(style, "\\s;\\s", ";");
  1376. style = Regex.Replace(style, "\\s:\\s", ":");
  1377. InitStyleDic(dic);
  1378. var cssItems = GetCssItems(style.TrimEnd(';'));
  1379. foreach (var cssitem in cssItems)
  1380. {
  1381. var kvp = GetCssKeyValue(cssitem);
  1382. if (dic.ContainsKey(kvp.Key))
  1383. dic[kvp.Key] = kvp.Value; //覆盖相同key的值
  1384. else
  1385. dic.Add(kvp.Key, kvp.Value);
  1386. }
  1387. var sortedCss = string.Join(";", dic.Select(s => $"{s.Key}:{s.Value}").ToArray());
  1388. return sortedCss;
  1389. }
  1390. /// <summary>
  1391. /// 获取样式数组
  1392. /// </summary>
  1393. /// <param name="style"></param>
  1394. /// <returns></returns>
  1395. private static string[] GetCssItems(string style)
  1396. {
  1397. var cssItems = Regex.Split(style, ";");
  1398. cssItems = cssItems.Where(w => !string.IsNullOrWhiteSpace(w)).ToArray();
  1399. return cssItems;
  1400. }
  1401. /// <summary>
  1402. /// 获取css样式
  1403. /// </summary>
  1404. /// <param name="css"></param>
  1405. /// <returns></returns>
  1406. private static KeyValuePair<string, string> GetCssKeyValue(string css)
  1407. {
  1408. var cssKeyValueArray = Regex.Split(css, ":").ToArray();
  1409. var cssKey = cssKeyValueArray[0].StandardCssKey();
  1410. var cssValue = cssKey == "font-name" ? cssKeyValueArray[1] : cssKeyValueArray[1].ToUpper(); //字体不应变大写
  1411. var kv = new KeyValuePair<string, string>(cssKey, cssValue);
  1412. return kv;
  1413. }
  1414. #endregion
  1415. #region 转换Css的 Key
  1416. /// <summary>
  1417. /// 缩写Key 转换成标准Key
  1418. /// </summary>
  1419. /// <param name="csskey"></param>
  1420. /// <returns></returns>
  1421. private static string StandardCssKey(this string csskey)
  1422. {
  1423. if (CssKeyDic.ContainsKey(csskey))
  1424. {
  1425. var sKey = CssKeyDic[csskey];
  1426. return sKey;
  1427. }
  1428. return csskey;
  1429. }
  1430. /// <summary>
  1431. /// key 转换字典
  1432. /// </summary>
  1433. private static Dictionary<string, string> CssKeyDic => new Dictionary<string, string>
  1434. {
  1435. {"color", "font-color"},
  1436. {"fc", "font-color"},
  1437. {"fw", "font-weight"},
  1438. {"fn", "font-name"},
  1439. {"fs", "font-size"},
  1440. {"italic", "font-italic"},
  1441. {"fi", "font-italic"},
  1442. {"underline", "font-underline"},
  1443. {"fu", "font-underline"},
  1444. {"u", "font-underline"},
  1445. {"deleteline", "font-strikeout"},
  1446. {"d-line", "font-strikeout"},
  1447. {"strikeout", "font-strikeout"},
  1448. {"fst", "font-strikeout"},
  1449. {"d", "font-strikeout"},
  1450. {"font-offset", "font-superscript"},
  1451. {"superscript", "font-superscript"},
  1452. {"fss", "font-superscript"},
  1453. {"ss", "font-superscript"},
  1454. {"bg-color", "background-color"},
  1455. {"bg-c", "background-color"},
  1456. {"bgc", "background-color"},
  1457. {"align", "text-align"},
  1458. {"wrap", "WrapText"},
  1459. {"inden", "Indention"},
  1460. {"in", "Indention"},
  1461. {"v-align", "vertical-align"},
  1462. {"b-t", "border-type"},
  1463. {"b", "border-type"},
  1464. {"bt", "top-border-type"},
  1465. {"br", "right-border-type"},
  1466. {"bb", "bottom-border-type"},
  1467. {"bl", "left-border-type"},
  1468. {"b-c", "border-color"},
  1469. {"bc", "border-color"},
  1470. {"btc", "top-border-color"},
  1471. {"brc", "right-border-color"},
  1472. {"bbc", "bottom-border-color"},
  1473. {"blc", "left-border-color"},
  1474. {"format", "data-format"},
  1475. {"df", "data-format"}
  1476. };
  1477. #endregion
  1478. #region 样式转换
  1479. #region font-weight
  1480. private static short FontWeight(this SortedDictionary<string, string> fontdic)
  1481. {
  1482. switch (fontdic["font-weight"])
  1483. {
  1484. case "NORMAL":
  1485. return 400;
  1486. case "BOLD":
  1487. return 700;
  1488. default:
  1489. return 0;
  1490. }
  1491. }
  1492. #endregion font-weight
  1493. #region font-name
  1494. private static string FontName(this SortedDictionary<string, string> fontdic)
  1495. {
  1496. return fontdic["font-name"];
  1497. }
  1498. #endregion font-name
  1499. #region font-size
  1500. private static short FontSize(this SortedDictionary<string, string> fontdic)
  1501. {
  1502. return short.TryParse(fontdic["font-size"], out var value) ? value : (short)10;
  1503. }
  1504. #endregion font-size
  1505. #region font-color
  1506. private static short FontColor(this SortedDictionary<string, string> fontdic)
  1507. {
  1508. var color = fontdic["font-color"].ConvertToColor();
  1509. return color;
  1510. }
  1511. #endregion font-color
  1512. #region font-italic
  1513. private static bool FontItalic(this SortedDictionary<string, string> fontdic)
  1514. {
  1515. return fontdic["font-italic"] == "TRUE";
  1516. }
  1517. #endregion font-italic
  1518. #region font-strikeout
  1519. /// <summary>
  1520. /// 删除线
  1521. /// </summary>
  1522. /// <param name="fontdic"></param>
  1523. /// <returns></returns>
  1524. private static bool FontStrikeout(this SortedDictionary<string, string> fontdic)
  1525. {
  1526. return fontdic["font-strikeout"] == "TRUE";
  1527. }
  1528. #endregion font-strikeout
  1529. #region WrapText
  1530. private static void TextWrap(this ICellStyle style, string v)
  1531. {
  1532. style.WrapText = v.ToUpper() == "TRUE";
  1533. }
  1534. #endregion WrapText
  1535. #region TextIndention
  1536. /// <summary>
  1537. /// 缩进
  1538. /// </summary>
  1539. /// <param name="style"></param>
  1540. /// <param name="v"></param>
  1541. private static void TextIndention(this ICellStyle style, string v)
  1542. {
  1543. if (short.TryParse(v, out var value))
  1544. {
  1545. style.Indention = value;
  1546. }
  1547. }
  1548. #endregion WrapText
  1549. #region text-align
  1550. private static void TextAlign(this ICellStyle style, string v)
  1551. {
  1552. style.Alignment = v.ConvertToHorizontalAlignment();
  1553. }
  1554. #endregion text-align
  1555. #region vertical-align
  1556. private static void VerticalAlign(this ICellStyle style, string v)
  1557. {
  1558. style.VerticalAlignment = v.ConvertToVerticalAlignment();
  1559. }
  1560. #endregion vertical-align
  1561. #region boder-type / boder-color
  1562. internal static void BorderTypes(this ICellStyle style, string v)
  1563. {
  1564. if (string.IsNullOrEmpty(v))
  1565. return;
  1566. string[] borderTypeNames = { string.Empty, string.Empty, string.Empty, string.Empty };
  1567. v = v.ToUpper();
  1568. var vs = v.Split(' ');
  1569. switch (vs.Length)
  1570. {
  1571. case 1:
  1572. borderTypeNames[0] = borderTypeNames[1] = borderTypeNames[2] = borderTypeNames[3] = vs[0];
  1573. break;
  1574. case 2:
  1575. borderTypeNames[0] = borderTypeNames[2] = vs[0];
  1576. borderTypeNames[1] = borderTypeNames[3] = vs[1];
  1577. break;
  1578. case 3:
  1579. borderTypeNames[0] = vs[0];
  1580. borderTypeNames[1] = borderTypeNames[3] = vs[1];
  1581. borderTypeNames[2] = vs[2];
  1582. break;
  1583. case 4:
  1584. borderTypeNames[0] = vs[0];
  1585. borderTypeNames[1] = vs[1];
  1586. borderTypeNames[2] = vs[2];
  1587. borderTypeNames[3] = vs[3];
  1588. break;
  1589. }
  1590. var borderTopTypeName = borderTypeNames[0];
  1591. var borderRightTypeName = borderTypeNames[1];
  1592. var borderBottomTypeName = borderTypeNames[2];
  1593. var borderLeftTypeName = borderTypeNames[3];
  1594. if (!string.IsNullOrWhiteSpace(borderTopTypeName))
  1595. style.BorderTop = borderTopTypeName.ConvertToBorderStyle();
  1596. if (!string.IsNullOrWhiteSpace(borderRightTypeName))
  1597. style.BorderRight = borderRightTypeName.ConvertToBorderStyle();
  1598. if (!string.IsNullOrWhiteSpace(borderBottomTypeName))
  1599. style.BorderBottom = borderBottomTypeName.ConvertToBorderStyle();
  1600. if (!string.IsNullOrWhiteSpace(borderLeftTypeName))
  1601. style.BorderLeft = borderLeftTypeName.ConvertToBorderStyle();
  1602. }
  1603. private static void BorderTopTypes(this ICellStyle style, string v)
  1604. {
  1605. if (!string.IsNullOrWhiteSpace(v))
  1606. style.BorderTop = v.ConvertToBorderStyle();
  1607. }
  1608. private static void BorderBottomTypes(this ICellStyle style, string v)
  1609. {
  1610. if (!string.IsNullOrWhiteSpace(v))
  1611. style.BorderBottom = v.ConvertToBorderStyle();
  1612. }
  1613. private static void BorderLeftTypes(this ICellStyle style, string v)
  1614. {
  1615. if (!string.IsNullOrWhiteSpace(v))
  1616. style.BorderLeft = v.ConvertToBorderStyle();
  1617. }
  1618. private static void BorderRightTypes(this ICellStyle style, string v)
  1619. {
  1620. if (!string.IsNullOrWhiteSpace(v))
  1621. style.BorderRight = v.ConvertToBorderStyle();
  1622. }
  1623. internal static void BorderColors(this ICellStyle style, string v)
  1624. {
  1625. if (string.IsNullOrEmpty(v))
  1626. return;
  1627. string[] borderColors = { string.Empty, string.Empty, string.Empty, string.Empty };
  1628. v = v.ToUpper();
  1629. var vs = v.Split(' ');
  1630. switch (vs.Length)
  1631. {
  1632. case 1:
  1633. borderColors[0] = borderColors[1] = borderColors[2] = borderColors[3] = vs[0];
  1634. break;
  1635. case 2:
  1636. borderColors[0] = borderColors[2] = vs[0];
  1637. borderColors[1] = borderColors[3] = vs[1];
  1638. break;
  1639. case 3:
  1640. borderColors[0] = vs[0];
  1641. borderColors[1] = borderColors[3] = vs[1];
  1642. borderColors[2] = vs[2];
  1643. break;
  1644. case 4:
  1645. borderColors[0] = vs[0];
  1646. borderColors[1] = vs[1];
  1647. borderColors[2] = vs[2];
  1648. borderColors[3] = vs[3];
  1649. break;
  1650. }
  1651. var borderTopColor = borderColors[0];
  1652. var borderRightColor = borderColors[1];
  1653. var borderBottomColor = borderColors[2];
  1654. var borderLeftColor = borderColors[3];
  1655. if (!string.IsNullOrWhiteSpace(borderTopColor))
  1656. style.TopBorderColor = borderTopColor.ConvertToColor();
  1657. if (!string.IsNullOrWhiteSpace(borderRightColor))
  1658. style.RightBorderColor = borderRightColor.ConvertToColor();
  1659. if (!string.IsNullOrWhiteSpace(borderBottomColor))
  1660. style.BottomBorderColor = borderBottomColor.ConvertToColor();
  1661. if (!string.IsNullOrWhiteSpace(borderLeftColor))
  1662. style.LeftBorderColor = borderLeftColor.ConvertToColor();
  1663. }
  1664. private static void BorderTopColors(this ICellStyle style, string v)
  1665. {
  1666. if (!string.IsNullOrWhiteSpace(v))
  1667. style.TopBorderColor = v.ConvertToColor();
  1668. }
  1669. private static void BorderBottomColors(this ICellStyle style, string v)
  1670. {
  1671. if (!string.IsNullOrWhiteSpace(v))
  1672. style.BottomBorderColor = v.ConvertToColor();
  1673. }
  1674. private static void BorderLeftColors(this ICellStyle style, string v)
  1675. {
  1676. if (!string.IsNullOrWhiteSpace(v))
  1677. style.LeftBorderColor = v.ConvertToColor();
  1678. }
  1679. private static void BorderRightColors(this ICellStyle style, string v)
  1680. {
  1681. if (!string.IsNullOrWhiteSpace(v))
  1682. style.RightBorderColor = v.ConvertToColor();
  1683. }
  1684. #endregion boder-type
  1685. #region data-format
  1686. private static void DataFormat(this ICellStyle style, IWorkbook workbook, string v)
  1687. {
  1688. if (string.IsNullOrEmpty(v))
  1689. return;
  1690. var df = workbook.CreateDataFormat();
  1691. style.DataFormat = df.GetFormat(v);
  1692. }
  1693. #endregion data-format
  1694. #region BackgroundColor
  1695. private static void BackgroundColor(this ICellStyle style, string v)
  1696. {
  1697. if (string.IsNullOrEmpty(v))
  1698. return;
  1699. style.FillPattern = FillPattern.SolidForeground;
  1700. style.FillForegroundColor = v.ConvertToColor();
  1701. }
  1702. #endregion
  1703. private static FontSuperScript ConvertToSuperScript(this SortedDictionary<string, string> fontdic)
  1704. {
  1705. var v = fontdic["font-superscript"];
  1706. switch (v)
  1707. {
  1708. case "SUPER":
  1709. return FontSuperScript.Super;
  1710. case "SUB":
  1711. return FontSuperScript.Sub;
  1712. default:
  1713. return FontSuperScript.None;
  1714. }
  1715. }
  1716. private static FontUnderlineType FontUnderline(this SortedDictionary<string, string> fontdic)
  1717. {
  1718. var v = fontdic["font-underline"];
  1719. switch (v)
  1720. {
  1721. case "SINGLE":
  1722. return FontUnderlineType.Single;
  1723. case "DOUBLE":
  1724. return FontUnderlineType.Double;
  1725. case "SINGLEACCOUNTING":
  1726. case "SINGLE_ACCOUNTING":
  1727. return FontUnderlineType.SingleAccounting;
  1728. case "DOUBLEACCOUNTING":
  1729. case "DOUBLE_ACCOUNTING":
  1730. return FontUnderlineType.DoubleAccounting;
  1731. default:
  1732. return FontUnderlineType.None;
  1733. }
  1734. }
  1735. public static short ConvertToColor(this string v)
  1736. {
  1737. if (string.IsNullOrEmpty(v))
  1738. return 32767;
  1739. switch (v.ToUpper())
  1740. {
  1741. case "AQUA":
  1742. return (short)ColorType.Aqua;
  1743. case "AUTOMATIC":
  1744. return (short)ColorType.Automatic;
  1745. case "BLACK":
  1746. return (short)ColorType.Black;
  1747. case "BLUE":
  1748. return (short)ColorType.Blue;
  1749. case "BLUE_GREY":
  1750. case "BLUEGREY":
  1751. return (short)ColorType.BlueGrey;
  1752. case "BRIGHT_GREEN":
  1753. case "BRIGHTGREEN":
  1754. return (short)ColorType.BrightGreen;
  1755. case "BROWN":
  1756. return (short)ColorType.Brown;
  1757. case "CORAL":
  1758. return (short)ColorType.Coral;
  1759. case "CORNFLOWER_BLUE":
  1760. case "CORNFLOWERBLUE":
  1761. return (short)ColorType.CornflowerBlue;
  1762. case "DARK_BLUE":
  1763. case "DARKBLUE":
  1764. return (short)ColorType.DarkBlue;
  1765. case "DARK_GREEN":
  1766. case "DARKGREEN":
  1767. return (short)ColorType.DarkGreen;
  1768. case "DARK_RED":
  1769. case "DARKRED":
  1770. return (short)ColorType.DarkRed;
  1771. case "DARK_TEAL":
  1772. case "DARKTEAL":
  1773. return (short)ColorType.DarkTeal;
  1774. case "DARK_YELLOW":
  1775. case "DARKYELLOW":
  1776. return (short)ColorType.DarkYellow;
  1777. case "GOLD":
  1778. return (short)ColorType.Gold;
  1779. case "GREEN":
  1780. return (short)ColorType.Green;
  1781. case "GREY_25_PERCENT":
  1782. case "GREY25PERCENT":
  1783. return (short)ColorType.Grey25Percent;
  1784. case "GREY_40_PERCENT":
  1785. case "GREY40PERCENT":
  1786. return (short)ColorType.Grey40Percent;
  1787. case "GREY_50_PERCENT":
  1788. case "GREY50PERCENT":
  1789. return (short)ColorType.Grey50Percent;
  1790. case "GREY_80_PERCENT":
  1791. case "GREY80PERCENT":
  1792. return (short)ColorType.Grey80Percent;
  1793. case "INDIGO":
  1794. return (short)ColorType.Indigo;
  1795. case "LAVENDER":
  1796. return (short)ColorType.Lavender;
  1797. case "LEMON_CHIFFON":
  1798. case "LEMONCHIFFON":
  1799. return (short)ColorType.LemonChiffon;
  1800. case "LIGHT_BLUE":
  1801. case "LIGHTBLUE":
  1802. return (short)ColorType.LightBlue;
  1803. case "LIGHT_CORNFLOWERBLUE":
  1804. case "LIGHTCORNFLOWERBLUE":
  1805. return (short)ColorType.LightCornflowerBlue;
  1806. case "LIGHT_GREEN":
  1807. case "LIGHTGREEN":
  1808. return (short)ColorType.LightGreen;
  1809. case "LIGHT_ORANGE":
  1810. case "LIGHTORANGE":
  1811. return (short)ColorType.LightOrange;
  1812. case "LIGHT_TURQUOISE":
  1813. case "LIGHTTURQUOISE":
  1814. return (short)ColorType.LightTurquoise;
  1815. case "LIGHT_YELLOW":
  1816. case "LIGHTYELLOW":
  1817. return (short)ColorType.LightYellow;
  1818. case "LIME":
  1819. return (short)ColorType.Lime;
  1820. case "MAROON":
  1821. return (short)ColorType.Maroon;
  1822. case "OLIVE_GREEN":
  1823. case "OLIVEGREEN":
  1824. return (short)ColorType.OliveGreen;
  1825. case "ORANGE":
  1826. return (short)ColorType.Orange;
  1827. case "ORCHID":
  1828. return (short)ColorType.Orchid;
  1829. case "PALE_BLUE":
  1830. case "PALEBLUE":
  1831. return (short)ColorType.PaleBlue;
  1832. case "PINK":
  1833. return (short)ColorType.Pink;
  1834. case "PLUM":
  1835. return (short)ColorType.Plum;
  1836. case "RED":
  1837. return (short)ColorType.Red;
  1838. case "ROSE":
  1839. return (short)ColorType.Rose;
  1840. case "ROYAL_BLUE":
  1841. case "ROYALBLUE":
  1842. return (short)ColorType.RoyalBlue;
  1843. case "SEA_GREEN":
  1844. case "SEAGREEN":
  1845. return (short)ColorType.SeaGreen;
  1846. case "SKY_BLUE":
  1847. case "SKYBLUE":
  1848. return (short)ColorType.SkyBlue;
  1849. case "TAN":
  1850. return (short)ColorType.Tan;
  1851. case "TEAL":
  1852. return (short)ColorType.Teal;
  1853. case "TURQUOISE":
  1854. return (short)ColorType.Turquoise;
  1855. case "VIOLET":
  1856. return (short)ColorType.Violet;
  1857. case "WHITE":
  1858. return (short)ColorType.White;
  1859. case "YELLOW":
  1860. return (short)ColorType.Yellow;
  1861. default:
  1862. return 32767;
  1863. }
  1864. }
  1865. private static HorizontalAlignment ConvertToHorizontalAlignment(this string v)
  1866. {
  1867. if (string.IsNullOrEmpty(v))
  1868. return HorizontalAlignment.General;
  1869. switch (v.ToUpper())
  1870. {
  1871. case "LEFT":
  1872. return HorizontalAlignment.Left;
  1873. case "CENTER":
  1874. return HorizontalAlignment.Center;
  1875. case "CENTERSELECTION":
  1876. case "CENTER_SELECTION":
  1877. return HorizontalAlignment.CenterSelection;
  1878. case "RIGHT":
  1879. return HorizontalAlignment.Right;
  1880. case "DISTRIBUTED":
  1881. return HorizontalAlignment.Distributed;
  1882. case "FILL":
  1883. return HorizontalAlignment.Fill;
  1884. case "JUSTIFY":
  1885. return HorizontalAlignment.Justify;
  1886. default:
  1887. return HorizontalAlignment.General;
  1888. }
  1889. }
  1890. private static VerticalAlignment ConvertToVerticalAlignment(this string v)
  1891. {
  1892. if (string.IsNullOrEmpty(v))
  1893. return VerticalAlignment.Justify;
  1894. switch (v.ToUpper())
  1895. {
  1896. case "TOP":
  1897. return VerticalAlignment.Top;
  1898. case "CENTER":
  1899. return VerticalAlignment.Center;
  1900. case "BOTTOM":
  1901. return VerticalAlignment.Bottom;
  1902. case "DISTRIBUTED":
  1903. return VerticalAlignment.Distributed;
  1904. default:
  1905. return VerticalAlignment.Justify;
  1906. }
  1907. }
  1908. public static BorderStyle ConvertToBorderStyle(this string v)
  1909. {
  1910. if (string.IsNullOrEmpty(v))
  1911. return BorderStyle.None;
  1912. switch (v.ToUpper())
  1913. {
  1914. case "THIN":
  1915. return BorderStyle.Thin;
  1916. case "MEDIUM":
  1917. return BorderStyle.Medium;
  1918. case "DASHED":
  1919. return BorderStyle.Dashed;
  1920. case "HAIR":
  1921. return BorderStyle.Hair;
  1922. case "THICK":
  1923. return BorderStyle.Thick;
  1924. case "DOUBLE":
  1925. return BorderStyle.Double;
  1926. case "DOTTED":
  1927. return BorderStyle.Dotted;
  1928. case "MEDIUMDASHED":
  1929. case "MEDIUM_DASHED":
  1930. return BorderStyle.MediumDashed;
  1931. case "DASHDOT":
  1932. case "DASH_DOT":
  1933. return BorderStyle.DashDot;
  1934. case "MEDIUMDASHDOT":
  1935. case "MEDIUM_DASH_DOT":
  1936. return BorderStyle.MediumDashDot;
  1937. case "DASHDOTDOT":
  1938. case "DASH_DOT_DOT":
  1939. return BorderStyle.DashDotDot;
  1940. case "MEDIUMDASHDOTDOT":
  1941. case "MEDIUM_DASH_DOT_DOT":
  1942. return BorderStyle.MediumDashDotDot;
  1943. case "SLANTEDDASHDOT":
  1944. case "SLANTED_DASH_DOT":
  1945. return BorderStyle.SlantedDashDot;
  1946. default:
  1947. return BorderStyle.None;
  1948. }
  1949. }
  1950. #endregion
  1951. #endregion
  1952. }
  1953. public enum ColorType
  1954. {
  1955. Black = HSSFColor.Black.Index,
  1956. Brown = HSSFColor.Brown.Index,
  1957. OliveGreen = HSSFColor.OliveGreen.Index,
  1958. DarkGreen = HSSFColor.DarkGreen.Index,
  1959. DarkTeal = HSSFColor.DarkTeal.Index,
  1960. DarkBlue = HSSFColor.DarkBlue.Index,
  1961. Indigo = HSSFColor.Indigo.Index,
  1962. Grey80Percent = HSSFColor.Grey80Percent.Index,
  1963. Orange = HSSFColor.Orange.Index,
  1964. DarkYellow = HSSFColor.DarkYellow.Index,
  1965. Green = HSSFColor.Green.Index,
  1966. Teal = HSSFColor.Teal.Index,
  1967. Blue = HSSFColor.Blue.Index,
  1968. BlueGrey = HSSFColor.BlueGrey.Index,
  1969. Grey50Percent = HSSFColor.Grey50Percent.Index,
  1970. Red = HSSFColor.Red.Index,
  1971. LightOrange = HSSFColor.LightOrange.Index,
  1972. Lime = HSSFColor.Lime.Index,
  1973. SeaGreen = HSSFColor.SeaGreen.Index,
  1974. Aqua = HSSFColor.Aqua.Index,
  1975. LightBlue = HSSFColor.LightBlue.Index,
  1976. Violet = HSSFColor.Violet.Index,
  1977. Grey40Percent = HSSFColor.Grey40Percent.Index,
  1978. Pink = HSSFColor.Pink.Index,
  1979. Gold = HSSFColor.Gold.Index,
  1980. Yellow = HSSFColor.Yellow.Index,
  1981. BrightGreen = HSSFColor.BrightGreen.Index,
  1982. Turquoise = HSSFColor.Turquoise.Index,
  1983. DarkRed = HSSFColor.DarkRed.Index,
  1984. SkyBlue = HSSFColor.SkyBlue.Index,
  1985. Plum = HSSFColor.Plum.Index,
  1986. Grey25Percent = HSSFColor.Grey25Percent.Index,
  1987. Rose = HSSFColor.Rose.Index,
  1988. LightYellow = HSSFColor.LightYellow.Index,
  1989. LightGreen = HSSFColor.LightGreen.Index,
  1990. LightTurquoise = HSSFColor.LightTurquoise.Index,
  1991. PaleBlue = HSSFColor.PaleBlue.Index,
  1992. Lavender = HSSFColor.Lavender.Index,
  1993. White = HSSFColor.White.Index,
  1994. CornflowerBlue = HSSFColor.CornflowerBlue.Index,
  1995. LemonChiffon = HSSFColor.LemonChiffon.Index,
  1996. Maroon = HSSFColor.Maroon.Index,
  1997. Orchid = HSSFColor.Orchid.Index,
  1998. Coral = HSSFColor.Coral.Index,
  1999. RoyalBlue = HSSFColor.RoyalBlue.Index,
  2000. LightCornflowerBlue = HSSFColor.LightCornflowerBlue.Index,
  2001. Tan = HSSFColor.Tan.Index,
  2002. Automatic = HSSFColor.Automatic.Index
  2003. }
  2004. public static class TransExp<TIn, TOut>
  2005. {
  2006. private static readonly Func<TIn, TOut> Cache = GetFunc();
  2007. private static Func<TIn, TOut> GetFunc()
  2008. {
  2009. ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
  2010. List<MemberBinding> memberBindingList = new List<MemberBinding>();
  2011. foreach (var item in typeof(TOut).GetProperties())
  2012. {
  2013. if (!item.CanWrite)
  2014. continue;
  2015. MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name) ?? throw new InvalidOperationException());
  2016. MemberBinding memberBinding = Expression.Bind(item, property);
  2017. memberBindingList.Add(memberBinding);
  2018. }
  2019. MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
  2020. Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[] { parameterExpression });
  2021. return lambda.Compile();
  2022. }
  2023. public static TOut Trans(TIn tIn)
  2024. {
  2025. return Cache(tIn);
  2026. }
  2027. }
  2028. public class ToExcelObj
  2029. {
  2030. public string ShowColumn { get; set; }
  2031. public string MapColumn { get; set; }
  2032. public string StyleStr { get; set; }
  2033. }
  2034. }