2017-03-08 13:08:19 -08:00
|
|
|
/*
|
|
|
|
* API List & Filter Component
|
|
|
|
*
|
|
|
|
* A page that displays a formatted list of the public Angular API entities.
|
|
|
|
* Clicking on a list item triggers navigation to the corresponding API entity document.
|
|
|
|
* Can add/remove API entity links based on filter settings.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { Component, ElementRef, OnInit, ViewChild } from '@angular/core';
|
|
|
|
|
2018-03-20 18:22:59 +02:00
|
|
|
import { combineLatest, Observable, ReplaySubject } from 'rxjs';
|
2017-03-08 13:08:19 -08:00
|
|
|
|
|
|
|
import { LocationService } from 'app/shared/location.service';
|
2017-10-11 21:36:33 +01:00
|
|
|
import { ApiSection, ApiService } from './api.service';
|
2017-03-08 13:08:19 -08:00
|
|
|
|
2017-06-06 10:09:46 +01:00
|
|
|
import { Option } from 'app/shared/select/select.component';
|
2017-03-08 13:08:19 -08:00
|
|
|
|
|
|
|
class SearchCriteria {
|
2018-04-14 00:44:09 -07:00
|
|
|
query ? = '';
|
|
|
|
status ? = 'all';
|
|
|
|
type ? = 'all';
|
2017-03-08 13:08:19 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
selector: 'aio-api-list',
|
2018-02-28 12:05:59 -08:00
|
|
|
templateUrl: './api-list.component.html',
|
2017-03-08 13:08:19 -08:00
|
|
|
})
|
|
|
|
export class ApiListComponent implements OnInit {
|
|
|
|
|
|
|
|
filteredSections: Observable<ApiSection[]>;
|
|
|
|
|
|
|
|
showStatusMenu = false;
|
|
|
|
showTypeMenu = false;
|
|
|
|
|
|
|
|
private criteriaSubject = new ReplaySubject<SearchCriteria>(1);
|
|
|
|
private searchCriteria = new SearchCriteria();
|
|
|
|
|
2017-06-06 10:09:46 +01:00
|
|
|
status: Option;
|
|
|
|
type: Option;
|
2017-03-08 13:08:19 -08:00
|
|
|
|
|
|
|
// API types
|
2017-06-06 10:09:46 +01:00
|
|
|
types: Option[] = [
|
|
|
|
{ value: 'all', title: 'All' },
|
|
|
|
{ value: 'class', title: 'Class' },
|
2017-12-26 14:11:58 -06:00
|
|
|
{ value: 'const', title: 'Const'},
|
|
|
|
{ value: 'decorator', title: 'Decorator' },
|
|
|
|
{ value: 'directive', title: 'Directive' },
|
2017-06-06 10:09:46 +01:00
|
|
|
{ value: 'enum', title: 'Enum' },
|
2017-12-26 14:11:58 -06:00
|
|
|
{ value: 'function', title: 'Function' },
|
|
|
|
{ value: 'interface', title: 'Interface' },
|
|
|
|
{ value: 'pipe', title: 'Pipe'},
|
2018-07-01 21:23:45 +01:00
|
|
|
{ value: 'type-alias', title: 'Type alias' },
|
|
|
|
{ value: 'package', title: 'Package'}
|
2017-03-08 13:08:19 -08:00
|
|
|
];
|
|
|
|
|
2017-06-06 10:09:46 +01:00
|
|
|
statuses: Option[] = [
|
|
|
|
{ value: 'all', title: 'All' },
|
|
|
|
{ value: 'stable', title: 'Stable' },
|
|
|
|
{ value: 'deprecated', title: 'Deprecated' },
|
|
|
|
{ value: 'experimental', title: 'Experimental' },
|
|
|
|
{ value: 'security-risk', title: 'Security Risk' }
|
2017-03-08 13:08:19 -08:00
|
|
|
];
|
|
|
|
|
|
|
|
@ViewChild('filter') queryEl: ElementRef;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
private apiService: ApiService,
|
|
|
|
private locationService: LocationService) { }
|
|
|
|
|
|
|
|
ngOnInit() {
|
|
|
|
this.filteredSections = combineLatest(
|
|
|
|
this.apiService.sections,
|
|
|
|
this.criteriaSubject,
|
|
|
|
(sections, criteria) => {
|
2018-07-01 21:23:45 +01:00
|
|
|
return sections
|
|
|
|
.map(section => ({ ...section, items: this.filterSection(section, criteria) }));
|
2017-03-08 13:08:19 -08:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
this.initializeSearchCriteria();
|
|
|
|
}
|
|
|
|
|
2018-04-08 18:19:25 -03:00
|
|
|
// TODO: may need to debounce as the original did
|
2017-03-08 13:08:19 -08:00
|
|
|
// although there shouldn't be any perf consequences if we don't
|
|
|
|
setQuery(query: string) {
|
|
|
|
this.setSearchCriteria({query: (query || '').toLowerCase().trim() });
|
|
|
|
}
|
|
|
|
|
2017-06-06 10:09:46 +01:00
|
|
|
setStatus(status: Option) {
|
2017-03-08 13:08:19 -08:00
|
|
|
this.toggleStatusMenu();
|
|
|
|
this.status = status;
|
2017-06-06 10:09:46 +01:00
|
|
|
this.setSearchCriteria({status: status.value});
|
2017-03-08 13:08:19 -08:00
|
|
|
}
|
|
|
|
|
2017-06-06 10:09:46 +01:00
|
|
|
setType(type: Option) {
|
2017-03-08 13:08:19 -08:00
|
|
|
this.toggleTypeMenu();
|
|
|
|
this.type = type;
|
2017-06-06 10:09:46 +01:00
|
|
|
this.setSearchCriteria({type: type.value});
|
2017-03-08 13:08:19 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
toggleStatusMenu() {
|
|
|
|
this.showStatusMenu = !this.showStatusMenu;
|
|
|
|
}
|
|
|
|
|
|
|
|
toggleTypeMenu() {
|
|
|
|
this.showTypeMenu = !this.showTypeMenu;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////// Private //////////
|
|
|
|
|
|
|
|
private filterSection(section: ApiSection, { query, status, type }: SearchCriteria) {
|
2018-07-01 21:23:45 +01:00
|
|
|
const items = section.items!.filter(item => {
|
|
|
|
return matchesType() && matchesStatus() && matchesQuery();
|
2017-03-08 13:08:19 -08:00
|
|
|
|
|
|
|
function matchesQuery() {
|
|
|
|
return !query ||
|
|
|
|
section.name.indexOf(query) !== -1 ||
|
|
|
|
item.name.indexOf(query) !== -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
function matchesStatus() {
|
|
|
|
return status === 'all' ||
|
|
|
|
status === item.stability ||
|
|
|
|
(status === 'security-risk' && item.securityRisk);
|
|
|
|
};
|
|
|
|
|
|
|
|
function matchesType() {
|
|
|
|
return type === 'all' || type === item.docType;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-07-01 21:23:45 +01:00
|
|
|
// If there are no items we still return an empty array if the section name matches and the type is 'package'
|
|
|
|
return items.length ? items : (type === 'package' && (!query || section.name.indexOf(query) !== -1)) ? [] : null;
|
2017-03-08 13:08:19 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get initial search criteria from URL search params
|
|
|
|
private initializeSearchCriteria() {
|
|
|
|
const {query, status, type} = this.locationService.search();
|
|
|
|
|
|
|
|
const q = (query || '').toLowerCase();
|
|
|
|
// Hack: can't bind to query because input cursor always forced to end-of-line.
|
|
|
|
this.queryEl.nativeElement.value = q;
|
|
|
|
|
2017-06-06 10:09:46 +01:00
|
|
|
this.status = this.statuses.find(x => x.value === status) || this.statuses[0];
|
|
|
|
this.type = this.types.find(x => x.value === type) || this.types[0];
|
2017-03-08 13:08:19 -08:00
|
|
|
|
|
|
|
this.searchCriteria = {
|
|
|
|
query: q,
|
2017-06-06 10:09:46 +01:00
|
|
|
status: this.status.value,
|
|
|
|
type: this.type.value
|
2017-03-08 13:08:19 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
this.criteriaSubject.next(this.searchCriteria);
|
|
|
|
}
|
|
|
|
|
|
|
|
private setLocationSearch() {
|
|
|
|
const {query, status, type} = this.searchCriteria;
|
|
|
|
const params = {
|
|
|
|
query: query ? query : undefined,
|
|
|
|
status: status !== 'all' ? status : undefined,
|
|
|
|
type: type !== 'all' ? type : undefined
|
|
|
|
};
|
|
|
|
|
|
|
|
this.locationService.setSearch('API Search', params);
|
|
|
|
}
|
|
|
|
|
|
|
|
private setSearchCriteria(criteria: SearchCriteria) {
|
|
|
|
this.criteriaSubject.next(Object.assign(this.searchCriteria, criteria));
|
|
|
|
this.setLocationSearch();
|
|
|
|
}
|
|
|
|
}
|