Compare commits
	
		
			51 Commits
		
	
	
		
			new-redesi
			...
			45c625994b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 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 | |||
| 9528b39d6d | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,2 +1,3 @@
 | 
				
			|||||||
.DS_Store
 | 
					.DS_Store
 | 
				
			||||||
node_modules
 | 
					node_modules
 | 
				
			||||||
 | 
					build
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,17 +6,35 @@
 | 
				
			|||||||
      depth: 15
 | 
					      depth: 15
 | 
				
			||||||
 | 
					
 | 
				
			||||||
steps:
 | 
					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:
 | 
					  build:
 | 
				
			||||||
    image: docker
 | 
					    image: docker
 | 
				
			||||||
    commands:
 | 
					    commands:
 | 
				
			||||||
      - apk add curl
 | 
					      - 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 build -t registry.yetaga.in/bookstores:latest .
 | 
				
			||||||
      - docker push 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/fetch -H "Authorization: Bearer $COMPOSE_TOKEN"'
 | 
				
			||||||
      - 'curl http://100.113.98.36:4000/api/update -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:
 | 
					    when:
 | 
				
			||||||
      branch: "master"
 | 
					      - event: push
 | 
				
			||||||
 | 
					        branch:
 | 
				
			||||||
 | 
					          include: ["main"]
 | 
				
			||||||
    volumes:
 | 
					    volumes:
 | 
				
			||||||
      - /var/run/docker.sock:/var/run/docker.sock
 | 
					      - /var/run/docker.sock:/var/run/docker.sock
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,12 +1,9 @@
 | 
				
			|||||||
FROM node:20 AS builder
 | 
					FROM node:24 AS builder
 | 
				
			||||||
 | 
					
 | 
				
			||||||
COPY . /src
 | 
					COPY . /src
 | 
				
			||||||
WORKDIR /src
 | 
					WORKDIR /src
 | 
				
			||||||
RUN npm install && node ./index.js
 | 
					RUN npm install && node ./index.js
 | 
				
			||||||
 | 
					
 | 
				
			||||||
FROM caddy:2.7.5
 | 
					FROM caddy:2.10.0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
