diff --git a/src/addons/mod/assign/components/feedback-plugin/addon-mod-assign-feedback-plugin.html b/src/addons/mod/assign/components/feedback-plugin/addon-mod-assign-feedback-plugin.html index 0c6116e3c..fce2551b7 100644 --- a/src/addons/mod/assign/components/feedback-plugin/addon-mod-assign-feedback-plugin.html +++ b/src/addons/mod/assign/components/feedback-plugin/addon-mod-assign-feedback-plugin.html @@ -1,4 +1,3 @@ - @@ -9,9 +8,8 @@ {{ 'addon.mod_assign.feedbacknotsupported' | translate }}

- +

@@ -9,9 +8,8 @@ {{ 'addon.mod_assign.submissionnotsupported' | translate }}

- +

{{ plugin.name }}

- +

- +
@@ -25,9 +23,8 @@ {{ plugin.name }} - + diff --git a/src/addons/mod/assign/submission/onlinetext/component/addon-mod-assign-submission-onlinetext.html b/src/addons/mod/assign/submission/onlinetext/component/addon-mod-assign-submission-onlinetext.html index 6660c6c4f..f17a5dca5 100644 --- a/src/addons/mod/assign/submission/onlinetext/component/addon-mod-assign-submission-onlinetext.html +++ b/src/addons/mod/assign/submission/onlinetext/component/addon-mod-assign-submission-onlinetext.html @@ -4,9 +4,8 @@

{{ plugin.name }}

{{ 'addon.mod_assign.numwords' | translate: {'$a': words} }}

- +

@@ -15,7 +14,9 @@
-

{{ plugin.name }}

+ +

{{ plugin.name }}

+
@@ -25,10 +26,10 @@ {{ plugin.name }} - +
diff --git a/src/addons/mod/workshop/components/index/addon-mod-workshop-index.html b/src/addons/mod/workshop/components/index/addon-mod-workshop-index.html index 35f7cac49..4004119c0 100644 --- a/src/addons/mod/workshop/components/index/addon-mod-workshop-index.html +++ b/src/addons/mod/workshop/components/index/addon-mod-workshop-index.html @@ -87,7 +87,7 @@

{{ 'addon.mod_workshop.conclusion' | translate }}

-
@@ -120,8 +120,8 @@

{{ 'addon.mod_workshop.areainstructauthors' | translate }}

- +
@@ -184,7 +184,7 @@

{{ 'addon.mod_workshop.areainstructreviewers' | translate }}

