mirror of
https://github.com/iv-org/invidious.git
synced 2025-01-19 05:02:41 +05:30
Compare commits
3 Commits
3509752b79
...
3615bb0e62
Author | SHA1 | Date | |
---|---|---|---|
|
3615bb0e62 | ||
|
7d435f082b | ||
|
1f7592e599 |
@ -7,7 +7,7 @@ module Invidious::Frontend::WatchPage
|
|||||||
getter full_videos : Array(Hash(String, JSON::Any))
|
getter full_videos : Array(Hash(String, JSON::Any))
|
||||||
getter video_streams : Array(Hash(String, JSON::Any))
|
getter video_streams : Array(Hash(String, JSON::Any))
|
||||||
getter audio_streams : Array(Hash(String, JSON::Any))
|
getter audio_streams : Array(Hash(String, JSON::Any))
|
||||||
getter captions : Array(Invidious::Videos::CaptionMetadata)
|
getter captions : Array(Invidious::Videos::Captions::Metadata)
|
||||||
|
|
||||||
def initialize(
|
def initialize(
|
||||||
@full_videos,
|
@full_videos,
|
||||||
|
@ -24,7 +24,7 @@ struct Video
|
|||||||
property updated : Time
|
property updated : Time
|
||||||
|
|
||||||
@[DB::Field(ignore: true)]
|
@[DB::Field(ignore: true)]
|
||||||
@captions = [] of Invidious::Videos::CaptionMetadata
|
@captions = [] of Invidious::Videos::Captions::Metadata
|
||||||
|
|
||||||
@[DB::Field(ignore: true)]
|
@[DB::Field(ignore: true)]
|
||||||
property adaptive_fmts : Array(Hash(String, JSON::Any))?
|
property adaptive_fmts : Array(Hash(String, JSON::Any))?
|
||||||
@ -215,9 +215,9 @@ struct Video
|
|||||||
keywords.includes? "YouTube Red"
|
keywords.includes? "YouTube Red"
|
||||||
end
|
end
|
||||||
|
|
||||||
def captions : Array(Invidious::Videos::CaptionMetadata)
|
def captions : Array(Invidious::Videos::Captions::Metadata)
|
||||||
if @captions.empty? && @info.has_key?("captions")
|
if @captions.empty? && @info.has_key?("captions")
|
||||||
@captions = Invidious::Videos::CaptionMetadata.from_yt_json(info["captions"])
|
@captions = Invidious::Videos::Captions::Metadata.from_yt_json(info["captions"])
|
||||||
end
|
end
|
||||||
|
|
||||||
return @captions
|
return @captions
|
||||||
|
@ -1,107 +1,106 @@
|
|||||||
require "json"
|
require "json"
|
||||||
|
|
||||||
module Invidious::Videos
|
module Invidious::Videos
|
||||||
struct CaptionMetadata
|
module Captions
|
||||||
property name : String
|
struct Metadata
|
||||||
property language_code : String
|
property name : String
|
||||||
property base_url : String
|
property language_code : String
|
||||||
|
property base_url : String
|
||||||
|
|
||||||
property auto_generated : Bool
|
property auto_generated : Bool
|
||||||
|
|
||||||
def initialize(@name, @language_code, @base_url, @auto_generated)
|
def initialize(@name, @language_code, @base_url, @auto_generated)
|
||||||
end
|
|
||||||
|
|
||||||
# Parse the JSON structure from Youtube
|
|
||||||
def self.from_yt_json(container : JSON::Any) : Array(CaptionMetadata)
|
|
||||||
caption_tracks = container
|
|
||||||
.dig?("playerCaptionsTracklistRenderer", "captionTracks")
|
|
||||||
.try &.as_a
|
|
||||||
|
|
||||||
captions_list = [] of CaptionMetadata
|
|
||||||
return captions_list if caption_tracks.nil?
|
|
||||||
|
|
||||||
caption_tracks.each do |caption|
|
|
||||||
name = caption["name"]["simpleText"]? || caption["name"]["runs"][0]["text"]
|
|
||||||
name = name.to_s.split(" - ")[0]
|
|
||||||
|
|
||||||
language_code = caption["languageCode"].to_s
|
|
||||||
base_url = caption["baseUrl"].to_s
|
|
||||||
|
|
||||||
auto_generated = false
|
|
||||||
if caption["kind"]? && caption["kind"] == "asr"
|
|
||||||
auto_generated = true
|
|
||||||
end
|
|
||||||
|
|
||||||
captions_list << CaptionMetadata.new(name, language_code, base_url, auto_generated)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
return captions_list
|
# Parse the JSON structure from Youtube
|
||||||
end
|
def self.from_yt_json(container : JSON::Any) : Array(Captions::Metadata)
|
||||||
|
caption_tracks = container
|
||||||
|
.dig?("playerCaptionsTracklistRenderer", "captionTracks")
|
||||||
|
.try &.as_a
|
||||||
|
|
||||||
def timedtext_to_vtt(timedtext : String, tlang = nil) : String
|
captions_list = [] of Captions::Metadata
|
||||||
# In the future, we could just directly work with the url. This is more of a POC
|
return captions_list if caption_tracks.nil?
|
||||||
cues = [] of XML::Node
|
|
||||||
tree = XML.parse(timedtext)
|
|
||||||
tree = tree.children.first
|
|
||||||
|
|
||||||
tree.children.each do |item|
|
caption_tracks.each do |caption|
|
||||||
if item.name == "body"
|
name = caption["name"]["simpleText"]? || caption["name"]["runs"][0]["text"]
|
||||||
item.children.each do |cue|
|
name = name.to_s.split(" - ")[0]
|
||||||
if cue.name == "p" && !(cue.children.size == 1 && cue.children[0].content == "\n")
|
|
||||||
cues << cue
|
language_code = caption["languageCode"].to_s
|
||||||
|
base_url = caption["baseUrl"].to_s
|
||||||
|
|
||||||
|
auto_generated = (caption["kind"]? == "asr")
|
||||||
|
|
||||||
|
captions_list << Captions::Metadata.new(name, language_code, base_url, auto_generated)
|
||||||
|
end
|
||||||
|
|
||||||
|
return captions_list
|
||||||
|
end
|
||||||
|
|
||||||
|
def timedtext_to_vtt(timedtext : String, tlang = nil) : String
|
||||||
|
# In the future, we could just directly work with the url. This is more of a POC
|
||||||
|
cues = [] of XML::Node
|
||||||
|
tree = XML.parse(timedtext)
|
||||||
|
tree = tree.children.first
|
||||||
|
|
||||||
|
tree.children.each do |item|
|
||||||
|
if item.name == "body"
|
||||||
|
item.children.each do |cue|
|
||||||
|
if cue.name == "p" && !(cue.children.size == 1 && cue.children[0].content == "\n")
|
||||||
|
cues << cue
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
break
|
||||||
end
|
end
|
||||||
break
|
|
||||||
end
|
end
|
||||||
end
|
result = String.build do |result|
|
||||||
result = String.build do |result|
|
result << <<-END_VTT
|
||||||
result << <<-END_VTT
|
WEBVTT
|
||||||
WEBVTT
|
Kind: captions
|
||||||
Kind: captions
|
Language: #{tlang || @language_code}
|
||||||
Language: #{tlang || @language_code}
|
|
||||||
|
|
||||||
|
|
||||||
END_VTT
|
END_VTT
|
||||||
|
|
||||||
result << "\n\n"
|
result << "\n\n"
|
||||||
|
|
||||||
cues.each_with_index do |node, i|
|
cues.each_with_index do |node, i|
|
||||||
start_time = node["t"].to_f.milliseconds
|
start_time = node["t"].to_f.milliseconds
|
||||||
|
|
||||||
duration = node["d"]?.try &.to_f.milliseconds
|
duration = node["d"]?.try &.to_f.milliseconds
|
||||||
|
|
||||||
duration ||= start_time
|
duration ||= start_time
|
||||||
|
|
||||||
if cues.size > i + 1
|
if cues.size > i + 1
|
||||||
end_time = cues[i + 1]["t"].to_f.milliseconds
|
end_time = cues[i + 1]["t"].to_f.milliseconds
|
||||||
else
|
else
|
||||||
end_time = start_time + duration
|
end_time = start_time + duration
|
||||||
|
end
|
||||||
|
|
||||||
|
# start_time
|
||||||
|
result << start_time.hours.to_s.rjust(2, '0')
|
||||||
|
result << ':' << start_time.minutes.to_s.rjust(2, '0')
|
||||||
|
result << ':' << start_time.seconds.to_s.rjust(2, '0')
|
||||||
|
result << '.' << start_time.milliseconds.to_s.rjust(3, '0')
|
||||||
|
|
||||||
|
result << " --> "
|
||||||
|
|
||||||
|
# end_time
|
||||||
|
result << end_time.hours.to_s.rjust(2, '0')
|
||||||
|
result << ':' << end_time.minutes.to_s.rjust(2, '0')
|
||||||
|
result << ':' << end_time.seconds.to_s.rjust(2, '0')
|
||||||
|
result << '.' << end_time.milliseconds.to_s.rjust(3, '0')
|
||||||
|
|
||||||
|
result << "\n"
|
||||||
|
|
||||||
|
node.children.each do |s|
|
||||||
|
result << s.content
|
||||||
|
end
|
||||||
|
result << "\n"
|
||||||
|
result << "\n"
|
||||||
end
|
end
|
||||||
|
|
||||||
# start_time
|
|
||||||
result << start_time.hours.to_s.rjust(2, '0')
|
|
||||||
result << ':' << start_time.minutes.to_s.rjust(2, '0')
|
|
||||||
result << ':' << start_time.seconds.to_s.rjust(2, '0')
|
|
||||||
result << '.' << start_time.milliseconds.to_s.rjust(3, '0')
|
|
||||||
|
|
||||||
result << " --> "
|
|
||||||
|
|
||||||
# end_time
|
|
||||||
result << end_time.hours.to_s.rjust(2, '0')
|
|
||||||
result << ':' << end_time.minutes.to_s.rjust(2, '0')
|
|
||||||
result << ':' << end_time.seconds.to_s.rjust(2, '0')
|
|
||||||
result << '.' << end_time.milliseconds.to_s.rjust(3, '0')
|
|
||||||
|
|
||||||
result << "\n"
|
|
||||||
|
|
||||||
node.children.each do |s|
|
|
||||||
result << s.content
|
|
||||||
end
|
|
||||||
result << "\n"
|
|
||||||
result << "\n"
|
|
||||||
end
|
end
|
||||||
|
return result
|
||||||
end
|
end
|
||||||
return result
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# List of all caption languages available on Youtube.
|
# List of all caption languages available on Youtube.
|
||||||
|
@ -4,16 +4,13 @@ module Invidious::Videos
|
|||||||
record TranscriptLine, start_ms : Time::Span, end_ms : Time::Span, line : String
|
record TranscriptLine, start_ms : Time::Span, end_ms : Time::Span, line : String
|
||||||
|
|
||||||
def self.generate_param(video_id : String, language_code : String, auto_generated : Bool) : String
|
def self.generate_param(video_id : String, language_code : String, auto_generated : Bool) : String
|
||||||
if !auto_generated
|
kind = auto_generated ? "asr" : ""
|
||||||
is_auto_generated = ""
|
|
||||||
elsif is_auto_generated = "asr"
|
|
||||||
end
|
|
||||||
|
|
||||||
object = {
|
object = {
|
||||||
"1:0:string" => video_id,
|
"1:0:string" => video_id,
|
||||||
|
|
||||||
"2:base64" => {
|
"2:base64" => {
|
||||||
"1:string" => is_auto_generated,
|
"1:string" => kind,
|
||||||
"2:string" => language_code,
|
"2:string" => language_code,
|
||||||
"3:string" => "",
|
"3:string" => "",
|
||||||
},
|
},
|
||||||
@ -37,7 +34,7 @@ module Invidious::Videos
|
|||||||
# Convert into array of TranscriptLine
|
# Convert into array of TranscriptLine
|
||||||
lines = self.parse(initial_data)
|
lines = self.parse(initial_data)
|
||||||
|
|
||||||
# Taken from Invidious::Videos::CaptionMetadata.timedtext_to_vtt()
|
# Taken from Invidious::Videos::Captions::Metadata.timedtext_to_vtt()
|
||||||
vtt = String.build do |vtt|
|
vtt = String.build do |vtt|
|
||||||
vtt << <<-END_VTT
|
vtt << <<-END_VTT
|
||||||
WEBVTT
|
WEBVTT
|
||||||
|
@ -89,7 +89,7 @@
|
|||||||
<label for="captions[0]"><%= translate(locale, "preferences_captions_label") %></label>
|
<label for="captions[0]"><%= translate(locale, "preferences_captions_label") %></label>
|
||||||
<% preferences.captions.each_with_index do |caption, index| %>
|
<% preferences.captions.each_with_index do |caption, index| %>
|
||||||
<select class="pure-u-1-6" name="captions[<%= index %>]" id="captions[<%= index %>]">
|
<select class="pure-u-1-6" name="captions[<%= index %>]" id="captions[<%= index %>]">
|
||||||
<% Invidious::Videos::CaptionMetadata::LANGUAGES.each do |option| %>
|
<% Invidious::Videos::Captions::LANGUAGES.each do |option| %>
|
||||||
<option value="<%= option %>" <% if preferences.captions[index] == option %> selected <% end %>><%= translate(locale, option.blank? ? "none" : option) %></option>
|
<option value="<%= option %>" <% if preferences.captions[index] == option %> selected <% end %>><%= translate(locale, option.blank? ? "none" : option) %></option>
|
||||||
<% end %>
|
<% end %>
|
||||||
</select>
|
</select>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user