30239-notes/12.gis/slides.html

329 lines
118 KiB
HTML
Raw Normal View History

2024-11-13 23:50:20 +00:00
<!DOCTYPE html><html lang="en-US"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width,height=device-height,initial-scale=1.0"><meta name="apple-mobile-web-app-capable" content="yes"><meta http-equiv="X-UA-Compatible" content="ie=edge"><meta property="og:type" content="website"><meta name="twitter:card" content="summary"><style>@media screen{body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button,body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-note-container button{appearance:none;background-color:initial;border:0;color:inherit;cursor:pointer;font-size:inherit;opacity:.8;outline:none;padding:0;transition:opacity .2s linear;-webkit-tap-highlight-color:transparent}body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button:disabled,body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button:disabled,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button:disabled,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-note-container button:disabled{cursor:not-allowed;opacity:.15!important}body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button:hover,body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button:hover,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button:hover,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-note-container button:hover{opacity:1}body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button:hover:active,body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button:hover:active,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button:hover:active,body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-note-container button:hover:active{opacity:.6}body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button:hover:not(:disabled),body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button:hover:not(:disabled),body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button:hover:not(:disabled),body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-note-container button:hover:not(:disabled){transition:none}body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button[data-bespoke-marp-osc=prev],body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button[data-bespoke-marp-osc=prev],body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button.bespoke-marp-presenter-info-page-prev{background:#0000 url("") no-repeat 50%;background-size:contain;overflow:hidden;text-indent:100%;white-space:nowrap}body[data-bespoke-view=""] .bespoke-marp-parent>.bespoke-marp-osc>button[data-bespoke-marp-osc=next],body[data-bespoke-view=next] .bespoke-marp-parent>.bespoke-marp-osc>button[data-bespoke-marp-osc=next],body[data-bespoke-view=presenter] .bespoke-marp-presenter-container .bespoke-marp-presenter-info-container button.bespoke-marp-presenter-info-page-next{background:#0000 url("
2024-11-11 01:01:51 +00:00
* Marp default theme.
*
* @theme default
* @author Yuki Hattori
*
* @auto-scaling true
* @size 16:9 1280px 720px
* @size 4:3 960px 720px
2024-11-13 23:50:20 +00:00
*/div#\:\$p > svg > foreignObject > section{--base-size-4:calc(var(--marpit-root-font-size, 1rem) * 0.25);--base-size-8:calc(var(--marpit-root-font-size, 1rem) * 0.5);--base-size-16:calc(var(--marpit-root-font-size, 1rem) * 1);--base-text-weight-normal:400;--base-text-weight-medium:500;--base-text-weight-semibold:600;--fontStack-monospace:ui-monospace, SFMono-Regular, SF Mono, Menlo, Consolas, Liberation Mono, monospace;}div#\:\$p > svg > foreignObject > section [data-theme=light],div#\:\$p > svg > foreignObject > section{color-scheme:light;--focus-outlineColor:#0969da;--fgColor-default:#1f2328;--fgColor-muted:#636c76;--fgColor-accent:#0969da;--fgColor-success:#1a7f37;--fgColor-attention:#9a6700;--fgColor-danger:#d1242f;--fgColor-done:#8250df;--bgColor-default:#fff;--bgColor-muted:#f6f8fa;--bgColor-neutral-muted:#afb8c133;--bgColor-attention-muted:#fff8c5;--borderColor-default:#d0d7de;--borderColor-muted:#d0d7deb3;--borderColor-neutral-muted:#afb8c133;--borderColor-accent-emphasis:#0969da;--borderColor-success-emphasis:#1a7f37;--borderColor-attention-emphasis:#bf8700;--borderColor-danger-emphasis:#cf222e;--borderColor-done-emphasis:#8250df;--color-prettylights-syntax-comment:#57606a;--color-prettylights-syntax-constant:#0550ae;--color-prettylights-syntax-constant-other-reference-link:#0a3069;--color-prettylights-syntax-entity:#6639ba;--color-prettylights-syntax-storage-modifier-import:#24292f;--color-prettylights-syntax-entity-tag:#0550ae;--color-prettylights-syntax-keyword:#cf222e;--color-prettylights-syntax-string:#0a3069;--color-prettylights-syntax-variable:#953800;--color-prettylights-syntax-brackethighlighter-unmatched:#82071e;--color-prettylights-syntax-brackethighlighter-angle:#57606a;--color-prettylights-syntax-invalid-illegal-text:#f6f8fa;--color-prettylights-syntax-invalid-illegal-bg:#82071e;--color-prettylights-syntax-carriage-return-text:#f6f8fa;--color-prettylights-syntax-carriage-return-bg:#cf222e;--color-prettylights-syntax-string-regexp:#116329;--color-prettylights-syntax-markup-list:#3b2300;--color-prettylights-syntax-markup-heading:#0550ae;--color-prettylights-syntax-markup-italic:#24292f;--color-prettylights-syntax-markup-bold:#24292f;--color-prettylights-syntax-markup-deleted-text:#82071e;--color-prettylights-syntax-markup-deleted-bg:#ffebe9;--color-prettylights-syntax-markup-inserted-text:#116329;--color-prettylights-syntax-markup-inserted-bg:#dafbe1;--color-prettylights-syntax-markup-changed-text:#953800;--color-prettylights-syntax-markup-changed-bg:#ffd8b5;--color-prettylights-syntax-markup-ignored-text:#eaeef2;--color-prettylights-syntax-markup-ignored-bg:#0550ae;--color-prettylights-syntax-meta-diff-range:#8250df;--color-prettylights-syntax-sublimelinter-gutter-mark:#8c959f;}div#\:\$p > svg > foreignObject > section [data-theme=dark],div#\:\$p > svg > foreignObject > section:where(.invert){color-scheme:dark;--focus-outlineColor:#1f6feb;--fgColor-default:#e6edf3;--fgColor-muted:#8d96a0;--fgColor-accent:#4493f8;--fgColor-success:#3fb950;--fgColor-attention:#d29922;--fgColor-danger:#f85149;--fgColor-done:#ab7df8;--bgColor-default:#0d1117;--bgColor-muted:#161b22;--bgColor-neutral-muted:#6e768166;--bgColor-attention-muted:#bb800926;--borderColor-default:#30363d;--borderColor-muted:#30363db3;--borderColor-neutral-muted:#6e768166;--borderColor-accent-emphasis:#1f6feb;--borderColor-success-emphasis:#238636;--borderColor-attention-emphasis:#9e6a03;--borderColor-danger-emphasis:#da3633;--borderColor-done-emphasis:#8957e5;--color-prettylights-syntax-comment:#8b949e;--color-prettylights-syntax-constant:#79c0ff;--color-prettylights-syntax-constant-other-reference-link:#a5d6ff;--color-prettylights-syntax-entity:#d2a8ff;--color-prettylights-syntax-storage-modifier-import:#c9d1d9;--color-prettylights-syntax-entity-tag:#7ee787;--color-prettylights-syntax-keyword:#ff7b72;--color-prettylights-syntax-string:#a5d6ff;--color-prettylights-syntax-variable:#ffa657;--color-prettylights-syntax-brackethighlighter-unmatched:#f85149;--color-prettylights-syntax-brackethighlighter-angle:#8b949e;--color-prettylights-syntax-inv
</style></head><body><div class="bespoke-marp-osc"><button data-bespoke-marp-osc="prev" tabindex="-1" title="Previous slide">Previous slide</button><span data-bespoke-marp-osc="page"></span><button data-bespoke-marp-osc="next" tabindex="-1" title="Next slide">Next slide</button><button data-bespoke-marp-osc="fullscreen" tabindex="-1" title="Toggle fullscreen (f)">Toggle fullscreen</button><button data-bespoke-marp-osc="presenter" tabindex="-1" title="Open presenter view (p)">Open presenter view</button></div><div id=":$p"><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="1" lang="en-US">
2024-11-11 01:01:51 +00:00
<h1 id="gis-fundamentals">GIS Fundamentals</h1>
<h2 id="capp-30239">CAPP 30239</h2>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="2" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="today">Today</h2>
<p>Establish a baseline understanding of GIS data.</p>
<p>(Not focused on Data Viz)</p>
2024-11-13 23:50:20 +00:00
<h3 id="next">Next</h3>
<p>Browser Mapping Libraries</p>
2024-11-11 01:01:51 +00:00
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section lang="en-US" data-marpit-advanced-background="background" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;"><div data-marpit-advanced-background-container="true" data-marpit-advanced-background-direction="horizontal"><figure style="background-image:url(&quot;282eb677-c3c3-4d85-a652-5b1b7eb30c8d.jpg&quot;);background-size:contain;"></figure></div></section></foreignObject><foreignObject width="50%" height="720"><section id="3" lang="en-US" data-marpit-advanced-background="content" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;">
2024-11-11 01:01:51 +00:00
<h2 id="gis-fundamentals-1">GIS Fundamentals</h2>
<p><strong>GIS</strong> - Geographic Information System - Specialized set of tools for dealing with geospatial data.</p>
<h3 id="whats-special-about-gis">What's special about GIS?</h3>
<p><br />
(source: NASA.gov)</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject><foreignObject width="1280" height="720" data-marpit-advanced-background="pseudo"><section lang="en-US" data-marpit-advanced-background="pseudo" data-marpit-advanced-background-split="right" style=""></section></foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section lang="en-US" data-marpit-advanced-background="background" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;"><div data-marpit-advanced-background-container="true" data-marpit-advanced-background-direction="horizontal"><figure style="background-image:url(&quot;a4c744e0-7b3b-4a1e-9989-cf67a9b53583.png&quot;);background-size:contain;"></figure></div></section></foreignObject><foreignObject width="50%" height="720"><section id="4" lang="en-US" data-marpit-advanced-background="content" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;">
2024-11-11 01:01:51 +00:00
<h2 id="quantity-of-data-coastline-paradox">Quantity of data: coastline paradox</h2>
<p><br />
(source: Coastline Paradox on Wikipedia)</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject><foreignObject width="1280" height="720" data-marpit-advanced-background="pseudo"><section lang="en-US" data-marpit-advanced-background="pseudo" data-marpit-advanced-background-split="right" style=""></section></foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="5" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="gis-data-caveats">GIS Data Caveats</h2>
<h3 id="resolution">Resolution</h3>
<p>We have to decide what resolution is important. Even at a fairly low fidelity, a shape is likely to have tens of thousands of points.</p>
<h3 id="projection">Projection</h3>
<p>The earth is not flat, but we typically work in (x, y, [altitude]). This requires <strong>projection</strong>.</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section lang="en-US" data-marpit-advanced-background="background" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;"><div data-marpit-advanced-background-container="true" data-marpit-advanced-background-direction="horizontal"><figure style="background-image:url(&quot;f65f091f-54e8-4249-833f-a5ff19bf9dff.JPG&quot;);"></figure></div></section></foreignObject><foreignObject width="50%" height="720"><section id="6" lang="en-US" data-marpit-advanced-background="content" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;">
2024-11-11 01:01:51 +00:00
<p><strong>What's wrong with this picture?</strong></p>
<p><br />
Mercator Projection, Wikipedia</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject><foreignObject width="1280" height="720" data-marpit-advanced-background="pseudo"><section lang="en-US" data-marpit-advanced-background="pseudo" data-marpit-advanced-background-split="right" style=""></section></foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="7" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="africa-is-14x-larger">Africa is 14x larger</h2>
<p><img src="d63823a9-23a1-4ac3-999c-6a72798236e5.jpg" alt="greenland-on-africa.jpg" /><br />
<a href="https://mortenjonassen.dk/maps/greenland-vs-africa-size-comparison">https://mortenjonassen.dk/maps/greenland-vs-africa-size-comparison</a></p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="8" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="spatial-data-types">Spatial Data Types</h2>
<table>
<thead>
<tr>
<th></th>
<th>Raster</th>
<th>Vector</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>Image Format</strong></td>
<td>JPEG/PNG</td>
<td>SVG/PDF</td>
</tr>
<tr>
<td><strong>GIS File Types</strong></td>
<td>GeoTIFF/NetCDF</td>
<td>Shapefile/GeoJSON/Geopackage/KML</td>
</tr>
<tr>
<td><strong>Example Data</strong></td>
<td>Satellite imagery</td>
<td>- Points: cities<br />- Lines: roads<br />- Polygons: land parcels</td>
</tr>
<tr>
<td><strong>Typical Tool</strong></td>
<td>Camera</td>
<td>GPS</td>
</tr>
</tbody>
</table>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="9" lang="en-US">
2024-11-11 01:01:51 +00:00
<table>
<thead>
<tr>
<th></th>
<th>Raster</th>
<th>Vector</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>Pros</strong></td>
<td>Efficient for continuous surfaces (can sample any point)</td>
<td>Arbitrary precision; Ease of calculation; Network efficiency</td>
</tr>
<tr>
<td><strong>Cons</strong></td>
<td>Limited by image resolution (sensor &amp; storage)</td>
<td>Harder to create as specific relationships must be calculated and defined</td>
</tr>
</tbody>
</table>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="10" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="gis-vector-data-types">GIS Vector Data Types</h2>
<ul>
<li><strong>Point</strong> - <code>(lat, lon)</code> or <code>(lat, lon, altitude)</code> - example: address, tree, bus stop</li>
<li><strong>Line</strong> aka <strong>Polyline</strong> - series of ordered, connected points - examples: roads, rivers</li>
<li><strong>Polygon</strong> - enclosed areas - lakes, land parcels, county</li>
</ul>
<p>Container Types:</p>
<ul>
<li><strong>MultiPoint</strong> - example: fleet location, fire hydrants, GPS pings</li>
<li><strong>MultiLineString</strong> - example: branching transit line</li>
<li><strong>MultiPolygon</strong> - example: islands</li>
</ul>
<p>This allows us to think about a group of related polygons as one:<br />
e.g. does this storm path (Polyline) intersect with Hawaii (MultiPolygon)?</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="11" lang="en-US">
2024-11-11 01:01:51 +00:00
<h3 id="coordinate-reference-systems">Coordinate Reference Systems</h3>
<p>These systems define what the <code>x</code> and <code>y</code> values actually mean in a given file.</p>
<p>Two kinds:</p>
<ul>
<li><strong>Geographic Coordinate Systems</strong> - attempt to map a lat, lon to every point on Earth</li>
<li><strong>Projected Coordinate Systems</strong> - attempt to project earth's curved surface onto a plane (as needed for drawing)</li>
</ul>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="12" lang="en-US">
2024-11-11 01:01:51 +00:00
<p>Two kinds, infinite variations.</p>
<p><strong>Why are there multiple of each?</strong></p>
<p>GCS - regional or local variations in earth's curvature, and different estimates as to exact size of earth.<br />
PCS - no perfect mapping of sphere to x,y plane exists, as seen with Mercator projection, accuracy in one area of globe means distortion in others</p>
<p><strong>WGS84</strong> is used in GPS and therefore is by far the most common GCS.</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section lang="en-US" data-marpit-advanced-background="background" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;"><div data-marpit-advanced-background-container="true" data-marpit-advanced-background-direction="horizontal"><figure style="background-image:url(&quot;392c3b25-9766-47f9-b7af-8e1ffab814f9.png&quot;);"></figure></div></section></foreignObject><foreignObject width="50%" height="720"><section id="13" lang="en-US" data-marpit-advanced-background="content" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;">
2024-11-11 01:01:51 +00:00
<p>PCS is not so uniform, but most common is the <strong>UTM</strong> is the Transverse Mercator projection, which minimizes distortion within a region via a set of guarantees.</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject><foreignObject width="1280" height="720" data-marpit-advanced-background="pseudo"><section lang="en-US" data-marpit-advanced-background="pseudo" data-marpit-advanced-background-split="right" style=""></section></foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section lang="en-US" data-marpit-advanced-background="background" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;"><div data-marpit-advanced-background-container="true" data-marpit-advanced-background-direction="horizontal"><figure style="background-image:url(&quot;a3585f84-8cf4-451c-b079-0bb2ac338c6d.jpg&quot;);"></figure></div></section></foreignObject><foreignObject width="50%" height="720"><section id="14" lang="en-US" data-marpit-advanced-background="content" data-marpit-advanced-background-split="right" style="--marpit-advanced-background-split:50%;">
2024-11-11 01:01:51 +00:00
<p>Localities will often publish in a more local projections:</p>
2024-11-13 23:50:20 +00:00
<p>SPCS - State Plane Coordinate System, actually 125 zones with their own system across US. They are fine-tuned to fit the shape of each state to maximize accuracy. (Some use transverse Mercator, some use a Lambert conformal conic projection, all with different parameters.)</p>
2024-11-11 01:01:51 +00:00
<p><br />
(ce: conservation.ca.gov)</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject><foreignObject width="1280" height="720" data-marpit-advanced-background="pseudo"><section lang="en-US" data-marpit-advanced-background="pseudo" data-marpit-advanced-background-split="right" style=""></section></foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section lang="en-US" data-marpit-advanced-background="background" data-marpit-advanced-background-split="left" style="--marpit-advanced-background-split:50%;"><div data-marpit-advanced-background-container="true" data-marpit-advanced-background-direction="horizontal"><figure style="background-image:url(&quot;bad-proj.jpg&quot;);background-size:contain;"></figure></div></section></foreignObject><foreignObject width="50%" height="720" x="50%"><section id="15" lang="en-US" data-marpit-advanced-background="content" data-marpit-advanced-background-split="left" style="--marpit-advanced-background-split:50%;">
<h2 id="correct-projection-is-essential">Correct Projection is Essential</h2>
<p>In practice, it is <strong>essential</strong> that you know what CRS your data is in and what CRS you are displaying it in.</p>
<p>Failure leads to incorrect results, heavily distorted visualizations, and real-world trouble.</p>
2024-11-11 01:01:51 +00:00
</section>
2024-11-13 23:50:20 +00:00
</foreignObject><foreignObject width="1280" height="720" data-marpit-advanced-background="pseudo"><section lang="en-US" data-marpit-advanced-background="pseudo" data-marpit-advanced-background-split="left" style=""></section></foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="16" lang="en-US">
2024-11-11 01:01:51 +00:00
<h3 id="features">Features</h3>
<p>Another property of most GIS filetypes is that they have <strong>features</strong>. You can think of features as a database or spreadsheet of attributes associated with a given shape. It might take the form:</p>
<table>
<thead>
<tr>
<th>County Name</th>
<th>Population (2020)</th>
<th>Area (sq mi)</th>
<th>County Seat</th>
<th>Georeference</th>
</tr>
</thead>
<tbody>
<tr>
<td>Cook</td>
<td>5,275,541</td>
<td>1,635</td>
<td>Chicago</td>
<td>{...}</td>
</tr>
<tr>
<td>DuPage</td>
<td>932,877</td>
<td>334</td>
<td>Wheaton</td>
<td>{...}</td>
</tr>
<tr>
<td>Lake</td>
<td>714,342</td>
<td>448</td>
<td>Waukegan</td>
<td>{...}</td>
</tr>
<tr>
<td>Will</td>
<td>690,743</td>
<td>837</td>
<td>Joliet</td>
<td>{...}</td>
</tr>
<tr>
<td>Kane</td>
<td>516,522</td>
<td>524</td>
<td>Geneva</td>
<td>{...}</td>
</tr>
</tbody>
</table>
<p>Georeference column is either the geometry definition itself, or a foreign key (primary ID) for the given datum.</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="17" lang="en-US">
2024-11-11 01:01:51 +00:00
<h3 id="formats">Formats</h3>
<ul>
<li>
<p><strong>shapefile</strong> - actually a bunch of files that need to live in the same directory</p>
<ul>
<li>.shp - geometries</li>
<li>.shx - indexes for searching</li>
<li>.dbf - csv-like features</li>
<li>.prj - projection (CRS)</li>
<li>.sbn/.sbx/.shp.xml/.cpg - additional optional files that provide more metadata and features</li>
</ul>
</li>
<li>
<p><strong>GeoJSON/TopoJSON</strong>: JSON representation of features and geometries</p>
</li>
<li>
<p><strong>KML</strong>: Google XML-based format, mostly out of favor but still commonly found on GIS sites</p>
</li>
<li>
<p><strong>Geopackage</strong>: Still uncommon, sqlite based successor to shapefile.</p>
</li>
</ul>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="18" data-marpit-fragments="6" lang="en-US">
2024-11-11 01:01:51 +00:00
<h3 id="spatial-queries">Spatial Queries</h3>
<p>These form the common spatial queries, which mostly take the form:</p>
<p><code>operation(geometry1, geometry2, optional_tolerance) -&gt; geometry_result</code></p>
<p>Example operations: <code>nearest</code>, <code>contains</code>, <code>adjacent_to</code></p>
<p>Let's convert all of these to GIS terms:</p>
<ul>
<li data-marpit-fragment="1">Where is the nearest Waffle House?
<ul>
<li data-marpit-fragment="2">nearest(my_location, waffle_house_multipoint)</li>
</ul>
</li>
<li data-marpit-fragment="3">What congressional district am I in?
<ul>
<li data-marpit-fragment="4">contains(congressional_district_multipoly, my_location)</li>
</ul>
</li>
<li data-marpit-fragment="5">What cool attractions are close to my drive?
<ul>
<li data-marpit-fragment="6">nearest(my_route, roadside_attractions_multipoint, tolerance=10miles)</li>
</ul>
</li>
</ul>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="19" data-marpit-fragments="9" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="spatial-overlays">Spatial Overlays</h2>
<ul>
<li data-marpit-fragment="1">
<p>Union: combine two or more geometries into one</p>
</li>
<li data-marpit-fragment="2">
<p>Intersection: identify overlap (IL &amp; fresh water)</p>
</li>
<li data-marpit-fragment="3">
<p>Difference: Cook County - Chicago</p>
</li>
<li data-marpit-fragment="4">
<p>What part of the US is either national park <em>or</em> state park?</p>
<ul>
<li data-marpit-fragment="5"><code>union(national_parks_mp, state_parks_mp)</code></li>
</ul>
</li>
<li data-marpit-fragment="6">
<p>How much of Illinois is fresh water?</p>
<ul>
<li data-marpit-fragment="7"><code>intersection(il_boundaries, fresh_water_mp)</code></li>
</ul>
</li>
<li data-marpit-fragment="8">
<p>What parts of Cook County are <em>not</em> within Chicago?</p>
<ul>
<li data-marpit-fragment="9"><code>difference(cook_co, chicago)</code></li>
</ul>
</li>
</ul>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="20" lang="en-US">
2024-11-11 01:01:51 +00:00
<h3 id="buffering--joins">Buffering &amp; Joins</h3>
<p>Some other common functionality:</p>
<p><strong>Buffering</strong> is taking an existing shape and pushing the edge out by some distance. This turns a point into a circle, or a line into a polygon, or expands polygons.</p>
<p>An example buffering operation:</p>
<p><code>current_location.buffer(1000).contains(waffle_houses)</code></p>
<p>Turn a point into a circle with radius 1km, and see if that circle contains any waffle houses.</p>
<p><strong>Geospatial Joins</strong> combine two or more datasets on a condition (like overlaps or contains). This technique is used to discover overlaps between datasets, for example, to find what counties are in which cities:</p>
<p><code>joined_data = [(county, city) for county in counties for city in cities if county.geometry.intersects(city.geometry)]</code></p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="21" lang="en-US">
2024-11-11 01:01:51 +00:00
<h2 id="geospatial-software">Geospatial Software</h2>
<p>(skip in lecture)</p>
<p><strong>OGC Simple Features</strong> - Defines common interface (types and operations) for GIS.</p>
<p>Most geospatial software implements some version of this.</p>
<p>The most common of these is <strong>GEOS</strong> a C++ implementation of the specification that provides the common types and operations. ( Geometry Engine, Open Source )</p>
<p>A related library is <strong>GDAL</strong> which handles loading file formats:</p>
<p><a href="https://gdal.org/index.html">https://gdal.org/index.html</a> for list of formats</p>
<h3 id="postgis">PostGIS</h3>
<p>GIS primitives as first-class types in database. Allows querying directly from database, saving conversion and minimizing memory footprint. (Like most things Postgres, one of the most solid pieces of software ever built.)</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="22" lang="en-US">
2024-11-11 01:01:51 +00:00
<h3 id="geopandas">GeoPandas</h3>
<p>Wraps Shapely and provides a pandas like interface.</p>
<p><em>NOT EFFICIENT</em> fine for simple use, but quickly baloons to unusably large memory footprints without polygon simplification. (Reducing detail by eliding points.)</p>
<h3 id="arcgis">ArcGIS</h3>
<p>Commercial/proprietary service. Developed by ESRI, the key commercial player in GIS for decades.</p>
<p>Offers cloud/server/desktop based tools.</p>
<p>Favored in government, large corporations, full suite is most feature complete offering around. Under increasing competition from OSS competitors which are often more lean, cutting lesser-used features and antiquated formats.</p>
</section>
2024-11-13 23:50:20 +00:00
</foreignObject></svg><svg data-marpit-svg="" viewBox="0 0 1280 720"><foreignObject width="1280" height="720"><section id="23" lang="en-US">
<h2 id="next-1">Next</h2>
2024-11-11 01:01:51 +00:00
<p>Mapping Frameworks</p>
</section>
2024-11-13 23:50:20 +00:00
<script>!function(){"use strict";const t={h1:{proto:()=>HTMLHeadingElement,attrs:{role:"heading","aria-level":"1"},style:"display: block; font-size: 2em; margin-block-start: 0.67em; margin-block-end: 0.67em; margin-inline-start: 0px; margin-inline-end: 0px; font-weight: bold;"},h2:{proto:()=>HTMLHeadingElement,attrs:{role:"heading","aria-level":"2"},style:"display: block; font-size: 1.5em; margin-block-start: 0.83em; margin-block-end: 0.83em; margin-inline-start: 0px; margin-inline-end: 0px; font-weight: bold;"},h3:{proto:()=>HTMLHeadingElement,attrs:{role:"heading","aria-level":"3"},style:"display: block; font-size: 1.17em; margin-block-start: 1em; margin-block-end: 1em; margin-inline-start: 0px; margin-inline-end: 0px; font-weight: bold;"},h4:{proto:()=>HTMLHeadingElement,attrs:{role:"heading","aria-level":"4"},style:"display: block; margin-block-start: 1.33em; margin-block-end: 1.33em; margin-inline-start: 0px; margin-inline-end: 0px; font-weight: bold;"},h5:{proto:()=>HTMLHeadingElement,attrs:{role:"heading","aria-level":"5"},style:"display: block; font-size: 0.83em; margin-block-start: 1.67em; margin-block-end: 1.67em; margin-inline-start: 0px; margin-inline-end: 0px; font-weight: bold;"},h6:{proto:()=>HTMLHeadingElement,attrs:{role:"heading","aria-level":"6"},style:"display: block; font-size: 0.67em; margin-block-start: 2.33em; margin-block-end: 2.33em; margin-inline-start: 0px; margin-inline-end: 0px; font-weight: bold;"},span:{proto:()=>HTMLSpanElement},pre:{proto:()=>HTMLElement,style:"display: block; font-family: monospace; white-space: pre; margin: 1em 0; --marp-auto-scaling-white-space: pre;"}},e="data-marp-auto-scaling-wrapper",i="data-marp-auto-scaling-svg",n="data-marp-auto-scaling-container";class s extends HTMLElement{container;containerSize;containerObserver;svg;svgComputedStyle;svgPreserveAspectRatio="xMinYMid meet";wrapper;wrapperSize;wrapperObserver;constructor(){super();const t=t=>([e])=>{const{width:i,height:n}=e.contentRect;this[t]={width:i,height:n},this.updateSVGRect()};this.attachShadow({mode:"open"}),this.containerObserver=new ResizeObserver(t("containerSize")),this.wrapperObserver=new ResizeObserver(((...e)=>{t("wrapperSize")(...e),this.flushSvgDisplay()}))}static get observedAttributes(){return["data-downscale-only"]}connectedCallback(){this.shadowRoot.innerHTML=`\n<style>\n svg[${i}] { display: block; width: 100%; height: auto; vertical-align: top; }\n span[${n}] { display: table; white-space: var(--marp-auto-scaling-white-space, nowrap); width: max-content; }\n</style>\n<div ${e}>\n <svg part="svg" ${i}>\n <foreignObject><span ${n}><slot></slot></span></foreignObject>\n </svg>\n</div>\n `.split(/\n\s*/).join(""),this.wrapper=this.shadowRoot.querySelector(`div[${e}]`)??void 0;const t=this.svg;this.svg=this.wrapper?.querySelector(`svg[${i}]`)??void 0,this.svg!==t&&(this.svgComputedStyle=this.svg?window.getComputedStyle(this.svg):void 0),this.container=this.svg?.querySelector(`span[${n}]`)??void 0,this.observe()}disconnectedCallback(){this.svg=void 0,this.svgComputedStyle=void 0,this.wrapper=void 0,this.container=void 0,this.observe()}attributeChangedCallback(){this.observe()}flushSvgDisplay(){const{svg:t}=this;t&&(t.style.display="inline",requestAnimationFrame((()=>{t.style.display=""})))}observe(){this.containerObserver.disconnect(),this.wrapperObserver.disconnect(),this.wrapper&&this.wrapperObserver.observe(this.wrapper),this.container&&this.containerObserver.observe(this.container),this.svgComputedStyle&&this.observeSVGStyle(this.svgComputedStyle)}observeSVGStyle(t){const e=()=>{const i=(()=>{const e=t.getPropertyValue("--preserve-aspect-ratio");if(e)return e.trim();return`x${(({textAlign:t,direction:e})=>{if(t.endsWith("left"))return"Min";if(t.endsWith("right"))return"Max";if("start"===t||"end"===t){let i="rtl"===e;return"end"===t&&(i=!i),i?"Max":"Min"}return"Mid"})(t)}YMid meet`})();i!==this.svgPreserveAspectRatio&&(this.svgPreserveAspectRatio=i,this.updateSVGRect()),t===this.svgComputedStyle&&requestAnimationFrame(e)};e()}updateSVGRect(){let t=Math.ceil(this.containerSize?.w
2024-11-11 01:01:51 +00:00
</script></foreignObject></svg></div><div class="bespoke-marp-note" data-index="8" tabindex="0"><p>This is not unique to GIS, image files can be raster or vector as well, with nearly the same meaning.
A raster is a grid or matrix, such as the pixels that make up an image. In GIS terms this is often a satellite image, but any representation of the space where each cell in a grid has properties with a given meaning is a raster.
A raster can be though of as a series of values `(x, y, val)` where `val` is a color or similar. The `x` and `y` coordinates could be specified in any unit, allowing one &quot;pixel&quot; to represent 1 sq km or 1 sq mm depending on the needs of the program.
2024-11-13 23:50:20 +00:00
Vector data is instead a series of geometric primitives. They are defined in relationship to one another, allowing groups and other relationships to be estabished. We'll address the specifics more after we compare Vector data to Raster data.</p></div><script>/*!! License: https://unpkg.com/@marp-team/marp-cli@4.0.3/lib/bespoke.js.LICENSE.txt */
!function(){"use strict";function e(e){return e&&e.__esModule&&Object.prototype.hasOwnProperty.call(e,"default")?e.default:e}var t,n,r=(n||(n=1,t={from:function(e,t){var n,r=1===(e.parent||e).nodeType?e.parent||e:document.querySelector(e.parent||e),o=[].filter.call("string"==typeof e.slides?r.querySelectorAll(e.slides):e.slides||r.children,(function(e){return"SCRIPT"!==e.nodeName})),a={},i=function(e,t){return(t=t||{}).index=o.indexOf(e),t.slide=e,t},s=function(e,t){a[e]=(a[e]||[]).filter((function(e){return e!==t}))},c=function(e,t){return(a[e]||[]).reduce((function(e,n){return e&&!1!==n(t)}),!0)},l=function(e,t){o[e]&&(n&&c("deactivate",i(n,t)),n=o[e],c("activate",i(n,t)))},d=function(e,t){var r=o.indexOf(n)+e;c(e>0?"next":"prev",i(n,t))&&l(r,t)},u={off:s,on:function(e,t){return(a[e]||(a[e]=[])).push(t),s.bind(null,e,t)},fire:c,slide:function(e,t){if(!arguments.length)return o.indexOf(n);c("slide",i(o[e],t))&&l(e,t)},next:d.bind(null,1),prev:d.bind(null,-1),parent:r,slides:o,destroy:function(e){c("destroy",i(n,e)),a={}}};return(t||[]).forEach((function(e){e(u)})),n||l(0),u}}),t),o=e(r);const a=document.body,i=(...e)=>history.replaceState(...e),s="",c="presenter",l="next",d=["",c,l],u="bespoke-marp-",f=`data-${u}`,m=(e,{protocol:t,host:n,pathname:r,hash:o}=location)=>{const a=e.toString();return`${t}//${n}${r}${a?"?":""}${a}${o}`},g=()=>a.dataset.bespokeView,p=e=>new URLSearchParams(location.search).get(e),v=(e,t={})=>{const n={location,setter:i,...t},r=new URLSearchParams(n.location.search);for(const t of Object.keys(e)){const n=e[t];"string"==typeof n?r.set(t,n):r.delete(t)}try{n.setter({...window.history.state??{}},"",m(r,n.location))}catch(e){console.error(e)}},h=(()=>{const e="bespoke-marp";try{return localStorage.setItem(e,e),localStorage.removeItem(e),!0}catch{return!1}})(),y=e=>{try{return localStorage.getItem(e)}catch{return null}},b=(e,t)=>{try{return localStorage.setItem(e,t),!0}catch{return!1}},w=e=>{try{return localStorage.removeItem(e),!0}catch{return!1}},x=(e,t)=>{const n="aria-hidden";t?e.setAttribute(n,"true"):e.removeAttribute(n)},k=e=>{e.parent.classList.add(`${u}parent`),e.slides.forEach((e=>e.classList.add(`${u}slide`))),e.on("activate",(t=>{const n=`${u}active`,r=t.slide,o=r.classList,a=!o.contains(n);if(e.slides.forEach((e=>{e.classList.remove(n),x(e,!0)})),o.add(n),x(r,!1),a){const e=`${n}-ready`;o.add(e),document.body.clientHeight,o.remove(e)}}))},$=e=>{let t=0,n=0;Object.defineProperty(e,"fragments",{enumerable:!0,value:e.slides.map((e=>[null,...e.querySelectorAll("[data-marpit-fragment]")]))});const r=r=>void 0!==e.fragments[t][n+r],o=(r,o)=>{t=r,n=o,e.fragments.forEach(((e,t)=>{e.forEach(((e,n)=>{if(null==e)return;const a=t<r||t===r&&n<=o;e.setAttribute(`${f}fragment`,(a?"":"in")+"active");const i=`${f}current-fragment`;t===r&&n===o?e.setAttribute(i,"current"):e.removeAttribute(i)}))})),e.fragmentIndex=o;const a={slide:e.slides[r],index:r,fragments:e.fragments[r],fragmentIndex:o};e.fire("fragment",a)};e.on("next",(({fragment:a=!0})=>{if(a){if(r(1))return o(t,n+1),!1;const a=t+1;e.fragments[a]&&o(a,0)}else{const r=e.fragments[t].length;if(n+1<r)return o(t,r-1),!1;const a=e.fragments[t+1];a&&o(t+1,a.length-1)}})),e.on("prev",(({fragment:a=!0})=>{if(r(-1)&&a)return o(t,n-1),!1;const i=t-1;e.fragments[i]&&o(i,e.fragments[i].length-1)})),e.on("slide",(({index:t,fragment:n})=>{let r=0;if(void 0!==n){const o=e.fragments[t];if(o){const{length:e}=o;r=-1===n?e-1:Math.min(Math.max(n,0),e-1)}}o(t,r)})),o(0,0)},E=document,L=()=>!(!E.fullscreenEnabled&&!E.webkitFullscreenEnabled),S=()=>!(!E.fullscreenElement&&!E.webkitFullscreenElement),P=e=>{e.fullscreen=()=>{L()&&(async()=>{S()?(E.exitFullscreen||E.webkitExitFullscreen)?.call(E):((e=E.body)=>{(e.requestFullscreen||e.webkitRequestFullscreen)?.call(e)})()})()},document.addEventListener("keydown",(t=>{"f"!==t.key&&"F11"!==t.key||t.altKey||t.ctrlKey||t.metaKey||!L()||(e.fullscreen(),t.preventDefault())}))},_=`${u}inactive`,T=(e=2e3)=>({parent:t,fire:n})=>{const r=t.classList,o=e=>n(`marp-${e?"":"in"}active`);let a;const i=()=>{a&&clearTimeout(a),a=setTimeou