- diff --git a/src/core/directives/format-text.ts b/src/core/directives/format-text.ts index ea1ed3e5e..10c220e72 100644 --- a/src/core/directives/format-text.ts +++ b/src/core/directives/format-text.ts @@ -61,8 +61,6 @@ export class CoreFormatTextDirective implements OnChanges { @Input() adaptImg?: boolean | string = true; // Whether to adapt images to screen width. @Input() clean?: boolean | string; // Whether all the HTML tags should be removed. @Input() singleLine?: boolean | string; // Whether new lines should be removed (all text in single line). Only if clean=true. - @Input() fullOnClick?: boolean | string; // Whether it should open a new page with the full contents on click. - @Input() fullTitle?: string; // Title to use in full view. Defaults to "Description". @Input() highlight?: string; // Text to highlight. @Input() filter?: boolean | string; // Whether to filter the text. If not defined, true if contextLevel and instanceId are set. @Input() contextLevel?: string; // The context level of the text. @@ -73,6 +71,8 @@ export class CoreFormatTextDirective implements OnChanges { @Input() openLinksInApp?: boolean; // Whether links should be opened in InAppBrowser. @Input() hideIfEmpty = false; // If true, the tag will contain nothing if text is empty. + @Input() fullOnClick?: boolean | string; // @deprecated on 4.0 Won't do anything. + @Input() fullTitle?: string; // @deprecated on 4.0 Won't do anything.. /** * Max height in pixels to render the content box. It should be 50 at least to make sense. * Using this parameter will force display: block to calculate height better. @@ -84,10 +84,11 @@ export class CoreFormatTextDirective implements OnChanges { @Output() onClick: EventEmitter = new EventEmitter(); // Called when clicked. protected element: HTMLElement; - protected showMoreDisplayed = false; + protected expanded = false; protected loadingChangedListener?: CoreEventObserver; protected emptyText = ''; protected contentSpan: HTMLElement; + protected toggleExpandEnabled = false; constructor( element: ElementRef, @@ -114,11 +115,12 @@ export class CoreFormatTextDirective implements OnChanges { } /** - * Detect changes on input properties. + * @inheritdoc */ ngOnChanges(changes: { [name: string]: SimpleChange }): void { if (changes.text || changes.filter || changes.contextLevel || changes.contextInstanceId) { - this.hideShowMore(); + this.setExpandButtonEnabled(false); + this.formatAndRenderContents(); } } @@ -267,37 +269,61 @@ export class CoreFormatTextDirective implements OnChanges { this.element.style.maxHeight = initialMaxHeight; // If cannot calculate height, shorten always. - if (!height || height > this.maxHeight) { - if (!this.showMoreDisplayed) { - this.displayShowMore(); - } - } else if (this.showMoreDisplayed) { - this.hideShowMore(); - } + this.setExpandButtonEnabled(!height || height > this.maxHeight); } /** - * Display the "Show more" in the element. + * Sets if expand button is enabled or not. + * + * @param enable Wether enable or disable. */ - protected displayShowMore(): void { - const expandInFullview = CoreUtils.isTrueOrOne(this.fullOnClick) || false; - const showMoreButton = document.createElement('ion-button'); + protected setExpandButtonEnabled(enable: boolean): void { + this.toggleExpandEnabled = enable; + this.element.classList.toggle('core-text-formatted', enable); - showMoreButton.classList.add('core-show-more'); - showMoreButton.setAttribute('fill', 'clear'); - showMoreButton.innerHTML = Translate.instant('core.showmore'); - this.element.appendChild(showMoreButton); - - if (expandInFullview) { - this.element.classList.add('core-expand-in-fullview'); - } else { - showMoreButton.setAttribute('aria-expanded', 'false'); + if (!enable || this.element.querySelector('ion-button.core-format-text-toggle')) { + return; } - this.element.classList.add('core-text-formatted'); - this.element.classList.add('core-shortened'); - this.element.style.maxHeight = this.maxHeight + 'px'; - this.showMoreDisplayed = true; + // Add expand/collapse buttons + const toggleButton = document.createElement('ion-button'); + toggleButton.classList.add('core-format-text-toggle'); + toggleButton.setAttribute('fill', 'clear'); + + const toggleText = document.createElement('span'); + toggleText.classList.add('core-format-text-toggle-text'); + toggleButton.appendChild(toggleText); + + const expandArrow = document.createElement('span'); + expandArrow.classList.add('core-format-text-arrow'); + toggleButton.appendChild(expandArrow); + + this.element.appendChild(toggleButton); + + this.toggleExpand(this.expanded); + } + + /** + * Expand or collapse text. + * + * @param expand Wether expand or collapse text. If undefined, will toggle. + */ + protected toggleExpand(expand?: boolean): void { + if (expand === undefined) { + expand = !this.expanded; + } + this.expanded = expand; + this.element.classList.toggle('core-text-format-expanded', expand); + this.element.classList.toggle('core-text-format-collapsed', !expand); + this.element.style.maxHeight = expand ? '' : this.maxHeight + 'px'; + + const toggleButton = this.element.querySelector('ion-button.core-format-text-toggle'); + const toggleText = toggleButton?.querySelector('.core-format-text-toggle-text'); + if (!toggleButton || !toggleText) { + return; + } + toggleText.innerHTML = expand ? Translate.instant('core.showless') : Translate.instant('core.showmore'); + toggleButton.setAttribute('aria-expanded', expand ? 'true' : 'false'); } /** @@ -321,9 +347,7 @@ export class CoreFormatTextDirective implements OnChanges { return; } - const expandInFullview = CoreUtils.isTrueOrOne(this.fullOnClick) || false; - - if (!expandInFullview && !this.showMoreDisplayed) { + if (!this.toggleExpandEnabled) { // Nothing to do on click, just stop. return; } @@ -331,28 +355,7 @@ export class CoreFormatTextDirective implements OnChanges { e.preventDefault(); e.stopPropagation(); - if (!expandInFullview) { - // Change class. - this.element.classList.toggle('core-shortened'); - - return; - } else { - // Open a new state with the contents. - const filter = typeof this.filter != 'undefined' ? CoreUtils.isTrueOrOne(this.filter) : undefined; - - CoreTextUtils.viewText( - this.fullTitle || Translate.instant('core.description'), - this.text, - { - component: this.component, - componentId: this.componentId, - filter: filter, - contextLevel: this.contextLevel, - instanceId: this.contextInstanceId, - courseId: this.courseId, - }, - ); - } + this.toggleExpand(); } /** @@ -361,6 +364,7 @@ export class CoreFormatTextDirective implements OnChanges { protected finishRender(): void { // Show the element again. this.element.classList.remove('core-format-text-loading'); + // Emit the afterRender output. this.afterRender.emit(); } @@ -393,7 +397,7 @@ export class CoreFormatTextDirective implements OnChanges { this.contentSpan.innerHTML = ''; // Remove current contents. if (this.maxHeight && result.div.innerHTML != '' && - (this.fullOnClick || (window.innerWidth < 576 || window.innerHeight < 576))) { // Don't collapse in big screens. + (window.innerWidth < 576 || window.innerHeight < 576)) { // Don't collapse in big screens. // Move the children to the current element to be able to calculate the height. CoreDomUtils.moveChildren(result.div, this.contentSpan); @@ -655,20 +659,6 @@ export class CoreFormatTextDirective implements OnChanges { return CoreDomUtils.getElementHeight(element) || 0; } - /** - * "Hide" the "Show more" in the element if it's shown. - */ - protected hideShowMore(): void { - const showMoreButton = this.element.querySelector('ion-button.core-show-more'); - showMoreButton?.remove(); - - this.element.classList.remove('core-expand-in-fullview'); - this.element.classList.remove('core-text-formatted'); - this.element.classList.remove('core-shortened'); - this.element.style.maxHeight = ''; - this.showMoreDisplayed = false; - } - /** * Add media adapt class and apply CoreExternalContentDirective to the media element and its sources and tracks. * diff --git a/src/core/features/course/components/module-description/core-course-module-description.html b/src/core/features/course/components/module-description/core-course-module-description.html index e07546c5a..0f6d34c0a 100644 --- a/src/core/features/course/components/module-description/core-course-module-description.html +++ b/src/core/features/course/components/module-description/core-course-module-description.html @@ -1,9 +1,8 @@ - + @@ -12,4 +11,4 @@ {{ note }}
- \ No newline at end of file + diff --git a/src/core/features/course/components/module-info/core-course-module-info.html b/src/core/features/course/components/module-info/core-course-module-info.html index acd0b90e5..8aac777b4 100644 --- a/src/core/features/course/components/module-info/core-course-module-info.html +++ b/src/core/features/course/components/module-info/core-course-module-info.html @@ -13,7 +13,7 @@ + [contextInstanceId]="module.id" [courseId]="courseId" [maxHeight]="120"> diff --git a/src/core/features/course/components/module/core-course-module.html b/src/core/features/course/components/module/core-course-module.html index 630c19e11..ea1bec5df 100644 --- a/src/core/features/course/components/module/core-course-module.html +++ b/src/core/features/course/components/module/core-course-module.html @@ -1,17 +1,12 @@ - + }" [button]="module.handlerData.action && module.uservisible" detail="false"> @@ -23,11 +18,8 @@ [courseId]="courseId" [attr.aria-label]="module.handlerData.a11yTitle + ', ' + modNameTranslated">

