mirror of
https://github.com/iv-org/invidious.git
synced 2025-10-23 17:28:27 -05:00
Merge 126f8fd09751d7c01d02960c60ca2bcd29ae34a1 into b173d4acf21563d47d26718eca7932878fb424e6
This commit is contained in:
commit
a659a779ae
@ -498,5 +498,8 @@
|
|||||||
"toggle_theme": "Toggle Theme",
|
"toggle_theme": "Toggle Theme",
|
||||||
"carousel_slide": "Slide {{current}} of {{total}}",
|
"carousel_slide": "Slide {{current}} of {{total}}",
|
||||||
"carousel_skip": "Skip the Carousel",
|
"carousel_skip": "Skip the Carousel",
|
||||||
"carousel_go_to": "Go to slide `x`"
|
"carousel_go_to": "Go to slide `x`",
|
||||||
|
"error_from_youtube_unplayable": "Video unplayable due to an error from YouTube:",
|
||||||
|
"error_processing_data_youtube": "Error while processing the data sent by YouTube",
|
||||||
|
"refresh_page": "Refresh the page"
|
||||||
}
|
}
|
||||||
|
@ -73,10 +73,6 @@ def error_template_helper(env : HTTP::Server::Context, status_code : Int32, exce
|
|||||||
</div>
|
</div>
|
||||||
END_HTML
|
END_HTML
|
||||||
|
|
||||||
# Don't show the usual "next steps" widget. The same options are
|
|
||||||
# proposed above the error message, just worded differently.
|
|
||||||
next_steps = ""
|
|
||||||
|
|
||||||
return templated "error"
|
return templated "error"
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -86,8 +82,13 @@ def error_template_helper(env : HTTP::Server::Context, status_code : Int32, mess
|
|||||||
|
|
||||||
locale = env.get("preferences").as(Preferences).locale
|
locale = env.get("preferences").as(Preferences).locale
|
||||||
|
|
||||||
error_message = translate(locale, message)
|
error_message = <<-END_HTML
|
||||||
next_steps = error_redirect_helper(env)
|
<div class="error_message">
|
||||||
|
<h2>#{translate(locale, "error_processing_data_youtube")}</h2>
|
||||||
|
<p>#{translate(locale, message)}</p>
|
||||||
|
#{error_redirect_helper(env)}
|
||||||
|
</div>
|
||||||
|
END_HTML
|
||||||
|
|
||||||
return templated "error"
|
return templated "error"
|
||||||
end
|
end
|
||||||
|
@ -117,6 +117,7 @@ module Invidious::Routes::Watch
|
|||||||
comment_html ||= ""
|
comment_html ||= ""
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if video.reason.nil?
|
||||||
fmt_stream = video.fmt_stream
|
fmt_stream = video.fmt_stream
|
||||||
adaptive_fmts = video.adaptive_fmts
|
adaptive_fmts = video.adaptive_fmts
|
||||||
|
|
||||||
@ -189,6 +190,9 @@ module Invidious::Routes::Watch
|
|||||||
audio_streams: audio_streams,
|
audio_streams: audio_streams,
|
||||||
captions: video.captions
|
captions: video.captions
|
||||||
)
|
)
|
||||||
|
else
|
||||||
|
env.response.status_code = 500
|
||||||
|
end
|
||||||
|
|
||||||
templated "watch"
|
templated "watch"
|
||||||
end
|
end
|
||||||
|
@ -313,7 +313,7 @@ def get_video(id, refresh = true, region = nil, force_refresh = false)
|
|||||||
end
|
end
|
||||||
else
|
else
|
||||||
video = fetch_video(id, region)
|
video = fetch_video(id, region)
|
||||||
Invidious::Database::Videos.insert(video) if !region
|
Invidious::Database::Videos.insert(video) if !region && !video.info.dig?("reason")
|
||||||
end
|
end
|
||||||
|
|
||||||
return video
|
return video
|
||||||
@ -326,13 +326,18 @@ end
|
|||||||
def fetch_video(id, region)
|
def fetch_video(id, region)
|
||||||
info = extract_video_info(video_id: id)
|
info = extract_video_info(video_id: id)
|
||||||
|
|
||||||
if reason = info["reason"]?
|
if info["reason"]? && info["subreason"]?
|
||||||
|
reason = info["reason"].as_s
|
||||||
|
puts info
|
||||||
|
if info.dig?("subreason").nil?
|
||||||
|
subreason = info["subreason"].as_s
|
||||||
|
else
|
||||||
|
subreason = "No additional reason"
|
||||||
|
end
|
||||||
if reason == "Video unavailable"
|
if reason == "Video unavailable"
|
||||||
raise NotFoundException.new(reason.as_s || "")
|
raise NotFoundException.new(reason + ": Video not found" || "")
|
||||||
elsif !reason.as_s.starts_with? "Premieres"
|
elsif {"Private video"}.any?(reason)
|
||||||
# dont error when it's a premiere.
|
raise InfoException.new(reason + ": " + subreason || "")
|
||||||
# we already parsed most of the data and display the premiere date
|
|
||||||
raise InfoException.new(reason.as_s || "")
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -64,18 +64,20 @@ def extract_video_info(video_id : String)
|
|||||||
playability_status = player_response.dig?("playabilityStatus", "status").try &.as_s
|
playability_status = player_response.dig?("playabilityStatus", "status").try &.as_s
|
||||||
|
|
||||||
if playability_status != "OK"
|
if playability_status != "OK"
|
||||||
subreason = player_response.dig?("playabilityStatus", "errorScreen", "playerErrorMessageRenderer", "subreason")
|
reason = player_response.dig?("playabilityStatus", "reason").try &.as_s
|
||||||
reason = subreason.try &.[]?("simpleText").try &.as_s
|
reason ||= player_response.dig("playabilityStatus", "errorScreen", "playerErrorMessageRenderer", "reason", "simpleText").as_s
|
||||||
reason ||= subreason.try &.[]("runs").as_a.map(&.[]("text")).join("")
|
subreason_main = player_response.dig?("playabilityStatus", "errorScreen", "playerErrorMessageRenderer", "subreason")
|
||||||
reason ||= player_response.dig("playabilityStatus", "reason").as_s
|
subreason = subreason_main.try &.[]?("simpleText").try &.as_s
|
||||||
|
subreason ||= subreason_main.try &.[]("runs").as_a.map(&.[]("text")).join("")
|
||||||
|
|
||||||
# Stop here if video is not a scheduled livestream or
|
# Stop if private video or video not found.
|
||||||
# for LOGIN_REQUIRED when videoDetails element is not found because retrying won't help
|
# But for video unavailable, only stop if playability_status is ERROR because playability_status UNPLAYABLE
|
||||||
if !{"LIVE_STREAM_OFFLINE", "LOGIN_REQUIRED"}.any?(playability_status) ||
|
# still gives all the necessary info for displaying the video page (title, description and more)
|
||||||
playability_status == "LOGIN_REQUIRED" && !player_response.dig?("videoDetails")
|
if {"Private video", "Video unavailable"}.any?(reason) && !{"UNPLAYABLE"}.any?(playability_status)
|
||||||
return {
|
return {
|
||||||
"version" => JSON::Any.new(Video::SCHEMA_VERSION.to_i64),
|
"version" => JSON::Any.new(Video::SCHEMA_VERSION.to_i64),
|
||||||
"reason" => JSON::Any.new(reason),
|
"reason" => JSON::Any.new(reason),
|
||||||
|
"subreason" => JSON::Any.new(subreason),
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
elsif video_id != player_response.dig("videoDetails", "videoId")
|
elsif video_id != player_response.dig("videoDetails", "videoId")
|
||||||
@ -95,11 +97,8 @@ def extract_video_info(video_id : String)
|
|||||||
reason = nil
|
reason = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
# Don't fetch the next endpoint if the video is unavailable.
|
|
||||||
if {"OK", "LIVE_STREAM_OFFLINE", "LOGIN_REQUIRED"}.any?(playability_status)
|
|
||||||
next_response = YoutubeAPI.next({"videoId": video_id, "params": ""})
|
next_response = YoutubeAPI.next({"videoId": video_id, "params": ""})
|
||||||
player_response = player_response.merge(next_response)
|
player_response = player_response.merge(next_response)
|
||||||
end
|
|
||||||
|
|
||||||
params = parse_video_info(video_id, player_response)
|
params = parse_video_info(video_id, player_response)
|
||||||
params["reason"] = JSON::Any.new(reason) if reason
|
params["reason"] = JSON::Any.new(reason) if reason
|
||||||
@ -205,17 +204,22 @@ def parse_video_info(video_id : String, player_response : Hash(String, JSON::Any
|
|||||||
raise BrokenTubeException.new("videoSecondaryInfoRenderer") if !video_secondary_renderer
|
raise BrokenTubeException.new("videoSecondaryInfoRenderer") if !video_secondary_renderer
|
||||||
end
|
end
|
||||||
|
|
||||||
video_details = player_response.dig?("videoDetails")
|
if !(video_details = player_response.dig?("videoDetails"))
|
||||||
|
video_details = {} of String => JSON::Any
|
||||||
|
end
|
||||||
if !(microformat = player_response.dig?("microformat", "playerMicroformatRenderer"))
|
if !(microformat = player_response.dig?("microformat", "playerMicroformatRenderer"))
|
||||||
microformat = {} of String => JSON::Any
|
microformat = {} of String => JSON::Any
|
||||||
end
|
end
|
||||||
|
|
||||||
raise BrokenTubeException.new("videoDetails") if !video_details
|
|
||||||
|
|
||||||
# Basic video infos
|
# Basic video infos
|
||||||
|
|
||||||
title = video_details["title"]?.try &.as_s
|
title = video_details["title"]?.try &.as_s
|
||||||
|
|
||||||
|
title ||= extract_text(
|
||||||
|
video_primary_renderer
|
||||||
|
.try &.dig?("title")
|
||||||
|
)
|
||||||
|
|
||||||
# We have to try to extract viewCount from videoPrimaryInfoRenderer first,
|
# We have to try to extract viewCount from videoPrimaryInfoRenderer first,
|
||||||
# then from videoDetails, as the latter is "0" for livestreams (we want
|
# then from videoDetails, as the latter is "0" for livestreams (we want
|
||||||
# to get the amount of viewers watching).
|
# to get the amount of viewers watching).
|
||||||
@ -226,17 +230,34 @@ def parse_video_info(video_id : String, player_response : Hash(String, JSON::Any
|
|||||||
views_txt ||= video_details["viewCount"]?.try &.as_s || ""
|
views_txt ||= video_details["viewCount"]?.try &.as_s || ""
|
||||||
views = views_txt.gsub(/\D/, "").to_i64?
|
views = views_txt.gsub(/\D/, "").to_i64?
|
||||||
|
|
||||||
length_txt = (microformat["lengthSeconds"]? || video_details["lengthSeconds"])
|
length_txt = (microformat["lengthSeconds"]? || video_details["lengthSeconds"]?)
|
||||||
.try &.as_s.to_i64
|
.try &.as_s.to_i64
|
||||||
|
|
||||||
published = microformat["publishDate"]?
|
published = microformat["publishDate"]?
|
||||||
.try { |t| Time.parse(t.as_s, "%Y-%m-%d", Time::Location::UTC) } || Time.utc
|
.try { |t| Time.parse(t.as_s, "%Y-%m-%d", Time::Location::UTC) }
|
||||||
|
|
||||||
|
if published.nil?
|
||||||
|
published_txt = video_primary_renderer
|
||||||
|
.try &.dig?("dateText", "simpleText")
|
||||||
|
|
||||||
|
if published_txt.try &.as_s.includes?("ago") && !published_txt.nil?
|
||||||
|
published = decode_date(published_txt.as_s.lchop("Started streaming "))
|
||||||
|
elsif published_txt && published_txt.try &.as_s.matches?(/(\w{3} \d{1,2}, \d{4})$/)
|
||||||
|
published = Time.parse(published_txt.as_s.match!(/(\w{3} \d{1,2}, \d{4})$/)[0], "%b %-d, %Y", Time::Location::UTC)
|
||||||
|
else
|
||||||
|
published = Time.utc
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
premiere_timestamp = microformat.dig?("liveBroadcastDetails", "startTimestamp")
|
premiere_timestamp = microformat.dig?("liveBroadcastDetails", "startTimestamp")
|
||||||
.try { |t| Time.parse_rfc3339(t.as_s) }
|
.try { |t| Time.parse_rfc3339(t.as_s) }
|
||||||
|
|
||||||
live_now = microformat.dig?("liveBroadcastDetails", "isLiveNow")
|
live_now = microformat.dig?("liveBroadcastDetails", "isLiveNow")
|
||||||
.try &.as_bool || false
|
.try &.as_bool
|
||||||
|
if live_now.nil?
|
||||||
|
live_now = video_primary_renderer
|
||||||
|
.try &.dig?("viewCount", "videoViewCountRenderer", "isLive").try &.as_bool || false
|
||||||
|
end
|
||||||
|
|
||||||
post_live_dvr = video_details.dig?("isPostLiveDvr")
|
post_live_dvr = video_details.dig?("isPostLiveDvr")
|
||||||
.try &.as_bool || false
|
.try &.as_bool || false
|
||||||
@ -247,8 +268,24 @@ def parse_video_info(video_id : String, player_response : Hash(String, JSON::Any
|
|||||||
.try &.as_a.map &.as_s || [] of String
|
.try &.as_a.map &.as_s || [] of String
|
||||||
|
|
||||||
allow_ratings = video_details["allowRatings"]?.try &.as_bool
|
allow_ratings = video_details["allowRatings"]?.try &.as_bool
|
||||||
|
|
||||||
family_friendly = microformat["isFamilySafe"]?.try &.as_bool
|
family_friendly = microformat["isFamilySafe"]?.try &.as_bool
|
||||||
|
if family_friendly.nil?
|
||||||
|
family_friendly = true # if isFamilySafe not found then assume is safe
|
||||||
|
end
|
||||||
|
|
||||||
is_listed = video_details["isCrawlable"]?.try &.as_bool
|
is_listed = video_details["isCrawlable"]?.try &.as_bool
|
||||||
|
if video_badges = video_primary_renderer.try &.dig?("badges")
|
||||||
|
if has_unlisted_badge?(video_badges)
|
||||||
|
is_listed ||= false
|
||||||
|
else
|
||||||
|
is_listed ||= true
|
||||||
|
end
|
||||||
|
# if no badges but videoDetails not available then assume isListed
|
||||||
|
else
|
||||||
|
is_listed ||= true
|
||||||
|
end
|
||||||
|
|
||||||
is_upcoming = video_details["isUpcoming"]?.try &.as_bool
|
is_upcoming = video_details["isUpcoming"]?.try &.as_bool
|
||||||
|
|
||||||
keywords = video_details["keywords"]?
|
keywords = video_details["keywords"]?
|
||||||
@ -414,6 +451,9 @@ def parse_video_info(video_id : String, player_response : Hash(String, JSON::Any
|
|||||||
subs_text = author_info["subscriberCountText"]?
|
subs_text = author_info["subscriberCountText"]?
|
||||||
.try { |t| t["simpleText"]? || t.dig?("runs", 0, "text") }
|
.try { |t| t["simpleText"]? || t.dig?("runs", 0, "text") }
|
||||||
.try &.as_s.split(" ", 2)[0]
|
.try &.as_s.split(" ", 2)[0]
|
||||||
|
|
||||||
|
author ||= author_info.dig?("title", "runs", 0, "text").try &.as_s
|
||||||
|
ucid ||= author_info.dig?("title", "runs", 0, "navigationEndpoint", "browseEndpoint", "browseId").try &.as_s
|
||||||
end
|
end
|
||||||
|
|
||||||
# Return data
|
# Return data
|
||||||
@ -438,8 +478,8 @@ def parse_video_info(video_id : String, player_response : Hash(String, JSON::Any
|
|||||||
# Extra video infos
|
# Extra video infos
|
||||||
"allowedRegions" => JSON::Any.new(allowed_regions.map { |v| JSON::Any.new(v) }),
|
"allowedRegions" => JSON::Any.new(allowed_regions.map { |v| JSON::Any.new(v) }),
|
||||||
"allowRatings" => JSON::Any.new(allow_ratings || false),
|
"allowRatings" => JSON::Any.new(allow_ratings || false),
|
||||||
"isFamilyFriendly" => JSON::Any.new(family_friendly || false),
|
"isFamilyFriendly" => JSON::Any.new(family_friendly),
|
||||||
"isListed" => JSON::Any.new(is_listed || false),
|
"isListed" => JSON::Any.new(is_listed),
|
||||||
"isUpcoming" => JSON::Any.new(is_upcoming || false),
|
"isUpcoming" => JSON::Any.new(is_upcoming || false),
|
||||||
"keywords" => JSON::Any.new(keywords.map { |v| JSON::Any.new(v) }),
|
"keywords" => JSON::Any.new(keywords.map { |v| JSON::Any.new(v) }),
|
||||||
"isPostLiveDvr" => JSON::Any.new(post_live_dvr),
|
"isPostLiveDvr" => JSON::Any.new(post_live_dvr),
|
||||||
@ -448,7 +488,7 @@ def parse_video_info(video_id : String, player_response : Hash(String, JSON::Any
|
|||||||
# Description
|
# Description
|
||||||
"description" => JSON::Any.new(description || ""),
|
"description" => JSON::Any.new(description || ""),
|
||||||
"descriptionHtml" => JSON::Any.new(description_html || "<p></p>"),
|
"descriptionHtml" => JSON::Any.new(description_html || "<p></p>"),
|
||||||
"shortDescription" => JSON::Any.new(short_description.try &.as_s || nil),
|
"shortDescription" => JSON::Any.new(short_description.try &.as_s || ""),
|
||||||
# Video metadata
|
# Video metadata
|
||||||
"genre" => JSON::Any.new(genre.try &.as_s || ""),
|
"genre" => JSON::Any.new(genre.try &.as_s || ""),
|
||||||
"genreUcid" => JSON::Any.new(genre_ucid.try &.as_s?),
|
"genreUcid" => JSON::Any.new(genre_ucid.try &.as_s?),
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
<% if audio_streams && fmt_stream && preferred_captions && captions %>
|
||||||
<video style="outline:none;width:100%;background-color:#000" playsinline poster="<%= thumbnail %>"
|
<video style="outline:none;width:100%;background-color:#000" playsinline poster="<%= thumbnail %>"
|
||||||
id="player" class="on-video_player video-js player-style-<%= params.player_style %>"
|
id="player" class="on-video_player video-js player-style-<%= params.player_style %>"
|
||||||
preload="<% if params.preload %>auto<% else %>none<% end %>"
|
preload="<% if params.preload %>auto<% else %>none<% end %>"
|
||||||
@ -79,3 +80,4 @@
|
|||||||
%>
|
%>
|
||||||
</script>
|
</script>
|
||||||
<script src="/js/player.js?v=<%= ASSET_COMMIT %>"></script>
|
<script src="/js/player.js?v=<%= ASSET_COMMIT %>"></script>
|
||||||
|
<% end %>
|
||||||
|
@ -31,7 +31,15 @@
|
|||||||
%>
|
%>
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<% if video.reason.nil? %>
|
||||||
|
<h3>
|
||||||
|
<%= video.reason %>
|
||||||
|
</h3>
|
||||||
|
<% else %>
|
||||||
|
<div id="player-container" class="h-box">
|
||||||
<%= rendered "components/player" %>
|
<%= rendered "components/player" %>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
<script src="/js/embed.js?v=<%= ASSET_COMMIT %>"></script>
|
<script src="/js/embed.js?v=<%= ASSET_COMMIT %>"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -4,5 +4,4 @@
|
|||||||
|
|
||||||
<div class="h-box">
|
<div class="h-box">
|
||||||
<%= error_message %>
|
<%= error_message %>
|
||||||
<%= next_steps %>
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -70,9 +70,11 @@ we're going to need to do it here in order to allow for translations.
|
|||||||
%>
|
%>
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<% if video.reason.nil? %>
|
||||||
<div id="player-container" class="h-box">
|
<div id="player-container" class="h-box">
|
||||||
<%= rendered "components/player" %>
|
<%= rendered "components/player" %>
|
||||||
</div>
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<div class="h-box">
|
<div class="h-box">
|
||||||
<h1>
|
<h1>
|
||||||
@ -96,7 +98,10 @@ we're going to need to do it here in order to allow for translations.
|
|||||||
|
|
||||||
<% if video.reason %>
|
<% if video.reason %>
|
||||||
<h3>
|
<h3>
|
||||||
<%= video.reason %>
|
<%= translate(locale, "error_from_youtube_unplayable") %> <%= video.reason %>
|
||||||
|
</h3>
|
||||||
|
<h3>
|
||||||
|
<%= translate(locale, "next_steps_error_message") %>
|
||||||
</h3>
|
</h3>
|
||||||
<% elsif video.premiere_timestamp.try &.> Time.utc %>
|
<% elsif video.premiere_timestamp.try &.> Time.utc %>
|
||||||
<h3>
|
<h3>
|
||||||
@ -112,7 +117,11 @@ we're going to need to do it here in order to allow for translations.
|
|||||||
<div class="pure-g">
|
<div class="pure-g">
|
||||||
<div class="pure-u-1 pure-u-lg-1-5">
|
<div class="pure-u-1 pure-u-lg-1-5">
|
||||||
<div class="h-box">
|
<div class="h-box">
|
||||||
<span id="watch-on-youtube">
|
<span id="refresh-page">
|
||||||
|
<a href="<% env.request.resource %>"><%= translate(locale, "refresh_page") %></a>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<p id="watch-on-youtube">
|
||||||
<%-
|
<%-
|
||||||
link_yt_watch = URI.new(scheme: "https", host: "www.youtube.com", path: "/watch", query: "v=#{video.id}")
|
link_yt_watch = URI.new(scheme: "https", host: "www.youtube.com", path: "/watch", query: "v=#{video.id}")
|
||||||
link_yt_embed = URI.new(scheme: "https", host: "www.youtube.com", path: "/embed/#{video.id}")
|
link_yt_embed = URI.new(scheme: "https", host: "www.youtube.com", path: "/embed/#{video.id}")
|
||||||
@ -125,7 +134,7 @@ we're going to need to do it here in order to allow for translations.
|
|||||||
-%>
|
-%>
|
||||||
<a id="link-yt-watch" rel="noreferrer noopener" data-base-url="<%= link_yt_watch %>" href="<%= link_yt_watch %>"><%= translate(locale, "videoinfo_watch_on_youTube") %></a>
|
<a id="link-yt-watch" rel="noreferrer noopener" data-base-url="<%= link_yt_watch %>" href="<%= link_yt_watch %>"><%= translate(locale, "videoinfo_watch_on_youTube") %></a>
|
||||||
(<a id="link-yt-embed" rel="noreferrer noopener" data-base-url="<%= link_yt_embed %>" href="<%= link_yt_embed %>"><%= translate(locale, "videoinfo_youTube_embed_link") %></a>)
|
(<a id="link-yt-embed" rel="noreferrer noopener" data-base-url="<%= link_yt_embed %>" href="<%= link_yt_embed %>"><%= translate(locale, "videoinfo_youTube_embed_link") %></a>)
|
||||||
</span>
|
</p>
|
||||||
|
|
||||||
<p id="watch-on-another-invidious-instance">
|
<p id="watch-on-another-invidious-instance">
|
||||||
<%- link_iv_other = IV::Frontend::Misc.redirect_url(env) -%>
|
<%- link_iv_other = IV::Frontend::Misc.redirect_url(env) -%>
|
||||||
@ -186,11 +195,14 @@ we're going to need to do it here in order to allow for translations.
|
|||||||
<% end %>
|
<% end %>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
||||||
|
<% if video_assets %>
|
||||||
<%= Invidious::Frontend::WatchPage.download_widget(locale, video, video_assets) %>
|
<%= Invidious::Frontend::WatchPage.download_widget(locale, video, video_assets) %>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<p id="views"><i class="icon ion-ios-eye"></i> <%= number_with_separator(video.views) %></p>
|
<p id="views"><i class="icon ion-ios-eye"></i> <%= number_with_separator(video.views) %></p>
|
||||||
<p id="likes"><i class="icon ion-ios-thumbs-up"></i> <%= number_with_separator(video.likes) %></p>
|
<p id="likes"><i class="icon ion-ios-thumbs-up"></i> <%= number_with_separator(video.likes) %></p>
|
||||||
<p id="dislikes" style="display: none; visibility: hidden;"></p>
|
<p id="dislikes" style="display: none; visibility: hidden;"></p>
|
||||||
|
<% if video.genre %>
|
||||||
<p id="genre"><%= translate(locale, "Genre: ") %>
|
<p id="genre"><%= translate(locale, "Genre: ") %>
|
||||||
<% if !video.genre_url %>
|
<% if !video.genre_url %>
|
||||||
<%= video.genre %>
|
<%= video.genre %>
|
||||||
@ -198,6 +210,7 @@ we're going to need to do it here in order to allow for translations.
|
|||||||
<a href="<%= video.genre_url %>"><%= video.genre %></a>
|
<a href="<%= video.genre_url %>"><%= video.genre %></a>
|
||||||
<% end %>
|
<% end %>
|
||||||
</p>
|
</p>
|
||||||
|
<% end %>
|
||||||
<% if video.license %>
|
<% if video.license %>
|
||||||
<% if video.license.empty? %>
|
<% if video.license.empty? %>
|
||||||
<p id="license"><%= translate(locale, "License: ") %><%= translate(locale, "Standard YouTube license") %></p>
|
<p id="license"><%= translate(locale, "License: ") %><%= translate(locale, "Standard YouTube license") %></p>
|
||||||
|
@ -68,6 +68,23 @@ rescue ex
|
|||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def has_unlisted_badge?(badges : JSON::Any?)
|
||||||
|
return false if badges.nil?
|
||||||
|
|
||||||
|
badges.as_a.each do |badge|
|
||||||
|
icon_type = badge.dig("metadataBadgeRenderer", "icon", "iconType").as_s
|
||||||
|
|
||||||
|
return true if icon_type == "PRIVACY_UNLISTED"
|
||||||
|
end
|
||||||
|
|
||||||
|
return false
|
||||||
|
rescue ex
|
||||||
|
LOGGER.debug("Unable to parse owner badges. Got exception: #{ex.message}")
|
||||||
|
LOGGER.trace("Owner badges data: #{badges.to_json}")
|
||||||
|
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
# This function extracts SearchVideo items from a Category.
|
# This function extracts SearchVideo items from a Category.
|
||||||
# Categories are commonly returned in search results and trending pages.
|
# Categories are commonly returned in search results and trending pages.
|
||||||
def extract_category(category : Category) : Array(SearchVideo)
|
def extract_category(category : Category) : Array(SearchVideo)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user