{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "<img src=\"states_outlines.png\" width=\"500\"/>\n",
    "\n",
    "# US states outlines\n",
    "\n",
    "The following proof of concept exercise illustrates how to combine\n",
    "dual canvas features in a relatively complex visualization.\n",
    "\n",
    "The visualization overlays state border polygons onto a map image\n",
    "and adds a mouse over event handler which shows the name of the state\n",
    "under the mouse in a floating dialog."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "# https://stackoverflow.com/questions/1814169/geographical-boundaries-of-states-provinces-google-maps-polygon\n",
    "\n",
    "# Parse State border polygons from XML\n",
    "xml_file = \"states.xml\"\n",
    "import xml.etree.ElementTree as ET\n",
    "tree = ET.parse(xml_file)\n",
    "root = tree.getroot()\n",
    "name_to_boundary = {}\n",
    "allpoints = []\n",
    "for state in root:\n",
    "    name = state.attrib[\"name\"]\n",
    "    #print name\n",
    "    boundary = []\n",
    "    for point in state:\n",
    "        lat = float(point.attrib[\"lat\"])\n",
    "        lng = float(point.attrib[\"lng\"])\n",
    "        #pt = [lat, lng]\n",
    "        pt = [lng, lat]\n",
    "        boundary.append(pt)\n",
    "        allpoints.append(pt)\n",
    "    name_to_boundary[name] = boundary"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": false,
    "deletable": true,
    "editable": true,
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# Draw a color map image with state polygons and borders\n",
    "from jp_doodle import dual_canvas\n",
    "from IPython.display import display\n",
    "\n",
    "states = dual_canvas.DualCanvasWidget(width=520, height=320)\n",
    "states.check_jquery()\n",
    "\n",
    "states.text(text=\"Longitude\", y=-60, x=250, align=\"center\", font=\"bold 20px Arial\",)\n",
    "states.text(text=\"Latitude\", y=150, x=-50, align=\"center\", degrees=90, font=\"bold 20px Arial\",)\n",
    "minlng = min(x[0] for x in allpoints)\n",
    "maxlng = max(x[0] for x in allpoints)\n",
    "minlat = min(x[1] for x in allpoints)\n",
    "maxlat = max(x[1] for x in allpoints)\n",
    "display(states)\n",
    "\n",
    "# image underlay\n",
    "earth_image = 'Earthmap1000x500.jpg'\n",
    "# local link does not work (as written) in Jupyter Lab\n",
    "earth_image = 'https://upload.wikimedia.org/wikipedia/commons/a/ac/Earthmap1000x500.jpg'\n",
    "def latitude_pixel(lat):\n",
    "    \"pixel from top edge = (90 - latitude) / 0,36\"\n",
    "    return (90.0 - lat) / 0.36\n",
    "def longitude_pixel(lng):\n",
    "    \"pixel from left hand side = (180 + longitude) / 0,36\"\n",
    "    return (180 + lng) / 0.36\n",
    "sx = longitude_pixel(minlng)\n",
    "sy = latitude_pixel(maxlat)\n",
    "sWidth = (maxlng - minlng) / 0.36\n",
    "sHeight = (maxlat - minlat) / 0.36\n",
    "\n",
    "\n",
    "states.name_image_url(\"earth\", earth_image)\n",
    "states.named_image(\"earth\", 0,0,500,300, degrees=0, \n",
    "                   sx=sx, sy=sy, sWidth=sWidth, sHeight=sHeight)\n",
    "\n",
    "# States polygons overlay.\n",
    "frame = states.frame_region(0,0,500,300,minlng, minlat, maxlng, maxlat)\n",
    "for name in name_to_boundary:\n",
    "    points = name_to_boundary[name]\n",
    "    # add semi-transparent filled polygon for state interior\n",
    "    frame.polygon(points=points, name=name, color=\"rgba(100,200,0,0.5)\")\n",
    "    # add unnamed state border\n",
    "    frame.polygon(points=points, color=\"#38f\", fill=False)\n",
    "\n",
    "# add reference axes\n",
    "frame.right_axis(\n",
    "    min_value= minlat,\n",
    "    max_value= maxlat,\n",
    "    max_tick_count= 6,\n",
    "    axis_origin= dict(x=maxlng+2, y=0),\n",
    "    tick_line_config= dict(color=\"#66f\"),\n",
    "    tick_text_config= dict(color=\"#875\"),\n",
    ")\n",
    "\n",
    "frame.bottom_axis(\n",
    "    min_value= minlng,\n",
    "    max_value= maxlng,\n",
    "    max_tick_count= 8,\n",
    "    axis_origin= dict(x=0, y= minlat-2),\n",
    "    tick_line_config= dict(color=\"#66f\"),\n",
    "    tick_text_config= dict(color=\"#875\", align=\"center\", valign=\"center\", degrees=0),\n",
    "    add_end_points= True,\n",
    ")\n",
    "\n",
    "#print minlng, minlat, maxlng, maxlat\n",
    "states.fit()\n",
    "\n",
    "# Use javascript to add mouse over event handling and a pop-up dialog.\n",
    "\n",
    "states.js_init(\"\"\"\n",
    "\n",
    "// Text info area and JQueryUI dialog\n",
    "var info = $(\"<div>info here</div>\").appendTo(element);\n",
    "var dialog = $(\"<div>dialog text</div>\").appendTo(element);\n",
    "dialog.dialog();\n",
    "\n",
    "// https://stackoverflow.com/questions/17358622/dialog-box-hide-and-show-jquery\n",
    "dialog.dialog(\"open\");\n",
    "// Using dialog.parent().hide() and *.show() avoids scrolling anomalies.\n",
    "dialog.parent().hide()\n",
    "\n",
    "element.text\n",
    "var last_name = null\n",
    "count = 0;\n",
    "var mouse_move = function(event) {\n",
    "    count += 1;\n",
    "    var name = event.canvas_name;\n",
    "    var pos = { my: \"left+10 top+10\", at: \"left bottom\", of: event }\n",
    "    dialog.dialog(\"option\", \"position\", pos);\n",
    "    if ((last_name) && ((last_name!=name) || event.type==\"mouseout\")) {\n",
    "        // obscure the interior)\n",
    "        element.change(last_name, {color: \"rgba(100,200,0,0.5)\"})\n",
    "        last_name = null;\n",
    "        dialog.parent().hide();\n",
    "    }\n",
    "    if (name) {\n",
    "        // make the interior transparent\n",
    "        element.change(name, {color: \"rgba(0,0,0,0)\"})\n",
    "        last_name = name;\n",
    "        dialog.html(\"<div>\"+name+\"</div>\");\n",
    "        dialog.parent().show();\n",
    "    }\n",
    "    info.html(\"<div>name=\"+name+\"; last_name=\"+last_name\n",
    "      +\"; count=\"+count+\"</div>\");\n",
    "}\n",
    "element.on_canvas_event(\"mousemove\", mouse_move);\n",
    "\"\"\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true,
    "deletable": true,
    "editable": true
   },
   "outputs": [],
   "source": [
    "# https://commons.wikimedia.org/wiki/File:Earthmap1000x500.jpg\n",
    "#pixel from top edge = (90 - latitude) / 0,36\n",
    "#pixel from left hand side = (180 + longitude) / 0,36"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": false,
    "deletable": true,
    "editable": true
   },
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 2",
   "language": "python",
   "name": "python2"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 2
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython2",
   "version": "2.7.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}