commit
3a27eda256
152
package.json
152
package.json
|
@ -47,96 +47,96 @@
|
||||||
"windows.store": "npx electron-windows-store --input-directory .\\desktop\\dist\\win-unpacked --output-directory .\\desktop\\store -a .\\resources\\desktop -m .\\desktop\\assets\\windows\\AppXManifest.xml --package-version 0.0.0.0 --package-name MoodleDesktop"
|
"windows.store": "npx electron-windows-store --input-directory .\\desktop\\dist\\win-unpacked --output-directory .\\desktop\\store -a .\\resources\\desktop -m .\\desktop\\assets\\windows\\AppXManifest.xml --package-version 0.0.0.0 --package-name MoodleDesktop"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@angular/animations": "^5.2.11",
|
"@angular/animations": "5.2.11",
|
||||||
"@angular/common": "^5.2.11",
|
"@angular/common": "5.2.11",
|
||||||
"@angular/compiler": "^5.2.11",
|
"@angular/compiler": "5.2.11",
|
||||||
"@angular/compiler-cli": "^5.2.11",
|
"@angular/compiler-cli": "5.2.11",
|
||||||
"@angular/core": "^5.2.11",
|
"@angular/core": "5.2.11",
|
||||||
"@angular/forms": "^5.2.11",
|
"@angular/forms": "5.2.11",
|
||||||
"@angular/platform-browser": "^5.2.11",
|
"@angular/platform-browser": "5.2.11",
|
||||||
"@angular/platform-browser-dynamic": "^5.2.11",
|
"@angular/platform-browser-dynamic": "5.2.11",
|
||||||
"@ionic-native/badge": "^4.20.0",
|
"@ionic-native/badge": "4.20.0",
|
||||||
"@ionic-native/camera": "^4.20.0",
|
"@ionic-native/camera": "4.20.0",
|
||||||
"@ionic-native/chooser": "^4.20.0",
|
"@ionic-native/chooser": "4.20.0",
|
||||||
"@ionic-native/clipboard": "^4.20.0",
|
"@ionic-native/clipboard": "4.20.0",
|
||||||
"@ionic-native/core": "^4.20.0",
|
"@ionic-native/core": "4.20.0",
|
||||||
"@ionic-native/device": "^4.20.0",
|
"@ionic-native/device": "4.20.0",
|
||||||
"@ionic-native/diagnostic": "^4.2.0",
|
"@ionic-native/diagnostic": "4.2.0",
|
||||||
"@ionic-native/file": "^4.20.0",
|
"@ionic-native/file": "4.20.0",
|
||||||
"@ionic-native/file-opener": "^4.20.0",
|
"@ionic-native/file-opener": "4.20.0",
|
||||||
"@ionic-native/file-transfer": "^4.20.0",
|
"@ionic-native/file-transfer": "4.20.0",
|
||||||
"@ionic-native/geolocation": "^4.20.0",
|
"@ionic-native/geolocation": "4.20.0",
|
||||||
"@ionic-native/globalization": "^4.20.0",
|
"@ionic-native/globalization": "4.20.0",
|
||||||
"@ionic-native/http": "^4.20.0",
|
"@ionic-native/http": "4.20.0",
|
||||||
"@ionic-native/in-app-browser": "^4.20.0",
|
"@ionic-native/in-app-browser": "4.20.0",
|
||||||
"@ionic-native/keyboard": "^4.20.0",
|
"@ionic-native/keyboard": "4.20.0",
|
||||||
"@ionic-native/local-notifications": "^4.20.0",
|
"@ionic-native/local-notifications": "4.20.0",
|
||||||
"@ionic-native/media": "^4.20.0",
|
"@ionic-native/media": "4.20.0",
|
||||||
"@ionic-native/media-capture": "^4.20.0",
|
"@ionic-native/media-capture": "4.20.0",
|
||||||
"@ionic-native/network": "^4.20.0",
|
"@ionic-native/network": "4.20.0",
|
||||||
"@ionic-native/push": "^4.20.0",
|
"@ionic-native/push": "4.20.0",
|
||||||
"@ionic-native/qr-scanner": "^4.20.0",
|
"@ionic-native/qr-scanner": "4.20.0",
|
||||||
"@ionic-native/screen-orientation": "^4.20.0",
|
"@ionic-native/screen-orientation": "4.20.0",
|
||||||
"@ionic-native/splash-screen": "^4.20.0",
|
"@ionic-native/splash-screen": "4.20.0",
|
||||||
"@ionic-native/sqlite": "^4.20.0",
|
"@ionic-native/sqlite": "4.20.0",
|
||||||
"@ionic-native/status-bar": "^4.20.0",
|
"@ionic-native/status-bar": "4.20.0",
|
||||||
"@ionic-native/web-intent": "^4.20.0",
|
"@ionic-native/web-intent": "4.20.0",
|
||||||
"@ionic-native/zip": "^4.20.0",
|
"@ionic-native/zip": "4.20.0",
|
||||||
"@ngx-translate/core": "^8.0.0",
|
"@ngx-translate/core": "8.0.0",
|
||||||
"@ngx-translate/http-loader": "^2.0.1",
|
"@ngx-translate/http-loader": "2.0.1",
|
||||||
"ajv": "^6.11.0",
|
"ajv": "6.11.0",
|
||||||
"chart.js": "^2.9.3",
|
"chart.js": "2.9.3",
|
||||||
"com-darryncampbell-cordova-plugin-intent": "^1.3.0",
|
"com-darryncampbell-cordova-plugin-intent": "1.3.0",
|
||||||
"cordova": "^10.0.0",
|
"cordova": "10.0.0",
|
||||||
"cordova-android": "^8.1.0",
|
"cordova-android": "8.1.0",
|
||||||
"cordova-android-support-gradle-release": "^3.0.1",
|
"cordova-android-support-gradle-release": "3.0.1",
|
||||||
"cordova-clipboard": "^1.3.0",
|
"cordova-clipboard": "1.3.0",
|
||||||
"cordova-ios": "^5.1.1",
|
"cordova-ios": "5.1.1",
|
||||||
"cordova-plugin-advanced-http": "^2.4.1",
|
"cordova-plugin-advanced-http": "2.4.1",
|
||||||
"cordova-plugin-badge": "^0.8.8",
|
"cordova-plugin-badge": "0.8.8",
|
||||||
"cordova-plugin-camera": "^4.1.0",
|
"cordova-plugin-camera": "4.1.0",
|
||||||
"cordova-plugin-chooser": "^1.3.2",
|
"cordova-plugin-chooser": "1.3.2",
|
||||||
"cordova-plugin-customurlscheme": "^5.0.1",
|
"cordova-plugin-customurlscheme": "5.0.1",
|
||||||
"cordova-plugin-device": "^2.0.3",
|
"cordova-plugin-device": "2.0.3",
|
||||||
"cordova-plugin-file": "^6.0.2",
|
"cordova-plugin-file": "6.0.2",
|
||||||
"cordova-plugin-file-opener2": "^3.0.4",
|
"cordova-plugin-file-opener2": "3.0.4",
|
||||||
"cordova-plugin-file-transfer": "^1.7.1",
|
"cordova-plugin-file-transfer": "1.7.1",
|
||||||
"cordova-plugin-geolocation": "git+https://github.com/apache/cordova-plugin-geolocation.git#89cf51d222e8f225bdfb661965b3007d669c40ff",
|
"cordova-plugin-geolocation": "git+https://github.com/apache/cordova-plugin-geolocation.git#89cf51d222e8f225bdfb661965b3007d669c40ff",
|
||||||
"cordova-plugin-globalization": "^1.11.0",
|
"cordova-plugin-globalization": "1.11.0",
|
||||||
"cordova-plugin-inappbrowser": "git+https://github.com/moodlemobile/cordova-plugin-inappbrowser.git#moodle",
|
"cordova-plugin-inappbrowser": "git+https://github.com/moodlemobile/cordova-plugin-inappbrowser.git#moodle",
|
||||||
"cordova-plugin-ionic-keyboard": "2.1.3",
|
"cordova-plugin-ionic-keyboard": "2.1.3",
|
||||||
"cordova-plugin-ionic-webview": "git+https://github.com/moodlemobile/cordova-plugin-ionic-webview.git#500-moodle",
|
"cordova-plugin-ionic-webview": "git+https://github.com/moodlemobile/cordova-plugin-ionic-webview.git#500-moodle",
|
||||||
"cordova-plugin-local-notification": "git+https://github.com/moodlemobile/cordova-plugin-local-notification.git#moodle",
|
"cordova-plugin-local-notification": "git+https://github.com/moodlemobile/cordova-plugin-local-notification.git#moodle",
|
||||||
"cordova-plugin-media": "^5.0.3",
|
"cordova-plugin-media": "5.0.3",
|
||||||
"cordova-plugin-media-capture": "^3.0.3",
|
"cordova-plugin-media-capture": "3.0.3",
|
||||||
"cordova-plugin-network-information": "^2.0.2",
|
"cordova-plugin-network-information": "2.0.2",
|
||||||
"cordova-plugin-qrscanner": "git+https://github.com/moodlemobile/cordova-plugin-qrscanner.git#dist",
|
"cordova-plugin-qrscanner": "git+https://github.com/moodlemobile/cordova-plugin-qrscanner.git#dist",
|
||||||
"cordova-plugin-screen-orientation": "^3.0.2",
|
"cordova-plugin-screen-orientation": "3.0.2",
|
||||||
"cordova-plugin-splashscreen": "^6.0.0",
|
"cordova-plugin-splashscreen": "6.0.0",
|
||||||
"cordova-plugin-statusbar": "^2.4.3",
|
"cordova-plugin-statusbar": "2.4.3",
|
||||||
"cordova-plugin-whitelist": "^1.3.4",
|
"cordova-plugin-whitelist": "1.3.4",
|
||||||
"cordova-plugin-wkuserscript": "git+https://github.com/moodlemobile/cordova-plugin-wkuserscript.git",
|
"cordova-plugin-wkuserscript": "git+https://github.com/moodlemobile/cordova-plugin-wkuserscript.git",
|
||||||
"cordova-plugin-wkwebview-cookies": "git+https://github.com/moodlemobile/cordova-plugin-wkwebview-cookies.git",
|
"cordova-plugin-wkwebview-cookies": "git+https://github.com/moodlemobile/cordova-plugin-wkwebview-cookies.git",
|
||||||
"cordova-plugin-zip": "^3.1.0",
|
"cordova-plugin-zip": "3.1.0",
|
||||||
"cordova-sqlite-storage": "^4.0.0",
|
"cordova-sqlite-storage": "4.0.0",
|
||||||
"cordova-support-google-services": "^1.3.2",
|
"cordova-support-google-services": "1.3.2",
|
||||||
"es6-promise-plugin": "^4.2.2",
|
"es6-promise-plugin": "4.2.2",
|
||||||
"font-awesome": "^4.7.0",
|
"font-awesome": "4.7.0",
|
||||||
"inquirer": "^7.3.2",
|
"inquirer": "^7.3.2",
|
||||||
"ionic-angular": "3.9.9",
|
"ionic-angular": "3.9.9",
|
||||||
"ionicons": "3.0.0",
|
"ionicons": "3.0.0",
|
||||||
"jszip": "^3.1.5",
|
"jszip": "3.1.5",
|
||||||
"mathjax": "2.7.7",
|
"mathjax": "2.7.7",
|
||||||
"moment": "^2.24.0",
|
"moment": "2.24.0",
|
||||||
"nl.kingsquare.cordova.background-audio": "^1.0.1",
|
"nl.kingsquare.cordova.background-audio": "1.0.1",
|
||||||
"phonegap-plugin-multidex": "^1.0.0",
|
"phonegap-plugin-multidex": "1.0.0",
|
||||||
"phonegap-plugin-push": "git+https://github.com/moodlemobile/phonegap-plugin-push.git#moodle-v3",
|
"phonegap-plugin-push": "git+https://github.com/moodlemobile/phonegap-plugin-push.git#moodle-v3",
|
||||||
"promise.prototype.finally": "3.1.0",
|
"promise.prototype.finally": "3.1.0",
|
||||||
"rxjs": "^5.5.12",
|
"rxjs": "5.5.12",
|
||||||
"sw-toolbox": "^3.6.0",
|
"sw-toolbox": "3.6.0",
|
||||||
"ts-md5": "^1.2.7",
|
"ts-md5": "1.2.7",
|
||||||
"web-animations-js": "^2.3.2",
|
"web-animations-js": "2.3.2",
|
||||||
"zone.js": "^0.8.29"
|
"zone.js": "0.8.29"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@ionic/app-scripts": "3.2.3",
|
"@ionic/app-scripts": "3.2.3",
|
||||||
|
|
|
@ -24,9 +24,6 @@ import { CoreUrlUtilsProvider } from '@providers/utils/url';
|
||||||
import { CoreIframeUtilsProvider } from '@providers/utils/iframe';
|
import { CoreIframeUtilsProvider } from '@providers/utils/iframe';
|
||||||
import { CoreUtilsProvider } from '@providers/utils/utils';
|
import { CoreUtilsProvider } from '@providers/utils/utils';
|
||||||
import { CoreSplitViewComponent } from '@components/split-view/split-view';
|
import { CoreSplitViewComponent } from '@components/split-view/split-view';
|
||||||
import { CoreUrl } from '@singletons/url';
|
|
||||||
import { CoreApp } from '@providers/app';
|
|
||||||
import { WKWebViewCookiesWindow } from 'cordova-plugin-wkwebview-cookies';
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'core-iframe',
|
selector: 'core-iframe',
|
||||||
|
@ -107,24 +104,7 @@ export class CoreIframeComponent implements OnChanges {
|
||||||
if (changes.src) {
|
if (changes.src) {
|
||||||
const url = this.urlUtils.getYoutubeEmbedUrl(changes.src.currentValue) || changes.src.currentValue;
|
const url = this.urlUtils.getYoutubeEmbedUrl(changes.src.currentValue) || changes.src.currentValue;
|
||||||
|
|
||||||
if (CoreApp.instance.isIOS() && url && !this.urlUtils.isLocalFileUrl(url)) {
|
await this.iframeUtils.fixIframeCookies(url);
|
||||||
// Save a "fake" cookie for the iframe's domain to fix a bug in WKWebView.
|
|
||||||
try {
|
|
||||||
const win = <WKWebViewCookiesWindow> window;
|
|
||||||
const urlParts = CoreUrl.parse(url);
|
|
||||||
|
|
||||||
if (urlParts.domain) {
|
|
||||||
await win.WKWebViewCookies.setCookie({
|
|
||||||
name: 'MoodleAppCookieForWKWebView',
|
|
||||||
value: '1',
|
|
||||||
domain: urlParts.domain,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
// Ignore errors.
|
|
||||||
this.logger.error('Error setting cookie', err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.safeUrl = this.sanitizer.bypassSecurityTrustResourceUrl(CoreFile.instance.convertFileSrc(url));
|
this.safeUrl = this.sanitizer.bypassSecurityTrustResourceUrl(CoreFile.instance.convertFileSrc(url));
|
||||||
|
|
||||||
|
|
|
@ -191,7 +191,7 @@ export class CoreContentLinksDelegate {
|
||||||
|
|
||||||
// Add them to the list.
|
// Add them to the list.
|
||||||
linkActions.push({
|
linkActions.push({
|
||||||
priority: handler.priority,
|
priority: handler.priority || 0,
|
||||||
actions: actions
|
actions: actions
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -405,7 +405,7 @@ export class CoreCourseOptionsDelegate extends CoreDelegate {
|
||||||
promises.push(Promise.resolve(getFunction.call(handler, injector, course)).then((data) => {
|
promises.push(Promise.resolve(getFunction.call(handler, injector, course)).then((data) => {
|
||||||
handlersToDisplay.push({
|
handlersToDisplay.push({
|
||||||
data: data,
|
data: data,
|
||||||
priority: handler.priority,
|
priority: handler.priority || 0,
|
||||||
prefetch: handler.prefetch && handler.prefetch.bind(handler),
|
prefetch: handler.prefetch && handler.prefetch.bind(handler),
|
||||||
name: handler.name
|
name: handler.name
|
||||||
});
|
});
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
import { Component, Input, Output, EventEmitter, ViewChild, ElementRef, AfterContentInit, OnDestroy, Optional }
|
import { Component, Input, Output, EventEmitter, ViewChild, ElementRef, AfterContentInit, OnDestroy, Optional }
|
||||||
from '@angular/core';
|
from '@angular/core';
|
||||||
import { TextInput, Content, Platform, Slides } from 'ionic-angular';
|
import { TextInput, Content, Platform, Slides } from 'ionic-angular';
|
||||||
|
import { Device } from '@ionic-native/device';
|
||||||
import { CoreApp } from '@providers/app';
|
import { CoreApp } from '@providers/app';
|
||||||
import { CoreSitesProvider } from '@providers/sites';
|
import { CoreSitesProvider } from '@providers/sites';
|
||||||
import { CoreFilepoolProvider } from '@providers/filepool';
|
import { CoreFilepoolProvider } from '@providers/filepool';
|
||||||
|
@ -118,7 +119,8 @@ export class CoreEditorRichTextEditorComponent implements AfterContentInit, OnDe
|
||||||
protected events: CoreEventsProvider,
|
protected events: CoreEventsProvider,
|
||||||
protected utils: CoreUtilsProvider,
|
protected utils: CoreUtilsProvider,
|
||||||
protected platform: Platform,
|
protected platform: Platform,
|
||||||
protected editorOffline: CoreEditorOfflineProvider) {
|
protected editorOffline: CoreEditorOfflineProvider,
|
||||||
|
protected device: Device) {
|
||||||
this.contentChanged = new EventEmitter<string>();
|
this.contentChanged = new EventEmitter<string>();
|
||||||
this.element = elementRef.nativeElement as HTMLDivElement;
|
this.element = elementRef.nativeElement as HTMLDivElement;
|
||||||
this.pageInstance = 'app_' + Date.now(); // Generate a "unique" ID based on timestamp.
|
this.pageInstance = 'app_' + Date.now(); // Generate a "unique" ID based on timestamp.
|
||||||
|
@ -236,7 +238,7 @@ export class CoreEditorRichTextEditorComponent implements AfterContentInit, OnDe
|
||||||
if (CoreApp.instance.isAndroid()) {
|
if (CoreApp.instance.isAndroid()) {
|
||||||
// In Android we ignore the keyboard height because it is not part of the web view.
|
// In Android we ignore the keyboard height because it is not part of the web view.
|
||||||
height = this.domUtils.getContentHeight(this.content) - this.getSurroundingHeight(this.element);
|
height = this.domUtils.getContentHeight(this.content) - this.getSurroundingHeight(this.element);
|
||||||
} else if (CoreApp.instance.isIOS() && this.kbHeight > 0 && this.platform.version().major < 12) {
|
} else if (CoreApp.instance.isIOS() && this.kbHeight > 0 && Number(this.device.version.split('.')[0]) < 12) {
|
||||||
// Keyboard open in iOS 11 or previous. The window height changes when the keyboard is open.
|
// Keyboard open in iOS 11 or previous. The window height changes when the keyboard is open.
|
||||||
height = window.innerHeight - this.getSurroundingHeight(this.element);
|
height = window.innerHeight - this.getSurroundingHeight(this.element);
|
||||||
|
|
||||||
|
|
|
@ -177,7 +177,7 @@ export class CoreFileUploaderDelegate extends CoreDelegate {
|
||||||
}
|
}
|
||||||
|
|
||||||
const data: CoreFileUploaderHandlerDataToReturn = handler.getData();
|
const data: CoreFileUploaderHandlerDataToReturn = handler.getData();
|
||||||
data.priority = handler.priority;
|
data.priority = handler.priority || 0;
|
||||||
data.mimetypes = supportedMimetypes;
|
data.mimetypes = supportedMimetypes;
|
||||||
handlers.push(data);
|
handlers.push(data);
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,8 +44,7 @@ export class CoreFileUploaderFileHandler implements CoreFileUploaderHandler {
|
||||||
* @return True or promise resolved with true if enabled.
|
* @return True or promise resolved with true if enabled.
|
||||||
*/
|
*/
|
||||||
isEnabled(): boolean | Promise<boolean> {
|
isEnabled(): boolean | Promise<boolean> {
|
||||||
return CoreApp.instance.isAndroid() || !CoreApp.instance.isMobile() ||
|
return true;
|
||||||
(CoreApp.instance.isIOS() && this.platform.version().major >= 9);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -46,6 +46,7 @@ export class CoreMainMenuPage implements OnDestroy {
|
||||||
protected urlToOpen: string;
|
protected urlToOpen: string;
|
||||||
protected mainMenuId: number;
|
protected mainMenuId: number;
|
||||||
protected keyboardObserver: any;
|
protected keyboardObserver: any;
|
||||||
|
protected resizeFunction;
|
||||||
|
|
||||||
@ViewChild('mainTabs') mainTabs: CoreIonTabsComponent;
|
@ViewChild('mainTabs') mainTabs: CoreIonTabsComponent;
|
||||||
|
|
||||||
|
@ -115,7 +116,8 @@ export class CoreMainMenuPage implements OnDestroy {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
window.addEventListener('resize', this.initHandlers.bind(this));
|
this.resizeFunction = this.initHandlers.bind(this);
|
||||||
|
window.addEventListener('resize', this.resizeFunction);
|
||||||
|
|
||||||
if (CoreApp.instance.isIOS()) {
|
if (CoreApp.instance.isIOS()) {
|
||||||
// In iOS, the resize event is triggered before the keyboard is opened/closed and not triggered again once done.
|
// In iOS, the resize event is triggered before the keyboard is opened/closed and not triggered again once done.
|
||||||
|
@ -241,7 +243,7 @@ export class CoreMainMenuPage implements OnDestroy {
|
||||||
ngOnDestroy(): void {
|
ngOnDestroy(): void {
|
||||||
this.subscription && this.subscription.unsubscribe();
|
this.subscription && this.subscription.unsubscribe();
|
||||||
this.redirectObs && this.redirectObs.off();
|
this.redirectObs && this.redirectObs.off();
|
||||||
window.removeEventListener('resize', this.initHandlers.bind(this));
|
window.removeEventListener('resize', this.resizeFunction);
|
||||||
CoreApp.instance.setMainMenuOpen(this.mainMenuId, false);
|
CoreApp.instance.setMainMenuOpen(this.mainMenuId, false);
|
||||||
this.keyboardObserver && this.keyboardObserver.off();
|
this.keyboardObserver && this.keyboardObserver.off();
|
||||||
}
|
}
|
||||||
|
|
|
@ -162,7 +162,7 @@ export class CoreMainMenuDelegate extends CoreDelegate {
|
||||||
handlersData.push({
|
handlersData.push({
|
||||||
name: name,
|
name: name,
|
||||||
data: data,
|
data: data,
|
||||||
priority: handler.priority
|
priority: handler.priority || 0,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -177,6 +177,7 @@ export class CorePushNotificationsDelegate {
|
||||||
|
|
||||||
this.logger.log(`Registered addon '${handler.name}'`);
|
this.logger.log(`Registered addon '${handler.name}'`);
|
||||||
this.clickHandlers[handler.name] = handler;
|
this.clickHandlers[handler.name] = handler;
|
||||||
|
handler.priority = handler.priority || 0;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -108,7 +108,7 @@ export class CoreSettingsDelegate extends CoreDelegate {
|
||||||
|
|
||||||
handlersData.push({
|
handlersData.push({
|
||||||
data: data,
|
data: data,
|
||||||
priority: handler.priority
|
priority: handler.priority || 0,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -35,7 +35,7 @@ export class CoreSitePluginsCourseOptionHandler extends CoreSitePluginsBaseHandl
|
||||||
protected utils: CoreUtilsProvider) {
|
protected utils: CoreUtilsProvider) {
|
||||||
super(name);
|
super(name);
|
||||||
|
|
||||||
this.priority = handlerSchema.priority;
|
this.priority = handlerSchema.priority || 0;
|
||||||
this.isMenuHandler = !!handlerSchema.ismenuhandler;
|
this.isMenuHandler = !!handlerSchema.ismenuhandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,7 @@ export class CoreSitePluginsMainMenuHandler extends CoreSitePluginsBaseHandler i
|
||||||
protected initResult: any) {
|
protected initResult: any) {
|
||||||
super(name);
|
super(name);
|
||||||
|
|
||||||
this.priority = handlerSchema.priority;
|
this.priority = handlerSchema.priority || 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -32,7 +32,7 @@ export class CoreSitePluginsMessageOutputHandler extends CoreSitePluginsBaseHand
|
||||||
*/
|
*/
|
||||||
getDisplayData(): AddonMessageOutputHandlerData {
|
getDisplayData(): AddonMessageOutputHandlerData {
|
||||||
return {
|
return {
|
||||||
priority: this.handlerSchema.priority,
|
priority: this.handlerSchema.priority || 0,
|
||||||
label: this.title,
|
label: this.title,
|
||||||
icon: this.handlerSchema.displaydata.icon,
|
icon: this.handlerSchema.displaydata.icon,
|
||||||
page: 'CoreSitePluginsPluginPage',
|
page: 'CoreSitePluginsPluginPage',
|
||||||
|
|
|
@ -25,7 +25,7 @@ export class CoreSitePluginsSettingsHandler extends CoreSitePluginsBaseHandler i
|
||||||
protected initResult: any) {
|
protected initResult: any) {
|
||||||
super(name);
|
super(name);
|
||||||
|
|
||||||
this.priority = handlerSchema.priority;
|
this.priority = handlerSchema.priority || 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -43,7 +43,7 @@ export class CoreSitePluginsUserProfileHandler extends CoreSitePluginsBaseHandle
|
||||||
protected utils: CoreUtilsProvider) {
|
protected utils: CoreUtilsProvider) {
|
||||||
super(name);
|
super(name);
|
||||||
|
|
||||||
this.priority = handlerSchema.priority;
|
this.priority = handlerSchema.priority || 0;
|
||||||
|
|
||||||
// Only support TYPE_COMMUNICATION and TYPE_NEW_PAGE.
|
// Only support TYPE_COMMUNICATION and TYPE_NEW_PAGE.
|
||||||
this.type = handlerSchema.type != CoreUserDelegate.TYPE_COMMUNICATION ?
|
this.type = handlerSchema.type != CoreUserDelegate.TYPE_COMMUNICATION ?
|
||||||
|
|
|
@ -260,7 +260,7 @@ export class CoreUserDelegate extends CoreDelegate {
|
||||||
userData.handlers.push({
|
userData.handlers.push({
|
||||||
name: name,
|
name: name,
|
||||||
data: handler.getDisplayData(user, courseId),
|
data: handler.getDisplayData(user, courseId),
|
||||||
priority: handler.priority,
|
priority: handler.priority || 0,
|
||||||
type: handler.type || CoreUserDelegate.TYPE_NEW_PAGE
|
type: handler.type || CoreUserDelegate.TYPE_NEW_PAGE
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -450,6 +450,7 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
const div = document.createElement('div'),
|
const div = document.createElement('div'),
|
||||||
canTreatVimeo = site && site.isVersionGreaterEqualThan(['3.3.4', '3.4']),
|
canTreatVimeo = site && site.isVersionGreaterEqualThan(['3.3.4', '3.4']),
|
||||||
navCtrl = this.svComponent ? this.svComponent.getMasterNav() : this.navCtrl;
|
navCtrl = this.svComponent ? this.svComponent.getMasterNav() : this.navCtrl;
|
||||||
|
const promises = [];
|
||||||
|
|
||||||
div.innerHTML = formatted;
|
div.innerHTML = formatted;
|
||||||
|
|
||||||
|
@ -504,7 +505,7 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
});
|
});
|
||||||
|
|
||||||
iframes.forEach((iframe) => {
|
iframes.forEach((iframe) => {
|
||||||
this.treatIframe(iframe, site, canTreatVimeo, navCtrl);
|
promises.push(this.treatIframe(iframe, site, canTreatVimeo, navCtrl));
|
||||||
});
|
});
|
||||||
|
|
||||||
svgImages.forEach((image) => {
|
svgImages.forEach((image) => {
|
||||||
|
@ -543,10 +544,9 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
this.domUtils.handleBootstrapTooltips(div);
|
this.domUtils.handleBootstrapTooltips(div);
|
||||||
|
|
||||||
// Wait for images to load.
|
// Wait for images to load.
|
||||||
let promise: Promise<any> = null;
|
|
||||||
if (externalImages.length) {
|
if (externalImages.length) {
|
||||||
// Automatically reject the promise after 5 seconds to prevent blocking the user forever.
|
// Automatically reject the promise after 5 seconds to prevent blocking the user forever.
|
||||||
promise = this.utils.timeoutPromise(this.utils.allPromises(externalImages.map((externalImage): any => {
|
promises.push(this.utils.timeoutPromise(this.utils.allPromises(externalImages.map((externalImage): any => {
|
||||||
if (externalImage.loaded) {
|
if (externalImage.loaded) {
|
||||||
// Image has already been loaded, no need to wait.
|
// Image has already been loaded, no need to wait.
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
|
@ -558,12 +558,10 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
resolve();
|
resolve();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
})), 5000);
|
})), 5000));
|
||||||
} else {
|
|
||||||
promise = Promise.resolve();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return promise.catch(() => {
|
return Promise.all(promises).catch(() => {
|
||||||
// Ignore errors. So content gets always shown.
|
// Ignore errors. So content gets always shown.
|
||||||
}).then(() => {
|
}).then(() => {
|
||||||
result.div = div;
|
result.div = div;
|
||||||
|
@ -665,7 +663,8 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
* @param canTreatVimeo Whether Vimeo videos can be treated in the site.
|
* @param canTreatVimeo Whether Vimeo videos can be treated in the site.
|
||||||
* @param navCtrl NavController to use.
|
* @param navCtrl NavController to use.
|
||||||
*/
|
*/
|
||||||
protected treatIframe(iframe: HTMLIFrameElement, site: CoreSite, canTreatVimeo: boolean, navCtrl: NavController): void {
|
protected async treatIframe(iframe: HTMLIFrameElement, site: CoreSite, canTreatVimeo: boolean, navCtrl: NavController)
|
||||||
|
: Promise<void> {
|
||||||
const src = iframe.src,
|
const src = iframe.src,
|
||||||
currentSite = this.sitesProvider.getCurrentSite();
|
currentSite = this.sitesProvider.getCurrentSite();
|
||||||
|
|
||||||
|
@ -673,15 +672,19 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
|
|
||||||
if (currentSite && currentSite.containsUrl(src)) {
|
if (currentSite && currentSite.containsUrl(src)) {
|
||||||
// URL points to current site, try to use auto-login.
|
// URL points to current site, try to use auto-login.
|
||||||
currentSite.getAutoLoginUrl(src, false).then((finalUrl) => {
|
const finalUrl = await currentSite.getAutoLoginUrl(src, false);
|
||||||
iframe.src = finalUrl;
|
|
||||||
|
|
||||||
this.iframeUtils.treatFrame(iframe, false, navCtrl);
|
await this.iframeUtils.fixIframeCookies(finalUrl);
|
||||||
});
|
|
||||||
|
iframe.src = finalUrl;
|
||||||
|
|
||||||
|
this.iframeUtils.treatFrame(iframe, false, navCtrl);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await this.iframeUtils.fixIframeCookies(src);
|
||||||
|
|
||||||
if (src && canTreatVimeo) {
|
if (src && canTreatVimeo) {
|
||||||
// Check if it's a Vimeo video. If it is, use the wsplayer script instead to make restricted videos work.
|
// Check if it's a Vimeo video. If it is, use the wsplayer script instead to make restricted videos work.
|
||||||
const matches = iframe.src.match(/https?:\/\/player\.vimeo\.com\/video\/([0-9]+)/);
|
const matches = iframe.src.match(/https?:\/\/player\.vimeo\.com\/video\/([0-9]+)/);
|
||||||
|
@ -714,6 +717,9 @@ export class CoreFormatTextDirective implements OnChanges {
|
||||||
if (site && !site.isVersionGreaterEqualThan('3.7')) {
|
if (site && !site.isVersionGreaterEqualThan('3.7')) {
|
||||||
newUrl += '&width=' + width + '&height=' + height;
|
newUrl += '&width=' + width + '&height=' + height;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await this.iframeUtils.fixIframeCookies(newUrl);
|
||||||
|
|
||||||
iframe.src = newUrl;
|
iframe.src = newUrl;
|
||||||
|
|
||||||
if (!iframe.width) {
|
if (!iframe.width) {
|
||||||
|
|
|
@ -1549,11 +1549,16 @@ export class CoreSitesProvider {
|
||||||
await this.dbReady;
|
await this.dbReady;
|
||||||
|
|
||||||
const site = await this.getSite(siteId);
|
const site = await this.getSite(siteId);
|
||||||
const newValues = {
|
const newValues: any = {
|
||||||
token: '', // Erase the token for security.
|
|
||||||
loggedOut: loggedOut ? 1 : 0
|
loggedOut: loggedOut ? 1 : 0
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (loggedOut) {
|
||||||
|
// Erase the token for security.
|
||||||
|
newValues.token = '';
|
||||||
|
site.token = '';
|
||||||
|
}
|
||||||
|
|
||||||
site.setLoggedOut(loggedOut);
|
site.setLoggedOut(loggedOut);
|
||||||
|
|
||||||
return this.appDB.updateRecords(CoreSitesProvider.SITES_TABLE, newValues, { id: siteId });
|
return this.appDB.updateRecords(CoreSitesProvider.SITES_TABLE, newValues, { id: siteId });
|
||||||
|
|
|
@ -30,6 +30,7 @@ import { makeSingleton } from '@singletons/core.singletons';
|
||||||
import { CoreUrl } from '@singletons/url';
|
import { CoreUrl } from '@singletons/url';
|
||||||
import { CoreWindow } from '@singletons/window';
|
import { CoreWindow } from '@singletons/window';
|
||||||
import { WKUserScriptWindow, WKUserScriptInjectionTime } from 'cordova-plugin-wkuserscript';
|
import { WKUserScriptWindow, WKUserScriptInjectionTime } from 'cordova-plugin-wkuserscript';
|
||||||
|
import { WKWebViewCookiesWindow } from 'cordova-plugin-wkwebview-cookies';
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* "Utils" service with helper functions for iframes, embed and similar.
|
* "Utils" service with helper functions for iframes, embed and similar.
|
||||||
|
@ -488,6 +489,36 @@ export class CoreIframeUtilsProvider {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fix cookies for an iframe URL.
|
||||||
|
*
|
||||||
|
* @param url URL of the iframe.
|
||||||
|
* @return Promise resolved when done.
|
||||||
|
*/
|
||||||
|
async fixIframeCookies(url: string): Promise<void> {
|
||||||
|
if (!CoreApp.instance.isIOS() || !url || this.urlUtils.isLocalFileUrl(url)) {
|
||||||
|
// No need to fix cookies.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save a "fake" cookie for the iframe's domain to fix a bug in WKWebView.
|
||||||
|
try {
|
||||||
|
const win = <WKWebViewCookiesWindow> window;
|
||||||
|
const urlParts = CoreUrl.parse(url);
|
||||||
|
|
||||||
|
if (urlParts.domain) {
|
||||||
|
await win.WKWebViewCookies.setCookie({
|
||||||
|
name: 'MoodleAppCookieForWKWebView',
|
||||||
|
value: '1',
|
||||||
|
domain: urlParts.domain,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
// Ignore errors.
|
||||||
|
this.logger.error('Error setting cookie', err);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class CoreIframeUtils extends makeSingleton(CoreIframeUtilsProvider) {}
|
export class CoreIframeUtils extends makeSingleton(CoreIframeUtilsProvider) {}
|
||||||
|
|
Loading…
Reference in New Issue