ExcelHelper.cs 118 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Linq.Expressions;
  7. using System.Reflection;
  8. using System.Runtime.CompilerServices;
  9. using System.Security.Cryptography;
  10. using System.Text;
  11. using System.Text.RegularExpressions;
  12. using IwbZero.ToolCommon.LogHelpers;
  13. using NPOI.HSSF.UserModel;
  14. using NPOI.HSSF.Util;
  15. using NPOI.OpenXmlFormats.Spreadsheet;
  16. using NPOI.SS.UserModel;
  17. using NPOI.SS.Util;
  18. using NPOI.XSSF.UserModel;
  19. namespace IwbZero.ToolCommon.FileHelpers
  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="errMsg">错误信息</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 string errMsg, int startIndex = 1) where T : new()
  33. {
  34. List<T> enlist = new List<T>();
  35. var 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. }
  47. catch (Exception ex)
  48. {
  49. errMsg = ex.Message;
  50. typeof(ExcelHelper).LogError(ex);
  51. return default(List<T>);
  52. }
  53. errMsg = errorMsg.ToString();
  54. return enlist;
  55. }
  56. /// <summary>
  57. /// 从Excel2003取数据并记录到List集合里
  58. /// </summary>
  59. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  60. /// <param name="filePath">保存文件绝对路径</param>
  61. /// <param name="errorMsg">错误信息</param>
  62. /// <param name="startIndex"></param>
  63. /// <returns>转换好的List对象集合</returns>
  64. private static List<T> Excel2003ToEntityList<T>(this Dictionary<string, string> cellHeader, string filePath, out StringBuilder errorMsg, int startIndex = 1) where T : new()
  65. {
  66. errorMsg = new StringBuilder(); // 错误信息,Excel转换到实体对象时,会有格式的错误信息
  67. List<T> enlist = new List<T>(); // 转换后的集合
  68. try
  69. {
  70. using (FileStream fs = File.OpenRead(filePath))
  71. {
  72. HSSFWorkbook workbook = new HSSFWorkbook(fs);
  73. HSSFSheet sheet = (HSSFSheet)workbook.GetSheetAt(0); // 获取此文件第一个Sheet页
  74. for (int rowIndex = startIndex; rowIndex <= sheet.LastRowNum; rowIndex++)
  75. {
  76. // 1.判断当前行是否空行,若空行就不在进行读取下一行操作,结束Excel读取操作
  77. IRow row = sheet.GetRow(rowIndex);
  78. if (row == null)
  79. {
  80. break;
  81. }
  82. // 2.每一个Excel row转换为一个实体对象
  83. T en = new T();
  84. ExcelRowToEntity(cellHeader, row, rowIndex, en, ref errorMsg);
  85. enlist.Add(en);
  86. }
  87. }
  88. return enlist;
  89. }
  90. catch (Exception ex)
  91. {
  92. typeof(ExcelHelper).LogError(ex);
  93. return default(List<T>);
  94. }
  95. }
  96. /// <summary>
  97. /// 从Excel2007取数据并记录到List集合里
  98. /// </summary>
  99. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  100. /// <param name="filePath">保存文件绝对路径</param>
  101. /// <param name="errorMsg">错误信息</param>
  102. /// <param name="startIndex">数据行开始序列,默认为1(即第二列,从0开始)</param>
  103. /// <returns>转换好的List对象集合</returns>
  104. private static List<T> Excel2007ToEntityList<T>(this Dictionary<string, string> cellHeader, string filePath, out StringBuilder errorMsg, int startIndex = 1)
  105. where T : new()
  106. {
  107. errorMsg = new StringBuilder(); // 错误信息,Excel转换到实体对象时,会有格式的错误信息
  108. List<T> enlist = new List<T>(); // 转换后的集合
  109. try
  110. {
  111. using (FileStream fs = File.OpenRead(filePath))
  112. {
  113. XSSFWorkbook workbook = new XSSFWorkbook(fs);
  114. XSSFSheet sheet = (XSSFSheet)workbook.GetSheetAt(0); // 获取此文件第一个Sheet页
  115. for (int rowIndex = startIndex; rowIndex <= sheet.LastRowNum; rowIndex++)
  116. {
  117. // 1.判断当前行是否空行,若空行就不在进行读取下一行操作,结束Excel读取操作
  118. IRow row = sheet.GetRow(rowIndex);
  119. if (row == null)
  120. {
  121. break;
  122. }
  123. // 2.每一个Excel row转换为一个实体对象
  124. T en = new T();
  125. ExcelRowToEntity(cellHeader, row, rowIndex, en, ref errorMsg);
  126. enlist.Add(en);
  127. }
  128. }
  129. return enlist;
  130. }
  131. catch (Exception ex)
  132. {
  133. typeof(ExcelHelper).LogError(ex);
  134. return default(List<T>);
  135. }
  136. }
  137. #endregion Excel导入
  138. #region Excel导出
  139. /// <summary>
  140. /// 实体类集合导出到EXCEL2003
  141. /// </summary>
  142. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  143. /// <param name="enList">数据源</param>
  144. /// <param name="sheetName">工作表名称</param>
  145. /// <param name="filePath">文件的下载地址</param>
  146. /// <returns></returns>
  147. public static string EntityListToExcel2003(this Dictionary<string, string> cellHeader, IList enList, string sheetName, string filePath)
  148. {
  149. var lcRetVal = "error@";
  150. try
  151. {
  152. string fileName = $"D-{sheetName}-{DateTime.Now:yyyyMMddHHmmssfff}.xls"; // 文件名称
  153. filePath = filePath.StartsWith("/") ? filePath : ("/" + filePath);
  154. filePath = filePath.EndsWith("/") ? filePath : (filePath + "/");
  155. string path = $"{AppDomain.CurrentDomain.BaseDirectory}{filePath}";
  156. if (!Directory.Exists(path))
  157. Directory.CreateDirectory(path);
  158. // 2.解析单元格头部,设置单元头的中文名称
  159. HSSFWorkbook workbook = new HSSFWorkbook(); // 工作簿
  160. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  161. IRow row = sheet.CreateRow(0);
  162. List<string> keys = cellHeader.Keys.ToList();
  163. for (int i = 0; i < keys.Count; i++)
  164. {
  165. row.CreateCell(i).SetCellValue(cellHeader[keys[i]]); // 列名为Key的值
  166. }
  167. // 3.List对象的值赋值到Excel的单元格里
  168. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  169. object propertyValue = null;
  170. foreach (var en in enList)
  171. {
  172. IRow rowTmp = sheet.CreateRow(rowIndex);
  173. for (int i = 0; i < keys.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  174. {
  175. string cellValue = "";
  176. PropertyInfo propertyInfo = null; // 属性的信息
  177. // 3.1 若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.UserName
  178. string[] propertyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  179. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  180. {
  181. // 3.1.1 解析子类属性(这里只解析1层子类,多层子类未处理)
  182. string subClassName = propertyArray[0]; // '.'前面的为子类的名称
  183. string subClassPropertyName = propertyArray[1]; // '.'后面的为子类的属性名称
  184. PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  185. if (subClassInfo != null)
  186. {
  187. // 3.1.2 获取子类的实例
  188. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  189. // 3.1.3 根据属性名称获取子类里的属性类型
  190. propertyInfo = subClassInfo.PropertyType.GetProperty(subClassPropertyName);
  191. if (propertyInfo != null)
  192. {
  193. propertyValue = propertyInfo.GetValue(subClassEn, null); // 获取子类属性的值
  194. }
  195. }
  196. }
  197. else
  198. {
  199. // 3.2 若不是子类的属性,直接根据属性名称获取对象对应的属性
  200. propertyInfo = en.GetType().GetProperty(keys[i]);
  201. if (propertyInfo != null)
  202. {
  203. propertyValue = propertyInfo.GetValue(en, null);
  204. }
  205. }
  206. // 3.3 属性值经过转换赋值给单元格值
  207. if (propertyValue != null)
  208. {
  209. if (propertyInfo?.PropertyType.Name == "int" || propertyInfo?.PropertyType.Name == "float" || propertyInfo?.PropertyType.Name == "double" || propertyInfo?.PropertyType.Name == "decimal")
  210. {
  211. var value = Convert.ToDouble(propertyValue);
  212. rowTmp.CreateCell(i).SetCellValue(value);
  213. }
  214. else if (propertyInfo?.PropertyType.Name == "bool")
  215. {
  216. var value = Convert.ToBoolean(propertyValue);
  217. rowTmp.CreateCell(i).SetCellValue(value);
  218. }
  219. else
  220. {
  221. cellValue = propertyValue.ToString();
  222. // 3.3.1 对时间初始值赋值为空
  223. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  224. {
  225. cellValue = "";
  226. }
  227. // 3.4 填充到Excel的单元格里
  228. rowTmp.CreateCell(i).SetCellValue(cellValue);
  229. }
  230. }
  231. else
  232. {
  233. // 3.4 填充到Excel的单元格里
  234. rowTmp.CreateCell(i).SetCellValue(cellValue);
  235. }
  236. }
  237. rowIndex++;
  238. }
  239. // 4.生成文件
  240. FileStream file = new FileStream($"{path}{fileName}", FileMode.Create);
  241. workbook.Write(file);
  242. file.Close();
  243. lcRetVal = $"{filePath}{fileName}";
  244. }
  245. catch (Exception ex)
  246. {
  247. lcRetVal += ex.Message;
  248. typeof(ExcelHelper).LogError(ex);
  249. }
  250. return lcRetVal;
  251. }
  252. public static string EntityListToExcel2003(this HSSFWorkbook workbook,
  253. string sheetName, string filePath)
  254. {
  255. var lcRetVal = "error@";
  256. try
  257. {
  258. string fileName = $"D-{sheetName}-{DateTime.Now:yyyyMMddHHmmssfff}.xls"; // 文件名称
  259. filePath = filePath.StartsWith("/") ? filePath : ("/" + filePath);
  260. filePath = filePath.EndsWith("/") ? filePath : (filePath + "/");
  261. string path = $"{AppDomain.CurrentDomain.BaseDirectory}{filePath}";
  262. if (!Directory.Exists(path))
  263. Directory.CreateDirectory(path);
  264. FileStream file = new FileStream($"{path}{fileName}", FileMode.Create);
  265. workbook.Write(file);
  266. file.Close();
  267. lcRetVal = $"{filePath}{fileName}";
  268. }
  269. catch (Exception ex)
  270. {
  271. lcRetVal += ex.Message;
  272. typeof(ExcelHelper).LogError(ex);
  273. }
  274. return lcRetVal;
  275. }
  276. public static HSSFWorkbook EntityListToExcel2003Book(this Dictionary<string, string> cellHeader, IList enList,
  277. string sheetName)
  278. {
  279. HSSFWorkbook workbook = new HSSFWorkbook(); // 工作簿
  280. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  281. IRow row = sheet.CreateRow(0);
  282. List<string> keys = cellHeader.Keys.ToList();
  283. for (int i = 0; i < keys.Count; i++)
  284. {
  285. row.CreateCell(i).SetCellValue(cellHeader[keys[i]]); // 列名为Key的值
  286. }
  287. // 3.List对象的值赋值到Excel的单元格里
  288. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  289. foreach (var en in enList)
  290. {
  291. IRow rowTmp = sheet.CreateRow(rowIndex);
  292. for (int i = 0; i < keys.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  293. {
  294. string cellValue = ""; // 单元格的值
  295. object propertyValue = null; // 属性的值
  296. PropertyInfo propertyInfo; // 属性的信息
  297. // 3.1 若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.UserName
  298. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  299. {
  300. // 3.1.1 解析子类属性(这里只解析1层子类,多层子类未处理)
  301. string[] propertyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  302. string subClassName = propertyArray[0]; // '.'前面的为子类的名称
  303. string subClassPropertyName = propertyArray[1]; // '.'后面的为子类的属性名称
  304. PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  305. if (subClassInfo != null)
  306. {
  307. // 3.1.2 获取子类的实例
  308. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  309. // 3.1.3 根据属性名称获取子类里的属性类型
  310. propertyInfo = subClassInfo.PropertyType.GetProperty(subClassPropertyName);
  311. if (propertyInfo != null)
  312. {
  313. propertyValue = propertyInfo.GetValue(subClassEn, null); // 获取子类属性的值
  314. }
  315. }
  316. }
  317. else
  318. {
  319. // 3.2 若不是子类的属性,直接根据属性名称获取对象对应的属性
  320. propertyInfo = en.GetType().GetProperty(keys[i]);
  321. if (propertyInfo != null)
  322. {
  323. propertyValue = propertyInfo.GetValue(en, null);
  324. }
  325. }
  326. // 3.3 属性值经过转换赋值给单元格值
  327. if (propertyValue != null)
  328. {
  329. cellValue = propertyValue.ToString();
  330. // 3.3.1 对时间初始值赋值为空
  331. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  332. {
  333. cellValue = "";
  334. }
  335. }
  336. // 3.4 填充到Excel的单元格里
  337. rowTmp.CreateCell(i).SetCellValue(cellValue);
  338. }
  339. rowIndex++;
  340. }
  341. return workbook;
  342. }
  343. /// <summary>
  344. /// 实体类集合导出到EXCEL2007
  345. /// </summary>
  346. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  347. /// <param name="enList">数据源</param>
  348. /// <param name="sheetName">工作表名称</param>
  349. /// <param name="filePath">文件的下载地址</param>
  350. /// <returns></returns>
  351. public static string EntityListToExcel2007(this Dictionary<string, string> cellHeader, IList enList, string sheetName, string filePath)
  352. {
  353. var lcRetVal = "error@";
  354. try
  355. {
  356. string fileName = $"D-{sheetName}-{DateTime.Now:yyyyMMddHHmmssfff}.xlsx"; // 文件名称
  357. filePath = filePath.StartsWith("/") ? filePath : ("/" + filePath);
  358. filePath = filePath.EndsWith("/") ? filePath : (filePath + "/");
  359. string path = $"{AppDomain.CurrentDomain.BaseDirectory}{filePath}";
  360. if (!Directory.Exists(path))
  361. Directory.CreateDirectory(path);
  362. // 2.解析单元格头部,设置单元头的中文名称
  363. XSSFWorkbook workbook = new XSSFWorkbook(); // 工作簿
  364. ISheet sheet = workbook.CreateSheet(sheetName); // 工作表
  365. IRow row = sheet.CreateRow(0);
  366. List<string> keys = cellHeader.Keys.ToList();
  367. for (int i = 0; i < keys.Count; i++)
  368. {
  369. row.CreateCell(i).SetCellValue(cellHeader[keys[i]]); // 列名为Key的值
  370. }
  371. // 3.List对象的值赋值到Excel的单元格里
  372. int rowIndex = 1; // 从第二行开始赋值(第一行已设置为单元头)
  373. foreach (var en in enList)
  374. {
  375. IRow rowTmp = sheet.CreateRow(rowIndex);
  376. for (int i = 0; i < keys.Count; i++) // 根据指定的属性名称,获取对象指定属性的值
  377. {
  378. string cellValue = ""; // 单元格的值
  379. object propertyValue = null; // 属性的值
  380. PropertyInfo propertyInfo = null; // 属性的信息
  381. // 3.1 若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.UserName
  382. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  383. {
  384. // 3.1.1 解析子类属性(这里只解析1层子类,多层子类未处理)
  385. string[] propertyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  386. string subClassName = propertyArray[0]; // '.'前面的为子类的名称
  387. var subClassPropertyName = propertyArray[1]; // '.'后面的为子类的属性名称
  388. PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  389. if (subClassInfo != null)
  390. {
  391. // 3.1.2 获取子类的实例
  392. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  393. // 3.1.3 根据属性名称获取子类里的属性类型
  394. propertyInfo = subClassInfo.PropertyType.GetProperty(subClassPropertyName);
  395. if (propertyInfo != null)
  396. {
  397. propertyValue = propertyInfo.GetValue(subClassEn, null); // 获取子类属性的值
  398. }
  399. }
  400. }
  401. else
  402. {
  403. // 3.2 若不是子类的属性,直接根据属性名称获取对象对应的属性
  404. propertyInfo = en.GetType().GetProperty(keys[i]);
  405. if (propertyInfo != null)
  406. {
  407. propertyValue = propertyInfo.GetValue(en, null);
  408. }
  409. }
  410. // 3.3 属性值经过转换赋值给单元格值
  411. if (propertyValue != null)
  412. {
  413. if (propertyInfo.PropertyType.Name.ToLower() == "int" || propertyInfo.PropertyType.Name.ToLower() == "int16" || propertyInfo.PropertyType.Name.ToLower() == "int32" || propertyInfo.PropertyType.Name.ToLower() == "int64" || propertyInfo.PropertyType.Name.ToLower() == "double" || propertyInfo.PropertyType.Name.ToLower() == "decimal")
  414. {
  415. var value = Convert.ToDouble(propertyValue);
  416. rowTmp.CreateCell(i).SetCellValue(value);
  417. }
  418. else if (propertyInfo.PropertyType.Name.ToLower() == "bool")
  419. {
  420. var value = Convert.ToBoolean(propertyValue);
  421. rowTmp.CreateCell(i).SetCellValue(value);
  422. }
  423. else if (propertyInfo.PropertyType.Name.ToLower() == "nullable`1")
  424. {
  425. if (propertyInfo.PropertyType.FullName == null)
  426. {
  427. }
  428. else if (propertyInfo.PropertyType.FullName.ToLower().Contains("datetime"))
  429. {
  430. cellValue = propertyValue.ToString();
  431. // 3.3.1 对时间初始值赋值为空
  432. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  433. {
  434. cellValue = "";
  435. }
  436. rowTmp.CreateCell(i).SetCellValue(cellValue);
  437. }
  438. else if (propertyInfo.PropertyType.FullName.ToLower().Contains("int"))
  439. {
  440. var value = (int)Convert.ChangeType(propertyValue, typeof(int));
  441. rowTmp.CreateCell(i).SetCellValue(value);
  442. }
  443. else if (propertyInfo.PropertyType.FullName.ToLower().Contains("float") || propertyInfo.PropertyType.FullName.ToLower().Contains("double") || propertyInfo.PropertyType.FullName.ToLower().Contains("decimal") || propertyInfo.PropertyType.FullName.ToLower().Contains("single"))
  444. {
  445. var value = (float)Convert.ChangeType(propertyValue, typeof(float));
  446. rowTmp.CreateCell(i).SetCellValue(value);
  447. }
  448. else if (propertyInfo.PropertyType.FullName.ToLower().Contains("bool"))
  449. {
  450. var value = (bool)Convert.ChangeType(propertyValue, typeof(bool));
  451. rowTmp.CreateCell(i).SetCellValue(value);
  452. }
  453. }
  454. else
  455. {
  456. cellValue = propertyValue.ToString();
  457. // 3.3.1 对时间初始值赋值为空
  458. if (cellValue.Trim() == "0001/1/1 0:00:00" || cellValue.Trim() == "0001/1/1 23:59:59")
  459. {
  460. cellValue = "";
  461. }
  462. // 3.4 填充到Excel的单元格里
  463. rowTmp.CreateCell(i).SetCellValue(cellValue);
  464. }
  465. }
  466. else
  467. {
  468. // 3.4 填充到Excel的单元格里
  469. rowTmp.CreateCell(i).SetCellValue(cellValue);
  470. }
  471. }
  472. rowIndex++;
  473. }
  474. // 4.生成文件
  475. FileStream file = new FileStream($"{path}{fileName}", FileMode.Create);
  476. workbook.Write(file);
  477. file.Close();
  478. lcRetVal = $"{filePath}{fileName}";
  479. }
  480. catch (Exception ex)
  481. {
  482. lcRetVal += ex.Message;
  483. typeof(ExcelHelper).LogError(ex);
  484. }
  485. return lcRetVal;
  486. }
  487. #endregion Excel导出
  488. #region Common
  489. /// <summary>
  490. /// Excel row转换为实体对象
  491. /// </summary>
  492. /// <typeparam name="T"></typeparam>
  493. /// <param name="cellHeader">单元头的Key和Value:{ { "UserName", "姓名" }, { "Age", "年龄" } };</param>
  494. /// <param name="row">Excel row</param>
  495. /// <param name="rowIndex">row index</param>
  496. /// <param name="en">实体</param>
  497. /// <param name="errorMsg">错误信息</param>
  498. private static void ExcelRowToEntity<T>(Dictionary<string, string> cellHeader, IRow row, int rowIndex, T en, ref StringBuilder errorMsg)
  499. {
  500. List<string> keys = cellHeader.Keys.ToList(); // 要赋值的实体对象属性名称
  501. string errStr = ""; // 当前行转换时,是否有错误信息,格式为:第1行数据转换异常:XXX列;
  502. for (int i = 0; i < keys.Count; i++)
  503. {
  504. // 1.若属性头的名称包含'.',就表示是子类里的属性,那么就要遍历子类,eg:UserEn.TrueName
  505. if (keys[i].IndexOf(".", StringComparison.Ordinal) >= 0)
  506. {
  507. // 1)解析子类属性
  508. string[] propertyArray = keys[i].Split(new[] { "." }, StringSplitOptions.RemoveEmptyEntries);
  509. string subClassName = propertyArray[0]; // '.'前面的为子类的名称
  510. string classPropertyName = propertyArray[1]; // '.'后面的为子类的属性名称
  511. PropertyInfo subClassInfo = en.GetType().GetProperty(subClassName); // 获取子类的类型
  512. if (subClassInfo != null)
  513. {
  514. // 2)获取子类的实例
  515. var subClassEn = en.GetType().GetProperty(subClassName)?.GetValue(en, null);
  516. // 3)根据属性名称获取子类里的属性信息
  517. PropertyInfo propertyInfo = subClassInfo.PropertyType.GetProperty(classPropertyName);
  518. if (propertyInfo != null)
  519. {
  520. try
  521. {
  522. // Excel单元格的值转换为对象属性的值,若类型不对,记录出错信息
  523. propertyInfo.SetValue(subClassEn, GetExcelCellToProperty(propertyInfo.PropertyType, row.GetCell(i)), null);
  524. }
  525. catch (Exception e)
  526. {
  527. typeof(ExcelHelper).LogError(e);
  528. if (errStr.Length == 0)
  529. {
  530. errStr = "第" + rowIndex + "行数据转换异常:";
  531. }
  532. errStr += cellHeader[keys[i]] + "列;";
  533. }
  534. }
  535. }
  536. }
  537. else
  538. {
  539. // 2.给指定的属性赋值
  540. PropertyInfo propertyInfo = en.GetType().GetProperty(keys[i]);
  541. if (propertyInfo != null)
  542. {
  543. try
  544. {
  545. // Excel单元格的值转换为对象属性的值,若类型不对,记录出错信息
  546. propertyInfo.SetValue(en, GetExcelCellToProperty(propertyInfo.PropertyType, row.GetCell(i)), null);
  547. }
  548. catch (Exception e)
  549. {
  550. typeof(ExcelHelper).LogError(e);
  551. if (errStr.Length == 0)
  552. {
  553. errStr = "第" + rowIndex + "行数据转换异常:";
  554. }
  555. errStr += cellHeader[keys[i]] + "列;";
  556. }
  557. }
  558. }
  559. }
  560. // 若有错误信息,就添加到错误信息里
  561. if (errStr.Length > 0)
  562. {
  563. errorMsg.AppendLine(errStr);
  564. }
  565. }
  566. /// <summary>
  567. /// Excel Cell转换为实体的属性值
  568. /// </summary>
  569. /// <param name="distanceType">目标对象类型</param>
  570. /// <param name="sourceCell">对象属性的值</param>
  571. private static object GetExcelCellToProperty(Type distanceType, ICell sourceCell)
  572. {
  573. object rs = distanceType.IsValueType ? Activator.CreateInstance(distanceType) : null;
  574. // 1.判断传递的单元格是否为空
  575. if (sourceCell == null || string.IsNullOrEmpty(sourceCell.ToString()))
  576. {
  577. return rs;
  578. }
  579. // 2.Excel文本和数字单元格转换,在Excel里文本和数字是不能进行转换,所以这里预先存值
  580. object sourceValue = null;
  581. switch (sourceCell.CellType)
  582. {
  583. case CellType.Blank:
  584. break;
  585. case CellType.Boolean:
  586. sourceValue = sourceCell.BooleanCellValue;
  587. break;
  588. case CellType.Error:
  589. break;
  590. case CellType.Formula:
  591. break;
  592. case CellType.Numeric:
  593. sourceValue = sourceCell.NumericCellValue;
  594. break;
  595. case CellType.String:
  596. sourceValue = sourceCell.StringCellValue;
  597. break;
  598. case CellType.Unknown:
  599. break;
  600. default:
  601. sourceValue = sourceCell.StringCellValue;
  602. break;
  603. }
  604. string valueDataType = distanceType.Name;
  605. // 在这里进行特定类型的处理
  606. switch (valueDataType.ToLower()) // 以防出错,全部小写
  607. {
  608. case "string":
  609. rs = sourceValue?.ToString();
  610. break;
  611. case "int":
  612. case "int32":
  613. rs = (int)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  614. break;
  615. case "int16":
  616. rs = (short)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  617. break;
  618. case "int64":
  619. rs = (long)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  620. break;
  621. case "float":
  622. case "single":
  623. rs = (float)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  624. break;
  625. case "double":
  626. rs = (double)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  627. break;
  628. case "decimal":
  629. rs = (decimal)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  630. break;
  631. case "datetime":
  632. rs = (DateTime)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  633. break;
  634. case "guid":
  635. rs = (Guid)Convert.ChangeType(sourceValue?.ToString() ?? "", distanceType);
  636. break;
  637. case "nullable`1":
  638. if (distanceType.FullName == null)
  639. {
  640. rs = "";
  641. }
  642. else if (distanceType.FullName.ToLower().Contains("datetime"))
  643. {
  644. rs = (DateTime)Convert.ChangeType(sourceValue?.ToString() ?? "", typeof(DateTime));
  645. }
  646. else if ((bool)distanceType.FullName?.ToLower().Contains("int"))
  647. {
  648. rs = (int)Convert.ChangeType(sourceValue?.ToString() ?? "", typeof(int));
  649. }
  650. else if ((bool)distanceType.FullName?.ToLower().Contains("float") || (bool)distanceType.FullName?.ToLower().Contains("double") || (bool)distanceType.FullName?.ToLower().Contains("decimal") || (bool)distanceType.FullName?.ToLower().Contains("single"))
  651. {
  652. rs = (float)Convert.ChangeType(sourceValue?.ToString() ?? "", typeof(float));
  653. }
  654. else if ((bool)distanceType.FullName?.ToLower().Contains("bool"))
  655. {
  656. rs = (bool)Convert.ChangeType(sourceValue?.ToString() ?? "", typeof(bool));
  657. }
  658. break;
  659. }
  660. return rs;
  661. }
  662. #endregion Common
  663. public static IWorkbook CreateWorkBook(string filePath = "")
  664. {
  665. return filePath.ToLower().EndsWith(".xlsx") ? CreateWorkBook07(filePath) :
  666. filePath.ToLower().EndsWith(".xls") ? (IWorkbook)CreateWorkBook03(filePath) : new XSSFWorkbook();
  667. }
  668. public static HSSFWorkbook CreateWorkBook03(string filePath = null)
  669. {
  670. if (string.IsNullOrEmpty(filePath))
  671. {
  672. return new HSSFWorkbook();
  673. }
  674. FileStream file = new FileStream(filePath, FileMode.Open);
  675. return new HSSFWorkbook(file);
  676. }
  677. public static XSSFWorkbook CreateWorkBook07(string filePath = null)
  678. {
  679. if (string.IsNullOrEmpty(filePath))
  680. {
  681. return new XSSFWorkbook();
  682. }
  683. return new XSSFWorkbook(filePath);
  684. }
  685. /// <summary>
  686. /// 创建Sheet
  687. /// </summary>
  688. /// <param name="sheetName"></param>
  689. /// <param name="defaultWidth"></param>
  690. /// <param name="defaultHeight"></param>
  691. /// <param name="is07"></param>
  692. /// <param name="workbook"></param>
  693. /// <returns></returns>
  694. public static ISheet GenerateSheet(this string sheetName, int defaultWidth = 20, int defaultHeight = 20, bool is07 = true, IWorkbook workbook = null)
  695. {
  696. workbook = is07 ? workbook ?? CreateWorkBook07() : workbook ?? CreateWorkBook03();
  697. var sheet = workbook.CreateSheet(sheetName);
  698. sheet.DefaultColumnWidth = defaultWidth;
  699. sheet.DefaultRowHeight = (short)(defaultHeight * 20);
  700. return sheet;
  701. }
  702. /// <summary>
  703. /// 创建行
  704. /// </summary>
  705. /// <param name="sheet">表</param>
  706. /// <param name="rowIndex">第几行(从1开始计数)</param>
  707. /// <returns></returns>
  708. public static IRow GenerateRow(this ISheet sheet, int rowIndex)
  709. {
  710. rowIndex = rowIndex <= 0 ? 1 : rowIndex;
  711. var row = sheet.GetRow(rowIndex - 1) ?? sheet.CreateRow(rowIndex - 1);
  712. return row;
  713. }
  714. /// <summary>
  715. /// 插入行
  716. /// </summary>
  717. /// <param name="sheet">表</param>
  718. /// <param name="startRow">第几行后开始插入(源行)</param>
  719. /// <param name="count">插入的行数</param>
  720. /// <returns></returns>
  721. public static void InsertRows(this ISheet sheet, int startRow, int count)
  722. {
  723. var rowSource = sheet.GenerateRow(startRow);
  724. sheet.ShiftRows(startRow, sheet.LastRowNum, count, true, false);
  725. if (rowSource == null)
  726. return;
  727. var rowStyle = rowSource.RowStyle;
  728. for (int i = startRow + 1; i <= startRow + count; i++)
  729. {
  730. var rowInsert = sheet.GenerateRow(i);
  731. rowInsert.RowStyle = rowStyle;
  732. rowInsert.Height = rowSource.Height;
  733. for (int col = 1; col <= rowSource.LastCellNum; col++)
  734. {
  735. var cellSource = rowSource.GenerateCell(col);
  736. var cellInsert = rowInsert.GenerateCell(col);
  737. var cellStyle = cellSource.CellStyle;
  738. //设置单元格样式    
  739. if (cellStyle != null)
  740. cellInsert.CellStyle = cellSource.CellStyle;
  741. }
  742. }
  743. }
  744. /// <summary>
  745. /// 创建单元格
  746. /// </summary>
  747. /// <param name="sheet">表</param>
  748. /// <param name="rowIndex">第几行(从1开始计数)</param>
  749. /// <param name="columnIndex">第几列(从1开始计数)</param>
  750. /// <param name="val">值</param>
  751. /// <param name="cellType">单元格类型</param>
  752. /// <returns></returns>
  753. public static ICell GenerateCell(this ISheet sheet, int rowIndex, int columnIndex, string val = null, CellType cellType = CellType.String)
  754. {
  755. var row = sheet.GenerateRow(rowIndex);
  756. var cell = GenerateCell(row, columnIndex, val, cellType);
  757. return cell;
  758. }
  759. /// <summary>
  760. /// 创建单元格
  761. /// </summary>
  762. /// <param name="row">行</param>
  763. /// <param name="columnIndex">第几列(从1开始计数)</param>
  764. /// <param name="val">值</param>
  765. /// <param name="cellType">单元格类型</param>
  766. /// <returns></returns>
  767. public static ICell GenerateCell(this IRow row, int columnIndex, string val = null, CellType cellType = CellType.String)
  768. {
  769. columnIndex = columnIndex <= 0 ? 1 : columnIndex;
  770. var cell = row.GetCell(columnIndex - 1) ?? row.CreateCell(columnIndex - 1);
  771. if (!string.IsNullOrEmpty(val))
  772. {
  773. cell.SetCellValue(val);
  774. cell.SetCellType(cellType);
  775. }
  776. return cell;
  777. }
  778. /// <summary>
  779. /// 设置列宽度
  780. /// </summary>
  781. /// <param name="sheet">表</param>
  782. /// <param name="columnIndex">第几列(从1开始计数)</param>
  783. /// <param name="width">宽度</param>
  784. /// <returns></returns>
  785. public static ISheet SetCellWidth(this ISheet sheet, int columnIndex, int width)
  786. {
  787. sheet.SetColumnWidth(columnIndex - 1, width * 256);
  788. return sheet;
  789. }
  790. /// <summary>
  791. /// 设置列宽度
  792. /// </summary>
  793. /// <param name="row">行</param>
  794. /// <param name="columnIndex">第几列(从1开始计数)</param>
  795. /// <param name="width">宽度</param>
  796. /// <returns></returns>
  797. public static IRow SetCellWidth(this IRow row, int columnIndex, int width)
  798. {
  799. row.Sheet.SetColumnWidth(columnIndex - 1, width * 256);
  800. return row;
  801. }
  802. /// <summary>
  803. /// 设置列宽度
  804. /// </summary>
  805. /// <param name="cell">单元格</param>
  806. /// <param name="columnIndex">第几列(从1开始计数)</param>
  807. /// <param name="width">宽度</param>
  808. /// <returns></returns>
  809. public static ICell SetCellWidth(this ICell cell, int columnIndex, int width)
  810. {
  811. cell.Sheet.SetColumnWidth(columnIndex - 1, width * 256);
  812. return cell;
  813. }
  814. /// <summary>
  815. /// 设置行高度
  816. /// </summary>
  817. /// <param name="sheet">表</param>
  818. /// <param name="rowIndex">第几行(从1开始计数)</param>
  819. /// <param name="height">宽度</param>
  820. /// <returns></returns>
  821. public static ISheet SetRowHeight(this ISheet sheet, int rowIndex, int height)
  822. {
  823. sheet.GetRow(rowIndex - 1).SetRowHeight(height);
  824. return sheet;
  825. }
  826. /// <summary>
  827. /// 设置行高度
  828. /// </summary>
  829. /// <param name="row">行</param>
  830. /// <param name="height">宽度</param>
  831. /// <returns></returns>
  832. public static IRow SetRowHeight(this IRow row, int height)
  833. {
  834. row.Height = (short)(height * 20);
  835. return row;
  836. }
  837. /// <summary>
  838. /// 设置行高度
  839. /// </summary>
  840. /// <param name="cell">单元格</param>
  841. /// <param name="height">宽度</param>
  842. /// <returns></returns>
  843. public static ICell SetRowHeight(this ICell cell, int height)
  844. {
  845. cell.Row.SetRowHeight(height);
  846. return cell;
  847. }
  848. /// <summary>
  849. /// 设置单元格样式
  850. /// </summary>
  851. /// <param name="cell"></param>
  852. /// <param name="styleStr">样式字符串</param>
  853. /// <param name="instants">样式转换器</param>
  854. /// <returns></returns>
  855. public static ICell SetCellCss(this ICell cell, CellStyleCss instants = null, string styleStr = null)
  856. {
  857. instants = instants ?? CellStyleCss.Instants;
  858. return instants.Css(cell, styleStr);
  859. }
  860. /// <summary>
  861. /// 设置单元格的值
  862. /// </summary>
  863. /// <param name="cell"></param>
  864. /// <param name="obj"></param>
  865. /// <returns></returns>
  866. public static ICell SetValue(this ICell cell, string obj)
  867. {
  868. return SetValue<string>(cell, obj);
  869. }
  870. /// <summary>
  871. /// 设置单元格的值
  872. /// </summary>
  873. /// <typeparam name="T">值类型(sting,bool,double,DateTime)</typeparam>
  874. /// <param name="cell"></param>
  875. /// <param name="obj"></param>
  876. /// <returns></returns>
  877. public static ICell SetValue<T>(this ICell cell, object obj)
  878. {
  879. if (obj == null)
  880. {
  881. return cell;
  882. }
  883. Type type = typeof(T);
  884. if (type == typeof(string))
  885. {
  886. cell.SetCellValue(obj.ToString());
  887. }
  888. else if (type == typeof(bool) && bool.TryParse(obj.ToString(), out var bValue))
  889. {
  890. cell.SetCellValue(bValue);
  891. }
  892. else if ((type == typeof(int) || type == typeof(double) || type == typeof(decimal)) && double.TryParse(obj.ToString(), out var iValue))
  893. {
  894. cell.SetCellValue(iValue);
  895. }
  896. else if (type == typeof(DateTime) && DateTime.TryParse(obj.ToString(), out var dValue))
  897. {
  898. cell.SetCellValue(dValue);
  899. }
  900. return cell;
  901. }
  902. #region 设置下拉框
  903. public static void SetCellDropdownList03(this ISheet sheet, ICell cell, string[] val, string str = "", int startRow=1, string title = "提示")
  904. {
  905. //设置生成下拉框的行和列
  906. SetCellDropdownList03(sheet, cell.ColumnIndex+1, cell.ColumnIndex+1, val, str, startRow,title);
  907. }
  908. public static void SetCellDropdownList03(this ISheet sheet, int firstCol, int lastCol, string[] val, string str = "", int startRow = 1, string title = "提示")
  909. {
  910. //设置生成下拉框的行和列
  911. var cellRegions = new CellRangeAddressList(startRow, 65535, firstCol-1, lastCol-1);
  912. //设置 下拉框内容
  913. DVConstraint constraint = DVConstraint.CreateExplicitListConstraint(val);
  914. //绑定下拉框和作用区域,并设置错误提示信息
  915. HSSFDataValidation dataValidate = new HSSFDataValidation(cellRegions, constraint);
  916. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  917. dataValidate.ShowPromptBox = true;
  918. if (!string.IsNullOrEmpty(str))
  919. {
  920. dataValidate.CreatePromptBox(title, str);
  921. }
  922. sheet.AddValidationData(dataValidate);
  923. }
  924. public static void SetCellDropdownList07(this ISheet sheet, ICell cell, string[] val, string str = "", int startRow = 1, string title = "提示")
  925. {
  926. //设置生成下拉框的行和列
  927. SetCellDropdownList07(sheet, cell.ColumnIndex+1, cell.ColumnIndex+1, val, str,startRow,title);
  928. }
  929. public static void SetCellDropdownList07(this ISheet sheet, int firstCol, int lastCol, string[] val,string str="", int startRow = 1, string title = "提示")
  930. {
  931. //设置生成下拉框的行和列
  932. var cellRegions = new CellRangeAddressList(startRow, 65535, firstCol-1, lastCol-1);
  933. XSSFDataValidationHelper dvHelper = new XSSFDataValidationHelper((XSSFSheet)sheet);
  934. //设置 下拉框内容
  935. XSSFDataValidationConstraint constraint = (XSSFDataValidationConstraint)dvHelper.CreateExplicitListConstraint(val);
  936. //绑定下拉框和作用区域,并设置错误提示信息
  937. XSSFDataValidation dataValidate = (XSSFDataValidation)dvHelper.CreateValidation(constraint, cellRegions);
  938. dataValidate.ShowErrorBox = true;
  939. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  940. if (!string.IsNullOrEmpty(str))
  941. {
  942. dataValidate.CreatePromptBox(title,str);
  943. }
  944. dataValidate.ShowPromptBox = true;
  945. sheet.AddValidationData(dataValidate);
  946. }
  947. public static void SetCellDropdownList03(this ISheet sheet, HSSFWorkbook workbook, ICell cell, string[] val, string str = "", int startRow = 1, string title = "提示")
  948. {
  949. //设置生成下拉框的行和列
  950. SetCellDropdownList03(sheet, workbook, cell.ColumnIndex+1, cell.ColumnIndex+1, val, str, startRow,title);
  951. }
  952. public static void SetCellDropdownList03(this ISheet sheet, HSSFWorkbook workbook, int firstCol, int lastCol, string[] val, string str = "", int startRow = 1, string title = "提示")
  953. {
  954. //先创建一个Sheet专门用于存储下拉项的值
  955. var name = $"Dropdown{DateTime.Now:HHmmssffff}List";
  956. ISheet sheet2 = workbook.CreateSheet(name);
  957. //隐藏
  958. workbook.SetSheetHidden(workbook.GetSheetIndex(name), true);
  959. int index = 0;
  960. foreach (var item in val)
  961. {
  962. sheet2.CreateRow(index).CreateCell(0).SetCellValue(item);
  963. index++;
  964. }
  965. //创建的下拉项的区域:
  966. var rangeName = name + "Range";
  967. IName range = workbook.CreateName();
  968. range.RefersToFormula = name + "!$A$1:$A$" + index;
  969. range.NameName = rangeName;
  970. CellRangeAddressList regions = new CellRangeAddressList(startRow, 65535, firstCol-1, lastCol-1);
  971. DVConstraint constraint = DVConstraint.CreateFormulaListConstraint(rangeName);
  972. HSSFDataValidation dataValidate = new HSSFDataValidation(regions, constraint);
  973. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  974. dataValidate.ShowPromptBox = true;
  975. if (!string.IsNullOrEmpty(str))
  976. {
  977. dataValidate.CreatePromptBox(title, str);
  978. }
  979. sheet.AddValidationData(dataValidate);
  980. }
  981. public static void SetCellDropdownList07(this ISheet sheet, XSSFWorkbook workbook, ICell cell, string[] val, string str = "", int startRow = 1, string title = "提示")
  982. {
  983. //设置生成下拉框的行和列
  984. SetCellDropdownList07(sheet, workbook, cell.ColumnIndex+1, cell.ColumnIndex+1, val, str,startRow,title);
  985. }
  986. public static void SetCellDropdownList07(this ISheet sheet, XSSFWorkbook workbook, int firstCol, int lastCol, string[] val, string str = "", int startRow = 1, string title = "提示")
  987. {
  988. //先创建一个Sheet专门用于存储下拉项的值
  989. var name = $"Dropdown{DateTime.Now:HHmmssffff}List";
  990. ISheet sheet2 = workbook.CreateSheet(name);
  991. //隐藏
  992. workbook.SetSheetHidden(workbook.GetSheetIndex(name), true);
  993. int index = 0;
  994. foreach (var item in val)
  995. {
  996. sheet2.CreateRow(index).CreateCell(0).SetCellValue(item);
  997. index++;
  998. }
  999. //创建的下拉项的区域:
  1000. var rangeName = name + "Range";
  1001. IName range = workbook.CreateName();
  1002. range.RefersToFormula = $"'{name}'!$A$1:$A${index}";
  1003. range.NameName = rangeName;
  1004. CellRangeAddressList addressList = new CellRangeAddressList(startRow, 65535, firstCol-1, lastCol-1);
  1005. var dvHelper= sheet.GetDataValidationHelper();
  1006. //设置 下拉框内容
  1007. var constraint = (XSSFDataValidationConstraint)dvHelper.CreateFormulaListConstraint(rangeName);
  1008. XSSFDataValidation dataValidate = (XSSFDataValidation)dvHelper.CreateValidation( constraint, addressList);
  1009. dataValidate.ShowErrorBox = true;
  1010. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  1011. if (!string.IsNullOrEmpty(str))
  1012. {
  1013. dataValidate.CreatePromptBox(title, str);
  1014. }
  1015. dataValidate.ShowPromptBox = true;
  1016. sheet.AddValidationData(dataValidate);
  1017. }
  1018. public static void BuildListData(this ISheet dataSheet, DropdownListData listData,ref int rowIndex)
  1019. {
  1020. int columnIndex=0;
  1021. var row = dataSheet.CreateRow(rowIndex);
  1022. var rowIndex2 = rowIndex + 1;
  1023. foreach (var l in listData.Children)
  1024. {
  1025. var cell = row.CreateCell(columnIndex);
  1026. cell.SetCellValue(l.Value);
  1027. columnIndex++;
  1028. if (l.Children.Count > 0)
  1029. {
  1030. rowIndex++;
  1031. BuildListData(dataSheet, l, ref rowIndex);
  1032. }
  1033. }
  1034. if (columnIndex > 0)
  1035. {
  1036. IName name = dataSheet.Workbook.CreateName();
  1037. name.RefersToFormula = $"'{dataSheet.SheetName}'!$A${rowIndex2}:${ConvertColumnName(columnIndex-1)}${rowIndex2}";
  1038. name.NameName = listData.Name;
  1039. }
  1040. }
  1041. public static string ConvertColumnName(this int index)
  1042. {
  1043. char[] chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZ".ToCharArray();
  1044. var last = index % 25;
  1045. var lastStr = chars[last] + "";
  1046. var first = index / 25;
  1047. string str = (first > 26 ? ConvertColumnName(first) : first > 0 ? chars[first] + "" : "") + lastStr;
  1048. return str;
  1049. }
  1050. public static IDataValidationConstraint CreateValidationConstraint(this ISheet sheet,string formula)
  1051. {
  1052. var dvHelper = sheet.GetDataValidationHelper();
  1053. var constraint = dvHelper.CreateFormulaListConstraint(formula);
  1054. return constraint;
  1055. }
  1056. public static IDataValidationConstraint CreateLinkValidationConstraint(this ISheet sheet,string formula)
  1057. {
  1058. var dvHelper = sheet.GetDataValidationHelper();
  1059. var constraint = dvHelper.CreateFormulaListConstraint($"INDIRECT(UPPER({formula}))");
  1060. return constraint;
  1061. }
  1062. public static IDataValidationConstraint CreateLinkValidationConstraint(this ISheet sheet,ICell cell)
  1063. {
  1064. var dvHelper = sheet.GetDataValidationHelper();
  1065. var constraint = dvHelper.CreateFormulaListConstraint($"INDIRECT(UPPER(${ConvertColumnName(cell.ColumnIndex)}${cell.RowIndex+1}))");
  1066. return constraint;
  1067. }
  1068. public static void SetCellDropdownList07(this ISheet sheet, IDataValidationConstraint constraint, CellRangeAddress range, string str = "", string title = "提示")
  1069. {
  1070. var dvHelper= sheet.GetDataValidationHelper();
  1071. CellRangeAddressList addressList = new CellRangeAddressList();
  1072. addressList.AddCellRangeAddress(range);
  1073. XSSFDataValidation dataValidate = (XSSFDataValidation)dvHelper.CreateValidation(constraint, addressList);
  1074. dataValidate.ShowErrorBox = true;
  1075. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  1076. if (!string.IsNullOrEmpty(str))
  1077. {
  1078. dataValidate.CreatePromptBox(title, str);
  1079. }
  1080. dataValidate.ShowPromptBox = true;
  1081. sheet.AddValidationData(dataValidate);
  1082. }
  1083. public static void SetCellDropdownList07(this ISheet sheet, IDataValidationConstraint constraint1, CellRangeAddress range1, IDataValidationConstraint constraint2, CellRangeAddress range2, string str1 = "",string str2 = "", string title1 = "提示", string title2 = "提示")
  1084. {
  1085. var dvHelper= sheet.GetDataValidationHelper();
  1086. CellRangeAddressList addressList = new CellRangeAddressList();
  1087. addressList.AddCellRangeAddress(range1);
  1088. XSSFDataValidation dataValidate = (XSSFDataValidation)dvHelper.CreateValidation(constraint1, addressList);
  1089. dataValidate.ShowErrorBox = true;
  1090. dataValidate.ShowPromptBox = true;
  1091. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  1092. if (!string.IsNullOrEmpty(str1))
  1093. {
  1094. dataValidate.CreatePromptBox(title1, str1);
  1095. }
  1096. sheet.AddValidationData(dataValidate);
  1097. addressList = new CellRangeAddressList();
  1098. addressList.AddCellRangeAddress(range2);
  1099. dataValidate = (XSSFDataValidation)dvHelper.CreateValidation(constraint2, addressList);
  1100. dataValidate.ShowErrorBox = true;
  1101. dataValidate.ShowPromptBox = true;
  1102. dataValidate.CreateErrorBox("输入不合法", "请输入或选择下拉列表中的值。");
  1103. if (!string.IsNullOrEmpty(str2))
  1104. {
  1105. dataValidate.CreatePromptBox(title2, str2);
  1106. }
  1107. sheet.AddValidationData(dataValidate);
  1108. }
  1109. #endregion
  1110. #region 列提示信息
  1111. public static void SetColumnMsg03(this ISheet sheet, string str, int firstCol, int lastCol, int startRow = 1)
  1112. {
  1113. var cellRegions = new CellRangeAddressList(startRow, 65535, firstCol - 1, lastCol - 1);
  1114. DVConstraint constraint = DVConstraint.CreateCustomFormulaConstraint("A65535");
  1115. //绑定下拉框和作用区域,并设置提示信息
  1116. HSSFDataValidation dataValidate = new HSSFDataValidation(cellRegions, constraint);
  1117. dataValidate.CreatePromptBox("提示", str);
  1118. dataValidate.ShowPromptBox = true;
  1119. sheet.AddValidationData(dataValidate);
  1120. }
  1121. public static void SetColumnMsg07(this ISheet sheet, string str, int firstCol, int lastCol, int startRow = 1, string title = "提示")
  1122. {
  1123. var cellRegions = new CellRangeAddressList(startRow, 65535, firstCol - 1, lastCol - 1);
  1124. XSSFDataValidationHelper dvHelper = new XSSFDataValidationHelper((XSSFSheet)sheet);
  1125. //var constraint =(XSSFDataValidationConstraint)dvHelper.CreateTextLengthConstraint((int)ST_DataValidationOperator.greaterThanOrEqual,"0","");
  1126. var constraint = new XSSFDataValidationConstraint(ValidationType.ANY, "");
  1127. //绑定下拉框和作用区域,并设置提示信息
  1128. XSSFDataValidation dataValidate = (XSSFDataValidation)dvHelper.CreateValidation(constraint, cellRegions);
  1129. dataValidate.CreatePromptBox(title, str);
  1130. dataValidate.ShowPromptBox = true;
  1131. sheet.AddValidationData(dataValidate);
  1132. }
  1133. #endregion
  1134. #region 批注
  1135. public static void SetCellPatriarch03(this ISheet sheet, ICell cell, string str)
  1136. {
  1137. IDrawing patriarch = (HSSFPatriarch)sheet.CreateDrawingPatriarch();
  1138. int row = cell.RowIndex, col = cell.ColumnIndex;
  1139. HSSFComment comment = (HSSFComment)patriarch.CreateCellComment(new HSSFClientAnchor(0, 0, 0, 0, col + 1, row - 1, col + 3, row + 4));
  1140. comment.SetFillColor(204, 236, 255);
  1141. comment.String = new HSSFRichTextString(str);
  1142. cell.CellComment = comment;
  1143. }
  1144. public static void SetCellPatriarch07(this ISheet sheet, ICell cell, string str)
  1145. {
  1146. IDrawing patriarch = (XSSFDrawing)sheet.CreateDrawingPatriarch();
  1147. int row = cell.RowIndex, col = cell.ColumnIndex;
  1148. XSSFComment comment = (XSSFComment)patriarch.CreateCellComment(new XSSFClientAnchor(0, 0, 0, 0, col + 1, row - 1, col + 2, row + 4));
  1149. comment.SetString(str);
  1150. cell.CellComment = comment;
  1151. }
  1152. public static void SetColPatriarch07(this ISheet sheet, ICell cell, string str)
  1153. {
  1154. IDrawing patriarch = (XSSFDrawing)sheet.CreateDrawingPatriarch();
  1155. int row = cell.RowIndex, col = cell.ColumnIndex;
  1156. XSSFComment comment = (XSSFComment)patriarch.CreateCellComment(new XSSFClientAnchor(0, 0, 0, 0, col + 1, row - 1, col + 2, row + 3));
  1157. comment.SetString(str);
  1158. cell.CellComment = comment;
  1159. }
  1160. #endregion
  1161. public static void SetCellMinNumeric07(this ISheet sheet, int firstCol,int minValue, string str = "", int startRow = 1, string title = "提示", int? lastCol = null)
  1162. {
  1163. lastCol = lastCol ?? firstCol;
  1164. CellRangeAddressList regions = new CellRangeAddressList(startRow, 65535, firstCol - 1, (int)lastCol - 1);
  1165. var dvHelper = sheet.GetDataValidationHelper();
  1166. //设置 下拉框内容
  1167. var constraint = (XSSFDataValidationConstraint)dvHelper.CreateNumericConstraint(ValidationType.DECIMAL,OperatorType.GREATER_OR_EQUAL, $"{minValue}", "");
  1168. XSSFDataValidation dataValidate = (XSSFDataValidation)dvHelper.CreateValidation(constraint, regions);
  1169. dataValidate.ShowErrorBox = true;
  1170. dataValidate.CreateErrorBox("输入不合法", "请输入大于0的数值。");
  1171. if (!string.IsNullOrEmpty(str))
  1172. {
  1173. dataValidate.CreatePromptBox(title, str);
  1174. }
  1175. dataValidate.ShowPromptBox = true;
  1176. sheet.AddValidationData(dataValidate);
  1177. }
  1178. #region DATA-FORMAT
  1179. public static ICellStyle SetCellDateTime(this ICell cell, DateTime value, string formatStr = "yyyy-MM-dd HH:mm:ss")
  1180. {
  1181. cell.SetCellValue(value);
  1182. cell.SetDataFormat(formatStr);
  1183. return cell.CellStyle;
  1184. }
  1185. public static ICellStyle SetCellString(this ICell cell, double value, string formatStr = "@")
  1186. {
  1187. cell.SetCellValue(value);
  1188. cell.SetDataFormat(formatStr);
  1189. return cell.CellStyle;
  1190. }
  1191. public static ICellStyle SetCellDouble(this ICell cell, double value, string formatStr = "0.00")
  1192. {
  1193. cell.SetCellValue(value);
  1194. cell.SetDataFormat(formatStr);
  1195. return cell.CellStyle;
  1196. }
  1197. public static ICellStyle SetCellCurrency(this ICell cell, double value, string formatStr = "¥#,###.##")
  1198. {
  1199. cell.SetCellValue(value);
  1200. cell.SetDataFormat(formatStr);
  1201. return cell.CellStyle;
  1202. }
  1203. public static ICellStyle SetCellPercent(this ICell cell, double value, string formatStr = "0.00%")
  1204. {
  1205. cell.SetCellValue(value);
  1206. cell.SetDataFormat(formatStr);
  1207. return cell.CellStyle;
  1208. }
  1209. public static void SetDataFormat(this ICell cell, string formatStr = "¥#,##0.00")
  1210. {
  1211. var cellStyle = cell.CellStyle ?? GetCellStyle(cell);
  1212. var format = cell.Sheet.Workbook.CreateDataFormat();
  1213. cellStyle.DataFormat = format.GetFormat(formatStr);
  1214. cell.SetCellStyle(cellStyle);
  1215. }
  1216. #endregion DATA-FORMAT
  1217. #region 合并单元格
  1218. /// <summary>
  1219. /// 合并单元格
  1220. /// </summary>
  1221. /// <param name="sheet"></param>
  1222. /// <param name="firstRow">开始行数(从1开始计数)</param>
  1223. /// <param name="lastRow">结束行数(从1开始计数)</param>
  1224. /// <param name="firstColumn">开始列数(从1开始计数)</param>
  1225. /// <param name="lastColumn">结束列数(从1开始计数)</param>
  1226. /// <returns></returns>
  1227. public static ISheet MergedRegion(this ISheet sheet, int firstRow, int lastRow, int firstColumn, int lastColumn)
  1228. {
  1229. var region = GetCellRegion(firstRow, lastRow, firstColumn, lastColumn);
  1230. sheet.MergedRegion(region);
  1231. return sheet;
  1232. }
  1233. public static ISheet MergedRegion(this ISheet sheet, CellRangeAddress region)
  1234. {
  1235. sheet.AddMergedRegion(region);
  1236. //SetRegionBorder(sheet, region, borderTypeStr, borderColorStr);
  1237. return sheet;
  1238. }
  1239. public static CellRangeAddress GetCellRegion(this int firstRow, int lastRow, int firstColumn, int lastColumn)
  1240. {
  1241. firstRow = firstRow <= 0 ? 1 : firstRow;
  1242. firstColumn = firstColumn <= 0 ? 1 : firstColumn;
  1243. lastRow = lastRow <= 0 ? 1 : lastRow;
  1244. lastColumn = lastColumn <= 0 ? 1 : lastColumn;
  1245. return new CellRangeAddress(firstRow - 1, lastRow - 1, firstColumn - 1, lastColumn - 1);
  1246. }
  1247. public static ISheet SetRegionBorder(this CellRangeAddress region, ISheet sheet, string borderTypeStr, string borderColorStr)
  1248. {
  1249. ICellStyle style = sheet.Workbook.CreateCellStyle();
  1250. borderTypeStr = borderTypeStr ?? "Thin";
  1251. borderColorStr = borderColorStr ?? "Black";
  1252. style.BorderColors(borderColorStr);
  1253. style.BorderTypes(borderTypeStr);
  1254. for (int i = region.FirstRow; i <= region.LastRow; i++)
  1255. {
  1256. IRow row = sheet.GenerateRow(i + 1);
  1257. if (i == region.FirstRow)
  1258. {
  1259. }
  1260. else if (i == region.LastRow)
  1261. {
  1262. }
  1263. var leftCell = row.GenerateCell(region.FirstColumn + 1);
  1264. leftCell.CellStyle.BorderLeft = style.BorderLeft;
  1265. leftCell.CellStyle.LeftBorderColor = style.LeftBorderColor;
  1266. var rightCell = row.GenerateCell(region.LastColumn + 1);
  1267. rightCell.CellStyle.BorderRight = style.BorderRight;
  1268. rightCell.CellStyle.RightBorderColor = style.RightBorderColor;
  1269. //for (int j = region.FirstColumn; j <= region.LastColumn; j++)
  1270. //{
  1271. // ICell singleCell = row.GenerateCell((short)j);
  1272. //}
  1273. }
  1274. return sheet;
  1275. }
  1276. #endregion 合并单元格
  1277. public static ICellStyle GetCellStyle(this ICell cell)
  1278. {
  1279. return cell.CellStyle ?? cell.Sheet.Workbook.CreateCellStyle();
  1280. }
  1281. public static void SetCellStyle(this ICell cell, ICellStyle cellStyle)
  1282. {
  1283. cell.CellStyle = cellStyle;
  1284. }
  1285. /// <summary>
  1286. /// 保存工作簿
  1287. /// </summary>
  1288. /// <param name="sheet"></param>
  1289. /// <param name="filePath"></param>
  1290. /// <param name="fileName"></param>
  1291. /// <returns></returns>
  1292. public static string SaveWorkBook(this ISheet sheet, string filePath, string fileName)
  1293. {
  1294. try
  1295. {
  1296. if (!Directory.Exists(filePath))
  1297. {
  1298. Directory.CreateDirectory(filePath);
  1299. }
  1300. filePath = filePath.EndsWith("\\") ? filePath : filePath + "\\";
  1301. FileStream file = new FileStream(filePath + fileName, FileMode.Create);
  1302. sheet.Workbook.Write(file);
  1303. file.Close();
  1304. return "";
  1305. }
  1306. catch (Exception e)
  1307. {
  1308. return e.Message;
  1309. }
  1310. }
  1311. /// <summary>
  1312. /// 保存工作簿
  1313. /// </summary>
  1314. /// <param name="workbook"></param>
  1315. /// <param name="filePath"></param>
  1316. /// <param name="fileName"></param>
  1317. /// <returns></returns>
  1318. public static string SaveWorkBook(this IWorkbook workbook, string filePath, string fileName)
  1319. {
  1320. try
  1321. {
  1322. if (!Directory.Exists(filePath))
  1323. {
  1324. Directory.CreateDirectory(filePath);
  1325. }
  1326. filePath = filePath.EndsWith("\\") ? filePath : filePath + "\\";
  1327. FileStream file = new FileStream(filePath + fileName, FileMode.Create);
  1328. workbook.Write(file);
  1329. file.Close();
  1330. return "";
  1331. }
  1332. catch (Exception e)
  1333. {
  1334. return e.Message;
  1335. }
  1336. }
  1337. }
  1338. public class DropdownListData
  1339. {
  1340. public DropdownListData()
  1341. {
  1342. Children = new List<DropdownListData>();
  1343. }
  1344. public DropdownListData(string name):this()
  1345. {
  1346. Name = name;
  1347. Value = name;
  1348. }
  1349. public DropdownListData(string name,string value):this(name)
  1350. {
  1351. Value = value;
  1352. }
  1353. public void AddList(DropdownListData data)
  1354. {
  1355. Children.Add(data);
  1356. }
  1357. public string Name { get; set; }
  1358. public string Value { get; set; }
  1359. public List<DropdownListData> Children { get; set; }
  1360. }
  1361. /// <summary>
  1362. /// 表示单元格的维度,通常用于表达合并单元格的维度
  1363. /// </summary>
  1364. public struct Dimension
  1365. {
  1366. /// <summary>
  1367. /// 含有数据的单元格(通常表示合并单元格的第一个跨度行第一个跨度列),该字段可能为null
  1368. /// </summary>
  1369. public ICell DataCell;
  1370. /// <summary>
  1371. /// 行跨度(跨越了多少行)
  1372. /// </summary>
  1373. public int RowSpan;
  1374. /// <summary>
  1375. /// 列跨度(跨越了多少列)
  1376. /// </summary>
  1377. public int ColumnSpan;
  1378. /// <summary>
  1379. /// 合并单元格的起始行索引
  1380. /// </summary>
  1381. public int FirstRowIndex;
  1382. /// <summary>
  1383. /// 合并单元格的结束行索引
  1384. /// </summary>
  1385. public int LastRowIndex;
  1386. /// <summary>
  1387. /// 合并单元格的起始列索引
  1388. /// </summary>
  1389. public int FirstColumnIndex;
  1390. /// <summary>
  1391. /// 合并单元格的结束列索引
  1392. /// </summary>
  1393. public int LastColumnIndex;
  1394. }
  1395. public static class ExcelExtension
  1396. {
  1397. /// <summary>
  1398. /// 判断指定行列所在的单元格是否为合并单元格,并且输出该单元格的维度
  1399. /// </summary>
  1400. /// <param name="sheet">Excel工作表</param>
  1401. /// <param name="rowIndex">行索引,从0开始</param>
  1402. /// <param name="columnIndex">列索引,从0开始</param>
  1403. /// <param name="dimension">单元格维度</param>
  1404. /// <returns>返回是否为合并单元格的布尔(Boolean)值</returns>
  1405. public static bool IsMergeCell(this ISheet sheet, int rowIndex, int columnIndex, out Dimension dimension)
  1406. {
  1407. dimension = new Dimension
  1408. {
  1409. DataCell = null,
  1410. RowSpan = 1,
  1411. ColumnSpan = 1,
  1412. FirstRowIndex = rowIndex,
  1413. LastRowIndex = rowIndex,
  1414. FirstColumnIndex = columnIndex,
  1415. LastColumnIndex = columnIndex
  1416. };
  1417. for (int i = 0; i < sheet.NumMergedRegions; i++)
  1418. {
  1419. CellRangeAddress range = sheet.GetMergedRegion(i);
  1420. sheet.IsMergedRegion(range);
  1421. //这种算法只有当指定行列索引刚好是合并单元格的第一个跨度行第一个跨度列时才能取得合并单元格的跨度
  1422. //if (range.FirstRow == rowIndex && range.FirstColumn == columnIndex)
  1423. //{
  1424. // dimension.DataCell = sheet.GetRow(range.FirstRow).GetCell(range.FirstColumn);
  1425. // dimension.RowSpan = range.LastRow - range.FirstRow + 1;
  1426. // dimension.ColumnSpan = range.LastColumn - range.FirstColumn + 1;
  1427. // dimension.FirstRowIndex = range.FirstRow;
  1428. // dimension.LastRowIndex = range.LastRow;
  1429. // dimension.FirstColumnIndex = range.FirstColumn;
  1430. // dimension.LastColumnIndex = range.LastColumn;
  1431. // break;
  1432. //}
  1433. if ((rowIndex >= range.FirstRow && range.LastRow >= rowIndex) && (columnIndex >= range.FirstColumn && range.LastColumn >= columnIndex))
  1434. {
  1435. dimension.DataCell = sheet.GetRow(range.FirstRow).GetCell(range.FirstColumn);
  1436. dimension.RowSpan = range.LastRow - range.FirstRow + 1;
  1437. dimension.ColumnSpan = range.LastColumn - range.FirstColumn + 1;
  1438. dimension.FirstRowIndex = range.FirstRow;
  1439. dimension.LastRowIndex = range.LastRow;
  1440. dimension.FirstColumnIndex = range.FirstColumn;
  1441. dimension.LastColumnIndex = range.LastColumn;
  1442. break;
  1443. }
  1444. }
  1445. bool result;
  1446. if (rowIndex >= 0 && sheet.LastRowNum > rowIndex)
  1447. {
  1448. IRow row = sheet.GetRow(rowIndex);
  1449. if (columnIndex >= 0 && row.LastCellNum > columnIndex)
  1450. {
  1451. ICell cell = row.GetCell(columnIndex);
  1452. result = cell.IsMergedCell;
  1453. if (dimension.DataCell == null)
  1454. {
  1455. dimension.DataCell = cell;
  1456. }
  1457. }
  1458. else
  1459. {
  1460. result = false;
  1461. }
  1462. }
  1463. else
  1464. {
  1465. result = false;
  1466. }
  1467. return result;
  1468. }
  1469. /// <summary>
  1470. /// 判断指定行列所在的单元格是否为合并单元格,并且输出该单元格的行列跨度
  1471. /// </summary>
  1472. /// <param name="sheet">Excel工作表</param>
  1473. /// <param name="rowIndex">行索引,从0开始</param>
  1474. /// <param name="columnIndex">列索引,从0开始</param>
  1475. /// <param name="rowSpan">行跨度,返回值最小为1,同时表示没有行合并</param>
  1476. /// <param name="columnSpan">列跨度,返回值最小为1,同时表示没有列合并</param>
  1477. /// <returns>返回是否为合并单元格的布尔(Boolean)值</returns>
  1478. public static bool IsMergeCell(this ISheet sheet, int rowIndex, int columnIndex, out int rowSpan, out int columnSpan)
  1479. {
  1480. Dimension dimension;
  1481. bool result = sheet.IsMergeCell(rowIndex, columnIndex, out dimension);
  1482. rowSpan = dimension.RowSpan;
  1483. columnSpan = dimension.ColumnSpan;
  1484. return result;
  1485. }
  1486. /// <summary>
  1487. /// 判断指定单元格是否为合并单元格,并且输出该单元格的维度
  1488. /// </summary>
  1489. /// <param name="cell">单元格</param>
  1490. /// <param name="dimension">单元格维度</param>
  1491. /// <returns>返回是否为合并单元格的布尔(Boolean)值</returns>
  1492. public static bool IsMergeCell(this ICell cell, out Dimension dimension)
  1493. {
  1494. return cell.Sheet.IsMergeCell(cell.RowIndex, cell.ColumnIndex, out dimension);
  1495. }
  1496. /// <summary>
  1497. /// 判断指定单元格是否为合并单元格,并且输出该单元格的行列跨度
  1498. /// </summary>
  1499. /// <param name="cell">单元格</param>
  1500. /// <param name="rowSpan">行跨度,返回值最小为1,同时表示没有行合并</param>
  1501. /// <param name="columnSpan">列跨度,返回值最小为1,同时表示没有列合并</param>
  1502. /// <returns>返回是否为合并单元格的布尔(Boolean)值</returns>
  1503. public static bool IsMergeCell(this ICell cell, out int rowSpan, out int columnSpan)
  1504. {
  1505. return cell.Sheet.IsMergeCell(cell.RowIndex, cell.ColumnIndex, out rowSpan, out columnSpan);
  1506. }
  1507. /// <summary>
  1508. /// 返回上一个跨度行,如果rowIndex为第一行,则返回null
  1509. /// </summary>
  1510. /// <param name="sheet">Excel工作表</param>
  1511. /// <param name="rowIndex">行索引,从0开始</param>
  1512. /// <param name="columnIndex">列索引,从0开始</param>
  1513. /// <returns>返回上一个跨度行</returns>
  1514. public static IRow PrevSpanRow(this ISheet sheet, int rowIndex, int columnIndex)
  1515. {
  1516. return sheet.FuncSheet(rowIndex, columnIndex, (currentDimension, isMerge) =>
  1517. {
  1518. //上一个单元格维度
  1519. Dimension prevDimension;
  1520. sheet.IsMergeCell(currentDimension.FirstRowIndex - 1, columnIndex, out prevDimension);
  1521. return prevDimension.DataCell.Row;
  1522. });
  1523. }
  1524. /// <summary>
  1525. /// 返回下一个跨度行,如果rowIndex为最后一行,则返回null
  1526. /// </summary>
  1527. /// <param name="sheet">Excel工作表</param>
  1528. /// <param name="rowIndex">行索引,从0开始</param>
  1529. /// <param name="columnIndex">列索引,从0开始</param>
  1530. /// <returns>返回下一个跨度行</returns>
  1531. public static IRow NextSpanRow(this ISheet sheet, int rowIndex, int columnIndex)
  1532. {
  1533. return sheet.FuncSheet(rowIndex, columnIndex, (currentDimension, isMerge) =>
  1534. isMerge ? sheet.GetRow(currentDimension.FirstRowIndex + currentDimension.RowSpan) : sheet.GetRow(rowIndex));
  1535. }
  1536. /// <summary>
  1537. /// 返回上一个跨度行,如果row为第一行,则返回null
  1538. /// </summary>
  1539. /// <param name="row">行</param>
  1540. /// <returns>返回上一个跨度行</returns>
  1541. public static IRow PrevSpanRow(this IRow row)
  1542. {
  1543. return row.Sheet.PrevSpanRow(row.RowNum, row.FirstCellNum);
  1544. }
  1545. /// <summary>
  1546. /// 返回下一个跨度行,如果row为最后一行,则返回null
  1547. /// </summary>
  1548. /// <param name="row">行</param>
  1549. /// <returns>返回下一个跨度行</returns>
  1550. public static IRow NextSpanRow(this IRow row)
  1551. {
  1552. return row.Sheet.NextSpanRow(row.RowNum, row.FirstCellNum);
  1553. }
  1554. /// <summary>
  1555. /// 返回上一个跨度列,如果columnIndex为第一列,则返回null
  1556. /// </summary>
  1557. /// <param name="row">行</param>
  1558. /// <param name="columnIndex">列索引,从0开始</param>
  1559. /// <returns>返回上一个跨度列</returns>
  1560. public static ICell PrevSpanCell(this IRow row, int columnIndex)
  1561. {
  1562. return row.Sheet.FuncSheet(row.RowNum, columnIndex, (currentDimension, isMerge) =>
  1563. {
  1564. //上一个单元格维度
  1565. Dimension prevDimension;
  1566. row.Sheet.IsMergeCell(row.RowNum, currentDimension.FirstColumnIndex - 1, out prevDimension);
  1567. return prevDimension.DataCell;
  1568. });
  1569. }
  1570. /// <summary>
  1571. /// 返回下一个跨度列,如果columnIndex为最后一列,则返回null
  1572. /// </summary>
  1573. /// <param name="row">行</param>
  1574. /// <param name="columnIndex">列索引,从0开始</param>
  1575. /// <returns>返回下一个跨度列</returns>
  1576. public static ICell NextSpanCell(this IRow row, int columnIndex)
  1577. {
  1578. return row.Sheet.FuncSheet(row.RowNum, columnIndex, (currentDimension, isMerge) =>
  1579. row.GetCell(currentDimension.FirstColumnIndex + currentDimension.ColumnSpan));
  1580. }
  1581. /// <summary>
  1582. /// 返回上一个跨度列,如果cell为第一列,则返回null
  1583. /// </summary>
  1584. /// <param name="cell">单元格</param>
  1585. /// <returns>返回上一个跨度列</returns>
  1586. public static ICell PrevSpanCell(this ICell cell)
  1587. {
  1588. return cell.Row.PrevSpanCell(cell.ColumnIndex);
  1589. }
  1590. /// <summary>
  1591. /// 返回下一个跨度列,如果columnIndex为最后一列,则返回null
  1592. /// </summary>
  1593. /// <param name="cell">单元格</param>
  1594. /// <returns>返回下一个跨度列</returns>
  1595. public static ICell NextSpanCell(this ICell cell)
  1596. {
  1597. return cell.Row.NextSpanCell(cell.ColumnIndex);
  1598. }
  1599. /// <summary>
  1600. /// 返回指定行索引所在的合并单元格(区域)中的第一行(通常是含有数据的行)
  1601. /// </summary>
  1602. /// <param name="sheet">Excel工作表</param>
  1603. /// <param name="rowIndex">行索引,从0开始</param>
  1604. /// <returns>返回指定列索引所在的合并单元格(区域)中的第一行</returns>
  1605. public static IRow GetDataRow(this ISheet sheet, int rowIndex)
  1606. {
  1607. return sheet.FuncSheet(rowIndex, 0, (currentDimension, isMerge) => sheet.GetRow(currentDimension.FirstRowIndex));
  1608. }
  1609. /// <summary>
  1610. /// 返回指定列索引所在的合并单元格(区域)中的第一行第一列(通常是含有数据的单元格)
  1611. /// </summary>
  1612. /// <param name="row">行</param>
  1613. /// <param name="columnIndex">列索引</param>
  1614. /// <returns>返回指定列索引所在的合并单元格(区域)中的第一行第一列</returns>
  1615. public static ICell GetDataCell(this IRow row, int columnIndex)
  1616. {
  1617. return row.Sheet.FuncSheet(row.RowNum, columnIndex, (currentDimension, isMerge) => currentDimension.DataCell);
  1618. }
  1619. private static T FuncSheet<T>(this ISheet sheet, int rowIndex, int columnIndex, Func<Dimension, bool, T> func)
  1620. {
  1621. //当前单元格维度
  1622. Dimension currentDimension;
  1623. //是否为合并单元格
  1624. bool isMerge = sheet.IsMergeCell(rowIndex, columnIndex, out currentDimension);
  1625. return func(currentDimension, isMerge);
  1626. }
  1627. }
  1628. public class CellStyleCss
  1629. {
  1630. public static CellStyleCss Instants => new CellStyleCss()
  1631. {
  1632. //缩写
  1633. DefaultStyle = $"bgc:{ColorType.White.ToString()};" +
  1634. $"warp:{HorizontalAlignment.Left.ToString()};" +
  1635. $"align:{HorizontalAlignment.Left.ToString()};" +
  1636. $"v-align:{VerticalAlignment.Center.ToString()};" +
  1637. //$"b:{BorderStyle.None.ToString()};" +
  1638. //$"bc:{ColorType.Black.ToString()};" +
  1639. "inden:0;" +
  1640. "df:@;"
  1641. };
  1642. public void SetDefaultStyle(string styleStr)
  1643. {
  1644. var dic = new SortedDictionary<string, string>();
  1645. DefaultStyle = dic.GetCleanStyle("");
  1646. DefaultStyle = dic.GetCleanStyle(styleStr);
  1647. }
  1648. /// <summary>
  1649. /// 默认样式css
  1650. /// </summary>
  1651. private string DefaultStyle { get; set; }
  1652. //标准写法
  1653. //private static readonly string DefaultFontStyle = "font-color:black;" +
  1654. // "font-name:Arial;" +
  1655. // "font-size:10;" +
  1656. // "font-weight:normal;" +
  1657. // "font-underline:none;" +
  1658. // "font-italic:false;" +
  1659. // "font-strikeout:false;" +
  1660. // "font-superscript:none;"+
  1661. // "background-color:white;"+
  1662. // "text-align:none;"+
  1663. // "vertical-align:none;"+
  1664. // "data-format:none;border-type:Thin";
  1665. /// <summary>
  1666. /// 把css样式设置给单元格
  1667. /// </summary>
  1668. /// <param name="cell"></param>
  1669. /// <param name="styleStr"></param>
  1670. /// <returns></returns>
  1671. public ICell Css(ICell cell, string styleStr = null)
  1672. {
  1673. var dic = new SortedDictionary<string, string>();
  1674. var sortedCss = dic.GetCleanStyle(DefaultStyle);
  1675. if (!string.IsNullOrEmpty(styleStr))
  1676. sortedCss = dic.GetCleanStyle(styleStr);
  1677. var cssKey = $"CellStyle_{sortedCss.Md5()}";
  1678. var workbook = cell.Sheet.Workbook;
  1679. ICellStyle cellStyle = workbook.GetCellStyle(cssKey);
  1680. if (cellStyle == null)
  1681. {
  1682. cellStyle = workbook.GetCellStyle(dic, cell);
  1683. workbook.AttachedCellStyle(cssKey, cellStyle);
  1684. }
  1685. cell.CellStyle = cellStyle;
  1686. //cell.CellStyle = workbook.GetCellStyle(dic, cell);
  1687. return cell;//返回单元格方便流水式编程
  1688. }
  1689. public ICellStyle GetCssStyle(IWorkbook workbook, string styleStr)
  1690. {
  1691. var dic = new SortedDictionary<string, string>();
  1692. var sortedCss = dic.GetCleanStyle(DefaultStyle);
  1693. if (!string.IsNullOrEmpty(styleStr))
  1694. sortedCss = dic.GetCleanStyle(styleStr);
  1695. var cssKey = $"CellStyle_{sortedCss.Md5()}";
  1696. var cellStyle = workbook.GetCellStyle(cssKey);
  1697. if (cellStyle == null)
  1698. {
  1699. cellStyle = workbook.GetCellStyle(dic);
  1700. workbook.AttachedCellStyle(cssKey, cellStyle);
  1701. }
  1702. return cellStyle;
  1703. }
  1704. }
  1705. internal static class CellStyleRender
  1706. {
  1707. #region 解析css样式
  1708. /// <summary>
  1709. /// 默认字体样式
  1710. /// </summary>
  1711. private static string DefaultFontStyle { get; } = $"fc:{ColorType.Black.ToString()};" +//font-color
  1712. "fn:宋体;" +//font-name
  1713. "fs:12;" +//font-size
  1714. "fw:normal;" +//font-weight
  1715. "fu:none;" +//font-underline
  1716. "fi:false;" +//font-italic
  1717. "fst:false;" +//font-strikeout
  1718. "fss:none;";//font-superscript
  1719. #region 设置样式
  1720. /// <summary>
  1721. /// 缓存
  1722. /// </summary>
  1723. private static readonly ConditionalWeakTable<IWorkbook, Dictionary<string, ICellStyle>> Table =
  1724. new ConditionalWeakTable<IWorkbook, Dictionary<string, ICellStyle>>();
  1725. /// <summary>
  1726. /// 获取CellStyle
  1727. /// </summary>
  1728. /// <param name="workbook"></param>
  1729. /// <param name="dic"></param>
  1730. /// <param name="cell"></param>
  1731. /// <returns></returns>
  1732. public static ICellStyle GetCellStyle(this IWorkbook workbook, SortedDictionary<string, string> dic, ICell cell = null)
  1733. {
  1734. ICellStyle cellStyle = workbook.CreateCellStyle();
  1735. //if (cell != null)
  1736. //{
  1737. // cellStyle.CloneStyleFrom(cell.CellStyle);
  1738. //}
  1739. var fontStyles = dic.Where(w => w.Key.StartsWith("font-")).ToArray();
  1740. var fontDic = new SortedDictionary<string, string>();
  1741. foreach (var kv in fontStyles)
  1742. {
  1743. fontDic.Add(kv.Key, kv.Value);
  1744. }
  1745. var font = workbook.GetFont(fontDic);
  1746. cellStyle.SetFont(font);//TODO 在基于style.xls基础的样式上增加css时,会造成原字体设置的丢失
  1747. var xdic = dic.Where(w => !w.Key.StartsWith("font-")).ToArray();
  1748. foreach (var kvp in xdic)
  1749. {
  1750. FireCssAccess(cellStyle, workbook, kvp);
  1751. }
  1752. return cellStyle;
  1753. }
  1754. /// <summary>
  1755. /// 从缓存读取CellStyle
  1756. /// </summary>
  1757. /// <param name="workbook"></param>
  1758. /// <param name="propertyName"></param>
  1759. /// <returns></returns>
  1760. public static ICellStyle GetCellStyle(this IWorkbook workbook, string propertyName)
  1761. {
  1762. if (!Table.TryGetValue(workbook, out var values)) return null;
  1763. if (values.TryGetValue(propertyName, out var temp))
  1764. return temp;
  1765. return null;
  1766. }
  1767. /// <summary>
  1768. /// 缓存CellStyle
  1769. /// </summary>
  1770. /// <param name="workbook"></param>
  1771. /// <param name="propertyName"></param>
  1772. /// <param name="value"></param>
  1773. public static void AttachedCellStyle(this IWorkbook workbook, string propertyName, ICellStyle value)
  1774. {
  1775. if (!Table.TryGetValue(workbook, out var values))
  1776. {
  1777. values = new Dictionary<string, ICellStyle>();
  1778. Table.Add(workbook, values);
  1779. }
  1780. values[propertyName] = value;
  1781. }
  1782. /// <summary>
  1783. /// Md5 key
  1784. /// </summary>
  1785. /// <param name="input"></param>
  1786. /// <returns></returns>
  1787. public static string Md5(this string input)
  1788. {
  1789. if (input == null)
  1790. input = string.Empty;
  1791. byte[] data = Encoding.UTF8.GetBytes(input.Trim().ToLowerInvariant());
  1792. using (var md5 = new MD5CryptoServiceProvider())
  1793. {
  1794. data = md5.ComputeHash(data);
  1795. }
  1796. var ret = new StringBuilder();
  1797. foreach (byte b in data)
  1798. {
  1799. ret.Append(b.ToString("x2").ToLowerInvariant());
  1800. }
  1801. return ret.ToString();
  1802. }
  1803. /// <summary>
  1804. /// 设置不是字体的样式
  1805. /// </summary>
  1806. /// <param name="style"></param>
  1807. /// <param name="workbook"></param>
  1808. /// <param name="kvp"></param>
  1809. private static void FireCssAccess(ICellStyle style, IWorkbook workbook, KeyValuePair<string, string> kvp)
  1810. {
  1811. switch (kvp.Key)
  1812. {
  1813. case "WrapText":
  1814. style.TextWrap(kvp.Value);
  1815. break;
  1816. case "Indention":
  1817. style.TextIndention(kvp.Value);
  1818. break;
  1819. case "text-align":
  1820. style.TextAlign(kvp.Value);
  1821. break;
  1822. case "vertical-align":
  1823. style.VerticalAlign(kvp.Value);
  1824. break;
  1825. case "background-color":
  1826. style.BackgroundColor(kvp.Value);
  1827. break;
  1828. case "border-type":
  1829. style.BorderTypes(kvp.Value);
  1830. break;
  1831. case "top-border-type":
  1832. style.BorderTopTypes(kvp.Value);
  1833. break;
  1834. case "right-border-type":
  1835. style.BorderRightTypes(kvp.Value);
  1836. break;
  1837. case "bottom-border-type":
  1838. style.BorderBottomTypes(kvp.Value);
  1839. break;
  1840. case "left-border-type":
  1841. style.BorderLeftTypes(kvp.Value);
  1842. break;
  1843. case "border-color":
  1844. style.BorderColors(kvp.Value);
  1845. break;
  1846. case "top-border-color":
  1847. style.BorderTopColors(kvp.Value);
  1848. break;
  1849. case "right-border-color":
  1850. style.BorderRightColors(kvp.Value);
  1851. break;
  1852. case "bottom-border-color":
  1853. style.BorderBottomColors(kvp.Value);
  1854. break;
  1855. case "left-border-color":
  1856. style.BorderLeftColors(kvp.Value);
  1857. break;
  1858. case "data-format":
  1859. style.DataFormat(workbook, kvp.Value);
  1860. break;
  1861. }
  1862. }
  1863. /// <summary>
  1864. /// 获取字体样式
  1865. /// </summary>
  1866. /// <param name="workbook"></param>
  1867. /// <param name="fontdic"></param>
  1868. /// <returns></returns>
  1869. private static IFont GetFont(this IWorkbook workbook, SortedDictionary<string, string> fontdic)
  1870. {
  1871. var weight = fontdic.FontWeight();
  1872. var color = fontdic.FontColor();
  1873. var size = fontdic.FontSize();
  1874. var name = fontdic.FontName();
  1875. var underline = fontdic.FontUnderline();
  1876. var italic = fontdic.FontItalic();
  1877. var strikeout = fontdic.FontStrikeout();
  1878. var offset = fontdic.ConvertToSuperScript();
  1879. var findHeight = (short)(size * 20);
  1880. var font = workbook.FindFont(weight, color, findHeight, name, italic, strikeout, offset, underline);
  1881. if (font == null)
  1882. {
  1883. font = workbook.CreateFont();
  1884. font.Boldweight = weight;
  1885. font.Color = color;
  1886. font.FontHeightInPoints = size;
  1887. font.FontName = name;
  1888. font.Underline = underline;
  1889. font.IsItalic = italic;
  1890. font.IsStrikeout = strikeout;
  1891. font.TypeOffset = offset;
  1892. }
  1893. return font;
  1894. }
  1895. #endregion 设置样式
  1896. #region 获取样式
  1897. /// <summary>
  1898. /// 默认设置
  1899. /// </summary>
  1900. /// <param name="dic"></param>
  1901. public static void InitStyleDic(this SortedDictionary<string, string> dic)
  1902. {
  1903. var cssItems = GetCssItems(DefaultFontStyle);
  1904. foreach (var cssitem in cssItems)
  1905. {
  1906. var kvp = GetCssKeyValue(cssitem);
  1907. if (dic.ContainsKey(kvp.Key))
  1908. dic[kvp.Key] = kvp.Value; //覆盖相同key的值
  1909. else
  1910. dic.Add(kvp.Key, kvp.Value);
  1911. }
  1912. }
  1913. /// <summary>
  1914. /// 获取样式简洁字符串
  1915. /// </summary>
  1916. /// <param name="dic"></param>
  1917. /// <param name="style"></param>
  1918. /// <returns></returns>
  1919. public static string GetCleanStyle(this SortedDictionary<string, string> dic, string style)
  1920. {
  1921. style = Regex.Replace(style.Trim(), "\\s+", " ");
  1922. style = Regex.Replace(style, "\\s;\\s", ";");
  1923. style = Regex.Replace(style, "\\s:\\s", ":");
  1924. InitStyleDic(dic);
  1925. var cssItems = GetCssItems(style.TrimEnd(';'));
  1926. foreach (var cssitem in cssItems)
  1927. {
  1928. var kvp = GetCssKeyValue(cssitem);
  1929. if (dic.ContainsKey(kvp.Key))
  1930. dic[kvp.Key] = kvp.Value; //覆盖相同key的值
  1931. else
  1932. dic.Add(kvp.Key, kvp.Value);
  1933. }
  1934. var sortedCss = string.Join(";", dic.Select(s => $"{s.Key}:{s.Value}").ToArray());
  1935. return sortedCss;
  1936. }
  1937. /// <summary>
  1938. /// 获取样式数组
  1939. /// </summary>
  1940. /// <param name="style"></param>
  1941. /// <returns></returns>
  1942. private static string[] GetCssItems(string style)
  1943. {
  1944. var cssItems = Regex.Split(style, ";");
  1945. cssItems = cssItems.Where(w => !string.IsNullOrWhiteSpace(w)).ToArray();
  1946. return cssItems;
  1947. }
  1948. /// <summary>
  1949. /// 获取css样式
  1950. /// </summary>
  1951. /// <param name="css"></param>
  1952. /// <returns></returns>
  1953. private static KeyValuePair<string, string> GetCssKeyValue(string css)
  1954. {
  1955. var cssKeyValueArray = Regex.Split(css, ":").ToArray();
  1956. var cssKey = cssKeyValueArray[0].StandardCssKey();
  1957. var cssValue = cssKey == "font-name" ? cssKeyValueArray[1] : cssKeyValueArray[1].ToUpper(); //字体不应变大写
  1958. var kv = new KeyValuePair<string, string>(cssKey, cssValue);
  1959. return kv;
  1960. }
  1961. #endregion 获取样式
  1962. #region 转换Css的 Key
  1963. /// <summary>
  1964. /// 缩写Key 转换成标准Key
  1965. /// </summary>
  1966. /// <param name="csskey"></param>
  1967. /// <returns></returns>
  1968. private static string StandardCssKey(this string csskey)
  1969. {
  1970. if (CssKeyDic.ContainsKey(csskey))
  1971. {
  1972. var sKey = CssKeyDic[csskey];
  1973. return sKey;
  1974. }
  1975. return csskey;
  1976. }
  1977. /// <summary>
  1978. /// key 转换字典
  1979. /// </summary>
  1980. private static Dictionary<string, string> CssKeyDic => new Dictionary<string, string>
  1981. {
  1982. {"color", "font-color"},
  1983. {"fc", "font-color"},
  1984. {"fw", "font-weight"},
  1985. {"fn", "font-name"},
  1986. {"fs", "font-size"},
  1987. {"italic", "font-italic"},
  1988. {"fi", "font-italic"},
  1989. {"underline", "font-underline"},
  1990. {"fu", "font-underline"},
  1991. {"u", "font-underline"},
  1992. {"deleteline", "font-strikeout"},
  1993. {"d-line", "font-strikeout"},
  1994. {"strikeout", "font-strikeout"},
  1995. {"fst", "font-strikeout"},
  1996. {"d", "font-strikeout"},
  1997. {"font-offset", "font-superscript"},
  1998. {"superscript", "font-superscript"},
  1999. {"fss", "font-superscript"},
  2000. {"ss", "font-superscript"},
  2001. {"bg-color", "background-color"},
  2002. {"bg-c", "background-color"},
  2003. {"bgc", "background-color"},
  2004. {"align", "text-align"},
  2005. {"wrap", "WrapText"},
  2006. {"inden", "Indention"},
  2007. {"in", "Indention"},
  2008. {"v-align", "vertical-align"},
  2009. {"b-t", "border-type"},
  2010. {"b", "border-type"},
  2011. {"bt", "top-border-type"},
  2012. {"br", "right-border-type"},
  2013. {"bb", "bottom-border-type"},
  2014. {"bl", "left-border-type"},
  2015. {"b-c", "border-color"},
  2016. {"bc", "border-color"},
  2017. {"btc", "top-border-color"},
  2018. {"brc", "right-border-color"},
  2019. {"bbc", "bottom-border-color"},
  2020. {"blc", "left-border-color"},
  2021. {"format", "data-format"},
  2022. {"df", "data-format"}
  2023. };
  2024. #endregion 转换Css的 Key
  2025. #region 样式转换
  2026. #region font-weight
  2027. private static short FontWeight(this SortedDictionary<string, string> fontdic)
  2028. {
  2029. switch (fontdic["font-weight"])
  2030. {
  2031. case "NORMAL":
  2032. return 400;
  2033. case "BOLD":
  2034. return 700;
  2035. default:
  2036. return 0;
  2037. }
  2038. }
  2039. #endregion font-weight
  2040. #region font-name
  2041. private static string FontName(this SortedDictionary<string, string> fontdic)
  2042. {
  2043. return fontdic["font-name"];
  2044. }
  2045. #endregion font-name
  2046. #region font-size
  2047. private static short FontSize(this SortedDictionary<string, string> fontdic)
  2048. {
  2049. return short.TryParse(fontdic["font-size"], out var value) ? value : (short)10;
  2050. }
  2051. #endregion font-size
  2052. #region font-color
  2053. private static short FontColor(this SortedDictionary<string, string> fontdic)
  2054. {
  2055. var color = fontdic["font-color"].ConvertToColor();
  2056. return color;
  2057. }
  2058. #endregion font-color
  2059. #region font-italic
  2060. private static bool FontItalic(this SortedDictionary<string, string> fontdic)
  2061. {
  2062. return fontdic["font-italic"] == "TRUE";
  2063. }
  2064. #endregion font-italic
  2065. #region font-strikeout
  2066. /// <summary>
  2067. /// 删除线
  2068. /// </summary>
  2069. /// <param name="fontdic"></param>
  2070. /// <returns></returns>
  2071. private static bool FontStrikeout(this SortedDictionary<string, string> fontdic)
  2072. {
  2073. return fontdic["font-strikeout"] == "TRUE";
  2074. }
  2075. #endregion font-strikeout
  2076. #region WrapText
  2077. private static void TextWrap(this ICellStyle style, string v)
  2078. {
  2079. style.WrapText = v.ToUpper() == "TRUE";
  2080. }
  2081. #endregion WrapText
  2082. #region TextIndention
  2083. /// <summary>
  2084. /// 缩进
  2085. /// </summary>
  2086. /// <param name="style"></param>
  2087. /// <param name="v"></param>
  2088. private static void TextIndention(this ICellStyle style, string v)
  2089. {
  2090. if (short.TryParse(v, out var value))
  2091. {
  2092. style.Indention = value;
  2093. }
  2094. }
  2095. #endregion TextIndention
  2096. #region text-align
  2097. private static void TextAlign(this ICellStyle style, string v)
  2098. {
  2099. style.Alignment = v.ConvertToHorizontalAlignment();
  2100. }
  2101. #endregion text-align
  2102. #region vertical-align
  2103. private static void VerticalAlign(this ICellStyle style, string v)
  2104. {
  2105. style.VerticalAlignment = v.ConvertToVerticalAlignment();
  2106. }
  2107. #endregion vertical-align
  2108. #region boder-type / boder-color
  2109. internal static void BorderTypes(this ICellStyle style, string v)
  2110. {
  2111. if (string.IsNullOrEmpty(v))
  2112. return;
  2113. string[] borderTypeNames = { string.Empty, string.Empty, string.Empty, string.Empty };
  2114. v = v.ToUpper();
  2115. var vs = v.Split(' ');
  2116. switch (vs.Length)
  2117. {
  2118. case 1:
  2119. borderTypeNames[0] = borderTypeNames[1] = borderTypeNames[2] = borderTypeNames[3] = vs[0];
  2120. break;
  2121. case 2:
  2122. borderTypeNames[0] = borderTypeNames[2] = vs[0];
  2123. borderTypeNames[1] = borderTypeNames[3] = vs[1];
  2124. break;
  2125. case 3:
  2126. borderTypeNames[0] = vs[0];
  2127. borderTypeNames[1] = borderTypeNames[3] = vs[1];
  2128. borderTypeNames[2] = vs[2];
  2129. break;
  2130. case 4:
  2131. borderTypeNames[0] = vs[0];
  2132. borderTypeNames[1] = vs[1];
  2133. borderTypeNames[2] = vs[2];
  2134. borderTypeNames[3] = vs[3];
  2135. break;
  2136. }
  2137. var borderTopTypeName = borderTypeNames[0];
  2138. var borderRightTypeName = borderTypeNames[1];
  2139. var borderBottomTypeName = borderTypeNames[2];
  2140. var borderLeftTypeName = borderTypeNames[3];
  2141. if (!string.IsNullOrWhiteSpace(borderTopTypeName))
  2142. style.BorderTop = borderTopTypeName.ConvertToBorderStyle();
  2143. if (!string.IsNullOrWhiteSpace(borderRightTypeName))
  2144. style.BorderRight = borderRightTypeName.ConvertToBorderStyle();
  2145. if (!string.IsNullOrWhiteSpace(borderBottomTypeName))
  2146. style.BorderBottom = borderBottomTypeName.ConvertToBorderStyle();
  2147. if (!string.IsNullOrWhiteSpace(borderLeftTypeName))
  2148. style.BorderLeft = borderLeftTypeName.ConvertToBorderStyle();
  2149. }
  2150. private static void BorderTopTypes(this ICellStyle style, string v)
  2151. {
  2152. if (!string.IsNullOrWhiteSpace(v))
  2153. style.BorderTop = v.ConvertToBorderStyle();
  2154. }
  2155. private static void BorderBottomTypes(this ICellStyle style, string v)
  2156. {
  2157. if (!string.IsNullOrWhiteSpace(v))
  2158. style.BorderBottom = v.ConvertToBorderStyle();
  2159. }
  2160. private static void BorderLeftTypes(this ICellStyle style, string v)
  2161. {
  2162. if (!string.IsNullOrWhiteSpace(v))
  2163. style.BorderLeft = v.ConvertToBorderStyle();
  2164. }
  2165. private static void BorderRightTypes(this ICellStyle style, string v)
  2166. {
  2167. if (!string.IsNullOrWhiteSpace(v))
  2168. style.BorderRight = v.ConvertToBorderStyle();
  2169. }
  2170. internal static void BorderColors(this ICellStyle style, string v)
  2171. {
  2172. if (string.IsNullOrEmpty(v))
  2173. return;
  2174. string[] borderColors = { string.Empty, string.Empty, string.Empty, string.Empty };
  2175. v = v.ToUpper();
  2176. var vs = v.Split(' ');
  2177. switch (vs.Length)
  2178. {
  2179. case 1:
  2180. borderColors[0] = borderColors[1] = borderColors[2] = borderColors[3] = vs[0];
  2181. break;
  2182. case 2:
  2183. borderColors[0] = borderColors[2] = vs[0];
  2184. borderColors[1] = borderColors[3] = vs[1];
  2185. break;
  2186. case 3:
  2187. borderColors[0] = vs[0];
  2188. borderColors[1] = borderColors[3] = vs[1];
  2189. borderColors[2] = vs[2];
  2190. break;
  2191. case 4:
  2192. borderColors[0] = vs[0];
  2193. borderColors[1] = vs[1];
  2194. borderColors[2] = vs[2];
  2195. borderColors[3] = vs[3];
  2196. break;
  2197. }
  2198. var borderTopColor = borderColors[0];
  2199. var borderRightColor = borderColors[1];
  2200. var borderBottomColor = borderColors[2];
  2201. var borderLeftColor = borderColors[3];
  2202. if (!string.IsNullOrWhiteSpace(borderTopColor))
  2203. style.TopBorderColor = borderTopColor.ConvertToColor();
  2204. if (!string.IsNullOrWhiteSpace(borderRightColor))
  2205. style.RightBorderColor = borderRightColor.ConvertToColor();
  2206. if (!string.IsNullOrWhiteSpace(borderBottomColor))
  2207. style.BottomBorderColor = borderBottomColor.ConvertToColor();
  2208. if (!string.IsNullOrWhiteSpace(borderLeftColor))
  2209. style.LeftBorderColor = borderLeftColor.ConvertToColor();
  2210. }
  2211. private static void BorderTopColors(this ICellStyle style, string v)
  2212. {
  2213. if (!string.IsNullOrWhiteSpace(v))
  2214. style.TopBorderColor = v.ConvertToColor();
  2215. }
  2216. private static void BorderBottomColors(this ICellStyle style, string v)
  2217. {
  2218. if (!string.IsNullOrWhiteSpace(v))
  2219. style.BottomBorderColor = v.ConvertToColor();
  2220. }
  2221. private static void BorderLeftColors(this ICellStyle style, string v)
  2222. {
  2223. if (!string.IsNullOrWhiteSpace(v))
  2224. style.LeftBorderColor = v.ConvertToColor();
  2225. }
  2226. private static void BorderRightColors(this ICellStyle style, string v)
  2227. {
  2228. if (!string.IsNullOrWhiteSpace(v))
  2229. style.RightBorderColor = v.ConvertToColor();
  2230. }
  2231. #endregion boder-type / boder-color
  2232. #region data-format
  2233. private static void DataFormat(this ICellStyle style, IWorkbook workbook, string v)
  2234. {
  2235. if (string.IsNullOrEmpty(v))
  2236. return;
  2237. var df = workbook.CreateDataFormat();
  2238. style.DataFormat = df.GetFormat(v);
  2239. }
  2240. #endregion data-format
  2241. #region BackgroundColor
  2242. private static void BackgroundColor(this ICellStyle style, string v)
  2243. {
  2244. if (string.IsNullOrEmpty(v))
  2245. return;
  2246. style.FillPattern = FillPattern.SolidForeground;
  2247. style.FillForegroundColor = v.ConvertToColor();
  2248. }
  2249. #endregion BackgroundColor
  2250. private static FontSuperScript ConvertToSuperScript(this SortedDictionary<string, string> fontdic)
  2251. {
  2252. var v = fontdic["font-superscript"];
  2253. switch (v)
  2254. {
  2255. case "SUPER":
  2256. return FontSuperScript.Super;
  2257. case "SUB":
  2258. return FontSuperScript.Sub;
  2259. default:
  2260. return FontSuperScript.None;
  2261. }
  2262. }
  2263. private static FontUnderlineType FontUnderline(this SortedDictionary<string, string> fontdic)
  2264. {
  2265. var v = fontdic["font-underline"];
  2266. switch (v)
  2267. {
  2268. case "SINGLE":
  2269. return FontUnderlineType.Single;
  2270. case "DOUBLE":
  2271. return FontUnderlineType.Double;
  2272. case "SINGLEACCOUNTING":
  2273. case "SINGLE_ACCOUNTING":
  2274. return FontUnderlineType.SingleAccounting;
  2275. case "DOUBLEACCOUNTING":
  2276. case "DOUBLE_ACCOUNTING":
  2277. return FontUnderlineType.DoubleAccounting;
  2278. default:
  2279. return FontUnderlineType.None;
  2280. }
  2281. }
  2282. public static short ConvertToColor(this string v)
  2283. {
  2284. if (string.IsNullOrEmpty(v))
  2285. return 32767;
  2286. switch (v.ToUpper())
  2287. {
  2288. case "AQUA":
  2289. return (short)ColorType.Aqua;
  2290. case "AUTOMATIC":
  2291. return (short)ColorType.Automatic;
  2292. case "BLACK":
  2293. return (short)ColorType.Black;
  2294. case "BLUE":
  2295. return (short)ColorType.Blue;
  2296. case "BLUE_GREY":
  2297. case "BLUEGREY":
  2298. return (short)ColorType.BlueGrey;
  2299. case "BRIGHT_GREEN":
  2300. case "BRIGHTGREEN":
  2301. return (short)ColorType.BrightGreen;
  2302. case "BROWN":
  2303. return (short)ColorType.Brown;
  2304. case "CORAL":
  2305. return (short)ColorType.Coral;
  2306. case "CORNFLOWER_BLUE":
  2307. case "CORNFLOWERBLUE":
  2308. return (short)ColorType.CornflowerBlue;
  2309. case "DARK_BLUE":
  2310. case "DARKBLUE":
  2311. return (short)ColorType.DarkBlue;
  2312. case "DARK_GREEN":
  2313. case "DARKGREEN":
  2314. return (short)ColorType.DarkGreen;
  2315. case "DARK_RED":
  2316. case "DARKRED":
  2317. return (short)ColorType.DarkRed;
  2318. case "DARK_TEAL":
  2319. case "DARKTEAL":
  2320. return (short)ColorType.DarkTeal;
  2321. case "DARK_YELLOW":
  2322. case "DARKYELLOW":
  2323. return (short)ColorType.DarkYellow;
  2324. case "GOLD":
  2325. return (short)ColorType.Gold;
  2326. case "GREEN":
  2327. return (short)ColorType.Green;
  2328. case "GREY_25_PERCENT":
  2329. case "GREY25PERCENT":
  2330. return (short)ColorType.Grey25Percent;
  2331. case "GREY_40_PERCENT":
  2332. case "GREY40PERCENT":
  2333. return (short)ColorType.Grey40Percent;
  2334. case "GREY_50_PERCENT":
  2335. case "GREY50PERCENT":
  2336. return (short)ColorType.Grey50Percent;
  2337. case "GREY_80_PERCENT":
  2338. case "GREY80PERCENT":
  2339. return (short)ColorType.Grey80Percent;
  2340. case "INDIGO":
  2341. return (short)ColorType.Indigo;
  2342. case "LAVENDER":
  2343. return (short)ColorType.Lavender;
  2344. case "LEMON_CHIFFON":
  2345. case "LEMONCHIFFON":
  2346. return (short)ColorType.LemonChiffon;
  2347. case "LIGHT_BLUE":
  2348. case "LIGHTBLUE":
  2349. return (short)ColorType.LightBlue;
  2350. case "LIGHT_CORNFLOWERBLUE":
  2351. case "LIGHTCORNFLOWERBLUE":
  2352. return (short)ColorType.LightCornflowerBlue;
  2353. case "LIGHT_GREEN":
  2354. case "LIGHTGREEN":
  2355. return (short)ColorType.LightGreen;
  2356. case "LIGHT_ORANGE":
  2357. case "LIGHTORANGE":
  2358. return (short)ColorType.LightOrange;
  2359. case "LIGHT_TURQUOISE":
  2360. case "LIGHTTURQUOISE":
  2361. return (short)ColorType.LightTurquoise;
  2362. case "LIGHT_YELLOW":
  2363. case "LIGHTYELLOW":
  2364. return (short)ColorType.LightYellow;
  2365. case "LIME":
  2366. return (short)ColorType.Lime;
  2367. case "MAROON":
  2368. return (short)ColorType.Maroon;
  2369. case "OLIVE_GREEN":
  2370. case "OLIVEGREEN":
  2371. return (short)ColorType.OliveGreen;
  2372. case "ORANGE":
  2373. return (short)ColorType.Orange;
  2374. case "ORCHID":
  2375. return (short)ColorType.Orchid;
  2376. case "PALE_BLUE":
  2377. case "PALEBLUE":
  2378. return (short)ColorType.PaleBlue;
  2379. case "PINK":
  2380. return (short)ColorType.Pink;
  2381. case "PLUM":
  2382. return (short)ColorType.Plum;
  2383. case "RED":
  2384. return (short)ColorType.Red;
  2385. case "ROSE":
  2386. return (short)ColorType.Rose;
  2387. case "ROYAL_BLUE":
  2388. case "ROYALBLUE":
  2389. return (short)ColorType.RoyalBlue;
  2390. case "SEA_GREEN":
  2391. case "SEAGREEN":
  2392. return (short)ColorType.SeaGreen;
  2393. case "SKY_BLUE":
  2394. case "SKYBLUE":
  2395. return (short)ColorType.SkyBlue;
  2396. case "TAN":
  2397. return (short)ColorType.Tan;
  2398. case "TEAL":
  2399. return (short)ColorType.Teal;
  2400. case "TURQUOISE":
  2401. return (short)ColorType.Turquoise;
  2402. case "VIOLET":
  2403. return (short)ColorType.Violet;
  2404. case "WHITE":
  2405. return (short)ColorType.White;
  2406. case "YELLOW":
  2407. return (short)ColorType.Yellow;
  2408. default:
  2409. return 32767;
  2410. }
  2411. }
  2412. private static HorizontalAlignment ConvertToHorizontalAlignment(this string v)
  2413. {
  2414. if (string.IsNullOrEmpty(v))
  2415. return HorizontalAlignment.General;
  2416. switch (v.ToUpper())
  2417. {
  2418. case "LEFT":
  2419. return HorizontalAlignment.Left;
  2420. case "CENTER":
  2421. return HorizontalAlignment.Center;
  2422. case "CENTERSELECTION":
  2423. case "CENTER_SELECTION":
  2424. return HorizontalAlignment.CenterSelection;
  2425. case "RIGHT":
  2426. return HorizontalAlignment.Right;
  2427. case "DISTRIBUTED":
  2428. return HorizontalAlignment.Distributed;
  2429. case "FILL":
  2430. return HorizontalAlignment.Fill;
  2431. case "JUSTIFY":
  2432. return HorizontalAlignment.Justify;
  2433. default:
  2434. return HorizontalAlignment.General;
  2435. }
  2436. }
  2437. private static VerticalAlignment ConvertToVerticalAlignment(this string v)
  2438. {
  2439. if (string.IsNullOrEmpty(v))
  2440. return VerticalAlignment.Justify;
  2441. switch (v.ToUpper())
  2442. {
  2443. case "TOP":
  2444. return VerticalAlignment.Top;
  2445. case "CENTER":
  2446. return VerticalAlignment.Center;
  2447. case "BOTTOM":
  2448. return VerticalAlignment.Bottom;
  2449. case "DISTRIBUTED":
  2450. return VerticalAlignment.Distributed;
  2451. default:
  2452. return VerticalAlignment.Justify;
  2453. }
  2454. }
  2455. public static BorderStyle ConvertToBorderStyle(this string v)
  2456. {
  2457. if (string.IsNullOrEmpty(v))
  2458. return BorderStyle.None;
  2459. switch (v.ToUpper())
  2460. {
  2461. case "THIN":
  2462. return BorderStyle.Thin;
  2463. case "MEDIUM":
  2464. return BorderStyle.Medium;
  2465. case "DASHED":
  2466. return BorderStyle.Dashed;
  2467. case "HAIR":
  2468. return BorderStyle.Hair;
  2469. case "THICK":
  2470. return BorderStyle.Thick;
  2471. case "DOUBLE":
  2472. return BorderStyle.Double;
  2473. case "DOTTED":
  2474. return BorderStyle.Dotted;
  2475. case "MEDIUMDASHED":
  2476. case "MEDIUM_DASHED":
  2477. return BorderStyle.MediumDashed;
  2478. case "DASHDOT":
  2479. case "DASH_DOT":
  2480. return BorderStyle.DashDot;
  2481. case "MEDIUMDASHDOT":
  2482. case "MEDIUM_DASH_DOT":
  2483. return BorderStyle.MediumDashDot;
  2484. case "DASHDOTDOT":
  2485. case "DASH_DOT_DOT":
  2486. return BorderStyle.DashDotDot;
  2487. case "MEDIUMDASHDOTDOT":
  2488. case "MEDIUM_DASH_DOT_DOT":
  2489. return BorderStyle.MediumDashDotDot;
  2490. case "SLANTEDDASHDOT":
  2491. case "SLANTED_DASH_DOT":
  2492. return BorderStyle.SlantedDashDot;
  2493. default:
  2494. return BorderStyle.None;
  2495. }
  2496. }
  2497. #endregion 样式转换
  2498. #endregion 解析css样式
  2499. }
  2500. public enum ColorType
  2501. {
  2502. Black = HSSFColor.Black.Index,
  2503. Brown = HSSFColor.Brown.Index,
  2504. OliveGreen = HSSFColor.OliveGreen.Index,
  2505. DarkGreen = HSSFColor.DarkGreen.Index,
  2506. DarkTeal = HSSFColor.DarkTeal.Index,
  2507. DarkBlue = HSSFColor.DarkBlue.Index,
  2508. Indigo = HSSFColor.Indigo.Index,
  2509. Grey80Percent = HSSFColor.Grey80Percent.Index,
  2510. Orange = HSSFColor.Orange.Index,
  2511. DarkYellow = HSSFColor.DarkYellow.Index,
  2512. Green = HSSFColor.Green.Index,
  2513. Teal = HSSFColor.Teal.Index,
  2514. Blue = HSSFColor.Blue.Index,
  2515. BlueGrey = HSSFColor.BlueGrey.Index,
  2516. Grey50Percent = HSSFColor.Grey50Percent.Index,
  2517. Red = HSSFColor.Red.Index,
  2518. LightOrange = HSSFColor.LightOrange.Index,
  2519. Lime = HSSFColor.Lime.Index,
  2520. SeaGreen = HSSFColor.SeaGreen.Index,
  2521. Aqua = HSSFColor.Aqua.Index,
  2522. LightBlue = HSSFColor.LightBlue.Index,
  2523. Violet = HSSFColor.Violet.Index,
  2524. Grey40Percent = HSSFColor.Grey40Percent.Index,
  2525. Pink = HSSFColor.Pink.Index,
  2526. Gold = HSSFColor.Gold.Index,
  2527. Yellow = HSSFColor.Yellow.Index,
  2528. BrightGreen = HSSFColor.BrightGreen.Index,
  2529. Turquoise = HSSFColor.Turquoise.Index,
  2530. DarkRed = HSSFColor.DarkRed.Index,
  2531. SkyBlue = HSSFColor.SkyBlue.Index,
  2532. Plum = HSSFColor.Plum.Index,
  2533. Grey25Percent = HSSFColor.Grey25Percent.Index,
  2534. Rose = HSSFColor.Rose.Index,
  2535. LightYellow = HSSFColor.LightYellow.Index,
  2536. LightGreen = HSSFColor.LightGreen.Index,
  2537. LightTurquoise = HSSFColor.LightTurquoise.Index,
  2538. PaleBlue = HSSFColor.PaleBlue.Index,
  2539. Lavender = HSSFColor.Lavender.Index,
  2540. White = HSSFColor.White.Index,
  2541. CornflowerBlue = HSSFColor.CornflowerBlue.Index,
  2542. LemonChiffon = HSSFColor.LemonChiffon.Index,
  2543. Maroon = HSSFColor.Maroon.Index,
  2544. Orchid = HSSFColor.Orchid.Index,
  2545. Coral = HSSFColor.Coral.Index,
  2546. RoyalBlue = HSSFColor.RoyalBlue.Index,
  2547. LightCornflowerBlue = HSSFColor.LightCornflowerBlue.Index,
  2548. Tan = HSSFColor.Tan.Index,
  2549. Automatic = HSSFColor.Automatic.Index
  2550. }
  2551. public static class TransExp<TIn, TOut>
  2552. {
  2553. private static readonly Func<TIn, TOut> Cache = GetFunc();
  2554. private static Func<TIn, TOut> GetFunc()
  2555. {
  2556. ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
  2557. List<MemberBinding> memberBindingList = new List<MemberBinding>();
  2558. foreach (var item in typeof(TOut).GetProperties())
  2559. {
  2560. if (!item.CanWrite)
  2561. continue;
  2562. MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name) ?? throw new InvalidOperationException());
  2563. MemberBinding memberBinding = Expression.Bind(item, property);
  2564. memberBindingList.Add(memberBinding);
  2565. }
  2566. MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
  2567. Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[] { parameterExpression });
  2568. return lambda.Compile();
  2569. }
  2570. public static TOut Trans(TIn tIn)
  2571. {
  2572. return Cache(tIn);
  2573. }
  2574. }
  2575. }