FindDockWindow.cs
// Copyright 2010 ESRI // // All rights reserved under the copyright laws of the United States // and applicable international laws, treaties, and conventions. // // You may freely redistribute and use this sample code, with or // without modification, provided you include the original copyright // notice and use restrictions. // // See the use restrictions. // using System; using System.ComponentModel; using System.IO; using System.Linq; using System.Net; using System.Windows.Forms; using System.Xml; using ESRI.ArcGISExplorer.Mapping; namespace GeoNamesFind { /// <summary> /// Implements a custom find DockWindow. /// </summary> /// <remarks> /// Uses the geonames web service to search for placenames. The web service takes /// a URL query and returns an XML file with the results. /// The results are parsed and Graphics are created from them. The Graphics are attached /// to the Tag property of the TreeNode and are drawn on the map when the node is checked. /// </remarks> public partial class FindDockWindow : ESRI.ArcGISExplorer.Application.DockWindow { WebClient _webClient; // used to download results from the web service string _resultFile; // temp file that contains the result xml GraphicCollection _graphics = null; /// <summary> /// Creates a new FindDockWindow instance. /// </summary> public FindDockWindow() { InitializeComponent(); _webClient = new WebClient(); _webClient.DownloadFileCompleted += new AsyncCompletedEventHandler(_webClient_DownloadCompleted); _graphics = ESRI.ArcGISExplorer.Application.Application.ActiveMapDisplay.Graphics; } #region Button Events /// <summary> /// Raised when the Clear button is clicked. /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void buttonClear_Click(object sender, EventArgs e) { ClearResults(); } /// <summary> /// Clears the results from the tree and graphics from the map. /// </summary> void ClearResults() { foreach (TreeNode node in treeView1.Nodes) { _graphics.Remove(node.Tag as Graphic); } treeView1.Nodes.Clear(); } /// <summary> /// Enables/disables the controls on the DockWindow. /// </summary> /// <param name="enable"></param> void EnableControls(bool enable) { buttonFind.Enabled = enable; buttonClear.Enabled = enable; textBox1.Enabled = enable; // if we are enabling the controls, select the search string // and give the TextBox focus // if (enable) { textBox1.SelectAll(); textBox1.Focus(); } } /// <summary> /// Raised when the Find button is clicked. Issues the query to the web service. /// </summary> private void buttonFind_Click(object sender, EventArgs e) { ClearResults(); EnableControls(false); // generate a temporary filename to hold the result // _resultFile = Path.GetTempFileName(); // build the url and make the async call to execute the query (just get the first 10 matches) // string url = "http://ws.geonames.org/search?q=" + textBox1.Text + "&maxRows=10&style=full"; _webClient.DownloadFileAsync(new Uri(url), _resultFile); } #endregion #region WebClient Events /// <summary> /// Raised when the xml result file has been downloaded. /// </summary> void _webClient_DownloadCompleted(object sender, AsyncCompletedEventArgs e) { // read the xml file // XmlDocument doc = new XmlDocument(); doc.Load(_resultFile); foreach (XmlNode xmlNode in doc.SelectNodes("/geonames/geoname")) { double lat = double.Parse(xmlNode.SelectSingleNode("lat").InnerText); double lon = double.Parse(xmlNode.SelectSingleNode("lng").InnerText); double elev = double.NaN; double.TryParse(xmlNode.SelectSingleNode("elevation").InnerText, out elev); Graphic graphic = CreateGraphic(xmlNode.SelectSingleNode("name").InnerText, xmlNode.SelectSingleNode("adminName1").InnerText, xmlNode.SelectSingleNode("countryName").InnerText, lat, lon, elev); treeView1.Nodes.Add(new TreeNode(graphic.Label) { Tag = graphic }); } EnableControls(true); // zoom to the first node in the tree // if (treeView1.Nodes.Count > 0) ZoomTo(treeView1.Nodes[0]); } /// <summary> /// Zooms to the location that corresponds to the specified TreeNode and makes it visible. /// </summary> /// <param name="item"></param> void ZoomTo(TreeNode node) { node.Checked = true; this.Update(); // force redraw of the DockView before the flying to the item this.Cursor = Cursors.AppStarting; Graphic g = node.Tag as Graphic; if (g != null) ESRI.ArcGISExplorer.Application.Application.ActiveMapDisplay.ZoomTo(g.Geometry); this.Cursor = Cursors.Default; } /// <summary> /// Creates a Graphic using the specified names and location. /// </summary> Graphic CreateGraphic(string name, string adminName, string countryName, double lat, double lon, double elev) { Graphic graphic = new Graphic(new ESRI.ArcGISExplorer.Geometry.Point(lon, lat, elev), Symbol.Marker.Stickpin.White); if (double.IsNaN(elev)) // is the elevation undefined? graphic.Placement3D = Placement3D.AttachToSurface; else graphic.Placement3D = Placement3D.AbsoluteElevation; if (!string.IsNullOrEmpty(adminName)) name += ", " + adminName; if (!string.IsNullOrEmpty(countryName)) name += ", " + countryName; graphic.Label = name; return graphic; } #endregion #region TreeView Events /// <summary> /// Raised when a TreeNode is checked or unchecked. /// </summary> private void treeView1_AfterCheck(object sender, TreeViewEventArgs e) { Graphic g = e.Node.Tag as Graphic; if (g == null) return; // if the node is checked, show the graphic on the map, otherwise, hide it. if (_graphics.Contains(g)) { g.Visible = (e.Node.Checked); } else { // If this is the first time that the graphic is shown, then add it to the Map. // It will be visible by default. _graphics.Add(g); } } /// <summary> /// Raised when an tree node is double-clicked. /// </summary> private void treeView1_NodeMouseDoubleClick(object sender, TreeNodeMouseClickEventArgs e) { ZoomTo(e.Node); } #endregion #region TextBox Events /// <summary> /// Raised when a key is pressed in the TextBox control. /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void textBox1_KeyDown(object sender, KeyEventArgs e) { if (e.KeyCode == Keys.Return) { e.Handled = true; buttonFind_Click(this, EventArgs.Empty); } } #endregion #region Context Menu Events /// <summary> /// Delete context menu item. /// </summary> private void deleteMenuItem_Click(object sender, EventArgs e) { if (treeView1.SelectedNode == null) return; // Remove tree node and associated graphic from the map. Graphic g = treeView1.SelectedNode.Tag as Graphic; if (g != null) _graphics.Remove(g); treeView1.SelectedNode.Remove(); } /// <summary> /// Move To Map context menu item - creates a note for the selected /// TreeNode and adds it to the map. /// </summary> private void moveToMapMenuItem_Click(object sender, EventArgs e) { if (treeView1.SelectedNode == null) return; Map map = ESRI.ArcGISExplorer.Application.Application.ActiveMapDisplay.Map; Graphic g = treeView1.SelectedNode.Tag as Graphic; treeView1.SelectedNode.Checked = false; // causes the associated graphic to be removed from the map _graphics.Remove(g); if (g != null) { Note note = new Note(g.Label, g.Geometry, Symbol.Marker.Stickpin.Orange); // change the symbol note.Viewpoint = new Viewpoint(g.Geometry.GetEnvelope()); map.ChildItems.Insert(0, note); // add the note to the map ESRI.ArcGISExplorer.Application.Application.SelectedItems.Select(note); } treeView1.SelectedNode.Remove(); } /// <summary> /// Delete All context menu item. /// </summary> private void deleteAllMenuItem_Click(object sender, EventArgs e) { ClearResults(); } /// <summary> /// Raised before the context menu is shown - enable/disable items. /// </summary> private void contextMenuStrip1_Opening(object sender, CancelEventArgs e) { // enable/disable the context menu items deleteMenuItem.Enabled = treeView1.SelectedNode != null; moveToMapMenuItem.Enabled = treeView1.SelectedNode != null; deleteAllMenuItem.Enabled = treeView1.Nodes.Count > 0; } #endregion } }