| /******************************************************************************* |
| * 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 2012-11-25 |
| *******************************************************************************/ |
| |
| using System; |
| using System.Collections; |
| using System.Collections.Generic; |
| using OfficeOpenXml; |
| |
| internal class IndexBase : IComparable<IndexBase> { |
| internal short Index; |
| |
| public int CompareTo(IndexBase other) { |
| return Index < other.Index |
| ? -1 |
| : Index > other.Index |
| ? 1 |
| : 0; |
| } |
| } |
| |
| internal class IndexItem : IndexBase { |
| internal int IndexPointer { get; set; } |
| } |
| |
| internal class ColumnIndex : IndexBase { |
| private readonly IndexBase _searchIx = new(); |
| |
| internal int GetPosition(int row) { |
| var page = (short)(row >> CellStore<int>._pageBits); |
| _searchIx.Index = page; |
| var res = |
| (_pages != null |
| && page >= 0 |
| && page < PageCount |
| && page < _pages.Length |
| && _pages[page] != null |
| && _pages[page].Index == page) |
| ? page |
| : Array.BinarySearch(_pages, 0, PageCount, _searchIx); |
| if (res >= 0) { |
| GetPage(row, ref res); |
| return res; |
| } |
| var p = ~res; |
| |
| if (GetPage(row, ref p)) { |
| return p; |
| } |
| return res; |
| } |
| |
| private bool GetPage(int row, ref int res) { |
| if (res < PageCount && _pages[res].MinIndex <= row && _pages[res].MaxIndex >= row) { |
| return true; |
| } |
| if (res + 1 < PageCount && (_pages[res + 1].MinIndex <= row)) { |
| do { |
| res++; |
| } while (res + 1 < PageCount && _pages[res + 1].MinIndex <= row); |
| //if (res + 1 < PageCount && _pages[res + 1].MaxIndex >= Row) |
| //{ |
| return true; |
| //} |
| //else |
| //{ |
| // return false; |
| //} |
| } |
| if (res - 1 >= 0 && _pages[res - 1].MaxIndex >= row) { |
| do { |
| res--; |
| } while (res - 1 > 0 && _pages[res - 1].MaxIndex >= row); |
| //if (res > 0) |
| //{ |
| return true; |
| //} |
| //else |
| //{ |
| // return false; |
| //} |
| } |
| return false; |
| } |
| |
| internal int GetNextRow(int row) { |
| //var page = (int)((ulong)row >> CellStore<int>.pageBits); |
| var p = GetPosition(row); |
| if (p < 0) { |
| p = ~p; |
| if (p >= PageCount) { |
| return -1; |
| } |
| if (_pages[p].IndexOffset + _pages[p].Rows[0].Index < row) { |
| if (p + 1 >= PageCount) { |
| return -1; |
| } |
| return _pages[p + 1].IndexOffset + _pages[p].Rows[0].Index; |
| } |
| return _pages[p].IndexOffset + _pages[p].Rows[0].Index; |
| } |
| if (p < PageCount) { |
| var r = _pages[p].GetNextRow(row); |
| if (r >= 0) { |
| return _pages[p].IndexOffset + _pages[p].Rows[r].Index; |
| } else { |
| if (++p < PageCount) { |
| return _pages[p].IndexOffset + _pages[p].Rows[0].Index; |
| } else { |
| return -1; |
| } |
| } |
| } |
| return -1; |
| } |
| |
| internal PageIndex[] _pages = new PageIndex[CellStore<int>._pagesPerColumnMin]; |
| internal int PageCount; |
| } |
| |
| internal class PageIndex : IndexBase { |
| private readonly IndexBase _searchIx = new(); |
| |
| public PageIndex() { |
| Rows = new IndexItem[CellStore<int>._pageSizeMin]; |
| RowCount = 0; |
| } |
| |
| public PageIndex(IndexItem[] rows, int count) { |
| Rows = rows; |
| RowCount = count; |
| } |
| |
| public PageIndex(PageIndex pageItem, int start, int size) |
| : this(pageItem, start, size, pageItem.Index, pageItem.Offset) {} |
| |
| public PageIndex(PageIndex pageItem, int start, int size, short index, int offset) { |
| Rows = new IndexItem[CellStore<int>.GetSize(size)]; |
| Array.Copy(pageItem.Rows, start, Rows, 0, size); |
| RowCount = size; |
| Index = index; |
| Offset = offset; |
| } |
| |
| internal int Offset; |
| |
| internal int IndexOffset => IndexExpanded + Offset; |
| |
| internal int IndexExpanded => (Index << CellStore<int>._pageBits); |
| |
| internal IndexItem[] Rows { get; set; } |
| |
| internal int RowCount; |
| |
| internal int GetPosition(int offset) { |
| _searchIx.Index = (short)offset; |
| return (Rows != null |
| && offset > 0 |
| && offset - 1 < RowCount |
| && offset - 1 < Rows.Length |
| && Rows[offset - 1] != null |
| && Rows[offset - 1].Index == offset) |
| ? offset - 1 |
| : Array.BinarySearch(Rows, 0, RowCount, _searchIx); |
| } |
| |
| internal int GetNextRow(int row) { |
| int offset = row - IndexOffset; |
| var o = GetPosition(offset); |
| if (o < 0) { |
| o = ~o; |
| if (o < RowCount) { |
| return o; |
| } |
| return -1; |
| } |
| return o; |
| } |
| |
| public int MinIndex { |
| get { |
| if (Rows.Length > 0) { |
| return IndexOffset + Rows[0].Index; |
| } |
| return -1; |
| } |
| } |
| |
| public int MaxIndex { |
| get { |
| if (RowCount > 0) { |
| return IndexOffset + Rows[RowCount - 1].Index; |
| } |
| return -1; |
| } |
| } |
| |
| public int GetIndex(int pos) { |
| return IndexOffset + Rows[pos].Index; |
| } |
| } |
| |
| /// <summary> |
| /// This is the store for all Rows, Columns and Cells. |
| /// It is a Dictionary implementation that allows you to change the Key (the RowID, ColumnID or CellID ) |
| /// </summary> |
| internal class CellStore<T> { |
| /**** Size constants ****/ |
| internal const int _pageBits = 10; //13bits=8192 Note: Maximum is 13 bits since short is used (PageMax=16K) |
| internal const int _pageSize = 1 << _pageBits; |
| internal const int _pageSizeMin = 1 << 10; |
| internal const int _pageSizeMax = _pageSize << 1; //Double page size |
| internal const int _colSizeMin = 32; |
| internal const int _pagesPerColumnMin = 32; |
| |
| private List<T> _values = new(); |
| internal ColumnIndex[] _columnIndex = new ColumnIndex[_colSizeMin]; |
| internal IndexBase _searchIx = new(); |
| internal int ColumnCount; |
| |
| ~CellStore() { |
| if (_values != null) { |
| _values.Clear(); |
| _values = null; |
| } |
| _columnIndex = null; |
| } |
| |
| internal int GetPosition(int column) { |
| _searchIx.Index = (short)column; |
| return (_columnIndex != null |
| && column > 0 |
| && column - 1 < ColumnCount |
| && column - 1 < _columnIndex.Length |
| && _columnIndex[column - 1] != null |
| && _columnIndex[column - 1].Index == column) |
| ? column - 1 |
| : Array.BinarySearch(_columnIndex, 0, ColumnCount, _searchIx); |
| } |
| |
| internal bool GetDimension(out int fromRow, out int fromCol, out int toRow, out int toCol) { |
| if (ColumnCount == 0) { |
| fromRow = fromCol = toRow = toCol = 0; |
| return false; |
| } |
| fromCol = _columnIndex[0].Index; |
| var fromIndex = 0; |
| if (fromCol <= 0 && ColumnCount > 1) { |
| fromCol = _columnIndex[1].Index; |
| fromIndex = 1; |
| } else if (ColumnCount == 1 && fromCol <= 0) { |
| fromRow = fromCol = toRow = toCol = 0; |
| return false; |
| } |
| var col = ColumnCount - 1; |
| while (col > 0) { |
| if (_columnIndex[col].PageCount == 0 |
| || _columnIndex[col]._pages[0].RowCount > 1 |
| || _columnIndex[col]._pages[0].Rows[0].Index > 0) { |
| break; |
| } |
| col--; |
| } |
| toCol = _columnIndex[col].Index; |
| if (toCol == 0) { |
| fromRow = fromCol = toRow = toCol = 0; |
| return false; |
| } |
| fromRow = toRow = 0; |
| |
| for (int c = fromIndex; c < ColumnCount; c++) { |
| int first, |
| last; |
| if (_columnIndex[c].PageCount == 0) { |
| continue; |
| } |
| if (_columnIndex[c]._pages[0].RowCount > 0 && _columnIndex[c]._pages[0].Rows[0].Index > 0) { |
| first = _columnIndex[c]._pages[0].IndexOffset + _columnIndex[c]._pages[0].Rows[0].Index; |
| } else { |
| if (_columnIndex[c]._pages[0].RowCount > 1) { |
| first = _columnIndex[c]._pages[0].IndexOffset + _columnIndex[c]._pages[0].Rows[1].Index; |
| } else if (_columnIndex[c].PageCount > 1) { |
| first = _columnIndex[c]._pages[0].IndexOffset + _columnIndex[c]._pages[1].Rows[0].Index; |
| } else { |
| first = 0; |
| } |
| } |
| var lp = _columnIndex[c].PageCount - 1; |
| while (_columnIndex[c]._pages[lp].RowCount == 0 && lp != 0) { |
| lp--; |
| } |
| var p = _columnIndex[c]._pages[lp]; |
| if (p.RowCount > 0) { |
| last = p.IndexOffset + p.Rows[p.RowCount - 1].Index; |
| } else { |
| last = first; |
| } |
| if (first > 0 && (first < fromRow || fromRow == 0)) { |
| fromRow = first; |
| } |
| if (first > 0 && (last > toRow || toRow == 0)) { |
| toRow = last; |
| } |
| } |
| if (fromRow <= 0 || toRow <= 0) { |
| fromRow = fromCol = toRow = toCol = 0; |
| return false; |
| } |
| return true; |
| } |
| |
| internal int FindNext(int column) { |
| var c = GetPosition(column); |
| if (c < 0) { |
| return ~c; |
| } |
| return c; |
| } |
| |
| internal T GetValue(int row, int column) { |
| int i = GetPointer(row, column); |
| if (i >= 0) { |
| return _values[i]; |
| } |
| return default(T); |
| } |
| |
| private int GetPointer(int row, int column) { |
| var col = GetPosition(column); |
| if (col >= 0) { |
| var pos = _columnIndex[col].GetPosition(row); |
| if (pos >= 0 && pos < _columnIndex[col].PageCount) { |
| var pageItem = _columnIndex[col]._pages[pos]; |
| if (pageItem.MinIndex > row) { |
| pos--; |
| if (pos < 0) { |
| return -1; |
| } |
| pageItem = _columnIndex[col]._pages[pos]; |
| } |
| short ix = (short)(row - pageItem.IndexOffset); |
| _searchIx.Index = ix; |
| var cellPos = |
| (pageItem.Rows != null |
| && ix > 0 |
| && ix - 1 < pageItem.RowCount |
| && ix - 1 < pageItem.Rows.Length |
| && pageItem.Rows[ix - 1] != null |
| && pageItem.Rows[ix - 1].Index == ix) |
| ? ix - 1 |
| : Array.BinarySearch(pageItem.Rows, 0, pageItem.RowCount, _searchIx); |
| if (cellPos >= 0) { |
| return pageItem.Rows[cellPos].IndexPointer; |
| } //Cell does not exist |
| return -1; |
| } //Page does not exist |
| return -1; |
| } //Column does not exist |
| return -1; |
| } |
| |
| internal bool Exists(int row, int column) { |
| return GetPointer(row, column) >= 0; |
| } |
| |
| internal bool Exists(int row, int column, ref T value) { |
| var p = GetPointer(row, column); |
| if (p >= 0) { |
| value = _values[p]; |
| return true; |
| } |
| return false; |
| } |
| |
| internal void SetValue(int row, int column, T value) { |
| var col = |
| (_columnIndex != null |
| && column > 0 |
| && column - 1 < ColumnCount |
| && column - 1 < _columnIndex.Length |
| && _columnIndex[column - 1] != null |
| && _columnIndex[column - 1].Index == column) |
| ? column - 1 |
| : Array.BinarySearch( |
| _columnIndex, |
| 0, |
| ColumnCount, |
| new IndexBase { |
| Index = (short)(column), |
| }); |
| var page = (short)(row >> _pageBits); |
| if (col >= 0) { |
| //var pos = Array.BinarySearch(_columnIndex[col].Pages, 0, _columnIndex[col].Count, new IndexBase() { Index = page }); |
| var pos = _columnIndex[col].GetPosition(row); |
| if (pos < 0) { |
| pos = ~pos; |
| if (pos - 1 < 0 || _columnIndex[col]._pages[pos - 1].IndexOffset + _pageSize - 1 < row) { |
| AddPage(_columnIndex[col], pos, page); |
| } else { |
| pos--; |
| } |
| } |
| if (pos >= _columnIndex[col].PageCount) { |
| AddPage(_columnIndex[col], pos, page); |
| } |
| var pageItem = _columnIndex[col]._pages[pos]; |
| if (pageItem.IndexOffset > row) { |
| pos--; |
| page--; |
| if (pos < 0) { |
| throw (new("Unexpected error when setting value")); |
| } |
| pageItem = _columnIndex[col]._pages[pos]; |
| } |
| |
| short ix = (short)(row - ((pageItem.Index << _pageBits) + pageItem.Offset)); |
| _searchIx.Index = ix; |
| var cellPos = |
| (pageItem.Rows != null |
| && ix >= 0 |
| && ix < pageItem.RowCount |
| && ix < pageItem.Rows.Length |
| && pageItem.Rows[ix] != null |
| && pageItem.Rows[ix].Index == ix) |
| ? ix |
| : Array.BinarySearch(pageItem.Rows, 0, pageItem.RowCount, _searchIx); |
| if (cellPos < 0) { |
| cellPos = ~cellPos; |
| AddCell(_columnIndex[col], pos, cellPos, ix, value); |
| } else { |
| _values[pageItem.Rows[cellPos].IndexPointer] = value; |
| } |
| } else //Column does not exist |
| { |
| col = ~col; |
| AddColumn(col, column); |
| AddPage(_columnIndex[col], 0, page); |
| short ix = (short)(row - (page << _pageBits)); |
| AddCell(_columnIndex[col], 0, 0, ix, value); |
| } |
| } |
| |
| internal void Insert(int fromRow, int fromCol, int rows, int columns) { |
| if (columns > 0) { |
| var col = GetPosition(fromCol); |
| if (col < 0) { |
| col = ~col; |
| } |
| for (var c = col; c < ColumnCount; c++) { |
| _columnIndex[c].Index += (short)columns; |
| } |
| } else { |
| var page = fromRow >> _pageBits; |
| for (int c = 0; c < ColumnCount; c++) { |
| var column = _columnIndex[c]; |
| var pagePos = column.GetPosition(fromRow); |
| if (pagePos >= 0) { |
| if (fromRow >= column._pages[pagePos].MinIndex |
| && fromRow |
| <= column._pages[pagePos].MaxIndex) //The row is inside the page |
| { |
| int offset = fromRow - column._pages[pagePos].IndexOffset; |
| var rowPos = column._pages[pagePos].GetPosition(offset); |
| if (rowPos < 0) { |
| rowPos = ~rowPos; |
| } |
| UpdateIndexOffset(column, pagePos, rowPos, fromRow, rows); |
| } else if (column._pages[pagePos].MinIndex > fromRow - 1 |
| && pagePos |
| > 0) //The row is on the page before. |
| { |
| int offset = fromRow - ((page - 1) << _pageBits); |
| var rowPos = column._pages[pagePos - 1].GetPosition(offset); |
| if (rowPos > 0 && pagePos > 0) { |
| UpdateIndexOffset(column, pagePos - 1, rowPos, fromRow, rows); |
| } |
| } else if (column.PageCount >= pagePos + 1) { |
| int offset = fromRow - column._pages[pagePos].IndexOffset; |
| var rowPos = column._pages[pagePos].GetPosition(offset); |
| if (rowPos < 0) { |
| rowPos = ~rowPos; |
| } |
| if (column._pages[pagePos].RowCount > rowPos) { |
| UpdateIndexOffset(column, pagePos, rowPos, fromRow, rows); |
| } else { |
| UpdateIndexOffset(column, pagePos + 1, 0, fromRow, rows); |
| } |
| } |
| } else { |
| UpdateIndexOffset(column, ~pagePos, 0, fromRow, rows); |
| } |
| } |
| } |
| } |
| |
| internal void Clear(int fromRow, int fromCol, int rows, int columns) { |
| Delete(fromRow, fromCol, rows, columns, false); |
| } |
| |
| internal void Delete(int fromRow, int fromCol, int rows, int columns) { |
| Delete(fromRow, fromCol, rows, columns, true); |
| } |
| |
| internal void Delete(int fromRow, int fromCol, int rows, int columns, bool shift) { |
| if (columns > 0 && fromRow == 1 && rows >= ExcelPackage.MaxRows) { |
| DeleteColumns(fromCol, columns, shift); |
| } else { |
| var toCol = fromCol + columns - 1; |
| var pageFromRow = fromRow >> _pageBits; |
| for (int c = 0; c < ColumnCount; c++) { |
| var column = _columnIndex[c]; |
| if (column.Index >= fromCol) { |
| if (column.Index > toCol) { |
| break; |
| } |
| var pagePos = column.GetPosition(fromRow); |
| if (pagePos < 0) { |
| pagePos = ~pagePos; |
| } |
| if (pagePos < column.PageCount) { |
| var page = column._pages[pagePos]; |
| if (shift |
| && page.RowCount > 0 |
| && page.MinIndex > fromRow |
| && page.MaxIndex >= fromRow + rows) { |
| var o = page.MinIndex - fromRow; |
| if (o < rows) { |
| rows -= o; |
| page.Offset -= o; |
| UpdatePageOffset(column, pagePos, o); |
| } else { |
| page.Offset -= rows; |
| UpdatePageOffset(column, pagePos, rows); |
| continue; |
| } |
| } |
| if (page.RowCount > 0 |
| && page.MinIndex <= fromRow + rows - 1 |
| && page.MaxIndex |
| >= fromRow) //The row is inside the page |
| { |
| var endRow = fromRow + rows; |
| var delEndRow = DeleteCells(column._pages[pagePos], fromRow, endRow, shift); |
| if (shift && delEndRow != fromRow) { |
| UpdatePageOffset(column, pagePos, delEndRow - fromRow); |
| } |
| if (endRow > delEndRow |
| && pagePos < column.PageCount |
| && column._pages[pagePos].MinIndex < endRow) { |
| pagePos = (delEndRow == fromRow ? pagePos : pagePos + 1); |
| var rowsLeft = DeletePage( |
| shift ? fromRow : delEndRow, |
| endRow - delEndRow, |
| column, |
| pagePos, |
| shift); |
| //if (shift) UpdatePageOffset(column, pagePos, endRow - fromRow - rowsLeft); |
| if (rowsLeft > 0) { |
| var fr = shift ? fromRow : endRow - rowsLeft; |
| pagePos = column.GetPosition(fr); |
| DeleteCells(column._pages[pagePos], fr, shift ? fr + rowsLeft : endRow, shift); |
| if (shift) { |
| UpdatePageOffset(column, pagePos, rowsLeft); |
| } |
| } |
| } |
| } else if (pagePos > 0 |
| && column._pages[pagePos].IndexOffset |
| > fromRow) //The row is on the page before. |
| { |
| int offset = fromRow + rows - 1 - ((pageFromRow - 1) << _pageBits); |
| var rowPos = column._pages[pagePos - 1].GetPosition(offset); |
| if (rowPos > 0 && pagePos > 0) { |
| if (shift) { |
| UpdateIndexOffset(column, pagePos - 1, rowPos, fromRow + rows - 1, -rows); |
| } |
| } |
| } else { |
| if (shift && pagePos + 1 < column.PageCount) { |
| UpdateIndexOffset( |
| column, |
| pagePos + 1, |
| 0, |
| column._pages[pagePos + 1].MinIndex, |
| -rows); |
| } |
| } |
| } |
| } |
| } |
| } |
| } |
| |
| private void UpdatePageOffset(ColumnIndex column, int pagePos, int rows) { |
| //Update Pageoffset |
| |
| if (++pagePos < column.PageCount) { |
| for (int p = pagePos; p < column.PageCount; p++) { |
| if (column._pages[p].Offset - rows <= -_pageSize) { |
| column._pages[p].Index--; |
| column._pages[p].Offset -= rows - _pageSize; |
| } else { |
| column._pages[p].Offset -= rows; |
| } |
| } |
| |
| if (Math.Abs(column._pages[pagePos].Offset) > _pageSize |
| || Math.Abs(column._pages[pagePos].Rows[column._pages[pagePos].RowCount - 1].Index) |
| > _pageSizeMax) //Split or Merge??? |
| { |
| ResetPageOffset(column, pagePos, rows); |
| } |
| } |
| } |
| |
| private void ResetPageOffset(ColumnIndex column, int pagePos, int rows) { |
| PageIndex fromPage = column._pages[pagePos]; |
| PageIndex toPage; |
| short pageAdd = 0; |
| if (fromPage.Offset < -_pageSize) { |
| toPage = column._pages[pagePos - 1]; |
| pageAdd = -1; |
| if (fromPage.Index - 1 == toPage.Index) { |
| if (fromPage.IndexOffset |
| + fromPage.Rows[fromPage.RowCount - 1].Index |
| - toPage.IndexOffset |
| + toPage.Rows[0].Index |
| <= _pageSizeMax) { |
| MergePage(column, pagePos - 1); |
| } |
| } else //No page after |
| { |
| fromPage.Index -= pageAdd; |
| fromPage.Offset += _pageSize; |
| } |
| } else if (fromPage.Offset > _pageSize) { |
| toPage = column._pages[pagePos + 1]; |
| pageAdd = 1; |
| if (fromPage.Index + 1 == toPage.Index) {} else { |
| fromPage.Index += pageAdd; |
| fromPage.Offset += _pageSize; |
| } |
| } |
| } |
| |
| private int DeletePage(int fromRow, int rows, ColumnIndex column, int pagePos, bool shift) { |
| PageIndex page = column._pages[pagePos]; |
| var startRows = rows; |
| while (page != null |
| && page.MinIndex >= fromRow |
| && ((shift && page.MaxIndex < fromRow + rows) |
| || (!shift && page.MaxIndex < fromRow + startRows))) { |
| //Delete entire page. |
| var delSize = page.MaxIndex - page.MinIndex + 1; |
| rows -= delSize; |
| var prevOffset = page.Offset; |
| Array.Copy( |
| column._pages, |
| pagePos + 1, |
| column._pages, |
| pagePos, |
| column.PageCount - pagePos + 1); |
| column.PageCount--; |
| if (column.PageCount == 0) { |
| return 0; |
| } |
| if (shift) { |
| for (int i = pagePos; i < column.PageCount; i++) { |
| column._pages[i].Offset -= delSize; |
| if (column._pages[i].Offset <= -_pageSize) { |
| column._pages[i].Index--; |
| column._pages[i].Offset += _pageSize; |
| } |
| } |
| } |
| if (column.PageCount > pagePos) { |
| page = column._pages[pagePos]; |
| //page.Offset = pagePos == 0 ? 1 : prevOffset; //First page can only reference to rows starting from Index == 1 |
| } else { |
| //No more pages, return 0 |
| return 0; |
| } |
| } |
| return rows; |
| } |
| |
| private int DeleteCells(PageIndex page, int fromRow, int toRow, bool shift) { |
| var fromPos = page.GetPosition(fromRow - (page.IndexOffset)); |
| if (fromPos < 0) { |
| fromPos = ~fromPos; |
| } |
| var maxRow = page.MaxIndex; |
| var offset = toRow - page.IndexOffset; |
| if (offset > _pageSizeMax) { |
| offset = _pageSizeMax; |
| } |
| var toPos = page.GetPosition(offset); |
| if (toPos < 0) { |
| toPos = ~toPos; |
| } |
| |
| if (fromPos <= toPos && fromPos < page.RowCount && page.GetIndex(fromPos) < toRow) { |
| if (toRow > page.MaxIndex) { |
| if (fromRow |
| == page.MinIndex) //Delete entire page, late in the page delete method |
| { |
| return fromRow; |
| } |
| var r = page.MaxIndex; |
| var deletedRow = page.RowCount - fromPos; |
| page.RowCount -= deletedRow; |
| return r + 1; |
| } |
| var rows = toRow - fromRow; |
| if (shift) { |
| UpdateRowIndex(page, toPos, rows); |
| } |
| Array.Copy(page.Rows, toPos, page.Rows, fromPos, page.RowCount - toPos); |
| page.RowCount -= toPos - fromPos; |
| |
| return toRow; |
| } |
| if (shift) { |
| UpdateRowIndex(page, toPos, toRow - fromRow); |
| } |
| return toRow < maxRow ? toRow : maxRow; |
| } |
| |
| private static void UpdateRowIndex(PageIndex page, int toPos, int rows) { |
| for (int r = toPos; r < page.RowCount; r++) { |
| page.Rows[r].Index -= (short)rows; |
| } |
| } |
| |
| private void DeleteColumns(int fromCol, int columns, bool shift) { |
| var fPos = GetPosition(fromCol); |
| if (fPos < 0) { |
| fPos = ~fPos; |
| } |
| int tPos = fPos; |
| for (var c = fPos; c <= ColumnCount; c++) { |
| tPos = c; |
| if (tPos == ColumnCount || _columnIndex[c].Index >= fromCol + columns) { |
| break; |
| } |
| } |
| |
| if (ColumnCount <= fPos) { |
| return; |
| } |
| |
| if (_columnIndex[fPos].Index >= fromCol && _columnIndex[fPos].Index <= fromCol + columns) { |
| if (tPos < ColumnCount) { |
| Array.Copy(_columnIndex, tPos, _columnIndex, fPos, ColumnCount - tPos); |
| } |
| ColumnCount -= (tPos - fPos); |
| } |
| if (shift) { |
| for (var c = fPos; c < ColumnCount; c++) { |
| _columnIndex[c].Index -= (short)columns; |
| } |
| } |
| } |
| |
| private void UpdateIndexOffset(ColumnIndex column, int pagePos, int rowPos, int row, int rows) { |
| if (pagePos >= column.PageCount) { |
| return; //A page after last cell. |
| } |
| var page = column._pages[pagePos]; |
| if (rows > _pageSize) { |
| short addPages = (short)(rows >> _pageBits); |
| int offset = +(rows - (_pageSize * addPages)); |
| for (int p = pagePos + 1; p < column.PageCount; p++) { |
| if (column._pages[p].Offset + offset > _pageSize) { |
| column._pages[p].Index += (short)(addPages + 1); |
| column._pages[p].Offset += offset - _pageSize; |
| } else { |
| column._pages[p].Index += addPages; |
| column._pages[p].Offset += offset; |
| } |
| } |
| |
| var size = page.RowCount - rowPos; |
| if (page.RowCount > rowPos) { |
| if (column.PageCount - 1 |
| == pagePos) //No page after, create a new one. |
| { |
| //Copy rows to next page. |
| var newPage = CopyNew(page, rowPos, size); |
| newPage.Index = (short)((row + rows) >> _pageBits); |
| newPage.Offset = row + rows - (newPage.Index * _pageSize) - newPage.Rows[0].Index; |
| if (newPage.Offset > _pageSize) { |
| newPage.Index++; |
| newPage.Offset -= _pageSize; |
| } |
| AddPage(column, pagePos + 1, newPage); |
| page.RowCount = rowPos; |
| } else { |
| if (column._pages[pagePos + 1].RowCount + size |
| > _pageSizeMax) //Split Page |
| { |
| SplitPageInsert(column, pagePos, rowPos, rows, size, addPages); |
| } else //Copy Page. |
| { |
| CopyMergePage(page, rowPos, rows, size, column._pages[pagePos + 1]); |
| } |
| } |
| } |
| } else { |
| //Add to Pages. |
| for (int r = rowPos; r < page.RowCount; r++) { |
| page.Rows[r].Index += (short)rows; |
| } |
| if (page.Offset + page.Rows[page.RowCount - 1].Index |
| >= _pageSizeMax) //Can not be larger than the max size of the page. |
| { |
| AdjustIndex(column, pagePos); |
| if (page.Offset + page.Rows[page.RowCount - 1].Index >= _pageSizeMax) { |
| pagePos = SplitPage(column, pagePos); |
| } |
| //IndexItem[] newRows = new IndexItem[GetSize(page.RowCount - page.Rows[r].Index)]; |
| //var newPage = new PageIndex(newRows, r); |
| //newPage.Index = (short)(pagePos + 1); |
| //TODO: MoveRows to next page. |
| } |
| |
| for (int p = pagePos + 1; p < column.PageCount; p++) { |
| if (column._pages[p].Offset + rows < _pageSize) { |
| column._pages[p].Offset += rows; |
| } else { |
| column._pages[p].Index++; |
| column._pages[p].Offset = (column._pages[p].Offset + rows) % _pageSize; |
| } |
| } |
| } |
| } |
| |
| private void SplitPageInsert( |
| ColumnIndex column, |
| int pagePos, |
| int rowPos, |
| int rows, |
| int size, |
| int addPages) { |
| var newRows = new IndexItem[GetSize(size)]; |
| var page = column._pages[pagePos]; |
| |
| var rStart = -1; |
| for (int r = rowPos; r < page.RowCount; r++) { |
| if (page.IndexExpanded - (page.Rows[r].Index + rows) > _pageSize) { |
| rStart = r; |
| break; |
| } |
| page.Rows[r].Index += (short)rows; |
| } |
| var rc = page.RowCount - rStart; |
| page.RowCount = rStart; |
| if (rc > 0) { |
| //Copy to a new page |
| var row = page.IndexOffset; |
| var newPage = CopyNew(page, rStart, rc); |
| var ix = (short)(page.Index + addPages); |
| var offset = page.IndexOffset + rows - (ix * _pageSize); |
| if (offset > _pageSize) { |
| ix += (short)(offset / _pageSize); |
| offset %= _pageSize; |
| } |
| newPage.Index = ix; |
| newPage.Offset = offset; |
| AddPage(column, pagePos + 1, newPage); |
| } |
| |
| //Copy from next Row |
| } |
| |
| private void CopyMergePage(PageIndex page, int rowPos, int rows, int size, PageIndex toPage) { |
| var startRow = page.IndexOffset + page.Rows[rowPos].Index + rows; |
| var newRows = new IndexItem[GetSize(toPage.RowCount + size)]; |
| page.RowCount -= size; |
| Array.Copy(page.Rows, rowPos, newRows, 0, size); |
| for (int r = 0; r < size; r++) { |
| newRows[r].Index += (short)(page.IndexOffset + rows - toPage.IndexOffset); |
| } |
| |
| Array.Copy(toPage.Rows, 0, newRows, size, toPage.RowCount); |
| toPage.Rows = newRows; |
| toPage.RowCount += size; |
| } |
| |
| private void MergePage(ColumnIndex column, int pagePos) { |
| PageIndex page1 = column._pages[pagePos]; |
| PageIndex page2 = column._pages[pagePos + 1]; |
| |
| var newPage = new PageIndex(page1, 0, page1.RowCount + page2.RowCount); |
| newPage.RowCount = page1.RowCount + page2.RowCount; |
| Array.Copy(page1.Rows, 0, newPage.Rows, 0, page1.RowCount); |
| Array.Copy(page2.Rows, 0, newPage.Rows, page1.RowCount, page2.RowCount); |
| for (int r = page1.RowCount; r < newPage.RowCount; r++) { |
| newPage.Rows[r].Index += (short)(page2.IndexOffset - page1.IndexOffset); |
| } |
| |
| column._pages[pagePos] = newPage; |
| column.PageCount--; |
| |
| if (column.PageCount > (pagePos + 1)) { |
| Array.Copy( |
| column._pages, |
| pagePos + 2, |
| column._pages, |
| pagePos + 1, |
| column.PageCount - (pagePos + 1)); |
| for (int p = pagePos + 1; p < column.PageCount; p++) { |
| column._pages[p].Index--; |
| column._pages[p].Offset += _pageSize; |
| } |
| } |
| } |
| |
| private PageIndex CopyNew(PageIndex pageFrom, int rowPos, int size) { |
| IndexItem[] newRows = new IndexItem[GetSize(size)]; |
| Array.Copy(pageFrom.Rows, rowPos, newRows, 0, size); |
| return new(newRows, size); |
| } |
| |
| internal static int GetSize(int size) { |
| var newSize = 256; |
| while (newSize < size) { |
| newSize <<= 1; |
| } |
| return newSize; |
| } |
| |
| private void AddCell(ColumnIndex columnIndex, int pagePos, int pos, short ix, T value) { |
| PageIndex pageItem = columnIndex._pages[pagePos]; |
| if (pageItem.RowCount == pageItem.Rows.Length) { |
| if (pageItem.RowCount |
| == _pageSizeMax) //Max size-->Split |
| { |
| pagePos = SplitPage(columnIndex, pagePos); |
| if (columnIndex._pages[pagePos - 1].RowCount > pos) { |
| pagePos--; |
| } else { |
| pos -= columnIndex._pages[pagePos - 1].RowCount; |
| } |
| pageItem = columnIndex._pages[pagePos]; |
| } else //Expand to double size. |
| { |
| var rowsTmp = new IndexItem[pageItem.Rows.Length << 1]; |
| Array.Copy(pageItem.Rows, 0, rowsTmp, 0, pageItem.RowCount); |
| pageItem.Rows = rowsTmp; |
| } |
| } |
| if (pos < pageItem.RowCount) { |
| Array.Copy(pageItem.Rows, pos, pageItem.Rows, pos + 1, pageItem.RowCount - pos); |
| } |
| pageItem.Rows[pos] = new() { |
| Index = ix, |
| IndexPointer = _values.Count, |
| }; |
| _values.Add(value); |
| pageItem.RowCount++; |
| } |
| |
| private int SplitPage(ColumnIndex columnIndex, int pagePos) { |
| var page = columnIndex._pages[pagePos]; |
| if (page.Offset != 0) { |
| var offset = page.Offset; |
| page.Offset = 0; |
| for (int r = 0; r < page.RowCount; r++) { |
| page.Rows[r].Index -= (short)offset; |
| } |
| } |
| //Find Split pos |
| int splitPos = 0; |
| for (int r = 0; r < page.RowCount; r++) { |
| if (page.Rows[r].Index > _pageSize) { |
| splitPos = r; |
| break; |
| } |
| } |
| var newPage = new PageIndex(page, 0, splitPos); |
| var nextPage = new PageIndex( |
| page, |
| splitPos, |
| page.RowCount - splitPos, |
| (short)(page.Index + 1), |
| page.Offset); |
| |
| for (int r = 0; r < nextPage.RowCount; r++) { |
| nextPage.Rows[r].Index = (short)(nextPage.Rows[r].Index - _pageSize); |
| } |
| |
| columnIndex._pages[pagePos] = newPage; |
| if (columnIndex.PageCount + 1 > columnIndex._pages.Length) { |
| var pageTmp = new PageIndex[columnIndex._pages.Length << 1]; |
| Array.Copy(columnIndex._pages, 0, pageTmp, 0, columnIndex.PageCount); |
| columnIndex._pages = pageTmp; |
| } |
| Array.Copy( |
| columnIndex._pages, |
| pagePos + 1, |
| columnIndex._pages, |
| pagePos + 2, |
| columnIndex.PageCount - pagePos - 1); |
| columnIndex._pages[pagePos + 1] = nextPage; |
| page = nextPage; |
| //pos -= PageSize; |
| columnIndex.PageCount++; |
| return pagePos + 1; |
| } |
| |
| private PageIndex AdjustIndex(ColumnIndex columnIndex, int pagePos) { |
| PageIndex page = columnIndex._pages[pagePos]; |
| //First Adjust indexes |
| if (page.Offset + page.Rows[0].Index >= _pageSize |
| || page.Offset >= _pageSize |
| || page.Rows[0].Index >= _pageSize) { |
| page.Index++; |
| page.Offset -= _pageSize; |
| } else if (page.Offset + page.Rows[0].Index <= -_pageSize |
| || page.Offset <= -_pageSize |
| || page.Rows[0].Index <= -_pageSize) { |
| page.Index--; |
| page.Offset += _pageSize; |
| } |
| return page; |
| } |
| |
| private void AddPage(ColumnIndex column, int pos, short index) { |
| AddPage(column, pos); |
| column._pages[pos] = new() { |
| Index = index, |
| }; |
| if (pos > 0) { |
| var pp = column._pages[pos - 1]; |
| if (pp.RowCount > 0 && pp.Rows[pp.RowCount - 1].Index > _pageSize) { |
| column._pages[pos].Offset = pp.Rows[pp.RowCount - 1].Index - _pageSize; |
| } |
| } |
| } |
| |
| /// <summary> |
| /// Add a new page to the collection |
| /// </summary> |
| /// <param name="column">The column</param> |
| /// <param name="pos">Position</param> |
| /// <param name="page">The new page object to add</param> |
| private void AddPage(ColumnIndex column, int pos, PageIndex page) { |
| AddPage(column, pos); |
| column._pages[pos] = page; |
| } |
| |
| /// <summary> |
| /// Add a new page to the collection |
| /// </summary> |
| /// <param name="column">The column</param> |
| /// <param name="pos">Position</param> |
| private void AddPage(ColumnIndex column, int pos) { |
| if (column.PageCount == column._pages.Length) { |
| var pageTmp = new PageIndex[column._pages.Length * 2]; |
| Array.Copy(column._pages, 0, pageTmp, 0, column.PageCount); |
| column._pages = pageTmp; |
| } |
| if (pos < column.PageCount) { |
| Array.Copy(column._pages, pos, column._pages, pos + 1, column.PageCount - pos); |
| } |
| column.PageCount++; |
| } |
| |
| private void AddColumn(int pos, int column) { |
| if (ColumnCount == _columnIndex.Length) { |
| var colTmp = new ColumnIndex[_columnIndex.Length * 2]; |
| Array.Copy(_columnIndex, 0, colTmp, 0, ColumnCount); |
| _columnIndex = colTmp; |
| } |
| if (pos < ColumnCount) { |
| Array.Copy(_columnIndex, pos, _columnIndex, pos + 1, ColumnCount - pos); |
| } |
| _columnIndex[pos] = new() { |
| Index = (short)(column), |
| }; |
| ColumnCount++; |
| } |
| |
| internal bool NextCell(ref int row, ref int col) { |
| return NextCell(ref row, ref col, 0, 0, ExcelPackage.MaxRows, ExcelPackage.MaxColumns); |
| } |
| |
| internal bool NextCell( |
| ref int row, |
| ref int col, |
| int minRow, |
| int minColPos, |
| int maxRow, |
| int maxColPos) { |
| if (minColPos >= ColumnCount) { |
| return false; |
| } |
| if (maxColPos >= ColumnCount) { |
| maxColPos = ColumnCount - 1; |
| } |
| var c = GetPosition(col); |
| if (c >= 0) { |
| if (c > maxColPos) { |
| if (col <= minColPos) { |
| return false; |
| } |
| col = minColPos; |
| return NextCell(ref row, ref col); |
| } |
| var r = GetNextCell(ref row, ref c, minColPos, maxRow, maxColPos); |
| col = _columnIndex[c].Index; |
| return r; |
| } |
| c = ~c; |
| if (c > _columnIndex[c].Index) { |
| if (col <= minColPos) { |
| return false; |
| } |
| col = minColPos; |
| return NextCell(ref row, ref col, minRow, minColPos, maxRow, maxColPos); |
| } |
| { |
| var r = GetNextCell(ref c, ref row, minColPos, maxRow, maxColPos); |
| col = _columnIndex[c].Index; |
| return r; |
| } |
| } |
| |
| internal bool GetNextCell( |
| ref int row, |
| ref int colPos, |
| int startColPos, |
| int endRow, |
| int endColPos) { |
| if (ColumnCount == 0) { |
| return false; |
| } |
| if (++colPos < ColumnCount && colPos <= endColPos) { |
| var r = _columnIndex[colPos].GetNextRow(row); |
| if (r |
| == row) //Exists next Row |
| { |
| return true; |
| } |
| int minRow, |
| minCol; |
| if (r > row) { |
| minRow = r; |
| minCol = colPos; |
| } else { |
| minRow = int.MaxValue; |
| minCol = 0; |
| } |
| |
| var c = colPos + 1; |
| while (c < ColumnCount && c <= endColPos) { |
| r = _columnIndex[c].GetNextRow(row); |
| if (r |
| == row) //Exists next Row |
| { |
| colPos = c; |
| return true; |
| } |
| if (r > row && r < minRow) { |
| minRow = r; |
| minCol = c; |
| } |
| c++; |
| } |
| c = startColPos; |
| if (row < endRow) { |
| row++; |
| while (c < colPos) { |
| r = _columnIndex[c].GetNextRow(row); |
| if (r |
| == row) //Exists next Row |
| { |
| colPos = c; |
| return true; |
| } |
| if (r > row && (r < minRow || (r == minRow && c < minCol)) && r <= endRow) { |
| minRow = r; |
| minCol = c; |
| } |
| c++; |
| } |
| } |
| |
| if (minRow == int.MaxValue || minRow > endRow) { |
| return false; |
| } |
| row = minRow; |
| colPos = minCol; |
| return true; |
| } |
| if (colPos <= startColPos || row >= endRow) { |
| return false; |
| } |
| colPos = startColPos - 1; |
| row++; |
| return GetNextCell(ref row, ref colPos, startColPos, endRow, endColPos); |
| } |
| |
| internal bool PrevCell(ref int row, ref int col) { |
| return PrevCell(ref row, ref col, 0, 0, ExcelPackage.MaxRows, ExcelPackage.MaxColumns); |
| } |
| |
| private bool PrevCell( |
| ref int row, |
| ref int col, |
| int minRow, |
| int minColPos, |
| int maxRow, |
| int maxColPos) { |
| if (minColPos >= ColumnCount) { |
| return false; |
| } |
| if (maxColPos >= ColumnCount) { |
| maxColPos = ColumnCount - 1; |
| } |
| var c = GetPosition(col); |
| if (c >= 0) { |
| if (c == 0) { |
| if (col >= maxColPos) { |
| return false; |
| } |
| if (row == minRow) { |
| return false; |
| } |
| row--; |
| col = maxColPos; |
| return PrevCell(ref row, ref col, minRow, minColPos, maxRow, maxColPos); |
| } |
| var ret = GetPrevCell(ref row, ref c, minRow, minColPos, maxColPos); |
| if (ret) { |
| col = _columnIndex[c].Index; |
| } |
| return ret; |
| } |
| c = ~c; |
| if (c == 0) { |
| if (col >= maxColPos || row <= 0) { |
| return false; |
| } |
| col = maxColPos; |
| row--; |
| return PrevCell(ref row, ref col, minRow, minColPos, maxRow, maxColPos); |
| } |
| { |
| var ret = GetPrevCell(ref row, ref c, minRow, minColPos, maxColPos); |
| if (ret) { |
| col = _columnIndex[c].Index; |
| } |
| return ret; |
| } |
| } |
| |
| internal bool GetPrevCell( |
| ref int row, |
| ref int colPos, |
| int startRow, |
| int startColPos, |
| int endColPos) { |
| if (ColumnCount == 0) { |
| return false; |
| } |
| if (--colPos >= startColPos) |
| // if (++colPos < ColumnCount && colPos <= endColPos) |
| { |
| var r = _columnIndex[colPos].GetNextRow(row); |
| if (r |
| == row) //Exists next Row |
| { |
| return true; |
| } |
| int minRow, |
| minCol; |
| if (r > row && r >= startRow) { |
| minRow = r; |
| minCol = colPos; |
| } else { |
| minRow = int.MaxValue; |
| minCol = 0; |
| } |
| |
| var c = colPos - 1; |
| if (c >= startColPos) { |
| while (c >= startColPos) { |
| r = _columnIndex[c].GetNextRow(row); |
| if (r |
| == row) //Exists next Row |
| { |
| colPos = c; |
| return true; |
| } |
| if (r > row && r < minRow && r >= startRow) { |
| minRow = r; |
| minCol = c; |
| } |
| c--; |
| } |
| } |
| if (row > startRow) { |
| c = endColPos; |
| row--; |
| while (c > colPos) { |
| r = _columnIndex[c].GetNextRow(row); |
| if (r |
| == row) //Exists next Row |
| { |
| colPos = c; |
| return true; |
| } |
| if (r > row && r < minRow && r >= startRow) { |
| minRow = r; |
| minCol = c; |
| } |
| c--; |
| } |
| } |
| if (minRow == int.MaxValue || startRow < minRow) { |
| return false; |
| } |
| row = minRow; |
| colPos = minCol; |
| return true; |
| } |
| colPos = ColumnCount; |
| row--; |
| if (row < startRow) { |
| return false; |
| } |
| return GetPrevCell(ref colPos, ref row, startRow, startColPos, endColPos); |
| } |
| } |
| |
| internal class CellsStoreEnumerator<T> : IEnumerable<T>, IEnumerator<T> { |
| private readonly CellStore<T> _cellStore; |
| private int row, |
| colPos; |
| private int[] pagePos, |
| cellPos; |
| private readonly int _startRow; |
| private readonly int _startCol; |
| private readonly int _endRow; |
| private readonly int _endCol; |
| private int minRow, |
| minColPos, |
| maxRow, |
| maxColPos; |
| |
| public CellsStoreEnumerator(CellStore<T> cellStore) |
| : this(cellStore, 0, 0, ExcelPackage.MaxRows, ExcelPackage.MaxColumns) {} |
| |
| public CellsStoreEnumerator( |
| CellStore<T> cellStore, |
| int startRow, |
| int startCol, |
| int endRow, |
| int endCol) { |
| _cellStore = cellStore; |
| |
| _startRow = startRow; |
| _startCol = startCol; |
| _endRow = endRow; |
| _endCol = endCol; |
| |
| Init(); |
| } |
| |
| internal void Init() { |
| minRow = _startRow; |
| maxRow = _endRow; |
| |
| minColPos = _cellStore.GetPosition(_startCol); |
| if (minColPos < 0) { |
| minColPos = ~minColPos; |
| } |
| maxColPos = _cellStore.GetPosition(_endCol); |
| if (maxColPos < 0) { |
| maxColPos = ~maxColPos - 1; |
| } |
| row = minRow; |
| colPos = minColPos - 1; |
| |
| var cols = maxColPos - minColPos + 1; |
| pagePos = new int[cols]; |
| cellPos = new int[cols]; |
| for (int i = 0; i < cols; i++) { |
| pagePos[i] = -1; |
| cellPos[i] = -1; |
| } |
| } |
| |
| internal int Row => row; |
| |
| internal int Column { |
| get { |
| if (colPos == -1) { |
| MoveNext(); |
| } |
| if (colPos == -1) { |
| return 0; |
| } |
| return _cellStore._columnIndex[colPos].Index; |
| } |
| } |
| |
| internal T Value { |
| get => _cellStore.GetValue(row, Column); |
| set => _cellStore.SetValue(row, Column, value); |
| } |
| |
| internal bool Next() { |
| return _cellStore.GetNextCell(ref row, ref colPos, minColPos, maxRow, maxColPos); |
| } |
| |
| public string CellAddress => ExcelCellBase.GetAddress(Row, Column); |
| |
| public IEnumerator<T> GetEnumerator() { |
| Reset(); |
| return this; |
| } |
| |
| IEnumerator IEnumerable.GetEnumerator() { |
| Reset(); |
| return this; |
| } |
| |
| public T Current => Value; |
| |
| public void Dispose() {} |
| |
| object IEnumerator.Current { |
| get { |
| Reset(); |
| return this; |
| } |
| } |
| |
| public bool MoveNext() { |
| return Next(); |
| } |
| |
| public void Reset() { |
| Init(); |
| } |
| } |
| |
| internal class FlagCellStore : CellStore<byte> { |
| internal void SetFlagValue(int row, int col, bool value, CellFlags cellFlags) { |
| CellFlags currentValue = (CellFlags)GetValue(row, col); |
| if (value) { |
| SetValue(row, col, (byte)(currentValue | cellFlags)); // add the CellFlag bit |
| } else { |
| SetValue(row, col, (byte)(currentValue & ~cellFlags)); // remove the CellFlag bit |
| } |
| } |
| |
| internal bool GetFlagValue(int row, int col, CellFlags cellFlags) { |
| return !(((byte)cellFlags & GetValue(row, col)) == 0); |
| } |
| } |