COPY img /usr/share/caddy/img
 | 
					COPY --from=builder /src/build /usr/share/caddy
 | 
				
			||||||
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
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										941
									
								
								index.html
									
									
									
									
									
								
							
							
						
						
									
										941
									
								
								index.html
									
									
									
									
									
								
							@@ -1,941 +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?1697075405981"
 | 
					 | 
				
			||||||
    />
 | 
					 | 
				
			||||||
    <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">114</span> stores indexed
 | 
					 | 
				
			||||||
            on this page. Last updated
 | 
					 | 
				
			||||||
            <span id="updatedOn">October 11, 2023</span>.
 | 
					 | 
				
			||||||
          </p>
 | 
					 | 
				
			||||||
          <details>
 | 
					 | 
				
			||||||
            <summary>Recent Changes</summary>
 | 
					 | 
				
			||||||
            <ul id="changesList">
 | 
					 | 
				
			||||||
              <li>October 11, 2023 - add this fancy recent changes module</li>
 | 
					 | 
				
			||||||
              <li>
 | 
					 | 
				
			||||||
                October 8, 2023 - move Burnt Books into Hey Kids!, update
 | 
					 | 
				
			||||||
                archestratus to note there's no cafe anymore
 | 
					 | 
				
			||||||
              </li>
 | 
					 | 
				
			||||||
              <li>September 30, 2023 - add Lofty Pigeon Books</li>
 | 
					 | 
				
			||||||
              <li>September 19, 2023 - Move Yu and Me to Essex Market</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>234 Water St, 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">Black Spring Books</td>
 | 
					 | 
				
			||||||
                <td>672 Driggs Ave, 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>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">Idlewild Books</td>
 | 
					 | 
				
			||||||
                <td>170 7th Avenue S, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="48" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">James Cummins Bookseller</td>
 | 
					 | 
				
			||||||
                <td>699 Madison Ave, 7th Floor, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="49" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Joanne Hendricks Cookbooks</td>
 | 
					 | 
				
			||||||
                <td>488 Greenwich St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="50" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Karma Bookstore</td>
 | 
					 | 
				
			||||||
                <td>136 East Third St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="51" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Kew & Willow Books</td>
 | 
					 | 
				
			||||||
                <td>81-63 Lefferts Boulevard, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="52" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Kinokunya</td>
 | 
					 | 
				
			||||||
                <td>1073 Avenue of the Americas, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="53" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Kitchen Arts & Letters</td>
 | 
					 | 
				
			||||||
                <td>1435 Lexington Ave, 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 (Upper West Side)
 | 
					 | 
				
			||||||
                </td>
 | 
					 | 
				
			||||||
                <td>2020 Broadway, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="86" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Sister's Uptown Bookstore</td>
 | 
					 | 
				
			||||||
                <td>1942 Amsterdam Ave, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="87" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Spoonbill & Sugartown, Booksellers</td>
 | 
					 | 
				
			||||||
                <td>218 Bedford Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="88" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Standards Manual</td>
 | 
					 | 
				
			||||||
                <td>212 Franklin Street, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="89" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Strand Bookstore</td>
 | 
					 | 
				
			||||||
                <td>828 Broadway, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="90" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Sweet Pickle Books</td>
 | 
					 | 
				
			||||||
                <td>47 Orchard St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="91" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Taylor & Co. Books</td>
 | 
					 | 
				
			||||||
                <td>1021 Cortelyou Rd, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="92" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Terrace Books</td>
 | 
					 | 
				
			||||||
                <td>242 Prospect Park West, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="93" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Austin Book Shop</td>
 | 
					 | 
				
			||||||
                <td>104-29 Jamaica Ave, Richmond Hill</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="94" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Book Cellar</td>
 | 
					 | 
				
			||||||
                <td>1465 York Ave, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="95" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Center for Fiction</td>
 | 
					 | 
				
			||||||
                <td>15 Lafayette Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="96" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Corner Bookstore</td>
 | 
					 | 
				
			||||||
                <td>1313 Madison Ave, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="97" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Drama Book Shop</td>
 | 
					 | 
				
			||||||
                <td>266 W 39th St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="98" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Lit. Bar</td>
 | 
					 | 
				
			||||||
                <td>131 Alexander Ave, Bronx</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="99" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Mysterious Bookshop</td>
 | 
					 | 
				
			||||||
                <td>58 Warren St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="100" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Ripped Bodice</td>
 | 
					 | 
				
			||||||
                <td>218 5th Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="101" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Strand At Columbus Ave</td>
 | 
					 | 
				
			||||||
                <td>450 Columbus Ave, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="102" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">The Word Is Change</td>
 | 
					 | 
				
			||||||
                <td>368 Tompkins Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="103" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Three Lives & Company</td>
 | 
					 | 
				
			||||||
                <td>238 West 10th St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="104" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Topos Bookstore Cafe</td>
 | 
					 | 
				
			||||||
                <td>788 Woodward Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="105" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Troubled Sleep Books</td>
 | 
					 | 
				
			||||||
                <td>129 6th Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="106" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Turn The Page... Again</td>
 | 
					 | 
				
			||||||
                <td>39-15a Bell Blvd, Flushing</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="107" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Unnameable Books</td>
 | 
					 | 
				
			||||||
                <td>600 Vanderbilt Ave, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="108" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Ursus Books</td>
 | 
					 | 
				
			||||||
                <td>50 East 78th St, Suite 1C, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="109" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Westsider Rare & Used Books</td>
 | 
					 | 
				
			||||||
                <td>2246 Broadway, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="110" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Westsider Records</td>
 | 
					 | 
				
			||||||
                <td>233 West 72nd St, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="111" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Word Bookstore</td>
 | 
					 | 
				
			||||||
                <td>126 Franklin St, Brooklyn</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="112" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Word Up Books</td>
 | 
					 | 
				
			||||||
                <td>2113 Amsterdam Ave, New York</td>
 | 
					 | 
				
			||||||
              </tr>
 | 
					 | 
				
			||||||
              <tr id="113" class="spotRow">
 | 
					 | 
				
			||||||
                <td class="name">Yu and Me Books</td>
 | 
					 | 
				
			||||||
                <td>115 Delancey St Stall #11, 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>
 | 
					 | 
				
			||||||
							
								
								
									
										153
									
								
								index.js
									
									
									
									
									
								
							
							
						
						
									
										153
									
								
								index.js
									
									
									
									
									
								
							@@ -1,7 +1,58 @@
 | 
				
			|||||||
import { load } from "cheerio";
 | 
					import { load } from "cheerio";
 | 
				
			||||||