- + @@ -47,12 +39,8 @@ -
+
- + + (click)="buttonClicked($event, button)" [attr.aria-label]="button.label | translate:{$a: module.handlerData.title}">
- + }" detail="false">
@@ -94,12 +75,12 @@
- -
@@ -107,14 +88,11 @@ - - + [ngClass]="['core-course-module-handler', 'core-module-loading', module.handlerData.class]" detail="false"> + + + diff --git a/src/core/features/course/pages/preview/preview.html b/src/core/features/course/pages/preview/preview.html index a69893d67..49a16a7ab 100644 --- a/src/core/features/course/pages/preview/preview.html +++ b/src/core/features/course/pages/preview/preview.html @@ -15,7 +15,7 @@
- +
@@ -27,8 +27,10 @@ -

+

+ + +

{{course.startdate * 1000 | coreFormatDate:"strftimedatefullshort" }} - {{course.enddate * 1000 | coreFormatDate:"strftimedatefullshort" }} @@ -38,7 +40,7 @@ - + @@ -51,8 +53,7 @@ - +

{{contact.fullname}}

@@ -63,20 +64,20 @@ - -
- - - - : - - - - -
-
+ +
+ + + + : + + + + +
+
@@ -100,19 +101,19 @@ -

{{ 'core.courses.notenrollable' | translate }}

