commit
8925e5ccb4
|
@ -73,6 +73,7 @@
|
||||||
"addon.block_timeline.noevents": "block_timeline",
|
"addon.block_timeline.noevents": "block_timeline",
|
||||||
"addon.block_timeline.overdue": "block_timeline",
|
"addon.block_timeline.overdue": "block_timeline",
|
||||||
"addon.block_timeline.pluginname": "block_timeline",
|
"addon.block_timeline.pluginname": "block_timeline",
|
||||||
|
"addon.block_timeline.searchevents": "block_timeline",
|
||||||
"addon.block_timeline.sortbycourses": "block_timeline",
|
"addon.block_timeline.sortbycourses": "block_timeline",
|
||||||
"addon.block_timeline.sortbydates": "block_timeline",
|
"addon.block_timeline.sortbydates": "block_timeline",
|
||||||
"addon.blog.blog": "blog",
|
"addon.blog.blog": "blog",
|
||||||
|
|
|
@ -18,6 +18,7 @@ import { CoreSharedModule } from '@/core/shared.module';
|
||||||
|
|
||||||
import { AddonBlockTimelineComponent } from './timeline/timeline';
|
import { AddonBlockTimelineComponent } from './timeline/timeline';
|
||||||
import { AddonBlockTimelineEventsComponent } from './events/events';
|
import { AddonBlockTimelineEventsComponent } from './events/events';
|
||||||
|
import { CoreSearchComponentsModule } from '@features/search/components/components.module';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
|
@ -26,6 +27,7 @@ import { AddonBlockTimelineEventsComponent } from './events/events';
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
CoreSharedModule,
|
CoreSharedModule,
|
||||||
|
CoreSearchComponentsModule,
|
||||||
],
|
],
|
||||||
exports: [
|
exports: [
|
||||||
AddonBlockTimelineComponent,
|
AddonBlockTimelineComponent,
|
||||||
|
|
|
@ -52,7 +52,7 @@ export class AddonBlockTimelineEventsComponent implements OnChanges {
|
||||||
this.showCourse = !this.course;
|
this.showCourse = !this.course;
|
||||||
|
|
||||||
if (changes.events || changes.from || changes.to) {
|
if (changes.events || changes.from || changes.to) {
|
||||||
if (this.events && this.events.length > 0) {
|
if (this.events) {
|
||||||
const filteredEvents = await this.filterEventsByTime(this.from, this.to);
|
const filteredEvents = await this.filterEventsByTime(this.from, this.to);
|
||||||
this.empty = !filteredEvents || filteredEvents.length <= 0;
|
this.empty = !filteredEvents || filteredEvents.length <= 0;
|
||||||
|
|
||||||
|
|
|
@ -2,18 +2,11 @@
|
||||||
<ion-label>
|
<ion-label>
|
||||||
<h2>{{ 'addon.block_timeline.pluginname' | translate }}</h2>
|
<h2>{{ 'addon.block_timeline.pluginname' | translate }}</h2>
|
||||||
</ion-label>
|
</ion-label>
|
||||||
<core-context-menu slot="end">
|
|
||||||
<core-context-menu-item *ngIf="loaded" [priority]="900" [content]="'addon.block_timeline.sortbydates' | translate"
|
|
||||||
(action)="switchSort('sortbydates')" [iconAction]="sort == 'sortbydates' ? 'far-dot-circle' : 'far-circle'">
|
|
||||||
</core-context-menu-item>
|
|
||||||
<core-context-menu-item *ngIf="loaded" [priority]="800" [content]="'addon.block_timeline.sortbycourses' | translate"
|
|
||||||
(action)="switchSort('sortbycourses')" [iconAction]="sort == 'sortbycourses' ? 'far-dot-circle' : 'far-circle'">
|
|
||||||
</core-context-menu-item>
|
|
||||||
</core-context-menu>
|
|
||||||
</ion-item-divider>
|
</ion-item-divider>
|
||||||
<core-loading [hideUntil]="loaded" [fullscreen]="false">
|
<core-loading [hideUntil]="loaded" [fullscreen]="false">
|
||||||
<div class="safe-area-padding-horizontal">
|
<ion-row class="ion-no-padding ion-justify-content-between ion-align-items-center">
|
||||||
<core-combobox [selection]="filter" (onChange)="switchFilter($event)">
|
<ion-col size="auto" class="ion-no-padding">
|
||||||
|
<core-combobox [selection]="filter" (onChange)="switchFilter($event)" icon="fas-filter">
|
||||||
<ion-select-option class="ion-text-wrap" value="all">
|
<ion-select-option class="ion-text-wrap" value="all">
|
||||||
{{ 'core.all' | translate }}
|
{{ 'core.all' | translate }}
|
||||||
</ion-select-option>
|
</ion-select-option>
|
||||||
|
@ -36,7 +29,34 @@
|
||||||
{{ 'addon.block_timeline.next6months' | translate }}
|
{{ 'addon.block_timeline.next6months' | translate }}
|
||||||
</ion-select-option>
|
</ion-select-option>
|
||||||
</core-combobox>
|
</core-combobox>
|
||||||
</div>
|
</ion-col>
|
||||||
|
<ion-col class="ion-no-padding ion-hide-md-down" *ngIf="searchEnabled">
|
||||||
|
<!-- Filter courses. -->
|
||||||
|
<core-search-box (onSubmit)="searchTextChanged($event)" (onClear)="searchTextChanged()"
|
||||||
|
[placeholder]="'addon.block_timeline.searchevents' | translate" autocorrect="off" spellcheck="false" lengthCheck="2"
|
||||||
|
searchArea="AddonBlockTimeline"></core-search-box>
|
||||||
|
</ion-col>
|
||||||
|
<ion-col size="auto" class="ion-no-padding">
|
||||||
|
<core-combobox [label]="'core.sortby' | translate" [selection]="sort" (onChange)="switchSort($event)"
|
||||||
|
icon="fas-sort-amount-down-alt">
|
||||||
|
<ion-select-option class="ion-text-wrap" value="sortbydates">
|
||||||
|
{{'addon.block_timeline.sortbydates' | translate}}
|
||||||
|
</ion-select-option>
|
||||||
|
<ion-select-option class="ion-text-wrap" value="sortbycourses">
|
||||||
|
{{'addon.block_timeline.sortbycourses' | translate}}
|
||||||
|
</ion-select-option>
|
||||||
|
</core-combobox>
|
||||||
|
</ion-col>
|
||||||
|
</ion-row>
|
||||||
|
<ion-row class="ion-no-padding ion-hide-md-up" *ngIf="searchEnabled">
|
||||||
|
<ion-col class="ion-no-padding">
|
||||||
|
<!-- Filter courses. -->
|
||||||
|
<core-search-box (onSubmit)="searchTextChanged($event)" (onClear)="searchTextChanged()"
|
||||||
|
[placeholder]="'addon.block_timeline.searchevents' | translate" autocorrect="off" spellcheck="false" lengthCheck="2"
|
||||||
|
searchArea="AddonBlockTimeline"></core-search-box>
|
||||||
|
</ion-col>
|
||||||
|
</ion-row>
|
||||||
|
|
||||||
<core-loading [hideUntil]="timeline.loaded" [hidden]="sort != 'sortbydates'" [fullscreen]="false">
|
<core-loading [hideUntil]="timeline.loaded" [hidden]="sort != 'sortbydates'" [fullscreen]="false">
|
||||||
<addon-block-timeline-events [events]="timeline.events" [canLoadMore]="timeline.canLoadMore" (loadMore)="loadMore()"
|
<addon-block-timeline-events [events]="timeline.events" [canLoadMore]="timeline.canLoadMore" (loadMore)="loadMore()"
|
||||||
[from]="dataFrom" [to]="dataTo"></addon-block-timeline-events>
|
[from]="dataFrom" [to]="dataTo"></addon-block-timeline-events>
|
||||||
|
|
|
@ -59,6 +59,9 @@ export class AddonBlockTimelineComponent extends CoreBlockBaseComponent implemen
|
||||||
dataFrom?: number;
|
dataFrom?: number;
|
||||||
dataTo?: number;
|
dataTo?: number;
|
||||||
|
|
||||||
|
searchEnabled = false;
|
||||||
|
searchText = '';
|
||||||
|
|
||||||
protected courseIds: number[] = [];
|
protected courseIds: number[] = [];
|
||||||
protected fetchContentDefaultError = 'Error getting timeline data.';
|
protected fetchContentDefaultError = 'Error getting timeline data.';
|
||||||
|
|
||||||
|
@ -85,6 +88,8 @@ export class AddonBlockTimelineComponent extends CoreBlockBaseComponent implemen
|
||||||
|
|
||||||
this.sort = await this.currentSite.getLocalSiteConfig('AddonBlockTimelineSort', this.sort);
|
this.sort = await this.currentSite.getLocalSiteConfig('AddonBlockTimelineSort', this.sort);
|
||||||
|
|
||||||
|
this.searchEnabled = this.currentSite.isVersionGreaterEqualThan('4.0');
|
||||||
|
|
||||||
super.ngOnInit();
|
super.ngOnInit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,7 +140,8 @@ export class AddonBlockTimelineComponent extends CoreBlockBaseComponent implemen
|
||||||
async loadMore(course?: AddonBlockTimelineCourse): Promise<void> {
|
async loadMore(course?: AddonBlockTimelineCourse): Promise<void> {
|
||||||
try {
|
try {
|
||||||
if (course) {
|
if (course) {
|
||||||
const courseEvents = await AddonBlockTimeline.getActionEventsByCourse(course.id, course.canLoadMore);
|
const courseEvents =
|
||||||
|
await AddonBlockTimeline.getActionEventsByCourse(course.id, course.canLoadMore, this.searchText);
|
||||||
course.events = course.events?.concat(courseEvents.events);
|
course.events = course.events?.concat(courseEvents.events);
|
||||||
course.canLoadMore = courseEvents.canLoadMore;
|
course.canLoadMore = courseEvents.canLoadMore;
|
||||||
} else {
|
} else {
|
||||||
|
@ -153,7 +159,7 @@ export class AddonBlockTimelineComponent extends CoreBlockBaseComponent implemen
|
||||||
* @return Promise resolved when done.
|
* @return Promise resolved when done.
|
||||||
*/
|
*/
|
||||||
protected async fetchMyOverviewTimeline(afterEventId?: number): Promise<void> {
|
protected async fetchMyOverviewTimeline(afterEventId?: number): Promise<void> {
|
||||||
const events = await AddonBlockTimeline.getActionEventsByTimesort(afterEventId);
|
const events = await AddonBlockTimeline.getActionEventsByTimesort(afterEventId, this.searchText);
|
||||||
|
|
||||||
this.timeline.events = events.events;
|
this.timeline.events = events.events;
|
||||||
this.timeline.canLoadMore = events.canLoadMore;
|
this.timeline.canLoadMore = events.canLoadMore;
|
||||||
|
@ -174,7 +180,7 @@ export class AddonBlockTimelineComponent extends CoreBlockBaseComponent implemen
|
||||||
if (this.timelineCourses.courses.length > 0) {
|
if (this.timelineCourses.courses.length > 0) {
|
||||||
this.courseIds = this.timelineCourses.courses.map((course) => course.id);
|
this.courseIds = this.timelineCourses.courses.map((course) => course.id);
|
||||||
|
|
||||||
const courseEvents = await AddonBlockTimeline.getActionEventsByCourses(this.courseIds);
|
const courseEvents = await AddonBlockTimeline.getActionEventsByCourses(this.courseIds, this.searchText);
|
||||||
|
|
||||||
this.timelineCourses.courses = this.timelineCourses.courses.filter((course) => {
|
this.timelineCourses.courses = this.timelineCourses.courses.filter((course) => {
|
||||||
if (courseEvents[course.id].events.length == 0) {
|
if (courseEvents[course.id].events.length == 0) {
|
||||||
|
@ -243,6 +249,17 @@ export class AddonBlockTimelineComponent extends CoreBlockBaseComponent implemen
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Search text changed.
|
||||||
|
*
|
||||||
|
* @param searchValue Search value
|
||||||
|
*/
|
||||||
|
searchTextChanged(searchValue = ''): void {
|
||||||
|
this.searchText = searchValue || '';
|
||||||
|
|
||||||
|
this.fetchContent();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export type AddonBlockTimelineCourse = CoreEnrolledCourseDataWithOptions & {
|
export type AddonBlockTimelineCourse = CoreEnrolledCourseDataWithOptions & {
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
"noevents": "No upcoming activities due",
|
"noevents": "No upcoming activities due",
|
||||||
"overdue": "Overdue",
|
"overdue": "Overdue",
|
||||||
"pluginname": "Timeline",
|
"pluginname": "Timeline",
|
||||||
|
"searchevents": "Search by activity type or name",
|
||||||
"sortbycourses": "Sort by courses",
|
"sortbycourses": "Sort by courses",
|
||||||
"sortbydates": "Sort by dates"
|
"sortbydates": "Sort by dates"
|
||||||
}
|
}
|
|
@ -26,7 +26,6 @@ import {
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import { makeSingleton } from '@singletons';
|
import { makeSingleton } from '@singletons';
|
||||||
import { CoreSiteWSPreSets } from '@classes/site';
|
import { CoreSiteWSPreSets } from '@classes/site';
|
||||||
import { CoreError } from '@classes/errors/error';
|
|
||||||
|
|
||||||
// Cache key was maintained from block myoverview when blocks were splitted.
|
// Cache key was maintained from block myoverview when blocks were splitted.
|
||||||
const ROOT_CACHE_KEY = 'myoverview:';
|
const ROOT_CACHE_KEY = 'myoverview:';
|
||||||
|
@ -45,12 +44,14 @@ export class AddonBlockTimelineProvider {
|
||||||
*
|
*
|
||||||
* @param courseId Only events in this course.
|
* @param courseId Only events in this course.
|
||||||
* @param afterEventId The last seen event id.
|
* @param afterEventId The last seen event id.
|
||||||
|
* @param searchValue The value a user wishes to search against.
|
||||||
* @param siteId Site ID. If not defined, use current site.
|
* @param siteId Site ID. If not defined, use current site.
|
||||||
* @return Promise resolved when the info is retrieved.
|
* @return Promise resolved when the info is retrieved.
|
||||||
*/
|
*/
|
||||||
async getActionEventsByCourse(
|
async getActionEventsByCourse(
|
||||||
courseId: number,
|
courseId: number,
|
||||||
afterEventId?: number,
|
afterEventId?: number,
|
||||||
|
searchValue = '',
|
||||||
siteId?: string,
|
siteId?: string,
|
||||||
): Promise<{ events: AddonCalendarEvent[]; canLoadMore?: number }> {
|
): Promise<{ events: AddonCalendarEvent[]; canLoadMore?: number }> {
|
||||||
const site = await CoreSites.getSite(siteId);
|
const site = await CoreSites.getSite(siteId);
|
||||||
|
@ -70,19 +71,20 @@ export class AddonBlockTimelineProvider {
|
||||||
cacheKey: this.getActionEventsByCourseCacheKey(courseId),
|
cacheKey: this.getActionEventsByCourseCacheKey(courseId),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (searchValue != '') {
|
||||||
|
data.searchvalue = searchValue;
|
||||||
|
preSets.getFromCache = false;
|
||||||
|
}
|
||||||
|
|
||||||
const courseEvents = await site.read<AddonCalendarEvents>(
|
const courseEvents = await site.read<AddonCalendarEvents>(
|
||||||
'core_calendar_get_action_events_by_course',
|
'core_calendar_get_action_events_by_course',
|
||||||
data,
|
data,
|
||||||
preSets,
|
preSets,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (courseEvents && courseEvents.events) {
|
|
||||||
return this.treatCourseEvents(courseEvents, time);
|
return this.treatCourseEvents(courseEvents, time);
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new CoreError('No events returned on core_calendar_get_action_events_by_course.');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get cache key for get calendar action events for the given course value WS call.
|
* Get cache key for get calendar action events for the given course value WS call.
|
||||||
*
|
*
|
||||||
|
@ -98,10 +100,12 @@ export class AddonBlockTimelineProvider {
|
||||||
*
|
*
|
||||||
* @param courseIds Course IDs.
|
* @param courseIds Course IDs.
|
||||||
* @param siteId Site ID. If not defined, use current site.
|
* @param siteId Site ID. If not defined, use current site.
|
||||||
|
* @param searchValue The value a user wishes to search against.
|
||||||
* @return Promise resolved when the info is retrieved.
|
* @return Promise resolved when the info is retrieved.
|
||||||
*/
|
*/
|
||||||
async getActionEventsByCourses(
|
async getActionEventsByCourses(
|
||||||
courseIds: number[],
|
courseIds: number[],
|
||||||
|
searchValue = '',
|
||||||
siteId?: string,
|
siteId?: string,
|
||||||
): Promise<{[courseId: string]: { events: AddonCalendarEvent[]; canLoadMore?: number } }> {
|
): Promise<{[courseId: string]: { events: AddonCalendarEvent[]; canLoadMore?: number } }> {
|
||||||
const site = await CoreSites.getSite(siteId);
|
const site = await CoreSites.getSite(siteId);
|
||||||
|
@ -117,6 +121,11 @@ export class AddonBlockTimelineProvider {
|
||||||
cacheKey: this.getActionEventsByCoursesCacheKey(),
|
cacheKey: this.getActionEventsByCoursesCacheKey(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (searchValue != '') {
|
||||||
|
data.searchvalue = searchValue;
|
||||||
|
preSets.getFromCache = false;
|
||||||
|
}
|
||||||
|
|
||||||
const events = await site.read<AddonCalendarEventsGroupedByCourse>(
|
const events = await site.read<AddonCalendarEventsGroupedByCourse>(
|
||||||
'core_calendar_get_action_events_by_courses',
|
'core_calendar_get_action_events_by_courses',
|
||||||
data,
|
data,
|
||||||
|
@ -145,11 +154,13 @@ export class AddonBlockTimelineProvider {
|
||||||
* Get calendar action events based on the timesort value.
|
* Get calendar action events based on the timesort value.
|
||||||
*
|
*
|
||||||
* @param afterEventId The last seen event id.
|
* @param afterEventId The last seen event id.
|
||||||
|
* @param searchValue The value a user wishes to search against.
|
||||||
* @param siteId Site ID. If not defined, use current site.
|
* @param siteId Site ID. If not defined, use current site.
|
||||||
* @return Promise resolved when the info is retrieved.
|
* @return Promise resolved when the info is retrieved.
|
||||||
*/
|
*/
|
||||||
async getActionEventsByTimesort(
|
async getActionEventsByTimesort(
|
||||||
afterEventId?: number,
|
afterEventId?: number,
|
||||||
|
searchValue = '',
|
||||||
siteId?: string,
|
siteId?: string,
|
||||||
): Promise<{ events: AddonCalendarEvent[]; canLoadMore?: number }> {
|
): Promise<{ events: AddonCalendarEvent[]; canLoadMore?: number }> {
|
||||||
const site = await CoreSites.getSite(siteId);
|
const site = await CoreSites.getSite(siteId);
|
||||||
|
@ -171,13 +182,17 @@ export class AddonBlockTimelineProvider {
|
||||||
uniqueCacheKey: true,
|
uniqueCacheKey: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (searchValue != '') {
|
||||||
|
data.searchvalue = searchValue;
|
||||||
|
preSets.getFromCache = false;
|
||||||
|
}
|
||||||
|
|
||||||
const result = await site.read<AddonCalendarEvents>(
|
const result = await site.read<AddonCalendarEvents>(
|
||||||
'core_calendar_get_action_events_by_timesort',
|
'core_calendar_get_action_events_by_timesort',
|
||||||
data,
|
data,
|
||||||
preSets,
|
preSets,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (result && result.events) {
|
|
||||||
const canLoadMore = result.events.length >= limitnum ? result.lastid : undefined;
|
const canLoadMore = result.events.length >= limitnum ? result.lastid : undefined;
|
||||||
|
|
||||||
// Filter events by time in case it uses cache.
|
// Filter events by time in case it uses cache.
|
||||||
|
@ -189,9 +204,6 @@ export class AddonBlockTimelineProvider {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new CoreError('No events returned on core_calendar_get_action_events_by_timesort.');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get prefix cache key for calendar action events based on the timesort value WS calls.
|
* Get prefix cache key for calendar action events based on the timesort value WS calls.
|
||||||
*
|
*
|
||||||
|
|
|
@ -1785,6 +1785,7 @@ export type AddonCalendarGetActionEventsByCoursesWSParams = {
|
||||||
timesortfrom?: number; // Time sort from.
|
timesortfrom?: number; // Time sort from.
|
||||||
timesortto?: number; // Time sort to.
|
timesortto?: number; // Time sort to.
|
||||||
limitnum?: number; // Limit number.
|
limitnum?: number; // Limit number.
|
||||||
|
searchvalue?: string; // The value a user wishes to search against.
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1804,6 +1805,7 @@ export type AddonCalendarGetActionEventsByCourseWSParams = {
|
||||||
timesortto?: number; // Time sort to.
|
timesortto?: number; // Time sort to.
|
||||||
aftereventid?: number; // The last seen event id.
|
aftereventid?: number; // The last seen event id.
|
||||||
limitnum?: number; // Limit number.
|
limitnum?: number; // Limit number.
|
||||||
|
searchvalue?: string; // The value a user wishes to search against.
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1816,6 +1818,7 @@ export type AddonCalendarGetActionEventsByTimesortWSParams = {
|
||||||
limitnum?: number; // Limit number.
|
limitnum?: number; // Limit number.
|
||||||
limittononsuspendedevents?: boolean; // Limit the events to courses the user is not suspended in.
|
limittononsuspendedevents?: boolean; // Limit the events to courses the user is not suspended in.
|
||||||
userid?: number; // The user id.
|
userid?: number; // The user id.
|
||||||
|
searchvalue?: string; // The value a user wishes to search against.
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
:host {
|
:host {
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
|
display: block;
|
||||||
@include margin-horizontal(var(--ion-safe-area-left), var(--ion-safe-area-right));
|
@include margin-horizontal(var(--ion-safe-area-left), var(--ion-safe-area-right));
|
||||||
|
|
||||||
ion-select,
|
ion-select,
|
||||||
|
@ -87,6 +88,10 @@ ion-select {
|
||||||
background: var(--background-focused);
|
background: var(--background-focused);
|
||||||
opacity: var(--background-focused-opacity);
|
opacity: var(--background-focused-opacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&[hidden] {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ion-button {
|
ion-button {
|
||||||
|
|
|
@ -12,10 +12,11 @@
|
||||||
// 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 { Component, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core';
|
import { Component, EventEmitter, Input, Output, ViewChild, ViewEncapsulation } from '@angular/core';
|
||||||
import { Translate } from '@singletons';
|
import { Translate } from '@singletons';
|
||||||
import { ModalOptions } from '@ionic/core';
|
import { ModalOptions } from '@ionic/core';
|
||||||
import { CoreDomUtils } from '@services/utils/dom';
|
import { CoreDomUtils } from '@services/utils/dom';
|
||||||
|
import { IonSelect } from '@ionic/angular';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Component that show a combo select button (combobox).
|
* Component that show a combo select button (combobox).
|
||||||
|
@ -43,6 +44,8 @@ import { CoreDomUtils } from '@services/utils/dom';
|
||||||
})
|
})
|
||||||
export class CoreComboboxComponent {
|
export class CoreComboboxComponent {
|
||||||
|
|
||||||
|
@ViewChild(IonSelect) select!: IonSelect;
|
||||||
|
|
||||||
@Input() interface: 'popover' | 'modal' = 'popover';
|
@Input() interface: 'popover' | 'modal' = 'popover';
|
||||||
@Input() label = Translate.instant('core.show'); // Aria label.
|
@Input() label = Translate.instant('core.show'); // Aria label.
|
||||||
@Input() disabled = false;
|
@Input() disabled = false;
|
||||||
|
@ -51,12 +54,20 @@ export class CoreComboboxComponent {
|
||||||
|
|
||||||
// Additional options when interface modal is selected.
|
// Additional options when interface modal is selected.
|
||||||
@Input() icon?: string; // Icon for modal interface.
|
@Input() icon?: string; // Icon for modal interface.
|
||||||
|
@Input() badge?: number; // Badge number to show near the icon.
|
||||||
@Input() modalOptions?: ModalOptions; // Will emit an event the value changed.
|
@Input() modalOptions?: ModalOptions; // Will emit an event the value changed.
|
||||||
@Input() listboxId = '';
|
@Input() listboxId = '';
|
||||||
|
|
||||||
expanded = false;
|
expanded = false;
|
||||||
|
|
||||||
async showModal(): Promise<void> {
|
/**
|
||||||
|
* Shows combobox modal.
|
||||||
|
*
|
||||||
|
* @param event Event.
|
||||||
|
* @return Promise resolved when done.
|
||||||
|
*/
|
||||||
|
async openSelect(event?: UIEvent): Promise<void> {
|
||||||
|
if (this.interface == 'modal') {
|
||||||
if (this.expanded || !this.modalOptions) {
|
if (this.expanded || !this.modalOptions) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -72,6 +83,9 @@ export class CoreComboboxComponent {
|
||||||
if (data) {
|
if (data) {
|
||||||
this.onChange.emit(data);
|
this.onChange.emit(data);
|
||||||
}
|
}
|
||||||
|
} else if (this.select) {
|
||||||
|
this.select.open(event);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,27 +1,20 @@
|
||||||
<ion-select
|
<ion-button (click)="openSelect($event)" color="light" *ngIf="interface != 'modal' && icon" [disabled]="disabled">
|
||||||
*ngIf="interface != 'modal'"
|
<ion-icon [name]="icon" [attr.aria-label]="label" slot="start">
|
||||||
class="ion-text-start"
|
</ion-icon>
|
||||||
[(ngModel)]="selection"
|
<ion-badge *ngIf="badge && badge > 0" slot="start">{{badge}}</ion-badge>
|
||||||
(ngModelChange)="onChange.emit(selection)"
|
<div class="select-icon" role="presentation" aria-hidden="true">
|
||||||
[interface]="interface"
|
<div class="select-icon-inner"></div>
|
||||||
[attr.aria-label]="label + ': ' + selection"
|
</div>
|
||||||
[disabled]="disabled"
|
</ion-button>
|
||||||
>
|
<ion-select *ngIf="interface != 'modal'" class="ion-text-start" [(ngModel)]="selection" (ngModelChange)="onChange.emit(selection)"
|
||||||
|
[interface]="interface" [attr.aria-label]="label + ': ' + selection" [disabled]="disabled" [hidden]="!!icon">
|
||||||
<ng-content></ng-content>
|
<ng-content></ng-content>
|
||||||
</ion-select>
|
</ion-select>
|
||||||
|
|
||||||
<ion-button
|
<ion-button *ngIf="interface == 'modal'" aria-haspopup="listbox" [attr.aria-controls]="listboxId" [attr.aria-owns]="listboxId"
|
||||||
*ngIf="interface == 'modal'"
|
[attr.aria-expanded]="expanded" (click)="openSelect()" [disabled]="disabled" expand="block" role="combobox">
|
||||||
aria-haspopup="listbox"
|
|
||||||
aria-controls="addon-mod-forum-sort-order-selector"
|
|
||||||
[attr.aria-owns]="listboxId"
|
|
||||||
[attr.aria-expanded]="expanded"
|
|
||||||
(click)="showModal()"
|
|
||||||
[disabled]="disabled"
|
|
||||||
expand="block"
|
|
||||||
role="combobox"
|
|
||||||
>
|
|
||||||
<ion-icon *ngIf="icon" [name]="icon" slot="start" aria-hidden="true"></ion-icon>
|
<ion-icon *ngIf="icon" [name]="icon" slot="start" aria-hidden="true"></ion-icon>
|
||||||
|
<ion-badge *ngIf="badge && badge > 0" slot="start">{{badge}}</ion-badge>
|
||||||
<span class="sr-only" *ngIf="label">{{ label }}:</span>
|
<span class="sr-only" *ngIf="label">{{ label }}:</span>
|
||||||
<div class="select-text">
|
<div class="select-text">
|
||||||
<slot name="text">{{selection}}</slot>
|
<slot name="text">{{selection}}</slot>
|
||||||
|
|
|
@ -52,7 +52,7 @@ export class CoreAutoFocusDirective implements AfterViewInit {
|
||||||
/**
|
/**
|
||||||
* Function to focus the element.
|
* Function to focus the element.
|
||||||
*
|
*
|
||||||
* @param retries Internal param to stop retrying then 0.
|
* @param retries Internal param to stop retrying on 0.
|
||||||
*/
|
*/
|
||||||
protected setFocus(retries = 10): void {
|
protected setFocus(retries = 10): void {
|
||||||
if (retries == 0) {
|
if (retries == 0) {
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
@import "~theme/globals";
|
||||||
|
|
||||||
|
:host ::ng-deep core-block {
|
||||||
|
@include media-breakpoint-up(md) {
|
||||||
|
.ion-hide-md-down {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ion-hide-md-up {
|
||||||
|
display: block !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -28,6 +28,7 @@ import { CoreCoursesDashboard } from '@features/courses/services/dashboard';
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'core-block-side-blocks',
|
selector: 'core-block-side-blocks',
|
||||||
templateUrl: 'side-blocks.html',
|
templateUrl: 'side-blocks.html',
|
||||||
|
styleUrls: ['side-blocks.scss'],
|
||||||
})
|
})
|
||||||
export class CoreBlockSideBlocksComponent implements OnInit {
|
export class CoreBlockSideBlocksComponent implements OnInit {
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
:host {
|
:host {
|
||||||
height: 73px;
|
min-height: 61px;
|
||||||
display: block;
|
display: block;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
|
||||||
|
@ -8,13 +8,12 @@
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
z-index: 4;
|
z-index: 4;
|
||||||
margin-top: 10px;
|
margin-top: 8px;
|
||||||
margin-bottom: 10px;
|
margin-bottom: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
ion-button.button {
|
ion-button.button {
|
||||||
margin-left: 0;
|
margin: 0;
|
||||||
margin-right: 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.core-search-history {
|
.core-search-history {
|
||||||
|
@ -37,4 +36,9 @@
|
||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
padding-right: 0;
|
padding-right: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ion-item {
|
||||||
|
--min-height: var(--a11y-min-target-size);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -41,8 +41,8 @@ export class CoreSearchBoxComponent implements OnInit {
|
||||||
@Input() searchLabel?: string; // Label to be used on action button.
|
@Input() searchLabel?: string; // Label to be used on action button.
|
||||||
@Input() placeholder?: string; // Placeholder text for search text input.
|
@Input() placeholder?: string; // Placeholder text for search text input.
|
||||||
@Input() autocorrect = 'on'; // Enables/disable Autocorrection on search text input.
|
@Input() autocorrect = 'on'; // Enables/disable Autocorrection on search text input.
|
||||||
@Input() spellcheck?: string | boolean = true; // Enables/disable Spellchecker on search text input.
|
@Input() spellcheck: string | boolean = true; // Enables/disable Spellchecker on search text input.
|
||||||
@Input() autoFocus?: string | boolean; // Enables/disable Autofocus when entering view.
|
@Input() autoFocus: string | boolean = false; // Enables/disable Autofocus when entering view.
|
||||||
@Input() lengthCheck = 3; // Check value length before submit. If 0, any string will be submitted.
|
@Input() lengthCheck = 3; // Check value length before submit. If 0, any string will be submitted.
|
||||||
@Input() showClear = true; // Show/hide clear button.
|
@Input() showClear = true; // Show/hide clear button.
|
||||||
@Input() disabled = false; // Disables the input text.
|
@Input() disabled = false; // Disables the input text.
|
||||||
|
|
|
@ -172,6 +172,7 @@ ion-app.ios ion-header h2 {
|
||||||
.item.ion-text-wrap ion-label core-format-text .core-format-text-content > *,
|
.item.ion-text-wrap ion-label core-format-text .core-format-text-content > *,
|
||||||
.fake-ion-item.ion-text-wrap core-format-text .core-format-text-content > * {
|
.fake-ion-item.ion-text-wrap core-format-text .core-format-text-content > * {
|
||||||
white-space: normal;
|
white-space: normal;
|
||||||
|
overflow: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
.item.ion-text-wrap ion-label {
|
.item.ion-text-wrap ion-label {
|
||||||
|
|
Loading…
Reference in New Issue