-
Notifications
You must be signed in to change notification settings - Fork 132
/
Copy pathmain.qml
694 lines (607 loc) · 24.2 KB
/
main.qml
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
import QtQuick 2.7
import QtWebEngine 1.4
import QtWebChannel 1.0
import QtQuick.Window 2.2 // for Window instead of ApplicationWindow; also for Screen
import QtQuick.Controls 1.4 // for ApplicationWindow
import QtQuick.Dialogs 1.2
import com.stremio.process 1.0
import com.stremio.screensaver 1.0
import com.stremio.libmpv 1.0
import com.stremio.clipboard 1.0
import QtQml 2.2
import "autoupdater.js" as Autoupdater
ApplicationWindow {
id: root
visible: true
minimumWidth: 1000
minimumHeight: 650
readonly property int initialWidth: Math.max(root.minimumWidth, Math.min(1600, Screen.desktopAvailableWidth * 0.8))
readonly property int initialHeight: Math.max(root.minimumHeight, Math.min(1000, Screen.desktopAvailableHeight * 0.8))
width: root.initialWidth
height: root.initialHeight
property bool quitting: false
color: "#0c0b11";
title: appTitle
property var previousVisibility: Window.Windowed
property bool wasFullScreen: false
function setFullScreen(fullscreen) {
if (fullscreen) {
root.visibility = Window.FullScreen;
root.wasFullScreen = true;
} else {
root.visibility = root.previousVisibility;
root.wasFullScreen = false;
}
}
function showWindow() {
if (root.wasFullScreen) {
root.visibility = Window.FullScreen;
} else {
root.visibility = root.previousVisibility;
}
root.raise();
root.requestActivate();
}
function updatePreviousVisibility() {
if (root.visible && root.visibility != Window.FullScreen && root.visibility != Window.Minimized) {
root.previousVisibility = root.visibility;
}
}
// Transport
QtObject {
id: transport
readonly property string shellVersion: Qt.application.version
property string serverAddress: "http://127.0.0.1:11470" // will be set to something else if server inits on another port
readonly property bool isFullscreen: root.visibility === Window.FullScreen // just to send the initial state
signal event(var ev, var args)
function onEvent(ev, args) {
if (ev === "quit") quitApp()
if (ev === "app-ready") transport.flushQueue()
if (ev === "mpv-command" && args && args[0] !== "run") mpv.command(args)
if (ev === "mpv-set-prop") {
mpv.setProperty(args[0], args[1]);
if (args[0] === "pause") {
shouldDisableScreensaver(!args[1]);
}
}
if (ev === "mpv-observe-prop") mpv.observeProperty(args)
if (ev === "control-event") wakeupEvent()
if (ev === "wakeup") wakeupEvent()
if (ev === "set-window-mode") onWindowMode(args)
if (ev === "open-external") Qt.openUrlExternally(args)
if (ev === "win-focus" && !root.visible) {
showWindow();
}
if (ev === "win-set-visibility") {
if (args.hasOwnProperty('fullscreen')) {
setFullScreen(args.fullscreen);
}
}
if (ev === "autoupdater-notif-clicked" && autoUpdater.onNotifClicked) {
autoUpdater.onNotifClicked();
}
//if (ev === "chroma-toggle") { args.enabled ? chroma.enable() : chroma.disable() }
if (ev === "screensaver-toggle") shouldDisableScreensaver(args.disabled)
if (ev === "file-close") fileDialog.close()
if (ev === "file-open") {
if (typeof args !== "undefined") {
var fileDialogDefaults = {
title: "Please choose",
selectExisting: true,
selectFolder: false,
selectMultiple: false,
nameFilters: [],
selectedNameFilter: "",
data: null
}
Object.keys(fileDialogDefaults).forEach(function(key) {
fileDialog[key] = args.hasOwnProperty(key) ? args[key] : fileDialogDefaults[key]
})
}
fileDialog.open()
}
}
// events that we want to wait for the app to initialize
property variant queued: []
function queueEvent() {
if (transport.queued) transport.queued.push(arguments)
else transport.event.apply(transport, arguments)
}
function flushQueue() {
if (transport.queued) transport.queued.forEach(function(args) { transport.event.apply(transport, args) })
transport.queued = null;
}
}
// Utilities
function onWindowMode(mode) {
shouldDisableScreensaver(mode === "player")
}
function wakeupEvent() {
shouldDisableScreensaver(true)
timerScreensaver.restart()
}
function shouldDisableScreensaver(condition) {
if (condition === screenSaver.disabled) return;
condition ? screenSaver.disable() : screenSaver.enable();
screenSaver.disabled = condition;
}
function isPlayerPlaying() {
return root.visible && typeof(mpv.getProperty("path"))==="string" && !mpv.getProperty("pause")
}
// Received external message
function onAppMessageReceived(instance, message) {
message = message.toString(); // cause it may be QUrl
showWindow();
if (message !== "SHOW") {
onAppOpenMedia(message);
}
}
// May be called from a message (from another app instance) or when app is initialized with an arg
function onAppOpenMedia(message) {
var url = (message.indexOf('://') > -1 || message.indexOf('magnet:') === 0) ? message : 'file://'+message;
transport.queueEvent("open-media", url)
}
function quitApp() {
root.quitting = true;
webView.destroy();
systemTray.hideIconTray();
streamingServer.kill();
streamingServer.waitForFinished(1500);
Qt.quit();
}
/* With help Connections object
* set connections with System tray class
* */
Connections {
target: systemTray
function onSignalIconMenuAboutToShow() {
systemTray.updateIsOnTop((root.flags & Qt.WindowStaysOnTopHint) === Qt.WindowStaysOnTopHint);
systemTray.updateVisibleAction(root.visible);
}
function onSignalShow() {
if(root.visible) {
root.hide();
} else {
showWindow();
}
}
function onSignalAlwaysOnTop() {
root.raise()
if (root.flags & Qt.WindowStaysOnTopHint) {
root.flags &= ~Qt.WindowStaysOnTopHint;
} else {
root.flags |= Qt.WindowStaysOnTopHint;
}
}
// The signal - close the application by ignoring the check-box
function onSignalQuit() {
quitApp();
}
// Minimize / maximize the window by clicking on the default system tray
function onSignalIconActivated() {
showWindow();
}
}
// Screen saver - enable & disable
ScreenSaver {
id: screenSaver
property bool disabled: false // track last state so we don't call it multiple times
}
// This is needed so that 300s after the remote control has been used, we can re-enable the screensaver
// (if the player is not playing)
Timer {
id: timerScreensaver
interval: 300000
running: false
onTriggered: function () { shouldDisableScreensaver(isPlayerPlaying()) }
}
// Clipboard proxy
Clipboard {
id: clipboard
}
//
// Streaming server
//
Process {
id: streamingServer
property string errMessage:
"Error while starting streaming server. Please try to restart stremio. If it happens again please contact the Stremio support team for assistance"
property int errors: 0
property bool fastReload: false
onStarted: function() { stayAliveStreamingServer.stop() }
onFinished: function(code, status) {
// status -> QProcess::CrashExit is 1
if (!streamingServer.fastReload && errors < 5 && (code !== 0 || status !== 0) && !root.quitting) {
transport.queueEvent("server-crash", {"code": code, "log": streamingServer.getErrBuff()});
errors++
showStreamingServerErr(code)
}
if (streamingServer.fastReload) {
console.log("Streaming server: performing fast re-load")
streamingServer.fastReload = false
root.launchServer()
} else {
stayAliveStreamingServer.start()
}
}
onAddressReady: function (address) {
transport.serverAddress = address
transport.event("server-address", address)
}
onErrorThrown: function (error) {
if (root.quitting) return; // inhibits errors during quitting
if (streamingServer.fastReload && error == 1) return; // inhibit errors during fast reload mode;
// we'll unset that after we've restarted the server
transport.queueEvent("server-crash", {"code": error, "log": streamingServer.getErrBuff()});
showStreamingServerErr(error)
}
}
function showStreamingServerErr(code) {
errorDialog.text = streamingServer.errMessage
errorDialog.detailedText = 'Stremio streaming server has thrown an error \nQProcess::ProcessError code: '
+ code + '\n\n'
+ streamingServer.getErrBuff();
errorDialog.visible = true
}
function launchServer() {
var node_executable = applicationDirPath + "/node"
if (Qt.platform.os === "windows") node_executable = applicationDirPath + "/stremio-runtime.exe"
streamingServer.start(node_executable,
[applicationDirPath +"/server.js"].concat(Qt.application.arguments.slice(1)),
"EngineFS server started at "
)
}
// TimerStreamingServer
Timer {
id: stayAliveStreamingServer
interval: 10000
running: false
onTriggered: function () { root.launchServer() }
}
//
// Player
//
MpvObject {
id: mpv
anchors.fill: parent
onMpvEvent: function(ev, args) { transport.event(ev, args) }
}
//
// Main UI (via WebEngineView)
//
function getWebUrl() {
var params = "?loginFlow=desktop"
var args = Qt.application.arguments
var shortVer = Qt.application.version.split('.').slice(0, 2).join('.')
var webuiArg = "--webui-url="
for (var i=0; i!=args.length; i++) {
if (args[i].indexOf(webuiArg) === 0) return args[i].slice(webuiArg.length)
}
if (args.indexOf("--development") > -1 || debug)
return "http://127.0.0.1:11470/#"+params
if (args.indexOf("--staging") > -1)
return "https://staging.strem.io/#"+params
return "https://app.strem.io/shell-v"+shortVer+"/#"+params;
}
Timer {
id: retryTimer
interval: 1000
running: false
onTriggered: function () {
webView.tries++
// we want to revert to the mainUrl in case the URL we were at was the one that caused the crash
//webView.reload()
webView.url = webView.mainUrl;
}
}
function injectJS() {
splashScreen.visible = false
pulseOpacity.running = false
removeSplashTimer.running = false
webView.webChannel.registerObject( 'transport', transport )
// Try-catch to be able to return the error as result, but still throw it in the client context
// so it can be caught and reported
var injectedJS = "try { initShellComm() } " +
"catch(e) { setTimeout(function() { throw e }); e.message || JSON.stringify(e) }"
webView.runJavaScript(injectedJS, function(err) {
if (!err) {
webView.tries = 0
} else {
errorDialog.text = "User Interface could not be loaded.\n\nPlease try again later or contact the Stremio support team for assistance."
errorDialog.detailedText = err
errorDialog.visible = true
console.error(err)
}
});
}
// We want to remove the splash after a minute
Timer {
id: removeSplashTimer
interval: 90000
running: true
repeat: false
onTriggered: function () {
webView.backgroundColor = "transparent"
injectJS()
}
}
WebEngineView {
id: webView;
focus: true
readonly property string mainUrl: getWebUrl()
url: webView.mainUrl;
anchors.fill: parent
backgroundColor: "transparent";
property int tries: 0
readonly property int maxTries: 20
Component.onCompleted: function() {
console.log("Loading web UI from URL: "+webView.mainUrl)
webView.profile.httpUserAgent = webView.profile.httpUserAgent+' StremioShell/'+Qt.application.version
// for more info, see
// https://github.com/adobe/chromium/blob/master/net/disk_cache/backend_impl.cc - AdjustMaxCacheSize,
// https://github.com/adobe/chromium/blob/master/net/disk_cache/backend_impl.cc#L2094
webView.profile.httpCacheMaximumSize = 209715200 // 200 MB
}
onLoadingChanged: function(loadRequest) {
// hack for webEngineView changing it's background color on crashes
webView.backgroundColor = "transparent"
var successfullyLoaded = loadRequest.status == WebEngineView.LoadSucceededStatus
if (successfullyLoaded || webView.tries > 0) {
// show the webview if the loading is failing
// can fail because of many reasons, including captive portals
splashScreen.visible = false
pulseOpacity.running = false
}
if (successfullyLoaded) {
injectJS()
}
var shouldRetry = loadRequest.status == WebEngineView.LoadFailedStatus ||
loadRequest.status == WebEngineView.LoadStoppedStatus
if ( shouldRetry && webView.tries < webView.maxTries) {
retryTimer.restart()
}
}
onRenderProcessTerminated: function(terminationStatus, exitCode) {
console.log("render process terminated with code "+exitCode+" and status: "+terminationStatus)
// hack for webEngineView changing it's background color on crashes
webView.backgroundColor = "black"
retryTimer.restart()
// send an event for the crash, but since the web UI is not working, reset the queue and queue it
transport.queued = []
transport.queueEvent("render-process-terminated", { exitCode: exitCode, terminationStatus: terminationStatus, url: webView.url })
}
// WARNING: does not work..for some reason: "Scripts may close only the windows that were opened by it."
// onWindowCloseRequested: function() {
// root.visible = false;
// Qt.quit()
// }
// In the app, we use open-external IPC signal, but make sure this works anyway
property string hoveredUrl: ""
onLinkHovered: webView.hoveredUrl = hoveredUrl
onNewViewRequested: function(req) { if (req.userInitiated) Qt.openUrlExternally(webView.hoveredUrl) }
// FIXME: When is this called?
onFullScreenRequested: function(req) {
setFullScreen(req.toggleOn);
req.accept();
}
// Prevent navigation
onNavigationRequested: function(req) {
// WARNING: @TODO: perhaps we need a better way to parse URLs here
var allowedHost = webView.mainUrl.split('/')[2]
var targetHost = req.url.toString().split('/')[2]
if (allowedHost != targetHost && (req.isMainFrame || targetHost !== 'www.youtube.com')) {
console.log("onNavigationRequested: disallowed URL "+req.url.toString());
req.action = WebEngineView.IgnoreRequest;
}
}
Menu {
id: ctxMenu
MenuItem {
text: "Undo"
shortcut: StandardKey.Undo
onTriggered: webView.triggerWebAction(WebEngineView.Undo)
}
MenuItem {
text: "Redo"
shortcut: StandardKey.Redo
onTriggered: webView.triggerWebAction(WebEngineView.Redo)
}
MenuSeparator { }
MenuItem {
text: "Cut"
shortcut: StandardKey.Cut
onTriggered: webView.triggerWebAction(WebEngineView.Cut)
}
MenuItem {
text: "Copy"
shortcut: StandardKey.Copy
onTriggered: webView.triggerWebAction(WebEngineView.Copy)
}
MenuItem {
text: "Paste"
shortcut: StandardKey.Paste
onTriggered: webView.triggerWebAction(WebEngineView.Paste)
}
MenuSeparator { }
MenuItem {
text: "Select All"
shortcut: StandardKey.SelectAll
onTriggered: webView.triggerWebAction(WebEngineView.SelectAll)
}
}
// Prevent ctx menu
onContextMenuRequested: function(request) {
request.accepted = true;
// Allow menu inside editalbe objects
if (request.isContentEditable) {
ctxMenu.popup();
}
}
Action {
shortcut: StandardKey.Paste
onTriggered: webView.triggerWebAction(WebEngineView.Paste)
}
DropArea {
anchors.fill: parent
onDropped: function(dropargs){
var args = JSON.parse(JSON.stringify(dropargs))
transport.event("dragdrop", args.urls)
}
}
webChannel: wChannel
}
WebChannel {
id: wChannel
}
//
// Splash screen
// Must be over the UI
//
Rectangle {
id: splashScreen;
color: "#0c0b11";
anchors.fill: parent;
Image {
id: splashLogo
source: "qrc:///images/stremio.png"
anchors.horizontalCenter: parent.horizontalCenter
anchors.verticalCenter: parent.verticalCenter
SequentialAnimation {
id: pulseOpacity
running: true
NumberAnimation { target: splashLogo; property: "opacity"; to: 1.0; duration: 600;
easing.type: Easing.Linear; }
NumberAnimation { target: splashLogo; property: "opacity"; to: 0.3; duration: 600;
easing.type: Easing.Linear; }
loops: Animation.Infinite
}
}
}
//
// Err dialog
//
MessageDialog {
id: errorDialog
title: "Stremio - Application Error"
// onAccepted handler does not work
//icon: StandardIcon.Critical
//standardButtons: StandardButton.Ok
}
FileDialog {
id: fileDialog
folder: shortcuts.home
onAccepted: {
var fileProtocol = "file://"
var onWindows = Qt.platform.os === "windows" ? 1 : 0
var pathSeparators = ["/", "\\"]
var files = fileDialog.fileUrls.filter(function(fileUrl) {
// Ignore network drives and alike
return fileUrl.startsWith(fileProtocol)
})
.map(function(fileUrl) {
// Send actual path and not file protocol URL
return decodeURIComponent(fileUrl
.substring(fileProtocol.length + onWindows))
.replace(/\//g, pathSeparators[onWindows])
})
transport.event("file-selected", {
files: files,
title: fileDialog.title,
selectExisting: fileDialog.selectExisting,
selectFolder: fileDialog.selectFolder,
selectMultiple: fileDialog.selectMultiple,
nameFilters: fileDialog.nameFilters,
selectedNameFilter: fileDialog.selectedNameFilter,
data: fileDialog.data
})
}
onRejected: {
transport.event("file-rejected", {
title: fileDialog.title,
selectExisting: fileDialog.selectExisting,
selectFolder: fileDialog.selectFolder,
selectMultiple: fileDialog.selectMultiple,
nameFilters: fileDialog.nameFilters,
selectedNameFilter: fileDialog.selectedNameFilter,
data: fileDialog.data
})
}
property var data: {}
}
//
// Binding window -> app events
//
onWindowStateChanged: function(state) {
updatePreviousVisibility();
transport.event("win-state-changed", { state: state })
}
onVisibilityChanged: {
var enabledAlwaysOnTop = root.visible && root.visibility != Window.FullScreen;
systemTray.alwaysOnTopEnabled(enabledAlwaysOnTop);
if (!enabledAlwaysOnTop) {
root.flags &= ~Qt.WindowStaysOnTopHint;
}
updatePreviousVisibility();
transport.event("win-visibility-changed", { visible: root.visible, visibility: root.visibility,
isFullscreen: root.visibility === Window.FullScreen })
}
property int appState: Qt.application.state;
onAppStateChanged: {
// WARNING: we should load the app through https to avoid MITM attacks on the clipboard
var clipboardUrl
if (clipboard.text.match(/^(magnet|http|https|file|stremio|ipfs):/)) clipboardUrl = clipboard.text
transport.event("app-state-changed", { state: appState, clipboard: clipboardUrl })
// WARNING: CAVEAT: this works when you've focused ANOTHER app and then get back to this one
if (Qt.platform.os === "osx" && appState === Qt.ApplicationActive && !root.visible) {
root.show()
}
}
onClosing: function(event){
event.accepted = false
root.hide()
}
//
// AUTO UPDATER
//
signal autoUpdaterErr(var msg, var err);
signal autoUpdaterRestartTimer();
// Explanation: when the long timer expires, we schedule the short timer; we do that,
// because in case the computer has been asleep for a long time, we want another short timer so we don't check
// immediately (network not connected yet, etc)
// we also schedule the short timer if the computer is offline
Timer {
id: autoUpdaterLongTimer
interval: 2 * 60 * 60 * 1000
running: false
onTriggered: function() { autoUpdaterShortTimer.restart() }
}
Timer {
id: autoUpdaterShortTimer
interval: 5 * 60 * 1000
running: false
onTriggered: function() { } // empty, set if auto-updater is enabled in initAutoUpdater()
}
//
// On complete handler
//
Component.onCompleted: function() {
console.log('Stremio Shell version: '+Qt.application.version)
// Kind of hacky way to ensure there are no Qt bindings going on; otherwise when we go to fullscreen
// Qt tries to restore original window size
root.height = root.initialHeight
root.width = root.initialWidth
// Start streaming server
var args = Qt.application.arguments
if (args.indexOf("--development") > -1 && args.indexOf("--streaming-server") === -1)
console.log("Skipping launch of streaming server under --development");
else
launchServer();
// Handle file opens
var lastArg = args[1]; // not actually last, but we want to be consistent with what happens when we open
// a second instance (main.cpp)
if (args.length > 1 && !lastArg.match('^--')) onAppOpenMedia(lastArg)
// Check for updates
console.info(" **** Completed. Loading Autoupdater ***")
Autoupdater.initAutoUpdater(autoUpdater, root.autoUpdaterErr, autoUpdaterShortTimer, autoUpdaterLongTimer, autoUpdaterRestartTimer, webView.profile.httpUserAgent);
}
}