import { readFile, writeFile } from "fs";
 | 
					import { readFile, writeFileSync, mkdirSync, cpSync, rmSync } from "fs";
 | 
				
			||||||
import process from "child_process";
 | 
					import process from "child_process";
 | 
				
			||||||
import stores from "./stores.json" assert { type: "json" };
 | 
					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() {
 | 
					function GetRecentChanges() {
 | 
				
			||||||
  const res = process
 | 
					  const res = process
 | 
				
			||||||
@@ -17,6 +68,7 @@ function ChangeLog(logs) {
 | 
				
			|||||||
    if (
 | 
					    if (
 | 
				
			||||||
      i > 3 ||
 | 
					      i > 3 ||
 | 
				
			||||||
      l.includes("[skip]") ||
 | 
					      l.includes("[skip]") ||
 | 
				
			||||||
 | 
					      l.includes("[ignore]") ||
 | 
				
			||||||
      l.includes("caddy") ||
 | 
					      l.includes("caddy") ||
 | 
				
			||||||
      l.includes("renovate")
 | 
					      l.includes("renovate")
 | 
				
			||||||
    ) {
 | 
					    ) {
 | 
				
			||||||
@@ -43,15 +95,67 @@ function TableViewTemplate(rows) {
 | 
				
			|||||||
  return table + "</table>";
 | 
					  return table + "</table>";
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function TableRowTemplate({ rowNumber, name, address, city }) {
 | 
					function TableRowTemplate({ rowNumber, name, slug, address, city }) {
 | 
				
			||||||
  return `
 | 
					  return `
 | 
				
			||||||
  <tr id="${rowNumber}" class="spotRow">
 | 
					  <tr id="${rowNumber}" class="spotRow">
 | 
				
			||||||
    <td class="name">${name}</td>
 | 
					    <td class="name"><a href="/${slugify(name)}/">${name}</a></td>
 | 
				
			||||||
    <td>${address}, ${city}</td>
 | 
					    <td><a href="/${slugify(name)}/">${address}, ${city}</a></td>
 | 
				
			||||||
  </tr>`;
 | 
					  </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();
 | 
					  const changeList = GetRecentChanges();
 | 
				
			||||||
  if (err) {
 | 
					  if (err) {
 | 
				
			||||||
    throw err;
 | 
					    throw err;
 | 
				
			||||||
@@ -76,8 +180,39 @@ readFile("./index.html", function (err, data) {
 | 
				
			|||||||
  $("#changesList").html(ChangeLog(changeList));
 | 
					  $("#changesList").html(ChangeLog(changeList));
 | 
				
			||||||
  const cssurl = $("link[type='text/css']").attr("href").split("?")[0];
 | 
					  const cssurl = $("link[type='text/css']").attr("href").split("?")[0];
 | 
				
			||||||
  $("link[type='text/css']").attr("href", cssurl + "?" + new Date().getTime());
 | 
					  $("link[type='text/css']").attr("href", cssurl + "?" + new Date().getTime());
 | 
				
			||||||
  writeFile("./index.html", $.html(), (err) => {
 | 
					
 | 
				
			||||||
    if (err) throw err;
 | 
					  rmSync("./build", { recursive: true, force: true });
 | 
				
			||||||
    console.log("Default view updated.");
 | 
					
 | 
				
			||||||
 | 
					  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>
 | 
				
			||||||
							
								
								
									
										355
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										355
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							@@ -9,29 +9,55 @@
 | 
				
			|||||||
      "version": "1.0.0",
 | 
					      "version": "1.0.0",
 | 
				
			||||||
      "license": "BSD-3-Clause",
 | 
					      "license": "BSD-3-Clause",
 | 
				
			||||||
      "dependencies": {
 | 
					      "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": {
 | 
					    "node_modules/boolbase": {
 | 
				
			||||||
      "version": "1.0.0",
 | 
					      "version": "1.0.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww=="
 | 
					      "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "node_modules/cheerio": {
 | 
					    "node_modules/cheerio": {
 | 
				
			||||||
      "version": "1.0.0-rc.12",
 | 
					      "version": "1.1.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.1.2.tgz",
 | 
				
			||||||
      "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==",
 | 
					      "integrity": "sha512-IkxPpb5rS/d1IiLbHMgfPuS0FgiWTtFIm/Nj+2woXDLTZ7fOT2eqzgYbdMlLweqlHbsZjxEChoVK+7iph7jyQg==",
 | 
				
			||||||
      "dependencies": {
 | 
					      "dependencies": {
 | 
				
			||||||
        "cheerio-select": "^2.1.0",
 | 
					        "cheerio-select": "^2.1.0",
 | 
				
			||||||
        "dom-serializer": "^2.0.0",
 | 
					        "dom-serializer": "^2.0.0",
 | 
				
			||||||
        "domhandler": "^5.0.3",
 | 
					        "domhandler": "^5.0.3",
 | 
				
			||||||
        "domutils": "^3.0.1",
 | 
					        "domutils": "^3.2.2",
 | 
				
			||||||
        "htmlparser2": "^8.0.1",
 | 
					        "encoding-sniffer": "^0.2.1",
 | 
				
			||||||
        "parse5": "^7.0.0",
 | 
					        "htmlparser2": "^10.0.0",
 | 
				
			||||||
        "parse5-htmlparser2-tree-adapter": "^7.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": {
 | 
					      "engines": {
 | 
				
			||||||
        "node": ">= 6"
 | 
					        "node": ">=20.18.1"
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
      "funding": {
 | 
					      "funding": {
 | 
				
			||||||
        "url": "https://github.com/cheeriojs/cheerio?sponsor=1"
 | 
					        "url": "https://github.com/cheeriojs/cheerio?sponsor=1"
 | 
				
			||||||
@@ -118,22 +144,34 @@
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "node_modules/domutils": {
 | 
					    "node_modules/domutils": {
 | 
				
			||||||
      "version": "3.0.1",
 | 
					      "version": "3.2.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz",
 | 
				
			||||||
      "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==",
 | 
					      "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==",
 | 
				
			||||||
      "dependencies": {
 | 
					      "dependencies": {
 | 
				
			||||||
        "dom-serializer": "^2.0.0",
 | 
					        "dom-serializer": "^2.0.0",
 | 
				
			||||||
        "domelementtype": "^2.3.0",
 | 
					        "domelementtype": "^2.3.0",
 | 
				
			||||||
        "domhandler": "^5.0.1"
 | 
					        "domhandler": "^5.0.3"
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
      "funding": {
 | 
					      "funding": {
 | 
				
			||||||
        "url": "https://github.com/fb55/domutils?sponsor=1"
 | 
					        "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": {
 | 
					    "node_modules/entities": {
 | 
				
			||||||
      "version": "4.4.0",
 | 
					      "version": "4.5.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA==",
 | 
					      "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==",
 | 
				
			||||||
      "engines": {
 | 
					      "engines": {
 | 
				
			||||||
        "node": ">=0.12"
 | 
					        "node": ">=0.12"
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
@@ -142,9 +180,9 @@
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "node_modules/htmlparser2": {
 | 
					    "node_modules/htmlparser2": {
 | 
				
			||||||
      "version": "8.0.1",
 | 
					      "version": "10.0.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-10.0.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-4lVbmc1diZC7GUJQtRQ5yBAeUCL1exyMwmForWkRLnwyzWBFxN633SALPMGYaWZvKe9j1pRZJpauvmxENSp/EA==",
 | 
					      "integrity": "sha512-TwAZM+zE5Tq3lrEHvOlvwgj1XLWQCtaaibSN11Q+gGBAS7Y1uZSWwXXRe4iF6OXnaq1riyQAPFOBtYc77Mxq0g==",
 | 
				
			||||||
      "funding": [
 | 
					      "funding": [
 | 
				
			||||||
        "https://github.com/fb55/htmlparser2?sponsor=1",
 | 
					        "https://github.com/fb55/htmlparser2?sponsor=1",
 | 
				
			||||||
        {
 | 
					        {
 | 
				
			||||||
@@ -154,9 +192,31 @@
 | 
				
			|||||||
      ],
 | 
					      ],
 | 
				
			||||||
      "dependencies": {
 | 
					      "dependencies": {
 | 
				
			||||||
        "domelementtype": "^2.3.0",
 | 
					        "domelementtype": "^2.3.0",
 | 
				
			||||||
        "domhandler": "^5.0.2",
 | 
					        "domhandler": "^5.0.3",
 | 
				
			||||||
        "domutils": "^3.0.1",
 | 
					        "domutils": "^3.2.1",
 | 
				
			||||||
        "entities": "^4.3.0"
 | 
					        "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": {
 | 
					    "node_modules/nth-check": {
 | 
				
			||||||
@@ -171,47 +231,148 @@
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "node_modules/parse5": {
 | 
					    "node_modules/parse5": {
 | 
				
			||||||
      "version": "7.1.2",
 | 
					      "version": "7.3.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.3.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==",
 | 
					      "integrity": "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==",
 | 
				
			||||||
      "dependencies": {
 | 
					      "dependencies": {
 | 
				
			||||||
        "entities": "^4.4.0"
 | 
					        "entities": "^6.0.0"
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
      "funding": {
 | 
					      "funding": {
 | 
				
			||||||
        "url": "https://github.com/inikulin/parse5?sponsor=1"
 | 
					        "url": "https://github.com/inikulin/parse5?sponsor=1"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "node_modules/parse5-htmlparser2-tree-adapter": {
 | 
					    "node_modules/parse5-htmlparser2-tree-adapter": {
 | 
				
			||||||
      "version": "7.0.0",
 | 
					      "version": "7.1.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.1.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==",
 | 
					      "integrity": "sha512-ruw5xyKs6lrpo9x9rCZqZZnIUntICjQAd0Wsmp396Ul9lN/h+ifgVV1x1gZHi8euej6wTfpqX8j+BFQxF0NS/g==",
 | 
				
			||||||
      "dependencies": {
 | 
					      "dependencies": {
 | 
				
			||||||
        "domhandler": "^5.0.2",
 | 
					        "domhandler": "^5.0.3",
 | 
				
			||||||
        "parse5": "^7.0.0"
 | 
					        "parse5": "^7.0.0"
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
      "funding": {
 | 
					      "funding": {
 | 
				
			||||||
        "url": "https://github.com/inikulin/parse5?sponsor=1"
 | 
					        "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.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/sitemap/-/sitemap-8.0.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-+AbdxhM9kJsHtruUF39bwS/B0Fytw6Fr1o4ZAIAEqA6cke2xcoO2GleBw9Zw7nRzILVEgz7zBM5GiTJjie1G9A==",
 | 
				
			||||||
 | 
					      "license": "MIT",
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "@types/node": "^17.0.5",
 | 
				
			||||||
 | 
					        "@types/sax": "^1.2.1",
 | 
				
			||||||
 | 
					        "arg": "^5.0.0",
 | 
				
			||||||
 | 
					        "sax": "^1.2.4"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "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": {
 | 
					  "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": {
 | 
					    "boolbase": {
 | 
				
			||||||
      "version": "1.0.0",
 | 
					      "version": "1.0.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww=="
 | 
					      "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "cheerio": {
 | 
					    "cheerio": {
 | 
				
			||||||
      "version": "1.0.0-rc.12",
 | 
					      "version": "1.1.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.1.2.tgz",
 | 
				
			||||||
      "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==",
 | 
					      "integrity": "sha512-IkxPpb5rS/d1IiLbHMgfPuS0FgiWTtFIm/Nj+2woXDLTZ7fOT2eqzgYbdMlLweqlHbsZjxEChoVK+7iph7jyQg==",
 | 
				
			||||||
      "requires": {
 | 
					      "requires": {
 | 
				
			||||||
        "cheerio-select": "^2.1.0",
 | 
					        "cheerio-select": "^2.1.0",
 | 
				
			||||||
        "dom-serializer": "^2.0.0",
 | 
					        "dom-serializer": "^2.0.0",
 | 
				
			||||||
        "domhandler": "^5.0.3",
 | 
					        "domhandler": "^5.0.3",
 | 
				
			||||||
        "domutils": "^3.0.1",
 | 
					        "domutils": "^3.2.2",
 | 
				
			||||||
        "htmlparser2": "^8.0.1",
 | 
					        "encoding-sniffer": "^0.2.1",
 | 
				
			||||||
        "parse5": "^7.0.0",
 | 
					        "htmlparser2": "^10.0.0",
 | 
				
			||||||
        "parse5-htmlparser2-tree-adapter": "^7.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": {
 | 
					    "cheerio-select": {
 | 
				
			||||||
@@ -268,29 +429,53 @@
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "domutils": {
 | 
					    "domutils": {
 | 
				
			||||||
      "version": "3.0.1",
 | 
					      "version": "3.2.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz",
 | 
				
			||||||
      "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==",
 | 
					      "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==",
 | 
				
			||||||
      "requires": {
 | 
					      "requires": {
 | 
				
			||||||
        "dom-serializer": "^2.0.0",
 | 
					        "dom-serializer": "^2.0.0",
 | 
				
			||||||
        "domelementtype": "^2.3.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": {
 | 
					    "entities": {
 | 
				
			||||||
      "version": "4.4.0",
 | 
					      "version": "4.5.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA=="
 | 
					      "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "htmlparser2": {
 | 
					    "htmlparser2": {
 | 
				
			||||||
      "version": "8.0.1",
 | 
					      "version": "10.0.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-10.0.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-4lVbmc1diZC7GUJQtRQ5yBAeUCL1exyMwmForWkRLnwyzWBFxN633SALPMGYaWZvKe9j1pRZJpauvmxENSp/EA==",
 | 
					      "integrity": "sha512-TwAZM+zE5Tq3lrEHvOlvwgj1XLWQCtaaibSN11Q+gGBAS7Y1uZSWwXXRe4iF6OXnaq1riyQAPFOBtYc77Mxq0g==",
 | 
				
			||||||
      "requires": {
 | 
					      "requires": {
 | 
				
			||||||
        "domelementtype": "^2.3.0",
 | 
					        "domelementtype": "^2.3.0",
 | 
				
			||||||
        "domhandler": "^5.0.2",
 | 
					        "domhandler": "^5.0.3",
 | 
				
			||||||
        "domutils": "^3.0.1",
 | 
					        "domutils": "^3.2.1",
 | 
				
			||||||
        "entities": "^4.3.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=="
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "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": {
 | 
					    "nth-check": {
 | 
				
			||||||
@@ -302,21 +487,75 @@
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "parse5": {
 | 
					    "parse5": {
 | 
				
			||||||
      "version": "7.1.2",
 | 
					      "version": "7.3.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.3.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==",
 | 
					      "integrity": "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==",
 | 
				
			||||||
      "requires": {
 | 
					      "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": {
 | 
					    "parse5-htmlparser2-tree-adapter": {
 | 
				
			||||||
      "version": "7.0.0",
 | 
					      "version": "7.1.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.1.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==",
 | 
					      "integrity": "sha512-ruw5xyKs6lrpo9x9rCZqZZnIUntICjQAd0Wsmp396Ul9lN/h+ifgVV1x1gZHi8euej6wTfpqX8j+BFQxF0NS/g==",
 | 
				
			||||||
      "requires": {
 | 
					      "requires": {
 | 
				
			||||||
        "domhandler": "^5.0.2",
 | 
					        "domhandler": "^5.0.3",
 | 
				
			||||||
        "parse5": "^7.0.0"
 | 
					        "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.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/sitemap/-/sitemap-8.0.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-+AbdxhM9kJsHtruUF39bwS/B0Fytw6Fr1o4ZAIAEqA6cke2xcoO2GleBw9Zw7nRzILVEgz7zBM5GiTJjie1G9A==",
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@types/node": "^17.0.5",
 | 
				
			||||||
 | 
					        "@types/sax": "^1.2.1",
 | 
				
			||||||
 | 
					        "arg": "^5.0.0",
 | 
				
			||||||
 | 
					        "sax": "^1.2.4"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "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/",
 | 
					  "homepage": "https://www.nycbookstores.org/",
 | 
				
			||||||
  "dependencies": {
 | 
					  "dependencies": {
 | 
				
			||||||
    "cheerio": "^1.0.0-rc.12"
 | 
					    "cheerio": "^1.0.0",
 | 
				
			||||||
 | 
					    "sitemap": "^8.0.0"
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "type": "module"
 | 
					  "type": "module"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -0,0 +1 @@
 | 
				
			|||||||
 | 
					Sitemap: https://nycbookstores.org/sitemap-index.xml
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -180,7 +180,7 @@ ul.nav li a:hover {
 | 
				
			|||||||
  cursor: pointer;
 | 
					  cursor: pointer;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#Stores tr td {
 | 
					#Stores tr td {
 | 
				
			||||||
  padding: 4px 10px;
 | 
					  padding: 4px 4px 12px;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#Stores tr:not(:last-child) td {
 | 
					#Stores tr:not(:last-child) td {
 | 
				
			||||||
  border-bottom: 1px #ddd solid;
 | 
					  border-bottom: 1px #ddd solid;
 | 
				
			||||||
							
								
								
									
										535
									
								
								stores.json
									
									
									
									
									
								
							
							
						
						
									
										535
									
								
								stores.json
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user