Compare commits
	
		
			84 Commits
		
	
	
		
			60b2df5e90
			...
			renovate/s
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 490071326e | |||
| 93369a2d62 | |||
| 6a922433cc | |||
| 2d6ba8a5a7 | |||
| cca8af0b93 | |||
| a8d8cb232d | |||
| 2840715cb6 | |||
| 5c6a07e87b | |||
| cd0dd430c6 | |||
| 115f5692e3 | |||
| 45c625994b | |||
| 93aea2e385 | |||
| d5a43e291c | |||
| f499b594da | |||
| 7beb704808 | |||
| 3f642f0a47 | |||
| c0450242ea | |||
| cb627abe06 | |||
| b56bcbc26d | |||
| d5f71fa043 | |||
| 0bde4eb0d1 | |||
| 22bad851e0 | |||
| 2c861f868a | |||
| a31c158cc5 | |||
| 78b35a7d83 | |||
| 1115a7afdd | |||
| ef8683a155 | |||
| ff51d6bc89 | |||
| faf901cd3a | |||
| 442b336c9e | |||
| d5cd3d76c1 | |||
| 7dff9b5700 | |||
| 28d7948261 | |||
| 4385f07aa7 | |||
| 29820d41b9 | |||
| c4c6265852 | |||
| 8d2f7d238f | |||
| ae5f2b7270 | |||
| a3f4c45733 | |||
| d1600e763a | |||
| 92cea97430 | |||
| dc9c4c0ca7 | |||
| 48c4e817d7 | |||
| 0f954d30a9 | |||
| 24f9aff11d | |||
| e5f0a70f97 | |||
| 1e476c767b | |||
| e28078402f | |||
| 66c1682b0b | |||
| bc49f31237 | |||
| 3def1a40ce | |||
| 1b3dc47f4e | |||
| cd97756ec2 | |||
| 0d796eb63c | |||
| f49a1d3bc7 | |||
| 5c49dc346c | |||
| 0467591053 | |||
| c3f917faa5 | |||
| 97503c3d68 | |||
| cf0dddd4a6 | |||
| 2a72d1cecb | |||
| 537be9ed04 | |||
| f23f669f76 | |||
| e32bbee062 | |||
| 9228769274 | |||
| 9528b39d6d | |||
| 0b850b97e1 | |||
| 72f517fdbb | |||
| e2392081bf | |||
| 52be65f6ba | |||
| 3af97b1d4d | |||
| 0297cca236 | |||
| 03e0a54399 | |||
| 18d7a3e7b1 | |||
| a33c4ca92b | |||
| b2ad4697f0 | |||
| 084a613467 | |||
| 60b0023a9d | |||
| 44e4ae5fa9 | |||
| 63efd8c449 | |||
| bd390adcc1 | |||
| 439b1fe95e | |||
| 898e2b6136 | |||
| a595129d6d | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,2 +1,3 @@ | ||||
| .DS_Store | ||||
| node_modules | ||||
| build | ||||
|   | ||||
| @@ -1,15 +1,40 @@ | ||||
| pipeline: | ||||
| clone: | ||||
|   git: | ||||
|     image: woodpeckerci/plugin-git | ||||
|     settings: | ||||
|       partial: false | ||||
|       depth: 15 | ||||
|  | ||||
| steps: | ||||
|   test: | ||||
|     image: docker | ||||
|     commands: | ||||
|       - apk add curl | ||||
|       - docker build . | ||||
|     when: | ||||
|       - event: push | ||||
|         branch: | ||||
|           exclude: ["main"] | ||||
|     volumes: | ||||
|       - /var/run/docker.sock:/var/run/docker.sock | ||||
|  | ||||
|   build: | ||||
|     image: docker | ||||
|     commands: | ||||
|       - apk add curl | ||||
|       - docker login -u $DOCKER_USERNAME -p $DOCKER_PASSWORD registry.yetaga.in | ||||
|       - docker login -u docker -p $DOCKER_PASSWORD registry.yetaga.in | ||||
|       - docker build -t registry.yetaga.in/bookstores:latest . | ||||
|       - docker push registry.yetaga.in/bookstores:latest | ||||
|       - 'curl http://100.113.98.36:4000/api/fetch -H "Authorization: Bearer $COMPOSE_TOKEN"' | ||||
|       - 'curl http://100.113.98.36:4000/api/update -H "Authorization: Bearer $COMPOSE_TOKEN"' | ||||
|     secrets: [docker_username, docker_password, compose_token] | ||||
|     environment: | ||||
|       DOCKER_PASSWORD: | ||||
|         from_secret: docker_password | ||||
|       COMPOSE_TOKEN: | ||||
|         from_secret: compose_token | ||||
|     when: | ||||
|       branch: "master" | ||||
|       - event: push | ||||
|         branch: | ||||
|           include: ["main"] | ||||
|     volumes: | ||||
|       - /var/run/docker.sock:/var/run/docker.sock | ||||
|   | ||||
| @@ -1,12 +1,9 @@ | ||||
| FROM node:20-slim AS builder | ||||
| FROM node:24 AS builder | ||||
|  | ||||
| COPY . /src | ||||
| WORKDIR /src | ||||
| RUN npm install && node ./index.js | ||||
|  | ||||
| FROM caddy:2.6.4 | ||||
| FROM caddy:2.10.2 | ||||
|  | ||||
| COPY img /usr/share/caddy/img | ||||
| COPY css /usr/share/caddy/css | ||||
| COPY stores.json robots.txt /usr/share/caddy/ | ||||
| COPY --from=builder /src/index.html /usr/share/caddy/index.html | ||||
| COPY --from=builder /src/build /usr/share/caddy | ||||
|   | ||||
							
								
								
									
										919
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										919
									
								
								index.html
									
									
									
									
									
								
							| @@ -1,919 +0,0 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <script  | ||||
