using DocumentFormat.OpenXml.Packaging; using DocumentFormat.OpenXml.Spreadsheet; using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; using System.Text; using System.Text.RegularExpressions; using System.Threading.Tasks; using System.Windows.Media; namespace Tango.Documents { public class ExcelWriter : IDisposable { private Stream _stream; private SpreadsheetDocument _document; private List _column_names; private Worksheet _currentWorkSheet; private Stylesheet _currentStyleSheet; public ExcelWriter(Stream source) { _stream = source; _document = SpreadsheetDocument.Open(_stream, true); _column_names = new List(); if (_document.WorkbookPart.Workbook.GetFirstChild() != null) { foreach (DefinedName name in _document.WorkbookPart.Workbook.GetFirstChild()) { _column_names.Add(name); } } } public ExcelWriter(String fileName) : this(new FileStream(fileName, FileMode.OpenOrCreate, FileAccess.ReadWrite)) { } public void WriteData(IEnumerable data, String sheetName, int? headerRowNumber = null) { var props = typeof(T).GetProperties(BindingFlags.Public | BindingFlags.Instance).ToList(); WorksheetPart work_sheet_part = GetWorkSheetPart(sheetName); Worksheet workSheet = work_sheet_part.Worksheet; _currentWorkSheet = workSheet; SheetData sheetData = workSheet.GetFirstChild(); List rows = sheetData.Elements().ToList(); var list = data.ToList(); int lastRowIndex = headerRowNumber.HasValue ? headerRowNumber.Value : 0; for (int i = 0; i < list.Count; i++) { var item = list[i]; Row last_row = null; if (headerRowNumber.HasValue) { last_row = sheetData.Elements().ElementAt(lastRowIndex - 1); } else { last_row = sheetData.Elements().Last(); } Row newRow = new Row(); newRow.RowIndex = last_row.RowIndex + 1; lastRowIndex = (int)newRow.RowIndex.Value; sheetData.InsertAfter(newRow, last_row); foreach (var prop in props) { var definedName = _column_names.SingleOrDefault(x => x.Name == prop.Name); if (definedName != null) { SetCellRow(newRow, GetCellReference(definedName, (int)(last_row.RowIndex + 1)), prop.GetValue(item)); } } } if (_currentStyleSheet != null) { _currentStyleSheet.Save(); _currentStyleSheet = null; } } private void SetCellRow(Row row, string cellReference, object value) { Cell cell = row.Descendants().FirstOrDefault(c => c.CellReference == cellReference); bool append = false; if (cell == null) { cell = new Cell(); cell.CellReference = cellReference; append = true; } cell.CellValue = new CellValue(value.ToString()); cell.DataType = CellValues.Number; if (value is String) { cell.DataType = CellValues.String; } else if (value is DateTime) { cell.DataType = CellValues.Date; } else if (value is Boolean) { cell.DataType = CellValues.Boolean; } else if (value is System.Windows.Media.Color) { cell.DataType = CellValues.String; var color = (System.Windows.Media.Color)value; if (_currentStyleSheet == null) { WorkbookPart wbPart = _document.WorkbookPart; WorkbookStylesPart stylesPart = wbPart.WorkbookStylesPart; _currentStyleSheet = stylesPart.Stylesheet; } var fill = new Fill(new PatternFill( new ForegroundColor() { Rgb = new DocumentFormat.OpenXml.HexBinaryValue() { Value = new ColorConverter().ConvertToString(color).Replace("#", ""), }, }, new BackgroundColor() { Rgb = new DocumentFormat.OpenXml.HexBinaryValue() { Value = new ColorConverter().ConvertToString(color).Replace("#", ""), }, }) { PatternType = PatternValues.Solid, }); if (_currentStyleSheet.Fills == null) { _currentStyleSheet.Fills = new Fills(); } _currentStyleSheet.Fills.Append(fill); _currentStyleSheet.CellFormats.Append(new CellFormat() { ApplyFill = true, FillId = _currentStyleSheet.Fills.Count, NumberFormatId = (DocumentFormat.OpenXml.UInt32Value)0U, FontId = (DocumentFormat.OpenXml.UInt32Value)0U, BorderId = (DocumentFormat.OpenXml.UInt32Value)0U, FormatId = (DocumentFormat.OpenXml.UInt32Value)0U, }); cell.CellValue = new CellValue(String.Empty); cell.StyleIndex = new DocumentFormat.OpenXml.UInt32Value(_currentStyleSheet.CellFormats.Count); _currentStyleSheet.CellFormats.Count++; _currentStyleSheet.Fills.Count++; } if (append) { row.Append(cell); } } private WorksheetPart GetWorkSheetPart(String sheetName) { var sheet = _document.WorkbookPart.Workbook.Descendants().FirstOrDefault(x => x.Name.Value == sheetName); string relId = sheet.Id; return (WorksheetPart)_document.WorkbookPart.GetPartById(relId); } private string GetCellReference(DefinedName name, int rowIndex) { Regex rowPatern = new Regex("^.*\\!\\$(.*)\\$\\d*$"); Match match = rowPatern.Match(name.Text); String s = match.Groups[1].Value + rowIndex.ToString(); return s; } /// /// In case a sheet has a table with unknown rows update the size of the table . For example newRef="A1:B328" where 328 is count of rows table included header. /// /// Name of the sheet. /// The new reference. public void UpdateTableSize( string sheetName, string newRef) { WorksheetPart work_sheet_part = GetWorkSheetPart(sheetName); if (work_sheet_part.TableDefinitionParts != null) { TableDefinitionPart tableDefinitionPart = work_sheet_part.TableDefinitionParts.FirstOrDefault(); if (tableDefinitionPart == null) return; Table excelTable = tableDefinitionPart.Table; if(excelTable != null) { excelTable.Reference = newRef; } } } public void Dispose() { _document.Save(); _stream.Dispose(); } } }