| /******************************************************************************* |
| * 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; |
| using System.Collections.Generic; |
| using System.Xml; |
| |
| namespace OfficeOpenXml.Table; |
| |
| /// <summary> |
| /// A collection of table objects |
| /// </summary> |
| public class ExcelTableCollection : IEnumerable<ExcelTable> { |
| private List<ExcelTable> _tables = new(); |
| internal Dictionary<string, int> _tableNames = new(StringComparer.InvariantCultureIgnoreCase); |
| private ExcelWorksheet _ws; |
| |
| internal ExcelTableCollection(ExcelWorksheet ws) { |
| var pck = ws._package.Package; |
| _ws = ws; |
| foreach (XmlElement node in ws.WorksheetXml.SelectNodes( |
| "//d:tableParts/d:tablePart", |
| ws.NameSpaceManager)) { |
| var rel = ws.Part.GetRelationship(node.GetAttribute("id", ExcelPackage._schemaRelationships)); |
| var tbl = new ExcelTable(rel, ws); |
| _tableNames.Add(tbl.Name, _tables.Count); |
| _tables.Add(tbl); |
| } |
| } |
| |
| private ExcelTable Add(ExcelTable tbl) { |
| _tables.Add(tbl); |
| _tableNames.Add(tbl.Name, _tables.Count - 1); |
| if (tbl.Id >= _ws.Workbook._nextTableID) { |
| _ws.Workbook._nextTableID = tbl.Id + 1; |
| } |
| return tbl; |
| } |
| |
| /// <summary> |
| /// Create a table on the supplied range |
| /// </summary> |
| /// <param name="range">The range address including header and total row</param> |
| /// <param name="name">The name of the table. Must be unique </param> |
| /// <returns>The table object</returns> |
| public ExcelTable Add(ExcelAddressBase range, string name) { |
| if (range.WorkSheet != null && range.WorkSheet != _ws.Name) { |
| throw new ArgumentException("Range does not belong to worksheet", "range"); |
| } |
| |
| if (string.IsNullOrEmpty(name)) { |
| name = GetNewTableName(); |
| } else if (_ws.Workbook.ExistsTableName(name)) { |
| throw (new ArgumentException("Tablename is not unique")); |
| } |
| |
| ValidateTableName(name); |
| |
| foreach (var t in _tables) { |
| if (t.Address.Collide(range) != ExcelAddressBase.eAddressCollition.No) { |
| throw (new ArgumentException(string.Format("Table range collides with table {0}", t.Name))); |
| } |
| } |
| return Add(new(_ws, range, name, _ws.Workbook._nextTableID)); |
| } |
| |
| private void ValidateTableName(string name) { |
| if (string.IsNullOrEmpty(name)) { |
| throw new ArgumentException("Tablename is null or empty"); |
| } |
| |
| char firstLetterOfName = name[0]; |
| if (Char.IsLetter(firstLetterOfName) == false |
| && firstLetterOfName != '_' |
| && firstLetterOfName != '\\') { |
| throw new ArgumentException("Tablename start with invalid character"); |
| } |
| |
| if (name.Contains(" ")) { |
| throw new ArgumentException("Tablename has spaces"); |
| } |
| } |
| |
| public void Delete(int index, bool clearRange = false) { |
| Delete(this[index], clearRange); |
| } |
| |
| public void Delete(string name, bool clearRange = false) { |
| if (this[name] == null) { |
| throw new ArgumentOutOfRangeException( |
| string.Format("Cannot delete non-existant table {0} in sheet {1}.", name, _ws.Name)); |
| } |
| Delete(this[name], clearRange); |
| } |
| |
| public void Delete(ExcelTable excelTable, bool clearRange = false) { |
| if (!_tables.Contains(excelTable)) { |
| throw new ArgumentOutOfRangeException( |
| "excelTable", |
| String.Format("Table {0} does not exist in this collection", excelTable.Name)); |
| } |
| lock (this) { |
| var range = _ws.Cells[excelTable.Address.Address]; |
| _tableNames.Remove(excelTable.Name); |
| _tables.Remove(excelTable); |
| foreach (var sheet in excelTable.WorkSheet.Workbook.Worksheets) { |
| foreach (var table in sheet.Tables) { |
| if (table.Id > excelTable.Id) { |
| table.Id--; |
| } |
| } |
| excelTable.WorkSheet.Workbook._nextTableID--; |
| } |
| if (clearRange) { |
| range.Clear(); |
| } |
| } |
| } |
| |
| internal string GetNewTableName() { |
| string name = "Table1"; |
| int i = 2; |
| while (_ws.Workbook.ExistsTableName(name)) { |
| name = string.Format("Table{0}", i++); |
| } |
| return name; |
| } |
| |
| /// <summary> |
| /// Number of items in the collection |
| /// </summary> |
| public int Count => _tables.Count; |
| |
| /// <summary> |
| /// Get the table object from a range. |
| /// </summary> |
| /// <param name="range">The range</param> |
| /// <returns>The table. Null if no range matches</returns> |
| public ExcelTable GetFromRange(ExcelRangeBase range) { |
| foreach (var tbl in range.Worksheet.Tables) { |
| if (tbl.Address._address == range._address) { |
| return tbl; |
| } |
| } |
| return null; |
| } |
| |
| /// <summary> |
| /// The table Index. Base 0. |
| /// </summary> |
| /// <param name="index"></param> |
| /// <returns></returns> |
| public ExcelTable this[int index] { |
| get { |
| if (index < 0 || index >= _tables.Count) { |
| throw (new ArgumentOutOfRangeException("Table index out of range")); |
| } |
| return _tables[index]; |
| } |
| } |
| |
| /// <summary> |
| /// Indexer |
| /// </summary> |
| /// <param name="name">The name of the table</param> |
| /// <returns>The table. Null if the table name is not found in the collection</returns> |
| public ExcelTable this[string name] { |
| get { |
| if (_tableNames.ContainsKey(name)) { |
| return _tables[_tableNames[name]]; |
| } |
| return null; |
| } |
| } |
| |
| public IEnumerator<ExcelTable> GetEnumerator() { |
| return _tables.GetEnumerator(); |
| } |
| |
| IEnumerator IEnumerable.GetEnumerator() { |
| return _tables.GetEnumerator(); |
| } |
| } |