rewrite-into-separate-pages (#16)
	
		
			
	
		
	
	
		
	
		
			All checks were successful
		
		
	
	
		
			
				
	
				ci/woodpecker/push/woodpecker Pipeline was successful
				
			
		
		
	
	
				
					
				
			
		
			All checks were successful
		
		
	
	ci/woodpecker/push/woodpecker Pipeline was successful
				
			Reviewed-on: #16 Co-authored-by: David Ashby <delta.mu.alpha@gmail.com> Co-committed-by: David Ashby <delta.mu.alpha@gmail.com>
This commit is contained in:
		
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,2 +1,3 @@ | ||||
| .DS_Store | ||||
| node_modules | ||||
| build | ||||
|   | ||||
| @@ -6,7 +6,4 @@ RUN npm install && node ./index.js | ||||
|  | ||||
| FROM caddy:2.9.1 | ||||
|  | ||||
| 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 | ||||
|   | ||||
							
								
								
									
										968
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										968
									
								
								index.html
									
									
									
									
									
								
							| @@ -1,968 +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?1709780697849" | ||||
|     /> | ||||
|     <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="#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">118</span> stores indexed | ||||
|             on this page. Last updated | ||||
|             <span id="updatedOn">March 6, 2024</span>. | ||||
|           </p> | ||||
|           <details> | ||||
|             <summary>Recent Changes</summary> | ||||
|             <ul id="changesList"> | ||||
|               <li> | ||||
|                 January 28, 2024 - move Yu and Me back to its original location | ||||
|                 (congrats!) | ||||
|               </li> | ||||
|               <li> | ||||
|                 January 1, 2024 - Add Bibliotheque and The World's Borough | ||||
|                 Bookshop | ||||
|               </li> | ||||
|               <li> | ||||
|                 December 27, 2023 - add La Joie de Vivre, a French/English | ||||
|                 bookstore in the Flatiron Distirct | ||||
|               </li> | ||||
|               <li>December 21, 2023 - add canonical link</li> | ||||
|             </ul> | ||||
|           </details> | ||||
|         </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 St, 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>115 Ralph Ave, 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>164 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>36-19 30th St, Astoria</td> | ||||
|               </tr> | ||||
|               <tr id="9" class="spotRow"> | ||||
|                 <td class="name">Berl's Brooklyn Poetry Shop</td> | ||||
|                 <td>126A Front St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="10" class="spotRow"> | ||||
|                 <td class="name">Better Read Than Dead</td> | ||||
|                 <td>867 Broadway, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="11" class="spotRow"> | ||||
|                 <td class="name">Better Read Than Dead & Burly Coffee</td> | ||||
|                 <td>90 Kosciuszko St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="12" class="spotRow"> | ||||
|                 <td class="name">Bibliotheque</td> | ||||
|                 <td>54 Mercer St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="13" class="spotRow"> | ||||
|                 <td class="name">Black Spring Books</td> | ||||
|                 <td>672 Driggs Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="14" class="spotRow"> | ||||
|                 <td class="name">Bluestockings Cooperative</td> | ||||
|                 <td>116 Suffolk Street, New York</td> | ||||
|               </tr> | ||||
|               <tr id="15" class="spotRow"> | ||||
|                 <td class="name">Bonnie Slotnick Cookbooks</td> | ||||
|                 <td>28 East 2nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="16" class="spotRow"> | ||||
|                 <td class="name">Book Club Bar</td> | ||||
|                 <td>197 E 3rd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="17" class="spotRow"> | ||||
|                 <td class="name">Book Culture</td> | ||||
|                 <td>536 W 112th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="18" class="spotRow"> | ||||
|                 <td class="name">Book Culture LIC</td> | ||||
|                 <td>26-09 Jackson Ave, Queens</td> | ||||
|               </tr> | ||||
|               <tr id="19" class="spotRow"> | ||||
|                 <td class="name">Book Culture on Broadway</td> | ||||
|                 <td>2915 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="20" class="spotRow"> | ||||
|                 <td class="name">Book Thug Nation</td> | ||||
|                 <td>100 North 3rd St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="21" class="spotRow"> | ||||
|                 <td class="name">Bookmarc</td> | ||||
|                 <td>400 Bleecker St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="22" class="spotRow"> | ||||
|                 <td class="name">BookMark Shoppe</td> | ||||
|                 <td>8415 3rd Avenue, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="23" class="spotRow"> | ||||
|                 <td class="name">Bookoff</td> | ||||
|                 <td>49 W 45nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="24" class="spotRow"> | ||||
|                 <td class="name">Bookoff Brooklyn</td> | ||||
|                 <td>934 3rd Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="25" class="spotRow"> | ||||
|                 <td class="name">Books Are Magic (Montague St.)</td> | ||||
|                 <td>122 Montague St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="26" class="spotRow"> | ||||
|                 <td class="name">Books Are Magic (Smith St.)</td> | ||||
|                 <td>225 Smith St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="27" class="spotRow"> | ||||
|                 <td class="name">Books Of Wonder</td> | ||||
|                 <td>42 West 17th St, New York</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>157 Huron St, 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>985 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 Ave, 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">Here's A Book Store</td> | ||||
|                 <td>1964 Coney Island Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="44" class="spotRow"> | ||||
|                 <td class="name">Hey Kids! Comics</td> | ||||
|                 <td>157 Huron St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="45" class="spotRow"> | ||||
|                 <td class="name">Housing Works Bookstore Café</td> | ||||
|                 <td>126 Crosby St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="46" class="spotRow"> | ||||
|                 <td class="name">Human Relations Books</td> | ||||
|                 <td>1067 Flushing Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="47" class="spotRow"> | ||||
|                 <td class="name">James Cummins Bookseller</td> | ||||
|                 <td>699 Madison Ave, 7th Floor, New York</td> | ||||
|               </tr> | ||||
|               <tr id="48" class="spotRow"> | ||||
|                 <td class="name">Joanne Hendricks Cookbooks</td> | ||||
|                 <td>488 Greenwich St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="49" class="spotRow"> | ||||
|                 <td class="name">Karma Bookstore</td> | ||||
|                 <td>136 East Third St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="50" class="spotRow"> | ||||
|                 <td class="name">Kew & Willow Books</td> | ||||
|                 <td>81-63 Lefferts Boulevard, New York</td> | ||||
|               </tr> | ||||
|               <tr id="51" class="spotRow"> | ||||
|                 <td class="name">Kinokunya</td> | ||||
|                 <td>1073 Avenue of the Americas, New York</td> | ||||
|               </tr> | ||||
|               <tr id="52" class="spotRow"> | ||||
|                 <td class="name">Kitchen Arts & Letters</td> | ||||
|                 <td>1435 Lexington Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="53" class="spotRow"> | ||||
|                 <td class="name">La Joie de Vivre</td> | ||||
|                 <td>145 W 27th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="54" class="spotRow"> | ||||
|                 <td class="name">Leaves Bookstore</td> | ||||
|                 <td>140 Nassau Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="55" class="spotRow"> | ||||
|                 <td class="name">Left Bank Books</td> | ||||
|                 <td>41 Perry St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="56" class="spotRow"> | ||||
|                 <td class="name">Lofty Pigeon Books</td> | ||||
|                 <td>743 Church Ave, Brooklyn</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>134 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 St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="70" class="spotRow"> | ||||
|                 <td class="name">Passageway Books</td> | ||||
|                 <td>150 9th Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="71" class="spotRow"> | ||||
|                 <td class="name">Pillow-Cat Books</td> | ||||
|                 <td>328 East 9th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="72" class="spotRow"> | ||||
|                 <td class="name">Posman Books Chelsea Market</td> | ||||
|                 <td>75 9th Avenue, New York</td> | ||||
|               </tr> | ||||
|               <tr id="73" class="spotRow"> | ||||
|                 <td class="name">POWERHOUSE @ IC</td> | ||||
|                 <td>220 36th St, Building #2, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="74" class="spotRow"> | ||||
|                 <td class="name">POWERHOUSE Arena</td> | ||||
|                 <td>28 Adams St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="75" class="spotRow"> | ||||
|                 <td class="name">POWERHOUSE on 8th</td> | ||||
|                 <td>1111 8th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="76" class="spotRow"> | ||||
|                 <td class="name">Printed Matter</td> | ||||
|                 <td>231 11th Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="77" class="spotRow"> | ||||
|                 <td class="name">Printed Matter St. Marks</td> | ||||
|                 <td>38 St. Marks Pl, New York</td> | ||||
|               </tr> | ||||
|               <tr id="78" class="spotRow"> | ||||
|                 <td class="name">Quest Bookshop</td> | ||||
|                 <td>240 E 53rd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="79" class="spotRow"> | ||||
|                 <td class="name">Quimby's Bookstore</td> | ||||
|                 <td>536 Metropolitan Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="80" class="spotRow"> | ||||
|                 <td class="name">Recirculation</td> | ||||
|                 <td>876 Riverside Dr, New York</td> | ||||
|               </tr> | ||||
|               <tr id="81" class="spotRow"> | ||||
|                 <td class="name">Respect For Life Books-N-Things</td> | ||||
|                 <td>537 Nostrand Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="82" class="spotRow"> | ||||
|                 <td class="name">Revolution Books</td> | ||||
|                 <td>437 Malcolm X Blvd, New York</td> | ||||
|               </tr> | ||||
|               <tr id="83" class="spotRow"> | ||||
|                 <td class="name">Rizzoli Bookstore</td> | ||||
|                 <td>1133 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="84" class="spotRow"> | ||||
|                 <td class="name">Shakespeare & Company</td> | ||||
|                 <td>939 Lexington Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="85" class="spotRow"> | ||||
|                 <td class="name"> | ||||
|                   Shakespeare & Company (Manhattan Valley/Columbia | ||||
|                   University) | ||||
|                 </td> | ||||
|                 <td>2736 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="86" class="spotRow"> | ||||
|                 <td class="name"> | ||||
|                   Shakespeare & Company (Upper West Side) | ||||
|                 </td> | ||||
|                 <td>2020 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="87" class="spotRow"> | ||||
|                 <td class="name">Sister's Uptown Bookstore</td> | ||||
|                 <td>1942 Amsterdam Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="88" class="spotRow"> | ||||
|                 <td class="name">Spoonbill & Sugartown, Booksellers</td> | ||||
|                 <td>218 Bedford Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="89" class="spotRow"> | ||||
|                 <td class="name">Standards Manual</td> | ||||
|                 <td>212 Franklin Street, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="90" class="spotRow"> | ||||
|                 <td class="name">Strand Bookstore</td> | ||||
|                 <td>828 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="91" class="spotRow"> | ||||
|                 <td class="name">Sweet Pickle Books</td> | ||||
|                 <td>47 Orchard St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="92" class="spotRow"> | ||||
|                 <td class="name">Taylor & Co. Books</td> | ||||
|                 <td>1021 Cortelyou Rd, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="93" class="spotRow"> | ||||
|                 <td class="name">Terrace Books</td> | ||||
|                 <td>242 Prospect Park West, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="94" class="spotRow"> | ||||
|                 <td class="name">The Austin Book Shop</td> | ||||
|                 <td>104-29 Jamaica Ave, Richmond Hill</td> | ||||
|               </tr> | ||||
|               <tr id="95" class="spotRow"> | ||||
|                 <td class="name">The Book Cellar</td> | ||||
|                 <td>1465 York Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="96" class="spotRow"> | ||||
|                 <td class="name">The Center for Fiction</td> | ||||
|                 <td>15 Lafayette Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="97" class="spotRow"> | ||||
|                 <td class="name">The Corner Bookstore</td> | ||||
|                 <td>1313 Madison Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="98" class="spotRow"> | ||||
|                 <td class="name">The Drama Book Shop</td> | ||||
|                 <td>266 W 39th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="99" class="spotRow"> | ||||
|                 <td class="name">The Lit. Bar</td> | ||||
|                 <td>131 Alexander Ave, Bronx</td> | ||||
|               </tr> | ||||
|               <tr id="100" class="spotRow"> | ||||
|                 <td class="name">The Mysterious Bookshop</td> | ||||
|                 <td>58 Warren St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="101" class="spotRow"> | ||||
|                 <td class="name">The Ripped Bodice</td> | ||||
|                 <td>218 5th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="102" class="spotRow"> | ||||
|                 <td class="name">The Strand At Columbus Ave</td> | ||||
|                 <td>450 Columbus Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="103" class="spotRow"> | ||||
|                 <td class="name">The Word Is Change</td> | ||||
|                 <td>368 Tompkins Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="104" class="spotRow"> | ||||
|                 <td class="name">The World's Borough Bookshop</td> | ||||
|                 <td>3406 73rd St, Queens</td> | ||||
|               </tr> | ||||
|               <tr id="105" class="spotRow"> | ||||
|                 <td class="name">Three Lives & Company</td> | ||||
|                 <td>154 W 10th St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="106" class="spotRow"> | ||||
|                 <td class="name">Topos Bookstore Cafe</td> | ||||
|                 <td>788 Woodward Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="107" class="spotRow"> | ||||
|                 <td class="name">Topos Too</td> | ||||
|                 <td>59-22 Myrtle Ave, Queens</td> | ||||
|               </tr> | ||||
|               <tr id="108" class="spotRow"> | ||||
|                 <td class="name">Troubled Sleep Books</td> | ||||
|                 <td>129 6th Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="109" class="spotRow"> | ||||
|                 <td class="name">Turn The Page... Again</td> | ||||
|                 <td>39-15a Bell Blvd, Flushing</td> | ||||
|               </tr> | ||||
|               <tr id="110" class="spotRow"> | ||||
|                 <td class="name">Unnameable Books</td> | ||||
|                 <td>600 Vanderbilt Ave, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="111" class="spotRow"> | ||||
|                 <td class="name">Ursus Books</td> | ||||
|                 <td>50 East 78th St, Suite 1C, New York</td> | ||||
|               </tr> | ||||
|               <tr id="112" class="spotRow"> | ||||
|                 <td class="name">Village Works</td> | ||||
|                 <td>12 St. Marks Pl, New York</td> | ||||
|               </tr> | ||||
|               <tr id="113" class="spotRow"> | ||||
|                 <td class="name">Westsider Rare & Used Books</td> | ||||
|                 <td>2246 Broadway, New York</td> | ||||
|               </tr> | ||||
|               <tr id="114" class="spotRow"> | ||||
|                 <td class="name">Westsider Records</td> | ||||
|                 <td>233 West 72nd St, New York</td> | ||||
|               </tr> | ||||
|               <tr id="115" class="spotRow"> | ||||
|                 <td class="name">Word Bookstore</td> | ||||
|                 <td>126 Franklin St, Brooklyn</td> | ||||
|               </tr> | ||||
|               <tr id="116" class="spotRow"> | ||||
|                 <td class="name">Word Up Books</td> | ||||
|                 <td>2113 Amsterdam Ave, New York</td> | ||||
|               </tr> | ||||
|               <tr id="117" 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> | ||||
							
								
								
									
										130
									
								
								index.js
									
									
									
									
									
								
							
							
						
						
									
										130
									
								
								index.js
									
									
									
									
									
								
							| @@ -1,8 +1,52 @@ | ||||
| import { load } from "cheerio"; | ||||
| import { readFile, writeFile } from "fs"; | ||||
| import { readFile, writeFileSync, mkdirSync, cpSync, rmSync } from "fs"; | ||||
| import process from "child_process"; | ||||
| 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"') | ||||
| @@ -44,15 +88,67 @@ 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; | ||||
| @@ -77,8 +173,26 @@ readFile("./index.html", function (err, data) { | ||||
|   $("#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"); | ||||
|  | ||||
|   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)); | ||||
|     mkDir(`./build/${slugify(store.name)}`); | ||||
|     writeFile(`./build/${slugify(store.name)}/index.html`, $.html()); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
							
								
								
									
										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> | ||||
| @@ -180,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; | ||||
							
								
								
									
										372
									
								
								stores.json
									
									
									
									
									
								
							
							
						
						
									
										372
									
								
								stores.json
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user