MOBILE-3651 core: Fix scroll to element function
parent
1620dd47ea
commit
f682d89e67
|
@ -47,6 +47,7 @@ export class AddonModQuizPreflightModalComponent implements OnInit {
|
|||
|
||||
constructor(
|
||||
formBuilder: FormBuilder,
|
||||
protected elementRef: ElementRef,
|
||||
) {
|
||||
// Create an empty form group. The controls will be added by the access rules components.
|
||||
this.preflightForm = formBuilder.group({});
|
||||
|
@ -115,7 +116,12 @@ export class AddonModQuizPreflightModalComponent implements OnInit {
|
|||
|
||||
if (!this.preflightForm.valid) {
|
||||
// Form not valid. Scroll to the first element with errors.
|
||||
if (!CoreDomUtils.instance.scrollToInputError(this.content)) {
|
||||
const hasScrolled = CoreDomUtils.instance.scrollToInputError(
|
||||
this.elementRef.nativeElement,
|
||||
this.content,
|
||||
);
|
||||
|
||||
if (!hasScrolled) {
|
||||
// Input not found, show an error modal.
|
||||
CoreDomUtils.instance.showErrorModal('core.errorinvalidform', true);
|
||||
}
|
||||
|
|
|
@ -681,9 +681,11 @@ export class AddonModQuizPlayerPage implements OnInit, OnDestroy {
|
|||
* @param slot Slot of the question to scroll to.
|
||||
*/
|
||||
protected scrollToQuestion(slot: number): void {
|
||||
if (this.content) {
|
||||
CoreDomUtils.instance.scrollToElementBySelector(this.content, '#addon-mod_quiz-question-' + slot);
|
||||
}
|
||||
CoreDomUtils.instance.scrollToElementBySelector(
|
||||
this.elementRef.nativeElement,
|
||||
this.content,
|
||||
'#addon-mod_quiz-question-' + slot,
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -104,7 +104,11 @@ export class CoreLinkDirective implements OnInit {
|
|||
if (href.charAt(0) == '#') {
|
||||
// Look for id or name.
|
||||
href = href.substr(1);
|
||||
CoreDomUtils.instance.scrollToElementBySelector(this.content, '#' + href + ', [name=\'' + href + '\']');
|
||||
CoreDomUtils.instance.scrollToElementBySelector(
|
||||
this.element.closest('ion-content'),
|
||||
this.content,
|
||||
`#${href}, [name='${href}']`,
|
||||
);
|
||||
|
||||
return;
|
||||
}
|
||||
|
|
|
@ -25,6 +25,7 @@ import {
|
|||
QueryList,
|
||||
Type,
|
||||
ViewChild,
|
||||
ElementRef,
|
||||
} from '@angular/core';
|
||||
|
||||
import { CoreSites } from '@services/sites';
|
||||
|
@ -111,6 +112,7 @@ export class CoreCourseFormatComponent implements OnInit, OnChanges, OnDestroy {
|
|||
|
||||
constructor(
|
||||
protected content: IonContent,
|
||||
protected elementRef: ElementRef,
|
||||
) {
|
||||
// Pass this instance to all components so they can use its methods and properties.
|
||||
this.data.coreCourseFormatComponent = this;
|
||||
|
@ -402,7 +404,11 @@ export class CoreCourseFormatComponent implements OnInit, OnChanges, OnDestroy {
|
|||
|
||||
if (this.moduleId && typeof previousValue == 'undefined') {
|
||||
setTimeout(() => {
|
||||
CoreDomUtils.instance.scrollToElementBySelector(this.content, '#core-course-module-' + this.moduleId);
|
||||
CoreDomUtils.instance.scrollToElementBySelector(
|
||||
this.elementRef.nativeElement,
|
||||
this.content,
|
||||
'#core-course-module-' + this.moduleId,
|
||||
);
|
||||
}, 200);
|
||||
} else {
|
||||
this.content.scrollToTop(0);
|
||||
|
|
|
@ -12,7 +12,7 @@
|
|||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
import { Component, ViewChild, ElementRef, OnInit } from '@angular/core';
|
||||
import { Component, ViewChild, ElementRef, OnInit, ChangeDetectorRef } from '@angular/core';
|
||||
import { FormBuilder, FormGroup, Validators, FormControl } from '@angular/forms';
|
||||
import { IonContent, IonRefresher } from '@ionic/angular';
|
||||
|
||||
|
@ -81,6 +81,8 @@ export class CoreLoginEmailSignupPage implements OnInit {
|
|||
|
||||
constructor(
|
||||
protected fb: FormBuilder,
|
||||
protected elementRef: ElementRef,
|
||||
protected changeDetector: ChangeDetectorRef,
|
||||
) {
|
||||
// Create the ageVerificationForm.
|
||||
this.ageVerificationForm = this.fb.group({
|
||||
|
@ -272,8 +274,17 @@ export class CoreLoginEmailSignupPage implements OnInit {
|
|||
e.stopPropagation();
|
||||
|
||||
if (!this.signupForm.valid || (this.settings?.recaptchapublickey && !this.captcha.recaptcharesponse)) {
|
||||
// Form not valid. Scroll to the first element with errors.
|
||||
const errorFound = await CoreDomUtils.instance.scrollToInputError(this.content);
|
||||
// Form not valid. Mark all controls as dirty to display errors.
|
||||
for (const name in this.signupForm.controls) {
|
||||
this.signupForm.controls[name].markAsDirty();
|
||||
}
|
||||
this.changeDetector.detectChanges();
|
||||
|
||||
// Scroll to the first element with errors.
|
||||
const errorFound = CoreDomUtils.instance.scrollToInputError(
|
||||
this.elementRef.nativeElement,
|
||||
this.content,
|
||||
);
|
||||
|
||||
if (!errorFound) {
|
||||
// Input not found, show an error modal.
|
||||
|
|
|
@ -1119,24 +1119,26 @@ export class CoreDomUtilsProvider {
|
|||
/**
|
||||
* Scroll to a certain element using a selector to find it.
|
||||
*
|
||||
* @param container The element that contains the element that must be scrolled.
|
||||
* @param content The content that must be scrolled.
|
||||
* @param selector Selector to find the element to scroll to.
|
||||
* @param scrollParentClass Parent class where to stop calculating the position. Default inner-scroll.
|
||||
* @param duration Duration of the scroll animation in milliseconds.
|
||||
* @return True if the element is found, false otherwise.
|
||||
*/
|
||||
async scrollToElementBySelector(
|
||||
content: IonContent,
|
||||
scrollToElementBySelector(
|
||||
container: HTMLElement | null,
|
||||
content: IonContent | undefined,
|
||||
selector: string,
|
||||
scrollParentClass?: string,
|
||||
duration?: number,
|
||||
): Promise<boolean> {
|
||||
// @todo: This function is broken. Scroll element cannot be used because it uses shadow DOM so querySelector returns null.
|
||||
// Also, traversing using parentElement doesn't work either, offsetParent isn't part of the parentElement tree.
|
||||
try {
|
||||
const scrollElement = await content.getScrollElement();
|
||||
): boolean {
|
||||
if (!container || !content) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const position = this.getElementXY(scrollElement, selector, scrollParentClass);
|
||||
try {
|
||||
const position = this.getElementXY(container, selector, scrollParentClass);
|
||||
if (!position) {
|
||||
return false;
|
||||
}
|
||||
|
@ -1152,16 +1154,13 @@ export class CoreDomUtilsProvider {
|
|||
/**
|
||||
* Search for an input with error (core-input-error directive) and scrolls to it if found.
|
||||
*
|
||||
* @param container The element that contains the element that must be scrolled.
|
||||
* @param content The content that must be scrolled.
|
||||
* @param scrollParentClass Parent class where to stop calculating the position. Default inner-scroll.
|
||||
* @return True if the element is found, false otherwise.
|
||||
*/
|
||||
async scrollToInputError(content?: IonContent, scrollParentClass?: string): Promise<boolean> {
|
||||
if (!content) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return this.scrollToElementBySelector(content, '.core-input-error', scrollParentClass);
|
||||
scrollToInputError(container: HTMLElement | null, content?: IonContent, scrollParentClass?: string): boolean {
|
||||
return this.scrollToElementBySelector(container, content, '.core-input-error', scrollParentClass);
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
Loading…
Reference in New Issue