startpage

Simple text centric startpage
Log | Files | Refs | README

index.html (26351B)


      1 <!doctype html>
      2 <html lang="en">
      3   <head>
      4     <meta charset="UTF-8">
      5     <title>New Tab</title>
      6     <link rel="icon" type="image/png" sizes="16x16" href="">
      7     <link rel="icon" type="image/png" sizes="16x16" media="(prefers-color-scheme: light)" href="">
      8     <link rel="icon" type="image/png" sizes="16x16" media="(prefers-color-scheme: dark)" href="">
      9     <style>
     10       #prediction,body{font-family:monospace;font-size:1rem}
     11       #prediction:not(:empty){opacity:1}
     12       #prediction{background-color:#1d1d1d;color:#88c0d0cc;box-shadow:.1rem .1rem .7rem #151515;padding: 1rem 1.8rem 1rem 1.8rem;width:100%;margin:auto;opacity:0;border-radius: 0 0 .3rem .3rem;position:relative;z-index:-1}
     13       .input-container input{all:unset;background-color:#151515;color:#6c99bb;border-radius:.3rem;box-shadow:.1rem .1rem .7rem #151515;padding:1.8rem;width:100%;margin:2rem auto 0;display:block}
     14       .input-container:has(#prediction:not(:empty)) input{border-radius:.3rem .3rem 0 0}
     15       .input-container{position:relative;width:100%}
     16       body{width:56vw;margin:25vh auto;background-color:#2e3440;color:#d6dce8;overflow-y:hidden}
     17       input{margin:2rem auto}
     18       li.highlight,li:hover{text-shadow:2px 2px 8px #fff;text-decoration:underline;cursor:pointer}
     19       li{width:fit-content;float:left;margin:.3rem 1rem .3rem 0}
     20       p,svg{vertical-align:middle}
     21       svg{width:1rem;height:1rem;padding-right:.5rem}
     22       ul{list-style:none;padding:0;height:30vh;mask-image:linear-gradient(180deg, #000 10%, transparent)}
     23       ul:hover{mask-image:unset;height:unset}
     24     </style>
     25   </head>
     26   <body>
     27     <p id="datetime">
     28       <svg fill="none" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
     29         <path d="m12 8v4l3 3m6-3c0 4.9706-4.0294 9-9 9-4.97056 0-9-4.0294-9-9 0-4.97056 4.02944-9 9-9 4.9706 0 9 4.02944 9 9z" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"/>
     30       </svg><span id="datetime-text"></span>
     31     </p>
     32     <div class="input-container">
     33       <input type="text" id="queryInput" placeholder="Enter bookmark or search prefix..." autocomplete="off" autofocus>
     34       <p id="prediction"></p>
     35     </div>
     36     <ul id="bookmarkList"></ul>
     37     <script>
     38       const bookmarks = [
     39       { name: "/g/ Technology", url: "https://boards.4channel.org/g/" },
     40       { name: "/r/Roms Megathread", url: "https://r-roms.github.io/" },
     41       { name: "Aftenposten", url: "https://www.aftenposten.no/" },
     42       { name: "AuroraWatch", url: "https://aurorawatch.lancs.ac.uk/" },
     43       { name: "Bank Holidays", url: "https://www.gov.uk/bank-holidays" },
     44       { name: "BBC News", url: "https://www.bbc.co.uk/news" },
     45       { name: "Beeper", url: "https://chat.beeper.com" },
     46       { name: "BestInvest Funds", url: "https://www.bestinvest.co.uk/v2/investment-search#/" },
     47       { name: "Bitcoin Rainbow", url: "https://www.blockchaincenter.net/bitcoin-rainbow-chart/" },
     48       { name: "BorgBase", url: "https://www.borgbase.com/repositories" },
     49       { name: "ChatGPT", url: "https://chatgpt.com/" },
     50       { name: "Cloudflare", url: "https://dash.cloudflare.com/" },
     51       { name: "Codecademy", url: "https://www.codecademy.com/learn" },
     52       { name: "CoinMarketCap Portfolio", url: "https://coinmarketcap.com/portfolio-tracker/" },
     53       { name: "CryptoVision", url: "https://cryptovision.live/" },
     54       { name: "CSS Minifier", url: "https://www.toptal.com/developers/cssminifier/" },
     55       { name: "DrayTek Firmware", url: "https://fw.draytek.com.tw/" },
     56       { name: "Drive Prices", url: "https://diskprices.com/?locale=uk" },
     57       { name: "Duolingo", url: "https://www.duolingo.com/" },
     58       { name: "Duolinguists", url: "https://duolinguists.wordpress.com/i-speak-english/" },
     59       { name: "eBay Balance", url: "https://www.ebay.co.uk/mes/summary" },
     60       { name: "eBay Dispatch", url: "https://www.ebay.co.uk/sh/ord/?filter=status%3AAWAITING_SHIPMENT" },
     61       { name: "Emulation General Wiki", url: "https://emulation.gametechwiki.com/index.php/Main_Page" },
     62       { name: "Epic Games", url: "https://store.epicgames.com/en-US/free-games" },
     63       { name: "EpisodeFeed", url: "https://episodefeed.com/my-feed#currently-in-feed" },
     64       { name: "Fanatical", url: "https://www.fanatical.com/en/" },
     65       { name: "Filter Lists", url: "https://filterlists.com/" },
     66       { name: "FitGirl", url: "https://fitgirl-repacks.site/" },
     67       { name: "Flagle", url: "https://www.flagle.io/" },
     68       { name: "Flight Radar 24", url: "https://www.flightradar24.com/" },
     69       { name: "FreshRSS", url: "https://rss.minskio.co.uk/" },
     70       { name: "GitHub", url: "https://github.com/breadcat/" },
     71       { name: "HackerNews", url: "https://news.ycombinator.com/news" },
     72       { name: "Hemnet", url: "https://www.hemnet.se/" },
     73       { name: "Home AP", url: "https://192.168.1.2/" },
     74       { name: "Home Router", url: "https://192.168.1.1/" },
     75       { name: "Hugo Local", url: "http://127.0.0.1:1313/" },
     76       { name: "IMDb Top 250", url: "https://www.imdb.com/chart/top" },
     77       { name: "Japanese Class", url: "https://japaneseclass.jp/" },
     78       { name: "JavaScript Minifier", url: "https://www.toptal.com/developers/javascript-minifier/" },
     79       { name: "Klar Tale", url: "https://www.klartale.no/" },
     80       { name: "Kodi", url: "http://192.168.1.6:8080/" },
     81       { name: "Ladbrokes Promo", url: "https://promo.ladbrokes.com/en/promo/stw" },
     82       { name: "Linode", url: "https://cloud.linode.com/linodes" },
     83       { name: "Logical Increments", url: "https://www.logicalincrements.com/" },
     84       { name: "Marine Traffic", url: "https://www.marinetraffic.com/" },
     85       { name: "Meetup", url: "https://www.meetup.com/your-events/" },
     86       { name: "Met Office Weather", url: "https://www.metoffice.gov.uk/public/weather/forecast/gcw9rrxn7" },
     87       { name: "Minskio", url: "https://minskio.co.uk/" },
     88       { name: "Mistral", url: "https://chat.mistral.ai/chat" },
     89       { name: "Myrient", url: "https://myrient.erista.me/files/" },
     90       { name: "Nault NANO Wallet", url: "https://nault.cc/" },
     91       { name: "Nexo", url: "https://platform.nexo.io/" },
     92       { name: "Noisli", url: "https://www.noisli.com/" },
     93       { name: "Oracle", url: "https://cloud.oracle.com/compute/instances" },
     94       { name: "Paperspace", url: "https://console.paperspace.com/" },
     95       { name: "PC Part Picker", url: "https://uk.pcpartpicker.com/list/" },
     96       { name: "Postcode Finder", url: "https://www.royalmail.com/find-a-postcode" },
     97       { name: "Power Cut Map", url: "https://www.northernpowergrid.com/power-cuts-map" },
     98       { name: "Prolific", url: "https://app.prolific.co/studies" },
     99       { name: "Qutebrowser Cheat Sheet", url: "https://raw.githubusercontent.com/qutebrowser/qutebrowser/master/doc/img/cheatsheet-big.png" },
    100       { name: "Radiooooo", url: "https://radiooooo.com/" },
    101       { name: "Real Time Lightning Map", url: "https://www.lightningmaps.org/" },
    102       { name: "Redactle", url: "https://redactle.net/" },
    103       { name: "Reddit", url: "https://old.reddit.com/" },
    104       { name: "Rightmove", url: "https://www.rightmove.co.uk/" },
    105       { name: "Showcase", url: "https://www.showcasecinemas.co.uk/showtimes/showcase-cinema-de-lux-leeds" },
    106       { name: "Sliding Puzzle Solver", url: "https://jweilhammer.github.io/sliding-puzzle-solver/" },
    107       { name: "Spotify", url: "https://open.spotify.com/" },
    108       { name: "SQLite Viewer", url: "https://sqliteviewer.app/" },
    109       { name: "Stable Tracker", url: "https://newtrackon.com/list" },
    110       { name: "Stagit", url: "https://git.minskio.co.uk/" },
    111       { name: "Suckless", url: "https://suckless.org/" },
    112       { name: "Syncthing", url: "http://127.0.0.1:8384/" },
    113       { name: "Teleport Map", url: "https://oldschool.runescape.wiki/images/Teleport_map.jpg" },
    114       { name: "Time Zone Map", url: "https://upload.wikimedia.org/wikipedia/commons/e/ec/World_Time_Zones_Map.svg" },
    115       { name: "Tradle", url: "https://tradle.net/" },
    116       { name: "Transmission", url: "https://tor.minskio.co.uk" },
    117       { name: "Traveline Route Planner", url: "https://www.traveline.info/" },
    118       { name: "TTS NaturalReader", url: "https://www.naturalreaders.com/online/" },
    119       { name: "Tynan Portfolio", url: "https://dashboard.m1finance.com/share/H4sIAAAAAAACA53PzwuCMBQH8H9F3jUP05JK6BDYUSXQIiJirMea2RJdiMn-99ah0y6r43t8P-_HCGpoEGKQ2J8bgeCDpPdPoxgkld66rr09UnXF1gtJSCYm0NWCYQfxcQT1gHj8juiQPVuhBhMRF1OXZJfkFY-yqlyB9qHBlqFUlJtwMNO-K39tLb501Lc-LZilp-SH5anFF466DLNkY38euN9eHWxOfuD8_-08ygubz_VJvwFVDPxOMwIAAA?referrerCode=5QM0YPa-0rCz" },
    120       { name: "unRPA", url: "https://im7mortal.github.io/unrpa/" },
    121       { name: "Unsplash", url: "https://unsplash.com/" },
    122       { name: "VaultWarden", url: "https://pass.minskio.co.uk/" },
    123       { name: "Ventu Sky", url: "https://www.ventusky.com/" },
    124       { name: "Vidyascape Wiki", url: "http://vscape.wikidot.com/" },
    125       { name: "Vidyascape", url: "https://vidyascape.org/" },
    126       { name: "Vikunja", url: "https://todo.minskio.co.uk/" },
    127       { name: "Vuniper", url: "https://vuniper.com/movies" },
    128       { name: "W3C validator", url: "https://validator.w3.org/" },
    129       { name: "Walk Mapper", url: "https://onthegomap.com/" },
    130       { name: "WikiQuote Random", url: "https://en.wikiquote.org/wiki/Special:Random" },
    131       { name: "Winget", url: "https://winstall.app/apps/" },
    132       { name: "Wordle", url: "https://www.nytimes.com/games/wordle/index.html" },
    133       { name: "Worldle", url: "https://worldle.teuteuf.fr/" },
    134       { name: "Zoho Mail", url: "https://mail.zoho.eu/zm/#mail" },
    135       { name: "Zoho Sheets", url: "https://sheet.zoho.eu/spreadsheets" },
    136       { name: "Zoopla", url: "https://www.zoopla.co.uk/" }
    137       ];
    138 
    139       const commandMap = {
    140       a: {base: "https://www.amazon.co.uk/",query: "https://www.amazon.co.uk/s?k={q}"},
    141       abb: {base: "http://audiobookbay.fi/",query: "http://audiobookbay.fi/?s={q}"},
    142       alp: {base: "https://pkgs.alpinelinux.org/packages?name=&branch=edge&arch=x86_64",query: "https://pkgs.alpinelinux.org/packages?branch=edge&arch=x86_64&name={q}"},
    143       anna: {base: "https://annas-archive.org/",query: "https://annas-archive.org/search?q={q}"},
    144       ans: {base: "https://galaxy.ansible.com/", query: "https://galaxy.ansible.com/search?deprecated=false&keywords={q}"},
    145       ap: {base: "https://www.archlinux.org/packages/", query: "https://www.archlinux.org/packages/?sort=&q={q}"},
    146       archive: {base: "https://web.archive.org/", query: "https://web.archive.org/web/*/{q}"},
    147       aur: {base: "https://aur.archlinux.org/packages/", query: "https://aur.archlinux.org/packages/?O=0&K={q}"},
    148       aw: {base: "https://wiki.archlinux.org/", query: "https://wiki.archlinux.org/index.php?title=Special:Search&search={q}"},
    149       cb: {base: "https://comicvine.gamespot.com/", query: "https://comicvine.gamespot.com/search/?q={q}"},
    150       cg: {base: "https://www.coingecko.com/", query: "https://www.coingecko.com/en/coins/{q}"},
    151       choco: {base: "https://chocolatey.org/packages", query: "https://chocolatey.org/packages?q={q}"},
    152       cron: {base: "https://crontab.guru/", query: "https://crontab.guru/#{q}"},
    153       d: {base: "https://start.duckduckgo.com/", query: "https://start.duckduckgo.com/?q={q}"},
    154       deb: {base: "https://packages.debian.org/", query: "https://packages.debian.org/{q}"},
    155       dock: {base: "https://hub.docker.com/", query: "https://hub.docker.com/search/?pullCount=1&q={q}"},
    156       down: {base: "https://downforeveryoneorjustme.com/", query: "https://downforeveryoneorjustme.com/{q}"},
    157       e: {base: "https://www.ebay.co.uk/", query: "https://www.ebay.co.uk/sch/i.html?_nkw={q}"},
    158       emu: {base: "https://emulation.gametechwiki.com/index.php", query: "https://emulation.gametechwiki.com/index.php?title=Special:Search&search={q}"},
    159       f: {base: "https://f-droid.org/", query: "https://search.f-droid.org/?lang=en&q={q}"},
    160       fast: {base: "https://fast.com/"},
    161       g: {base: "https://www.google.co.uk", query: "https://www.google.co.uk/search?udm=14&q={q}"},
    162       gamma: {base: "https://www.gamma-portal.com/framework/home.jspa", query: "https://www.gamma-portal.com/voip/ipdc/searchInput.jspa#{q}"},
    163       ge: {base: "https://www.ge-tracker.com/",query: "https://www.ge-tracker.com/names/{q}"},
    164       gf: {base: "https://gamefaqs.gamespot.com/", query: "https://gamefaqs.gamespot.com/search?game={q}"},
    165       gist: {base: "https://gist.github.com/", query: "https://gist.github.com/search?utf8=%E2%9C%93&q={q}"},
    166       git: {base: "https://github.com/", query: "https://github.com/search?q={q}"},
    167       gl: {base: "https://www.google.co.uk", query: "https://www.google.co.uk/search?btnI=1&q={q}"},
    168       gm: {base: "https://www.google.co.uk/maps/", query: "https://www.google.co.uk/maps/search/{q}"},
    169       gmt: {base: "https://www.google.co.uk/maps/",query: "https://www.google.co.uk/maps/dir/Brighouse/{q}/Brighouse/"},
    170       gog: {base: "https://www.gog.com/", query: "https://www.gog.com/games?query={q}"},
    171       gogg: {base: "https://gog-games.to/", query: "https://gog-games.to/?search={q}"},
    172       gr: {base: "https://www.goodreads.com/", query: "https://www.goodreads.com/search?utf8=%E2%9C%93&query={q}"},
    173       gum: {base: "https://www.gumtree.com/", query: "https://www.gumtree.com/search?q={q}&search_location=Brighouse&distance=30"},
    174       hm: {base: "https://home-manager-options.extranix.com", query: "https://home-manager-options.extranix.com/?release=release-25.05&query={q}"},
    175       i: {base: "https://images.google.co.uk/", query: "https://www.google.co.uk/search?tbm=isch&q={q}"},
    176       im: {base: "https://imdb.com/", query: "https://www.imdb.com/find?ref_=nv_sr_fn&s=all&q={q}"},
    177       ip: {base: "https://ip.minskio.co.uk/", query: "https://www.ip2location.com/demo/{q}"},
    178       ipl: {base: "https://www.bbc.co.uk/iplayer/", query: "https://www.bbc.co.uk/iplayer/search?q={q}"},
    179       jack: {base: "https://jack.minskio.co.uk", query: "https://jack.minskio.co.uk/UI/Dashboard#search={q}"},
    180       last: {base: "https://www.last.fm/", query: "https://www.last.fm/search?q={q}"},
    181       lut: {base: "https://lutris.net/", query: "https://lutris.net/games?q={q}"},
    182       m: {base: "https://wego.here.com/", query: "https://wego.here.com/search/{q}"},
    183       mac: {base: "https://macvendors.com/", query: "https://api.macvendors.com/{q}"},
    184       man: {base: "https://man.archlinux.org/", query: "https://man.archlinux.org/search?go=Go&q={q}"},
    185       mb: {base: "https://musicbrainz.org/", query: "https://musicbrainz.org/search?type=artist&limit=100&method=indexed&query={q}"},
    186       mx: {base: "https://mxtoolbox.com/SuperTool.aspx", query: "https://mxtoolbox.com/SuperTool.aspx?action={q}"},
    187       nitter: {base: "https://nitter.net/", query: "https://nitter.net/search?f=users&q={q}"},
    188       nix: {base: "https://nixos.org/", query: "https://search.nixos.org/packages?query={q}"},
    189       no: {base: "https://enno.dict.cc/", query: "https://enno.dict.cc/?s={q}"},
    190       nyaa: {base: "https://nyaa.si/", query: "https://nyaa.si/?s=seeders&o=desc&q={q}"},
    191       osm: {base: "https://www.openstreetmap.org/", query: "https://www.openstreetmap.org/search?query={q}"},
    192       osrs: {base: "https://oldschool.runescape.wiki/", query: "https://oldschool.runescape.wiki/?title=Special%3ASearch&search={q}"},
    193       pass: {base: "https://pass.minskio.co.uk/"},
    194       pay: {base: "https://12ft.io/", query: "https://12ft.io/{q}"},
    195       pc: {base: "https://pcgamingwiki.com/w/index.php", query: "https://pcgamingwiki.com/w/index.php?title=Special:Search&search={q}"},
    196       play: {base: "https://play.google.com/store/", query: "https://play.google.com/store/search?c=apps&q={q}"},
    197       proton: {base: "https://protondb.com/explore", query: "https://protondb.com/search?q={q}"},
    198       rfc: {base: "https://datatracker.ietf.org/", query: "https://datatracker.ietf.org/doc/search?rfcs=on&name={q}"},
    199       rs3: {base: "https://runescape.wiki/", query: "https://runescape.wiki/?title=Special%3ASearch&go=Go&search={q}"},
    200       rshs: {base: "https://secure.runescape.com/m=hiscore_oldschool/", query: "https://secure.runescape.com/m=hiscore_oldschool/hiscorepersonal?user1={q}"},
    201       rss: {base: "https://rss.minskio.co.uk/"},
    202       rt: {base: "https://www.rottentomatoes.com/", query: "https://www.rottentomatoes.com/search/?search={q}"},
    203       s: {base: "https://store.steampowered.com/", query: "https://store.steampowered.com/search/?term={q}"},
    204       sc: {base: "https://github.com/koalaman/shellcheck/", query: "https://github.com/koalaman/shellcheck/wiki/{q}"},
    205       skey: {base: "https://store.steampowered.com/", query: "https://store.steampowered.com/account/registerkey?key={q}"},
    206       spot: {base: "https://open.spotify.com/", query: "https://open.spotify.com/search/{q}"},
    207       srx: {base: "https://searx.be/", query: "https://searx.be/?q={q}"},
    208       ss: {base: "https://steamspy.com/", query: "https://steamspy.com/search.php?s={q}"},
    209       st: {base: "https://www.startpage.com/", query: "https://www.startpage.com/do/metasearch.pl?query={q}"},
    210       svg: {base: "https://freesvgicons.com/", query: "https://freesvgicons.com/search?q={q}&collectionId=fluent"},
    211       t: {base: "https://tineye.com/", query: "https://tineye.com/parse?url={q}"},
    212       timer: {base: "https://www.google.co.uk/search?q=12.5%20minute%20timer", query: "https://www.google.co.uk/search?q={q}%20timer"},
    213       tm: {base: "https://www.themoviedb.org/", query: "https://www.themoviedb.org/search?query={q}"},
    214       tmc: {base: "https://login.themilesconsultancy.com/Driver/TripSummary.aspx"},
    215       tmt: {base: "https://www.themoviedb.org/", query: "https://www.themoviedb.org/search/tv?query={q}"},
    216       todo: {base: "https://todo.minskio.co.uk/"},
    217       tr: {base: "https://translate.google.com/", query: "https://translate.google.com/?sl=auto&tl=en&text={q}"},
    218       track: {base: "https://www.royalmail.com/track-your-item", query: "https://www.royalmail.com/track-your-item#/tracking-results/{q}"},
    219       tv: {base: "https://www.thetvdb.com/", query: "https://www.thetvdb.com/search?query={q}"},
    220       tvt: {base: "https://tvtropes.org/", query: "https://tvtropes.org/pmwiki/search_result.php?q={q}"},
    221       ud: {base: "https://www.urbandictionary.com/", query: "https://www.urbandictionary.com/define.php?term={q}"},
    222       ujs: {base: "https://greasyfork.org/en/scripts", query: "https://greasyfork.org/en/scripts?q={q}"},
    223       valid: {base: "https://validator.w3.org/nu/", query: "https://validator.w3.org/nu/?doc={q}"},
    224       w3c: {base: "https://validator.w3.org/", query: "https://validator.w3.org/nu/?doc={q}"},
    225       w3w: {base: "https://what3words.com/",query: "https://what3words.com/{q}"},
    226       w: {base: "https://en.wikipedia.org/w/index.php", query: "https://en.wikipedia.org/w/index.php?title=Special:Search&search={q}"},
    227       wa: {base: "https://www.wolframalpha.com/", query: "https://www.wolframalpha.com/input/?i={q}"},
    228       whois: {base: "https://whois.gandi.net/", query: "https://whois.gandi.net/en/results?search={q}"},
    229       wikt: {base: "https://en.wiktionary.org/w/index.php", query: "https://en.wiktionary.org/w/index.php?title=Special:Search&search={q}"},
    230       wiktno: {base: "https://no.wiktionary.org/w/index.php", query: "https://no.wiktionary.org/w/index.php?title=Special:Search&search={q}"},
    231       wine: {base: "https://www.winehq.org/",query: "https://www.winehq.org/search?q={q}"},
    232       wno: {base: "https://no.wikipedia.org/w/index.php",query: "https://no.wikipedia.org/w/index.php?title=Special:Search&search={q}"},
    233       y: {base: "https://www.youtube.com/",query: "https://www.youtube.com/results?search_query={q}"}
    234       };
    235 
    236       const input = document.getElementById("queryInput");
    237       const bookmarkList = document.getElementById("bookmarkList");
    238       const predictionDisplay = document.getElementById("prediction");
    239 
    240       let filteredBookmarks = [...bookmarks];
    241       let selectedBookmarkIndex = -1;
    242 
    243       function renderBookmarks() {
    244       bookmarkList.innerHTML = "";
    245       filteredBookmarks.forEach((bm, index) => {
    246       const li = document.createElement("li");
    247       li.textContent = bm.name;
    248       if (index === selectedBookmarkIndex) {
    249         li.classList.add("highlight");
    250         // Scroll the selected item into view
    251         setTimeout(() => li.scrollIntoView({ block: "nearest", behavior: "smooth" }), 0);
    252       }
    253       li.addEventListener("click", () => {
    254         window.open(bm.url, "_self");
    255       });
    256       bookmarkList.appendChild(li);
    257       });
    258 
    259       bookmarkList.style.display = filteredBookmarks.length === 0 ? "none" : "block";
    260       }
    261 
    262       function filterBookmarks(query) {
    263         selectedBookmarkIndex = -1;
    264         const lower = query.toLowerCase();
    265         filteredBookmarks = lower === ""
    266           ? [...bookmarks]
    267           : bookmarks.filter(b => b.name.toLowerCase().includes(lower));
    268         renderBookmarks();
    269       }
    270 
    271       function runCommand(prefix, args) {
    272         const cmd = commandMap[prefix];
    273         if (!cmd) return;
    274         const url = args.length === 0
    275           ? cmd.base
    276           : cmd.query.replace("{q}", encodeURIComponent(args.join(" ")));
    277         window.open(url, "_self");
    278       }
    279 
    280       function hintCommand(prefix, args) {
    281         const cmd = commandMap[prefix];
    282         if (!cmd) return "";
    283         return args.length === 0
    284           ? cmd.base
    285           : cmd.query.replace("{q}", encodeURIComponent(args.join(" ")));
    286       }
    287 
    288       function handleCommand(inputValue) {
    289         const parts = inputValue.trim().split(" ");
    290         const prefix = parts[0].toLowerCase();
    291         const args = parts.slice(1);
    292 
    293         if (commandMap[prefix]) {
    294           runCommand(prefix, args);
    295         } else {
    296           const query = encodeURIComponent(inputValue);
    297           window.open(`https://www.google.com/search?q=${query}`, "_self");
    298         }
    299       }
    300 
    301       function updatePrediction(inputValue) {
    302         const [prefix, ...args] = inputValue.trim().split(" ");
    303         predictionDisplay.textContent = commandMap[prefix?.toLowerCase()]
    304           ? hintCommand(prefix.toLowerCase(), args)
    305           : "";
    306         }
    307 
    308       input.addEventListener("input", (e) => {
    309         filterBookmarks(e.target.value);
    310         updatePrediction(e.target.value);
    311       });
    312 
    313       input.addEventListener("keydown", (e) => {
    314       if (e.key === "Tab") {
    315       e.preventDefault();
    316       if (filteredBookmarks.length > 0) {
    317       if (e.shiftKey) {
    318         selectedBookmarkIndex = (selectedBookmarkIndex - 1 + filteredBookmarks.length) % filteredBookmarks.length;
    319       } else {
    320         selectedBookmarkIndex = (selectedBookmarkIndex + 1) % filteredBookmarks.length;
    321       }
    322       renderBookmarks();
    323       }
    324       }
    325       else if (e.key === "ArrowDown") {
    326           e.preventDefault();
    327           if (filteredBookmarks.length > 0) {
    328             selectedBookmarkIndex = (selectedBookmarkIndex + 1) % filteredBookmarks.length;
    329             renderBookmarks();
    330           }
    331         } else if (e.key === "ArrowUp") {
    332           e.preventDefault();
    333           if (filteredBookmarks.length > 0) {
    334             selectedBookmarkIndex = (selectedBookmarkIndex - 1 + filteredBookmarks.length) % filteredBookmarks.length;
    335             renderBookmarks();
    336           }
    337         } else if (e.key === "Enter") {
    338           if (selectedBookmarkIndex >= 0) {
    339             window.open(filteredBookmarks[selectedBookmarkIndex].url, "_self");
    340           } else {
    341             handleCommand(input.value);
    342           }
    343         } else if (e.key === "Escape") {
    344           input.value = "";
    345           predictionDisplay.textContent = "";
    346           filterBookmarks("");
    347         }
    348       });
    349 
    350       // Initial load
    351       filterBookmarks("");
    352       function formatDateTime(date) {
    353       const days = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
    354       const months = ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"];
    355 
    356       const dayName = days[date.getDay()];
    357       const day = date.getDate();
    358       const month = months[date.getMonth()];
    359       const hours = String(date.getHours()).padStart(2, '0');
    360       const minutes = String(date.getMinutes()).padStart(2, '0');
    361 
    362       const getOrdinal = (n) => {
    363       if (n > 3 && n < 21) return "th";
    364       switch (n % 10) {
    365       case 1: return "st";
    366       case 2: return "nd";
    367       case 3: return "rd";
    368       default: return "th";
    369       }
    370       };
    371 
    372       return `${dayName} ${day}${getOrdinal(day)} ${month}, ${hours}:${minutes}`;
    373       }
    374 
    375       function updateDateTime() {
    376       const now = new Date();
    377       document.getElementById("datetime-text").textContent = formatDateTime(now);
    378       }
    379 
    380       setInterval(updateDateTime, 60000);
    381       updateDateTime();
    382 
    383     </script>
    384   </body>
    385 </html>