| /******************************************************************************* | 
 |  * 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		    Initial Release		       2009-10-01 | 
 |  * Jan Källman		    License changed GPL-->LGPL 2011-12-27 | 
 |  *******************************************************************************/ | 
 | using System; | 
 | using System.Xml; | 
 |  | 
 | namespace OfficeOpenXml | 
 | { | 
 | 	/// <summary> | 
 | 	/// Represents the different view states of the worksheet | 
 | 	/// </summary> | 
 | 	public class ExcelWorksheetView : XmlHelper | 
 | 	{ | 
 |         /// <summary> | 
 |         /// The worksheet panes after a freeze or split. | 
 |         /// </summary> | 
 |         public class ExcelWorksheetPanes : XmlHelper | 
 |         { | 
 |             XmlElement _selectionNode = null; | 
 |             internal ExcelWorksheetPanes(XmlNamespaceManager ns, XmlNode topNode) : | 
 |                 base(ns, topNode) | 
 |             { | 
 |                 if(topNode.Name=="selection") | 
 |                 { | 
 |                     _selectionNode=topNode as XmlElement; | 
 |                 } | 
 |             } | 
 |  | 
 |             const string _activeCellPath = "@activeCell"; | 
 |             /// <summary> | 
 |             /// Set the active cell. Must be set within the SelectedRange. | 
 |             /// </summary> | 
 |             public string ActiveCell | 
 |             { | 
 |                 get | 
 |                 { | 
 |                     string address = GetXmlNodeString(_activeCellPath); | 
 |                     if (address == "") | 
 |                     { | 
 |                         return "A1"; | 
 |                     } | 
 |                     return address; | 
 |                 } | 
 |                 set | 
 |                 { | 
 |                     int fromCol, fromRow, toCol, toRow; | 
 |                     if(_selectionNode==null) CreateSelectionElement(); | 
 |                     ExcelCellBase.GetRowColFromAddress(value, out fromRow, out fromCol, out toRow, out toCol); | 
 |                     SetXmlNodeString(_activeCellPath, value); | 
 |                     if (((XmlElement)TopNode).GetAttribute("sqref") == "") | 
 |                     { | 
 |  | 
 |                         SelectedRange = ExcelCellBase.GetAddress(fromRow, fromCol); | 
 |                     } | 
 |                     else | 
 |                     { | 
 |                         //TODO:Add fix for out of range here | 
 |                     } | 
 |                 } | 
 |             } | 
 |  | 
 |             private void CreateSelectionElement() | 
 |             { | 
 |  	            _selectionNode=TopNode.OwnerDocument.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                 TopNode.AppendChild(_selectionNode); | 
 |                 TopNode=_selectionNode;              | 
 |             } | 
 |             const string _selectionRangePath = "@sqref"; | 
 |             /// <summary> | 
 |             /// Selected Cells.Used in combination with ActiveCell | 
 |             /// </summary>         | 
 |             public string SelectedRange | 
 |             { | 
 |                 get | 
 |                 { | 
 |                     string address = GetXmlNodeString(_selectionRangePath); | 
 |                     if (address == "") | 
 |                     { | 
 |                         return "A1"; | 
 |                     } | 
 |                     return address; | 
 |                 } | 
 |                 set | 
 |                 { | 
 |                     int fromCol, fromRow, toCol, toRow; | 
 |                     if(_selectionNode==null) CreateSelectionElement(); | 
 |                     ExcelCellBase.GetRowColFromAddress(value, out fromRow, out fromCol, out toRow, out toCol); | 
 |                     SetXmlNodeString(_selectionRangePath, value); | 
 |                     if (((XmlElement)TopNode).GetAttribute("activeCell") == "") | 
 |                     { | 
 |  | 
 |                         ActiveCell = ExcelCellBase.GetAddress(fromRow, fromCol); | 
 |                     } | 
 |                     else | 
 |                     { | 
 |                         //TODO:Add fix for out of range here | 
 |                     } | 
 |                 } | 
 |             } | 
 |         } | 
 | 		private ExcelWorksheet _worksheet; | 
 |  | 
 | 		#region ExcelWorksheetView Constructor | 
 | 		/// <summary> | 
 | 		/// Creates a new ExcelWorksheetView which provides access to all the view states of the worksheet. | 
 | 		/// </summary> | 
 |         /// <param name="ns"></param> | 
 |         /// <param name="node"></param> | 
 |         /// <param name="xlWorksheet"></param> | 
 | 		internal ExcelWorksheetView(XmlNamespaceManager ns, XmlNode node,  ExcelWorksheet xlWorksheet) : | 
 |             base(ns, node) | 
 | 		{ | 
 |             _worksheet = xlWorksheet; | 
 |             SchemaNodeOrder = new string[] { "sheetViews", "sheetView", "pane", "selection" }; | 
 |             Panes = LoadPanes();  | 
 | 		} | 
 |  | 
 | 		#endregion | 
 |         private ExcelWorksheetPanes[] LoadPanes() | 
 |         { | 
 |             XmlNodeList nodes = TopNode.SelectNodes("//d:selection", NameSpaceManager); | 
 |             if(nodes.Count==0) | 
 |             { | 
 |                 return new ExcelWorksheetPanes[] { new ExcelWorksheetPanes(NameSpaceManager, TopNode) }; | 
 |             } | 
 |             else | 
 |             { | 
 |                 ExcelWorksheetPanes[] panes = new ExcelWorksheetPanes[nodes.Count]; | 
 |                 int i=0; | 
 |                 foreach(XmlElement elem in nodes) | 
 |                 { | 
 |                     panes[i++] = new ExcelWorksheetPanes(NameSpaceManager, elem); | 
 |                 } | 
 |                 return panes; | 
 |             } | 
 |         } | 
 | 		#region SheetViewElement | 
 | 		/// <summary> | 
 | 		/// Returns a reference to the sheetView element | 
 | 		/// </summary> | 
 | 		protected internal XmlElement SheetViewElement | 
 | 		{ | 
 | 			get  | 
 | 			{ | 
 | 				return (XmlElement)TopNode; | 
 | 			} | 
 | 		} | 
 | 		#endregion | 
 | 		#region TabSelected | 
 |         private XmlElement _selectionNode = null; | 
 |         private XmlElement SelectionNode | 
 |         { | 
 |             get | 
 |             { | 
 |                 _selectionNode = SheetViewElement.SelectSingleNode("//d:selection", _worksheet.NameSpaceManager) as XmlElement; | 
 |                 if (_selectionNode == null) | 
 |                 { | 
 |                     _selectionNode = _worksheet.WorksheetXml.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                     SheetViewElement.AppendChild(_selectionNode); | 
 |                 } | 
 |                 return _selectionNode; | 
 |             } | 
 |         } | 
 |         #endregion | 
 |         #region Public Methods & Properties | 
 |         /// <summary> | 
 |         /// The active cell. | 
 |         /// </summary> | 
 |         public string ActiveCell | 
 |         { | 
 |             get | 
 |             { | 
 |                 return Panes[Panes.GetUpperBound(0)].ActiveCell; | 
 |             } | 
 |             set  | 
 |             { | 
 |                 Panes[Panes.GetUpperBound(0)].ActiveCell = value; | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Selected Cells in the worksheet.Used in combination with ActiveCell | 
 |         /// </summary> | 
 |         public string SelectedRange | 
 |         { | 
 |             get | 
 |             { | 
 |                 return Panes[Panes.GetUpperBound(0)].SelectedRange; | 
 |             } | 
 |             set | 
 |             { | 
 |                 Panes[Panes.GetUpperBound(0)].SelectedRange = value; | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Indicates if the worksheet is selected within the workbook | 
 |         /// </summary> | 
 |         public bool TabSelected | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool("@tabSelected"); | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (value) | 
 |                 { | 
 |                     //    // ensure no other worksheet has its tabSelected attribute set to 1 | 
 |                     foreach (ExcelWorksheet sheet in _worksheet._package.Workbook.Worksheets) | 
 |                         sheet.View.TabSelected = false; | 
 |  | 
 |                     SheetViewElement.SetAttribute("tabSelected", "1"); | 
 |                     XmlElement bookView = _worksheet.Workbook.WorkbookXml.SelectSingleNode("//d:workbookView", _worksheet.NameSpaceManager) as XmlElement; | 
 |                     if (bookView != null) | 
 |                     { | 
 |                         bookView.SetAttribute("activeTab", (_worksheet.PositionID - 1).ToString()); | 
 |                     } | 
 |                 } | 
 |                 else | 
 |                     SetXmlNodeString("@tabSelected", "0"); | 
 |  | 
 |             } | 
 |         } | 
 |  | 
 | 		/// <summary> | 
 | 		/// Sets the view mode of the worksheet to pagelayout | 
 | 		/// </summary> | 
 | 		public bool PageLayoutView | 
 | 		{ | 
 | 			get | 
 | 			{ | 
 |                 return GetXmlNodeString("@view") == "pageLayout"; | 
 | 			} | 
 | 			set | 
 | 			{ | 
 |                 if (value) | 
 |                     SetXmlNodeString("@view", "pageLayout"); | 
 |                 else | 
 |                     SheetViewElement.RemoveAttribute("view"); | 
 | 			} | 
 | 		} | 
 |         /// <summary> | 
 |         /// Sets the view mode of the worksheet to pagebreak | 
 |         /// </summary> | 
 |         public bool PageBreakView | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeString("@view") == "pageBreakPreview"; | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (value) | 
 |                     SetXmlNodeString("@view", "pageBreakPreview"); | 
 |                 else | 
 |                     SheetViewElement.RemoveAttribute("view"); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Show gridlines in the worksheet | 
 |         /// </summary> | 
 |         public bool ShowGridLines  | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool("@showGridLines"); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeString("@showGridLines", value ? "1" : "0"); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Show the Column/Row headers (containg column letters and row numbers) | 
 |         /// </summary> | 
 |         public bool ShowHeaders | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool("@showRowColHeaders"); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeString("@showRowColHeaders", value ? "1" : "0"); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Window zoom magnification for current view representing percent values. | 
 |         /// </summary> | 
 |         public int ZoomScale | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeInt("@zoomScale"); | 
 |             } | 
 |             set | 
 |             { | 
 |                 if (value < 10 || value > 400) | 
 |                 { | 
 |                     throw new ArgumentOutOfRangeException("Zoome scale out of range (10-400)"); | 
 |                 } | 
 |                 SetXmlNodeString("@zoomScale", value.ToString()); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Flag indicating whether the sheet is in 'right to left' display mode. When in this mode,Column A is on the far right, Column B ;is one column left of Column A, and so on. Also,information in cells is displayed in the Right to Left format. | 
 |         /// </summary> | 
 |         public bool RightToLeft | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool("@rightToLeft"); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeString("@rightToLeft", value == true ? "1" : "0"); | 
 |             } | 
 |         } | 
 |         internal bool WindowProtection  | 
 |         { | 
 |             get | 
 |             { | 
 |                 return GetXmlNodeBool("@windowProtection",false); | 
 |             } | 
 |             set | 
 |             { | 
 |                 SetXmlNodeBool("@windowProtection",value,false); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Reference to the panes | 
 |         /// </summary> | 
 |         public ExcelWorksheetPanes[] Panes | 
 |         { | 
 |             get; | 
 |             internal set; | 
 |         } | 
 |         string _paneNodePath = "d:pane"; | 
 |         string _selectionNodePath = "d:selection"; | 
 |         /// <summary> | 
 |         /// Freeze the columns/rows to left and above the cell | 
 |         /// </summary> | 
 |         /// <param name="Row"></param> | 
 |         /// <param name="Column"></param> | 
 |         public void FreezePanes(int Row, int Column) | 
 |         { | 
 |             //TODO:fix this method to handle splits as well. | 
 |             if (Row == 1 && Column == 1) UnFreezePanes(); | 
 |             string sqRef = SelectedRange, activeCell = ActiveCell; | 
 |              | 
 |             XmlElement paneNode = TopNode.SelectSingleNode(_paneNodePath, NameSpaceManager) as XmlElement; | 
 |             if (paneNode == null) | 
 |             { | 
 |                 CreateNode(_paneNodePath); | 
 |                 paneNode = TopNode.SelectSingleNode(_paneNodePath, NameSpaceManager) as XmlElement; | 
 |             } | 
 |             paneNode.RemoveAll();   //Clear all attributes | 
 |             if (Column > 1) paneNode.SetAttribute("xSplit", (Column - 1).ToString()); | 
 |             if (Row > 1) paneNode.SetAttribute("ySplit", (Row - 1).ToString()); | 
 |             paneNode.SetAttribute("topLeftCell", ExcelCellBase.GetAddress(Row, Column)); | 
 |             paneNode.SetAttribute("state", "frozen"); | 
 |  | 
 |             RemoveSelection(); | 
 |  | 
 |             if (Row > 1 && Column==1) | 
 |             { | 
 |                 paneNode.SetAttribute("activePane", "bottomLeft"); | 
 |                 XmlElement sel=TopNode.OwnerDocument.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                 sel.SetAttribute("pane", "bottomLeft"); | 
 |                 if (activeCell != "") sel.SetAttribute("activeCell", activeCell); | 
 |                 if (sqRef != "") sel.SetAttribute("sqref", sqRef); | 
 |                 sel.SetAttribute("sqref", sqRef); | 
 |                 TopNode.InsertAfter(sel, paneNode); | 
 |             } | 
 |             else if (Column > 1 && Row == 1) | 
 |             { | 
 |                 paneNode.SetAttribute("activePane", "topRight"); | 
 |                 XmlElement sel = TopNode.OwnerDocument.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                 sel.SetAttribute("pane", "topRight"); | 
 |                 if (activeCell != "") sel.SetAttribute("activeCell", activeCell); | 
 |                 if (sqRef != "") sel.SetAttribute("sqref", sqRef); | 
 |                 TopNode.InsertAfter(sel, paneNode); | 
 |             } | 
 |             else | 
 |             { | 
 |                 paneNode.SetAttribute("activePane", "bottomRight"); | 
 |                 XmlElement sel1 = TopNode.OwnerDocument.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                 sel1.SetAttribute("pane", "topRight"); | 
 |                 string cell = ExcelCellBase.GetAddress(1, Column); | 
 |                 sel1.SetAttribute("activeCell", cell); | 
 |                 sel1.SetAttribute("sqref", cell); | 
 |                 paneNode.ParentNode.InsertAfter(sel1, paneNode); | 
 |  | 
 |                 XmlElement sel2 = TopNode.OwnerDocument.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                 cell = ExcelCellBase.GetAddress(Row, 1); | 
 |                 sel2.SetAttribute("pane", "bottomLeft"); | 
 |                 sel2.SetAttribute("activeCell", cell); | 
 |                 sel2.SetAttribute("sqref", cell); | 
 |                 sel1.ParentNode.InsertAfter(sel2, sel1); | 
 |  | 
 |                 XmlElement sel3 = TopNode.OwnerDocument.CreateElement("selection", ExcelPackage.schemaMain); | 
 |                 sel3.SetAttribute("pane", "bottomRight"); | 
 |                 if(activeCell!="") sel3.SetAttribute("activeCell", activeCell);                 | 
 |                 if(sqRef!="") sel3.SetAttribute("sqref", sqRef); | 
 |                 sel2.ParentNode.InsertAfter(sel3, sel2); | 
 |  | 
 |             } | 
 |             Panes=LoadPanes(); | 
 |         } | 
 |         private void RemoveSelection() | 
 |         { | 
 |             //Find selection nodes and remove them             | 
 |             XmlNodeList selections = TopNode.SelectNodes(_selectionNodePath, NameSpaceManager); | 
 |             foreach (XmlNode sel in selections) | 
 |             { | 
 |                 sel.ParentNode.RemoveChild(sel); | 
 |             } | 
 |         } | 
 |         /// <summary> | 
 |         /// Unlock all rows and columns to scroll freely | 
 |         /// /// </summary> | 
 |         public void UnFreezePanes() | 
 |         { | 
 |             string sqRef = SelectedRange, activeCell = ActiveCell; | 
 |  | 
 |             XmlElement paneNode = TopNode.SelectSingleNode(_paneNodePath, NameSpaceManager) as XmlElement; | 
 |             if (paneNode != null) | 
 |             { | 
 |                 paneNode.ParentNode.RemoveChild(paneNode); | 
 |             } | 
 |             RemoveSelection(); | 
 |  | 
 |             Panes=LoadPanes(); | 
 |  | 
 |             SelectedRange = sqRef; | 
 |             ActiveCell = activeCell; | 
 |         } | 
 |         #endregion | 
 |     } | 
 | } |