+ +

{{ 'core.courses.notenrollable' | translate }}

+
+ [attr.aria-label]="prefetchCourseData.statusTranslatable | translate" button> -
diff --git a/src/core/features/courses/pages/categories/categories.html b/src/core/features/courses/pages/categories/categories.html index 495001569..0baf6bdff 100644 --- a/src/core/features/courses/pages/categories/categories.html +++ b/src/core/features/courses/pages/categories/categories.html @@ -31,7 +31,7 @@

-

diff --git a/src/core/features/courses/services/courses-helper.ts b/src/core/features/courses/services/courses-helper.ts index 3cf1a014d..df8e4e92c 100644 --- a/src/core/features/courses/services/courses-helper.ts +++ b/src/core/features/courses/services/courses-helper.ts @@ -251,7 +251,7 @@ export class CoreCoursesHelperProvider { case 'lastaccess': courses.sort((a, b) => (b.lastaccess || 0) - (a.lastaccess || 0)); break; - // @todo Time modified property is not defined in CoreEnrolledCourseDataWithOptions, so it won't do nothing. + // @todo Time modified property is not defined in CoreEnrolledCourseDataWithOptions, so it Won't do anything. // case 'timemodified': // courses.sort((a, b) => b.timemodified - a.timemodified); // break; diff --git a/src/core/features/grades/pages/grade/grade.html b/src/core/features/grades/pages/grade/grade.html index 9abe2b1f3..f565c8388 100644 --- a/src/core/features/grades/pages/grade/grade.html +++ b/src/core/features/grades/pages/grade/grade.html @@ -14,26 +14,29 @@ - + - + -

-

+

+ + +

- + - + -

-

+

+ + +

@@ -89,8 +92,10 @@

{{ 'core.grades.feedback' | translate}}

-

+

+ + +

diff --git a/src/core/services/utils/text.ts b/src/core/services/utils/text.ts index 2ea31bf2b..362298e18 100644 --- a/src/core/services/utils/text.ts +++ b/src/core/services/utils/text.ts @@ -571,7 +571,7 @@ export class CoreTextUtilsProvider { const regex = new RegExp('(' + searchText + ')', 'gi'); - return text.replace(regex, '$1'); + return text.replace(regex, '$1'); } /** diff --git a/src/theme/components/format-text.scss b/src/theme/components/format-text.scss index 204c0612b..7ac616332 100644 --- a/src/theme/components/format-text.scss +++ b/src/theme/components/format-text.scss @@ -51,11 +51,15 @@ core-format-text { display: inline; } - .core-show-more { + .core-format-text-toggle { display: none !important; } } + .core-format-text-toggle { + display: none; + } + .core-format-text-content { opacity: 1; @include core-transition(opacity, 200ms); @@ -84,30 +88,45 @@ core-format-text { cursor: pointer; pointer-events: auto; - .core-show-more { - display: none; + .core-format-text-toggle { + display: block; + position: absolute; + bottom: 0; + left: 0; + right: 0; + text-align: center; + z-index: 7; + text-transform: none; + text-align: end; + font-size: 14px; + background-color: var(--core-format-text-background); + color: var(--text-color); + margin: 0; + + + .core-format-text-arrow { + width: var(--a11y-min-target-size); + height: var(--a11y-min-target-size); + + background-position: center; + background-repeat: no-repeat; + background-size: 14px 14px; + @include core-transition(transform, 500ms); + + @include push-arrow-color(626262, true); + + @include darkmode() { + @include push-arrow-color(ffffff, true); + } + } } - &:not(.core-shortened) { - max-height: none !important; - } - - &.core-shortened { + &.core-text-format-collapsed { overflow: hidden; min-height: 50px; - .core-show-more { - text-transform: none; - text-align: end; - font-size: 14px; - display: block; - position: absolute; - @include position(null, 0, 0, null); - z-index: 7; - background-color: var(--core-format-text-background); - color: var(--text-color); - @include padding(null, null, null, 10px); - margin: 0; + .core-format-text-arrow { + transform: rotate(90deg); } &:before { @@ -115,26 +134,19 @@ core-format-text { height: 100%; position: absolute; @include position(null, 0, 0, 0); - background: -webkit-linear-gradient(top, rgba(var(--core-format-text-background-gradient-rgb), 0) calc(100% - 50px), rgba(var(--core-format-text-background-gradient-rgb), 1) calc(100% - 15px)); - background: linear-gradient(to bottom, rgba(var(--core-format-text-background-gradient-rgb), 0) calc(100% - 50px), rgba(var(--core-format-text-background-gradient-rgb), 1) calc(100% - 15px)); + background: -webkit-linear-gradient(top, rgba(var(--core-format-text-background-gradient-rgb), 0) calc(100% - 60px), rgba(var(--core-format-text-background-gradient-rgb), 1) calc(100% - 40px)); + background: linear-gradient(to bottom, rgba(var(--core-format-text-background-gradient-rgb), 0) calc(100% - 60px), rgba(var(--core-format-text-background-gradient-rgb), 1) calc(100% - 40px)); z-index: 6; } } - } - &.core-expand-in-fullview { - cursor: pointer; + &.core-text-format-expanded { + max-height: none !important; - .core-show-more { - @include push-arrow-color(626262, true); - @include padding-horizontal(null, 5px); - @include background-position(end, 5px, center); + padding-bottom: 50px; // So the Show less button can fit. - background-repeat: no-repeat; - background-size: 14px 14px; - - @include darkmode() { - @include push-arrow-color(ffffff, true); + .core-format-text-arrow { + transform: rotate(-90deg); } } } @@ -143,10 +155,10 @@ core-format-text { @if ($core-format-text-never-shorten) { &[maxHeight], &[ng-reflect-max-height] { - &.core-text-formatted.core-shortened { + &.core-text-formatted.core-text-format-expanded { max-height: none !important; - .core-show-more { + .core-format-text-toggle { display: none !important; } @@ -354,7 +366,7 @@ core-rich-text-editor .core-rte-editor { height: 30px; display: inline-block; border: 1px solid var(--gray-dark); - background: var(--background-contrast); + background: var(--contrast-background); padding: 6px 8px; border-radius: 4px; margin-left: 2px; @@ -363,7 +375,7 @@ core-rich-text-editor .core-rte-editor { } select { - background-color: var(--background-contrast); + background-color: var(--contrast-background); background-image: url('data:image/svg+xml;charset=US-ASCII,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%22292.4%22%20height%3D%22292.4%22%3E%3Cpath%20fill%3D%22%23000000%22%20d%3D%22M287%2069.4a17.6%2017.6%200%200%200-13-5.4H18.4c-5%200-9.3%201.8-12.9%205.4A17.6%2017.6%200%200%200%200%2082.2c0%205%201.8%209.3%205.4%2012.9l128%20127.9c3.6%203.6%207.8%205.4%2012.8%205.4s9.2-1.8%2012.8-5.4L287%2095c3.5-3.5%205.4-7.8%205.4-12.8%200-5-1.9-9.2-5.5-12.8z%22%2F%3E%3C%2Fsvg%3E'); background-repeat: no-repeat, repeat; background-position: right .7em top 50%, 0 0; diff --git a/src/theme/theme.base.scss b/src/theme/theme.base.scss index c7150accd..12b4b04a1 100644 --- a/src/theme/theme.base.scss +++ b/src/theme/theme.base.scss @@ -848,7 +848,7 @@ core-block ion-item-divider .core-button-spinner { // Text formats. // Highlight text. -.matchtext { +mark, .matchtext { background-color: var(--text-hightlight-background-color); }