pub / yt2html

Fetches Youtube content via RSS and provides a chronological timeline
git clone https://src.jayvii.de/pub/yt2html.git
Home | Log | Files | Exports | Refs | README | RSS

index.php (10149B)


      1 <!-- SPDX-License-Identifier: AGPL-3.0-or-later
      2      SPDX-FileCopyrightText: 2021-2024 JayVii <jayvii[AT]posteo[DOT]de>
      3 -->
      4 
      5 <!DOCTYPE html>
      6 <html lang="en">
      7   <head>
      8     <meta charset="utf-8">
      9     <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
     10     <title>YouTube Videos</title>
     11     <link rel="icon" type="image/png" href="/assets/favicon.png">
     12     <link rel="icon" type="image/png" sizes="16x16" href="/assets/favicon_16.png">
     13     <link rel="icon" type="image/png" sizes="32x32" href="/assets/favicon_32.png">
     14     <link rel="icon" type="image/png" sizes="64x64" href="/assets/favicon_64.png">
     15     <link rel="icon" type="image/png" sizes="128x128" href="/assets/favicon_128.png">
     16     <link rel="apple-touch-icon" href="/assets/favicon.png">
     17     <link rel="stylesheet" type="text/css" href="/assets/css/simple.min.css"/>
     18     <link rel="stylesheet" type="text/css" href="/assets/css/custom.css"/>
     19     <script async src="assets/js/yt.js"></script>
     20     <link crossorigin="use-credentials" rel="manifest" href="/manifest.json">
     21     <meta name="viewport" content="width=device-width, initial-scale=1.0">
     22   </head>
     23 
     24 <?php
     25 
     26   /* Comparing function for sorting */
     27   function compare_numeric(array $a, array $b, string $col) {
     28     return $b[$col] - $a[$col];
     29   }
     30   function compare_string(array $a, array $b, string $col) {
     31     return strcmp(strtolower($a[$col]), strtolower($b[$col]));
     32   }
     33 
     34 
     35   /* Initilise videos template */
     36   $video_template = array(
     37     "vid" => array(),
     38     "title" => array(),
     39     "desc" => array(),
     40     "time" => array(),
     41     "aid" => array(),
     42     "author" => array()
     43   );
     44 
     45   /* Initilise channels arrays */
     46   $channels = array();
     47   $channels_list = array();
     48 
     49   /* Initilise videos array */
     50   $videos = array();
     51 
     52   // Fetch POST, GET arguments or the COOKIE
     53   // Cookie should have lowest priority (fallback), POST first (intend)
     54   // GET arguments are used for one-time usage (will not set cookie)
     55   if (!is_null($_COOKIE["channels"])) {
     56     $channels = explode(",", $_COOKIE["channels"]);
     57   }
     58   if (!is_null($_POST["channels"])) {
     59     $channels = urldecode($_POST["channels"]);
     60     $channels = preg_replace('/[^A-Za-z0-9\-\_\,]+/', '', $channels);
     61     $channels = explode(",", $channels);
     62   }
     63   if (!is_null($_GET["channels"])) {
     64     $channels = explode(",", $_GET["channels"]);
     65   }
     66 
     67   // Sort channels by alphabet and ensure each channel is unique
     68   $channels = array_unique($channels);
     69 
     70   /* refresh cookie */
     71   if (is_null($_GET["channels"])) {
     72     header(
     73       "Set-Cookie: " .
     74       "channels=" . implode(",", $channels) . ";" .
     75       "Max-Age=" . 31536000 . "; " . /* 60 x 60 x 24 x 365 = 1 year */
     76       "Domain=" . $_SERVER["SERVER_NAME"] . "; " .
     77       "SameSite=Strict;"
     78     );
     79   }
     80 
     81   for ($i = 0; $i < count($channels); $i++) {
     82     $channel = $channels[$i];
     83 
     84     // Fetch Youtube XML Feed
     85     $channel_xml = file(
     86     "https://www.youtube.com/feeds/videos.xml?channel_id=" . $channel
     87     );
     88 
     89     /* Skip to next entry, if channel could not be found */
     90     if ($channel_xml === false) {
     91       /* Remove entry from channels list */
     92       array_splice($channels, $i, 1);
     93       /* Skip item within loop */
     94       continue;
     95     }
     96 
     97     // Replace un-parsable items
     98     $channel_xml = str_replace(
     99       array("yt:", "media:"),
    100       array("yt_", "media_"),
    101       $channel_xml
    102     );
    103 
    104     // Cast Array to string
    105     $channel_xml = implode(PHP_EOL, $channel_xml);
    106 
    107     // Parse XML
    108     $channel_xml = simplexml_load_string($channel_xml);
    109     $channel_xml = json_encode($channel_xml);
    110     $channel_xml = json_decode($channel_xml, true);
    111 
    112     // Get Channel name
    113     $author = str_replace(
    114       array("&"),
    115       "&amp;",
    116       $channel_xml["entry"][0]["author"]["name"]
    117     );
    118 
    119     /* Fill channels list array */
    120     array_push($channels_list, array("aid" => $channel, "author" => $author));
    121 
    122     // Process Entries
    123     foreach ($channel_xml["entry"] as $entry) {
    124 
    125       /* Copy video template array */
    126       $video = $video_template;
    127 
    128       // Get Video ID
    129       $video["vid"] = str_replace(array("yt_video:"), "", $entry["id"]);
    130 
    131       // Get Video Title
    132       $video["title"] = str_replace(array("&"), "&amp;", $entry["title"]);
    133 
    134       // Get Video Description
    135       $video["desc"] = preg_replace(
    136         "/\n+/",
    137         "<br>",
    138         $entry["media_group"]["media_description"]
    139       );
    140       if (empty($video["desc"])) {
    141         $video["desc"] = "";
    142       }
    143 
    144       // Get Time
    145       $video["time"] = strtotime($entry["published"]);
    146 
    147       // Get Channel ID
    148       $video["aid"] = $channel;
    149 
    150       // Get Channel Name
    151       $video["author"] = $author;
    152 
    153       /* check whether indications of a YouTube-Short exist, skip if it is
    154        * We assume video is a short if any of the following applies:
    155        * has #short/#shorts in the title
    156        * has #short #shorts in the description
    157        * description only contains #hashtags
    158        * has no description at all
    159        */
    160       $vdesc = explode(" ", $video["desc"]);
    161       if (
    162         preg_match('/#short(s)*/', $video["title"]) > 0 ||
    163         preg_match('/#short(s)*/', $video["desc"]) > 0 ||
    164         preg_grep('/^(#|$)/', $vdesc) == $vdesc ||
    165         strlen($video["desc"]) == 0
    166       ) {
    167         continue;
    168       }
    169 
    170       /* Add video to videos array */
    171       array_push($videos, $video);
    172 
    173     }
    174 
    175   }
    176 
    177   /* Sort videos and channels arrays */
    178   function cmp_time(array $a, array $b) {
    179     return compare_numeric($a, $b, "time");
    180   }
    181   function cmp_name(array $a, array $b) {
    182     return compare_string($a, $b, "author");
    183   }
    184   usort($videos, "cmp_time");
    185   usort($channels_list, "cmp_name");
    186 
    187 ?>
    188 
    189   <body>
    190 
    191     <header>
    192       <nav>
    193         <a href="/">All Videos</a>
    194         <?php
    195           /* For GET-requests, draw simple reload button,
    196            * otherwise re-submit channels form
    197            */
    198           if (is_null($_GET["channels"])) {
    199             $jsfun = "document.getElementById('channel_form').submit();";
    200           } else {
    201             $jsfun = "window.location.reload();";
    202           }
    203         ?>
    204         <a href="#" onclick="<?php echo $jsfun; ?>">
    205           Refresh
    206         </a>
    207         <a href="https://src.jayvii.de/pub/yt2html" target="_blank">
    208           Development
    209         </a>
    210       </nav>
    211       <h1>Videos</h1>
    212     </header>
    213 
    214     <form action="https://www.youtube.com/results" method="GET">
    215       <input
    216         id="searchbar"
    217         type="text"
    218         id="searchInput"
    219         name="search_query"
    220         placeholder="Search on YouTube..."
    221       >
    222     </form>
    223 
    224     <!-- Channels List Form -->
    225     <details id="channels">
    226       <summary>List of Channels</summary>
    227       <form action="/" method="POST" id="channel_form">
    228         <details>
    229           <summary>Import / Export</summary>
    230           <p>
    231             Please enter the YouTube channel-IDs you want to check here, each
    232             separated with a <code>,</code>:
    233           </p>
    234           <input
    235             name="channels"
    236             type="text"
    237             value="<?php echo implode("," . PHP_EOL, $channels); ?>"
    238           >
    239         <input type="submit" value="Save &amp; Reload">
    240         </details>
    241         <div id="channels_list">
    242           <?php
    243             /* Draw Input fields for each channel ID */
    244             for ($i = 0; $i < count($channels_list); $i++) {
    245           ?>
    246           <label for="channel_<?php echo $i; ?>">
    247             <?php
    248               if ($channels_list[$i]["author"] != "") {
    249                 echo "<a href=\"/?channels=" . $channels_list[$i]["aid"] .
    250                   "\" target=\"_blank\">" . $channels_list[$i]["author"] .
    251                   "</a>";
    252               } else {
    253                 echo "<mark class=\"error\">Error</mark>";
    254               }
    255             ?>
    256           </label>
    257           <input
    258             name="channel_<?php echo $i; ?>"
    259             class="channels_input"
    260             type="text"
    261             value="<?php echo $channels_list[$i]["aid"]; ?>"
    262             oninput="yt2html_update_channels_list();"
    263           >
    264           <?php
    265             }
    266           ?>
    267           <!-- Draw one empty input field for new entry by the user -->
    268           <label for="channel_<?php echo ($i); ?>">
    269             <mark>Add a new channel here</mark>
    270           </label>
    271           <input
    272             name="channel<?php echo ($i); ?>"
    273             class="channels_input"
    274             type="text"
    275             oninput="yt2html_update_channels_list();"
    276           >
    277         </div>
    278         <!-- Submit Button -->
    279         <input type="submit" value="Save &amp; Reload">
    280       </form>
    281     </details>
    282 
    283     <!-- Video List -->
    284     <?php
    285       $index = 0;
    286       foreach ($videos as $video) {
    287 
    288         /* Stop if 500 videos are listed */
    289         if ($index >= 500) {
    290           break;
    291         }
    292     ?>
    293     <section id="video_<?php echo $index; ?>">
    294 
    295       <!-- Headline: streamer name -->
    296       <h2><?php echo $video["title"]; ?></h2>
    297 
    298       <!-- player Container -->
    299       <div class="player_container hidden"></div>
    300 
    301       <!-- Preview Image -->
    302       <img
    303         id="preview_<?php echo $index; ?>"
    304         class="preview"
    305         src="https://i4.ytimg.com/vi/<?php echo $video["vid"]; ?>/hqdefault.jpg"
    306         loading="lazy"
    307       >
    308 
    309       <!-- Video Information -->
    310       <div style="width:100%;margin-bottom:0.5em;">
    311         <span style="margin-right:0.25em;"><?php echo $video["author"]; ?></span>
    312         <span style="margin-left:0.25em;"><?php echo date("Y-m-d, H:i:s", $video["time"]); ?></span>
    313       </div>
    314 
    315       <!-- Video Buttons -->
    316       <?php $js_args = $index . ",'" . $video["vid"] . "'"; ?> 
    317       <button onclick="yt2html_toggle_player(<?php echo $js_args; ?>)">
    318         Toggle Player
    319       </button>
    320       <a
    321         class="button"
    322         href="/?channels=<?php echo $video["aid"]; ?>"
    323       >
    324         Open Channel
    325       </a>
    326       <a
    327         class="button"
    328         href="https://www.youtube.com/watch?v=<?php echo $video["vid"]; ?>"
    329         target="_blank"
    330       >
    331         Open on YouTube
    332       </a>
    333 
    334     <!-- Chat Collapsable -->
    335     <details>
    336         <summary>
    337             Description
    338         </summary>
    339         <?php echo $video["desc"]; ?>
    340     </details>
    341 
    342   </section>
    343 
    344   <?php
    345 
    346     $index++;
    347 
    348     }
    349 
    350   ?>
    351 
    352   </body>
    353 </html>