MOBILE-3947 slides: Use Swiper instead of IonSlides
parent
34147fceb7
commit
7c31e79bbd
|
@ -94,6 +94,7 @@
|
||||||
"nl.kingsquare.cordova.background-audio": "^1.0.1",
|
"nl.kingsquare.cordova.background-audio": "^1.0.1",
|
||||||
"ogv": "^1.8.9",
|
"ogv": "^1.8.9",
|
||||||
"rxjs": "~7.8.0",
|
"rxjs": "~7.8.0",
|
||||||
|
"swiper": "^11.0.3",
|
||||||
"ts-md5": "^1.2.7",
|
"ts-md5": "^1.2.7",
|
||||||
"tslib": "^2.3.0",
|
"tslib": "^2.3.0",
|
||||||
"video.js": "^7.21.1",
|
"video.js": "^7.21.1",
|
||||||
|
@ -27566,6 +27567,24 @@
|
||||||
"es6-symbol": "^3.1.1"
|
"es6-symbol": "^3.1.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/swiper": {
|
||||||
|
"version": "11.0.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/swiper/-/swiper-11.0.5.tgz",
|
||||||
|
"integrity": "sha512-rhCwupqSyRnWrtNzWzemnBLMoyYuoDgGgspAm/8iBD3jCvAWycPLH4Z3TB0O5520DHLzMx94yUMH/B9Efpa48w==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/swiperjs"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "open_collective",
|
||||||
|
"url": "http://opencollective.com/swiper"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 4.7.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/symbol-observable": {
|
"node_modules/symbol-observable": {
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-4.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-4.0.0.tgz",
|
||||||
|
|
|
@ -129,6 +129,7 @@
|
||||||
"nl.kingsquare.cordova.background-audio": "^1.0.1",
|
"nl.kingsquare.cordova.background-audio": "^1.0.1",
|
||||||
"ogv": "^1.8.9",
|
"ogv": "^1.8.9",
|
||||||
"rxjs": "~7.8.0",
|
"rxjs": "~7.8.0",
|
||||||
|
"swiper": "^11.0.3",
|
||||||
"ts-md5": "^1.2.7",
|
"ts-md5": "^1.2.7",
|
||||||
"tslib": "^2.3.0",
|
"tslib": "^2.3.0",
|
||||||
"video.js": "^7.21.1",
|
"video.js": "^7.21.1",
|
||||||
|
|
|
@ -142,7 +142,7 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ion-slide {
|
swiper-slide {
|
||||||
display: block;
|
display: block;
|
||||||
font-size: inherit;
|
font-size: inherit;
|
||||||
justify-content: start;
|
justify-content: start;
|
||||||
|
|
|
@ -64,7 +64,7 @@ import { Translate } from '@singletons';
|
||||||
})
|
})
|
||||||
export class AddonCalendarCalendarComponent implements OnInit, DoCheck, OnDestroy {
|
export class AddonCalendarCalendarComponent implements OnInit, DoCheck, OnDestroy {
|
||||||
|
|
||||||
@ViewChild(CoreSwipeSlidesComponent) slides?: CoreSwipeSlidesComponent<PreloadedMonth>;
|
@ViewChild(CoreSwipeSlidesComponent) swipeSlidesComponent?: CoreSwipeSlidesComponent<PreloadedMonth>;
|
||||||
|
|
||||||
@Input() initialYear?: number; // Initial year to load.
|
@Input() initialYear?: number; // Initial year to load.
|
||||||
@Input() initialMonth?: number; // Initial month to load.
|
@Input() initialMonth?: number; // Initial month to load.
|
||||||
|
@ -185,7 +185,7 @@ export class AddonCalendarCalendarComponent implements OnInit, DoCheck, OnDestro
|
||||||
this.hiddenDiffer = this.hidden;
|
this.hiddenDiffer = this.hidden;
|
||||||
|
|
||||||
if (!this.hidden) {
|
if (!this.hidden) {
|
||||||
this.slides?.slides?.getSwiper().then(swipper => swipper.update());
|
this.swipeSlidesComponent?.updateSlidesComponent();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -248,14 +248,14 @@ export class AddonCalendarCalendarComponent implements OnInit, DoCheck, OnDestro
|
||||||
* Load next month.
|
* Load next month.
|
||||||
*/
|
*/
|
||||||
loadNext(): void {
|
loadNext(): void {
|
||||||
this.slides?.slideNext();
|
this.swipeSlidesComponent?.slideNext();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load previous month.
|
* Load previous month.
|
||||||
*/
|
*/
|
||||||
loadPrevious(): void {
|
loadPrevious(): void {
|
||||||
this.slides?.slidePrev();
|
this.swipeSlidesComponent?.slidePrev();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -343,8 +343,7 @@ export class AddonCalendarCalendarComponent implements OnInit, DoCheck, OnDestro
|
||||||
*/
|
*/
|
||||||
async viewMonth(month: number, year: number): Promise<void> {
|
async viewMonth(month: number, year: number): Promise<void> {
|
||||||
const manager = this.manager;
|
const manager = this.manager;
|
||||||
const slides = this.slides;
|
if (!manager || !this.swipeSlidesComponent) {
|
||||||
if (!manager || !slides) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -360,7 +359,7 @@ export class AddonCalendarCalendarComponent implements OnInit, DoCheck, OnDestro
|
||||||
// Make sure the day is loaded.
|
// Make sure the day is loaded.
|
||||||
await manager.getSource().loadItem(item);
|
await manager.getSource().loadItem(item);
|
||||||
|
|
||||||
slides.slideToItem(item);
|
this.swipeSlidesComponent.slideToItem(item);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
CoreDomUtils.showErrorModalDefault(error, 'addon.calendar.errorloadevents', true);
|
CoreDomUtils.showErrorModalDefault(error, 'addon.calendar.errorloadevents', true);
|
||||||
} finally {
|
} finally {
|
||||||
|
|
|
@ -60,7 +60,7 @@ import { CoreTime } from '@singletons/time';
|
||||||
})
|
})
|
||||||
export class AddonCalendarDayPage implements OnInit, OnDestroy {
|
export class AddonCalendarDayPage implements OnInit, OnDestroy {
|
||||||
|
|
||||||
@ViewChild(CoreSwipeSlidesComponent) slides?: CoreSwipeSlidesComponent<PreloadedDay>;
|
@ViewChild(CoreSwipeSlidesComponent) swipeSlidesComponent?: CoreSwipeSlidesComponent<PreloadedDay>;
|
||||||
|
|
||||||
protected currentSiteId: string;
|
protected currentSiteId: string;
|
||||||
|
|
||||||
|
@ -434,8 +434,7 @@ export class AddonCalendarDayPage implements OnInit, OnDestroy {
|
||||||
*/
|
*/
|
||||||
async goToCurrentDay(): Promise<void> {
|
async goToCurrentDay(): Promise<void> {
|
||||||
const manager = this.manager;
|
const manager = this.manager;
|
||||||
const slides = this.slides;
|
if (!manager || !this.swipeSlidesComponent) {
|
||||||
if (!manager || !slides) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -448,7 +447,7 @@ export class AddonCalendarDayPage implements OnInit, OnDestroy {
|
||||||
// Make sure the day is loaded.
|
// Make sure the day is loaded.
|
||||||
await manager.getSource().loadItem(currentDay);
|
await manager.getSource().loadItem(currentDay);
|
||||||
|
|
||||||
slides.slideToItem(currentDay);
|
this.swipeSlidesComponent.slideToItem(currentDay);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
CoreDomUtils.showErrorModalDefault(error, 'addon.calendar.errorloadevents', true);
|
CoreDomUtils.showErrorModalDefault(error, 'addon.calendar.errorloadevents', true);
|
||||||
} finally {
|
} finally {
|
||||||
|
@ -460,14 +459,14 @@ export class AddonCalendarDayPage implements OnInit, OnDestroy {
|
||||||
* Load next day.
|
* Load next day.
|
||||||
*/
|
*/
|
||||||
async loadNext(): Promise<void> {
|
async loadNext(): Promise<void> {
|
||||||
this.slides?.slideNext();
|
this.swipeSlidesComponent?.slideNext();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load previous day.
|
* Load previous day.
|
||||||
*/
|
*/
|
||||||
async loadPrevious(): Promise<void> {
|
async loadPrevious(): Promise<void> {
|
||||||
this.slides?.slidePrev();
|
this.swipeSlidesComponent?.slidePrev();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
</ion-item>
|
</ion-item>
|
||||||
</ion-card>
|
</ion-card>
|
||||||
|
|
||||||
<core-swipe-slides [manager]="manager" [options]="slidesOpts">
|
<core-swipe-slides [manager]="manager" [options]="swiperOpts">
|
||||||
<ng-template let-chapter="item" let-active="active">
|
<ng-template let-chapter="item" let-active="active">
|
||||||
<div class="ion-padding">
|
<div class="ion-padding">
|
||||||
<core-format-text [component]="component" [componentId]="cmId" [text]="chapter.content" contextLevel="module"
|
<core-format-text [component]="component" [componentId]="cmId" [text]="chapter.content" contextLevel="module"
|
||||||
|
|
|
@ -41,6 +41,7 @@ import {
|
||||||
} from '../../services/book';
|
} from '../../services/book';
|
||||||
import { CoreAnalytics, CoreAnalyticsEventType } from '@services/analytics';
|
import { CoreAnalytics, CoreAnalyticsEventType } from '@services/analytics';
|
||||||
import { CoreUrlUtils } from '@services/utils/url';
|
import { CoreUrlUtils } from '@services/utils/url';
|
||||||
|
import { IonicSlides } from '@ionic/angular';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Page that displays a book contents.
|
* Page that displays a book contents.
|
||||||
|
@ -52,7 +53,7 @@ import { CoreUrlUtils } from '@services/utils/url';
|
||||||
})
|
})
|
||||||
export class AddonModBookContentsPage implements OnInit, OnDestroy {
|
export class AddonModBookContentsPage implements OnInit, OnDestroy {
|
||||||
|
|
||||||
@ViewChild(CoreSwipeSlidesComponent) slides?: CoreSwipeSlidesComponent;
|
@ViewChild(CoreSwipeSlidesComponent) swipeSlidesComponent?: CoreSwipeSlidesComponent;
|
||||||
|
|
||||||
title = '';
|
title = '';
|
||||||
cmId!: number;
|
cmId!: number;
|
||||||
|
@ -63,7 +64,8 @@ export class AddonModBookContentsPage implements OnInit, OnDestroy {
|
||||||
warning = '';
|
warning = '';
|
||||||
displayNavBar = true;
|
displayNavBar = true;
|
||||||
navigationItems: CoreNavigationBarItem<AddonModBookTocChapter>[] = [];
|
navigationItems: CoreNavigationBarItem<AddonModBookTocChapter>[] = [];
|
||||||
slidesOpts: CoreSwipeSlidesOptions = {
|
swiperOpts: CoreSwipeSlidesOptions = {
|
||||||
|
modules: [IonicSlides],
|
||||||
autoHeight: true,
|
autoHeight: true,
|
||||||
observer: true,
|
observer: true,
|
||||||
observeParents: true,
|
observeParents: true,
|
||||||
|
@ -222,7 +224,7 @@ export class AddonModBookContentsPage implements OnInit, OnDestroy {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.slides?.slideToItem({ id: chapterId });
|
this.swipeSlidesComponent?.slideToItem({ id: chapterId });
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -134,22 +134,22 @@ Feature: Test basic usage of book activity in app
|
||||||
But I should not find "This is the first chapter" in the app
|
But I should not find "This is the first chapter" in the app
|
||||||
|
|
||||||
# Navigate using swipe.
|
# Navigate using swipe.
|
||||||
When I swipe to the left in "Chapt 3" "ion-slides" in the app
|
When I swipe to the left in "Chapt 3" "swiper-container" in the app
|
||||||
Then I should find "Chapt 3" in the app
|
Then I should find "Chapt 3" in the app
|
||||||
And I should find "This is the third chapter" in the app
|
And I should find "This is the third chapter" in the app
|
||||||
And I should find "4 / 4" in the app
|
And I should find "4 / 4" in the app
|
||||||
|
|
||||||
When I swipe to the right in "Chapt 3" "ion-slides" in the app
|
When I swipe to the right in "Chapt 3" "swiper-container" in the app
|
||||||
Then I should find "Chapt 2" in the app
|
Then I should find "Chapt 2" in the app
|
||||||
And I should find "This is the second chapter" in the app
|
And I should find "This is the second chapter" in the app
|
||||||
And I should find "3 / 4" in the app
|
And I should find "3 / 4" in the app
|
||||||
|
|
||||||
When I swipe to the right in "Chapt 2" "ion-slides" in the app
|
When I swipe to the right in "Chapt 2" "swiper-container" in the app
|
||||||
Then I should find "Chapt 1.1" in the app
|
Then I should find "Chapt 1.1" in the app
|
||||||
And I should find "This is a subchapter" in the app
|
And I should find "This is a subchapter" in the app
|
||||||
And I should find "2 / 4" in the app
|
And I should find "2 / 4" in the app
|
||||||
|
|
||||||
When I swipe to the left in "Chapt 1.1" "ion-slides" in the app
|
When I swipe to the left in "Chapt 1.1" "swiper-container" in the app
|
||||||
Then I should find "Chapt 2" in the app
|
Then I should find "Chapt 2" in the app
|
||||||
And I should find "This is the second chapter" in the app
|
And I should find "This is the second chapter" in the app
|
||||||
And I should find "3 / 4" in the app
|
And I should find "3 / 4" in the app
|
||||||
|
@ -208,22 +208,22 @@ Scenario: View and navigate book contents (teacher)
|
||||||
But I should not find "This is the first chapter" in the app
|
But I should not find "This is the first chapter" in the app
|
||||||
|
|
||||||
# Navigate using swipe.
|
# Navigate using swipe.
|
||||||
When I swipe to the left in "Hidden subchapter" "ion-slides" in the app
|
When I swipe to the left in "Hidden subchapter" "swiper-container" in the app
|
||||||
Then I should find "Chapt 3" in the app
|
Then I should find "Chapt 3" in the app
|
||||||
And I should find "This is the third chapter" in the app
|
And I should find "This is the third chapter" in the app
|
||||||
And I should find "6 / 7" in the app
|
And I should find "6 / 7" in the app
|
||||||
|
|
||||||
When I swipe to the left in "Chapt 3" "ion-slides" in the app
|
When I swipe to the left in "Chapt 3" "swiper-container" in the app
|
||||||
Then I should find "Last hidden" in the app
|
Then I should find "Last hidden" in the app
|
||||||
And I should find "Another hidden subchapter" in the app
|
And I should find "Another hidden subchapter" in the app
|
||||||
And I should find "7 / 7" in the app
|
And I should find "7 / 7" in the app
|
||||||
|
|
||||||
When I swipe to the left in "Last hidden" "ion-slides" in the app
|
When I swipe to the left in "Last hidden" "swiper-container" in the app
|
||||||
Then I should find "Last hidden" in the app
|
Then I should find "Last hidden" in the app
|
||||||
And I should find "Another hidden subchapter" in the app
|
And I should find "Another hidden subchapter" in the app
|
||||||
And I should find "7 / 7" in the app
|
And I should find "7 / 7" in the app
|
||||||
|
|
||||||
When I swipe to the right in "Last hidden" "ion-slides" in the app
|
When I swipe to the right in "Last hidden" "swiper-container" in the app
|
||||||
Then I should find "Chapt 3" in the app
|
Then I should find "Chapt 3" in the app
|
||||||
And I should find "This is the third chapter" in the app
|
And I should find "This is the third chapter" in the app
|
||||||
And I should find "6 / 7" in the app
|
And I should find "6 / 7" in the app
|
||||||
|
|
|
@ -35,11 +35,14 @@ import { CorePlatform } from '@services/platform';
|
||||||
import { CoreUrl } from '@singletons/url';
|
import { CoreUrl } from '@singletons/url';
|
||||||
import { CoreLogger } from '@singletons/logger';
|
import { CoreLogger } from '@singletons/logger';
|
||||||
import { CorePromisedValue } from '@classes/promised-value';
|
import { CorePromisedValue } from '@classes/promised-value';
|
||||||
|
import { register } from 'swiper/element/bundle';
|
||||||
|
|
||||||
const MOODLE_SITE_URL_PREFIX = 'url-';
|
const MOODLE_SITE_URL_PREFIX = 'url-';
|
||||||
const MOODLE_VERSION_PREFIX = 'version-';
|
const MOODLE_VERSION_PREFIX = 'version-';
|
||||||
const MOODLEAPP_VERSION_PREFIX = 'moodleapp-';
|
const MOODLEAPP_VERSION_PREFIX = 'moodleapp-';
|
||||||
|
|
||||||
|
register();
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-root',
|
selector: 'app-root',
|
||||||
templateUrl: 'app.component.html',
|
templateUrl: 'app.component.html',
|
||||||
|
|
|
@ -25,7 +25,6 @@ import {
|
||||||
SimpleChange,
|
SimpleChange,
|
||||||
ElementRef,
|
ElementRef,
|
||||||
} from '@angular/core';
|
} from '@angular/core';
|
||||||
import { IonSlides } from '@ionic/angular';
|
|
||||||
import { BackButtonEvent } from '@ionic/core';
|
import { BackButtonEvent } from '@ionic/core';
|
||||||
import { Subscription } from 'rxjs';
|
import { Subscription } from 'rxjs';
|
||||||
|
|
||||||
|
@ -40,6 +39,9 @@ import { CorePromisedValue } from './promised-value';
|
||||||
import { AsyncDirective } from './async-directive';
|
import { AsyncDirective } from './async-directive';
|
||||||
import { CoreDirectivesRegistry } from '@singletons/directives-registry';
|
import { CoreDirectivesRegistry } from '@singletons/directives-registry';
|
||||||
import { CorePlatform } from '@services/platform';
|
import { CorePlatform } from '@services/platform';
|
||||||
|
import { Swiper } from 'swiper';
|
||||||
|
import { SwiperOptions } from 'swiper/types';
|
||||||
|
import { IonicSlides } from '@ionic/angular';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class to abstract some common code for tabs.
|
* Class to abstract some common code for tabs.
|
||||||
|
@ -56,7 +58,34 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
@Input() hideUntil = false; // Determine when should the contents be shown.
|
@Input() hideUntil = false; // Determine when should the contents be shown.
|
||||||
@Output() protected ionChange = new EventEmitter<T>(); // Emitted when the tab changes.
|
@Output() protected ionChange = new EventEmitter<T>(); // Emitted when the tab changes.
|
||||||
|
|
||||||
@ViewChild(IonSlides) protected slides?: IonSlides;
|
protected swiper?: Swiper;
|
||||||
|
@ViewChild('swiperRef')
|
||||||
|
set swiperRef(swiperRef: ElementRef) {
|
||||||
|
/**
|
||||||
|
* This setTimeout waits for Ionic's async initialization to complete.
|
||||||
|
* Otherwise, an outdated swiper reference will be used.
|
||||||
|
*/
|
||||||
|
setTimeout(() => {
|
||||||
|
if (swiperRef?.nativeElement?.swiper && !this.swiper) {
|
||||||
|
this.swiper = swiperRef.nativeElement.swiper as Swiper;
|
||||||
|
|
||||||
|
this.swiper.changeLanguageDirection(CorePlatform.isRTL ? 'rtl' : 'ltr');
|
||||||
|
|
||||||
|
Object.keys(this.swiperOpts).forEach((key) => {
|
||||||
|
if (this.swiper) {
|
||||||
|
this.swiper.params[key] = this.swiperOpts[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Subscribe to changes.
|
||||||
|
this.swiper.on('slideChangeTransitionEnd', () => {
|
||||||
|
this.slideChanged();
|
||||||
|
});
|
||||||
|
|
||||||
|
this.init();
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
|
||||||
tabs: T[] = []; // List of tabs.
|
tabs: T[] = []; // List of tabs.
|
||||||
|
|
||||||
|
@ -66,18 +95,14 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
showNextButton = false;
|
showNextButton = false;
|
||||||
maxSlides = 3;
|
maxSlides = 3;
|
||||||
numTabsShown = 0;
|
numTabsShown = 0;
|
||||||
direction = 'ltr';
|
|
||||||
description = '';
|
description = '';
|
||||||
slidesOpts = {
|
swiperOpts: SwiperOptions = {
|
||||||
initialSlide: 0,
|
modules: [IonicSlides],
|
||||||
slidesPerView: 3,
|
slidesPerView: 3,
|
||||||
centerInsufficientSlides: true,
|
centerInsufficientSlides: true,
|
||||||
threshold: 10,
|
threshold: 10,
|
||||||
};
|
};
|
||||||
|
|
||||||
protected slidesElement?: HTMLIonSlidesElement;
|
|
||||||
protected initialized = false;
|
|
||||||
|
|
||||||
protected resizeListener?: CoreEventObserver;
|
protected resizeListener?: CoreEventObserver;
|
||||||
protected isDestroyed = false;
|
protected isDestroyed = false;
|
||||||
protected isCurrentView = true;
|
protected isCurrentView = true;
|
||||||
|
@ -87,7 +112,7 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
|
|
||||||
protected firstSelectedTab?: string; // ID of the first selected tab to control history.
|
protected firstSelectedTab?: string; // ID of the first selected tab to control history.
|
||||||
protected backButtonFunction: (event: BackButtonEvent) => void;
|
protected backButtonFunction: (event: BackButtonEvent) => void;
|
||||||
// Swiper 6 documentation: https://swiper6.vercel.app/
|
// Swiper documentation: https://swiperjs.com/swiper-api
|
||||||
protected isInTransition = false; // Wether Slides is in transition.
|
protected isInTransition = false; // Wether Slides is in transition.
|
||||||
protected subscriptions: Subscription[] = [];
|
protected subscriptions: Subscription[] = [];
|
||||||
protected onReadyPromise = new CorePromisedValue<void>();
|
protected onReadyPromise = new CorePromisedValue<void>();
|
||||||
|
@ -106,12 +131,10 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
async ngOnInit(): Promise<void> {
|
async ngOnInit(): Promise<void> {
|
||||||
this.direction = CorePlatform.isRTL ? 'rtl' : 'ltr';
|
|
||||||
|
|
||||||
// Change the side when the language changes.
|
// Change the side when the language changes.
|
||||||
this.subscriptions.push(Translate.onLangChange.subscribe(() => {
|
this.subscriptions.push(Translate.onLangChange.subscribe(() => {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.direction = CorePlatform.isRTL ? 'rtl' : 'ltr';
|
this.swiper?.changeLanguageDirection(CorePlatform.isRTL ? 'rtl' : 'ltr');
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
@ -125,6 +148,10 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
}
|
}
|
||||||
|
|
||||||
this.init();
|
this.init();
|
||||||
|
|
||||||
|
this.resizeListener = CoreDom.onWindowResize(() => {
|
||||||
|
this.calculateSlides();
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -136,7 +163,7 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* User entered the page that contains the component.
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
ionViewDidEnter(): void {
|
ionViewDidEnter(): void {
|
||||||
this.isCurrentView = true;
|
this.isCurrentView = true;
|
||||||
|
@ -179,7 +206,7 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* User left the page that contains the component.
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
ionViewDidLeave(): void {
|
ionViewDidLeave(): void {
|
||||||
// Unregister the custom back button action for this component.
|
// Unregister the custom back button action for this component.
|
||||||
|
@ -189,16 +216,15 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate slides.
|
* Updates the number of slides to show.
|
||||||
*/
|
*/
|
||||||
protected async calculateSlides(): Promise<void> {
|
protected async calculateSlides(): Promise<void> {
|
||||||
if (!this.isCurrentView || !this.initialized) {
|
if (!this.isCurrentView || !this.swiper) {
|
||||||
// Don't calculate if component isn't in current view, the calculations are wrong.
|
// Don't calculate if component isn't in current view, the calculations are wrong.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.numTabsShown = this.tabs.reduce((prev: number, current) => current.enabled ? prev + 1 : prev, 0);
|
this.numTabsShown = this.tabs.reduce((prev: number, current) => current.enabled ? prev + 1 : prev, 0);
|
||||||
|
|
||||||
if (this.numTabsShown <= 1) {
|
if (this.numTabsShown <= 1) {
|
||||||
this.hideTabs = true;
|
this.hideTabs = true;
|
||||||
|
|
||||||
|
@ -209,7 +235,32 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
|
|
||||||
await this.calculateMaxSlides();
|
await this.calculateMaxSlides();
|
||||||
|
|
||||||
await this.updateSlides();
|
this.swiperOpts.slidesPerView = Math.min(this.maxSlides, this.numTabsShown);
|
||||||
|
|
||||||
|
this.slideChanged();
|
||||||
|
|
||||||
|
this.swiper.update();
|
||||||
|
await CoreUtils.nextTick();
|
||||||
|
|
||||||
|
if (!this.hasSliddenToInitial && this.selectedIndex && this.selectedIndex >= this.swiper.slidesPerViewDynamic()) {
|
||||||
|
this.hasSliddenToInitial = true;
|
||||||
|
this.shouldSlideToInitial = true;
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
if (this.shouldSlideToInitial) {
|
||||||
|
this.swiper?.slideTo(this.selectedIndex, 0);
|
||||||
|
this.shouldSlideToInitial = false;
|
||||||
|
}
|
||||||
|
}, 400);
|
||||||
|
|
||||||
|
return;
|
||||||
|
} else if (this.selectedIndex) {
|
||||||
|
this.hasSliddenToInitial = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
this.slideChanged(); // Call slide changed again, sometimes the slide active index takes a while to be updated.
|
||||||
|
}, 400);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -218,8 +269,12 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
* @param tabId Tab ID.
|
* @param tabId Tab ID.
|
||||||
* @returns Selected tab.
|
* @returns Selected tab.
|
||||||
*/
|
*/
|
||||||
protected getTabIndex(tabId: string): number {
|
protected getTabIndex(tabId?: string): number {
|
||||||
return this.tabs.findIndex((tab) => tabId == tab.id);
|
if (!tabId) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.tabs.findIndex((tab) => tabId === tab.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -228,89 +283,39 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
* @returns Selected tab.
|
* @returns Selected tab.
|
||||||
*/
|
*/
|
||||||
getSelected(): T | undefined {
|
getSelected(): T | undefined {
|
||||||
const index = this.selected && this.getTabIndex(this.selected);
|
const index = this.getTabIndex(this.selected);
|
||||||
|
|
||||||
return index !== undefined && index >= 0 ? this.tabs[index] : undefined;
|
return index >= 0 ? this.tabs[index] : undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Init the component.
|
* Init the component.
|
||||||
*/
|
*/
|
||||||
protected async init(): Promise<void> {
|
protected async init(): Promise<void> {
|
||||||
if (!this.hideUntil) {
|
if (!this.hideUntil || !this.swiper) {
|
||||||
// Hidden, do nothing.
|
// Hidden, do nothing.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await this.initializeSlider();
|
const selectedTab = this.calculateInitialTab();
|
||||||
await this.initializeTabs();
|
if (!selectedTab) {
|
||||||
|
// No enabled tabs, return.
|
||||||
|
throw new CoreError('No enabled tabs.');
|
||||||
|
}
|
||||||
|
|
||||||
|
this.firstSelectedTab = selectedTab.id;
|
||||||
|
if (this.firstSelectedTab !== undefined) {
|
||||||
|
this.selectTab(this.firstSelectedTab);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check which arrows should be shown.
|
||||||
|
this.calculateSlides();
|
||||||
} catch {
|
} catch {
|
||||||
// Something went wrong, ignore.
|
// Something went wrong, ignore.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Initialize the slider elements.
|
|
||||||
*/
|
|
||||||
protected async initializeSlider(): Promise<void> {
|
|
||||||
if (this.initialized) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.slidesElement) {
|
|
||||||
// Already initializated, await for ready.
|
|
||||||
await this.slidesElement.componentOnReady();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!this.slides) {
|
|
||||||
await CoreUtils.nextTick();
|
|
||||||
}
|
|
||||||
const slidesSwiper = await this.slides?.getSwiper();
|
|
||||||
if (!slidesSwiper || !this.slides) {
|
|
||||||
throw new CoreError('Swiper not found, will try on next change.');
|
|
||||||
}
|
|
||||||
|
|
||||||
this.slidesElement = <HTMLIonSlidesElement>slidesSwiper.el;
|
|
||||||
await this.slidesElement.componentOnReady();
|
|
||||||
|
|
||||||
this.initialized = true;
|
|
||||||
|
|
||||||
// Subscribe to changes.
|
|
||||||
this.subscriptions.push(this.slides.ionSlideDidChange.subscribe(() => {
|
|
||||||
this.slideChanged();
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Initialize the tabs, determining the first tab to be shown.
|
|
||||||
*/
|
|
||||||
protected async initializeTabs(): Promise<void> {
|
|
||||||
if (!this.initialized || !this.slidesElement) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const selectedTab = this.calculateInitialTab();
|
|
||||||
if (!selectedTab) {
|
|
||||||
// No enabled tabs, return.
|
|
||||||
throw new CoreError('No enabled tabs.');
|
|
||||||
}
|
|
||||||
|
|
||||||
this.firstSelectedTab = selectedTab.id;
|
|
||||||
if (this.firstSelectedTab !== undefined) {
|
|
||||||
this.selectTab(this.firstSelectedTab);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check which arrows should be shown.
|
|
||||||
this.calculateSlides();
|
|
||||||
|
|
||||||
this.resizeListener = CoreDom.onWindowResize(() => {
|
|
||||||
this.calculateSlides();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate the initial tab to load.
|
* Calculate the initial tab to load.
|
||||||
*
|
*
|
||||||
|
@ -330,116 +335,71 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
/**
|
/**
|
||||||
* Method executed when the slides are changed.
|
* Method executed when the slides are changed.
|
||||||
*/
|
*/
|
||||||
async slideChanged(): Promise<void> {
|
slideChanged(): void {
|
||||||
if (!this.slidesElement) {
|
if (!this.swiper) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.isInTransition = false;
|
this.isInTransition = false;
|
||||||
const slidesCount = await this.slides?.length() || 0;
|
const slidesCount = this.swiper.slides.length || 0;
|
||||||
if (slidesCount > 0) {
|
if (slidesCount > 0) {
|
||||||
this.showPrevButton = !await this.slides?.isBeginning();
|
this.showPrevButton = !this.swiper.isBeginning;
|
||||||
this.showNextButton = !await this.slides?.isEnd();
|
this.showNextButton = !this.swiper.isEnd;
|
||||||
} else {
|
} else {
|
||||||
this.showPrevButton = false;
|
this.showPrevButton = false;
|
||||||
this.showNextButton = false;
|
this.showNextButton = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentIndex = await this.slides?.getActiveIndex();
|
const currentIndex = this.swiper.activeIndex;
|
||||||
if (this.shouldSlideToInitial && currentIndex != this.selectedIndex) {
|
if (this.shouldSlideToInitial && currentIndex != this.selectedIndex) {
|
||||||
// Current tab has changed, don't slide to initial anymore.
|
// Current tab has changed, don't slide to initial anymore.
|
||||||
this.shouldSlideToInitial = false;
|
this.shouldSlideToInitial = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Updates the number of slides to show.
|
|
||||||
*/
|
|
||||||
protected async updateSlides(): Promise<void> {
|
|
||||||
if (!this.slides) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.slidesOpts = { ...this.slidesOpts, slidesPerView: Math.min(this.maxSlides, this.numTabsShown) };
|
|
||||||
|
|
||||||
await this.slideChanged();
|
|
||||||
|
|
||||||
await this.slides.update();
|
|
||||||
|
|
||||||
if (!this.hasSliddenToInitial && this.selectedIndex && this.selectedIndex >= this.slidesOpts.slidesPerView) {
|
|
||||||
this.hasSliddenToInitial = true;
|
|
||||||
this.shouldSlideToInitial = true;
|
|
||||||
|
|
||||||
setTimeout(() => {
|
|
||||||
if (this.shouldSlideToInitial) {
|
|
||||||
this.slides?.slideTo(this.selectedIndex, 0);
|
|
||||||
this.shouldSlideToInitial = false;
|
|
||||||
}
|
|
||||||
}, 400);
|
|
||||||
|
|
||||||
return;
|
|
||||||
} else if (this.selectedIndex) {
|
|
||||||
this.hasSliddenToInitial = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
setTimeout(() => {
|
|
||||||
this.slideChanged(); // Call slide changed again, sometimes the slide active index takes a while to be updated.
|
|
||||||
}, 400);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate the number of slides that can fit on the screen.
|
* Calculate the number of slides that can fit on the screen.
|
||||||
*/
|
*/
|
||||||
protected async calculateMaxSlides(): Promise<void> {
|
protected async calculateMaxSlides(): Promise<void> {
|
||||||
if (!this.slidesElement || !this.slides) {
|
if (!this.swiper) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.maxSlides = 3;
|
this.maxSlides = 3;
|
||||||
await CoreUtils.nextTick();
|
await CoreUtils.nextTick();
|
||||||
|
|
||||||
let width: number = this.slidesElement.getBoundingClientRect().width;
|
if (!this.swiper.width) {
|
||||||
if (!width) {
|
return;
|
||||||
const slidesSwiper = await this.slides.getSwiper();
|
|
||||||
|
|
||||||
await slidesSwiper.updateSize();
|
|
||||||
await CoreUtils.nextTick();
|
|
||||||
|
|
||||||
width = slidesSwiper.width;
|
|
||||||
if (!width) {
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const zoomLevel = await CoreSettingsHelper.getZoom();
|
const zoomLevel = await CoreSettingsHelper.getZoom();
|
||||||
|
|
||||||
this.maxSlides = Math.floor(width / (zoomLevel / 100 * CoreTabsBaseComponent.MIN_TAB_WIDTH));
|
this.maxSlides = Math.floor(this.swiper.width / (zoomLevel / 100 * CoreTabsBaseComponent.MIN_TAB_WIDTH));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method that shows the next tab.
|
* Method that shows the next tab.
|
||||||
*/
|
*/
|
||||||
async slideNext(): Promise<void> {
|
slideNext(): void {
|
||||||
// Stop if slides are in transition.
|
// Stop if slides are in transition.
|
||||||
if (!this.showNextButton || this.isInTransition || !this.slides) {
|
if (!this.showNextButton || this.isInTransition || !this.swiper) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (await this.slides.isBeginning()) {
|
if (this.swiper.isBeginning) {
|
||||||
// Slide to the second page.
|
// Slide to the second page.
|
||||||
this.slides.slideTo(this.maxSlides);
|
this.swiper.slideTo(this.maxSlides);
|
||||||
} else {
|
} else {
|
||||||
const currentIndex = await this.slides.getActiveIndex();
|
const currentIndex = this.swiper.activeIndex;
|
||||||
if (currentIndex !== undefined) {
|
if (currentIndex !== undefined) {
|
||||||
const nextSlideIndex = currentIndex + this.maxSlides;
|
const nextSlideIndex = currentIndex + this.maxSlides;
|
||||||
this.isInTransition = true;
|
this.isInTransition = true;
|
||||||
if (nextSlideIndex < this.numTabsShown) {
|
if (nextSlideIndex < this.numTabsShown) {
|
||||||
// Slide to the next page.
|
// Slide to the next page.
|
||||||
await this.slides.slideTo(nextSlideIndex);
|
this.swiper.slideTo(nextSlideIndex);
|
||||||
} else {
|
} else {
|
||||||
// Slide to the latest slide.
|
// Slide to the latest slide.
|
||||||
await this.slides.slideTo(this.numTabsShown - 1);
|
this.swiper.slideTo(this.numTabsShown - 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -449,26 +409,26 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
/**
|
/**
|
||||||
* Method that shows the previous tab.
|
* Method that shows the previous tab.
|
||||||
*/
|
*/
|
||||||
async slidePrev(): Promise<void> {
|
slidePrev(): void {
|
||||||
// Stop if slides are in transition.
|
// Stop if slides are in transition.
|
||||||
if (!this.showPrevButton || this.isInTransition || !this.slides) {
|
if (!this.showPrevButton || this.isInTransition || !this.swiper) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (await this.slides.isEnd()) {
|
if (this.swiper.isEnd) {
|
||||||
this.slides.slideTo(this.numTabsShown - this.maxSlides * 2);
|
this.swiper.slideTo(this.numTabsShown - this.maxSlides * 2);
|
||||||
// Slide to the previous of the latest page.
|
// Slide to the previous of the latest page.
|
||||||
} else {
|
} else {
|
||||||
const currentIndex = await this.slides.getActiveIndex();
|
const currentIndex = this.swiper.activeIndex;
|
||||||
if (currentIndex !== undefined) {
|
if (currentIndex !== undefined) {
|
||||||
const prevSlideIndex = currentIndex - this.maxSlides;
|
const prevSlideIndex = currentIndex - this.maxSlides;
|
||||||
this.isInTransition = true;
|
this.isInTransition = true;
|
||||||
if (prevSlideIndex >= 0) {
|
if (prevSlideIndex >= 0) {
|
||||||
// Slide to the previous page.
|
// Slide to the previous page.
|
||||||
await this.slides.slideTo(prevSlideIndex);
|
this.swiper.slideTo(prevSlideIndex);
|
||||||
} else {
|
} else {
|
||||||
// Slide to the first page.
|
// Slide to the first page.
|
||||||
await this.slides.slideTo(0);
|
this.swiper.slideTo(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -517,12 +477,12 @@ export class CoreTabsBaseComponent<T extends CoreTabBase> implements OnInit, Aft
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.selected && this.slides) {
|
if (this.selected && this.swiper) {
|
||||||
// Check if we need to slide to the tab because it's not visible.
|
// Check if we need to slide to the tab because it's not visible.
|
||||||
const firstVisibleTab = await this.slides.getActiveIndex();
|
const firstVisibleTab = this.swiper.activeIndex;
|
||||||
const lastVisibleTab = firstVisibleTab + this.slidesOpts.slidesPerView - 1;
|
const lastVisibleTab = firstVisibleTab + this.swiper.slidesPerViewDynamic() - 1;
|
||||||
if (index < firstVisibleTab || index > lastVisibleTab) {
|
if (index < firstVisibleTab || index > lastVisibleTab) {
|
||||||
await this.slides.slideTo(index, 0, true);
|
this.swiper.slideTo(index, 0, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
import { NgModule } from '@angular/core';
|
import { CUSTOM_ELEMENTS_SCHEMA, NgModule } from '@angular/core';
|
||||||
import { CommonModule } from '@angular/common';
|
import { CommonModule } from '@angular/common';
|
||||||
import { IonicModule } from '@ionic/angular';
|
import { IonicModule } from '@ionic/angular';
|
||||||
import { TranslateModule } from '@ngx-translate/core';
|
import { TranslateModule } from '@ngx-translate/core';
|
||||||
|
@ -170,5 +170,6 @@ import { CoreSitesListComponent } from './sites-list/sites-list';
|
||||||
CoreSheetModalComponent,
|
CoreSheetModalComponent,
|
||||||
CoreSitesListComponent,
|
CoreSitesListComponent,
|
||||||
],
|
],
|
||||||
|
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
||||||
})
|
})
|
||||||
export class CoreComponentsModule {}
|
export class CoreComponentsModule {}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<ion-slides *ngIf="loaded" (ionSlideWillChange)="slideWillChange()" (ionSlideDidChange)="slideDidChange()" [options]="options">
|
<swiper-container #swiperRef *ngIf="loaded" (slidechangetransitionstart)="slideWillChange()" (slidechangetransitionend)="slideDidChange()"
|
||||||
<ion-slide *ngFor="let item of items; index as index" [attr.aria-hidden]="!isActive(index)">
|
[initialSlide]="options.initialSlide" [runCallbacksOnInit]="options.runCallbacksOnInit">
|
||||||
|
<swiper-slide *ngFor="let item of items; index as index" [attr.aria-hidden]="!isActive(index)">
|
||||||
<ng-container *ngIf="template" [ngTemplateOutlet]="template" [ngTemplateOutletContext]="{item: item, active: isActive(index)}">
|
<ng-container *ngIf="template" [ngTemplateOutlet]="template" [ngTemplateOutletContext]="{item: item, active: isActive(index)}">
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
</ion-slides>
|
</swiper-container>
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
:host {
|
:host {
|
||||||
ion-slides {
|
swiper-container {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
|
||||||
|
|
||||||
ion-slide {
|
swiper-slide {
|
||||||
display: block;
|
display: block;
|
||||||
font-size: inherit;
|
font-size: inherit;
|
||||||
justify-content: start;
|
justify-content: start;
|
||||||
align-items: start;
|
align-items: start;
|
||||||
text-align: start;
|
text-align: start;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,16 +13,17 @@
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, Output, TemplateRef, ViewChild,
|
Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, Output, SimpleChange, TemplateRef, ViewChild,
|
||||||
} from '@angular/core';
|
} from '@angular/core';
|
||||||
import { CoreSwipeSlidesItemsManager } from '@classes/items-management/swipe-slides-items-manager';
|
import { CoreSwipeSlidesItemsManager } from '@classes/items-management/swipe-slides-items-manager';
|
||||||
import { IonContent, IonSlides } from '@ionic/angular';
|
import { IonContent } from '@ionic/angular';
|
||||||
import { CoreDomUtils, VerticalPoint } from '@services/utils/dom';
|
import { CoreDomUtils, VerticalPoint } from '@services/utils/dom';
|
||||||
import { CoreUtils } from '@services/utils/utils';
|
import { CoreUtils } from '@services/utils/utils';
|
||||||
import { CoreDom } from '@singletons/dom';
|
import { CoreDom } from '@singletons/dom';
|
||||||
import { CoreEventObserver } from '@singletons/events';
|
import { CoreEventObserver } from '@singletons/events';
|
||||||
import { CoreMath } from '@singletons/math';
|
import { CoreMath } from '@singletons/math';
|
||||||
|
import { Swiper } from 'swiper';
|
||||||
|
import { SwiperOptions } from 'swiper/types';
|
||||||
/**
|
/**
|
||||||
* Helper component to display swipable slides.
|
* Helper component to display swipable slides.
|
||||||
*/
|
*/
|
||||||
|
@ -38,13 +39,31 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
@Output() onWillChange = new EventEmitter<CoreSwipeCurrentItemData<Item>>();
|
@Output() onWillChange = new EventEmitter<CoreSwipeCurrentItemData<Item>>();
|
||||||
@Output() onDidChange = new EventEmitter<CoreSwipeCurrentItemData<Item>>();
|
@Output() onDidChange = new EventEmitter<CoreSwipeCurrentItemData<Item>>();
|
||||||
|
|
||||||
@ViewChild(IonSlides) slides?: IonSlides;
|
protected swiper?: Swiper;
|
||||||
|
@ViewChild('swiperRef')
|
||||||
|
set swiperRef(swiperRef: ElementRef) {
|
||||||
|
/**
|
||||||
|
* This setTimeout waits for Ionic's async initialization to complete.
|
||||||
|
* Otherwise, an outdated swiper reference will be used.
|
||||||
|
*/
|
||||||
|
setTimeout(() => {
|
||||||
|
if (swiperRef?.nativeElement?.swiper) {
|
||||||
|
this.swiper = swiperRef.nativeElement.swiper as Swiper;
|
||||||
|
|
||||||
|
Object.keys(this.options).forEach((key) => {
|
||||||
|
if (this.swiper) {
|
||||||
|
this.swiper.params[key] = this.options[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
|
||||||
@ContentChild(TemplateRef) template?: TemplateRef<unknown>; // Template defined by the content.
|
@ContentChild(TemplateRef) template?: TemplateRef<unknown>; // Template defined by the content.
|
||||||
|
|
||||||
protected hostElement: HTMLElement;
|
protected hostElement: HTMLElement;
|
||||||
protected unsubscribe?: () => void;
|
protected unsubscribe?: () => void;
|
||||||
protected resizeListener: CoreEventObserver;
|
protected resizeListener: CoreEventObserver;
|
||||||
protected updateSlidesPromise?: Promise<void>;
|
|
||||||
protected activeSlideIndexes: number[] = [];
|
protected activeSlideIndexes: number[] = [];
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -53,18 +72,26 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
) {
|
) {
|
||||||
this.hostElement = elementRef.nativeElement;
|
this.hostElement = elementRef.nativeElement;
|
||||||
|
|
||||||
this.resizeListener = CoreDom.onWindowResize(async () => {
|
this.resizeListener = CoreDom.onWindowResize(() => {
|
||||||
await this.updateSlidesComponent();
|
this.updateSlidesComponent();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
ngOnChanges(): void {
|
ngOnChanges(changes: { [name: string]: SimpleChange }): void {
|
||||||
if (!this.unsubscribe && this.manager) {
|
if (!this.unsubscribe && this.manager) {
|
||||||
this.initialize(this.manager);
|
this.initialize(this.manager);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (changes.options) {
|
||||||
|
Object.keys(this.options).forEach((key) => {
|
||||||
|
if (this.swiper) {
|
||||||
|
this.swiper.params[key] = this.options[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get items(): Item[] {
|
get items(): Item[] {
|
||||||
|
@ -133,23 +160,20 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
* @param speed Animation speed.
|
* @param speed Animation speed.
|
||||||
* @param runCallbacks Whether to run callbacks.
|
* @param runCallbacks Whether to run callbacks.
|
||||||
*/
|
*/
|
||||||
async slideToIndex(index: number, speed?: number, runCallbacks?: boolean): Promise<void> {
|
slideToIndex(index: number, speed?: number, runCallbacks?: boolean): void {
|
||||||
// If slides are being updated, wait for the update to finish.
|
// If slides are being updated, wait for the update to finish.
|
||||||
await this.updateSlidesPromise;
|
if (!this.swiper) {
|
||||||
|
|
||||||
const slides = this.slides;
|
|
||||||
if (!slides) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verify that the number of slides matches the number of items.
|
// Verify that the number of slides matches the number of items.
|
||||||
const slidesLength = await slides.length();
|
const slidesLength = this.swiper.slides.length;
|
||||||
if (slidesLength !== this.items.length) {
|
if (slidesLength !== this.items.length) {
|
||||||
// Number doesn't match, do a new update to try to match them.
|
// Number doesn't match, do a new update to try to match them.
|
||||||
await this.updateSlidesComponent();
|
this.updateSlidesComponent();
|
||||||
}
|
}
|
||||||
|
|
||||||
this.slides?.slideTo(index, speed, runCallbacks);
|
this.swiper?.slideTo(index, speed, runCallbacks);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -173,7 +197,7 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
* @param runCallbacks Whether to run callbacks.
|
* @param runCallbacks Whether to run callbacks.
|
||||||
*/
|
*/
|
||||||
slideNext(speed?: number, runCallbacks?: boolean): void {
|
slideNext(speed?: number, runCallbacks?: boolean): void {
|
||||||
this.slides?.slideNext(speed, runCallbacks);
|
this.swiper?.slideNext(speed, runCallbacks);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -183,7 +207,7 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
* @param runCallbacks Whether to run callbacks.
|
* @param runCallbacks Whether to run callbacks.
|
||||||
*/
|
*/
|
||||||
slidePrev(speed?: number, runCallbacks?: boolean): void {
|
slidePrev(speed?: number, runCallbacks?: boolean): void {
|
||||||
this.slides?.slidePrev(speed, runCallbacks);
|
this.swiper?.slidePrev(speed, runCallbacks);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -194,7 +218,7 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
await CoreUtils.nextTick();
|
await CoreUtils.nextTick();
|
||||||
|
|
||||||
// Update the slides component so the slides list reflects the new items.
|
// Update the slides component so the slides list reflects the new items.
|
||||||
await this.updateSlidesComponent();
|
this.updateSlidesComponent();
|
||||||
|
|
||||||
const currentItem = this.manager?.getSelectedItem();
|
const currentItem = this.manager?.getSelectedItem();
|
||||||
|
|
||||||
|
@ -205,7 +229,7 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
// Keep the same slide in case the list has changed.
|
// Keep the same slide in case the list has changed.
|
||||||
const newIndex = this.manager.getSource().getItemIndex(currentItem) ?? -1;
|
const newIndex = this.manager.getSource().getItemIndex(currentItem) ?? -1;
|
||||||
if (newIndex != -1) {
|
if (newIndex != -1) {
|
||||||
this.slides?.slideTo(newIndex, 0, false);
|
this.swiper?.slideTo(newIndex, 0, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -270,11 +294,11 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
* @returns Promise resolved with current item data. Null if not found.
|
* @returns Promise resolved with current item data. Null if not found.
|
||||||
*/
|
*/
|
||||||
protected async getCurrentSlideItemData(): Promise<CoreSwipeCurrentItemData<Item> | null> {
|
protected async getCurrentSlideItemData(): Promise<CoreSwipeCurrentItemData<Item> | null> {
|
||||||
if (!this.slides || !this.manager) {
|
if (!this.swiper || !this.manager) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const index = await this.slides.getActiveIndex();
|
const index = this.swiper.activeIndex;
|
||||||
const items = this.manager.getSource().getItems();
|
const items = this.manager.getSource().getItems();
|
||||||
const currentItem = items && items[index];
|
const currentItem = items && items[index];
|
||||||
|
|
||||||
|
@ -291,19 +315,8 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
/**
|
/**
|
||||||
* Update slides component.
|
* Update slides component.
|
||||||
*/
|
*/
|
||||||
protected async updateSlidesComponent(): Promise<void> {
|
updateSlidesComponent(): void {
|
||||||
if (!this.slides) {
|
this.swiper?.update();
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const promise = this.slides.update();
|
|
||||||
this.updateSlidesPromise = promise;
|
|
||||||
|
|
||||||
await promise;
|
|
||||||
|
|
||||||
if (this.updateSlidesPromise === promise) {
|
|
||||||
delete this.updateSlidesPromise;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -321,8 +334,7 @@ export class CoreSwipeSlidesComponent<Item = unknown> implements OnChanges, OnDe
|
||||||
*
|
*
|
||||||
* @todo Change unknown with the right type once Swiper library is used.
|
* @todo Change unknown with the right type once Swiper library is used.
|
||||||
*/
|
*/
|
||||||
export type CoreSwipeSlidesOptions = Record<string, unknown> & {
|
export type CoreSwipeSlidesOptions = SwiperOptions & {
|
||||||
initialSlide?: number;
|
|
||||||
scrollOnChange?: 'top' | 'none'; // Scroll behaviour on change slide. By default, none.
|
scrollOnChange?: 'top' | 'none'; // Scroll behaviour on change slide. By default, none.
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -6,9 +6,9 @@
|
||||||
[attr.aria-label]="'core.previous' | translate">
|
[attr.aria-label]="'core.previous' | translate">
|
||||||
<ion-icon *ngIf="showPrevButton" name="fas-chevron-left" aria-hidden="true" slot="icon-only"></ion-icon>
|
<ion-icon *ngIf="showPrevButton" name="fas-chevron-left" aria-hidden="true" slot="icon-only"></ion-icon>
|
||||||
</ion-button>
|
</ion-button>
|
||||||
<ion-slides [options]="slidesOpts" [dir]="direction" role="tablist" [attr.aria-label]="description">
|
<swiper-container #swiperRef [slidesPerView]="swiperOpts.slidesPerView" role="tablist" [attr.aria-label]="description">
|
||||||
<ng-container *ngFor="let tab of tabs">
|
<ng-container *ngFor="let tab of tabs">
|
||||||
<ion-slide *ngIf="tab.id" role="presentation" [id]="tab.id + '-tab'" tabindex="-1"
|
<swiper-slide *ngIf="tab.id" role="presentation" [id]="tab.id + '-tab'" tabindex="-1"
|
||||||
[class.selected]="selected === tab.id" class="{{tab.class}}">
|
[class.selected]="selected === tab.id" class="{{tab.class}}">
|
||||||
<ion-tab-button (ionTabButtonClick)="selectTab(tab.id, $event)" (keydown)="tabAction.keyDown(tab.id, $event)"
|
<ion-tab-button (ionTabButtonClick)="selectTab(tab.id, $event)" (keydown)="tabAction.keyDown(tab.id, $event)"
|
||||||
(keyup)="tabAction.keyUp(tab.id, $event)" [tab]="tab.page" [layout]="layout" role="tab"
|
(keyup)="tabAction.keyUp(tab.id, $event)" [tab]="tab.page" [layout]="layout" role="tab"
|
||||||
|
@ -25,9 +25,9 @@
|
||||||
</ion-badge>
|
</ion-badge>
|
||||||
</ion-label>
|
</ion-label>
|
||||||
</ion-tab-button>
|
</ion-tab-button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</ion-slides>
|
</swiper-container>
|
||||||
<ion-button fill="clear" class="arrow-button" (click)="slideNext()" [disabled]="!showNextButton"
|
<ion-button fill="clear" class="arrow-button" (click)="slideNext()" [disabled]="!showNextButton"
|
||||||
[attr.aria-label]="'core.next' | translate">
|
[attr.aria-label]="'core.next' | translate">
|
||||||
<ion-icon *ngIf="showNextButton" name="fas-chevron-right" aria-hidden="true" slot="icon-only"></ion-icon>
|
<ion-icon *ngIf="showNextButton" name="fas-chevron-right" aria-hidden="true" slot="icon-only"></ion-icon>
|
||||||
|
|
|
@ -23,7 +23,6 @@ import {
|
||||||
SimpleChange,
|
SimpleChange,
|
||||||
} from '@angular/core';
|
} from '@angular/core';
|
||||||
import { IonRouterOutlet, IonTabs, ViewDidEnter, ViewDidLeave } from '@ionic/angular';
|
import { IonRouterOutlet, IonTabs, ViewDidEnter, ViewDidLeave } from '@ionic/angular';
|
||||||
import { Subscription } from 'rxjs';
|
|
||||||
|
|
||||||
import { CoreUtils } from '@services/utils/utils';
|
import { CoreUtils } from '@services/utils/utils';
|
||||||
import { Params } from '@angular/router';
|
import { Params } from '@angular/router';
|
||||||
|
@ -63,8 +62,6 @@ export class CoreTabsOutletComponent extends CoreTabsBaseComponent<CoreTabsOutle
|
||||||
|
|
||||||
@ViewChild(IonTabs) protected ionTabs!: IonTabs;
|
@ViewChild(IonTabs) protected ionTabs!: IonTabs;
|
||||||
|
|
||||||
protected stackEventsSubscription?: Subscription;
|
|
||||||
protected outletActivatedSubscription?: Subscription;
|
|
||||||
protected lastActiveComponent?: Partial<ViewDidLeave>;
|
protected lastActiveComponent?: Partial<ViewDidLeave>;
|
||||||
protected existsInNavigationStack = false;
|
protected existsInNavigationStack = false;
|
||||||
|
|
||||||
|
@ -90,7 +87,7 @@ export class CoreTabsOutletComponent extends CoreTabsBaseComponent<CoreTabsOutle
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.stackEventsSubscription = this.ionTabs.outlet.stackDidChange.subscribe(async (stackEvent: StackDidChangeEvent) => {
|
this.subscriptions.push(this.ionTabs.outlet.stackDidChange.subscribe(async (stackEvent: StackDidChangeEvent) => {
|
||||||
if (!this.isCurrentView) {
|
if (!this.isCurrentView) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -110,10 +107,10 @@ export class CoreTabsOutletComponent extends CoreTabsBaseComponent<CoreTabsOutle
|
||||||
}
|
}
|
||||||
|
|
||||||
this.showHideNavBarButtons(stackEvent.enteringView.element.tagName);
|
this.showHideNavBarButtons(stackEvent.enteringView.element.tagName);
|
||||||
});
|
}));
|
||||||
this.outletActivatedSubscription = this.ionTabs.outlet.activateEvents.subscribe(() => {
|
this.subscriptions.push(this.ionTabs.outlet.activateEvents.subscribe(() => {
|
||||||
this.lastActiveComponent = this.ionTabs.outlet.component;
|
this.lastActiveComponent = this.ionTabs.outlet.component;
|
||||||
});
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -221,8 +218,6 @@ export class CoreTabsOutletComponent extends CoreTabsBaseComponent<CoreTabsOutle
|
||||||
*/
|
*/
|
||||||
ngOnDestroy(): void {
|
ngOnDestroy(): void {
|
||||||
super.ngOnDestroy();
|
super.ngOnDestroy();
|
||||||
this.stackEventsSubscription?.unsubscribe();
|
|
||||||
this.outletActivatedSubscription?.unsubscribe();
|
|
||||||
this.existsInNavigationStack = false;
|
this.existsInNavigationStack = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,9 +5,9 @@
|
||||||
[attr.aria-label]="'core.previous' | translate">
|
[attr.aria-label]="'core.previous' | translate">
|
||||||
<ion-icon *ngIf="showPrevButton" name="fas-chevron-left" aria-hidden="true" slot="icon-only"></ion-icon>
|
<ion-icon *ngIf="showPrevButton" name="fas-chevron-left" aria-hidden="true" slot="icon-only"></ion-icon>
|
||||||
</ion-button>
|
</ion-button>
|
||||||
<ion-slides [options]="slidesOpts" [dir]="direction" role="tablist" [attr.aria-label]="description">
|
<swiper-container #swiperRef [slidesPerView]="swiperOpts.slidesPerView" role="tablist" [attr.aria-label]="description">
|
||||||
<ng-container *ngFor="let tab of tabs">
|
<ng-container *ngFor="let tab of tabs">
|
||||||
<ion-slide *ngIf="tab.enabled" role="presentation" [id]="tab.id! + '-tab'" [class.selected]="selected === tab.id"
|
<swiper-slide *ngIf="tab.enabled" role="presentation" [id]="tab.id! + '-tab'" [class.selected]="selected === tab.id"
|
||||||
class="{{tab.class}}">
|
class="{{tab.class}}">
|
||||||
<ion-tab-button (click)="selectTab(tab.id, $event)" (keydown)="tabAction.keyDown(tab.id, $event)"
|
<ion-tab-button (click)="selectTab(tab.id, $event)" (keydown)="tabAction.keyDown(tab.id, $event)"
|
||||||
(keyup)="tabAction.keyUp(tab.id, $event)" [layout]="layout" role="tab" [attr.aria-controls]="tab.id"
|
(keyup)="tabAction.keyUp(tab.id, $event)" [layout]="layout" role="tab" [attr.aria-controls]="tab.id"
|
||||||
|
@ -23,9 +23,9 @@
|
||||||
</ion-badge>
|
</ion-badge>
|
||||||
</ion-label>
|
</ion-label>
|
||||||
</ion-tab-button>
|
</ion-tab-button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
</ion-slides>
|
</swiper-container>
|
||||||
<ion-button fill="clear" class="arrow-button" (click)="slideNext()" [disabled]="!showNextButton"
|
<ion-button fill="clear" class="arrow-button" (click)="slideNext()" [disabled]="!showNextButton"
|
||||||
[attr.aria-label]="'core.next' | translate">
|
[attr.aria-label]="'core.next' | translate">
|
||||||
<ion-icon *ngIf="showNextButton" name="fas-chevron-right" aria-hidden="true" slot="icon-only"></ion-icon>
|
<ion-icon *ngIf="showNextButton" name="fas-chevron-right" aria-hidden="true" slot="icon-only"></ion-icon>
|
||||||
|
|
|
@ -44,12 +44,12 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ion-slides {
|
swiper-container {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
line-height: 1.6rem;
|
line-height: 1.6rem;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
|
||||||
ion-slide {
|
swiper-slide {
|
||||||
border-bottom: 2px solid transparent;
|
border-bottom: 2px solid transparent;
|
||||||
min-width: 100px;
|
min-width: 100px;
|
||||||
height: var(--height);
|
height: var(--height);
|
||||||
|
|
|
@ -47,7 +47,18 @@ export class CoreTabsComponent extends CoreTabsBaseComponent<CoreTabComponent> i
|
||||||
@Input() parentScrollable = false; // Determine if the scroll should be in the parent content or the tab itself.
|
@Input() parentScrollable = false; // Determine if the scroll should be in the parent content or the tab itself.
|
||||||
@Input() layout: 'icon-top' | 'icon-start' | 'icon-end' | 'icon-bottom' | 'icon-hide' | 'label-hide' = 'icon-hide';
|
@Input() layout: 'icon-top' | 'icon-start' | 'icon-end' | 'icon-bottom' | 'icon-hide' | 'label-hide' = 'icon-hide';
|
||||||
|
|
||||||
@ViewChild('originalTabs') originalTabsRef?: ElementRef;
|
@ViewChild('originalTabs')
|
||||||
|
set originalTabs(originalTabs: ElementRef) {
|
||||||
|
/**
|
||||||
|
* This setTimeout waits for Ionic's async initialization to complete.
|
||||||
|
* Otherwise, an outdated swiper reference will be used.
|
||||||
|
*/
|
||||||
|
setTimeout(() => {
|
||||||
|
if (originalTabs.nativeElement && !this.originalTabsContainer) {
|
||||||
|
this.originalTabsContainer = this.originalTabs?.nativeElement;
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
|
||||||
protected originalTabsContainer?: HTMLElement; // The container of the original tabs. It will include each tab's content.
|
protected originalTabsContainer?: HTMLElement; // The container of the original tabs. It will include each tab's content.
|
||||||
|
|
||||||
|
@ -60,15 +71,6 @@ export class CoreTabsComponent extends CoreTabsBaseComponent<CoreTabComponent> i
|
||||||
if (this.isDestroyed) {
|
if (this.isDestroyed) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.originalTabsContainer = this.originalTabsRef?.nativeElement;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Initialize the tabs, determining the first tab to be shown.
|
|
||||||
*/
|
|
||||||
protected async initializeTabs(): Promise<void> {
|
|
||||||
await super.initializeTabs();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
import { NgModule } from '@angular/core';
|
import { CUSTOM_ELEMENTS_SCHEMA, NgModule } from '@angular/core';
|
||||||
|
|
||||||
import { CoreEditorRichTextEditorComponent } from './rich-text-editor/rich-text-editor';
|
import { CoreEditorRichTextEditorComponent } from './rich-text-editor/rich-text-editor';
|
||||||
import { CoreSharedModule } from '@/core/shared.module';
|
import { CoreSharedModule } from '@/core/shared.module';
|
||||||
|
@ -29,5 +29,6 @@ import { CoreSharedModule } from '@/core/shared.module';
|
||||||
exports: [
|
exports: [
|
||||||
CoreEditorRichTextEditorComponent,
|
CoreEditorRichTextEditorComponent,
|
||||||
],
|
],
|
||||||
|
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
||||||
})
|
})
|
||||||
export class CoreEditorComponentsModule {}
|
export class CoreEditorComponentsModule {}
|
||||||
|
|
|
@ -21,103 +21,103 @@
|
||||||
[attr.aria-label]="'core.previous' | translate" [tabindex]="toolbarPrevHidden ? -1 : 0">
|
[attr.aria-label]="'core.previous' | translate" [tabindex]="toolbarPrevHidden ? -1 : 0">
|
||||||
<ion-icon name="fas-chevron-left" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-chevron-left" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
<ion-slides [options]="slidesOpts" [dir]="direction" (ionSlideDidChange)="updateToolbarArrows()">
|
<swiper-container #swiperRef [slidesPerView]="swiperOpts.slidesPerView" (slidechangetransitionend)="updateToolbarArrows()">
|
||||||
<!-- https://developer.mozilla.org/en-US/docs/Web/API/Document/execCommand -->
|
<!-- https://developer.mozilla.org/en-US/docs/Web/API/Document/execCommand -->
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.strong" [title]="'core.editor.bold' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.strong" [title]="'core.editor.bold' | translate"
|
||||||
(click)="buttonAction($event, 'bold', 'strong')" (keyup)="buttonAction($event, 'bold', 'strong')"
|
(click)="buttonAction($event, 'bold', 'strong')" (keyup)="buttonAction($event, 'bold', 'strong')"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-bold" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-bold" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.em" [title]="'core.editor.italic' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.em" [title]="'core.editor.italic' | translate"
|
||||||
(click)="buttonAction($event, 'italic', 'em')" (keyup)="buttonAction($event, 'italic', 'em')"
|
(click)="buttonAction($event, 'italic', 'em')" (keyup)="buttonAction($event, 'italic', 'em')"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-italic" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-italic" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.u" [title]="'core.editor.underline' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.u" [title]="'core.editor.underline' | translate"
|
||||||
(click)="buttonAction($event, 'underline', 'u')" (keyup)="buttonAction($event, 'underline', 'u')"
|
(click)="buttonAction($event, 'underline', 'u')" (keyup)="buttonAction($event, 'underline', 'u')"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-underline" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-underline" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.strike" [title]="'core.editor.strikethrough' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.strike" [title]="'core.editor.strikethrough' | translate"
|
||||||
(click)="buttonAction($event, 'strikethrough', 'strike')" (keyup)="buttonAction($event, 'strikethrough', 'strike')"
|
(click)="buttonAction($event, 'strikethrough', 'strike')" (keyup)="buttonAction($event, 'strikethrough', 'strike')"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-strikethrough" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-strikethrough" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.p" [title]="'core.editor.p' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.p" [title]="'core.editor.p' | translate"
|
||||||
(click)="buttonAction($event, 'p', 'block')" (keyup)="buttonAction($event, 'p', 'block')" (mousedown)="downAction($event)"
|
(click)="buttonAction($event, 'p', 'block')" (keyup)="buttonAction($event, 'p', 'block')" (mousedown)="downAction($event)"
|
||||||
(keydown)="downAction($event)" tabindex="0">
|
(keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-paragraph" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-paragraph" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.h3" [title]="'core.editor.h3' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.h3" [title]="'core.editor.h3' | translate"
|
||||||
(click)="buttonAction($event, 'h3', 'block')" (keyup)="buttonAction($event, 'h3', 'block')" (mousedown)="downAction($event)"
|
(click)="buttonAction($event, 'h3', 'block')" (keyup)="buttonAction($event, 'h3', 'block')" (mousedown)="downAction($event)"
|
||||||
(keydown)="downAction($event)" tabindex="0">
|
(keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-heading" aria-hidden="true"></ion-icon><span aria-hidden="true">3</span>
|
<ion-icon name="fas-heading" aria-hidden="true"></ion-icon><span aria-hidden="true">3</span>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.h4" [title]="'core.editor.h4' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.h4" [title]="'core.editor.h4' | translate"
|
||||||
(click)="buttonAction($event, 'h4', 'block')" (keyup)="buttonAction($event, 'h4', 'block')" (mousedown)="downAction($event)"
|
(click)="buttonAction($event, 'h4', 'block')" (keyup)="buttonAction($event, 'h4', 'block')" (mousedown)="downAction($event)"
|
||||||
(keydown)="downAction($event)" tabindex="0">
|
(keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-heading" aria-hidden="true"></ion-icon><span aria-hidden="true">4</span>
|
<ion-icon name="fas-heading" aria-hidden="true"></ion-icon><span aria-hidden="true">4</span>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.h5" [title]="'core.editor.h5' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.h5" [title]="'core.editor.h5' | translate"
|
||||||
(click)="buttonAction($event, 'h5', 'block')" (keyup)="buttonAction($event, 'h5', 'block')" (mousedown)="downAction($event)"
|
(click)="buttonAction($event, 'h5', 'block')" (keyup)="buttonAction($event, 'h5', 'block')" (mousedown)="downAction($event)"
|
||||||
(keydown)="downAction($event)" tabindex="0">
|
(keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-heading" aria-hidden="true"></ion-icon><span aria-hidden="true">5</span>
|
<ion-icon name="fas-heading" aria-hidden="true"></ion-icon><span aria-hidden="true">5</span>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.ul" [title]="'core.editor.unorderedlist' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.ul" [title]="'core.editor.unorderedlist' | translate"
|
||||||
(click)="buttonAction($event, 'insertUnorderedList')" (mousedown)="downAction($event)" (keydown)="downAction($event)"
|
(click)="buttonAction($event, 'insertUnorderedList')" (mousedown)="downAction($event)" (keydown)="downAction($event)"
|
||||||
tabindex="0">
|
tabindex="0">
|
||||||
<ion-icon name="fas-list-ul" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-list-ul" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.ol" [title]="'core.editor.orderedlist' | translate"
|
<button [disabled]="!rteEnabled" [attr.aria-pressed]="toolbarStyles.ol" [title]="'core.editor.orderedlist' | translate"
|
||||||
(click)="buttonAction($event, 'insertOrderedList')" (keyup)="buttonAction($event, 'insertOrderedList')"
|
(click)="buttonAction($event, 'insertOrderedList')" (keyup)="buttonAction($event, 'insertOrderedList')"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-list-ol" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-list-ol" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [disabled]="!rteEnabled" (click)="buttonAction($event, 'removeFormat')" (keyup)="buttonAction($event, 'removeFormat')"
|
<button [disabled]="!rteEnabled" (click)="buttonAction($event, 'removeFormat')" (keyup)="buttonAction($event, 'removeFormat')"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" [title]="'core.editor.clear' | translate">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" [title]="'core.editor.clear' | translate">
|
||||||
<ion-icon name="fas-eraser" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-eraser" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide *ngIf="canScanQR">
|
<swiper-slide *ngIf="canScanQR">
|
||||||
<button [disabled]="!rteEnabled" (click)="scanQR($event)" (keyup)="scanQR($event)" (mousedown)="stopBubble($event)"
|
<button [disabled]="!rteEnabled" (click)="scanQR($event)" (keyup)="scanQR($event)" (mousedown)="stopBubble($event)"
|
||||||
(keydown)="stopBubble($event)" [title]="'core.scanqr' | translate">
|
(keydown)="stopBubble($event)" [title]="'core.scanqr' | translate">
|
||||||
<ion-icon name="fas-qrcode" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-qrcode" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<button [attr.aria-pressed]="!rteEnabled" [title]="'core.editor.toggle' | translate" (click)="toggleEditor($event)"
|
<button [attr.aria-pressed]="!rteEnabled" [title]="'core.editor.toggle' | translate" (click)="toggleEditor($event)"
|
||||||
(keyup)="toggleEditor($event)" (mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(keyup)="toggleEditor($event)" (mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-code" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-code" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
<ion-slide *ngIf="isPhone">
|
<swiper-slide *ngIf="isPhone">
|
||||||
<button [title]="'core.editor.hidetoolbar' | translate" (click)="hideToolbar($event, true)" (keyup)="hideToolbar($event, true)"
|
<button [title]="'core.editor.hidetoolbar' | translate" (click)="hideToolbar($event, true)" (keyup)="hideToolbar($event, true)"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" tabindex="0">
|
||||||
<ion-icon name="fas-xmark" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-xmark" aria-hidden="true"></ion-icon>
|
||||||
</button>
|
</button>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
</ion-slides>
|
</swiper-container>
|
||||||
<button *ngIf="toolbarArrows" class="toolbar-arrow" [attr.disabled]="toolbarNextHidden ? 'true' : null"
|
<button *ngIf="toolbarArrows" class="toolbar-arrow" [attr.disabled]="toolbarNextHidden ? 'true' : null"
|
||||||
[attr.aria-label]="'core.next' | translate" (click)="toolbarNext($event)" (keyup)="toolbarNext($event)"
|
[attr.aria-label]="'core.next' | translate" (click)="toolbarNext($event)" (keyup)="toolbarNext($event)"
|
||||||
(mousedown)="downAction($event)" (keydown)="downAction($event)" [tabindex]="toolbarNextHidden ? -1 : 0">
|
(mousedown)="downAction($event)" (keydown)="downAction($event)" [tabindex]="toolbarNextHidden ? -1 : 0">
|
||||||
|
|
|
@ -108,7 +108,7 @@
|
||||||
padding-top: 5px;
|
padding-top: 5px;
|
||||||
border-top: 1px solid var(--stroke);
|
border-top: 1px solid var(--stroke);
|
||||||
|
|
||||||
ion-slides {
|
swiper-container {
|
||||||
width: 240px;
|
width: 240px;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
flex-shrink: 1;
|
flex-shrink: 1;
|
||||||
|
|
|
@ -25,7 +25,7 @@ import {
|
||||||
AfterViewInit,
|
AfterViewInit,
|
||||||
} from '@angular/core';
|
} from '@angular/core';
|
||||||
import { FormControl } from '@angular/forms';
|
import { FormControl } from '@angular/forms';
|
||||||
import { IonTextarea, IonContent, IonSlides } from '@ionic/angular';
|
import { IonTextarea, IonContent, IonicSlides } from '@ionic/angular';
|
||||||
import { Subscription } from 'rxjs';
|
import { Subscription } from 'rxjs';
|
||||||
|
|
||||||
import { CoreSites } from '@services/sites';
|
import { CoreSites } from '@services/sites';
|
||||||
|
@ -42,6 +42,8 @@ import { CoreScreen } from '@services/screen';
|
||||||
import { CoreCancellablePromise } from '@classes/cancellable-promise';
|
import { CoreCancellablePromise } from '@classes/cancellable-promise';
|
||||||
import { CoreDom } from '@singletons/dom';
|
import { CoreDom } from '@singletons/dom';
|
||||||
import { CorePlatform } from '@services/platform';
|
import { CorePlatform } from '@services/platform';
|
||||||
|
import { Swiper } from 'swiper';
|
||||||
|
import { SwiperOptions } from 'swiper/types';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Component to display a rich text editor if enabled.
|
* Component to display a rich text editor if enabled.
|
||||||
|
@ -78,7 +80,27 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
@ViewChild('editor') editor?: ElementRef; // WYSIWYG editor.
|
@ViewChild('editor') editor?: ElementRef; // WYSIWYG editor.
|
||||||
@ViewChild('textarea') textarea?: IonTextarea; // Textarea editor.
|
@ViewChild('textarea') textarea?: IonTextarea; // Textarea editor.
|
||||||
@ViewChild('toolbar') toolbar?: ElementRef;
|
@ViewChild('toolbar') toolbar?: ElementRef;
|
||||||
@ViewChild(IonSlides) toolbarSlides?: IonSlides;
|
protected toolbarSlides?: Swiper;
|
||||||
|
@ViewChild('swiperRef')
|
||||||
|
set swiperRef(swiperRef: ElementRef) {
|
||||||
|
/**
|
||||||
|
* This setTimeout waits for Ionic's async initialization to complete.
|
||||||
|
* Otherwise, an outdated swiper reference will be used.
|
||||||
|
*/
|
||||||
|
setTimeout(() => {
|
||||||
|
if (swiperRef.nativeElement?.swiper) {
|
||||||
|
this.toolbarSlides = swiperRef.nativeElement.swiper as Swiper;
|
||||||
|
|
||||||
|
this.toolbarSlides.changeLanguageDirection(CorePlatform.isRTL ? 'rtl' : 'ltr');
|
||||||
|
|
||||||
|
Object.keys(this.swiperOpts).forEach((key) => {
|
||||||
|
if (this.toolbarSlides) {
|
||||||
|
this.toolbarSlides.params[key] = this.swiperOpts[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
|
||||||
protected readonly DRAFT_AUTOSAVE_FREQUENCY = 30000;
|
protected readonly DRAFT_AUTOSAVE_FREQUENCY = 30000;
|
||||||
protected readonly RESTORE_MESSAGE_CLEAR_TIME = 6000;
|
protected readonly RESTORE_MESSAGE_CLEAR_TIME = 6000;
|
||||||
|
@ -119,7 +141,6 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
canScanQR = false;
|
canScanQR = false;
|
||||||
ariaLabelledBy?: string;
|
ariaLabelledBy?: string;
|
||||||
infoMessage?: string;
|
infoMessage?: string;
|
||||||
direction = 'ltr';
|
|
||||||
toolbarStyles = {
|
toolbarStyles = {
|
||||||
strong: 'false',
|
strong: 'false',
|
||||||
em: 'false',
|
em: 'false',
|
||||||
|
@ -133,11 +154,11 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
ol: 'false',
|
ol: 'false',
|
||||||
};
|
};
|
||||||
|
|
||||||
slidesOpts = {
|
swiperOpts: SwiperOptions = {
|
||||||
initialSlide: 0,
|
modules: [IonicSlides],
|
||||||
slidesPerView: 6,
|
slidesPerView: 6,
|
||||||
centerInsufficientSlides: true,
|
centerInsufficientSlides: true,
|
||||||
watchSlidesVisibility: true,
|
watchSlidesProgress: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -156,7 +177,6 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
this.canScanQR = CoreUtils.canScanQR();
|
this.canScanQR = CoreUtils.canScanQR();
|
||||||
this.isPhone = CoreScreen.isMobile;
|
this.isPhone = CoreScreen.isMobile;
|
||||||
this.toolbarHidden = this.isPhone;
|
this.toolbarHidden = this.isPhone;
|
||||||
this.direction = CorePlatform.isRTL ? 'rtl' : 'ltr';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -265,7 +285,7 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
// Change the side when the language changes.
|
// Change the side when the language changes.
|
||||||
this.languageChangedSubscription = Translate.onLangChange.subscribe(() => {
|
this.languageChangedSubscription = Translate.onLangChange.subscribe(() => {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.direction = CorePlatform.isRTL ? 'rtl' : 'ltr';
|
this.toolbarSlides?.changeLanguageDirection(CorePlatform.isRTL ? 'rtl' : 'ltr');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -774,8 +794,8 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
this.stopBubble(event);
|
this.stopBubble(event);
|
||||||
|
|
||||||
if (!this.toolbarNextHidden) {
|
if (!this.toolbarNextHidden) {
|
||||||
const currentIndex = await this.toolbarSlides?.getActiveIndex();
|
const currentIndex = this.toolbarSlides?.activeIndex;
|
||||||
this.toolbarSlides?.slideTo((currentIndex || 0) + this.slidesOpts.slidesPerView);
|
this.toolbarSlides?.slideTo((currentIndex || 0) + this.toolbarSlides.slidesPerViewDynamic());
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.updateToolbarArrows();
|
await this.updateToolbarArrows();
|
||||||
|
@ -792,8 +812,8 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
this.stopBubble(event);
|
this.stopBubble(event);
|
||||||
|
|
||||||
if (!this.toolbarPrevHidden) {
|
if (!this.toolbarPrevHidden) {
|
||||||
const currentIndex = await this.toolbarSlides?.getActiveIndex();
|
const currentIndex = this.toolbarSlides?.activeIndex;
|
||||||
this.toolbarSlides?.slideTo((currentIndex || 0) - this.slidesOpts.slidesPerView);
|
this.toolbarSlides?.slideTo((currentIndex || 0) - this.toolbarSlides.slidesPerViewDynamic());
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.updateToolbarArrows();
|
await this.updateToolbarArrows();
|
||||||
|
@ -808,7 +828,7 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const length = await this.toolbarSlides.length();
|
const length = this.toolbarSlides.slides.length;
|
||||||
|
|
||||||
// Cancel previous one, if any.
|
// Cancel previous one, if any.
|
||||||
this.buttonsDomPromise?.cancel();
|
this.buttonsDomPromise?.cancel();
|
||||||
|
@ -818,17 +838,16 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
const width = this.toolbar.nativeElement.getBoundingClientRect().width;
|
const width = this.toolbar.nativeElement.getBoundingClientRect().width;
|
||||||
|
|
||||||
if (length > 0 && width > length * this.toolbarButtonWidth) {
|
if (length > 0 && width > length * this.toolbarButtonWidth) {
|
||||||
this.slidesOpts = { ...this.slidesOpts, slidesPerView: length };
|
this.swiperOpts.slidesPerView = length;
|
||||||
this.toolbarArrows = false;
|
this.toolbarArrows = false;
|
||||||
} else {
|
} else {
|
||||||
const slidesPerView = Math.floor((width - this.toolbarArrowWidth * 2) / this.toolbarButtonWidth);
|
this.swiperOpts.slidesPerView = Math.floor((width - this.toolbarArrowWidth * 2) / this.toolbarButtonWidth);
|
||||||
this.slidesOpts = { ...this.slidesOpts, slidesPerView };
|
|
||||||
this.toolbarArrows = true;
|
this.toolbarArrows = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
await CoreUtils.nextTick();
|
await CoreUtils.nextTick();
|
||||||
|
|
||||||
await this.toolbarSlides.update();
|
this.toolbarSlides.update();
|
||||||
|
|
||||||
await this.updateToolbarArrows();
|
await this.updateToolbarArrows();
|
||||||
}
|
}
|
||||||
|
@ -841,10 +860,10 @@ export class CoreEditorRichTextEditorComponent implements OnInit, AfterViewInit,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentIndex = await this.toolbarSlides.getActiveIndex();
|
const currentIndex = this.toolbarSlides.activeIndex;
|
||||||
const length = await this.toolbarSlides.length();
|
const length = this.toolbarSlides.slides.length;
|
||||||
this.toolbarPrevHidden = currentIndex <= 0;
|
this.toolbarPrevHidden = currentIndex <= 0;
|
||||||
this.toolbarNextHidden = currentIndex + this.slidesOpts.slidesPerView >= length;
|
this.toolbarNextHidden = currentIndex + this.toolbarSlides.slidesPerViewDynamic() >= length;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
import { NgModule } from '@angular/core';
|
import { CUSTOM_ELEMENTS_SCHEMA, NgModule } from '@angular/core';
|
||||||
|
|
||||||
import { CoreSharedModule } from '@/core/shared.module';
|
import { CoreSharedModule } from '@/core/shared.module';
|
||||||
import { CoreViewerImageComponent } from './image/image';
|
import { CoreViewerImageComponent } from './image/image';
|
||||||
|
@ -33,5 +33,6 @@ import { CoreViewerTextComponent } from './text/text';
|
||||||
CoreViewerQRScannerComponent,
|
CoreViewerQRScannerComponent,
|
||||||
CoreViewerTextComponent,
|
CoreViewerTextComponent,
|
||||||
],
|
],
|
||||||
|
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
||||||
})
|
})
|
||||||
export class CoreViewerComponentsModule {}
|
export class CoreViewerComponentsModule {}
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
<ion-content>
|
<ion-content>
|
||||||
<ion-slides [options]="slidesOpts">
|
<swiper-container #swiperRef>
|
||||||
<ion-slide>
|
<swiper-slide>
|
||||||
<div class="swiper-zoom-container">
|
<div class="swiper-zoom-container">
|
||||||
<img [src]="image" [alt]="title" core-external-content [component]="component" [componentId]="componentId">
|
<img [src]="image" [alt]="title" core-external-content [component]="component" [componentId]="componentId">
|
||||||
</div>
|
</div>
|
||||||
</ion-slide>
|
</swiper-slide>
|
||||||
</ion-slides>
|
</swiper-container>
|
||||||
</ion-content>
|
</ion-content>
|
||||||
<ion-footer>
|
<ion-footer>
|
||||||
<ion-row class="ion-justify-content-between ion-align-items-center ion-no-padding">
|
<ion-row class="ion-justify-content-between ion-align-items-center ion-no-padding">
|
||||||
|
@ -15,12 +15,12 @@
|
||||||
</ion-button>
|
</ion-button>
|
||||||
</ion-col>
|
</ion-col>
|
||||||
<ion-col></ion-col>
|
<ion-col></ion-col>
|
||||||
<ion-col size="auto" *ngIf="slidesSwiper">
|
<ion-col size="auto" *ngIf="swiper">
|
||||||
<ion-button fill="clear" [attr.aria-label]="'core.zoomout' | translate" (click)="zoom(false)">
|
<ion-button fill="clear" [attr.aria-label]="'core.zoomout' | translate" (click)="zoom(false)">
|
||||||
<ion-icon name="fas-magnifying-glass-minus" slot="icon-only" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-magnifying-glass-minus" slot="icon-only" aria-hidden="true"></ion-icon>
|
||||||
</ion-button>
|
</ion-button>
|
||||||
</ion-col>
|
</ion-col>
|
||||||
<ion-col size="auto" *ngIf="slidesSwiper">
|
<ion-col size="auto" *ngIf="swiper">
|
||||||
<ion-button fill="clear" [attr.aria-label]="'core.zoomin' | translate" (click)="zoom(true)">
|
<ion-button fill="clear" [attr.aria-label]="'core.zoomin' | translate" (click)="zoom(true)">
|
||||||
<ion-icon name="fas-magnifying-glass-plus" slot="icon-only" aria-hidden="true"></ion-icon>
|
<ion-icon name="fas-magnifying-glass-plus" slot="icon-only" aria-hidden="true"></ion-icon>
|
||||||
</ion-button>
|
</ion-button>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
ion-slides {
|
swiper-container {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,10 +12,12 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
import { AfterViewInit, Component, ElementRef, Input, OnInit, ViewChild } from '@angular/core';
|
import { Component, ElementRef, Input, OnInit, ViewChild } from '@angular/core';
|
||||||
import { IonSlides } from '@ionic/angular';
|
|
||||||
import { ModalController, Translate } from '@singletons';
|
import { ModalController, Translate } from '@singletons';
|
||||||
import { CoreMath } from '@singletons/math';
|
import { CoreMath } from '@singletons/math';
|
||||||
|
import { Swiper } from 'swiper';
|
||||||
|
import { SwiperOptions } from 'swiper/types';
|
||||||
|
import { IonicSlides } from '@ionic/angular';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Modal component to view an image.
|
* Modal component to view an image.
|
||||||
|
@ -25,29 +27,49 @@ import { CoreMath } from '@singletons/math';
|
||||||
templateUrl: 'image.html',
|
templateUrl: 'image.html',
|
||||||
styleUrls: ['image.scss'],
|
styleUrls: ['image.scss'],
|
||||||
})
|
})
|
||||||
export class CoreViewerImageComponent implements OnInit, AfterViewInit {
|
export class CoreViewerImageComponent implements OnInit {
|
||||||
|
|
||||||
@ViewChild(IonSlides) protected slides?: IonSlides;
|
protected swiper?: Swiper;
|
||||||
|
@ViewChild('swiperRef')
|
||||||
|
set swiperRef(swiperRef: ElementRef) {
|
||||||
|
/**
|
||||||
|
* This setTimeout waits for Ionic's async initialization to complete.
|
||||||
|
* Otherwise, an outdated swiper reference will be used.
|
||||||
|
*/
|
||||||
|
setTimeout(() => {
|
||||||
|
if (swiperRef.nativeElement?.swiper) {
|
||||||
|
this.swiper = swiperRef.nativeElement.swiper as Swiper;
|
||||||
|
|
||||||
|
Object.keys(this.swiperOpts).forEach((key) => {
|
||||||
|
if (this.swiper) {
|
||||||
|
this.swiper.params[key] = this.swiperOpts[key];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
|
||||||
@Input() title = ''; // Modal title.
|
@Input() title = ''; // Modal title.
|
||||||
@Input() image = ''; // Image URL.
|
@Input() image = ''; // Image URL.
|
||||||
@Input() component?: string; // Component to use in external-content.
|
@Input() component?: string; // Component to use in external-content.
|
||||||
@Input() componentId?: string | number; // Component ID to use in external-content.
|
@Input() componentId?: string | number; // Component ID to use in external-content.
|
||||||
|
|
||||||
slidesOpts = {
|
private static readonly MAX_RATIO = 8;
|
||||||
|
|
||||||
|
protected swiperOpts: SwiperOptions = {
|
||||||
|
modules: [IonicSlides],
|
||||||
|
freeMode: true,
|
||||||
slidesPerView: 1,
|
slidesPerView: 1,
|
||||||
centerInsufficientSlides: true,
|
centerInsufficientSlides: true,
|
||||||
centerSlides: true,
|
centeredSlides: true,
|
||||||
zoom: {
|
zoom: {
|
||||||
maxRatio: 8,
|
maxRatio: CoreViewerImageComponent.MAX_RATIO,
|
||||||
minRatio: 0.5, // User can zoom out to 0.5 only using pinch gesture.
|
minRatio: 0.5, // User can zoom out to 0.5 only using pinch gesture.
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
protected zoomRatio = 1;
|
protected zoomRatio = 1;
|
||||||
|
|
||||||
slidesSwiper: any; // eslint-disable-line @typescript-eslint/no-explicit-any
|
|
||||||
|
|
||||||
constructor(protected element: ElementRef<HTMLElement>) {
|
constructor(protected element: ElementRef<HTMLElement>) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -58,12 +80,6 @@ export class CoreViewerImageComponent implements OnInit, AfterViewInit {
|
||||||
this.title = this.title || Translate.instant('core.imageviewer');
|
this.title = this.title || Translate.instant('core.imageviewer');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @inheritdoc
|
|
||||||
*/
|
|
||||||
async ngAfterViewInit(): Promise<void> {
|
|
||||||
this.slidesSwiper = await this.slides?.getSwiper();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Close modal.
|
* Close modal.
|
||||||
|
@ -80,7 +96,7 @@ export class CoreViewerImageComponent implements OnInit, AfterViewInit {
|
||||||
zoom(zoomIn = true): void {
|
zoom(zoomIn = true): void {
|
||||||
const imageElement = this.element.nativeElement.querySelector('img');
|
const imageElement = this.element.nativeElement.querySelector('img');
|
||||||
|
|
||||||
if (!this.slidesSwiper || !imageElement) {
|
if (!this.swiper || !imageElement) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,15 +105,15 @@ export class CoreViewerImageComponent implements OnInit, AfterViewInit {
|
||||||
: this.zoomRatio /= 2;
|
: this.zoomRatio /= 2;
|
||||||
|
|
||||||
// Using 1 as minimum for manual zoom.
|
// Using 1 as minimum for manual zoom.
|
||||||
this.zoomRatio = CoreMath.clamp(this.zoomRatio, 1, this.slidesOpts.zoom.maxRatio);
|
this.zoomRatio = CoreMath.clamp(this.zoomRatio, 1, CoreViewerImageComponent.MAX_RATIO);
|
||||||
|
|
||||||
if (this.zoomRatio > 1) {
|
if (this.zoomRatio > 1) {
|
||||||
this.slidesSwiper.zoom.in();
|
this.swiper.zoom.in();
|
||||||
|
|
||||||
imageElement.style.transform =
|
imageElement.style.transform =
|
||||||
'translate3d(0px, 0px, 0px) scale(' + this.zoomRatio + ')';
|
'translate3d(0px, 0px, 0px) scale(' + this.zoomRatio + ')';
|
||||||
} else {
|
} else {
|
||||||
this.slidesSwiper.zoom.out();
|
this.swiper.zoom.out();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -204,7 +204,7 @@ export class TestingBehatBlockingService {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const slide = element.closest('ion-slide');
|
const slide = element.closest('swiper-slide');
|
||||||
if (slide && !slide.classList.contains('swiper-slide-active')) {
|
if (slide && !slide.classList.contains('swiper-slide-active')) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -48,7 +48,7 @@ export class TestingBehatDomUtilsService {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (element.tagName === 'ION-SLIDE') {
|
if (element.tagName === 'SWIPER-SLIDE') {
|
||||||
// Check if the slide is visible (in the viewport).
|
// Check if the slide is visible (in the viewport).
|
||||||
const bounding = element.getBoundingClientRect();
|
const bounding = element.getBoundingClientRect();
|
||||||
if (bounding.right <= 0 || bounding.left >= window.innerWidth) {
|
if (bounding.right <= 0 || bounding.left >= window.innerWidth) {
|
||||||
|
|
|
@ -29,7 +29,7 @@ import { Injectable } from '@angular/core';
|
||||||
import { CoreSites, CoreSitesProvider } from '@services/sites';
|
import { CoreSites, CoreSitesProvider } from '@services/sites';
|
||||||
import { CoreNavigator, CoreNavigatorService } from '@services/navigator';
|
import { CoreNavigator, CoreNavigatorService } from '@services/navigator';
|
||||||
import { CoreSwipeNavigationDirective } from '@directives/swipe-navigation';
|
import { CoreSwipeNavigationDirective } from '@directives/swipe-navigation';
|
||||||
import { IonSlides } from '@ionic/angular';
|
import { Swiper } from 'swiper';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Behat runtime servive with public API.
|
* Behat runtime servive with public API.
|
||||||
|
@ -631,8 +631,8 @@ export class TestingBehatRuntimeService {
|
||||||
this.log('Action - Swipe', { direction, locator });
|
this.log('Action - Swipe', { direction, locator });
|
||||||
|
|
||||||
if (locator) {
|
if (locator) {
|
||||||
// Locator specified, try to find ion-slides first.
|
// Locator specified, try to find swiper-container first.
|
||||||
const instance = this.getAngularInstance<IonSlides>('ion-slides', 'IonSlides', locator);
|
const instance = this.getAngularInstance<Swiper>('swiper-container', 'Swiper', locator);
|
||||||
if (instance) {
|
if (instance) {
|
||||||
direction === 'left' ? instance.slideNext() : instance.slidePrev();
|
direction === 'left' ? instance.slideNext() : instance.slidePrev();
|
||||||
|
|
||||||
|
@ -640,7 +640,7 @@ export class TestingBehatRuntimeService {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// No locator specified or ion-slides not found, search swipe navigation now.
|
// No locator specified or swiper-container not found, search swipe navigation now.
|
||||||
const instance = this.getAngularInstance<CoreSwipeNavigationDirective>(
|
const instance = this.getAngularInstance<CoreSwipeNavigationDirective>(
|
||||||
'ion-content',
|
'ion-content',
|
||||||
'CoreSwipeNavigationDirective',
|
'CoreSwipeNavigationDirective',
|
||||||
|
|
|
@ -151,7 +151,7 @@ ion-toolbar {
|
||||||
|
|
||||||
// Header.
|
// Header.
|
||||||
ion-header {
|
ion-header {
|
||||||
z-index: 12; // To hide ion-slides on scroll.
|
z-index: 12; // To hide swiper-container on scroll.
|
||||||
|
|
||||||
ion-toolbar {
|
ion-toolbar {
|
||||||
ion-spinner {
|
ion-spinner {
|
||||||
|
@ -1807,13 +1807,53 @@ ion-header.no-title {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
|
||||||
ion-slides {
|
swiper-container {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
swiper-container {
|
||||||
|
--swiper-theme-color: var(--ion-color-primary, #3880ff);
|
||||||
|
--swiper-pagination-bullet-inactive-color: var(--ion-color-step-200, #cccccc);
|
||||||
|
--swiper-pagination-color: var(--swiper-theme-color);
|
||||||
|
--swiper-pagination-progressbar-bg-color: rgba(var(--ion-text-color-rgb, 0, 0, 0), 0.25);
|
||||||
|
--swiper-scrollbar-bg-color: rgba(var(--ion-text-color-rgb, 0, 0, 0), 0.1);
|
||||||
|
--swiper-scrollbar-drag-bg-color: rgba(var(--ion-text-color-rgb, 0, 0, 0), 0.5);
|
||||||
|
width: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
max-height: 100vh;
|
||||||
|
// CSS Grid/Flexbox bug size workaround
|
||||||
|
// @see https://github.com/kenwheeler/slick/issues/982
|
||||||
|
// @see https://github.com/nolimits4web/swiper/issues/3599
|
||||||
|
min-height: 0;
|
||||||
|
min-width: 0;
|
||||||
|
|
||||||
|
swiper-slide {
|
||||||
|
display: flex;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
flex-direction: column;
|
||||||
|
flex-shrink: 0;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
|
||||||
|
font-size: 18px;
|
||||||
|
|
||||||
|
text-align: center;
|
||||||
|
box-sizing: border-box;
|
||||||
|
|
||||||
|
img {
|
||||||
|
width: auto;
|
||||||
|
max-width: 100%;
|
||||||
|
height: auto;
|
||||||
|
max-height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.has-spacer,
|
.has-spacer,
|
||||||
|
|
|
@ -247,7 +247,7 @@ html {
|
||||||
--core-tab-font-weight-active: normal;
|
--core-tab-font-weight-active: normal;
|
||||||
--core-tabs-height: 48px;
|
--core-tabs-height: 48px;
|
||||||
core-tabs, core-tabs-outlet {
|
core-tabs, core-tabs-outlet {
|
||||||
ion-slide {
|
swiper-slide {
|
||||||
--background: var(--core-tab-background);
|
--background: var(--core-tab-background);
|
||||||
--color: var(--core-tab-color);
|
--color: var(--core-tab-color);
|
||||||
--border-color: var(--core-tab-border-color);
|
--border-color: var(--core-tab-border-color);
|
||||||
|
|
|
@ -8,6 +8,7 @@ For more information about upgrading, read the official documentation: https://m
|
||||||
- CoreCache has been deprecated, use plain object as in-memory stores instead.
|
- CoreCache has been deprecated, use plain object as in-memory stores instead.
|
||||||
- Renamed CoreLoginSitesComponent to CoreLoginSitesModalComponent to make it clear that it's a modal and to avoid confusing it with the new CoreSitesListComponent.
|
- Renamed CoreLoginSitesComponent to CoreLoginSitesModalComponent to make it clear that it's a modal and to avoid confusing it with the new CoreSitesListComponent.
|
||||||
- Removed CoreToLocaleStringPipe deprecated since 3.6.0
|
- Removed CoreToLocaleStringPipe deprecated since 3.6.0
|
||||||
|
- With the upgrade to Ionic 7 ion-slides is no longer supported and now you need to use swiper-container and swiper-slide. More info here: https://ionicframework.com/docs/angular/slides
|
||||||
|
|
||||||
=== 4.3.0 ===
|
=== 4.3.0 ===
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue