{
  "version": "1.0.0",
  "exported_at": "2026-05-31T09:20:00.000Z",
  "project": {
    "name": "Google Maps Store Listing Scraper Japan",
    "description": "Scrapes Google Maps Japan store listings for the default keyword '居酒屋　東京'. This corrected version avoids an infinite loop by using a bounded JavaScript infinite-scroll collector: it scrolls the Google Maps results feed, captures unique result cards into a hidden export table, then exports Octoparse-equivalent fields: keyword, title, Maps URL, rating, reviews, category, address, website/phone when visible, hours, coordinates, and scrape time. Google Maps may rate-limit automation, change class names, virtualize results, or hide website/phone unless detail pages are opened. Increase maxScrolls inside the JavaScript block for larger result sets.",
    "color": "bg-[#4589ff]",
    "template_id": "ai-generated"
  },
  "blocks": [
    {
      "block_id": "navigate-1",
      "block_type": "process",
      "title": "Navigate",
      "description": "Go to a URL",
      "position_x": 120,
      "position_y": 220,
      "config": {
        "url": "https://www.google.co.jp/maps/search/%E5%B1%85%E9%85%92%E5%B1%8B%E3%80%80%E6%9D%B1%E4%BA%AC?hl=ja",
        "color": "bg-[#4589ff]",
        "tags": [
          "google-maps",
          "japan",
          "keyword-search"
        ]
      }
    },
    {
      "block_id": "wait-for-page-load-1",
      "block_type": "process",
      "title": "Wait for Page Load",
      "description": "Wait for page to finish loading",
      "position_x": 480,
      "position_y": 220,
      "config": {
        "timeout": 45
      }
    },
    {
      "block_id": "wait-for-element-1",
      "block_type": "process",
      "title": "Wait for Element",
      "description": "Wait until element appears",
      "position_x": 840,
      "position_y": 220,
      "config": {
        "selector": "div[role=\"article\"][aria-label]",
        "timeout": 45,
        "visible": true
      }
    },
    {
      "block_id": "inject-javascript-1",
      "block_type": "process",
      "title": "Inject JavaScript",
      "description": "Execute custom JavaScript",
      "position_x": 1200,
      "position_y": 520,
      "config": {
        "waitForCompletion": true,
        "timeout": 180,
        "jsCode": "return (async () => { const sleep = (ms) => new Promise(r => setTimeout(r, ms)); const maxScrolls = 35; const scrollDelayMs = 1200; const keyword = (() => { try { const decoded = decodeURIComponent(location.pathname); const m = decoded.match(/\\/maps\\/search\\/([^/?]+)/); return m ? m[1].replace(/\\+/g, ' ') : '居酒屋　東京'; } catch (e) { return '居酒屋　東京'; } })(); const clean = (s) => (s || '').replace(/\\s+/g, ' ').trim(); const rows = new Map(); const parseCard = (card) => { const link = card.querySelector('a.hfpxzc'); const url = link ? link.href : ''; const title = clean((card.querySelector('.qBF1Pd') || card.querySelector('[class*=fontHeadline]') || {}).textContent) || clean(card.getAttribute('aria-label')); const rating = clean((card.querySelector('.MW4etd') || {}).textContent); let review_count = clean((card.querySelector('.UY7F9') || {}).textContent).replace(/[()]/g, ''); if (!review_count) { const label = Array.from(card.querySelectorAll('[aria-label]')).map(e => e.getAttribute('aria-label') || '').find(v => v.includes('クチコミ')); const m = label && label.match(/クチコミ\\s*([0-9,]+)\\s*件/); review_count = m ? m[1] : ''; } const lines = (card.innerText || '').split('\\n').map(clean).filter(Boolean); const infoLine = lines.find(l => l.includes('·') && !l.includes('営業') && !l.includes('￥') && !l.includes('つ星') && !l.startsWith('\"')) || ''; const infoParts = infoLine.split('·').map(clean).filter(p => p && !/[]/.test(p)); const category = infoParts[0] || ''; const address = infoParts.length > 1 ? infoParts[infoParts.length - 1] : ''; const hours = lines.find(l => l.includes('営業')) || ''; const websiteLink = Array.from(card.querySelectorAll('a[href^=\"http\"]')).find(a => !a.href.includes('google.') && !a.href.includes('maps/reserve')); const website = websiteLink ? websiteLink.href : ''; const phoneLink = card.querySelector('a[href^=\"tel:\"]'); const phone = phoneLink ? phoneLink.href.replace('tel:', '') : ''; const coordMatch = url.match(/!3d(-?\\d+(?:\\.\\d+)?)!4d(-?\\d+(?:\\.\\d+)?)/); const latitude = coordMatch ? coordMatch[1] : ''; const longitude = coordMatch ? coordMatch[2] : ''; const key = url || title; if (key && title) { rows.set(key, { search_keyword: keyword, title, maps_url: url, rating, review_count, category, address, website, phone, hours, longitude, latitude, scraped_at: new Date().toISOString() }); } }; const collect = () => Array.from(document.querySelectorAll('div[role=\"article\"][aria-label]')).forEach(parseCard); const feed = document.querySelector('div[role=\"feed\"]') || document.querySelector('.m6QErb[role=\"feed\"]'); let stagnantRounds = 0; let lastCount = 0; collect(); for (let i = 0; i < maxScrolls; i++) { if (feed) { feed.scrollTop = feed.scrollHeight; feed.dispatchEvent(new WheelEvent('wheel', { deltaY: 3500, bubbles: true, cancelable: true })); } else { window.scrollTo(0, document.body.scrollHeight); } await sleep(scrollDelayMs); collect(); const pageText = (feed ? feed.innerText : document.body.innerText) || ''; if (/リストの最後|すべての結果|これ以上/.test(pageText)) break; if (rows.size === lastCount) stagnantRounds += 1; else stagnantRounds = 0; lastCount = rows.size; if (stagnantRounds >= 6) break; } collect(); let host = document.querySelector('#uscraper-gmaps-data'); if (host) host.remove(); host = document.createElement('div'); host.id = 'uscraper-gmaps-data'; host.setAttribute('data-count', String(rows.size)); host.style.cssText = 'position:absolute;left:0;top:0;width:1px;height:1px;overflow:hidden;opacity:0;pointer-events:none;'; document.body.appendChild(host); for (const item of rows.values()) { const el = document.createElement('div'); el.className = 'uscraper-row'; for (const [k, v] of Object.entries(item)) el.setAttribute('data-' + k.replace(/_/g, '-'), v || ''); host.appendChild(el); } return { collected: rows.size }; })();"
      }
    },
    {
      "block_id": "wait-for-element-2",
      "block_type": "process",
      "title": "Wait for Element",
      "description": "Wait until element appears",
      "position_x": 1560,
      "position_y": 520,
      "config": {
        "selector": "#uscraper-gmaps-data .uscraper-row",
        "timeout": 30,
        "visible": false
      }
    },
    {
      "block_id": "structured-export-1",
      "block_type": "process",
      "title": "Structured Export",
      "description": "Export data with custom columns",
      "position_x": 1920,
      "position_y": 520,
      "config": {
        "rowSelector": "#uscraper-gmaps-data .uscraper-row",
        "fileName": "google-maps-store-listing-scraper.csv",
        "saveLocation": "C:\\Users\\theskd\\Documents\\UScraper\\templates",
        "includeHeaders": true,
        "fileMode": "create",
        "columns": [
          {
            "name": "search_keyword",
            "selector": "",
            "attribute": "data-search-keyword"
          },
          {
            "name": "title",
            "selector": "",
            "attribute": "data-title"
          },
          {
            "name": "maps_url",
            "selector": "",
            "attribute": "data-maps-url"
          },
          {
            "name": "rating",
            "selector": "",
            "attribute": "data-rating"
          },
          {
            "name": "review_count",
            "selector": "",
            "attribute": "data-review-count"
          },
          {
            "name": "category",
            "selector": "",
            "attribute": "data-category"
          },
          {
            "name": "address",
            "selector": "",
            "attribute": "data-address"
          },
          {
            "name": "website",
            "selector": "",
            "attribute": "data-website"
          },
          {
            "name": "phone",
            "selector": "",
            "attribute": "data-phone"
          },
          {
            "name": "hours",
            "selector": "",
            "attribute": "data-hours"
          },
          {
            "name": "longitude",
            "selector": "",
            "attribute": "data-longitude"
          },
          {
            "name": "latitude",
            "selector": "",
            "attribute": "data-latitude"
          },
          {
            "name": "scraped_at",
            "selector": "",
            "attribute": "data-scraped-at"
          }
        ]
      }
    },
    {
      "block_id": "end-1",
      "block_type": "output",
      "title": "End",
      "description": "Terminate execution flow",
      "position_x": 2280,
      "position_y": 520,
      "config": {}
    }
  ],
  "connections": [
    {
      "from_block_id": "navigate-1",
      "from_connector_id": "right",
      "to_block_id": "wait-for-page-load-1",
      "to_connector_id": "left"
    },
    {
      "from_block_id": "wait-for-page-load-1",
      "from_connector_id": "right",
      "to_block_id": "wait-for-element-1",
      "to_connector_id": "left"
    },
    {
      "from_block_id": "wait-for-element-1",
      "from_connector_id": "right",
      "to_block_id": "inject-javascript-1",
      "to_connector_id": "left"
    },
    {
      "from_block_id": "inject-javascript-1",
      "from_connector_id": "right",
      "to_block_id": "wait-for-element-2",
      "to_connector_id": "left"
    },
    {
      "from_block_id": "wait-for-element-2",
      "from_connector_id": "right",
      "to_block_id": "structured-export-1",
      "to_connector_id": "left"
    },
    {
      "from_block_id": "structured-export-1",
      "from_connector_id": "right",
      "to_block_id": "end-1",
      "to_connector_id": "left"
    }
  ],
  "canvas_elements": [
    {
      "id": "group-load",
      "element_type": "group",
      "title": "Page Load",
      "color": "#08bdba",
      "position_x": 48,
      "position_y": 116,
      "width": 1760,
      "height": 596,
      "z_index": 20,
      "data": {
        "memberBlockIds": [
          "navigate-1",
          "wait-for-page-load-1",
          "wait-for-element-1",
          "wait-for-element-2"
        ]
      }
    },
    {
      "id": "group-interaction",
      "element_type": "group",
      "title": "Interaction",
      "color": "#a56eff",
      "position_x": 1128,
      "position_y": 416,
      "width": 380,
      "height": 296,
      "z_index": 20,
      "data": {
        "memberBlockIds": [
          "inject-javascript-1"
        ]
      }
    },
    {
      "id": "group-extract",
      "element_type": "group",
      "title": "Data Extraction",
      "color": "#42be65",
      "position_x": 1848,
      "position_y": 416,
      "width": 380,
      "height": 296,
      "z_index": 20,
      "data": {
        "memberBlockIds": [
          "structured-export-1"
        ]
      }
    },
    {
      "id": "group-control",
      "element_type": "group",
      "title": "Control Flow",
      "color": "#8d8d8d",
      "position_x": 2208,
      "position_y": 416,
      "width": 380,
      "height": 296,
      "z_index": 20,
      "data": {
        "memberBlockIds": [
          "end-1"
        ]
      }
    },
    {
      "id": "note-overview",
      "element_type": "note",
      "title": "Overview",
      "content": "Scrapes Google Maps Japan store listings for the default keyword '居酒屋　東京'. This corrected version avoids an infinite loop by using a bounded JavaScript infinite-scroll collector: it scrolls the Google Maps results feed, captures unique result cards into a hidden export table, then exports Octoparse-equivalent fields: keyword, title, Maps URL, rating, reviews, category, address, website/phone when visible, hours, coordinates, and scrape time. Google Maps may rate-limit automation, change class names, virtualize results, or hide website/phone unless detail pages are opened. Increase maxScrolls inside the JavaScript block for larger result sets.",
      "color": "#f1c21b",
      "position_x": 80,
      "position_y": 20,
      "width": 480,
      "height": 160,
      "z_index": 22,
      "data": {}
    },
    {
      "id": "note-block-inject-javascript-1",
      "element_type": "note",
      "title": "Note: Inject JavaScript",
      "content": "Runs custom JavaScript in the page: `return (async () => { const sleep = (ms) => new Promise(r => setTimeout(r, ms)); const maxScrolls = ...` Verify in browser if results are empty.",
      "color": "#ee5396",
      "position_x": 1400,
      "position_y": 500,
      "width": 340,
      "height": 140,
      "z_index": 22,
      "data": {
        "block_id": "inject-javascript-1"
      }
    },
    {
      "id": "note-block-structured-export-1",
      "element_type": "note",
      "title": "Note: Structured Export",
      "content": "Extracts rows matching `#uscraper-gmaps-data .uscraper-row`. Confirm row count > 0 before running at scale.",
      "color": "#ee5396",
      "position_x": 2120,
      "position_y": 500,
      "width": 340,
      "height": 115,
      "z_index": 22,
      "data": {
        "block_id": "structured-export-1"
      }
    }
  ]
}