forked from cyl0/ModernX
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathmodernx.lua
4577 lines (4029 loc) · 177 KB
/
modernx.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
--[[
ModernX by zydezu
(https://github.com/zydezu/ModernX)
This script is a result of the original mpv-osc-modern by maoiscat
and it's subsequent forks:
* cyl0/ModernX
* dexeonify/ModernX
* Samillion/ModernZ
Based on the osc.lua from mpv
--]]
mp.assdraw = require("mp.assdraw")
mp.msg = require("mp.msg")
mp.utils = require("mp.utils")
-- ====================
-- declarations
-- ====================
local function update_tracklist() end
local function get_tracklist() end
local function set_track() end
local function get_track() end
local function window_controls_enabled() end
local function get_chapter() end
local function render_elements() end
local function render_persistent_progressbar() end
local function limited_list() end
local function checktitle() end
local function normaliseDate(date) end
local function exec_async() end
local function is_url() end
local function check_path_url() end
local function check_comments() end
local function loadSetOfComments() end
local function process_filesize() end
local function splitUTF8(str, maxLength) end
local function process_vid_stats() end
local function process_dislikes() end
local function add_commas_to_number() end
local function addLikeCountToTitle() end
local function format_file_size(file_size) end
local function get_playlist() end
local function get_chapterlist() end
local function show_message(text, duration) end
local function bind_keys() end
local function unbind_keys() end
local function destroyscrollingkeys() end
local function check_description() end
local function show_description(text) end
local function reset_desc_timer() end
local function render_message() end
local function window_controls() end
local function validate_user_opts() end
local function update_options(list) end
local function show_osc() end
local function hide_osc() end
local function osc_visible(visible) end
local function adjustSubtitles(visible) end
local function pause_state() end
local function cache_state() end
local function process_event() end
local function tick() end
local function reset_timeout() end
local function visibility_mode(mode) end
-- ====================
-- Parameters
-- default user option values
-- change them using modernx.conf
-- ====================
local user_opts = {
-- Language and display --
language = "en", -- en:English - .json translations need implementing
font = "mpv-osd-symbols", -- font for the OSC (default: mpv-osd-symbols or the one set in mpv.conf)
layout_option = "original", -- use the original/reduced layout
idle_screen = true, -- show mpv logo when idle
key_bindings = true, -- register additional key bindings, such as chapter scrubbing, pinning the window
window_top_bar = "auto", -- show OSC window top bar: "auto", "yes", or "no" (borderless/fullscreen)
show_windowed = true, -- show OSC when windowed
show_fullscreen = true, -- show OSC when fullscreen
show_on_pause = true, -- show OSC when paused
keep_on_pause = false, -- disable OSC hide timeout when paused
green_and_grumpy = false, -- disable Santa hat in December
visibility = "auto", -- only used at init to set visibility_mode(...)
-- OSC behaviour and scaling
hide_timeout = 1500, -- time (in ms) before OSC hides if no mouse movement
seek_resets_hide_timeout = true, -- if seeking should reset the hide_timeout
fade_duration = 150, -- fade-out duration (in ms), set to 0 for no fade
min_mouse_move = 0, -- minimum mouse movement (in pixels) required to show OSC
bottom_hover = true, -- show OSC only when hovering at the bottom
bottom_hover_zone = 200, -- height of hover zone for bottom_hover (in pixels)
osc_on_seek = false, -- show OSC when seeking
mouse_seek_pause = true, -- pause video while seeking with mouse move (on button hold)
vid_scale = false, -- scale osc with the video
scale_windowed = 1.0, -- osc scale factor when windowed
scale_fullscreen = 1.0, -- osc scale factor when fullscreen
scale_forced_window = 1.0, -- osc scale factor when forced (no video, like music files)
-- Time, title and description display
show_title = true, -- show title in the OSC (above seekbar)
title = "${media-title}", -- title above seekbar format: "${media-title}" or "${filename}"
title_font_size = 28, -- font size of the title text (above seekbar)
dynamic_title = true, -- change title if {media-title} and {filename} differ (eg: when playing URLs or audio)
show_chapter_title = true, -- show chapter title alongside timestamp (below seekbar)
chapter_fmt = "%s", -- format for chapter display on seekbar hover (set to "no" to disable)
show_chapter_markers = false, -- show chapter markers on the seekbar
time_total = true, -- show total time instead of remaining time
time_ms = false, -- show timecodes with milliseconds
unicode_minus = false, -- use the Unicode minus sign in remaining time
time_format = "dynamic", -- "dynamic" or "fixed" - dynamic shows MM:SS when possible, fixed always shows HH:MM:SS
time_font_size = 18, -- font size of the time display
show_description = true, -- show video description - description on web videos or metadata/stats on local video
show_file_size = true, -- show the current file's size in the description
description_font_size = 19, -- font size of the description text (below title)
description_alpha = 100, -- alpha of the description background box
scrolling_speed = 40, -- the speed of scrolling text in description/comment menus
date_format = "%Y-%m-%d", -- how dates should be formatted, when read from metadata (uses standard lua date formatting)
-- Title bar settings
window_title = true, -- show window title in borderless/fullscreen mode
window_controls = true, -- show window controls (close, minimize, maximize) in borderless/fullscreen
title_bar_box = false, -- show title bar as a box instead of a black fade
window_controls_title = "${media-title}", -- same as title but for window_controls
-- Subtitle display settings
raise_subtitles = true, -- whether to raise subtitles above the osc when it's shown
raise_subtitle_amount = 175, -- how much subtitles rise when the osc is shown
-- Buttons display and functionality
compact_mode = true, -- replace the jump buttons with the seek/chapter buttons
jump_buttons = true, -- show the jump backward and forward buttons
jump_amount = 10, -- change the jump amount in seconds
jump_more_amount = 60, -- change the jump amount in seconds when right-clicking jump buttons and shift-clicking chapter skip buttons
jump_icon_number = true, -- show different icon when jump_amount is set to 5, 10, or 30
jump_mode = "relative", -- seek mode for jump buttons
jump_softrepeat = true, -- enable continuous jumping when holding down seek buttons
chapter_skip_buttons = true, -- show the chapter skip backward and forward buttons
chapter_softrepeat = false, -- enable continuous skipping when holding down chapter skip buttons
track_nextprev_buttons = true, -- show next/previous playlist track buttons
volume_control = true, -- show mute button and volume slider
volume_control_type = "linear", -- volume scale type: "linear" or "logarithmic"
info_button = false, -- show info button
ontop_button = true, -- show window on top button
screenshot_button = false, -- show screenshot button
screenshot_flag = "subtitles", -- flag for screenshot button: "subtitles", "video", "window", "each-frame"
-- https://mpv.io/manual/master/#command-interface-screenshot-%3Cflags%3E
download_button = true, -- show download button on web videos (requires yt-dlp and ffmpeg)
download_path = "~~desktop/mpv/downloads", -- default download directory for videos (https://mpv.io/manual/master/#paths)
loop_button = false, -- show loop button
loop_in_pause = true, -- enable looping by right-clicking pause
playpause_size = 30, -- icon size for the play/pause button
midbuttons_size = 24, -- icon size for the middle buttons
sidebuttons_size = 24, -- icon size for the side buttons
-- Colors and style
osc_color = "#000000", -- accent color of the OSC and title bar
window_title_color = "#FFFFFF", -- color of the title in borderless/fullscreen mode
window_controls_color = "#FFFFFF", -- color of the window controls (close, minimize, maximize) in borderless/fullscreen mode
window_controls_close_hover = "#E81123", -- color of close window control on hover
window_controls_minmax_hover = "#53A4FC", -- color of min/max window controls on hover
title_color = "#FFFFFF", -- color of the title (above seekbar)
seekbarfg_color = "#1D96F5", -- color of the seekbar progress and handle, in Hex color format
seekbarbg_color = "#FFFFFF", -- color of the remaining seekbar, in Hex color format
seekbar_cache_color = "#1D96F5", -- color of the cache ranges on the seekbar
volumebar_match_seek_color = false, -- match volume bar color with seekbar color (ignores side_buttons_color)
time_color = "#FFFFFF", -- color of the timestamps (below seekbar)
chapter_title_color = "#FFFFFF", -- color of the chapter title next to timestamp (below seekbar)
side_buttons_color = "#FFFFFF", -- color of the side buttons (audio, subtitles, playlist, etc.)
middle_buttons_color = "#FFFFFF", -- color of the middle buttons (skip, jump, chapter, etc.)
playpause_color = "#FFFFFF", -- color of the play/pause button
held_element_color = "#999999", -- color of the element when held down (pressed)
hover_effect_color = "#FFFFFF", -- color of a hovered button when hover_effect includes "color"
thumbnail_border_color = "#FFFFFF", -- color of the border for thumbnails (with thumbfast)
thumbnail_border_outline = "#000000", -- color of the border outline for thumbnails
fade_alpha = 100, -- alpha of the title bar background box
fade_blur_strength = 75, -- blur strength for the OSC alpha fade - caution: high values can take a lot of CPU time to render
title_bar_fade_alpha = 150, -- alpha of the OSC background box
title_bar_fade_blur_strength = 100, -- blur strength for the title bar alpha fade
window_fade_alpha = 75, -- alpha of the window title bar
thumbnail_border = 3, -- width of the thumbnail border (for thumbfast)
thumbnail_border_radius = 3, -- rounded corner radius for thumbnail border (0 to disable)
-- Button hover effects
hover_effect = "size,glow,color", -- active button hover effects: "glow", "size", "color"; can use multiple separated by commas
hover_button_size = 115, -- relative size of a hovered button if "size" effect is active
button_glow_amount = 5, -- glow intensity when "glow" hover effect is active
hover_effect_for_sliders = false, -- apply hover effects to slider handles
-- Progress bar settings
seek_handle_size = 0.8, -- size ratio of the seekbar handle (range: 0 ~ 1)
progress_bar_height = 16, -- height of the progress bar
seek_range = true, -- show seek range overlay
seek_range_alpha = 175, -- transparency of the seek range
seekbar_keyframes = false, -- use keyframes when dragging the seekbar
automatic_keyframe_mode = true, -- automatically set keyframes for the seekbar based on video length
automatic_keyframe_limit = 600, -- videos longer than this (in seconds) will have keyframes on the seekbar
persistent_progress_default = false, -- always show a small progress line at the bottom of the screen
persistent_progress_height = 17, -- height of the persistent_progress bar
persistent_buffer = false, -- show the buffer on the persistent progress line
persistent_progress_toggle = true, -- enable toggling the persistent_progress bar
-- Web videos
title_youtube_stats = true, -- update the window/OSC title bar with YouTube video stats (views, likes, dislikes)
ytdl_format = "", -- optional parameteres for yt-dlp downloading, eg: '-f bestvideo+bestaudio/best'
-- sponsorblock features need https://github.com/zydezu/mpvconfig/blob/main/scripts/sponsorblock.lua to work!
show_sponsorblock_segments = true, -- show sponsorblock segments on the progress bar
add_sponsorblock_chapters = false, -- add sponsorblock chapters to the chapter list
sponsorblock_seek_range_alpha = 75, -- transparency of sponsorblock segments
sponsor_types = { -- what categories to show in the progress bar
"sponsor", -- all categories:
"intro", -- sponsor, intro, outro,
"outro", -- interaction, selfpromo, preview,
"interaction", -- music_offtopic, filler
"selfpromo",
"preview",
"music_offtopic",
"filler"
},
sponsorblock_sponsor_color = "#00D400", -- color for sponsors
sponsorblock_intro_color = "#00FFFF", -- color for intermission/intro animations
sponsorblock_outro_color = "#0202ED", -- color for endcards/credits
sponsorblock_interaction_color = "#CC00FF", -- color for interaction reminders (reminders to subscribe)
sponsorblock_selfpromo_color = "#FFFF00", -- color for unpaid/self promotion
sponsorblock_preview_color = "#008FD6", -- color for unpaid/self promotion
sponsorblock_music_offtopic_color = "#FF9900", -- color for unpaid/self promotion
sponsorblock_filler_color = "#7300FF", -- color for filler tangent/jokes
-- Experimental
show_youtube_comments = false, -- EXPERIMENTAL - show youtube comments
comments_download_path = "~~desktop/mpv/downloads/comments", -- EXPERIMENTAL - the download path for the comment JSON file
FORCE_fix_not_ontop = true, -- EXPERIMENTAL - try and mitigate https://github.com/zydezu/ModernX/issues/30, https://github.com/akiirui/mpv-handler/issues/48
}
-- read options from config and command-line
require("mp.options").read_options(user_opts, 'modernx', function(list) update_options(list) end)
mp.observe_property("osc", "bool", function(name, value) if value == true then mp.set_property("osc", "no") end end)
local osc_param = { -- calculated by osc_init()
playresy = 0, -- canvas size Y
playresx = 0, -- canvas size X
display_aspect = 1,
unscaled_y = 0,
areas = {},
}
local icons = {
play = "\238\166\143",
pause = "\238\163\140",
replay = "\238\189\191",
previous = "\239\152\167",
next = "\239\149\168",
rewind = "\238\168\158",
forward = "\238\152\135",
audio = "\238\175\139",
subtitle = "\238\175\141",
volume_mute = "\238\173\138",
volume_quiet = "\238\172\184",
volume_low = "\238\172\189",
volume_high = "\238\173\130",
download = "\239\133\144",
downloading = "\239\140\174",
loop_off = "\239\133\178",
loop_on = "\239\133\181",
info = "\239\146\164",
ontop_on = "\238\165\190",
ontop_off = "\238\166\129",
screenshot = "\239\154\142",
fullscreen = "\239\133\160",
fullscreen_exit = "\239\133\166",
jumpicons = {
[5] = {"\238\171\186", "\238\171\187"},
[10] = {"\238\171\188", "\238\172\129"},
[30] = {"\238\172\133", "\238\172\134"},
default = {"\238\172\138", "\238\172\138"}, -- second icon is mirrored in layout()
},
emoticon = {
view = "👁️",
comment = "💬",
like = "👍",
dislike = "👎"
},
playlist = "\238\161\159", -- unused rn
}
-- Localization
local language = {
['en'] = {
welcome = 'Drop files or URLs here to play', -- this text appears when mpv starts
off = 'OFF',
na = 'Not available',
none = 'None available',
video = 'Video',
audio = 'Audio',
subtitle = 'Subtitle',
nosub = 'No subtitles available',
noaudio = 'No audio tracks available',
track = ' tracks:',
playlist = 'Playlist',
nolist = 'Playlist is empty',
chapter = 'Chapter',
nochapter = 'No chapters available',
ontop = 'Pin window',
ontopdisable = 'Unpin window',
loopenable = 'Enable loop',
loopdisable = 'Disable loop',
screenshot = "Screenshot",
statsinfo = "Information",
download = "Download",
download_in_progress = "Download in progress",
downloading = "Downloading",
downloaded = "Already downloaded",
}
}
-- apply lang opts
local texts = language[user_opts.language] or language["en"]
local function contains(list, item)
local t = {}
if type(list) ~= "table" then
for str in string.gmatch(list, '([^,]+)') do
str = str:gsub("%s+", "")
table.insert(t, str)
end
else
t = list
end
for _, v in ipairs(t) do
if v == item then
return true
end
end
return false
end
local function dumptable(o)
if type(o) == 'table' then
local s = '{ '
for k,v in pairs(o) do
if type(k) ~= 'number' then k = '"'..k..'"' end
s = s .. '['..k..'] = ' .. dumptable(v) .. ','
end
return s .. '} '
else
return tostring(o)
end
end
local thumbfast = {
width = 0,
height = 0,
disabled = true,
available = false
}
local sponsorblock_color_map = {
sponsor = user_opts.sponsorblock_sponsor_color,
intro = user_opts.sponsorblock_intro_color,
outro = user_opts.sponsorblock_outro_color,
interaction = user_opts.sponsorblock_interaction_color,
selfpromo = user_opts.sponsorblock_selfpromo_color,
preview = user_opts.sponsorblock_preview_color,
music_offtopic = user_opts.sponsorblock_music_offtopic_color,
filler = user_opts.sponsorblock_filler_color
}
local tick_delay = 1 / 60 -- 60FPS
local audio_track_count = 0 -- TODO: implement
local sub_track_count = 0 -- TODO: implement
local window_control_box_width = 138
local max_descsize = 125
local comments_per_page = 25
local is_december = os.date("*t").month == 12
local UNICODE_MINUS = string.char(0xe2, 0x88, 0x92) -- UTF-8 for U+2212 MINUS SIGN
local iconfont = 'fluent-system-icons'
local function osc_color_convert(color)
return color:sub(6,7) .. color:sub(4,5) .. color:sub(2,3)
end
local playpause_size = user_opts.playpause_size or 30
local midbuttons_size = user_opts.midbuttons_size or 24
local sidebuttons_size = user_opts.sidebuttons_size or 24
local osc_styles = {
background_bar = "{\\1c&H" .. osc_color_convert(user_opts.osc_color) .. "&}",
box_bg = "{\\blur" .. user_opts.fade_blur_strength .. "\\bord" .. user_opts.fade_alpha .. "\\1c&H000000&\\3c&H" .. osc_color_convert(user_opts.osc_color) .. "&}",
title_bar_box_bg = "{\\blur" .. user_opts.title_bar_fade_blur_strength .. "\\bord" .. user_opts.title_bar_fade_alpha .. "\\1c&H000000&\\3c&H" .. osc_color_convert(user_opts.osc_color) .. "&}",
chapter_title = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.chapter_title_color) .. "&\\3c&H000000&\\fs" .. user_opts.time_font_size .. "\\fn" .. user_opts.font .. "}",
control_1 = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.playpause_color) .. "&\\3c&HFFFFFF&\\fs" .. playpause_size .. "\\fn" .. iconfont .. "}",
control_2 = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.middle_buttons_color) .. "&\\3c&HFFFFFF&\\fs" .. midbuttons_size .. "\\fn" .. iconfont .. "}",
control_2_flip = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.middle_buttons_color) .. "&\\3c&HFFFFFF&\\fs" .. midbuttons_size .. "\\fn" .. iconfont .. "\\fry180}",
control_3 = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.side_buttons_color) .. "&\\3c&HFFFFFF&\\fs" .. sidebuttons_size .. "\\fn" .. iconfont .. "}",
element_down = "{\\1c&H" .. osc_color_convert(user_opts.held_element_color) .. "&}",
element_hover = "{" .. (contains(user_opts.hover_effect, "color") and "\\1c&H" .. osc_color_convert(user_opts.hover_effect_color) .. "&" or "") .."\\2c&HFFFFFF&" .. (contains(user_opts.hover_effect, "size") and string.format("\\fscx%s\\fscy%s", user_opts.hover_button_size, user_opts.hover_button_size) or "") .. "}",
seekbar_bg = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.seekbarbg_color) .. "&}",
seekbar_fg = "{\\blur1\\bord1\\1c&H" .. osc_color_convert(user_opts.seekbarfg_color) .. "&}",
thumbnail = "{\\blur0\\bord1\\1c&H" .. osc_color_convert(user_opts.thumbnail_border_color) .. "&\\3c&H" .. osc_color_convert(user_opts.thumbnail_border_outline) .. "&}",
time = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.time_color) .. "&\\3c&H000000&\\fs" .. user_opts.time_font_size .. "\\fn" .. user_opts.font .. "}",
title = "{\\blur1\\bord0.5\\1c&H" .. osc_color_convert(user_opts.title_color) .. "&\\3c&H0&\\fs".. user_opts.title_font_size .."\\q2\\fn" .. user_opts.font .. "}",
tooltip = "{\\blur1\\bord0.5\\1c&HFFFFFF&\\3c&H000000&\\fs" .. user_opts.time_font_size .. "\\fn" .. user_opts.font .. "}",
volumebar_bg = "{\\blur0\\bord0\\1c&H999999&}",
volumebar_fg = "{\\blur1\\bord1\\1c&H" .. osc_color_convert(user_opts.side_buttons_color) .. "&}",
window_control = "{\\blur1\\bord0.5\\1c&H" .. osc_color_convert(user_opts.window_controls_color) .. "&\\3c&H0&\\fs20\\fnmpv-osd-symbols}",
window_title = "{\\blur1\\bord0.5\\1c&H" .. osc_color_convert(user_opts.window_title_color) .. "&\\3c&H0&\\fs20\\q2\\fn" .. user_opts.font .. "}",
description = '{\\blur1\\bord0.5\\1c&HFFFFFF&\\3c&H000000&\\fs'.. user_opts.description_font_size ..'\\q2\\fn' .. user_opts.font .. '}',
}
-- internal states, do not touch
local state = {
showtime = nil, -- time of last invocation (last mouse move)
osc_visible = false,
anistart = nil, -- time when the animation started
anitype = nil, -- current type of animation
animation = nil, -- current animation alpha
mouse_down_counter = 0, -- used for softrepeat
active_element = nil, -- nil = none, 0 = background, 1+ = see elements[]
active_event_source = nil, -- the "button" that issued the current event
tc_right_rem = not user_opts.time_total, -- if the right timecode should display total or remaining time
fulltime = user_opts.time_ms,
mp_screen_sizeX = nil, mp_screen_sizeY = nil, -- last screen-resolution, to detect resolution changes to issue reINITs
initREQ = false, -- is a re-init request pending?
last_mouseX = nil, last_mouseY = nil, -- last mouse position, to detect significant mouse movement
mouse_in_window = false,
fullscreen = false,
tick_timer = nil,
tick_last_time = 0, -- when the last tick() was run
hide_timer = nil,
cache_state = nil,
buffering = false,
idle = false,
enabled = true,
input_enabled = true,
showhide_enabled = false,
border = true,
title_bar = true,
maximized = false,
osd = mp.create_osd_overlay('ass-events'),
new_file_flag = false, -- flag to detect new file starts
chapter_list = {}, -- sorted by time
chapter_list_pre_sponsorblock = {},
mute = false,
looping = false,
sliderpos = 0,
touchingprogressbar = false, -- if the mouse is touching the progress bar
initialborder = mp.get_property('border'),
playingWhilstSeeking = false,
playingWhilstSeekingWaitingForEnd = false,
persistent_progresstoggle = user_opts.persistent_progress_default,
downloaded_once = false,
downloading = false,
file_size_bytes = 0,
file_size_normalized = "Approximating size...",
is_URL = false,
URL_path = "", -- used for yt-dlp downloading
videoCantBeDownloaded = false, -- TODO: needs to be removed
localDescription = nil,
localDescriptionClick = nil,
localDescriptionIsClickable = false,
videoDescription = "", -- use if it is a YouTube video
descriptionLoaded = false,
showingDescription = false,
scrolledlines = 25,
youtubeuploader = "",
jsoncomments= {},
youtubecomments = {},
commentsParsed = false,
currentCommentIndex = 0,
commentsPage = 0,
maxCommentPages = 0,
commentsAdditionalText = "",
sponsor_segments = {},
message_text = nil, -- TODO: needs to be removed
message_hide_timer = nil, -- TODO: needs to be removed
}
local logo_lines = {
-- White border
"{\\c&HE5E5E5&\\p6}m 895 10 b 401 10 0 410 0 905 0 1399 401 1800 895 1800 1390 1800 1790 1399 1790 905 1790 410 1390 10 895 10 {\\p0}",
-- Purple fill
"{\\c&H682167&\\p6}m 925 42 b 463 42 87 418 87 880 87 1343 463 1718 925 1718 1388 1718 1763 1343 1763 880 1763 418 1388 42 925 42{\\p0}",
-- Darker fill
"{\\c&H430142&\\p6}m 1605 828 b 1605 1175 1324 1456 977 1456 631 1456 349 1175 349 828 349 482 631 200 977 200 1324 200 1605 482 1605 828{\\p0}",
-- White fill
"{\\c&HDDDBDD&\\p6}m 1296 910 b 1296 1131 1117 1310 897 1310 676 1310 497 1131 497 910 497 689 676 511 897 511 1117 511 1296 689 1296 910{\\p0}",
-- Triangle
"{\\c&H691F69&\\p6}m 762 1113 l 762 708 b 881 776 1000 843 1119 911 1000 978 881 1046 762 1113{\\p0}",
}
local santa_hat_lines = {
-- Pompoms
"{\\c&HC0C0C0&\\p6}m 500 -323 b 491 -322 481 -318 475 -311 465 -312 456 -319 446 -318 434 -314 427 -304 417 -297 410 -290 404 -282 395 -278 390 -274 387 -267 381 -265 377 -261 379 -254 384 -253 397 -244 409 -232 425 -228 437 -228 446 -218 457 -217 462 -216 466 -213 468 -209 471 -205 477 -203 482 -206 491 -211 499 -217 508 -222 532 -235 556 -249 576 -267 584 -272 584 -284 578 -290 569 -305 550 -312 533 -309 523 -310 515 -316 507 -321 505 -323 503 -323 500 -323{\\p0}",
"{\\c&HE0E0E0&\\p6}m 315 -260 b 286 -258 259 -240 246 -215 235 -210 222 -215 211 -211 204 -188 177 -176 172 -151 170 -139 163 -128 154 -121 143 -103 141 -81 143 -60 139 -46 125 -34 129 -17 132 -1 134 16 142 30 145 56 161 80 181 96 196 114 210 133 231 144 266 153 303 138 328 115 373 79 401 28 423 -24 446 -73 465 -123 483 -174 487 -199 467 -225 442 -227 421 -232 402 -242 384 -254 364 -259 342 -250 322 -260 320 -260 317 -261 315 -260{\\p0}",
-- Main cap
"{\\c&H0000F0&\\p6}m 1151 -523 b 1016 -516 891 -458 769 -406 693 -369 624 -319 561 -262 526 -252 465 -235 479 -187 502 -147 551 -135 588 -111 1115 165 1379 232 1909 761 1926 800 1952 834 1987 858 2020 883 2053 912 2065 952 2088 1000 2146 962 2139 919 2162 836 2156 747 2143 662 2131 615 2116 567 2122 517 2120 410 2090 306 2089 199 2092 147 2071 99 2034 64 1987 5 1928 -41 1869 -86 1777 -157 1712 -256 1629 -337 1578 -389 1521 -436 1461 -476 1407 -509 1343 -507 1284 -515 1240 -519 1195 -521 1151 -523{\\p0}",
-- Cap shadow
"{\\c&H0000AA&\\p6}m 1657 248 b 1658 254 1659 261 1660 267 1669 276 1680 284 1689 293 1695 302 1700 311 1707 320 1716 325 1726 330 1735 335 1744 347 1752 360 1761 371 1753 352 1754 331 1753 311 1751 237 1751 163 1751 90 1752 64 1752 37 1767 14 1778 -3 1785 -24 1786 -45 1786 -60 1786 -77 1774 -87 1760 -96 1750 -78 1751 -65 1748 -37 1750 -8 1750 20 1734 78 1715 134 1699 192 1694 211 1689 231 1676 246 1671 251 1661 255 1657 248 m 1909 541 b 1914 542 1922 549 1917 539 1919 520 1921 502 1919 483 1918 458 1917 433 1915 407 1930 373 1942 338 1947 301 1952 270 1954 238 1951 207 1946 214 1947 229 1945 239 1939 278 1936 318 1924 356 1923 362 1913 382 1912 364 1906 301 1904 237 1891 175 1887 150 1892 126 1892 101 1892 68 1893 35 1888 2 1884 -9 1871 -20 1859 -14 1851 -6 1854 9 1854 20 1855 58 1864 95 1873 132 1883 179 1894 225 1899 273 1908 362 1910 451 1909 541{\\p0}",
-- Brim and tip pompom
"{\\c&HF8F8F8&\\p6}m 626 -191 b 565 -155 486 -196 428 -151 387 -115 327 -101 304 -47 273 2 267 59 249 113 219 157 217 213 215 265 217 309 260 302 285 283 373 264 465 264 555 257 608 252 655 292 709 287 759 294 816 276 863 298 903 340 972 324 1012 367 1061 394 1125 382 1167 424 1213 462 1268 482 1322 506 1385 546 1427 610 1479 662 1510 690 1534 725 1566 752 1611 796 1664 830 1703 880 1740 918 1747 986 1805 1005 1863 991 1897 932 1916 880 1914 823 1945 777 1961 725 1979 673 1957 622 1938 575 1912 534 1862 515 1836 473 1790 417 1755 351 1697 305 1658 266 1633 216 1593 176 1574 138 1539 116 1497 110 1448 101 1402 77 1371 37 1346 -16 1295 15 1254 6 1211 -27 1170 -62 1121 -86 1072 -104 1027 -128 976 -133 914 -130 851 -137 794 -162 740 -181 679 -168 626 -191 m 2051 917 b 1971 932 1929 1017 1919 1091 1912 1149 1923 1214 1970 1254 2000 1279 2027 1314 2066 1325 2139 1338 2212 1295 2254 1238 2281 1203 2287 1158 2282 1116 2292 1061 2273 1006 2229 970 2206 941 2167 938 2138 918{\\p0}",
}
--
-- Helper functions
--
local function kill_animation()
state.anistart = nil
state.animation = nil
state.anitype = nil
end
local function set_osd(res_x, res_y, text, z)
if state.osd.res_x == res_x and
state.osd.res_y == res_y and
state.osd.data == text then
return
end
state.osd.res_x = res_x
state.osd.res_y = res_y
state.osd.data = text
state.osd.z = z
state.osd:update()
end
-- scale factor for translating between real and virtual ASS coordinates
local function get_virt_scale_factor()
local w, h = mp.get_osd_size()
if w <= 0 or h <= 0 then
return 0, 0
end
return osc_param.playresx / w, osc_param.playresy / h
end
-- return mouse position in virtual ASS coordinates (playresx/y)
local function get_virt_mouse_pos()
if state.mouse_in_window then
local sx, sy = get_virt_scale_factor()
local x, y = mp.get_mouse_pos()
return x * sx, y * sy
else
return -1, -1
end
end
local function set_virt_mouse_area(x0, y0, x1, y1, name)
local sx, sy = get_virt_scale_factor()
mp.set_mouse_area(x0 / sx, y0 / sy, x1 / sx, y1 / sy, name)
end
local function scale_value(x0, x1, y0, y1, val)
local m = (y1 - y0) / (x1 - x0)
local b = y0 - (m * x0)
return (m * val) + b
end
-- returns hitbox spanning coordinates (top left, bottom right corner)
-- according to alignment
local function get_hitbox_coords(x, y, an, w, h)
local alignments = {
[1] = function () return x, y-h, x+w, y end,
[2] = function () return x-(w/2), y-h, x+(w/2), y end,
[3] = function () return x-w, y-h, x, y end,
[4] = function () return x, y-(h/2), x+w, y+(h/2) end,
[5] = function () return x-(w/2), y-(h/2), x+(w/2), y+(h/2) end,
[6] = function () return x-w, y-(h/2), x, y+(h/2) end,
[7] = function () return x, y, x+w, y+h end,
[8] = function () return x-(w/2), y, x+(w/2), y+h end,
[9] = function () return x-w, y, x, y+h end,
}
return alignments[an]()
end
local function get_hitbox_coords_geo(geometry)
return get_hitbox_coords(geometry.x, geometry.y, geometry.an,
geometry.w, geometry.h)
end
local function get_element_hitbox(element)
return element.hitbox.x1, element.hitbox.y1,
element.hitbox.x2, element.hitbox.y2
end
local function mouse_hit_coords(bX1, bY1, bX2, bY2)
local mX, mY = get_virt_mouse_pos()
return (mX >= bX1 and mX <= bX2 and mY >= bY1 and mY <= bY2)
end
local function mouse_hit(element)
return mouse_hit_coords(get_element_hitbox(element))
end
local function limit_range(min, max, val)
if val > max then
val = max
elseif val < min then
val = min
end
return val
end
-- translate value into element coordinates
local function get_slider_ele_pos_for(element, val)
local ele_pos = scale_value(
element.slider.min.value, element.slider.max.value,
element.slider.min.ele_pos, element.slider.max.ele_pos,
val)
return limit_range(
element.slider.min.ele_pos, element.slider.max.ele_pos,
ele_pos)
end
-- translates global (mouse) coordinates to value
local function get_slider_value_at(element, glob_pos)
if element then
local val = scale_value(
element.slider.min.glob_pos, element.slider.max.glob_pos,
element.slider.min.value, element.slider.max.value,
glob_pos)
return limit_range(
element.slider.min.value, element.slider.max.value,
val)
end
-- fall back incase of loading errors
return 0
end
-- get value at current mouse position
local function get_slider_value(element)
return get_slider_value_at(element, get_virt_mouse_pos())
end
-- multiplies two alpha values, formular can probably be improved
local function mult_alpha(alphaA, alphaB)
return 255 - (((1-(alphaA/255)) * (1-(alphaB/255))) * 255)
end
local function add_area(name, x1, y1, x2, y2)
-- create area if needed
if osc_param.areas[name] == nil then
osc_param.areas[name] = {}
end
table.insert(osc_param.areas[name], {x1=x1, y1=y1, x2=x2, y2=y2})
end
local function ass_append_alpha(ass, alpha, modifier, inverse)
local ar = {}
for ai, av in pairs(alpha) do
av = mult_alpha(av, modifier)
if state.animation then
local animpos = state.animation
if inverse then
animpos = 255 - animpos
end
av = mult_alpha(av, animpos)
end
ar[ai] = av
end
ass:append(string.format("{\\1a&H%X&\\2a&H%X&\\3a&H%X&\\4a&H%X&}",
ar[1], ar[2], ar[3], ar[4]))
end
local function ass_draw_cir_cw(ass, x, y, r)
ass:round_rect_cw(x-r, y-r, x+r, y+r, r)
end
local function ass_draw_rr_h_cw(ass, x0, y0, x1, y1, r1, hexagon, r2)
if hexagon then
ass:hexagon_cw(x0, y0, x1, y1, r1, r2)
else
ass:round_rect_cw(x0, y0, x1, y1, r1, r2)
end
end
local function get_hide_timeout()
if user_opts.visibility == "always" then
return -1 -- disable autohide
end
return user_opts.hide_timeout
end
-- Request that tick() is called (which typically re-renders the OSC).
-- The tick is then either executed immediately, or rate-limited if it was
-- called a small time ago.
local function request_tick()
if state.tick_timer == nil then
state.tick_timer = mp.add_timeout(0, tick)
end
if not state.tick_timer:is_enabled() then
local now = mp.get_time()
local timeout = tick_delay - (now - state.tick_last_time)
if timeout < 0 then
timeout = 0
end
state.tick_timer.timeout = timeout
state.tick_timer:resume()
end
end
local function request_init()
state.initREQ = true
request_tick()
end
-- Like request_init(), but also request an immediate update
local function request_init_resize()
request_init()
-- ensure immediate update
state.tick_timer:kill()
state.tick_timer.timeout = 0
state.tick_timer:resume()
end
local function render_wipe()
mp.msg.trace('render_wipe()')
state.osd.data = "" -- allows set_osd to immediately update on enable
state.osd:remove()
end
--
-- Tracklist Management
--
local nicetypes = {video = texts.video, audio = texts.audio, sub = texts.subtitle}
local tracks_osc, tracks_mpv
-- updates the OSC internal playlists, should be run each time the track-layout changes
function update_tracklist()
local tracktable = mp.get_property_native('track-list', {})
-- by osc_id
tracks_osc = {}
tracks_osc.video, tracks_osc.audio, tracks_osc.sub = {}, {}, {}
-- by mpv_id
tracks_mpv = {}
tracks_mpv.video, tracks_mpv.audio, tracks_mpv.sub = {}, {}, {}
for n = 1, #tracktable do
if not (tracktable[n].type == 'unknown') then
local type = tracktable[n].type
local mpv_id = tonumber(tracktable[n].id)
-- by osc_id
table.insert(tracks_osc[type], tracktable[n])
-- by mpv_id
tracks_mpv[type][mpv_id] = tracktable[n]
tracks_mpv[type][mpv_id].osc_id = #tracks_osc[type]
end
end
end
-- return a nice list of tracks of the given type (video, audio, sub)
function get_tracklist(type)
local message = nicetypes[type] .. texts.track
if not tracks_osc or #tracks_osc[type] == 0 then
message = texts.none
else
for n = 1, #tracks_osc[type] do
local track = tracks_osc[type][n]
local lang, title, selected = 'unknown', '', '○'
if not(track.lang == nil) then lang = track.lang end
if not(track.title == nil) then title = track.title end
if (track.id == tonumber(mp.get_property(type))) then
selected = '●'
end
message = message..'\n'..selected..' '..n..': ['..lang..'] '..title
end
end
return message
end
-- relatively change the track of given <type> by <next> tracks
--(+1 -> next, -1 -> previous)
function set_track(type, next)
local current_track_mpv, current_track_osc
current_track_osc = 0
if (mp.get_property(type) == 'no') then
current_track_osc = 0
else
current_track_mpv = tonumber(mp.get_property(type))
if (tracks_mpv[type][current_track_mpv]) then
current_track_osc = tracks_mpv[type][current_track_mpv].osc_id
end
end
local new_track_osc = (current_track_osc + next) % (#tracks_osc[type] + 1)
local new_track_mpv
if new_track_osc == 0 then
new_track_mpv = 'no'
else
new_track_mpv = tracks_osc[type][new_track_osc].id
end
mp.commandv('set', type, new_track_mpv)
end
-- get the currently selected track of <type>, OSC-style counted
function get_track(type)
local track = mp.get_property(type)
if track ~= 'no' and track ~= nil then
local tr = tracks_mpv[type][tonumber(track)]
if tr then
return tr.osc_id
end
end
return 0
end
-- convert slider_pos to logarithmic depending on volume_control user_opts
local function set_volume(slider_pos)
local volume = slider_pos
if user_opts.volume_control_type == "logarithmic" then
volume = slider_pos^2 / 100
end
return math.floor(volume)
end
-- WindowControl helpers
function window_controls_enabled()
local val = user_opts.window_top_bar
if val == 'auto' then
return (not state.border) or (not state.title_bar) or state.fullscreen
else
return val ~= 'no'
end
end
--
-- Element Management
--
local elements = {}
local function prepare_elements()
-- remove elements without layout or invisible
local elements2 = {}
for _, element in pairs(elements) do
if element.layout ~= nil and element.visible then
table.insert(elements2, element)
end
end
elements = elements2
local function elem_compare (a, b)
return a.layout.layer < b.layout.layer
end
table.sort(elements, elem_compare)
for _,element in pairs(elements) do
local elem_geo = element.layout.geometry
-- Calculate the hitbox
local bX1, bY1, bX2, bY2 = get_hitbox_coords_geo(elem_geo)
element.hitbox = {x1 = bX1, y1 = bY1, x2 = bX2, y2 = bY2}
local style_ass = mp.assdraw.ass_new()
-- prepare static elements
style_ass:append("{}") -- hack to troll new_event into inserting a \n
style_ass:new_event()
style_ass:pos(elem_geo.x, elem_geo.y)
style_ass:an(elem_geo.an)
style_ass:append(element.layout.style)
element.style_ass = style_ass
local static_ass = mp.assdraw.ass_new()
if element.type == "box" then
--draw box
static_ass:draw_start()
ass_draw_rr_h_cw(static_ass, 0, 0, elem_geo.w, elem_geo.h,
element.layout.box.radius, element.layout.box.hexagon)
static_ass:draw_stop()
elseif element.type == "slider" then
--draw static slider parts
local slider_lo = element.layout.slider
-- calculate positions of min and max points
element.slider.min.ele_pos = user_opts.seek_handle_size * elem_geo.h / 2
element.slider.max.ele_pos = elem_geo.w - element.slider.min.ele_pos
element.slider.min.glob_pos = element.hitbox.x1 + element.slider.min.ele_pos
element.slider.max.glob_pos = element.hitbox.x1 + element.slider.max.ele_pos
static_ass:draw_start()
-- a hack which prepares the whole slider area to allow center placements such like an=5
static_ass:rect_cw(0, 0, elem_geo.w, elem_geo.h)
static_ass:rect_ccw(0, 0, elem_geo.w, elem_geo.h)
-- chapter marker nibbles
if user_opts.show_chapter_markers and element.slider.markerF ~= nil and slider_lo.gap > 0 then
local markers = element.slider.markerF()
for _, marker in pairs(markers) do
if marker >= element.slider.min.value and marker <= element.slider.max.value then
local s = get_slider_ele_pos_for(element, marker)
if slider_lo.gap > 5 then -- draw triangles
--top
if slider_lo.nibbles_top then
static_ass:move_to(s - 3, slider_lo.gap - 5)
static_ass:line_to(s + 3, slider_lo.gap - 5)
static_ass:line_to(s, slider_lo.gap - 1)
end
--bottom
if slider_lo.nibbles_bottom then
static_ass:move_to(s - 3, elem_geo.h - slider_lo.gap + 5)
static_ass:line_to(s, elem_geo.h - slider_lo.gap + 1)
static_ass:line_to(s + 3, elem_geo.h - slider_lo.gap + 5)
end
else -- draw 2x1px nibbles
--top
if slider_lo.nibbles_top then
static_ass:rect_cw(s - 1, 0, s + 1, slider_lo.gap);
end
--bottom
if slider_lo.nibbles_bottom then
static_ass:rect_cw(s - 1, elem_geo.h - slider_lo.gap, s + 1, elem_geo.h);
end
end
end
end
end
end
element.static_ass = static_ass
-- if the element is supposed to be disabled,
-- style it accordingly and kill the eventresponders
if not element.enabled then
element.layout.alpha[1] = 215
if not (element.name == "sub_track" or element.name == "audio_track" or element.name == "tog_playlist") then -- keep these to display tooltips
element.eventresponder = nil
end
end
-- gray out the element if it is toggled off
if element.off then
element.layout.alpha[1] = 100
end
end
end
--
-- Element Rendering
--
-- returns nil or a chapter element from the native property chapter-list
function get_chapter(possec)
local cl = state.chapter_list -- sorted, get latest before possec, if any
for n=#cl,1,-1 do
if possec >= cl[n].time then
return cl[n]
end
end
end
local function draw_seekbar_handle(element, elem_ass, override_alpha)
local pos = element.slider.posF()
if not pos then
return 0, 0
end
local display_handle = user_opts.seek_handle_size > 0
local elem_geo = element.layout.geometry
local rh = display_handle and (user_opts.seek_handle_size * elem_geo.h / 2) or 0 -- handle radius