using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.IO;
using System.Xml.Linq;
using DocumentFormat.OpenXml.Spreadsheet;
using DocumentFormat.OpenXml;
using System.Text.RegularExpressions;
using System.Data;
namespace IronIntel.Contractor
{
internal class ChartFormatedData : IOpenXmlExcelStyleAndData
{
public ChartFormatedData()
{
FormatType = CellFormatType.CSharp;
}
#region IOpenXmlExcelStyleAndData Members
public CellAlignment CAlignment { get; set; }
public CellBorder CBorder { get; set; }
public CellFill CFill { get; set; }
public CellFont CFont { get; set; }
public CellDataType DataType { get; set; }
public string FormatCode { get; set; }
public CellFormatType FormatType { get; set; }
public object Value { get; set; }
public String MergedCellsPosition { get; set; }
#endregion
}
public class ExportToExcel
{
///
/// 将DataTable的数据导出到Excel
///
///
public byte[] CreateExcel(DataTable data, string caption, double[] columnWidths, string[] MergeTitles)
{
COpenXmlExcelSheet osheet = ConvertToOpenXmlObject(data, caption, columnWidths, MergeTitles);
MemoryStream ms = null;
try
{
ms = new MemoryStream();
ExcelXlsx xls = new ExcelXlsx();
xls.CreatePackage(ms, osheet);
ms.Position = 0;
byte[] bts = new byte[ms.Length];
int offset = 0;
while (offset < bts.Length)
{
offset += ms.Read(bts, offset, bts.Length - offset);
}
return bts;
}
catch { }
finally
{
if (ms != null)
ms.Close();
}
return null;
}
private object ConvertIvalidChars(object s)
{
if (s == null) return null;
if (s.GetType() != typeof(string)) return s;
//const string invalidCharsMatch =
// "(?ims)[\x0\x1\x2\x3\x4\x5\x6\x7\x8\x9\xa\xb\xc\xd\xe\xf" +
// "\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f\x7f]";
const string invalidCharsMatch =
"(?ims)[\x0\x1\x2\x3\x4\x5\x6\x7\x8\x9\xb\xc\xe\xf" +
"\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f\x7f]";
//取代其中無效字元, 通通換成空字串
s = Regex.Replace(
s.ToString(),
invalidCharsMatch, "");
return s;
}
private COpenXmlExcelSheet ConvertToOpenXmlObject(DataTable data, string caption, double[] columnWidths, string[] MergeTitles)
{
try
{
COpenXmlExcelSheet osheet = new COpenXmlExcelSheet();
// 设置数据和格式。
//所有数据和格式存放在此结构中。
List> dataMatrix = new List>();
osheet.DataMatrix = dataMatrix;
//行数据临时对象。
List rowData = null;
//单元格数据临时对象。
ChartFormatedData cellData = null;
DataRow rdr = null;
DataColumn rdc = null;
#region 设置宽度。
foreach (var w in columnWidths)
{
osheet.WidthList.Add(w);
}
//for (int k = 0; k < dataFromClient.Columns.Count; k++)
//{
// try
// {
// rdc = dataFromClient.Columns[k];
// if (rdc.Attributes == null || !rdc.Attributes.ContainsKey("Width"))
// {
// osheet.WidthList.Add(100);
// }
// else
// {
// osheet.WidthList.Add(Convert.ToDouble(rdc.Attributes["Width"]));
// }
// }
// catch
// {
// }
//}
#endregion
int rowIndex = 0;
#region caption
if (!string.IsNullOrEmpty(caption))
{
rowData = new List();
cellData = new ChartFormatedData();
cellData.CAlignment = new CellAlignment();
cellData.CAlignment.Align = GetAlignment("center");
cellData.DataType = CellDataType.String;
cellData.FormatCode = "";
cellData.CFont = new CellFont();
cellData.CFont.FontSize = 14;
cellData.Value = caption;
cellData.MergedCellsPosition = "A1:" + ConvertColNumber(data.Columns.Count) + "1";
rowData.Add(cellData);
rowIndex += 1;
dataMatrix.Add(rowData);
//设置第一行的高度。
osheet.RowHeightList.Add(rowIndex, 23);
//添加一个空行。
rowData = new List();
cellData = new ChartFormatedData();
cellData.MergedCellsPosition = "A2:" + ConvertColNumber(data.Columns.Count) + "2";
rowData.Add(cellData);
rowIndex += 1;
dataMatrix.Add(rowData);
}
#endregion
#region MergeTitles
if (MergeTitles != null && MergeTitles.Length % 2 == 0)
{
rowData = new List();
for (int q = 0; q < data.Columns.Count; q++)
{
cellData = new ChartFormatedData();
cellData.CAlignment = new CellAlignment();
cellData.CAlignment.Align = GetAlignment("center");
cellData.DataType = CellDataType.String;
cellData.FormatCode = "";
cellData.CFont = new CellFont();
cellData.CFont.FontSize = 14;
rowData.Add(cellData);
}
for (int i = 0; i < MergeTitles.Length; i += 2)
{
string[] tmp = MergeTitles[i + 1].Split('-');
if (tmp.Length == 1) continue;
cellData = (ChartFormatedData)rowData[(Convert.ToInt32(tmp[0]) - 1)];
cellData.Value = MergeTitles[i];
cellData.MergedCellsPosition = ConvertColNumber(Convert.ToInt32(tmp[0])) + "3:" + ConvertColNumber(Convert.ToInt32(tmp[1])) + "3";
}
rowIndex += 1;
dataMatrix.Add(rowData);
//设置第一行的高度。
osheet.RowHeightList.Add(rowIndex, 15);
}
#endregion
#region header。
rowData = new List();
for (int q = 0; q < data.Columns.Count; q++)
{
rdc = data.Columns[q];
cellData = new ChartFormatedData();
string alignment = "";
if (rdc != null)
{
//if (rdc.Attributes != null && rdc.Attributes.ContainsKey("Alignment"))
//{
// alignment = rdc.Attributes["Alignment"];
//}
cellData.CAlignment = new CellAlignment();
cellData.CAlignment.Align = GetAlignment(alignment);
cellData.CFont = new CellFont();
cellData.CFont.IsBold = true;
cellData.CFill = new CellFill();
cellData.CFill.FillColor = System.Drawing.Color.Gray;
cellData.DataType = CellDataType.String;
cellData.FormatCode = "";
}
cellData.Value = rdc.Caption;
rowData.Add(cellData);
}
rowIndex += 1;
osheet.RowHeightList.Add(rowIndex, 18);
dataMatrix.Add(rowData);
#endregion
#region real data 。
for (int k = 0; k <= data.Rows.Count - 1; k++)
{
rdr = data.Rows[k];
rowData = new List();
for (int q = 0; q < data.Columns.Count; q++)
{
rdc = data.Columns[q];
string format = "";
if (rdc != null)
{
if (rdc.ExtendedProperties != null && rdc.ExtendedProperties.ContainsKey("DataFormat"))
{
format = (rdc.ExtendedProperties["DataFormat"] == null
? ""
: rdc.ExtendedProperties["DataFormat"].ToString());
}
}
cellData = new ChartFormatedData();
//将特殊格式值处理成字符串显示。
CExcelCellValue cExcelValue = null;
if (rdr != null)
{
bool isProcessed = false;
object cValue = rdr[q];
if (cValue is CExcelCellValue)
{
cExcelValue = (cValue as CExcelCellValue);
cValue = cExcelValue.Value;
}
cellData.Value =
ConvertIvalidChars(
ProcessSpecialFormat(cValue, format, ref isProcessed));
if (isProcessed) format = "";
}
cellData.FormatCode = ProcessFormat(format);
cellData.DataType =
GetDataType((cellData.Value == null ? typeof(string) : cellData.Value.GetType()));
//cellData.FormatCode = ProcessFormat(format, cellData.DataType, cellData.Value);
//if ((cellData.DataType == CellDataType.Integer || cellData.DataType == CellDataType.Float) && cellData.FormatCode == "{0}")
//{
// try
// {
// var d = decimal.Parse(cellData.Value.ToString());
// var pad = (decimal.GetBits(d)[3] >> 16) & 0x7fff;
// if (pad > 0)
// {
// cellData.FormatCode = "{0:0." + new string('0', pad) + "}";
// }
// }
// catch { }
//}
string alignment = "";
if (rdc != null)
{
//if (rdc.Attributes != null && rdc.Attributes.ContainsKey("Alignment"))
//{
// alignment = rdc.Attributes["Alignment"];
//}
cellData.CAlignment = new CellAlignment();
cellData.CAlignment.Align = GetAlignment(alignment);
}
if (cExcelValue != null)
{
if (cExcelValue.BackgroundColor != null)
{
cellData.CFill = new CellFill();
cellData.CFill.FillColor = cExcelValue.BackgroundColor.Value;
}
}
//如果是合计行则以浅灰色显示。
//if (hasTotalRow && k == dataFromClient.Rows.Count - 1)
//{
// cellData.CFill = new CellFill();
// cellData.CFill.FillColor = System.Drawing.Color.LightGray;
//}
//如果是合计列则以浅灰色显示。
//if (hasTotalColumn && q == dataFromClient.Columns.Count - 1)
//{
// cellData.CFill = new CellFill();
// cellData.CFill.FillColor = System.Drawing.Color.LightGray;
//}
rowData.Add(cellData);
}
//rowIndex += 1;
//if (hasTotalRow && k == dataFromClient.Rows.Count - 1)
//{
// osheet.RowHeightList.Add(rowIndex, totalRowHeight);
//}
//else
//{
// osheet.RowHeightList.Add(rowIndex, 18);
//}
dataMatrix.Add(rowData);
}
#endregion
return osheet;
}
catch (Exception ex)
{
throw ex;
}
}
private string ProcessFormat(string format, CellDataType cellDataType, object value)
{
string resultFormat = format;
if (string.IsNullOrEmpty(resultFormat)) return resultFormat;
if (format.IndexOf("*") >= 0)
{
resultFormat = resultFormat.Replace("*", "\"*\"");
}
if (format.IndexOf("@") >= 0)
{
resultFormat = resultFormat.Replace("@", "\"@\"");
}
if (cellDataType == CellDataType.Integer && Regex.IsMatch(resultFormat, @"[.][#]+"))
{
resultFormat = Regex.Replace(resultFormat, @"[.][#]+", "");
}
else if (cellDataType == CellDataType.Float)
{
try
{
var s = string.Format(resultFormat, value);
if (s.IndexOf('.') < 0)
{
resultFormat = Regex.Replace(resultFormat, @"[.][0#]*(.*)[}]", "$1}");
}
}
catch { }
}
return resultFormat;
}
private string ConvertColNumber(int colnum)
{
string zzz = "Z+";
Regex reg = new Regex(zzz);
string result = "";
MatchCollection mc = null;
for (int k = 0; k < colnum; k++)
{
mc = reg.Matches(result);
if (mc.Count > 0)
{
//是zzz格式。
string first = result.Substring(0, mc[0].Index);
if (string.IsNullOrEmpty(first))
{
result = result.Replace("Z", "A") + "A";
}
else
{
char c = first[first.Length - 1];
c = Convert.ToChar(c + 1);
result = c.ToString() + result.Substring(mc[0].Index).Replace("Z", "A");
}
}
else
{
if (string.IsNullOrEmpty(result))
{
result = "A";
}
else
{
char c = result[result.Length - 1];
c = Convert.ToChar(c + 1);
result = result.Substring(0, result.Length - 1) + c.ToString();
}
}
}
return result;
}
private Alignment GetAlignment(string align)
{
Alignment horizon = new Alignment() { Horizontal = HorizontalAlignmentValues.Left, WrapText = true };
if (string.Equals(align, "center", StringComparison.OrdinalIgnoreCase))
{
horizon = new Alignment() { Horizontal = HorizontalAlignmentValues.Center, WrapText = true };
}
else if (string.Equals(align, "right", StringComparison.OrdinalIgnoreCase))
{
horizon = new Alignment() { Horizontal = HorizontalAlignmentValues.Right, WrapText = true };
}
return horizon;
}
private CellDataType GetDataType(Type typ)
{
CellDataType result = CellDataType.String;
switch (Type.GetTypeCode(typ))
{
case TypeCode.Int16:
case TypeCode.Int32:
case TypeCode.Int64:
case TypeCode.UInt16:
case TypeCode.UInt32:
case TypeCode.UInt64:
result = CellDataType.Integer;
break;
case TypeCode.Decimal:
case TypeCode.Double:
case TypeCode.Single:
result = CellDataType.Float;
break;
case TypeCode.DateTime:
result = CellDataType.Date;
break;
case TypeCode.Boolean:
result = CellDataType.Bool;
break;
default:
break;
}
return result;
}
private object ProcessSpecialFormat(object val, string format, ref bool isProcessed)
{
object result = val;
isProcessed = false;
if (val == null || string.IsNullOrEmpty(format.Trim())) return result;
CellDataType typ = GetDataType(result.GetType());
//第一个特殊格式:如果值是6位字符串,格式是"MMM-yyyy",则按日期处理。
if (typ == CellDataType.String && string.Equals(format.Replace(" ", ""), "{0:MMM-yyyy}"))
{
string str = result.ToString();
if (str.Length == 6)
{
try
{
result = new DateTime(Convert.ToInt32(str.Substring(0, 4)), Convert.ToInt32(str.Substring(4)), 1);
result = string.Format(format, result);
isProcessed = true;
}
catch { }
}
}
return result;
}
///
/// 处理Format格式,使一些特殊符号(如:*、@)可以直接在Excel中看到,而不是解释成Excel中对这些符号的定义
///
/// 格式字符串
///
private string ProcessFormat(string format)
{
string resultFormat = format;
if (string.IsNullOrEmpty(resultFormat)) return resultFormat;
if (format.IndexOf("*") >= 0)
{
resultFormat = resultFormat.Replace("*", "\"*\"");
}
if (format.IndexOf("@") >= 0)
{
resultFormat = resultFormat.Replace("@", "\"@\"");
}
if (format.IndexOf("\n") >= 0)
{
resultFormat = resultFormat.Replace("@", "\"@\"");
}
return resultFormat;
}
}
}