forked from Samillion/ModernZ
-
Notifications
You must be signed in to change notification settings - Fork 0
/
modernz.lua
3208 lines (2812 loc) · 128 KB
/
modernz.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
-- Samillion/ModernZ: https://github.com/Samillion/ModernZ
--- forked from zydezu/ModernX: https://github.com/zydezu/ModernX
---- forked from dexeonify: https://github.com/dexeonify/mpv-config/blob/main/scripts/modernx.lua
----- forked from cyl0: https://github.com/cyl0/ModernX
------ forked from maoiscat: https://github.com/maoiscat/mpv-osc-modern
local assdraw = require "mp.assdraw"
local msg = require "mp.msg"
local opt = require "mp.options"
local utils = require "mp.utils"
-- Parameters
-- default user option values
-- do not touch, change them in modernz.conf
local user_opts = {
-- General
language = "en", -- See https://github.com/Samillion/ModernZ/blob/main/docs/TRANSLATIONS.md for other languages
idlescreen = true, -- show mpv logo on idle
windowcontrols = "auto", -- whether to show OSC window controls, "auto", "yes" or "no"
showwindowed = true, -- show OSC when windowed?
showfullscreen = true, -- show OSC when fullscreen?
greenandgrumpy = false, -- disable santa hat in December
-- Colors
osc_color = "#000000", -- accent of the OSC and the title bar
window_title_color = "#FFFFFF", -- color of title in borderless/fullscreen mode
window_controls_color = "#FFFFFF", -- color of window controls (close, min, max) in borderless/fullscreen mode
title_color = "#FFFFFF", -- color of the title (above seekbar)
seekbarfg_color = "#BE4D25", -- color of the seekbar progress and handle
seekbarbg_color = "#FFFFFF", -- color of the remaining seekbar
vol_bar_match_seek = false, -- match volume bar color with seekbar color? ignores side_buttons_color
time_color = "#FFFFFF", -- color of timestamps (below seekbar)
chapter_title_color = "#FFFFFF", -- color of chapter title next to timestamp (below seekbar)
side_buttons_color = "#FFFFFF", -- color of side buttons (audio, sub, playlist, vol, loop, info..etc)
middle_buttons_color = "#FFFFFF", -- color of middle buttons (skip, jump, chapter...etc)
playpause_color = "#FFFFFF", -- color of play/pause button
held_element_color = "#999999", -- color of an element while held down
thumbnailborder_color = "#111111", -- color of border for thumbnail (with thumbfast)
hovereffect_color = "#CB7050", -- color of a hovered button when hovereffect is: color
-- Buttons
hovereffect = "size,glow,color", -- list of active button hover effects seperated by comma: glow, size, color
hover_button_size = 115, -- the relative size of a hovered button if the size effect is active
button_glow_amount = 5, -- the amount of glow a hovered button receives if the glow effect is active
hovereffect_for_sliders = true, -- apply button hovereffects to slide handles
showjump = true, -- show "jump forward/backward 10 seconds" buttons
showskip = false, -- show the chapter skip back and forward buttons
shownextprev = true, -- show the next/previous playlist track buttons
showplaylist = false, -- show playlist button? LClick: simple playlist, RClick: interactive playlist
hide_empty_playlist_button = true, -- hide playlist button when no playlist exists
gray_empty_playlist_button = true, -- grays out the playlist button when no playlist exists
showinfo = false, -- show the info button
showloop = true, -- show the loop button
showfullscreen_button = true, -- show fullscreen toggle button
showontop = true, -- show window on top button
showscreenshot = false, -- show screenshot button
screenshot_flag = "subtitles", -- flag for the screenshot button. subtitles, video, window, each-frame
-- https://mpv.io/manual/master/#command-interface-screenshot-%3Cflags%3E
chapter_softrepeat = true, -- holding chapter skip buttons repeats toggle
jump_softrepeat = true, -- holding jump seek buttons repeats toggle
downloadbutton = true, -- show download button on web videos (requires yt-dlp and ffmpeg)
download_path = "~~desktop/mpv", -- the download path for videos https://mpv.io/manual/master/#paths
-- Scaling
vidscale = "auto", -- whether to scale the controller with the video
scalewindowed = 1.0, -- scaling of the controller when windowed
scalefullscreen = 1.0, -- scaling of the controller when fullscreen
-- Time & Volume
unicodeminus = false, -- whether to use the Unicode minus sign character in remaining time
timetotal = true, -- display total time instead of remaining time?
timems = false, -- display timecodes with milliseconds?
time_format = "dynamic", -- "dynamic" or "fixed". dynamic shows MM:SS when possible, fixed always shows HH:MM:SS
timefontsize = 18, -- the font size of the time
jumpamount = 10, -- change the jump amount (in seconds by default)
jumpiconnumber = true, -- show different icon when jumpamount is 5, 10, or 30
jumpmode = "relative", -- seek mode for jump buttons
volumecontrol = true, -- whether to show mute button and volume slider
volumecontroltype = "linear", -- use "linear" or "log" (logarithmic) volume scale
-- Seeking
seekbarkeyframes = false, -- use keyframes when dragging the seekbar
seekbarhandlesize = 0.8, -- size ratio of the slider handle, range 0 ~ 1
seekrange = true, -- show seekrange overlay
seekrangealpha = 150, -- transparency of seekranges
livemarkers = true, -- update seekbar chapter markers on duration change
osc_on_seek = false, -- show osc when seeking? or input.conf: x script-message-to modernz osc-show
mouse_seek_pause = true, -- should the video pause while seeking with mouse move? (on button hold)
automatickeyframemode = true, -- set seekbarkeyframes based on video length to prevent laggy scrubbing on long videos
automatickeyframelimit = 600, -- videos of above this length (in seconds) will have seekbarkeyframes on
-- UI [elements]
showtitle = true, -- show title in OSC (above seekbar)
showwindowtitle = true, -- show window title in borderless/fullscreen mode
showwindowcontrols = true, -- show window controls (close, min, max) in borderless/fullscreen
show_chapter_title = true, -- show chapter title next to timestamp (below seekbar)
titleBarStrip = false, -- whether to make the title bar a singular bar instead of a black fade
title = "${media-title}", -- title above seekbar
windowcontrols_title = "${media-title}", -- title in windowcontrols
font = "mpv-osd-symbols", -- mpv-osd-symbols = default osc font (or the one set in mpv.conf)
titlefontsize = 30, -- the font size of the title text (above seekbar)
chapter_fmt = "%s", -- chapter print format for seekbar-hover. "no" to disable
tooltips_for_disabled_elements = true, -- enables tooltips for disabled buttons and elements
tooltip_hints = true, -- enables text hints for the information, loop, ontop and screenshot buttons
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
persistentprogress = false, -- always show a small progress line at the bottom of the screen
persistentprogressheight = 17, -- the height of the persistentprogress bar
persistentbuffer = false, -- on web videos, show the buffer on the persistent progress line
-- UI [behavior]
showonpause = true, -- whether to show osc when paused
keeponpause = true, -- whether to disable the hide timeout on pause
bottomhover = true, -- if the osc should only display when hovering at the bottom
bottomhover_zone = 160, -- height of show/hide zone for bottomhover
raisesubs = true, -- whether to raise subtitles above the osc when it's shown
raisesubamount = 175, -- how much subtitles rise when the osc is shown
thumbnailborder = 2, -- the width of the thumbnail border (thumbfast)
OSCfadealpha = 150, -- alpha of the background box for the OSC
boxalpha = 75, -- alpha of the window title bar
loopinpause = true, -- activate looping by right clicking pause
visibility = "auto", -- only used at init to set visibility_mode(...)
-- UI [time-based]
hidetimeout = 1500, -- duration in ms until OSC hides if no mouse movement
fadeduration = 250, -- duration of fade out in ms, 0 = no fade
minmousemove = 0, -- amount of pixels the mouse has to move for OSC to show
tick_delay = 1 / 60, -- minimum interval between OSC redraws in seconds
tick_delay_follow_display_fps = false, -- use display fps as the minimum interval
-- Mouse commands
-- customize the button function based on mouse action
-- seekbar mouse wheel mode. accepts: "speed" or "seek". speed adjusts playback speed.
seekbar_track_wheel_mode = "seek",
-- title above seekbar mouse actions
title_mbtn_left_command = "script-binding select/select-playlist; script-message-to modernz osc-hide",
title_mbtn_right_command = "script-binding stats/display-page-5",
-- playlist button mouse actions
playlist_mbtn_left_command = "script-binding select/select-playlist; script-message-to modernz osc-hide",
playlist_mbtn_right_command = "show-text ${playlist} 3000",
-- volume mouse actions
vol_ctrl_mbtn_right_command = "script-binding select/select-audio-device; script-message-to modernz osc-hide",
-- audio button mouse actions
audio_track_mbtn_left_command = "script-binding select/select-aid; script-message-to modernz osc-hide",
audio_track_mbtn_right_command = "osd-msg cycle audio",
audio_track_wheel_down_command = "osd-msg cycle audio",
audio_track_wheel_up_command = "osd-msg cycle audio down",
-- subtitle button mouse actions
sub_track_mbtn_left_command = "script-binding select/select-sid; script-message-to modernz osc-hide",
sub_track_mbtn_right_command = "osd-msg cycle sub",
sub_track_wheel_down_command = "osd-msg cycle sub",
sub_track_wheel_up_command = "osd-msg cycle sub down",
-- chapter skip buttons mouse actions
chapter_prev_mbtn_left_command = "osd-msg add chapter -1",
chapter_prev_mbtn_right_command = "script-binding select/select-chapter; script-message-to modernz osc-hide",
chapter_next_mbtn_left_command = "osd-msg add chapter 1",
chapter_next_mbtn_right_command = "script-binding select/select-chapter; script-message-to modernz osc-hide",
-- chapter title (below seekbar) mouse actions
chapter_title_mbtn_left_command = "script-binding select/select-chapter; script-message-to modernz osc-hide",
chapter_title_mbtn_right_command = "show-text ${chapter-list} 3000",
}
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",
playlist = "\238\161\159",
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",
screenshot = "\238\169\150",
ontop_on = "\238\165\190",
ontop_off = "\238\166\129",
loop_off = "\239\133\178",
loop_on = "\239\133\181",
info = "\239\146\164",
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()
}
}
--- Localization
local language = {
["en"] = {
welcome = "{\\fs24\\1c&H0&\\1c&HFFFFFF&}Drop files or URLs to play here",
off = "OFF",
na = "n/a",
none = "None available",
video = "Video",
audio = "Audio",
subtitle = "Subtitle",
nosub = "No subtitles available",
noaudio = "No audio tracks available",
track = " tracks:",
playlist = "Playlist",
nolist = "Empty playlist.",
chapter = "Chapter",
nochapter = "No chapters.",
ontop = "Pin window",
ontopdisable = "Unpin window",
loopenable = "Enable loop",
loopdisable = "Disable loop",
screenshot = "Screenshot",
screenshotsaved = "Screenshot saved",
statsinfo = "Information",
},
}
mp.observe_property("osc", "bool", function(name, value) if value == true then mp.set_property("osc", "no") end end)
-- Load external locales if available
local locale_file = mp.find_config_file("scripts/modernz-locale.lua")
if locale_file then
local success, external = pcall(function()
return loadfile(locale_file)()
end)
if success and external then
-- Merge external locales
for lang, strings in pairs(external) do
language[lang] = strings
-- Fill in any missing locales with English
for key, value in pairs(language["en"]) do
if strings[key] == nil then
strings[key] = value
end
end
end
end
end
local texts
local function set_osc_texts()
texts = language[user_opts.language] or language["en"]
end
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 thumbfast = {
width = 0,
height = 0,
disabled = true,
available = false
}
local tick_delay = 1 / 60
local audio_track_count = 0
local sub_track_count = 0
local window_control_box_width = 138
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 osc_styles
local function set_osc_styles()
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
osc_styles = {
background_bar = "{\\1c&H" .. osc_color_convert(user_opts.osc_color) .. "&}",
box_bg = "{\\blur100\\bord" .. user_opts.OSCfadealpha .. "\\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.timefontsize .. "\\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.hovereffect, "color") and "\\1c&H" .. osc_color_convert(user_opts.hovereffect_color) .. "&" or "") .."\\2c&HFFFFFF&" .. (contains(user_opts.hovereffect, "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 = "{\\blur1\\bord0.5\\1c&H" .. osc_color_convert(user_opts.thumbnailborder_color) .. "&\\3c&H000000&}",
time = "{\\blur0\\bord0\\1c&H" .. osc_color_convert(user_opts.time_color) .. "&\\3c&H000000&\\fs" .. user_opts.timefontsize .. "\\fn" .. user_opts.font .. "}",
title = "{\\blur1\\bord0.5\\1c&H" .. osc_color_convert(user_opts.title_color) .. "&\\3c&H0&\\fs".. user_opts.titlefontsize .."\\q2\\fn" .. user_opts.font .. "}",
tooltip = "{\\blur1\\bord0.5\\1c&HFFFFFF&\\3c&H000000&\\fs" .. user_opts.timefontsize .. "\\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&\\fs25\\fnmpv-osd-symbols}",
window_title = "{\\blur1\\bord0.5\\1c&H" .. osc_color_convert(user_opts.window_title_color) .. "&\\3c&H0&\\fs30\\q2\\fn" .. user_opts.font .. "}",
}
end
-- internal states, do not touch
local state = {
showtime = nil, -- time of last invocation (last mouse move)
touchtime = nil, -- time of last invocation (last touch event)
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.timetotal, -- if the right timecode should display total or remaining time
tc_ms = user_opts.timems, -- Should the timecodes display their time with milliseconds
screen_sizeX = nil, screen_sizeY = nil, -- last screen-resolution, to detect resolution changes to issue reINITs
initREQ = false, -- is a re-init request pending?
marginsREQ = false, -- is a margins update 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,
idle = false,
enabled = true,
input_enabled = true,
showhide_enabled = false,
windowcontrols_buttons = false,
windowcontrols_title = false,
border = true,
maximized = false,
osd = mp.create_osd_overlay("ass-events"),
chapter_list = {}, -- sorted by time
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,
persistentprogresstoggle = user_opts.persistentprogress,
downloadedOnce = false,
downloading = false,
fileSizeBytes = 0,
fileSizeNormalised = "Approximating size...",
isWebVideo = false,
web_video_path = "", -- used for yt-dlp downloading
videoCantBeDownloaded = false,
}
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
local function set_time_styles(timetotal_changed, timems_changed)
if timetotal_changed then
state.tc_right_rem = not user_opts.timetotal
end
if timems_changed then
state.tc_ms = user_opts.timems
end
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_hidetimeout()
if user_opts.visibility == "always" then
return -1 -- disable autohide
end
return user_opts.hidetimeout
end
local function get_touchtimeout()
if state.touchtime == nil then
return 0
end
return state.touchtime + (get_hidetimeout() / 1000) - mp.get_time()
end
local tick
-- 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()
msg.trace("render_wipe()")
state.osd.data = "" -- allows set_osd to immediately update on enable
state.osd:remove()
end
--
-- Tracklist Management
--
-- updates the OSC internal playlists, should be run each time the track-layout changes
local function update_tracklist()
audio_track_count, sub_track_count = 0, 0
for _, track in pairs(mp.get_property_native("track-list")) do
if track.type == "audio" then
audio_track_count = audio_track_count + 1
elseif track.type == "sub" then
sub_track_count = sub_track_count + 1
end
end
end
-- convert slider_pos to logarithmic depending on volumecontrol user_opts
local function set_volume(slider_pos)
local volume = slider_pos
if user_opts.volumecontroltype == "log" then
volume = slider_pos^2 / 100
end
return math.floor(volume)
end
-- WindowControl helpers
local function window_controls_enabled()
local val = user_opts.windowcontrols
if val == "auto" then
return not (state.border and 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 = 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 = 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.seekbarhandlesize * 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)
-- marker nibbles
if 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
local 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 render_elements(master_ass)
-- when the slider is dragged or hovered and we have a target chapter name
-- then we use it instead of the normal title. we calculate it before the
-- render iterations because the title may be rendered before the slider.
state.forced_title = nil
-- disable displaying chapter name in title when thumbfast is available
-- because thumbfast will render it above the thumbnail instead
if thumbfast.disabled then
local se, ae = state.slider_element, elements[state.active_element]
if user_opts.chapter_fmt ~= "no" and state.touchingprogressbar then
local dur = mp.get_property_number("duration", 0)
if dur > 0 then
local ch = get_chapter(state.sliderpos * dur / 100)
if ch and ch.title and ch.title ~= "" then
state.forced_title = string.format(user_opts.chapter_fmt, ch.title)
end
end
end
end
state.touchingprogressbar = false
for n=1, #elements do
local element = elements[n]
local style_ass = assdraw.ass_new()
style_ass:merge(element.style_ass)
ass_append_alpha(style_ass, element.layout.alpha, 0)
if element.eventresponder and (state.active_element == n) then
-- run render event functions
if element.eventresponder.render ~= nil then
element.eventresponder.render(element)
end
if mouse_hit(element) then
-- mouse down styling
if element.styledown then
style_ass:append(osc_styles.element_down)
end
if element.softrepeat and state.mouse_down_counter >= 15
and state.mouse_down_counter % 5 == 0 then
element.eventresponder[state.active_event_source.."_down"](element)
end
state.mouse_down_counter = state.mouse_down_counter + 1
end
end
local elem_ass = assdraw.ass_new()
elem_ass:merge(style_ass)
if element.type ~= "button" then
elem_ass:merge(element.static_ass)
end
if element.type == "slider" then
if element.name ~= "persistentseekbar" then
local slider_lo = element.layout.slider
local elem_geo = element.layout.geometry
local s_min = element.slider.min.value
local s_max = element.slider.max.value
-- draw pos marker
local pos = element.slider.posF()
local seekRanges = element.slider.seekRangesF()
local rh = user_opts.seekbarhandlesize * elem_geo.h / 2 -- Handle radius
local xp
if pos then
xp = get_slider_ele_pos_for(element, pos)
local handle_hovered = mouse_hit_coords(element.hitbox.x1+xp-rh, element.hitbox.y1+elem_geo.h/2-rh, element.hitbox.x1+xp+rh, element.hitbox.y1+elem_geo.h/2+rh)
if handle_hovered and user_opts.hovereffect_for_sliders then
-- apply size & color hovereffects (glow is not supported)
if contains(user_opts.hovereffect, "size") then
rh = rh*(user_opts.hover_button_size/100)
end
if contains(user_opts.hovereffect, "color") then
elem_ass.text = elem_ass.text:gsub(element.layout.style, element.layout.slider.hoverstyle)
end
end
ass_draw_cir_cw(elem_ass, xp, elem_geo.h/2, rh)
if handle_hovered and user_opts.hovereffect_for_sliders then
elem_ass:draw_stop()
elem_ass:merge(element.style_ass)
ass_append_alpha(elem_ass, element.layout.alpha, 0)
elem_ass:merge(element.static_ass)
end
elem_ass:rect_cw(0, slider_lo.gap, xp-rh, elem_geo.h - slider_lo.gap)
end
if seekRanges then
elem_ass:draw_stop()
elem_ass:merge(element.style_ass)
ass_append_alpha(elem_ass, element.layout.alpha, user_opts.seekrangealpha)
elem_ass:merge(element.static_ass)
for _,range in pairs(seekRanges) do
local pstart = get_slider_ele_pos_for(element, range["start"])
local pend = get_slider_ele_pos_for(element, range["end"])
elem_ass:rect_cw(pstart - rh, slider_lo.gap, pend + rh, elem_geo.h - slider_lo.gap)
end
end
elem_ass:draw_stop()
-- add tooltip
if element.slider.tooltipF ~= nil then
if mouse_hit(element) then
local sliderpos = get_slider_value(element)
local tooltiplabel = element.slider.tooltipF(sliderpos)
local an = slider_lo.tooltip_an
local ty
if an == 2 then
ty = element.hitbox.y1
else
ty = element.hitbox.y1 + elem_geo.h/2
end
local tx = get_virt_mouse_pos()
if slider_lo.adjust_tooltip then
if an == 2 then
if sliderpos < (s_min + 3) then
an = an - 1
elseif sliderpos > (s_max - 3) then
an = an + 1
end
elseif (sliderpos > (s_max+s_min)/2) then
an = an + 1
tx = tx - 5
else
an = an - 1
tx = tx + 10
end
end
if element.name == "seekbar" then
state.sliderpos = sliderpos
end
-- thumbfast
if element.thumbnailable and not thumbfast.disabled then
local osd_w = mp.get_property_number("osd-width")
local r_w, r_h = get_virt_scale_factor()
if osd_w then
local hover_sec = 0
if mp.get_property_number("duration") then hover_sec = mp.get_property_number("duration") * sliderpos / 100 end
local thumbPad = user_opts.thumbnailborder
local thumbMarginX = 18 / r_w
local thumbMarginY = user_opts.timefontsize + thumbPad + 2 / r_h
local thumbX = math.min(osd_w - thumbfast.width - thumbMarginX, math.max(thumbMarginX, tx / r_w - thumbfast.width / 2))
local thumbY = (ty - thumbMarginY) / r_h - thumbfast.height
thumbX = math.floor(thumbX + 0.5)
thumbY = math.floor(thumbY + 0.5)
if state.anitype == nil then
elem_ass:new_event()
elem_ass:pos(thumbX * r_w, ty - thumbMarginY - thumbfast.height * r_h)
elem_ass:an(7)
elem_ass:append(osc_styles.thumbnail)
elem_ass:draw_start()
elem_ass:rect_cw(-thumbPad * r_w, -thumbPad * r_h, (thumbfast.width + thumbPad) * r_w, (thumbfast.height + thumbPad) * r_h)
elem_ass:draw_stop()
-- force tooltip to be centered on the thumb, even at far left/right of screen
tx = (thumbX + thumbfast.width / 2) * r_w
an = 2
mp.commandv("script-message-to", "thumbfast", "thumb", hover_sec, thumbX, thumbY)