Spaces:
Sleeping
Sleeping
| <html lang="en"> | |
| <head> | |
| <meta charset="UTF-8"/> | |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"/> | |
| <title>PlayPulse | Single Explorer</title> | |
| <link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700;800&display=swap" rel="stylesheet"/> | |
| <style> | |
| :root{ | |
| --bg:#0b0e14; --surface:#151921; --surface2:#1c2333; --border:#232a35; | |
| --accent:#3b82f6; --accent-dim:rgba(59,130,246,0.12); --accent-glow:rgba(59,130,246,0.3); | |
| --green:#22c55e; --green-dim:rgba(34,197,94,0.12); | |
| --amber:#f59e0b; --amber-dim:rgba(245,158,11,0.12); | |
| --purple:#a78bfa; --purple-dim:rgba(167,139,250,0.08); | |
| --text:#f1f5f9; --muted:#64748b; --muted2:#94a3b8; | |
| } | |
| *{box-sizing:border-box;margin:0;padding:0;} | |
| ::-webkit-scrollbar{width:5px;height:5px;} | |
| ::-webkit-scrollbar-track{background:transparent;} | |
| ::-webkit-scrollbar-thumb{background:rgba(255,255,255,0.08);border-radius:10px;} | |
| ::-webkit-scrollbar-thumb:hover{background:rgba(255,255,255,0.18);} | |
| *{scrollbar-width:thin;scrollbar-color:rgba(255,255,255,0.08) transparent;} | |
| body{font-family:'Inter',sans-serif;background:var(--bg);color:var(--text);height:100vh;overflow:hidden;display:flex;flex-direction:column;} | |
| /* HEADER */ | |
| .header{height:52px;background:var(--surface);border-bottom:1px solid var(--border);display:flex;align-items:center;padding:0 16px;gap:12px;flex-shrink:0;} | |
| .logo{font-weight:800;font-size:16px;color:var(--accent);display:flex;align-items:center;gap:7px;text-decoration:none;} | |
| nav{display:flex;gap:3px;margin-left:12px;} | |
| .nav-link{color:var(--muted2);text-decoration:none;font-size:12px;font-weight:600;padding:5px 10px;border-radius:7px;transition:.15s;} | |
| .nav-link:hover{color:var(--text);background:var(--surface2);} | |
| .nav-link.active{color:var(--accent);background:var(--accent-dim);} | |
| /* LAYOUT */ | |
| .main{flex:1;display:flex;overflow:hidden;position:relative;} | |
| .sidebar{width:284px;min-width:220px;max-width:460px;background:var(--surface);border-right:1px solid var(--border);display:flex;flex-direction:column;overflow-y:auto;overflow-x:hidden;flex-shrink:0;transition:width .22s ease;} | |
| .sidebar.collapsed{width:0;min-width:0;overflow:hidden;} | |
| .sb-inner{padding:13px 14px;display:flex;flex-direction:column;gap:13px;min-width:220px;} | |
| .resize-handle{width:4px;background:transparent;cursor:col-resize;flex-shrink:0;transition:background .18s;z-index:10;} | |
| .resize-handle:hover,.resize-handle.active{background:var(--accent);} | |
| .collapse-btn{position:absolute;top:50%;transform:translateY(-50%);width:16px;height:36px;background:var(--surface2);border:1px solid var(--border);border-left:none;border-radius:0 7px 7px 0;display:flex;align-items:center;justify-content:center;cursor:pointer;z-index:20;color:var(--muted);transition:background .15s,color .15s;} | |
| .collapse-btn:hover{background:var(--surface);color:var(--text);} | |
| .collapse-btn svg{width:9px;height:9px;fill:none;stroke:currentColor;stroke-width:2.5;transition:transform .22s;} | |
| .content{flex:1;background:var(--bg);position:relative;display:flex;flex-direction:column;overflow:hidden;min-width:0;} | |
| /* SIDEBAR ELEMENTS */ | |
| .label{font-size:9px;font-weight:800;text-transform:uppercase;color:var(--muted);letter-spacing:.8px;display:flex;align-items:center;justify-content:space-between;} | |
| .input-group{display:flex;flex-direction:column;gap:7px;} | |
| input,select{background:var(--bg);border:1px solid var(--border);color:white;padding:8px 10px;border-radius:7px;font-size:12px;outline:none;width:100%;transition:border-color .15s;} | |
| input:focus,select:focus{border-color:var(--accent);} | |
| .toggle-grp{display:flex;background:var(--bg);padding:3px;border-radius:9px;border:1px solid var(--border);} | |
| .toggle-btn{flex:1;padding:7px;border:none;background:transparent;color:var(--muted);cursor:pointer;border-radius:6px;font-weight:600;font-size:11px;transition:.15s;} | |
| .toggle-btn.active{background:var(--accent);color:white;} | |
| .btn-main{background:var(--accent);color:white;border:none;padding:11px;border-radius:9px;font-weight:800;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:8px;transition:.2s;width:100%;font-size:12px;border-bottom:3px solid rgba(0,0,0,.2);} | |
| .btn-main:hover{transform:translateY(-1px);box-shadow:0 4px 14px var(--accent-glow);} | |
| .btn-main:disabled{opacity:.5;cursor:not-allowed;transform:none;box-shadow:none;} | |
| .star-filter-grid{display:flex;flex-direction:column;gap:5px;} | |
| .star-row{display:flex;align-items:center;gap:8px;padding:7px 9px;border-radius:7px;border:1px solid var(--border);background:var(--bg);cursor:pointer;transition:border-color .15s;user-select:none;} | |
| .star-row:hover{border-color:var(--accent);} | |
| .star-row input[type=checkbox]{width:13px;height:13px;accent-color:var(--accent);cursor:pointer;flex-shrink:0;} | |
| .star-label{display:flex;align-items:center;gap:4px;font-size:12px;font-weight:600;flex:1;} | |
| .stars-on{color:var(--amber);letter-spacing:-1px;} | |
| .stars-off{color:var(--border);letter-spacing:-1px;} | |
| .qbtn{font-size:9px;font-weight:700;color:var(--muted2);cursor:pointer;padding:2px 6px;border-radius:4px;border:1px solid var(--border);background:var(--surface2);transition:.15s;} | |
| .qbtn:hover{color:white;border-color:var(--accent);} | |
| .filter-chips{display:flex;flex-wrap:wrap;gap:5px;} | |
| .chip{font-size:10px;font-weight:700;padding:2px 7px;border-radius:16px;background:var(--accent-dim);color:var(--accent);border:1px solid rgba(59,130,246,.3);} | |
| .search-wrap{position:relative;} | |
| .suggestions-box{position:absolute;top:calc(100% + 6px);left:0;right:0;background:var(--surface2);border:1px solid var(--border);border-radius:10px;z-index:1000;max-height:340px;overflow-y:auto;box-shadow:0 14px 40px rgba(0,0,0,.65);} | |
| .suggestion-item{display:flex;align-items:center;padding:10px 12px;gap:10px;cursor:pointer;transition:.15s;border-bottom:1px solid var(--border);} | |
| .suggestion-item:last-child{border-bottom:none;} | |
| .suggestion-item:hover{background:var(--accent-dim);} | |
| .suggestion-item img{width:36px;height:36px;border-radius:8px;object-fit:cover;border:1px solid var(--border);flex-shrink:0;} | |
| .suggestion-info{flex:1;min-width:0;} | |
| .suggestion-title{font-size:12px;font-weight:700;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;color:var(--text);} | |
| .suggestion-sub{font-size:10px;color:var(--muted);margin-top:1px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;} | |
| .suggestion-score{font-size:10px;font-weight:700;color:var(--amber);background:var(--amber-dim);padding:2px 5px;border-radius:4px;flex-shrink:0;} | |
| .suggestion-loading{padding:22px 16px;text-align:center;color:var(--muted);font-size:11px;display:flex;flex-direction:column;align-items:center;gap:8px;} | |
| .spinner-small{width:18px;height:18px;border:2px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .8s linear infinite;} | |
| .recent-item{cursor:pointer;background:var(--bg);padding:8px 9px;border-radius:7px;display:flex;gap:8px;align-items:center;border:1px solid var(--border);transition:.15s;} | |
| .recent-item:hover{border-color:var(--accent);} | |
| /* PERSISTENT TOOLBAR */ | |
| .toolbar{display:flex;align-items:center;gap:12px;padding:6px 14px;background:var(--surface);border-bottom:1px solid var(--border);flex-shrink:0;min-height:48px;} | |
| .tb-search{display:flex;align-items:center;gap:8px;background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:0 12px;height:32px;flex:1;max-width:240px;transition:all .2s ease;} | |
| .tb-search:focus-within{border-color:var(--accent);box-shadow:0 0 0 2px var(--accent-dim);} | |
| .tb-search svg{width:13px;height:13px;fill:none;stroke:var(--muted);stroke-width:2.5;flex-shrink:0;} | |
| .tb-search input{background:transparent;border:none;color:var(--text);font-size:12px;outline:none;width:100%;height:100%;} | |
| .tb-filters{display:flex;align-items:center;gap:6px;} | |
| .tb-sel{background:var(--surface2);border:1px solid var(--border);color:var(--muted2);padding:0 8px;height:32px;border-radius:8px;font-size:11px;font-weight:600;cursor:pointer;outline:none;transition:all .15s;min-width:100px;} | |
| .tb-sel:hover{border-color:var(--muted);color:var(--text);} | |
| .tb-sel:focus{border-color:var(--accent);color:var(--text);background:var(--bg);} | |
| .tb-reset{font-size:11px;font-weight:700;color:var(--muted);cursor:pointer;height:32px;padding:0 12px;border-radius:8px;border:1px solid var(--border);background:transparent;white-space:nowrap;transition:.15s;display:flex;align-items:center;justify-content:center;} | |
| .tb-reset:hover{color:var(--text);border-color:var(--muted);background:var(--surface2);} | |
| .tb-right{display:flex;align-items:center;gap:10px;margin-left:auto;flex-shrink:0;} | |
| .tb-count{font-size:11px;color:var(--muted);font-weight:600;font-variant-numeric:tabular-nums;background:var(--surface2);padding:4px 8px;border-radius:6px;border:1px solid var(--border);} | |
| .view-sw{display:flex;background:var(--surface2);padding:2px;border-radius:8px;border:1px solid var(--border);} | |
| .vbtn{width:28px;height:28px;display:flex;align-items:center;justify-content:center;border-radius:6px;cursor:pointer;color:var(--muted);border:none;background:transparent;transition:.15s;} | |
| .vbtn.active{background:var(--accent);color:white;box-shadow:0 2px 8px var(--accent-glow);} | |
| .vbtn svg{width:14px;height:14px;fill:none;stroke:currentColor;stroke-width:2;} | |
| .view-tabs-sw{display:flex;background:var(--surface2);padding:2px;border-radius:8px;border:1px solid var(--border);} | |
| .vtab{padding:0 12px;height:28px;border-radius:6px;font-size:11px;font-weight:700;cursor:pointer;color:var(--muted);border:none;background:transparent;transition:.15s;display:flex;align-items:center;} | |
| .vtab.active{background:var(--accent);color:white;box-shadow:0 2px 8px var(--accent-glow);} | |
| .btn-icon{width:32px;height:32px;border-radius:8px;border:1px solid var(--border);background:var(--surface2);color:var(--muted);display:flex;align-items:center;justify-content:center;cursor:pointer;transition:.2s;flex-shrink:0;} | |
| .btn-icon:hover{color:white;border-color:var(--accent);background:var(--bg);} | |
| .btn-icon svg{width:15px;height:15px;fill:none;stroke:currentColor;stroke-width:2.5;} | |
| /* SCROLL VIEW */ | |
| .scroll-view{flex:1;overflow-y:auto;padding:14px 16px;display:flex;flex-direction:column;gap:12px;} | |
| /* APP HEADER */ | |
| .app-card{background:var(--surface);border:1px solid var(--border);padding:16px 18px;border-radius:14px;display:flex;gap:15px;align-items:flex-start;} | |
| .app-card img{width:66px;height:66px;border-radius:14px;object-fit:cover;flex-shrink:0;} | |
| .app-stats{display:flex;gap:8px;margin-top:10px;flex-wrap:wrap;} | |
| .stat-pill{background:var(--surface2);border:1px solid var(--border);border-radius:8px;padding:6px 11px;display:flex;flex-direction:column;align-items:center;min-width:54px;} | |
| .stat-val{font-size:14px;font-weight:800;line-height:1;} | |
| .stat-key{font-size:9px;font-weight:700;text-transform:uppercase;color:var(--muted);margin-top:3px;letter-spacing:.4px;} | |
| /* SUMMARY BAR */ | |
| .summary-bar{background:var(--surface);border:1px solid var(--border);border-radius:12px;padding:12px 16px;display:flex;gap:16px;align-items:center;flex-wrap:wrap;} | |
| .star-dist{flex:1;display:flex;flex-direction:column;gap:4px;min-width:140px;} | |
| .star-bar-row{display:flex;align-items:center;gap:6px;font-size:10px;} | |
| .star-bar-track{flex:1;height:4px;background:var(--border);border-radius:3px;overflow:hidden;} | |
| .star-bar-fill{height:100%;border-radius:3px;background:var(--amber);} | |
| /* REVIEW CARD */ | |
| .review-card{background:var(--surface);border:1px solid var(--border);border-radius:12px;overflow:hidden;transition:border-color .15s;} | |
| .review-card:hover{border-color:#2d3a4f;} | |
| .review-main{padding:13px 15px;} | |
| .review-header{display:flex;align-items:center;gap:9px;margin-bottom:8px;} | |
| .user-avatar{width:31px;height:31px;border-radius:50%;background:var(--surface2);flex-shrink:0;display:flex;align-items:center;justify-content:center;font-weight:700;font-size:11px;color:var(--muted2);border:1px solid var(--border);overflow:hidden;} | |
| .user-avatar img{width:31px;height:31px;border-radius:50%;object-fit:cover;} | |
| .review-meta{flex:1;min-width:0;} | |
| .review-username{font-weight:700;font-size:12px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;} | |
| .review-date{font-size:10px;color:var(--muted);margin-top:1px;} | |
| .review-stars{display:flex;gap:1px;flex-shrink:0;} | |
| .review-text{font-size:12px;color:#cbd5e1;line-height:1.6;} | |
| .review-footer{padding:8px 15px;background:var(--bg);border-top:1px solid var(--border);display:flex;flex-wrap:wrap;gap:6px;align-items:center;} | |
| .meta-pill{display:inline-flex;align-items:center;gap:4px;font-size:10px;font-weight:600;padding:3px 8px;border-radius:14px;border:1px solid var(--border);color:var(--muted2);background:var(--surface);} | |
| .meta-pill svg{width:10px;height:10px;fill:none;stroke:currentColor;stroke-width:2.5;flex-shrink:0;} | |
| .meta-pill.thumbs{color:var(--accent);border-color:rgba(59,130,246,.25);background:var(--accent-dim);} | |
| .meta-pill.version{color:var(--purple);border-color:rgba(167,139,250,.25);background:var(--purple-dim);} | |
| .meta-pill.replied{color:var(--green);border-color:rgba(34,197,94,.25);background:var(--green-dim);} | |
| .dev-reply{margin:0 15px 12px;background:var(--surface2);border:1px solid var(--border);border-left:3px solid var(--green);border-radius:8px;padding:10px 12px;} | |
| .dev-reply-header{font-size:10px;font-weight:700;color:var(--green);margin-bottom:5px;display:flex;align-items:center;gap:4px;} | |
| .dev-reply-text{font-size:11px;color:var(--muted2);line-height:1.55;} | |
| .dev-reply-date{font-size:9px;color:var(--muted);margin-top:4px;} | |
| /* SEARCH GRID */ | |
| .search-results-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(250px,1fr));gap:11px;} | |
| .search-app-card{background:var(--surface);border:1px solid var(--border);border-radius:12px;padding:13px;display:flex;gap:11px;cursor:pointer;transition:.18s;} | |
| .search-app-card:hover{border-color:var(--accent);background:var(--surface2);transform:translateY(-2px);} | |
| .search-app-card img{width:48px;height:48px;border-radius:10px;flex-shrink:0;} | |
| .search-app-info{flex:1;min-width:0;display:flex;flex-direction:column;justify-content:space-between;} | |
| .search-app-title{font-size:12px;font-weight:700;color:var(--text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap;} | |
| .search-app-dev{font-size:10px;color:var(--muted);margin-bottom:3px;} | |
| .search-app-meta{display:flex;align-items:center;gap:6px;font-size:10px;font-weight:600;} | |
| .search-app-score{color:var(--amber);} | |
| .search-app-installs{color:var(--muted2);} | |
| /* MISC */ | |
| .loader-overlay{position:absolute;inset:0;background:var(--bg);display:flex;flex-direction:column;align-items:center;justify-content:center;gap:14px;z-index:10;} | |
| .spinner{width:34px;height:34px;border:3px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .8s linear infinite;} | |
| .site-overlay{position:absolute;inset:0;background:var(--bg);display:flex;flex-direction:column;align-items:center;justify-content:center;text-align:center;gap:14px;padding:40px;} | |
| .site-overlay h3{font-size:17px;font-weight:700;} | |
| .site-overlay p{color:var(--muted);font-size:13px;max-width:360px;} | |
| .hidden{display:none;} | |
| @keyframes spin{to{transform:rotate(360deg);}} | |
| /* CHAT */ | |
| #chat-dialer{position:fixed;bottom:22px;right:22px;width:50px;height:50px;background:var(--accent);border-radius:50%;display:flex;align-items:center;justify-content:center;box-shadow:0 8px 28px rgba(59,130,246,.4);cursor:pointer;z-index:1000;transition:.3s cubic-bezier(.175,.885,.32,1.275);border:2px solid rgba(255,255,255,.1);} | |
| #chat-dialer:hover{transform:scale(1.1) rotate(5deg);} | |
| #chat-dialer svg{width:20px;height:20px;color:white;fill:none;stroke:currentColor;stroke-width:2.5;} | |
| #chat-window{position:fixed;bottom:82px;right:22px;width:400px;height:560px;background:var(--surface);border:1px solid var(--border);border-radius:18px;display:flex;flex-direction:column;box-shadow:0 20px 50px rgba(0,0,0,.5);z-index:1001;overflow:hidden;transform:translateY(20px) scale(.95);opacity:0;pointer-events:none;transition:.3s cubic-bezier(.4,0,.2,1);} | |
| #chat-window.open{transform:translateY(0) scale(1);opacity:1;pointer-events:auto;} | |
| .chat-hdr{padding:12px 16px;background:var(--accent);color:white;display:flex;align-items:center;gap:11px;flex-shrink:0;} | |
| .chat-hdr-info{flex:1;} | |
| .chat-hdr-title{font-weight:800;font-size:13px;} | |
| .chat-hdr-status{font-size:10px;opacity:.8;display:flex;align-items:center;gap:4px;} | |
| .sdot{width:5px;height:5px;background:#22c55e;border-radius:50%;} | |
| .chat-clr{background:rgba(255,255,255,.15);border:none;color:white;font-size:11px;padding:3px 8px;border-radius:7px;cursor:pointer;} | |
| .chat-clr:hover{background:rgba(255,255,255,.25);} | |
| .chat-msgs{flex:1;overflow-y:auto;padding:13px;display:flex;flex-direction:column;gap:9px;background-image:radial-gradient(var(--border) 1px,transparent 1px);background-size:20px 20px;} | |
| .msg-row{display:flex;flex-direction:column;gap:4px;} | |
| .msg-row.user{align-items:flex-end;} | |
| .msg-row.bot{align-items:flex-start;} | |
| .message{max-width:88%;padding:9px 13px;border-radius:14px;font-size:12px;line-height:1.6;} | |
| .message.user{background:var(--accent);color:white;border-bottom-right-radius:4px;} | |
| .message.bot{background:var(--surface2);color:var(--text);border:1px solid var(--border);border-bottom-left-radius:4px;white-space:pre-wrap;word-break:break-word;} | |
| .msg-section{margin-top:9px;font-weight:700;font-size:10px;color:var(--accent);letter-spacing:.05em;text-transform:uppercase;} | |
| .msg-item{display:flex;gap:7px;margin-top:4px;} | |
| .msg-item-num,.msg-bullet{font-weight:700;color:var(--accent);min-width:14px;} | |
| .ctw{max-width:100%;overflow-x:auto;border:1px solid var(--border);border-radius:9px;background:var(--surface2);margin-top:4px;} | |
| .ctt{padding:6px 10px;font-size:10px;font-weight:700;color:var(--accent);border-bottom:1px solid var(--border);text-transform:uppercase;} | |
| .ct{width:100%;border-collapse:collapse;font-size:11px;} | |
| .ct th{padding:5px 9px;text-align:left;font-weight:700;font-size:10px;color:var(--muted2);background:var(--bg);border-bottom:1px solid var(--border);} | |
| .ct td{padding:5px 9px;border-bottom:1px solid var(--border);color:var(--text);max-width:150px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;} | |
| .ct tr:last-child td{border-bottom:none;} | |
| .ct tr:hover td{background:var(--surface);} | |
| .typing{display:flex;gap:3px;padding:9px 13px;background:var(--surface2);border:1px solid var(--border);border-radius:13px;width:fit-content;} | |
| .dot{width:5px;height:5px;background:var(--muted);border-radius:50%;animation:bounce 1.4s infinite;} | |
| .dot:nth-child(2){animation-delay:.2s;} | |
| .dot:nth-child(3){animation-delay:.4s;} | |
| @keyframes bounce{0%,80%,100%{transform:translateY(0)}40%{transform:translateY(-5px)}} | |
| .chat-inp-area{padding:11px 13px;background:var(--surface);border-top:1px solid var(--border);display:flex;gap:7px;flex-shrink:0;} | |
| #chat-input{flex:1;background:var(--bg);border:1px solid var(--border);color:white;padding:8px 11px;border-radius:9px;font-size:12px;outline:none;} | |
| #chat-input:focus{border-color:var(--accent);} | |
| .btn-send{width:36px;height:36px;background:var(--accent);color:white;border:none;border-radius:8px;display:flex;align-items:center;justify-content:center;cursor:pointer;flex-shrink:0;} | |
| .btn-send:hover{opacity:.85;} | |
| .btn-send svg{width:14px;height:14px;fill:none;stroke:currentColor;stroke-width:2.5;} | |
| .chat-sugs{display:flex;flex-wrap:wrap;gap:4px;padding:0 13px 7px;} | |
| .sug{font-size:10px;padding:3px 8px;border-radius:16px;background:var(--surface2);border:1px solid var(--border);color:var(--muted2);cursor:pointer;transition:.2s;} | |
| .sug:hover{border-color:var(--accent);color:var(--accent);} | |
| </style> | |
| </head> | |
| <body> | |
| <div class="header"> | |
| <a href="/" class="logo"> | |
| <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"><path d="M12 2v20M17 5H9.5a3.5 3.5 0 0 0 0 7h5a3.5 3.5 0 0 1 0 7H6"/></svg> | |
| PLAYPULSE | |
| </a> | |
| <nav> | |
| <a href="/" class="nav-link">Home</a> | |
| <a href="/scraper" class="nav-link active">Single Explorer</a> | |
| <a href="/batch" class="nav-link">Batch Intelligence</a> | |
| </nav> | |
| <div style="flex:1"></div> | |
| </div> | |
| <div class="main" id="mainLayout"> | |
| <!-- SIDEBAR --> | |
| <aside class="sidebar" id="sidebar"> | |
| <div class="sb-inner"> | |
| <div class="input-group"> | |
| <div class="label">App Identity</div> | |
| <div class="search-wrap"> | |
| <input type="text" id="target" placeholder="App name or Play Store link" value="WhatsApp" autocomplete="off"> | |
| <div class="suggestions-box hidden" id="suggestionsBox"></div> | |
| </div> | |
| </div> | |
| <div class="input-group"> | |
| <div class="label">Amount of Data</div> | |
| <div class="toggle-grp"> | |
| <button class="toggle-btn" id="btnAll" onclick="setMode('all')">Fetch All</button> | |
| <button class="toggle-btn active" id="btnLimit" onclick="setMode('limit')">Custom</button> | |
| </div> | |
| <input type="number" id="manualCount" value="200" placeholder="e.g. 500"> | |
| </div> | |
| <div class="input-group"> | |
| <div class="label">Strategy</div> | |
| <select id="sort"> | |
| <option value="MOST_RELEVANT">Most Relevant</option> | |
| <option value="NEWEST">Newest</option> | |
| <option value="RATING">Top Ratings</option> | |
| </select> | |
| </div> | |
| <div class="input-group"> | |
| <div class="label"> | |
| <span>Star Filter</span> | |
| <div style="display:flex;gap:4px"> | |
| <button class="qbtn" onclick="selectAllStars(true)">All</button> | |
| <button class="qbtn" onclick="selectAllStars(false)">None</button> | |
| </div> | |
| </div> | |
| <div class="star-filter-grid"> | |
| <label class="star-row"><input type="checkbox" class="star-cb" value="5" checked><span class="star-label"><span class="stars-on">β β β β β </span> 5 Stars</span></label> | |
| <label class="star-row"><input type="checkbox" class="star-cb" value="4" checked><span class="star-label"><span class="stars-on">β β β β </span><span class="stars-off">β </span> 4 Stars</span></label> | |
| <label class="star-row"><input type="checkbox" class="star-cb" value="3" checked><span class="star-label"><span class="stars-on">β β β </span><span class="stars-off">β β </span> 3 Stars</span></label> | |
| <label class="star-row"><input type="checkbox" class="star-cb" value="2" checked><span class="star-label"><span class="stars-on">β β </span><span class="stars-off">β β β </span> 2 Stars</span></label> | |
| <label class="star-row"><input type="checkbox" class="star-cb" value="1" checked><span class="star-label"><span class="stars-on">β </span><span class="stars-off">β β β β </span> 1 Star</span></label> | |
| </div> | |
| <div class="filter-chips" id="filterChips"></div> | |
| </div> | |
| <button class="btn-main" id="go" onclick="run()"> | |
| <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="3"><polyline points="13 17 18 12 13 7"/><line x1="6" y1="17" x2="6" y2="7"/></svg> | |
| START SCRAPING | |
| </button> | |
| <div class="input-group"> | |
| <div class="label">Recent Sessions</div> | |
| <div id="recentList" style="display:flex;flex-direction:column;gap:6px;"></div> | |
| </div> | |
| </div> | |
| </aside> | |
| <div class="collapse-btn" id="collapseBtn" onclick="toggleSidebar()" title="Toggle sidebar"> | |
| <svg viewBox="0 0 24 24"><polyline points="15 18 9 12 15 6"/></svg> | |
| </div> | |
| <div class="resize-handle" id="resizeHandle"></div> | |
| <!-- CONTENT --> | |
| <div class="content"> | |
| <!-- PERSISTENT TOOLBAR β never re-rendered --> | |
| <div class="toolbar"> | |
| <div class="tb-search"> | |
| <svg viewBox="0 0 24 24"><circle cx="11" cy="11" r="8"/><line x1="21" y1="21" x2="16.65" y2="16.65"/></svg> | |
| <input type="text" id="tbSearch" placeholder="Search reviewsβ¦" oninput="applyFilters()"> | |
| </div> | |
| <div class="tb-filters"> | |
| <select class="tb-sel" id="tbSort" onchange="applyFilters()"> | |
| <option value="default">Sort: Default</option> | |
| <option value="score_desc">β High β Low</option> | |
| <option value="score_asc">β Low β High</option> | |
| <option value="helpful">Most Helpful</option> | |
| <option value="date_desc">Newest First</option> | |
| <option value="date_asc">Oldest First</option> | |
| </select> | |
| <select class="tb-sel" id="tbReplied" onchange="applyFilters()"> | |
| <option value="all">All Status</option> | |
| <option value="yes">Replied</option> | |
| <option value="no">Unreplied</option> | |
| </select> | |
| <select class="tb-sel" id="tbHelpful" onchange="applyFilters()"> | |
| <option value="all">Any Help</option> | |
| <option value="10">10+ Likes</option> | |
| <option value="50">50+ Likes</option> | |
| <option value="100">100+ Likes</option> | |
| </select> | |
| <button class="tb-reset" onclick="resetFilters()" title="Reset Filters"> | |
| <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" style="margin-right:4px"><path d="M3 12a9 9 0 1 0 9-9 9.75 9.75 0 0 0-6.74 2.74L3 8"/><path d="M3 3v5h5"/></svg> | |
| Reset | |
| </button> | |
| </div> | |
| <div class="tb-right"> | |
| <span class="tb-count" id="tbCount"></span> | |
| <div class="view-sw"> | |
| <button class="vbtn active" id="vbtnList" onclick="setView('list')" title="List view"> | |
| <svg viewBox="0 0 24 24"><line x1="8" y1="6" x2="21" y2="6"/><line x1="8" y1="12" x2="21" y2="12"/><line x1="8" y1="18" x2="21" y2="18"/><circle cx="3" cy="6" r="1" fill="currentColor" stroke="none"/><circle cx="3" cy="12" r="1" fill="currentColor" stroke="none"/><circle cx="3" cy="18" r="1" fill="currentColor" stroke="none"/></svg> | |
| </button> | |
| <button class="vbtn" id="vbtnGrid" onclick="setView('grid')" title="Grid view"> | |
| <svg viewBox="0 0 24 24"><rect x="3" y="3" width="7" height="7" rx="1"/><rect x="14" y="3" width="7" height="7" rx="1"/><rect x="14" y="14" width="7" height="7" rx="1"/><rect x="3" y="14" width="7" height="7" rx="1"/></svg> | |
| </button> | |
| </div> | |
| <div class="view-tabs-sw"> | |
| <button class="vtab active" id="tabData" onclick="switchPane('data')">Reviews</button> | |
| <button class="vtab" id="tabSite" onclick="switchPane('site')">Live Store</button> | |
| </div> | |
| <div class="btn-icon" onclick="downloadCSV()" title="Export results to CSV"> | |
| <svg viewBox="0 0 24 24"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4M7 10l5 5 5-5M12 15V3"/></svg> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- DATA PANE --> | |
| <div id="dataPane" class="scroll-view"> | |
| <div id="welcome" style="display:flex;flex-direction:column;align-items:center;justify-content:center;flex:1;color:var(--muted);gap:13px;text-align:center;padding:40px;"> | |
| <svg width="50" height="50" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2"><path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z"/><polyline points="14 2 14 8 20 8"/></svg> | |
| <div><p style="font-size:14px;font-weight:700;color:var(--muted2);margin-bottom:4px">No data loaded</p><p style="font-size:12px">Enter an app name or Play Store link and hit Start Scraping</p></div> | |
| </div> | |
| <div id="appHeader" class="hidden"></div> | |
| <div id="summaryBar" class="hidden"></div> | |
| <div id="reviewList" class="hidden"></div> | |
| </div> | |
| <!-- LOADER --> | |
| <div id="loader" class="loader-overlay hidden"> | |
| <div class="spinner"></div> | |
| <p style="color:var(--muted);font-size:13px" id="loaderMsg">Connecting to serversβ¦</p> | |
| </div> | |
| <!-- SITE PANE --> | |
| <div id="sitePane" class="hidden" style="flex:1;position:relative;"> | |
| <div class="site-overlay"> | |
| <h3>Web View Shielded</h3> | |
| <p>Google Play Store blocks embedding. Open it directly below.</p> | |
| <button class="btn-main" style="width:auto;padding:11px 22px" onclick="openTarget()"> | |
| Open on Google Play | |
| <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5"><path d="M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6M15 3h6v6M10 14L21 3"/></svg> | |
| </button> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- CHAT --> | |
| <div id="chat-dialer" onclick="toggleChat()"> | |
| <svg viewBox="0 0 24 24"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg> | |
| </div> | |
| <div id="chat-window"> | |
| <div class="chat-hdr"> | |
| <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="2.5"><circle cx="12" cy="12" r="3"/><path d="M12 1v4M12 19v4M4.22 4.22l2.83 2.83M16.95 16.95l2.83 2.83M1 12h4M19 12h4M4.22 19.78l2.83-2.83M16.95 7.05l2.83-2.83"/></svg> | |
| <div class="chat-hdr-info"> | |
| <div class="chat-hdr-title">PlayPulse Intelligence</div> | |
| <div class="chat-hdr-status"><span class="sdot"></span> Agent Online</div> | |
| </div> | |
| <div style="display:flex;gap:7px;align-items:center"> | |
| <button class="chat-clr" onclick="clearChat()">Clear</button> | |
| <div style="cursor:pointer;opacity:.7" onclick="toggleChat()"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="2.5"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></div> | |
| </div> | |
| </div> | |
| <div class="chat-msgs" id="chat-messages"> | |
| <div class="msg-row bot"><div class="message bot">π Hi! I'm PlayPulse Intelligence. Ask me anything about the loaded reviews β issues, sentiment, or say <em>"show in table"</em>.</div></div> | |
| </div> | |
| <div class="chat-sugs"> | |
| <div class="sug" onclick="fillChat('What are the main issues?')">Main issues</div> | |
| <div class="sug" onclick="fillChat('Show most helpful reviews')">Most helpful</div> | |
| <div class="sug" onclick="fillChat('Show 1 star reviews in table')">1β table</div> | |
| <div class="sug" onclick="fillChat('Summarize overall sentiment')">Sentiment</div> | |
| </div> | |
| <div class="chat-inp-area"> | |
| <input type="text" id="chat-input" placeholder="Ask about the reviewsβ¦" onkeydown="if(event.key==='Enter') sendChat()"> | |
| <button class="btn-send" onclick="sendChat()"><svg viewBox="0 0 24 24"><line x1="22" y1="2" x2="11" y2="13"/><polygon points="22 2 15 22 11 13 2 9 22 2"/></svg></button> | |
| </div> | |
| </div> | |
| <script> | |
| </script> | |
| </body> | |
| </html> |