| /******************************************************************************* | 
 |  * You may amend and distribute as you like, but don't remove this header! | 
 |  * | 
 |  * EPPlus provides server-side generation of Excel 2007/2010 spreadsheets. | 
 |  * See http://www.codeplex.com/EPPlus for details. | 
 |  * | 
 |  * Copyright (C) 2011  Jan Källman | 
 |  * | 
 |  * This library is free software; you can redistribute it and/or | 
 |  * modify it under the terms of the GNU Lesser General Public | 
 |  * License as published by the Free Software Foundation; either | 
 |  * version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |  * This library is distributed in the hope that it will be useful, | 
 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.   | 
 |  * See the GNU Lesser General Public License for more details. | 
 |  * | 
 |  * The GNU Lesser General Public License can be viewed at http://www.opensource.org/licenses/lgpl-license.php | 
 |  * If you unfamiliar with this license or have questions about it, here is an http://www.gnu.org/licenses/gpl-faq.html | 
 |  * | 
 |  * All code and executables are provided "as is" with no warranty either express or implied.  | 
 |  * The author accepts no liability for any damage or loss of business that this product may cause. | 
 |  * | 
 |  * Code change notes: | 
 |  *  | 
 |  * Author							Change						Date | 
 |  * ****************************************************************************** | 
 |  * Jan Källman		Added		30-AUG-2010 | 
 |  * Jan Källman		License changed GPL-->LGPL 2011-12-16 | 
 |  *******************************************************************************/ | 
 | using System; | 
 | using System.Collections.Generic; | 
 | using System.Text; | 
 | using System.Xml; | 
 | using System.Text.RegularExpressions; | 
 | using OfficeOpenXml.FormulaParsing.Excel.Functions.Math; | 
 | using OfficeOpenXml.Utils; | 
 |  | 
 | namespace OfficeOpenXml.Table | 
 | { | 
 |     /// <summary> | 
 |     /// Table style Enum | 
 |     /// </summary> | 
 |     public enum TableStyles | 
 |     { | 
 |         None, | 
 |         Custom, | 
 |         Light1, | 
 |         Light2, | 
 |         Light3, | 
 |         Light4, | 
 |         Light5, | 
 |         Light6, | 
 |         Light7, | 
 |         Light8, | 
 |         Light9, | 
 |         Light10, | 
 |         Light11, | 
 |         Light12, | 
 |         Light13, | 
 |         Light14, | 
 |         Light15, | 
 |         Light16, | 
 |         Light17, | 
 |         Light18, | 
 |         Light19, | 
 |         Light20, | 
 |         Light21, | 
 |         Medium1, | 
 |         Medium2, | 
 |         Medium3, | 
 |         Medium4, | 
 |         Medium5, | 
 |         Medium6, | 
 |         Medium7, | 
 |         Medium8, | 
 |         Medium9, | 
 |         Medium10, | 
 |         Medium11, | 
 |         Medium12, | 
 |         Medium13, | 
 |         Medium14, | 
 |         Medium15, | 
 |         Medium16, | 
 |         Medium17, | 
 |         Medium18, | 
 |         Medium19, | 
 |         Medium20, | 
 |         Medium21, | 
 |         Medium22, | 
 |         Medium23, | 
 |         Medium24, | 
 |         Medium25, | 
 |         Medium26, | 
 |         Medium27, | 
 |         Medium28, | 
 |         Dark1, | 
 |         Dark2, | 
 |         Dark3, | 
 |         Dark4, | 
 |         Dark5, | 
 |         Dark6, | 
 |         Dark7, | 
 |         Dark8, | 
 |         Dark9, | 
 |         Dark10, | 
 |         Dark11, | 
 |     } | 
 |     /// <summary> | 
 |     /// An Excel Table | 
 |     /// </summary> | 
 |     public class ExcelTable : XmlHelper, IEqualityComparer<ExcelTable> | 
 |     { | 
 |         internal ExcelTable(Packaging.ZipPackageRelationship rel, ExcelWorksheet sheet) :  | 
 |             base(sheet.NameSpaceManager) | 
 |         { | 
 |             WorkSheet = sheet; | 
 |             TableUri = UriHelper.ResolvePartUri(rel.SourceUri, rel.TargetUri); | 
 |             RelationshipID = rel.Id; | 
 |             var pck = sheet._package.Package; | 
 |             Part=pck.GetPart(TableUri); | 
 |  | 
 |             TableXml = new XmlDocument(); | 
 |             LoadXmlSafe(TableXml, Part.GetStream()); | 
 |             init(); | 
 |             Address = new ExcelAddressBase(GetXmlNodeString("@ref")); | 
 |         } | 
 |         internal ExcelTable(ExcelWorksheet sheet, ExcelAddressBase address, string name, int tblId) :  | 
 |             base(sheet.NameSpaceManager) | 
 | 	    { | 
 |             WorkSheet = sheet; | 
 |             Address = address; | 
 |             TableXml = new XmlDocument(); | 
 |             LoadXmlSafe(TableXml, GetStartXml(name, tblId), Encoding.UTF8);  | 
 |             TopNode = TableXml.DocumentElement; | 
 |  | 
 |             init(); | 
 |  | 
 |             //If the table is just one row we can not have a header. | 
 |             if (address._fromRow == address._toRow) | 
 |             { | 
 |                 ShowHeader = false; | 
 |             } | 
 |         } | 
 |  | 
 |         private void init() | 
 |         { | 
 |             TopNode = TableXml.DocumentElement; | 
 |             SchemaNodeOrder = new string[] { "autoFilter", "tableColumns", "tableStyleInfo" }; | 
 |         } | 
 |         private string GetStartXml(string name, int tblId) | 
 |         { | 
 |             string xml = "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\" ?>"; | 
 |             xml += string.Format("<table xmlns=\"http://schemas.openxmlformats.org/spreadsheetml/2006/main\" id=\"{0}\" name=\"{1}\" displayName=\"{2}\" ref=\"{3}\" headerRowCount=\"1\">", | 
 |             tblId, | 
 |             name, | 
 |             cleanDisplayName(name), | 
 |             Address.Address); | 
 |             xml += string.Format("<autoFilter ref=\"{0}\" />", Address.Address); | 
 |  | 
 |             int cols=Address._toCol-Address._fromCol+1; | 
 |             xml += string.Format("<tableColumns count=\"{0}\">",cols); | 
 |             var names = new Dictionary<string, string>();             | 
 |             for(int i=1;i<=cols;i++) | 
 |             { | 
 |                 var cell = WorkSheet.Cells[Address._fromRow, Address._fromCol+i-1]; | 
 |                 string colName; | 
 |                 if (cell.Value == null || names.ContainsKey(cell.Value.ToString())) | 
 |                 { | 
 |                     //Get an unique name | 
 |                     int a=i; | 
 |                     do | 
 |                     { | 
 |                         colName = string.Format("Column{0}", a++); | 
 |                     } | 
 |                     while (names.ContainsKey(colName)); | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     colName = System.Security.SecurityElement.Escape(cell.Value.ToString()); | 
 |                 } | 
 |                 names.Add(colName, colName); | 
 |                 xml += string.Format("<tableColumn id=\"{0}\" name=\"{1}\" />", i,colName); | 
 |             } | 
 |             xml += "</tableColumns>"; | 
 |             xml += "<tableStyleInfo name=\"TableStyleMedium9\" showFirstColumn=\"0\" showLastColumn=\"0\" showRowStripes=\"1\" showColumnStripes=\"0\" /> "; | 
 |             xml += "</table>"; | 
 |  | 
 |             return xml; | 
 |         } | 
 |         private string cleanDisplayName(string name)  | 
 |         { | 
 |             return Regex.Replace(name, @"[^\w\.-_]", "_"); | 
 |         } | 
 |         internal Packaging.ZipPackagePart Part | 
 |         { | 
 |             get; | 
 |             set; | 
 |         } | 
 |         /// <summary> | 
 |         /// Provides access to the XML data representing the table in the package. | 
 |         /// </summary> | 
 |         public XmlDocument TableXml | 
 |         { | 
 |             get; | 
 |             set; | 
 |         } | 
 |         /// <summary> | 
 |         /// The package internal URI to the Table Xml Document. | 
 |         /// </summary> | 
 |         public Uri TableUri | 
 |         { | 
 |             get; | 
 |             internal set; | 
 |         } | 
 |         internal string RelationshipID | 
 |         { | 
 |             get; | 
 |             set; | 
 |         } | 
 |         const string ID_PATH = "@id"; | 
 |         internal int Id  | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeInt(ID_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeString(ID_PATH, value.ToString()); | 
 |             } | 
 |         } | 
 |         const string NAME_PATH = "@name"; | 
 |         const string DISPLAY_NAME_PATH = "@displayName"; | 
 |         /// <summary> | 
 |         /// The name of the table object in Excel | 
 |         /// </summary> | 
 |         public string Name | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeString(NAME_PATH); | 
 |             } | 
 |             set  | 
 |             { | 
 |                 if(WorkSheet.Workbook.ExistsTableName(value)) | 
 |                 { | 
 |                     throw (new ArgumentException("Tablename is not unique")); | 
 |                 } | 
 |                 string prevName = Name; | 
 |                 if (WorkSheet.Tables._tableNames.ContainsKey(prevName)) | 
 |                 { | 
 |                     int ix=WorkSheet.Tables._tableNames[prevName]; | 
 |                     WorkSheet.Tables._tableNames.Remove(prevName); | 
 |                     WorkSheet.Tables._tableNames.Add(value,ix); | 
 |                 } | 
 |                 SetXmlNodeString(NAME_PATH, value); | 
 |                 SetXmlNodeString(DISPLAY_NAME_PATH, cleanDisplayName(value)); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// The worksheet of the table | 
 |         /// </summary> | 
 |         public ExcelWorksheet WorkSheet | 
 |         { | 
 |             get; | 
 |             set; | 
 |         } | 
 |  | 
 |         private ExcelAddressBase _address = null; | 
 |         /// <summary> | 
 |         /// The address of the table | 
 |         /// </summary> | 
 |         public ExcelAddressBase Address | 
 |         { | 
 |             get | 
 |             { | 
 |                 return _address; | 
 |             } | 
 |             set | 
 |             { | 
 |                 _address = value; | 
 |                 SetXmlNodeString("@ref",value.Address); | 
 |                 WriteAutoFilter(ShowTotal); | 
 |             } | 
 |         } | 
 |         internal ExcelTableColumnCollection _cols = null; | 
 |         /// <summary> | 
 |         /// Collection of the columns in the table | 
 |         /// </summary> | 
 |         public ExcelTableColumnCollection Columns | 
 |         { | 
 |             get | 
 |             { | 
 |                 if(_cols==null) | 
 |                 { | 
 |                     _cols = new ExcelTableColumnCollection(this); | 
 |                 } | 
 |                 return _cols; | 
 |             } | 
 |         } | 
 |         TableStyles _tableStyle = TableStyles.Medium6; | 
 |         /// <summary> | 
 |         /// The table style. If this property is cusom, the style from the StyleName propery is used. | 
 |         /// </summary> | 
 |         public TableStyles TableStyle | 
 |         { | 
 |             get | 
 |             { | 
 |                 return _tableStyle; | 
 |             } | 
 |             set | 
 |             { | 
 |                 _tableStyle=value; | 
 |                 if (value != TableStyles.Custom) | 
 |                 { | 
 |                     SetXmlNodeString(STYLENAME_PATH, "TableStyle" + value.ToString()); | 
 |                 } | 
 |             } | 
 |         } | 
 |         const string HEADERROWCOUNT_PATH = "@headerRowCount"; | 
 |         const string AUTOFILTER_PATH = "d:autoFilter/@ref"; | 
 |         /// <summary> | 
 |         /// If the header row is visible or not | 
 |         /// </summary> | 
 |         public bool ShowHeader | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeInt(HEADERROWCOUNT_PATH)!=0; | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (Address._toRow - Address._fromRow < 0 && value || | 
 |                     Address._toRow - Address._fromRow == 1 && value && ShowTotal) | 
 |                 { | 
 |                     throw (new Exception("Cant set ShowHeader-property. Table has too few rows")); | 
 |                 } | 
 |  | 
 |                 if(value) | 
 |                 { | 
 |                     DeleteNode(HEADERROWCOUNT_PATH); | 
 |                     WriteAutoFilter(ShowTotal); | 
 |                     //for (int i = 0; i < Columns.Count; i++) | 
 |                     //{ | 
 |                     //    var v = WorkSheet.GetValue<string>(Address._fromRow, Address._fromCol + i); | 
 |                     //    if (!string.IsNullOrEmpty(v) || v != _cols[i].Name) | 
 |                     //    { | 
 |                     //        _cols[i].Name = v; | 
 |                     //    } | 
 |                     //} | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     SetXmlNodeString(HEADERROWCOUNT_PATH, "0"); | 
 |                     DeleteAllNode(AUTOFILTER_PATH); | 
 |                 } | 
 |             } | 
 |         } | 
 |         internal ExcelAddressBase AutoFilterAddress | 
 |         { | 
 |             get | 
 |             { | 
 |                 string a=GetXmlNodeString(AUTOFILTER_PATH); | 
 |                 if (a == "") | 
 |                 { | 
 |                     return null; | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     return new ExcelAddressBase(a); | 
 |                 } | 
 |             } | 
 |         } | 
 |         private void WriteAutoFilter(bool showTotal) | 
 |         { | 
 |             string autofilterAddress; | 
 |             if (ShowHeader) | 
 |             { | 
 |                 if (showTotal) | 
 |                 { | 
 |                     autofilterAddress = ExcelCellBase.GetAddress(Address._fromRow, Address._fromCol, Address._toRow - 1, Address._toCol); | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     autofilterAddress = Address.Address; | 
 |                 } | 
 |                 SetXmlNodeString(AUTOFILTER_PATH, autofilterAddress); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// If the header row has an autofilter | 
 |         /// </summary> | 
 |         public bool ShowFilter  | 
 |         {  | 
 |             get | 
 |             { | 
 |                 return ShowHeader && AutoFilterAddress != null; | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (ShowHeader) | 
 |                 { | 
 |                     if (value) | 
 |                     { | 
 |                         WriteAutoFilter(ShowTotal); | 
 |                     } | 
 |                     else  | 
 |                     { | 
 |                         DeleteAllNode(AUTOFILTER_PATH); | 
 |                     } | 
 |                 } | 
 |                 else if(value) | 
 |                 { | 
 |                     throw(new InvalidOperationException("Filter can only be applied when ShowHeader is set to true")); | 
 |                 } | 
 |             } | 
 |         } | 
 |         const string TOTALSROWCOUNT_PATH = "@totalsRowCount"; | 
 |         const string TOTALSROWSHOWN_PATH = "@totalsRowShown"; | 
 |         /// <summary> | 
 |         /// If the total row is visible or not | 
 |         /// </summary> | 
 |         public bool ShowTotal | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeInt(TOTALSROWCOUNT_PATH) == 1; | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (value != ShowTotal) | 
 |                 { | 
 |                     if (value) | 
 |                     { | 
 |                         Address=new ExcelAddress(WorkSheet.Name, ExcelAddressBase.GetAddress(Address.Start.Row, Address.Start.Column, Address.End.Row+1, Address.End.Column)); | 
 |                     } | 
 |                     else | 
 |                     { | 
 |                         Address = new ExcelAddress(WorkSheet.Name, ExcelAddressBase.GetAddress(Address.Start.Row, Address.Start.Column, Address.End.Row - 1, Address.End.Column)); | 
 |                     } | 
 |                     SetXmlNodeString("@ref", Address.Address); | 
 |                     if (value) | 
 |                     { | 
 |                         SetXmlNodeString(TOTALSROWCOUNT_PATH, "1"); | 
 |                     } | 
 |                     else | 
 |                     { | 
 |                         DeleteNode(TOTALSROWCOUNT_PATH); | 
 |                     } | 
 |                     WriteAutoFilter(value); | 
 |                 } | 
 |             } | 
 |         } | 
 |         const string STYLENAME_PATH = "d:tableStyleInfo/@name"; | 
 |         /// <summary> | 
 |         /// The style name for custum styles | 
 |         /// </summary> | 
 |         public string StyleName | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeString(STYLENAME_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (value.StartsWith("TableStyle")) | 
 |                 { | 
 |                     try | 
 |                     { | 
 |                         _tableStyle = (TableStyles)Enum.Parse(typeof(TableStyles), value.Substring(10,value.Length-10), true); | 
 |                     } | 
 |                     catch | 
 |                     { | 
 |                         _tableStyle = TableStyles.Custom; | 
 |                     } | 
 |                 } | 
 |                 else if (value == "None") | 
 |                 { | 
 |                     _tableStyle = TableStyles.None; | 
 |                     value = ""; | 
 |                 } | 
 |                 else | 
 |                 { | 
 |                     _tableStyle = TableStyles.Custom; | 
 |                 } | 
 |                 SetXmlNodeString(STYLENAME_PATH,value,true); | 
 |             } | 
 |         } | 
 |         const string SHOWFIRSTCOLUMN_PATH = "d:tableStyleInfo/@showFirstColumn"; | 
 |         /// <summary> | 
 |         /// Display special formatting for the first row | 
 |         /// </summary> | 
 |         public bool ShowFirstColumn | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool(SHOWFIRSTCOLUMN_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeBool(SHOWFIRSTCOLUMN_PATH, value, false); | 
 |             }    | 
 |         } | 
 |         const string SHOWLASTCOLUMN_PATH = "d:tableStyleInfo/@showLastColumn"; | 
 |         /// <summary> | 
 |         /// Display special formatting for the last row | 
 |         /// </summary> | 
 |         public bool ShowLastColumn | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool(SHOWLASTCOLUMN_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeBool(SHOWLASTCOLUMN_PATH, value, false); | 
 |             } | 
 |         } | 
 |         const string SHOWROWSTRIPES_PATH = "d:tableStyleInfo/@showRowStripes"; | 
 |         /// <summary> | 
 |         /// Display banded rows | 
 |         /// </summary> | 
 |         public bool ShowRowStripes | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool(SHOWROWSTRIPES_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeBool(SHOWROWSTRIPES_PATH, value, false); | 
 |             } | 
 |         } | 
 |         const string SHOWCOLUMNSTRIPES_PATH = "d:tableStyleInfo/@showColumnStripes"; | 
 |         /// <summary> | 
 |         /// Display banded columns | 
 |         /// </summary> | 
 |         public bool ShowColumnStripes | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool(SHOWCOLUMNSTRIPES_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeBool(SHOWCOLUMNSTRIPES_PATH, value, false); | 
 |             } | 
 |         } | 
 |  | 
 |         const string TOTALSROWCELLSTYLE_PATH = "@totalsRowCellStyle"; | 
 |         /// <summary> | 
 |         /// Named style used for the total row | 
 |         /// </summary> | 
 |         public string TotalsRowCellStyle | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeString(TOTALSROWCELLSTYLE_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (WorkSheet.Workbook.Styles.NamedStyles.FindIndexByID(value) < 0) | 
 |                 { | 
 |                     throw (new Exception(string.Format("Named style {0} does not exist.", value))); | 
 |                 } | 
 |                 SetXmlNodeString(TopNode, TOTALSROWCELLSTYLE_PATH, value, true); | 
 |  | 
 |                 if (ShowTotal) | 
 |                 { | 
 |                     WorkSheet.Cells[Address._toRow, Address._fromCol, Address._toRow, Address._toCol].StyleName = value; | 
 |                 } | 
 |             } | 
 |         } | 
 |         const string DATACELLSTYLE_PATH = "@dataCellStyle"; | 
 |         /// <summary> | 
 |         /// Named style used for the data cells | 
 |         /// </summary> | 
 |         public string DataCellStyleName | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeString(DATACELLSTYLE_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (WorkSheet.Workbook.Styles.NamedStyles.FindIndexByID(value) < 0) | 
 |                 { | 
 |                     throw (new Exception(string.Format("Named style {0} does not exist.", value))); | 
 |                 } | 
 |                 SetXmlNodeString(TopNode, DATACELLSTYLE_PATH, value, true); | 
 |  | 
 |                 int fromRow = Address._fromRow + (ShowHeader ? 1 : 0), | 
 |                     toRow = Address._toRow - (ShowTotal ? 1 : 0); | 
 |  | 
 |                 if (fromRow < toRow) | 
 |                 { | 
 |                     WorkSheet.Cells[fromRow, Address._fromCol, toRow, Address._toCol].StyleName = value; | 
 |                 } | 
 |             } | 
 |         } | 
 |         const string HEADERROWCELLSTYLE_PATH = "@headerRowCellStyle"; | 
 |         /// <summary> | 
 |         /// Named style used for the header row | 
 |         /// </summary> | 
 |         public string HeaderRowCellStyle | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeString(HEADERROWCELLSTYLE_PATH); | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (WorkSheet.Workbook.Styles.NamedStyles.FindIndexByID(value) < 0) | 
 |                 { | 
 |                     throw (new Exception(string.Format("Named style {0} does not exist.", value))); | 
 |                 } | 
 |                 SetXmlNodeString(TopNode, HEADERROWCELLSTYLE_PATH, value, true); | 
 |  | 
 |                 if (ShowHeader) | 
 |                 { | 
 |                     WorkSheet.Cells[Address._fromRow, Address._fromCol, Address._fromRow, Address._toCol].StyleName = value; | 
 |                 } | 
 |  | 
 |             } | 
 |         } | 
 |  | 
 |         public bool Equals(ExcelTable x, ExcelTable y) | 
 |         { | 
 |             return x.WorkSheet == y.WorkSheet && x.Id == y.Id && x.TableXml.OuterXml == y.TableXml.OuterXml; | 
 |         } | 
 |  | 
 |         public int GetHashCode(ExcelTable obj) | 
 |         { | 
 |             return obj.TableXml.OuterXml.GetHashCode(); | 
 |         } | 
 |     } | 
 | } |