|       defer | ||||
|       data-domain="nycbookstores.org" | ||||
|       src="https://stats.yetaga.in/js/script.hash.outbound-links.js" | ||||
|     ></script> | ||||
|     <meta charset="utf-8" /> | ||||
|     <meta http-equiv="x-ua-compatible" content="ie=edge" /> | ||||
|     <title> | ||||
|       Independent Bookstores in New York City - Best Community Bookstores in NYC | ||||
|     </title> | ||||
|     <meta | ||||
|       name="google-site-verification" | ||||
|       content="hEfog9h0E3JQW91ZUZM5ayPb6DND0WbUa2_W8yTIuVw" | ||||
|     /> | ||||
|     <link rel="icon" type="image/png" href="/img/favicon.png" /> | ||||
|     <link rel="apple-touch-icon" href="/img/social.jpg" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|     <script src="https://api.mapbox.com/mapbox-gl-js/v2.13.0/mapbox-gl.js"></script> | ||||
|     <link | ||||
|       href="https://api.mapbox.com/mapbox-gl-js/v2.13.0/mapbox-gl.css" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Acme|Lato&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       media="screen" | ||||
|       rel="stylesheet" | ||||
|       type="text/css" | ||||
|       href="/css/site.css?1678239113605" | ||||
|     /> | ||||
|     <meta | ||||
|       property="title" | ||||
|       name="title" | ||||
|       content="Independent Bookstores in New York City - Best Community Bookstores in NYC" | ||||
|     /> | ||||
|     <meta | ||||
|       property="description" | ||||
|       name="description" | ||||
|       content="A guide to and map of every independent bookstore in New York City. We have a complete list of community bookstores in NYC with locations and descriptions." | ||||
|     /> | ||||
|     <meta name="twitter:card" content="summary" /> | ||||
|     <meta name="twitter:site" content="www.nycbookstores.org" /> | ||||
|     <meta name="twitter:title" content="NYC Bookstores" /> | ||||
|     <meta | ||||
|       name="twitter:description" | ||||
|       content="A Guide To The Many Independent Bookstores Of New York City" | ||||
|     /> | ||||
|     <meta | ||||
|       name="twitter:image" | ||||
|       content="https://www.nycbookstores.org/img/social.jpg" | ||||
|     /> | ||||
|     <meta property="og:url" content="https://www.nycbookstores.org/" /> | ||||
|     <meta property="og:title" content="NYC Bookstores" /> | ||||
|     <meta | ||||
|       property="og:description" | ||||
|       content="A Guide To The Many Independent Bookstores Of New York City" | ||||
|     /> | ||||
|     <meta | ||||
|       property="og:image" | ||||
|       content="https://www.nycbookstores.org/img/social.jpg" | ||||
|     /> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="wrapper"> | ||||
|       <h1>NYC Bookstores</h1> | ||||
|       <div> | ||||
|         <ul class="nav"> | ||||
|           <li> | ||||
|             <h2 id="subhed"> | ||||
|               The Many Independent Bookstores of New York City | ||||
|             </h2> | ||||
|           </li> | ||||
|           <li> | ||||
|             <a id="viewInfo" href="#info" onclick="event.preventDefault()" | ||||
|               >intro</a | ||||
|             > | ||||
|           </li> | ||||
|           <li> | ||||
|             <a | ||||
|               href="https://git.yetaga.in/alazyreader/nyc-bookstores/" | ||||
|               target="_blank" | ||||
|               >source</a | ||||
|             > | ||||
|           </li> | ||||
|           <li> | ||||
|             <a href="https://icosahedron.website/@lazyreader" target="_blank" | ||||
|               >@lazyreader</a | ||||
|             > | ||||
|           </li> | ||||
|         </ul> | ||||
|       </div> | ||||
|       <div class="container"> | ||||
|         <div id="map"></div> | ||||
|         <div id="info"> | ||||
|           <p> | ||||
|             New York City loves its independent bookstores. It | ||||
|             <a | ||||
|               href="https://www.nytimes.com/2006/10/15/nyregion/thecity/15book.html" | ||||
|               target="_blank" | ||||
|               >eulogizes those that have faded</a | ||||
|             > | ||||
|             and celebrates when new ventures are launched. And while the | ||||
|             historic | ||||
|             <a | ||||
|               href="https://untappedcities.com/2015/08/26/4th-avenue-the-history-of-nycs-book-row/" | ||||
|               target="_blank" | ||||
|               >Book Row may have passed away in the 80s</a | ||||
|             >, there are still many indie bookstores dotting the map, across all | ||||
|             five boroughs. Here, I have attempted to collect all of the | ||||
|             currently-open general-interest independent booksellers in NYC. Any | ||||
|             store with regular-ish hours (excluding religious booksellers and | ||||
|             appointment-only rare book sellers) is included. | ||||
|           </p> | ||||
|           <p> | ||||
|             While Manhattan and Brooklyn still lead the pack, Queens has a | ||||
|             respectable number of stores, and all five boroughs are represented, | ||||
|             with the Bronx and Staten Island both hosting lone independent | ||||
|             stores. Lower Manhattan has the highest density of booksellers. | ||||
|           </p> | ||||
|           <p> | ||||
|             The listings here are kept up-to-date to the best of my ability; | ||||
|             however, I make no promises about either the accuracy or reliability | ||||
|             of the information. If you spot an error, or I've missed a shop, | ||||
|             please let me know by | ||||
|             <a href="mailto:delta.mu.alpha@gmail.com" target="_blank">email</a>, | ||||
|             <a href="https://icosahedron.website/@lazyreader" target="_blank" | ||||
|               >mastodon</a | ||||
|             >, or | ||||
|             <a href="https://www.twitter.com/alazyreader" target="_blank" | ||||
|               >twitter</a | ||||
|             >. Originally based on the "<a | ||||
|               href="https://github.com/jlord/hack-spots" | ||||
|               target="_blank" | ||||
|               >Hack Spots</a | ||||
|             >" website by | ||||
|             <a href="https://www.twitter.com/jllord" target="_blank">@jllord</a> | ||||
|             (although I don't believe any of the actual underlying code still | ||||
|             survives at this point). | ||||
|           </p> | ||||
|           <p> | ||||
|             There are currently <span id="storeCount">111</span> stores indexed | ||||
|             on this page. Last updated | ||||
|             <span id="updatedOn">March 7, 2023</span>. | ||||
|           </p> | ||||
|         </div> | ||||
|         <div id="selected"></div> | ||||
|       </div> | ||||
|  | ||||
|       <div class="clearfix"></div> | ||||
|  | ||||
|       <div class="container"> | ||||
|         <div id="Stores"> | ||||
|           <table> | ||||
|             <tbody> | ||||
|               <tr id="0" class="spotRow"> | ||||
|                 <td class="name">1804 Books</td> | ||||
|                 <td>320 W 37th Street, New York</td> | ||||
|               </tr> | ||||
|               <tr id="1" class="spotRow"> | ||||
|                 <td class="name">192 Books</td> | ||||
|                 <td>192 10th Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="2" class="spotRow"> | ||||
|                 <td class="name">Adanne</td> | ||||
|                 <td>234 Water Street, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="3" class="spotRow"> | ||||
|                 <td class="name">Aeon Bookstore</td> | ||||
|                 <td>151 East Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="4" class="spotRow"> | ||||
|                 <td class="name">Alabaster Bookshop</td> | ||||
|                 <td>122 4th Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="5" class="spotRow"> | ||||
|                 <td class="name">Albertine Books</td> | ||||
|                 <td>972 Fifth Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="6" class="spotRow"> | ||||
|                 <td class="name">Archestratus</td> | ||||
|                 <td>160 Huron St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="7" class="spotRow"> | ||||
|                 <td class="name">Argosy Books</td> | ||||
|                 <td>116 East 59th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="8" class="spotRow"> | ||||
|                 <td class="name">Astoria Bookshop</td> | ||||
|                 <td>31-29 31st St, Astoria</td> | ||||
|               </tr> | ||||
|               <tr id="9" class="spotRow"> | ||||
|                 <td class="name">Avoid The Day Bookstore & Cafe</td> | ||||
|                 <td>99-04 A Rockaway Beach Blvd, Rockaway Park</td> | ||||
|               </tr> | ||||
|               <tr id="10" class="spotRow"> | ||||
|                 <td class="name">Berl's Brooklyn Poetry Shop</td> | ||||
|                 <td>126A Front St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="11" class="spotRow"> | ||||
|                 <td class="name">Better Read Than Dead</td> | ||||
|                 <td>867 Broadway, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="12" class="spotRow"> | ||||
|                 <td class="name">Better Read Than Dead & Burly Coffee</td> | ||||
|                 <td>90 Kosciuszko St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="13" class="spotRow"> | ||||
|                 <td class="name">Bluestockings Cooperative</td> | ||||
|                 <td>116 Suffolk Street, New York</td> | ||||
|               </tr> | ||||
|               <tr id="14" class="spotRow"> | ||||
|                 <td class="name">Bonnie Slotnick Cookbooks</td> | ||||
|                 <td>28 East 2nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="15" class="spotRow"> | ||||
|                 <td class="name">Book Club Bar</td> | ||||
|                 <td>197 E 3rd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="16" class="spotRow"> | ||||
|                 <td class="name">Book Culture</td> | ||||
|                 <td>536 W 112th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="17" class="spotRow"> | ||||
|                 <td class="name">Book Culture LIC</td> | ||||
|                 <td>26-09 Jackson Ave, Queens</td> | ||||
|               </tr> | ||||
|               <tr id="18" class="spotRow"> | ||||
|                 <td class="name">Book Culture on Broadway</td> | ||||
|                 <td>2915 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="19" class="spotRow"> | ||||
|                 <td class="name">Book Thug Nation</td> | ||||
|                 <td>100 North 3rd St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="20" class="spotRow"> | ||||
|                 <td class="name">Bookmarc</td> | ||||
|                 <td>400 Bleecker St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="21" class="spotRow"> | ||||
|                 <td class="name">BookMark Shoppe</td> | ||||
|                 <td>8415 3rd Avenue, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="22" class="spotRow"> | ||||
|                 <td class="name">Bookoff</td> | ||||
|                 <td>49 W 45nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="23" class="spotRow"> | ||||
|                 <td class="name">Bookoff Brooklyn</td> | ||||
|                 <td>934 3rd Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="24" class="spotRow"> | ||||
|                 <td class="name">Books Are Magic (Montague St.)</td> | ||||
|                 <td>122 Montague St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="25" class="spotRow"> | ||||
|                 <td class="name">Books Are Magic (Smith St.)</td> | ||||
|                 <td>225 Smith St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="26" class="spotRow"> | ||||
|                 <td class="name">Books Of Wonder</td> | ||||
|                 <td>42 West 17th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="27" class="spotRow"> | ||||
|                 <td class="name">Boulevard Books</td> | ||||
|                 <td>7518 13th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="28" class="spotRow"> | ||||
|                 <td class="name">Bravo's Book Nook</td> | ||||
|                 <td>115 MacDougal St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="29" class="spotRow"> | ||||
|                 <td class="name">Burnt Books</td> | ||||
|                 <td>1014 Manhattan Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="30" class="spotRow"> | ||||
|                 <td class="name">Cafe con Libros</td> | ||||
|                 <td>724 Prospect Place, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="31" class="spotRow"> | ||||
|                 <td class="name">Catland</td> | ||||
|                 <td>987 Flushing Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="32" class="spotRow"> | ||||
|                 <td class="name">Chartwell Booksellers</td> | ||||
|                 <td>55 E 52nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="33" class="spotRow"> | ||||
|                 <td class="name">Codex Books</td> | ||||
|                 <td>1 Bleecker St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="34" class="spotRow"> | ||||
|                 <td class="name">Community Bookstore</td> | ||||
|                 <td>143 7th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="35" class="spotRow"> | ||||
|                 <td class="name">Cups and Books</td> | ||||
|                 <td>2024 Bedford Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="36" class="spotRow"> | ||||
|                 <td class="name">Dashwood Books</td> | ||||
|                 <td>33 Bond St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="37" class="spotRow"> | ||||
|                 <td class="name">Dear Friend Books</td> | ||||
|                 <td>343A Tompkins Avenue, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="38" class="spotRow"> | ||||
|                 <td class="name">Desert Island Comics</td> | ||||
|                 <td>540 Metropolitan Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="39" class="spotRow"> | ||||
|                 <td class="name">East Village Books</td> | ||||
|                 <td>99 St. Mark's Place, New York</td> | ||||
|               </tr> | ||||
|               <tr id="40" class="spotRow"> | ||||
|                 <td class="name">ETG Book Cafe</td> | ||||
|                 <td>208 Bay St, Staten Island</td> | ||||
|               </tr> | ||||
|               <tr id="41" class="spotRow"> | ||||
|                 <td class="name">Freebird Books</td> | ||||
|                 <td>123 Columbia St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="42" class="spotRow"> | ||||
|                 <td class="name">Greenlight Bookstore</td> | ||||
|                 <td>686 Fulton St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="43" class="spotRow"> | ||||
|                 <td class="name"> | ||||
|                   Greenlight Bookstore (Prospect Lefferts Gardens) | ||||
|                 </td> | ||||
|                 <td>632 Flatbush Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="44" class="spotRow"> | ||||
|                 <td class="name">Here's A Book Store</td> | ||||
|                 <td>1964 Coney Island Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="45" class="spotRow"> | ||||
|                 <td class="name">Hey Kids! Comics</td> | ||||
|                 <td>157 Huron St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="46" class="spotRow"> | ||||
|                 <td class="name">Housing Works Bookstore Café</td> | ||||
|                 <td>126 Crosby St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="47" class="spotRow"> | ||||
|                 <td class="name">Human Relations Books</td> | ||||
|                 <td>1067 Flushing Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="48" class="spotRow"> | ||||
|                 <td class="name">Idlewild Books</td> | ||||
|                 <td>170 7th Avenue S, New York</td> | ||||
|               </tr> | ||||
|               <tr id="49" class="spotRow"> | ||||
|                 <td class="name">James Cummins Bookseller</td> | ||||
|                 <td>699 Madison Ave, 7th Floor, New York</td> | ||||
|               </tr> | ||||
|               <tr id="50" class="spotRow"> | ||||
|                 <td class="name">Joanne Hendricks Cookbooks</td> | ||||
|                 <td>488 Greenwich St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="51" class="spotRow"> | ||||
|                 <td class="name">Karma Bookstore</td> | ||||
|                 <td>136 East Third St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="52" class="spotRow"> | ||||
|                 <td class="name">Kew & Willow Books</td> | ||||
|                 <td>81-63 Lefferts Boulevard, New York</td> | ||||
|               </tr> | ||||
|               <tr id="53" class="spotRow"> | ||||
|                 <td class="name">Kinokunya</td> | ||||
|                 <td>1073 Avenue of the Americas, New York</td> | ||||
|               </tr> | ||||
|               <tr id="54" class="spotRow"> | ||||
|                 <td class="name">Kitchen Arts & Letters</td> | ||||
|                 <td>1435 Lexington Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="55" class="spotRow"> | ||||
|                 <td class="name">Leaves Bookstore</td> | ||||
|                 <td>140 Nassau Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="56" class="spotRow"> | ||||
|                 <td class="name">Left Bank Books</td> | ||||
|                 <td>41 Perry Street, New York</td> | ||||
|               </tr> | ||||
|               <tr id="57" class="spotRow"> | ||||
|                 <td class="name">Logos Bookstore</td> | ||||
|                 <td>1575 York Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="58" class="spotRow"> | ||||
|                 <td class="name">Mast Books</td> | ||||
|                 <td>72 Avenue A, New York</td> | ||||
|               </tr> | ||||
|               <tr id="59" class="spotRow"> | ||||
|                 <td class="name">McNally Jackson Books</td> | ||||
|                 <td>52 Prince St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="60" class="spotRow"> | ||||
|                 <td class="name">McNally Jackson Books City Point</td> | ||||
|                 <td>445 Albee Square W, Unit G112, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="61" class="spotRow"> | ||||
|                 <td class="name">McNally Jackson Books Seaport</td> | ||||
|                 <td>4 Fulton St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="62" class="spotRow"> | ||||
|                 <td class="name">McNally Jackson Books Williamsburg</td> | ||||
|                 <td>76 North 4th St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="63" class="spotRow"> | ||||
|                 <td class="name">McNally Jackson Rockefeller Center</td> | ||||
|                 <td>1 Rockefeller Plaza, New York</td> | ||||
|               </tr> | ||||
|               <tr id="64" class="spotRow"> | ||||
|                 <td class="name">Melville House Publishers</td> | ||||
|                 <td>46 John St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="65" class="spotRow"> | ||||
|                 <td class="name">Mercer Street Books & Records</td> | ||||
|                 <td>206 Mercer St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="66" class="spotRow"> | ||||
|                 <td class="name">Mil Mundos Books</td> | ||||
|                 <td>323 Linden St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="67" class="spotRow"> | ||||
|                 <td class="name">Molasses Books</td> | ||||
|                 <td>770 Hart St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="68" class="spotRow"> | ||||
|                 <td class="name">Namaste Bookshop</td> | ||||
|                 <td>2 W 14th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="69" class="spotRow"> | ||||
|                 <td class="name">P&T Knitwear</td> | ||||
|                 <td>180 Orchard Street, New York</td> | ||||
|               </tr> | ||||
|               <tr id="70" class="spotRow"> | ||||
|                 <td class="name">Pillow-Cat Books</td> | ||||
|                 <td>328 East 9th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="71" class="spotRow"> | ||||
|                 <td class="name">Posman Books Chelsea Market</td> | ||||
|                 <td>75 9th Avenue, New York</td> | ||||
|               </tr> | ||||
|               <tr id="72" class="spotRow"> | ||||
|                 <td class="name">POWERHOUSE @ IC</td> | ||||
|                 <td>220 36th Street, Building #2, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="73" class="spotRow"> | ||||
|                 <td class="name">POWERHOUSE Arena</td> | ||||
|                 <td>28 Adams St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="74" class="spotRow"> | ||||
|                 <td class="name">POWERHOUSE on 8th</td> | ||||
|                 <td>1111 8th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="75" class="spotRow"> | ||||
|                 <td class="name">Printed Matter</td> | ||||
|                 <td>231 11th Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="76" class="spotRow"> | ||||
|                 <td class="name">Printed Matter St. Marks</td> | ||||
|                 <td>38 St. Marks Pl, New York</td> | ||||
|               </tr> | ||||
|               <tr id="77" class="spotRow"> | ||||
|                 <td class="name">Quest Bookshop</td> | ||||
|                 <td>240 E 53rd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="78" class="spotRow"> | ||||
|                 <td class="name">Quimby's Bookstore</td> | ||||
|                 <td>536 Metropolitan Avenue, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="79" class="spotRow"> | ||||
|                 <td class="name">Recirculation</td> | ||||
|                 <td>876 Riverside Dr, New York</td> | ||||
|               </tr> | ||||
|               <tr id="80" class="spotRow"> | ||||
|                 <td class="name">Respect For Life Books-N-Things</td> | ||||
|                 <td>537 Nostrand Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="81" class="spotRow"> | ||||
|                 <td class="name">Revolution Books</td> | ||||
|                 <td>437 Malcolm X Blvd, New York</td> | ||||
|               </tr> | ||||
|               <tr id="82" class="spotRow"> | ||||
|                 <td class="name">Rizzoli Bookstore</td> | ||||
|                 <td>1133 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="83" class="spotRow"> | ||||
|                 <td class="name">Shakespeare & Company</td> | ||||
|                 <td>939 Lexington Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="84" class="spotRow"> | ||||
|                 <td class="name"> | ||||
|                   Shakespeare & Company (Upper West Side) | ||||
|                 </td> | ||||
|                 <td>2020 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="85" class="spotRow"> | ||||
|                 <td class="name">Sister's Uptown Bookstore</td> | ||||
|                 <td>1942 Amsterdam Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="86" class="spotRow"> | ||||
|                 <td class="name">Spoonbill & Sugartown, Booksellers</td> | ||||
|                 <td>218 Bedford Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="87" class="spotRow"> | ||||
|                 <td class="name">Standards Manual</td> | ||||
|                 <td>212 Franklin Street, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="88" class="spotRow"> | ||||
|                 <td class="name">Strand Bookstore</td> | ||||
|                 <td>828 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="89" class="spotRow"> | ||||
|                 <td class="name">Sweet Pickle Books</td> | ||||
|                 <td>47 Orchard St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="90" class="spotRow"> | ||||
|                 <td class="name">Terrace Books</td> | ||||
|                 <td>242 Prospect Park West, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="91" class="spotRow"> | ||||
|                 <td class="name">The Austin Book Shop</td> | ||||
|                 <td>104-29 Jamaica Ave, Richmond Hill</td> | ||||
|               </tr> | ||||
|               <tr id="92" class="spotRow"> | ||||
|                 <td class="name">The Book Cellar</td> | ||||
|                 <td>1465 York Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="93" class="spotRow"> | ||||
|                 <td class="name">The Center for Fiction</td> | ||||
|                 <td>15 Lafayette Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="94" class="spotRow"> | ||||
|                 <td class="name">The Corner Bookstore</td> | ||||
|                 <td>1313 Madison Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="95" class="spotRow"> | ||||
|                 <td class="name">The Drama Book Shop</td> | ||||
|                 <td>266 W 39th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="96" class="spotRow"> | ||||
|                 <td class="name">The Lit. Bar</td> | ||||
|                 <td>131 Alexander Ave, Bronx</td> | ||||
|               </tr> | ||||
|               <tr id="97" class="spotRow"> | ||||
|                 <td class="name">The Mysterious Bookshop</td> | ||||
|                 <td>58 Warren St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="98" class="spotRow"> | ||||
|                 <td class="name">The Strand At Columbus Ave</td> | ||||
|                 <td>450 Columbus Avenue, New York</td> | ||||
|               </tr> | ||||
|               <tr id="99" class="spotRow"> | ||||
|                 <td class="name">The Word Is Change</td> | ||||
|                 <td>368 Tompkins Avenue, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="100" class="spotRow"> | ||||
|                 <td class="name">Three Lives & Company</td> | ||||
|                 <td>238 West 10th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="101" class="spotRow"> | ||||
|                 <td class="name">Topos Bookstore Cafe</td> | ||||
|                 <td>788 Woodward Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="102" class="spotRow"> | ||||
|                 <td class="name">Troubled Sleep Books</td> | ||||
|                 <td>129 6th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="103" class="spotRow"> | ||||
|                 <td class="name">Turn The Page... Again</td> | ||||
|                 <td>39-15a Bell Blvd, Flushing</td> | ||||
|               </tr> | ||||
|               <tr id="104" class="spotRow"> | ||||
|                 <td class="name">Unnameable Books</td> | ||||
|                 <td>600 Vanderbilt Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="105" class="spotRow"> | ||||
|                 <td class="name">Ursus Books</td> | ||||
|                 <td>50 East 78th St, Suite 1C, New York</td> | ||||
|               </tr> | ||||
|               <tr id="106" class="spotRow"> | ||||
|                 <td class="name">Westsider Rare & Used Books</td> | ||||
|                 <td>2246 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="107" class="spotRow"> | ||||
|                 <td class="name">Westsider Records</td> | ||||
|                 <td>233 West 72nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="108" class="spotRow"> | ||||
|                 <td class="name">Word Bookstore</td> | ||||
|                 <td>126 Franklin St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="109" class="spotRow"> | ||||
|                 <td class="name">Word Up Books</td> | ||||
|                 <td>2113 Amsterdam Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="110" class="spotRow"> | ||||
|                 <td class="name">Yu and Me Books</td> | ||||
|                 <td>44 Mulberry St, New York</td> | ||||
|               </tr> | ||||
|             </tbody> | ||||
|           </table> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     <!-- end wrapper --> | ||||
|  | ||||
|     <script> | ||||
|       mapboxgl.accessToken = | ||||
|         "pk.eyJ1IjoiYWxhenlyZWFkZXIiLCJhIjoiY2lucDZhb2JxMHp6MHRxa2pvaTFoOWpuZyJ9.DILGYYxxt7A-A_lHHwp6tQ"; | ||||
|       var map = new mapboxgl.Map({ | ||||
|         container: "map", | ||||
|         style: "mapbox://styles/mapbox/basic-v9", | ||||
|         center: [-73.957292, 40.729071], // arbitrary center point | ||||
|         zoom: 9, | ||||
|         minZoom: 9, | ||||
|         maxZoom: 17, | ||||
|         dragRotate: false, | ||||
|       }); | ||||
|  | ||||
|       var popup = new mapboxgl.Popup({ | ||||
|         closeOnClick: false, | ||||
|         closeButton: false, | ||||
|       }); | ||||
|  | ||||
|       function TableViewTemplate(rows) { | ||||
|         table = "<table>"; | ||||
|         rows.forEach((row) => { | ||||
|           table = table + TableRowTemplate(row); | ||||
|         }); | ||||
|         return table + "</table>"; | ||||
|       } | ||||
|  | ||||
|       function TableRowTemplate({ rowNumber, name, address, city }) { | ||||
|         return `<tr id="${rowNumber}" class="spotRow"> | ||||
|           <td class="name">${name}</td><td>${address}, ${city}</td> | ||||
|         </tr>`; | ||||
|       } | ||||
|  | ||||
|       function SelectedStoreTemplate({ | ||||
|         name, | ||||
|         address, | ||||
|         city, | ||||
|         postcode, | ||||
|         website, | ||||
|         events, | ||||
|         cafe, | ||||
|         description, | ||||
|       }) { | ||||
|         const isAppleIsh = /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform); | ||||
|         return ` | ||||
|           <h2>${name}</h2> | ||||
|           <p class="address">${address}</p> | ||||
|           <p></p> | ||||
|           <p class="address"> | ||||
|             ${city}, NY ${postcode} | ||||
|           </p> | ||||
|           <p> | ||||
|             View in: | ||||
|             <a | ||||
|               href="https://maps.google.com/maps?q=${encodeURIComponent( | ||||
|                 name | ||||
|               )}+${address},${city},NY" | ||||
|               target="_blank" | ||||
|               >Google Maps</a | ||||
|             > | ||||
|             ${ | ||||
|               isAppleIsh | ||||
|                 ? ` | ||||
|             <a | ||||
|               href="http://maps.apple.com/?q=${encodeURIComponent( | ||||
|                 name | ||||
|               )}&address=${address},${city},NY" | ||||
|               target="_blank" | ||||
|               >Apple Maps</a | ||||
|             >` | ||||
|                 : "" | ||||
|             } | ||||
|           </p> | ||||
|           <ul> | ||||
|             ${ | ||||
|               website | ||||
|                 ? `<li><a href="${website}" target="_blank">${cleanWebsite( | ||||
|                     website | ||||
|                   )}</a></li>` | ||||
|                 : "" | ||||
|             } | ||||
|             <li class="storeDetails">Events: ${events}</li> | ||||
|             <li class="storeDetails">Café: ${cafe}</li> | ||||
|           </ul> | ||||
|           ${description ? `<p class="description">${description}</p>` : ""}`; | ||||
|       } | ||||
|  | ||||
|       function hideElementById(id) { | ||||
|         const element = document.getElementById(id); | ||||
|         if (element !== undefined) { | ||||
|           element.classList.add("hidden"); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function showElementById(id) { | ||||
|         const element = document.getElementById(id); | ||||
|         if (element !== undefined) { | ||||
|           element.classList.remove("hidden"); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function setContent(id, html) { | ||||
|         const element = document.getElementById(id); | ||||
|         if (element !== undefined) { | ||||
|           element.innerHTML = html; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       document.addEventListener("DOMContentLoaded", function () { | ||||
|         fetch("./stores.json") | ||||
|           .then((resp) => { | ||||
|             return resp.json(); | ||||
|           }) | ||||
|           .then((data) => { | ||||
|             data.sort(function (a, b) { | ||||
|               var aname = a.name.toLowerCase(); | ||||
|               var bname = b.name.toLowerCase(); | ||||
|               return aname === bname ? 0 : +(aname > bname) || -1; | ||||
|             }); | ||||
|             data.forEach((value, key) => { | ||||
|               value.rowNumber = key; | ||||
|               value.slug = slugify(value.name); | ||||
|             }); | ||||
|             setContent("storeCount", data.length); | ||||
|             window.data = data; | ||||
|             loadMap(data); | ||||
|           }) | ||||
|           .catch((err) => { | ||||
|             // we'll live with the static cache! | ||||
|             console.log(err); | ||||
|           }); | ||||
|       }); | ||||
|  | ||||
|       window.addEventListener( | ||||
|         "hashchange", | ||||
|         function (e) { | ||||
|           updateViewBySlug(e.newURL.split("#")[1]); | ||||
|         }, | ||||
|         false | ||||
|       ); | ||||
|  | ||||
|       function updateHash(slug) { | ||||
|         history.pushState(null, null, "#" + slug); | ||||
|       } | ||||
|  | ||||
|       function slugify(str) { | ||||
|         return str | ||||
|           .toLowerCase() | ||||
|           .replace(/é/g, "e") | ||||
|           .replace(/&/g, " and ") | ||||
|           .replace(/ /g, "-") | ||||
|           .replace(/[']+/g, "") | ||||
|           .replace(/[^\w-]+/g, "-") | ||||
|           .replace(/-+/g, "-") | ||||
|           .replace(/^-|-$/g, ""); | ||||
|       } | ||||
|  | ||||
|       function cleanWebsite(str) { | ||||
|         return str | ||||
|           .toLowerCase() | ||||
|           .replace(/^https?:\/\//g, "") | ||||
|           .replace(/^www./g, "") | ||||
|           .replace(/\/$/g, ""); | ||||
|       } | ||||
|  | ||||
|       function getStoreBySlug(slug) { | ||||
|         var ret = false; | ||||
|         window.data.forEach((value, key) => { | ||||
|           if (value.slug === slug) { | ||||
|             ret = value; | ||||
|             return false; | ||||
|           } | ||||
|         }); | ||||
|         return ret; | ||||
|       } | ||||
|  | ||||
|       function updateViewBySlug(slug) { | ||||
|         if (slug === "info" || slug === undefined) { | ||||
|           showInfo(false); | ||||
|         } else { | ||||
|           var store = getStoreBySlug(slug); | ||||
|           if (store) { | ||||
|             updateSelectedStore(store, false); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function boundingBox(point) { | ||||
|         // add some buffer to a point to give the user some leeway | ||||
|         return [ | ||||
|           [point.x - 5, point.y - 5], | ||||
|           [point.x + 5, point.y + 5], | ||||
|         ]; | ||||
|       } | ||||
|  | ||||
|       function updateSelectedStore(store, pushState = false) { | ||||
|         map.flyTo({ center: [store.long, store.lat], zoom: 12 }); | ||||
|  | ||||
|         popup.setLngLat([store.long, store.lat]).setHTML(store.name).addTo(map); | ||||
|  | ||||
|         hideElementById("info"); | ||||
|         setContent("selected", SelectedStoreTemplate(store)); | ||||
|         showElementById("selected"); | ||||
|         if (pushState) { | ||||
|           updateHash(store.slug); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function showInfo(pushState = true) { | ||||
|         hideElementById("selected"); | ||||
|         popup.remove(); | ||||
|         showElementById("info"); | ||||
|         if (pushState) { | ||||
|           updateHash("info"); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function loadMap(data) { | ||||
|         var geolocate = new mapboxgl.GeolocateControl(); | ||||
|  | ||||
|         var points = []; | ||||
|         data.forEach((value, key) => { | ||||
|           points.push({ | ||||
|             type: "Feature", | ||||
|             geometry: { | ||||
|               type: "Point", | ||||
|               coordinates: [value.long, value.lat], | ||||
|             }, | ||||
|             properties: value, | ||||
|           }); | ||||
|         }); | ||||
|         map.on("load", function () { | ||||
|           map.addLayer({ | ||||
|             id: "stores", | ||||
|             type: "circle", | ||||
|             source: { | ||||
|               type: "geojson", | ||||
|               data: { | ||||
|                 type: "FeatureCollection", | ||||
|                 features: points, | ||||
|               }, | ||||
|             }, | ||||
|             paint: { | ||||
|               "circle-radius": 5, | ||||
|               "circle-color": "#B9FCFC", | ||||
|               "circle-stroke-width": 2, | ||||
|               "circle-stroke-color": "#000000", | ||||
|             }, | ||||
|           }); | ||||
|           map.addControl(new mapboxgl.NavigationControl(), "top-left"); | ||||
|           map.addControl(geolocate, "top-right"); | ||||
|           updateViewBySlug(window.location.hash.split("#")[1]); | ||||
|         }); | ||||
|  | ||||
|         map.on("click", function (e) { | ||||
|           if (!map.getLayer("stores")) { | ||||
|             return; | ||||
|           } | ||||
|  | ||||
|           popup.remove(); | ||||
|           // Use queryRenderedFeatures to get features at a click event's point | ||||
|           var features = map.queryRenderedFeatures(boundingBox(e.point), { | ||||
|             layers: ["stores"], | ||||
|           }); | ||||
|           // fly to the location of the click event | ||||
|           if (features.length) { | ||||
|             var store = features[0]; | ||||
|             // Get coordinates from the symbol and center the map on those coordinates | ||||
|             updateSelectedStore(store.properties, true); | ||||
|           } | ||||
|         }); | ||||
|  | ||||
|         // indicate that the symbols are clickable by changing the cursor style to 'pointer'. | ||||
|         map.on("mousemove", function (e) { | ||||
|           if (!map.getLayer("stores")) { | ||||
|             return; | ||||
|           } | ||||
|           var features = map.queryRenderedFeatures(boundingBox(e.point), { | ||||
|             layers: ["stores"], | ||||
|           }); | ||||
|           map.getCanvas().style.cursor = features.length ? "pointer" : ""; | ||||
|         }); | ||||
|  | ||||
|         geolocate.on("geolocate", function (e) { | ||||
|           map.setZoom(14); | ||||
|           popup | ||||
|             .setLngLat([e.coords.longitude, e.coords.latitude]) | ||||
|             .setHTML("Current Location") | ||||
|             .addTo(map); | ||||
|         }); | ||||
|  | ||||
|         setContent("Stores", TableViewTemplate(data)); | ||||
|         document.querySelectorAll("#Stores tbody tr").forEach((element) => { | ||||
|           element.addEventListener("click", () => { | ||||
|             updateSelectedStore(data[element.id], true); | ||||
|             document | ||||
|               .getElementById("subhed") | ||||
|               .scrollIntoView({ behavior: "smooth" }); | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         document.getElementById("viewInfo").addEventListener("click", showInfo); | ||||
|       } | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										187
									
								
								index.js
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								index.js
									
									
									
									
									
								
							| @@ -1,6 +1,90 @@ | ||||
| import { load } from "cheerio"; | ||||
| import { readFile, writeFile } from "fs"; | ||||
| import stores from "./stores.json" assert { type: "json" }; | ||||
| import { readFile, writeFileSync, mkdirSync, cpSync, rmSync } from "fs"; | ||||
| import process from "child_process"; | ||||
| import { simpleSitemapAndIndex } from "sitemap"; | ||||
|  | ||||
| import stores from "./stores.json" with { type: "json" }; | ||||
|  | ||||
| function mkDir(path) { | ||||
|   try { | ||||
|     return mkdirSync(path); | ||||
|   } catch (err) { | ||||
|     if (err.code !== "EEXIST") throw err; | ||||
|   } | ||||
| } | ||||
|  | ||||
| function writeFile(path, content) { | ||||
|   try { | ||||
|     writeFileSync(path, content); | ||||
|   } catch (err) { | ||||
|     if (err) throw err; | ||||
|   } | ||||
|   console.log(`${path} updated.`); | ||||
| } | ||||
|  | ||||
| function slugify(str) { | ||||
|   return str | ||||
|     .toLowerCase() | ||||
|     .replace(/é/g, "e") | ||||
|     .replace(/&/g, " and ") | ||||
|     .replace(/ /g, "-") | ||||
|     .replace(/[']+/g, "") | ||||
|     .replace(/[^\w-]+/g, "-") | ||||
|     .replace(/-+/g, "-") | ||||
|     .replace(/^-|-$/g, ""); | ||||
| } | ||||
|  | ||||
| function cleanWebsite(str) { | ||||
|   return str | ||||
|     .toLowerCase() | ||||
|     .replace(/^https?:\/\//g, "") | ||||
|     .replace(/^www./g, "") | ||||
|     .replace(/\/$/g, ""); | ||||
| } | ||||
|  | ||||
| function metaDescription({ name, meta, description }) { | ||||
|   if (meta && (meta.length > 155 || (meta.length < 145 && meta.length !== 0))) { | ||||
|     console.log( | ||||
|       `warning: meta tag for ${name} is invalid: 145/${meta.length}/155` | ||||
|     ); | ||||
|   } | ||||
|   return meta || description.length > 155 | ||||
|     ? description.slice(0, 153) + "..." | ||||
|     : description || | ||||
|         "A guide to and map of every independent bookstore in New York City. We have a complete list of community bookstores in NYC with locations and descriptions."; | ||||
| } | ||||
|  | ||||
| function GetRecentChanges() { | ||||
|   const res = process | ||||
|     .execSync('git log -15 --pretty=format:"%ct %s"') | ||||
|     .toString(); | ||||
|   return res.split("\n"); | ||||
| } | ||||
|  | ||||
| function ChangeLog(logs) { | ||||
|   let res = "\n"; | ||||
|   let i = 0; | ||||
|   logs.forEach((l) => { | ||||
|     if ( | ||||
|       i > 3 || | ||||
|       l.includes("[skip]") || | ||||
|       l.includes("[ignore]") || | ||||
|       l.includes("caddy") || | ||||
|       l.includes("renovate") | ||||
|     ) { | ||||
|       return; | ||||
|     } | ||||
|     i++; | ||||
|     const s = l.split(" "); | ||||
|     const date = new Date(s[0] * 1000).toLocaleDateString("en-US", { | ||||
|       year: "numeric", | ||||
|       month: "long", | ||||
|       day: "numeric", | ||||
|     }); | ||||
|     res = res + `<li>${date} - ${s.slice(1).join(" ")}</li>\n`; | ||||
|   }); | ||||
|   return res; | ||||
| } | ||||
|  | ||||
| function TableViewTemplate(rows) { | ||||
|   let table = "<table>"; | ||||
| @@ -11,15 +95,68 @@ function TableViewTemplate(rows) { | ||||
|   return table + "</table>"; | ||||
| } | ||||
|  | ||||
| function TableRowTemplate({ rowNumber, name, address, city }) { | ||||
| function TableRowTemplate({ rowNumber, name, slug, address, city }) { | ||||
|   return ` | ||||
|   <tr id="${rowNumber}" class="spotRow"> | ||||
|     <td class="name">${name}</td> | ||||
|     <td>${address}, ${city}</td> | ||||
|     <td class="name"><a href="/${slugify(name)}/">${name}</a></td> | ||||
|     <td><a href="/${slugify(name)}/">${address}, ${city}</a></td> | ||||
|   </tr>`; | ||||
| } | ||||
|  | ||||
| readFile("./index.html", function (err, data) { | ||||
| function TitleTemplate({ name }) { | ||||
|   return `${name} | Independent Bookstores in New York City - Best Community Bookstores in NYC`; | ||||
| } | ||||
|  | ||||
| function SelectedStoreTemplate({ | ||||
|   name, | ||||
|   address, | ||||
|   city, | ||||
|   postcode, | ||||
|   website, | ||||
|   events, | ||||
|   cafe, | ||||
|   description, | ||||
| }) { | ||||
|   return ` | ||||
|     <h2>${name}</h2> | ||||
|     <p class="address">${address}</p> | ||||
|     <p></p> | ||||
|     <p class="address"> | ||||
|       ${city}, NY ${postcode} | ||||
|     </p> | ||||
|     <p> | ||||
|       View in: | ||||
|       <a | ||||
|         href="https://maps.google.com/maps?q=${encodeURIComponent( | ||||
|           name | ||||
|         )}+${address},${city},NY" | ||||
|         target="_blank" | ||||
|         >Google Maps</a | ||||
|       > | ||||
|       <a | ||||
|         href="http://maps.apple.com/?q=${encodeURIComponent( | ||||
|           name | ||||
|         )}&address=${address},${city},NY" | ||||
|         target="_blank" | ||||
|         >Apple Maps</a | ||||
|       > | ||||
|     </p> | ||||
|     <ul> | ||||
|       ${ | ||||
|         website | ||||
|           ? `<li><a href="${website}" target="_blank">${cleanWebsite( | ||||
|               website | ||||
|             )}</a></li>` | ||||
|           : "" | ||||
|       } | ||||
|       <li class="storeDetails">Events: ${events}</li> | ||||
|       <li class="storeDetails">Café: ${cafe}</li> | ||||
|     </ul> | ||||
|     ${description ? `<p class="description">${description}</p>` : ""}`; | ||||
| } | ||||
|  | ||||
| readFile("./index.tmpl.html", function (err, data) { | ||||
|   const changeList = GetRecentChanges(); | ||||
|   if (err) { | ||||
|     throw err; | ||||
|   } | ||||
| @@ -40,10 +177,42 @@ readFile("./index.html", function (err, data) { | ||||
|       day: "numeric", | ||||
|     }) | ||||
|   ); | ||||
|   $("#changesList").html(ChangeLog(changeList)); | ||||
|   const cssurl = $("link[type='text/css']").attr("href").split("?")[0]; | ||||
|   $("link[type='text/css']").attr("href", cssurl + "?" + new Date().getTime()); | ||||
|   writeFile("./index.html", $.html(), (err) => { | ||||
|     if (err) throw err; | ||||
|     console.log("Default view updated."); | ||||
|  | ||||
|   rmSync("./build", { recursive: true, force: true }); | ||||
|  | ||||
|   mkDir("./build"); | ||||
|  | ||||
|   writeFile("./build/index.html", $.html()); | ||||
|  | ||||
|   cpSync("./site.css", "./build/site.css"); | ||||
|   cpSync("./robots.txt", "./build/robots.txt"); | ||||
|   cpSync("./img", "./build/img", { recursive: true }); | ||||
|   cpSync("./stores.json", "./build/stores.json"); | ||||
|  | ||||
|   let pages = [{ url: `/` }]; | ||||
|  | ||||
|   stores.forEach((store) => { | ||||
|     $("#selected").html(SelectedStoreTemplate(store)); | ||||
|     $("#info").addClass("hidden"); | ||||
|     let title = TitleTemplate(store); | ||||
|     $("title").html(title); | ||||
|     $("meta[name='title']").attr("content", title); | ||||
|     $("meta[name='description']").attr("content", metaDescription(store)); | ||||
|     $("link[rel='canonical']").attr( | ||||
|       "href", | ||||
|       `https://www.nycbookstores.org/${slugify(store.name)}/` | ||||
|     ); | ||||
|     mkDir(`./build/${slugify(store.name)}`); | ||||
|     writeFile(`./build/${slugify(store.name)}/index.html`, $.html()); | ||||
|     pages.push({ url: `/${slugify(store.name)}/` }); | ||||
|   }); | ||||
|   simpleSitemapAndIndex({ | ||||
|     hostname: "https://www.nycbookstores.org/", | ||||
|     destinationDir: `./build/`, | ||||
|     sourceData: pages, | ||||
|     gzip: false, | ||||
|   }); | ||||
| }); | ||||
|   | ||||
							
								
								
									
										469
									
								
								index.tmpl.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										469
									
								
								index.tmpl.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,469 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <script | ||||
|       defer="" | ||||
|       data-domain="nycbookstores.org" | ||||
|       src="https://stats.yetaga.in/js/script.outbound-links.js" | ||||
|     ></script> | ||||
|     <meta charset="utf-8" /> | ||||
|     <meta http-equiv="x-ua-compatible" content="ie=edge" /> | ||||
|     <title> | ||||
|       Independent Bookstores in New York City - Best Community Bookstores in NYC | ||||
|     </title> | ||||
|     <meta | ||||
|       name="google-site-verification" | ||||
|       content="hEfog9h0E3JQW91ZUZM5ayPb6DND0WbUa2_W8yTIuVw" | ||||
|     /> | ||||
|     <link rel="icon" type="image/png" href="/img/favicon.png" /> | ||||
|     <link rel="apple-touch-icon" href="/img/social.jpg" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|     <script src="https://api.mapbox.com/mapbox-gl-js/v2.13.0/mapbox-gl.js"></script> | ||||
|     <link | ||||
|       href="https://api.mapbox.com/mapbox-gl-js/v2.13.0/mapbox-gl.css" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Acme|Lato&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <link | ||||
|       media="screen" | ||||
|       rel="stylesheet" | ||||
|       type="text/css" | ||||
|       href="/site.css?1734659401621" | ||||
|     /> | ||||
|     <meta | ||||
|       property="title" | ||||
|       name="title" | ||||
|       content="Independent Bookstores in New York City - Best Community Bookstores in NYC" | ||||
|     /> | ||||
|     <meta | ||||
|       property="description" | ||||
|       name="description" | ||||
|       content="A guide to and map of every independent bookstore in New York City. We have a complete list of community bookstores in NYC with locations and descriptions." | ||||
|     /> | ||||
|     <meta name="twitter:card" content="summary" /> | ||||
|     <meta name="twitter:site" content="www.nycbookstores.org" /> | ||||
|     <meta name="twitter:title" content="NYC Bookstores" /> | ||||
|     <meta | ||||
|       name="twitter:description" | ||||
|       content="A Guide To The Many Independent Bookstores Of New York City" | ||||
|     /> | ||||
|     <meta | ||||
|       name="twitter:image" | ||||
|       content="https://www.nycbookstores.org/img/social.jpg" | ||||
|     /> | ||||
|     <meta property="og:url" content="https://www.nycbookstores.org/" /> | ||||
|     <meta property="og:type" content="website" /> | ||||
|     <meta property="og:title" content="NYC Bookstores" /> | ||||
|     <meta | ||||
|       property="og:description" | ||||
|       content="A Guide To The Many Independent Bookstores Of New York City" | ||||
|     /> | ||||
|     <meta | ||||
|       property="og:image" | ||||
|       content="https://www.nycbookstores.org/img/social.jpg" | ||||
|     /> | ||||
|     <link rel="canonical" href="https://www.nycbookstores.org/" /> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="wrapper"> | ||||
|       <h1>NYC Bookstores</h1> | ||||
|       <div> | ||||
|         <ul class="nav"> | ||||
|           <li> | ||||
|             <h2 id="subhed"> | ||||
|               The Many Independent Bookstores of New York City | ||||
|             </h2> | ||||
|           </li> | ||||
|           <li> | ||||
|             <a id="viewInfo" href="/">intro</a> | ||||
|           </li> | ||||
|           <li> | ||||
|             <a | ||||
|               href="https://git.yetaga.in/alazyreader/nyc-bookstores/" | ||||
|               target="_blank" | ||||
|               >source</a | ||||
|             > | ||||
|           </li> | ||||
|           <li> | ||||
|             <a href="https://icosahedron.website/@lazyreader" target="_blank" | ||||
|               >@lazyreader</a | ||||
|             > | ||||
|           </li> | ||||
|         </ul> | ||||
|       </div> | ||||
|       <div class="container"> | ||||
|         <div id="map"></div> | ||||
|         <div id="info"> | ||||
|           <p> | ||||
|             New York City loves its independent bookstores. It | ||||
|             <a | ||||
|               href="https://www.nytimes.com/2006/10/15/nyregion/thecity/15book.html" | ||||
|               target="_blank" | ||||
|               >eulogizes those that have faded</a | ||||
|             > | ||||
|             and celebrates when new ventures are launched. And while the | ||||
|             historic | ||||
|             <a | ||||
|               href="https://untappedcities.com/2015/08/26/4th-avenue-the-history-of-nycs-book-row/" | ||||
|               target="_blank" | ||||
|               >Book Row may have passed away in the 80s</a | ||||
|             >, there are still many indie bookstores dotting the map, across all | ||||
|             five boroughs. Here, I have attempted to collect all of the | ||||
|             currently-open general-interest independent booksellers in NYC. Any | ||||
|             store with regular-ish hours (excluding religious booksellers and | ||||
|             appointment-only rare book sellers) is included. | ||||
|           </p> | ||||
|           <p> | ||||
|             While Manhattan and Brooklyn still lead the pack, Queens has a | ||||
|             respectable number of stores, and all five boroughs are represented, | ||||
|             with the Bronx and Staten Island both hosting lone independent | ||||
|             stores. Lower Manhattan has the highest density of booksellers. | ||||
|           </p> | ||||
|           <p> | ||||
|             The listings here are kept up-to-date to the best of my ability; | ||||
|             however, I make no promises about either the accuracy or reliability | ||||
|             of the information. If you spot an error, or I've missed a shop, | ||||
|             please let me know by | ||||
|             <a href="mailto:delta.mu.alpha@gmail.com" target="_blank">email</a>, | ||||
|             <a href="https://icosahedron.website/@lazyreader" target="_blank" | ||||
|               >mastodon</a | ||||
|             >, or | ||||
|             <a href="https://www.twitter.com/alazyreader" target="_blank" | ||||
|               >twitter</a | ||||
|             >. Originally based on the "<a | ||||
|               href="https://github.com/jlord/hack-spots" | ||||
|               target="_blank" | ||||
|               >Hack Spots</a | ||||
|             >" website by | ||||
|             <a href="https://www.twitter.com/jllord" target="_blank">@jllord</a> | ||||
|             (although I don't believe any of the actual underlying code still | ||||
|             survives at this point). | ||||
|           </p> | ||||
|           <p> | ||||
|             There are currently <span id="storeCount">121</span> stores indexed | ||||
|             on this page. Last updated | ||||
|             <span id="updatedOn">December 19, 2024</span>. | ||||
|           </p> | ||||
|           <details> | ||||
|             <summary>Recent Changes</summary> | ||||
|             <ul id="changesList"></ul> | ||||
|           </details> | ||||
|         </div> | ||||
|         <div id="selected"></div> | ||||
|       </div> | ||||
|  | ||||
|       <div class="clearfix"></div> | ||||
|  | ||||
|       <div class="container"> | ||||
|         <div id="Stores"> | ||||
|           <table> | ||||
|             <tbody></tbody> | ||||
|           </table> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|     <!-- end wrapper --> | ||||
|  | ||||
|     <script> | ||||
|       mapboxgl.accessToken = | ||||
|         "pk.eyJ1IjoiYWxhenlyZWFkZXIiLCJhIjoiY2lucDZhb2JxMHp6MHRxa2pvaTFoOWpuZyJ9.DILGYYxxt7A-A_lHHwp6tQ"; | ||||
|       var map = new mapboxgl.Map({ | ||||
|         container: "map", | ||||
|         style: "mapbox://styles/mapbox/basic-v9", | ||||
|         center: [-73.957292, 40.729071], // arbitrary center point | ||||
|         zoom: 9, | ||||
|         minZoom: 9, | ||||
|         maxZoom: 17, | ||||
|         dragRotate: false, | ||||
|       }); | ||||
|  | ||||
|       var popup = new mapboxgl.Popup({ | ||||
|         closeOnClick: false, | ||||
|         closeButton: false, | ||||
|       }); | ||||
|  | ||||
|       function TitleTemplate({ name }) { | ||||
|         return `${name} | Independent Bookstores in New York City - Best Community Bookstores in NYC`; | ||||
|       } | ||||
|  | ||||
|       function TableViewTemplate(rows) { | ||||
|         table = "<table>"; | ||||
|         rows.forEach((row) => { | ||||
|           table = table + TableRowTemplate(row); | ||||
|         }); | ||||
|         return table + "</table>"; | ||||
|       } | ||||
|  | ||||
|       function TableRowTemplate({ rowNumber, name, address, city }) { | ||||
|         return `<tr id="${rowNumber}" class="spotRow"> | ||||
|           <td class="name">${name}</td><td>${address}, ${city}</td> | ||||
|         </tr>`; | ||||
|       } | ||||
|  | ||||
|       function SelectedStoreTemplate({ | ||||
|         name, | ||||
|         address, | ||||
|         city, | ||||
|         postcode, | ||||
|         website, | ||||
|         events, | ||||
|         cafe, | ||||
|         description, | ||||
|       }) { | ||||
|         const isAppleIsh = /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform); | ||||
|         return ` | ||||
|           <h2>${name}</h2> | ||||
|           <p class="address">${address}</p> | ||||
|           <p></p> | ||||
|           <p class="address"> | ||||
|             ${city}, NY ${postcode} | ||||
|           </p> | ||||
|           <p> | ||||
|             View in: | ||||
|             <a | ||||
|               href="https://maps.google.com/maps?q=${encodeURIComponent( | ||||
|                 name | ||||
|               )}+${address},${city},NY" | ||||
|               target="_blank" | ||||
|               >Google Maps</a | ||||
|             > | ||||
|             ${ | ||||
|               isAppleIsh | ||||
|                 ? ` | ||||
|             <a | ||||
|               href="http://maps.apple.com/?q=${encodeURIComponent( | ||||
|                 name | ||||
|               )}&address=${address},${city},NY" | ||||
|               target="_blank" | ||||
|               >Apple Maps</a | ||||
|             >` | ||||
|                 : "" | ||||
|             } | ||||
|           </p> | ||||
|           <ul> | ||||
|             ${ | ||||
|               website | ||||
|                 ? `<li><a href="${website}" target="_blank">${cleanWebsite( | ||||
|                     website | ||||
|                   )}</a></li>` | ||||
|                 : "" | ||||
|             } | ||||
|             <li class="storeDetails">Events: ${events}</li> | ||||
|             <li class="storeDetails">Café: ${cafe}</li> | ||||
|           </ul> | ||||
|           ${description ? `<p class="description">${description}</p>` : ""}`; | ||||
|       } | ||||
|  | ||||
|       function hideElementById(id) { | ||||
|         const element = document.getElementById(id); | ||||
|         if (element !== undefined) { | ||||
|           element.classList.add("hidden"); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function showElementById(id) { | ||||
|         const element = document.getElementById(id); | ||||
|         if (element !== undefined) { | ||||
|           element.classList.remove("hidden"); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function setContent(id, html) { | ||||
|         const element = document.getElementById(id); | ||||
|         if (element !== undefined) { | ||||
|           element.innerHTML = html; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function setTitle(string) { | ||||
|         const element = document.getElementsByTagName("title"); | ||||
|         if (element !== undefined && element.length === 1) { | ||||
|           element[0].innerText = string; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       document.addEventListener("DOMContentLoaded", function () { | ||||
|         if (window.location.hash !== "") { | ||||
|           updateLocation(window.location.hash.substring(1)); | ||||
|         } | ||||
|  | ||||
|         fetch("/stores.json") | ||||
|           .then((resp) => { | ||||
|             return resp.json(); | ||||
|           }) | ||||
|           .then((data) => { | ||||
|             data.sort(function (a, b) { | ||||
|               var aname = a.name.toLowerCase(); | ||||
|               var bname = b.name.toLowerCase(); | ||||
|               return aname === bname ? 0 : +(aname > bname) || -1; | ||||
|             }); | ||||
|             data.forEach((value, key) => { | ||||
|               value.rowNumber = key; | ||||
|               value.slug = slugify(value.name); | ||||
|             }); | ||||
|             setContent("storeCount", data.length); | ||||
|             window.data = data; | ||||
|             loadMap(data); | ||||
|           }) | ||||
|           .catch((err) => { | ||||
|             // we'll live with the static cache! | ||||
|             console.log(err); | ||||
|           }); | ||||
|       }); | ||||
|  | ||||
|       function updateLocation(slug) { | ||||
|         history.pushState(null, null, `/${slug}`); | ||||
|       } | ||||
|  | ||||
|       function slugify(str) { | ||||
|         return str | ||||
|           .toLowerCase() | ||||
|           .replace(/é/g, "e") | ||||
|           .replace(/&/g, " and ") | ||||
|           .replace(/ /g, "-") | ||||
|           .replace(/[']+/g, "") | ||||
|           .replace(/[^\w-]+/g, "-") | ||||
|           .replace(/-+/g, "-") | ||||
|           .replace(/^-|-$/g, ""); | ||||
|       } | ||||
|  | ||||
|       function cleanWebsite(str) { | ||||
|         return str | ||||
|           .toLowerCase() | ||||
|           .replace(/^https?:\/\//g, "") | ||||
|           .replace(/^www./g, "") | ||||
|           .replace(/\/$/g, ""); | ||||
|       } | ||||
|  | ||||
|       function getStoreBySlug(slug) { | ||||
|         var ret = false; | ||||
|         window.data.forEach((value, key) => { | ||||
|           if (value.slug === slug) { | ||||
|             ret = value; | ||||
|             return false; | ||||
|           } | ||||
|         }); | ||||
|         return ret; | ||||
|       } | ||||
|  | ||||
|       function updateViewBySlug(slug) { | ||||
|         if (slug === undefined) { | ||||
|           showInfo(false); | ||||
|         } else { | ||||
|           var store = getStoreBySlug(slug); | ||||
|           if (store) { | ||||
|             updateSelectedStore(store, false); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function boundingBox(point) { | ||||
|         // add some buffer to a point to give the user some leeway | ||||
|         return [ | ||||
|           [point.x - 5, point.y - 5], | ||||
|           [point.x + 5, point.y + 5], | ||||
|         ]; | ||||
|       } | ||||
|  | ||||
|       function updateSelectedStore(store, pushState = false) { | ||||
|         map.flyTo({ center: [store.long, store.lat], zoom: 12 }); | ||||
|  | ||||
|         popup.setLngLat([store.long, store.lat]).setHTML(store.name).addTo(map); | ||||
|  | ||||
|         hideElementById("info"); | ||||
|         setContent("selected", SelectedStoreTemplate(store)); | ||||
|         showElementById("selected"); | ||||
|         setTitle(TitleTemplate(store)); | ||||
|         if (pushState) { | ||||
|           updateLocation(store.slug); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function showInfo(pushState = true) { | ||||
|         hideElementById("selected"); | ||||
|         popup.remove(); | ||||
|         showElementById("info"); | ||||
|         if (pushState) { | ||||
|           updateLocation("info"); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       function loadMap(data) { | ||||
|         var geolocate = new mapboxgl.GeolocateControl(); | ||||
|  | ||||
|         var points = []; | ||||
|         data.forEach((value, key) => { | ||||
|           points.push({ | ||||
|             type: "Feature", | ||||
|             geometry: { | ||||
|               type: "Point", | ||||
|               coordinates: [value.long, value.lat], | ||||
|             }, | ||||
|             properties: value, | ||||
|           }); | ||||
|         }); | ||||
|         map.on("load", function () { | ||||
|           map.addLayer({ | ||||
|             id: "stores", | ||||
|             type: "circle", | ||||
|             source: { | ||||
|               type: "geojson", | ||||
|               data: { | ||||
|                 type: "FeatureCollection", | ||||
|                 features: points, | ||||
|               }, | ||||
|             }, | ||||
|             paint: { | ||||
|               "circle-radius": 5, | ||||
|               "circle-color": "#B9FCFC", | ||||
|               "circle-stroke-width": 2, | ||||
|               "circle-stroke-color": "#000000", | ||||
|             }, | ||||
|           }); | ||||
|           map.addControl(new mapboxgl.NavigationControl(), "top-left"); | ||||
|           map.addControl(geolocate, "top-right"); | ||||
|           updateViewBySlug(window.location.pathname.split("/")[1]); | ||||
|         }); | ||||
|  | ||||
|         map.on("click", function (e) { | ||||
|           if (!map.getLayer("stores")) { | ||||
|             return; | ||||
|           } | ||||
|           popup.remove(); | ||||
|           // Use queryRenderedFeatures to get features at a click event's point | ||||
|           var features = map.queryRenderedFeatures(boundingBox(e.point), { | ||||
|             layers: ["stores"], | ||||
|           }); | ||||
|           // fly to the location of the click event | ||||
|           if (features.length) { | ||||
|             var store = features[0]; | ||||
|             // Get coordinates from the symbol and center the map on those coordinates | ||||
|             updateSelectedStore(store.properties, true); | ||||
|           } | ||||
|         }); | ||||
|  | ||||
|         // indicate that the symbols are clickable by changing the cursor style to 'pointer'. | ||||
|         map.on("mousemove", function (e) { | ||||
|           if (!map.getLayer("stores")) { | ||||
|             return; | ||||
|           } | ||||
|           var features = map.queryRenderedFeatures(boundingBox(e.point), { | ||||
|             layers: ["stores"], | ||||
|           }); | ||||
|           map.getCanvas().style.cursor = features.length ? "pointer" : ""; | ||||
|         }); | ||||
|  | ||||
|         geolocate.on("geolocate", function (e) { | ||||
|           map.setZoom(14); | ||||
|           popup | ||||
|             .setLngLat([e.coords.longitude, e.coords.latitude]) | ||||
|             .setHTML("Current Location") | ||||
|             .addTo(map); | ||||
|         }); | ||||
|       } | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										354
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										354
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -9,29 +9,55 @@ | ||||
|       "version": "1.0.0", | ||||
|       "license": "BSD-3-Clause", | ||||
|       "dependencies": { | ||||
|         "cheerio": "^1.0.0-rc.12" | ||||
|         "cheerio": "^1.0.0", | ||||
|         "sitemap": "^8.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@types/node": { | ||||
|       "version": "17.0.45", | ||||
|       "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.45.tgz", | ||||
|       "integrity": "sha512-w+tIMs3rq2afQdsPJlODhoUEKzFP1ayaoyl1CcnwtIlsVe7K7bA1NGm4s3PraqTLlXnbIN84zuBlxBWo1u9BLw==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/@types/sax": { | ||||
|       "version": "1.2.7", | ||||
|       "resolved": "https://registry.npmjs.org/@types/sax/-/sax-1.2.7.tgz", | ||||
|       "integrity": "sha512-rO73L89PJxeYM3s3pPPjiPgVVcymqU490g0YO5n5By0k2Erzj6tay/4lr1CHAAU4JyOWd1rpQ8bCf6cZfHU96A==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@types/node": "*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/arg": { | ||||
|       "version": "5.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz", | ||||
|       "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/boolbase": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", | ||||
|       "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==" | ||||
|     }, | ||||
|     "node_modules/cheerio": { | ||||
|       "version": "1.0.0-rc.12", | ||||
|       "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz", | ||||
|       "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==", | ||||
|       "version": "1.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.1.2.tgz", | ||||
|       "integrity": "sha512-IkxPpb5rS/d1IiLbHMgfPuS0FgiWTtFIm/Nj+2woXDLTZ7fOT2eqzgYbdMlLweqlHbsZjxEChoVK+7iph7jyQg==", | ||||
|       "dependencies": { | ||||
|         "cheerio-select": "^2.1.0", | ||||
|         "dom-serializer": "^2.0.0", | ||||
|         "domhandler": "^5.0.3", | ||||
|         "domutils": "^3.0.1", | ||||
|         "htmlparser2": "^8.0.1", | ||||
|         "parse5": "^7.0.0", | ||||
|         "parse5-htmlparser2-tree-adapter": "^7.0.0" | ||||
|         "domutils": "^3.2.2", | ||||
|         "encoding-sniffer": "^0.2.1", | ||||
|         "htmlparser2": "^10.0.0", | ||||
|         "parse5": "^7.3.0", | ||||
|         "parse5-htmlparser2-tree-adapter": "^7.1.0", | ||||
|         "parse5-parser-stream": "^7.1.2", | ||||
|         "undici": "^7.12.0", | ||||
|         "whatwg-mimetype": "^4.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 6" | ||||
|         "node": ">=20.18.1" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/cheeriojs/cheerio?sponsor=1" | ||||
| @@ -118,22 +144,34 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/domutils": { | ||||
|       "version": "3.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz", | ||||
|       "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==", | ||||
|       "version": "3.2.2", | ||||
|       "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz", | ||||
|       "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==", | ||||
|       "dependencies": { | ||||
|         "dom-serializer": "^2.0.0", | ||||
|         "domelementtype": "^2.3.0", | ||||
|         "domhandler": "^5.0.1" | ||||
|         "domhandler": "^5.0.3" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/fb55/domutils?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/encoding-sniffer": { | ||||
|       "version": "0.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/encoding-sniffer/-/encoding-sniffer-0.2.1.tgz", | ||||
|       "integrity": "sha512-5gvq20T6vfpekVtqrYQsSCFZ1wEg5+wW0/QaZMWkFr6BqD3NfKs0rLCx4rrVlSWJeZb5NBJgVLswK/w2MWU+Gw==", | ||||
|       "dependencies": { | ||||
|         "iconv-lite": "^0.6.3", | ||||
|         "whatwg-encoding": "^3.1.1" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/fb55/encoding-sniffer?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/entities": { | ||||
|       "version": "4.4.0", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz", | ||||
|       "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA==", | ||||
|       "version": "4.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", | ||||
|       "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", | ||||
|       "engines": { | ||||
|         "node": ">=0.12" | ||||
|       }, | ||||
| @@ -142,9 +180,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/htmlparser2": { | ||||
|       "version": "8.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.1.tgz", | ||||
|       "integrity": "sha512-4lVbmc1diZC7GUJQtRQ5yBAeUCL1exyMwmForWkRLnwyzWBFxN633SALPMGYaWZvKe9j1pRZJpauvmxENSp/EA==", | ||||
|       "version": "10.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-10.0.0.tgz", | ||||
|       "integrity": "sha512-TwAZM+zE5Tq3lrEHvOlvwgj1XLWQCtaaibSN11Q+gGBAS7Y1uZSWwXXRe4iF6OXnaq1riyQAPFOBtYc77Mxq0g==", | ||||
|       "funding": [ | ||||
|         "https://github.com/fb55/htmlparser2?sponsor=1", | ||||
|         { | ||||
| @@ -154,9 +192,31 @@ | ||||
|       ], | ||||
|       "dependencies": { | ||||
|         "domelementtype": "^2.3.0", | ||||
|         "domhandler": "^5.0.2", | ||||
|         "domutils": "^3.0.1", | ||||
|         "entities": "^4.3.0" | ||||
|         "domhandler": "^5.0.3", | ||||
|         "domutils": "^3.2.1", | ||||
|         "entities": "^6.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/htmlparser2/node_modules/entities": { | ||||
|       "version": "6.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-6.0.1.tgz", | ||||
|       "integrity": "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==", | ||||
|       "engines": { | ||||
|         "node": ">=0.12" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/fb55/entities?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/iconv-lite": { | ||||
|       "version": "0.6.3", | ||||
|       "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", | ||||
|       "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", | ||||
|       "dependencies": { | ||||
|         "safer-buffer": ">= 2.1.2 < 3.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/nth-check": { | ||||
| @@ -171,47 +231,147 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/parse5": { | ||||
|       "version": "7.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz", | ||||
|       "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==", | ||||
|       "version": "7.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.3.0.tgz", | ||||
|       "integrity": "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==", | ||||
|       "dependencies": { | ||||
|         "entities": "^4.4.0" | ||||
|         "entities": "^6.0.0" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/inikulin/parse5?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/parse5-htmlparser2-tree-adapter": { | ||||
|       "version": "7.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz", | ||||
|       "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==", | ||||
|       "version": "7.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.1.0.tgz", | ||||
|       "integrity": "sha512-ruw5xyKs6lrpo9x9rCZqZZnIUntICjQAd0Wsmp396Ul9lN/h+ifgVV1x1gZHi8euej6wTfpqX8j+BFQxF0NS/g==", | ||||
|       "dependencies": { | ||||
|         "domhandler": "^5.0.2", | ||||
|         "domhandler": "^5.0.3", | ||||
|         "parse5": "^7.0.0" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/inikulin/parse5?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/parse5-parser-stream": { | ||||
|       "version": "7.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/parse5-parser-stream/-/parse5-parser-stream-7.1.2.tgz", | ||||
|       "integrity": "sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow==", | ||||
|       "dependencies": { | ||||
|         "parse5": "^7.0.0" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/inikulin/parse5?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/parse5/node_modules/entities": { | ||||
|       "version": "6.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-6.0.1.tgz", | ||||
|       "integrity": "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==", | ||||
|       "engines": { | ||||
|         "node": ">=0.12" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/fb55/entities?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/safer-buffer": { | ||||
|       "version": "2.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", | ||||
|       "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" | ||||
|     }, | ||||
|     "node_modules/sax": { | ||||
|       "version": "1.4.1", | ||||
|       "resolved": "https://registry.npmjs.org/sax/-/sax-1.4.1.tgz", | ||||
|       "integrity": "sha512-+aWOz7yVScEGoKNd4PA10LZ8sk0A/z5+nXQG5giUO5rprX9jgYsTdov9qCchZiPIZezbZH+jRut8nPodFAX4Jg==", | ||||
|       "license": "ISC" | ||||
|     }, | ||||
|     "node_modules/sitemap": { | ||||
|       "version": "8.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/sitemap/-/sitemap-8.0.2.tgz", | ||||
|       "integrity": "sha512-LwktpJcyZDoa0IL6KT++lQ53pbSrx2c9ge41/SeLTyqy2XUNA6uR4+P9u5IVo5lPeL2arAcOKn1aZAxoYbCKlQ==", | ||||
|       "dependencies": { | ||||
|         "@types/node": "^17.0.5", | ||||
|         "@types/sax": "^1.2.1", | ||||
|         "arg": "^5.0.0", | ||||
|         "sax": "^1.4.1" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "sitemap": "dist/cli.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=14.0.0", | ||||
|         "npm": ">=6.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/undici": { | ||||
|       "version": "7.15.0", | ||||
|       "resolved": "https://registry.npmjs.org/undici/-/undici-7.15.0.tgz", | ||||
|       "integrity": "sha512-7oZJCPvvMvTd0OlqWsIxTuItTpJBpU1tcbVl24FMn3xt3+VSunwUasmfPJRE57oNO1KsZ4PgA1xTdAX4hq8NyQ==", | ||||
|       "engines": { | ||||
|         "node": ">=20.18.1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/whatwg-encoding": { | ||||
|       "version": "3.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-3.1.1.tgz", | ||||
|       "integrity": "sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==", | ||||
|       "dependencies": { | ||||
|         "iconv-lite": "0.6.3" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=18" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/whatwg-mimetype": { | ||||
|       "version": "4.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", | ||||
|       "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==", | ||||
|       "engines": { | ||||
|         "node": ">=18" | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@types/node": { | ||||
|       "version": "17.0.45", | ||||
|       "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.45.tgz", | ||||
|       "integrity": "sha512-w+tIMs3rq2afQdsPJlODhoUEKzFP1ayaoyl1CcnwtIlsVe7K7bA1NGm4s3PraqTLlXnbIN84zuBlxBWo1u9BLw==" | ||||
|     }, | ||||
|     "@types/sax": { | ||||
|       "version": "1.2.7", | ||||
|       "resolved": "https://registry.npmjs.org/@types/sax/-/sax-1.2.7.tgz", | ||||
|       "integrity": "sha512-rO73L89PJxeYM3s3pPPjiPgVVcymqU490g0YO5n5By0k2Erzj6tay/4lr1CHAAU4JyOWd1rpQ8bCf6cZfHU96A==", | ||||
|       "requires": { | ||||
|         "@types/node": "*" | ||||
|       } | ||||
|     }, | ||||
|     "arg": { | ||||
|       "version": "5.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz", | ||||
|       "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==" | ||||
|     }, | ||||
|     "boolbase": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", | ||||
|       "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==" | ||||
|     }, | ||||
|     "cheerio": { | ||||
|       "version": "1.0.0-rc.12", | ||||
|       "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz", | ||||
|       "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==", | ||||
|       "version": "1.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.1.2.tgz", | ||||
|       "integrity": "sha512-IkxPpb5rS/d1IiLbHMgfPuS0FgiWTtFIm/Nj+2woXDLTZ7fOT2eqzgYbdMlLweqlHbsZjxEChoVK+7iph7jyQg==", | ||||
|       "requires": { | ||||
|         "cheerio-select": "^2.1.0", | ||||
|         "dom-serializer": "^2.0.0", | ||||
|         "domhandler": "^5.0.3", | ||||
|         "domutils": "^3.0.1", | ||||
|         "htmlparser2": "^8.0.1", | ||||
|         "parse5": "^7.0.0", | ||||
|         "parse5-htmlparser2-tree-adapter": "^7.0.0" | ||||
|         "domutils": "^3.2.2", | ||||
|         "encoding-sniffer": "^0.2.1", | ||||
|         "htmlparser2": "^10.0.0", | ||||
|         "parse5": "^7.3.0", | ||||
|         "parse5-htmlparser2-tree-adapter": "^7.1.0", | ||||
|         "parse5-parser-stream": "^7.1.2", | ||||
|         "undici": "^7.12.0", | ||||
|         "whatwg-mimetype": "^4.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "cheerio-select": { | ||||
| @@ -268,29 +428,53 @@ | ||||
|       } | ||||
|     }, | ||||
|     "domutils": { | ||||
|       "version": "3.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz", | ||||
|       "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==", | ||||
|       "version": "3.2.2", | ||||
|       "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz", | ||||
|       "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==", | ||||
|       "requires": { | ||||
|         "dom-serializer": "^2.0.0", | ||||
|         "domelementtype": "^2.3.0", | ||||
|         "domhandler": "^5.0.1" | ||||
|         "domhandler": "^5.0.3" | ||||
|       } | ||||
|     }, | ||||
|     "encoding-sniffer": { | ||||
|       "version": "0.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/encoding-sniffer/-/encoding-sniffer-0.2.1.tgz", | ||||
|       "integrity": "sha512-5gvq20T6vfpekVtqrYQsSCFZ1wEg5+wW0/QaZMWkFr6BqD3NfKs0rLCx4rrVlSWJeZb5NBJgVLswK/w2MWU+Gw==", | ||||
|       "requires": { | ||||
|         "iconv-lite": "^0.6.3", | ||||
|         "whatwg-encoding": "^3.1.1" | ||||
|       } | ||||
|     }, | ||||
|     "entities": { | ||||
|       "version": "4.4.0", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz", | ||||
|       "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA==" | ||||
|       "version": "4.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", | ||||
|       "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==" | ||||
|     }, | ||||
|     "htmlparser2": { | ||||
|       "version": "8.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.1.tgz", | ||||
|       "integrity": "sha512-4lVbmc1diZC7GUJQtRQ5yBAeUCL1exyMwmForWkRLnwyzWBFxN633SALPMGYaWZvKe9j1pRZJpauvmxENSp/EA==", | ||||
|       "version": "10.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-10.0.0.tgz", | ||||
|       "integrity": "sha512-TwAZM+zE5Tq3lrEHvOlvwgj1XLWQCtaaibSN11Q+gGBAS7Y1uZSWwXXRe4iF6OXnaq1riyQAPFOBtYc77Mxq0g==", | ||||
|       "requires": { | ||||
|         "domelementtype": "^2.3.0", | ||||
|         "domhandler": "^5.0.2", | ||||
|         "domutils": "^3.0.1", | ||||
|         "entities": "^4.3.0" | ||||
|         "domhandler": "^5.0.3", | ||||
|         "domutils": "^3.2.1", | ||||
|         "entities": "^6.0.0" | ||||
|       }, | ||||
|       "dependencies": { | ||||
|         "entities": { | ||||
|           "version": "6.0.1", | ||||
|           "resolved": "https://registry.npmjs.org/entities/-/entities-6.0.1.tgz", | ||||
|           "integrity": "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==" | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "iconv-lite": { | ||||
|       "version": "0.6.3", | ||||
|       "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", | ||||
|       "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", | ||||
|       "requires": { | ||||
|         "safer-buffer": ">= 2.1.2 < 3.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "nth-check": { | ||||
| @@ -302,21 +486,75 @@ | ||||
|       } | ||||
|     }, | ||||
|     "parse5": { | ||||
|       "version": "7.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz", | ||||
|       "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==", | ||||
|       "version": "7.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.3.0.tgz", | ||||
|       "integrity": "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==", | ||||
|       "requires": { | ||||
|         "entities": "^4.4.0" | ||||
|         "entities": "^6.0.0" | ||||
|       }, | ||||
|       "dependencies": { | ||||
|         "entities": { | ||||
|           "version": "6.0.1", | ||||
|           "resolved": "https://registry.npmjs.org/entities/-/entities-6.0.1.tgz", | ||||
|           "integrity": "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==" | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "parse5-htmlparser2-tree-adapter": { | ||||
|       "version": "7.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz", | ||||
|       "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==", | ||||
|       "version": "7.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.1.0.tgz", | ||||
|       "integrity": "sha512-ruw5xyKs6lrpo9x9rCZqZZnIUntICjQAd0Wsmp396Ul9lN/h+ifgVV1x1gZHi8euej6wTfpqX8j+BFQxF0NS/g==", | ||||
|       "requires": { | ||||
|         "domhandler": "^5.0.2", | ||||
|         "domhandler": "^5.0.3", | ||||
|         "parse5": "^7.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "parse5-parser-stream": { | ||||
|       "version": "7.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/parse5-parser-stream/-/parse5-parser-stream-7.1.2.tgz", | ||||
|       "integrity": "sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow==", | ||||
|       "requires": { | ||||
|         "parse5": "^7.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "safer-buffer": { | ||||
|       "version": "2.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", | ||||
|       "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" | ||||
|     }, | ||||
|     "sax": { | ||||
|       "version": "1.4.1", | ||||
|       "resolved": "https://registry.npmjs.org/sax/-/sax-1.4.1.tgz", | ||||
|       "integrity": "sha512-+aWOz7yVScEGoKNd4PA10LZ8sk0A/z5+nXQG5giUO5rprX9jgYsTdov9qCchZiPIZezbZH+jRut8nPodFAX4Jg==" | ||||
|     }, | ||||
|     "sitemap": { | ||||
|       "version": "8.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/sitemap/-/sitemap-8.0.2.tgz", | ||||
|       "integrity": "sha512-LwktpJcyZDoa0IL6KT++lQ53pbSrx2c9ge41/SeLTyqy2XUNA6uR4+P9u5IVo5lPeL2arAcOKn1aZAxoYbCKlQ==", | ||||
|       "requires": { | ||||
|         "@types/node": "^17.0.5", | ||||
|         "@types/sax": "^1.2.1", | ||||
|         "arg": "^5.0.0", | ||||
|         "sax": "^1.4.1" | ||||
|       } | ||||
|     }, | ||||
|     "undici": { | ||||
|       "version": "7.15.0", | ||||
|       "resolved": "https://registry.npmjs.org/undici/-/undici-7.15.0.tgz", | ||||
|       "integrity": "sha512-7oZJCPvvMvTd0OlqWsIxTuItTpJBpU1tcbVl24FMn3xt3+VSunwUasmfPJRE57oNO1KsZ4PgA1xTdAX4hq8NyQ==" | ||||
|     }, | ||||
|     "whatwg-encoding": { | ||||
|       "version": "3.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-3.1.1.tgz", | ||||
|       "integrity": "sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==", | ||||
|       "requires": { | ||||
|         "iconv-lite": "0.6.3" | ||||
|       } | ||||
|     }, | ||||
|     "whatwg-mimetype": { | ||||
|       "version": "4.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", | ||||
|       "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==" | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -14,7 +14,8 @@ | ||||
|   }, | ||||
|   "homepage": "https://www.nycbookstores.org/", | ||||
|   "dependencies": { | ||||
|     "cheerio": "^1.0.0-rc.12" | ||||
|     "cheerio": "^1.0.0", | ||||
|     "sitemap": "^8.0.0" | ||||
|   }, | ||||
|   "type": "module" | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1 @@ | ||||
| Sitemap: https://nycbookstores.org/sitemap-index.xml | ||||
|   | ||||
| @@ -161,6 +161,16 @@ ul.nav li a:hover { | ||||
|   line-height: 22px; | ||||
|   text-align: justify; | ||||
| } | ||||
| #info details { | ||||
|   font-size: 15px; | ||||
|   padding: 4px; | ||||
|   line-height: 22px; | ||||
|   text-align: justify; | ||||
| } | ||||
| #info details summary { | ||||
|   text-decoration: underline; | ||||
|   cursor: pointer; | ||||
| } | ||||
| 
 | ||||
| #Stores { | ||||
|   margin: 0 auto; | ||||
| @@ -170,7 +180,7 @@ ul.nav li a:hover { | ||||
|   cursor: pointer; | ||||
| } | ||||
| #Stores tr td { | ||||
|   padding: 4px 10px; | ||||
|   padding: 4px 4px 12px; | ||||
| } | ||||
| #Stores tr:not(:last-child) td { | ||||
|   border-bottom: 1px #ddd solid; | ||||
							
								
								
									
										679
									
								
								stores.json
									
									
									
									
									
								
							
							
						
						
									
										679
									
								
								stores.json
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user