/*******************************************************************************
|
* 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 21-MAR-2011
|
* Jan Källman License changed GPL-->LGPL 2011-12-16
|
*******************************************************************************/
|
using System;
|
using System.Collections.Generic;
|
using System.Text;
|
using System.Xml;
|
|
namespace HH.WMS.Utils.EPPlus.Table.PivotTable
|
{
|
/// <summary>
|
/// Base collection class for pivottable fields
|
/// </summary>
|
/// <typeparam name="T"></typeparam>
|
public class ExcelPivotTableFieldCollectionBase<T> : IEnumerable<T>
|
{
|
protected ExcelPivotTable _table;
|
internal List<T> _list = new List<T>();
|
internal ExcelPivotTableFieldCollectionBase(ExcelPivotTable table)
|
{
|
_table = table;
|
}
|
public IEnumerator<T> GetEnumerator()
|
{
|
return _list.GetEnumerator();
|
}
|
|
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
|
{
|
return _list.GetEnumerator();
|
}
|
public int Count
|
{
|
get
|
{
|
return _list.Count;
|
}
|
}
|
internal void AddInternal(T field)
|
{
|
_list.Add(field);
|
}
|
internal void Clear()
|
{
|
_list.Clear();
|
}
|
public T this[int Index]
|
{
|
get
|
{
|
if (Index < 0 || Index >= _list.Count)
|
{
|
throw (new ArgumentOutOfRangeException("Index out of range"));
|
}
|
return _list[Index];
|
}
|
}
|
}
|
public class ExcelPivotTableFieldCollection : ExcelPivotTableFieldCollectionBase<ExcelPivotTableField>
|
{
|
internal ExcelPivotTableFieldCollection(ExcelPivotTable table, string topNode) :
|
base(table)
|
{
|
|
}
|
/// <summary>
|
/// Indexer by name
|
/// </summary>
|
/// <param name="name"></param>
|
/// <returns></returns>
|
public ExcelPivotTableField this[string name]
|
{
|
get
|
{
|
foreach (var field in _list)
|
{
|
if (field.Name == name)
|
{
|
return field;
|
}
|
}
|
return null;
|
}
|
}
|
/// <summary>
|
/// Returns the date group field.
|
/// </summary>
|
/// <param name="GroupBy">The type of grouping</param>
|
/// <returns>The matching field. If none is found null is returned</returns>
|
public ExcelPivotTableField GetDateGroupField(eDateGroupBy GroupBy)
|
{
|
foreach (var fld in _list)
|
{
|
if (fld.Grouping is ExcelPivotTableFieldDateGroup && (((ExcelPivotTableFieldDateGroup)fld.Grouping).GroupBy) == GroupBy)
|
{
|
return fld;
|
}
|
}
|
return null;
|
}
|
/// <summary>
|
/// Returns the numeric group field.
|
/// </summary>
|
/// <returns>The matching field. If none is found null is returned</returns>
|
public ExcelPivotTableField GetNumericGroupField()
|
{
|
foreach (var fld in _list)
|
{
|
if (fld.Grouping is ExcelPivotTableFieldNumericGroup)
|
{
|
return fld;
|
}
|
}
|
return null;
|
}
|
}
|
/// <summary>
|
/// Collection class for Row and column fields in a Pivottable
|
/// </summary>
|
public class ExcelPivotTableRowColumnFieldCollection : ExcelPivotTableFieldCollectionBase<ExcelPivotTableField>
|
{
|
internal string _topNode;
|
internal ExcelPivotTableRowColumnFieldCollection(ExcelPivotTable table, string topNode) :
|
base(table)
|
{
|
_topNode=topNode;
|
}
|
|
/// <summary>
|
/// Add a new row/column field
|
/// </summary>
|
/// <param name="Field">The field</param>
|
/// <returns>The new field</returns>
|
public ExcelPivotTableField Add(ExcelPivotTableField Field)
|
{
|
SetFlag(Field, true);
|
_list.Add(Field);
|
return Field;
|
}
|
/// <summary>
|
/// Insert a new row/column field
|
/// </summary>
|
/// <param name="Field">The field</param>
|
/// <param name="Index">The position to insert the field</param>
|
/// <returns>The new field</returns>
|
internal ExcelPivotTableField Insert(ExcelPivotTableField Field, int Index)
|
{
|
SetFlag(Field, true);
|
_list.Insert(Index, Field);
|
return Field;
|
}
|
private void SetFlag(ExcelPivotTableField field, bool value)
|
{
|
switch (_topNode)
|
{
|
case "rowFields":
|
if (field.IsColumnField || field.IsPageField)
|
{
|
throw(new Exception("This field is a column or page field. Can's add it to the RowFields collection"));
|
}
|
field.IsRowField = value;
|
field.Axis = ePivotFieldAxis.Row;
|
break;
|
case "colFields":
|
if (field.IsRowField || field.IsPageField)
|
{
|
throw (new Exception("This field is a row or page field. Can's add it to the ColumnFields collection"));
|
}
|
field.IsColumnField = value;
|
field.Axis = ePivotFieldAxis.Column;
|
break;
|
case "pageFields":
|
if (field.IsColumnField || field.IsRowField)
|
{
|
throw (new Exception("Field is a column or row field. Can's add it to the PageFields collection"));
|
}
|
if (_table.Address._fromRow < 3)
|
{
|
throw(new Exception(string.Format("A pivot table with page fields must be located above row 3. Currenct location is {0}", _table.Address.Address)));
|
}
|
field.IsPageField = value;
|
field.Axis = ePivotFieldAxis.Page;
|
break;
|
case "dataFields":
|
|
break;
|
}
|
}
|
/// <summary>
|
/// Remove a field
|
/// </summary>
|
/// <param name="Field"></param>
|
public void Remove(ExcelPivotTableField Field)
|
{
|
if(!_list.Contains(Field))
|
{
|
throw new ArgumentException("Field not in collection");
|
}
|
SetFlag(Field, false);
|
_list.Remove(Field);
|
}
|
/// <summary>
|
/// Remove a field at a specific position
|
/// </summary>
|
/// <param name="Index"></param>
|
public void RemoveAt(int Index)
|
{
|
if (Index > -1 && Index < _list.Count)
|
{
|
throw(new IndexOutOfRangeException());
|
}
|
SetFlag(_list[Index], false);
|
_list.RemoveAt(Index);
|
}
|
}
|
/// <summary>
|
/// Collection class for data fields in a Pivottable
|
/// </summary>
|
public class ExcelPivotTableDataFieldCollection : ExcelPivotTableFieldCollectionBase<ExcelPivotTableDataField>
|
{
|
internal ExcelPivotTableDataFieldCollection(ExcelPivotTable table) :
|
base(table)
|
{
|
|
}
|
/// <summary>
|
/// Add a new datafield
|
/// </summary>
|
/// <param name="field">The field</param>
|
/// <returns>The new datafield</returns>
|
public ExcelPivotTableDataField Add(ExcelPivotTableField field)
|
{
|
var dataFieldsNode = field.TopNode.SelectSingleNode("../../d:dataFields", field.NameSpaceManager);
|
if (dataFieldsNode == null)
|
{
|
_table.CreateNode("d:dataFields");
|
dataFieldsNode = field.TopNode.SelectSingleNode("../../d:dataFields", field.NameSpaceManager);
|
}
|
|
XmlElement node = _table.PivotTableXml.CreateElement("dataField", ExcelPackage.schemaMain);
|
node.SetAttribute("fld", field.Index.ToString());
|
dataFieldsNode.AppendChild(node);
|
|
//XmlElement node = field.AppendField(dataFieldsNode, field.Index, "dataField", "fld");
|
field.SetXmlNodeBool("@dataField", true,false);
|
|
var dataField = new ExcelPivotTableDataField(field.NameSpaceManager, node, field);
|
ValidateDupName(dataField);
|
|
_list.Add(dataField);
|
return dataField;
|
}
|
private void ValidateDupName(ExcelPivotTableDataField dataField)
|
{
|
if(ExistsDfName(dataField.Field.Name, null))
|
{
|
var index = 2;
|
string name;
|
do
|
{
|
name = dataField.Field.Name + "_" + index++.ToString();
|
}
|
while (ExistsDfName(name,null));
|
dataField.Name = name;
|
}
|
}
|
|
internal bool ExistsDfName(string name, ExcelPivotTableDataField datafield)
|
{
|
name = name.ToLower();
|
foreach (var df in _list)
|
{
|
if (((!string.IsNullOrEmpty(df.Name) && df.Name.ToLower() == name) ||
|
(string.IsNullOrEmpty(df.Name) && df.Field.Name.ToLower() == name)) && datafield != df)
|
{
|
return true;
|
}
|
}
|
return false;
|
}
|
/// <summary>
|
/// Remove a datafield
|
/// </summary>
|
/// <param name="dataField"></param>
|
public void Remove(ExcelPivotTableDataField dataField)
|
{
|
XmlElement node = dataField.Field.TopNode.SelectSingleNode(string.Format("../../d:dataFields/d:dataField[@fld={0}]", dataField.Index), dataField.NameSpaceManager) as XmlElement;
|
if (node != null)
|
{
|
node.ParentNode.RemoveChild(node);
|
}
|
_list.Remove(dataField);
|
}
|
}
|
}
|