mirror of
https://github.com/richard-loafle/fuse-angular.git
synced 2025-01-10 04:25:08 +00:00
Updated Angular Material elements
Updated the changelog
This commit is contained in:
parent
6fadc29e4c
commit
60ab983730
|
@ -9,7 +9,6 @@ import { ButtonToggleOverviewExample } from 'assets/angular-material-examples/bu
|
||||||
import { ButtonTypesExample } from 'assets/angular-material-examples/button-types/button-types-example';
|
import { ButtonTypesExample } from 'assets/angular-material-examples/button-types/button-types-example';
|
||||||
import { CardFancyExample } from 'assets/angular-material-examples/card-fancy/card-fancy-example';
|
import { CardFancyExample } from 'assets/angular-material-examples/card-fancy/card-fancy-example';
|
||||||
import { CardOverviewExample } from 'assets/angular-material-examples/card-overview/card-overview-example';
|
import { CardOverviewExample } from 'assets/angular-material-examples/card-overview/card-overview-example';
|
||||||
import { CdkTableBasicExample } from 'assets/angular-material-examples/cdk-table-basic/cdk-table-basic-example';
|
|
||||||
import { CheckboxConfigurableExample } from 'assets/angular-material-examples/checkbox-configurable/checkbox-configurable-example';
|
import { CheckboxConfigurableExample } from 'assets/angular-material-examples/checkbox-configurable/checkbox-configurable-example';
|
||||||
import { CheckboxOverviewExample } from 'assets/angular-material-examples/checkbox-overview/checkbox-overview-example';
|
import { CheckboxOverviewExample } from 'assets/angular-material-examples/checkbox-overview/checkbox-overview-example';
|
||||||
import { ChipsInputExample } from 'assets/angular-material-examples/chips-input/chips-input-example';
|
import { ChipsInputExample } from 'assets/angular-material-examples/chips-input/chips-input-example';
|
||||||
|
@ -111,8 +110,6 @@ import { FormFieldLabelExample } from 'assets/angular-material-examples/form-fie
|
||||||
import { AutocompleteOptgroupExample } from 'assets/angular-material-examples/autocomplete-optgroup/autocomplete-optgroup-example';
|
import { AutocompleteOptgroupExample } from 'assets/angular-material-examples/autocomplete-optgroup/autocomplete-optgroup-example';
|
||||||
import { BadgeOverviewExample } from 'assets/angular-material-examples/badge-overview/badge-overview-example';
|
import { BadgeOverviewExample } from 'assets/angular-material-examples/badge-overview/badge-overview-example';
|
||||||
import { BottomSheetOverviewExample, BottomSheetOverviewExampleSheet } from 'assets/angular-material-examples/bottom-sheet-overview/bottom-sheet-overview-example';
|
import { BottomSheetOverviewExample, BottomSheetOverviewExampleSheet } from 'assets/angular-material-examples/bottom-sheet-overview/bottom-sheet-overview-example';
|
||||||
import { CdkTreeFlatExample } from 'assets/angular-material-examples/cdk-tree-flat/cdk-tree-flat-example';
|
|
||||||
import { CdkTreeNestedExample } from 'assets/angular-material-examples/cdk-tree-nested/cdk-tree-nested-example';
|
|
||||||
import { ChipsAutocompleteExample } from 'assets/angular-material-examples/chips-autocomplete/chips-autocomplete-example';
|
import { ChipsAutocompleteExample } from 'assets/angular-material-examples/chips-autocomplete/chips-autocomplete-example';
|
||||||
import { DatepickerColorExample } from 'assets/angular-material-examples/datepicker-color/datepicker-color-example';
|
import { DatepickerColorExample } from 'assets/angular-material-examples/datepicker-color/datepicker-color-example';
|
||||||
import { DatepickerCustomHeaderExample, ExampleHeader } from 'assets/angular-material-examples/datepicker-custom-header/datepicker-custom-header-example';
|
import { DatepickerCustomHeaderExample, ExampleHeader } from 'assets/angular-material-examples/datepicker-custom-header/datepicker-custom-header-example';
|
||||||
|
@ -143,7 +140,6 @@ import { TreeDynamicExample } from 'assets/angular-material-examples/tree-dynami
|
||||||
import { TreeFlatOverviewExample } from 'assets/angular-material-examples/tree-flat-overview/tree-flat-overview-example';
|
import { TreeFlatOverviewExample } from 'assets/angular-material-examples/tree-flat-overview/tree-flat-overview-example';
|
||||||
import { TreeLoadmoreExample } from 'assets/angular-material-examples/tree-loadmore/tree-loadmore-example';
|
import { TreeLoadmoreExample } from 'assets/angular-material-examples/tree-loadmore/tree-loadmore-example';
|
||||||
import { TreeNestedOverviewExample } from 'assets/angular-material-examples/tree-nested-overview/tree-nested-overview-example';
|
import { TreeNestedOverviewExample } from 'assets/angular-material-examples/tree-nested-overview/tree-nested-overview-example';
|
||||||
import { CdkTableBasicFlexExample } from 'assets/angular-material-examples/cdk-table-basic-flex/cdk-table-basic-flex-example';
|
|
||||||
import { StepperVerticalExample } from 'assets/angular-material-examples/stepper-vertical/stepper-vertical-example';
|
import { StepperVerticalExample } from 'assets/angular-material-examples/stepper-vertical/stepper-vertical-example';
|
||||||
import { TabGroupBasicExample } from 'assets/angular-material-examples/tab-group-basic/tab-group-basic-example';
|
import { TabGroupBasicExample } from 'assets/angular-material-examples/tab-group-basic/tab-group-basic-example';
|
||||||
import { TabGroupAsyncExample } from 'assets/angular-material-examples/tab-group-async/tab-group-async-example';
|
import { TabGroupAsyncExample } from 'assets/angular-material-examples/tab-group-async/tab-group-async-example';
|
||||||
|
@ -169,6 +165,14 @@ import { TableStickyComplexExample } from 'assets/angular-material-examples/tabl
|
||||||
import { TableStickyComplexFlexExample } from 'assets/angular-material-examples/table-sticky-complex-flex/table-sticky-complex-flex-example';
|
import { TableStickyComplexFlexExample } from 'assets/angular-material-examples/table-sticky-complex-flex/table-sticky-complex-flex-example';
|
||||||
import { TableStickyFooterExample } from 'assets/angular-material-examples/table-sticky-footer/table-sticky-footer-example';
|
import { TableStickyFooterExample } from 'assets/angular-material-examples/table-sticky-footer/table-sticky-footer-example';
|
||||||
import { TableStickyHeaderExample } from 'assets/angular-material-examples/table-sticky-header/table-sticky-header-example';
|
import { TableStickyHeaderExample } from 'assets/angular-material-examples/table-sticky-header/table-sticky-header-example';
|
||||||
|
import { ButtonToggleAppearanceExample } from 'assets/angular-material-examples/button-toggle-appearance/button-toggle-appearance-example';
|
||||||
|
import { RippleOverviewExample } from 'assets/angular-material-examples/ripple-overview/ripple-overview-example';
|
||||||
|
import { StepperLabelPositionBottomExample } from 'assets/angular-material-examples/stepper-label-position-bottom/stepper-label-position-bottom-example';
|
||||||
|
import { StepperStatesExample } from 'assets/angular-material-examples/stepper-states/stepper-states-example';
|
||||||
|
import { StepperErrorsExample } from 'assets/angular-material-examples/stepper-errors/stepper-errors-example';
|
||||||
|
import { TabGroupAlignExample } from 'assets/angular-material-examples/tab-group-align/tab-group-align-example';
|
||||||
|
import { SimpleColumn, TableSimpleColumnExample } from 'assets/angular-material-examples/table-simple-column/table-simple-column-example';
|
||||||
|
import { TableWrappedExample, WrapperTable } from 'assets/angular-material-examples/table-wrapped/table-wrapped-example';
|
||||||
|
|
||||||
export const COMPONENT_MAP = {
|
export const COMPONENT_MAP = {
|
||||||
'autocomplete' : [
|
'autocomplete' : [
|
||||||
|
@ -191,20 +195,13 @@ export const COMPONENT_MAP = {
|
||||||
],
|
],
|
||||||
'button-toggle' : [
|
'button-toggle' : [
|
||||||
'button-toggle-overview',
|
'button-toggle-overview',
|
||||||
|
'button-toggle-appearance',
|
||||||
'button-toggle-exclusive'
|
'button-toggle-exclusive'
|
||||||
],
|
],
|
||||||
'card' : [
|
'card' : [
|
||||||
'card-overview',
|
'card-overview',
|
||||||
'card-fancy'
|
'card-fancy'
|
||||||
],
|
],
|
||||||
// 'cdk-table' : [
|
|
||||||
// 'cdk-table-basic',
|
|
||||||
// 'cdk-table-basic-flex',
|
|
||||||
// 'cdk-table-flat'
|
|
||||||
// ],
|
|
||||||
// 'cdk-tree' : [
|
|
||||||
// 'cdk-tree-nested'
|
|
||||||
// ],
|
|
||||||
'checkbox' : [
|
'checkbox' : [
|
||||||
'checkbox-overview',
|
'checkbox-overview',
|
||||||
'checkbox-configurable'
|
'checkbox-configurable'
|
||||||
|
@ -307,6 +304,9 @@ export const COMPONENT_MAP = {
|
||||||
'progress-spinner-overview',
|
'progress-spinner-overview',
|
||||||
'progress-spinner-configurable'
|
'progress-spinner-configurable'
|
||||||
],
|
],
|
||||||
|
'ripples' : [
|
||||||
|
'ripple-overview'
|
||||||
|
],
|
||||||
'radio-button' : [
|
'radio-button' : [
|
||||||
'radio-overview',
|
'radio-overview',
|
||||||
'radio-ng-model'
|
'radio-ng-model'
|
||||||
|
@ -357,7 +357,10 @@ export const COMPONENT_MAP = {
|
||||||
'stepper' : [
|
'stepper' : [
|
||||||
'stepper-overview',
|
'stepper-overview',
|
||||||
'stepper-editable',
|
'stepper-editable',
|
||||||
|
'stepper-errors',
|
||||||
|
'stepper-label-position-bottom',
|
||||||
'stepper-optional',
|
'stepper-optional',
|
||||||
|
'stepper-states',
|
||||||
'stepper-vertical'
|
'stepper-vertical'
|
||||||
],
|
],
|
||||||
'table' : [
|
'table' : [
|
||||||
|
@ -373,16 +376,19 @@ export const COMPONENT_MAP = {
|
||||||
'table-pagination',
|
'table-pagination',
|
||||||
'table-row-context',
|
'table-row-context',
|
||||||
'table-selection',
|
'table-selection',
|
||||||
|
'table-simple-column',
|
||||||
'table-sorting',
|
'table-sorting',
|
||||||
'table-sticky-columns',
|
'table-sticky-columns',
|
||||||
'table-sticky-complex',
|
'table-sticky-complex',
|
||||||
'table-sticky-complex-flex',
|
'table-sticky-complex-flex',
|
||||||
'table-sticky-header',
|
'table-sticky-header',
|
||||||
'table-sticky-footer',
|
'table-sticky-footer',
|
||||||
|
'table-wrapped'
|
||||||
],
|
],
|
||||||
'tabs' : [
|
'tabs' : [
|
||||||
'tab-group-basic',
|
'tab-group-basic',
|
||||||
'tab-group-async',
|
'tab-group-async',
|
||||||
|
'tab-group-align',
|
||||||
'tab-group-custom-label',
|
'tab-group-custom-label',
|
||||||
'tab-group-dynamic',
|
'tab-group-dynamic',
|
||||||
'tab-group-dynamic-height',
|
'tab-group-dynamic-height',
|
||||||
|
@ -460,6 +466,10 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Basic buttons',
|
title : 'Basic buttons',
|
||||||
component: ButtonOverviewExample
|
component: ButtonOverviewExample
|
||||||
},
|
},
|
||||||
|
'button-toggle-appearance' : {
|
||||||
|
title : 'Button toggle appearance',
|
||||||
|
component: ButtonToggleAppearanceExample
|
||||||
|
},
|
||||||
'button-toggle-exclusive' : {
|
'button-toggle-exclusive' : {
|
||||||
title : 'Exclusive selection',
|
title : 'Exclusive selection',
|
||||||
component: ButtonToggleExclusiveExample
|
component: ButtonToggleExclusiveExample
|
||||||
|
@ -480,22 +490,6 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Basic cards',
|
title : 'Basic cards',
|
||||||
component: CardOverviewExample
|
component: CardOverviewExample
|
||||||
},
|
},
|
||||||
'cdk-table-basic' : {
|
|
||||||
title : 'Basic CDK data-table',
|
|
||||||
component: CdkTableBasicExample
|
|
||||||
},
|
|
||||||
'cdk-table-basic-flex' : {
|
|
||||||
title : 'Basic use of `<cdk-table>` (uses display flex)',
|
|
||||||
component: CdkTableBasicFlexExample
|
|
||||||
},
|
|
||||||
'cdk-tree-flat' : {
|
|
||||||
title : 'Tree with flat nodes',
|
|
||||||
component: CdkTreeFlatExample
|
|
||||||
},
|
|
||||||
'cdk-tree-nested' : {
|
|
||||||
title : 'Tree with nested nodes',
|
|
||||||
component: CdkTreeNestedExample
|
|
||||||
},
|
|
||||||
'checkbox-configurable' : {
|
'checkbox-configurable' : {
|
||||||
title : 'Configurable checkbox',
|
title : 'Configurable checkbox',
|
||||||
component: CheckboxConfigurableExample
|
component: CheckboxConfigurableExample
|
||||||
|
@ -787,6 +781,10 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Basic radios',
|
title : 'Basic radios',
|
||||||
component: RadioOverviewExample
|
component: RadioOverviewExample
|
||||||
},
|
},
|
||||||
|
'ripple-overview' : {
|
||||||
|
title : 'MatRipple basic usage',
|
||||||
|
component: RippleOverviewExample
|
||||||
|
},
|
||||||
'select-custom-trigger' : {
|
'select-custom-trigger' : {
|
||||||
title : 'Select with custom trigger text',
|
title : 'Select with custom trigger text',
|
||||||
component: SelectCustomTriggerExample
|
component: SelectCustomTriggerExample
|
||||||
|
@ -921,6 +919,14 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Stepper with editable steps',
|
title : 'Stepper with editable steps',
|
||||||
component: StepperEditableExample
|
component: StepperEditableExample
|
||||||
},
|
},
|
||||||
|
'stepper-errors' : {
|
||||||
|
title : 'Stepper that displays errors in the steps\n',
|
||||||
|
component: StepperErrorsExample
|
||||||
|
},
|
||||||
|
'stepper-label-position-bottom' : {
|
||||||
|
title : 'Stepper label position bottom',
|
||||||
|
component: StepperLabelPositionBottomExample
|
||||||
|
},
|
||||||
'stepper-optional' : {
|
'stepper-optional' : {
|
||||||
title : 'Stepper with optional steps',
|
title : 'Stepper with optional steps',
|
||||||
component: StepperOptionalExample
|
component: StepperOptionalExample
|
||||||
|
@ -929,6 +935,10 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Stepper overview',
|
title : 'Stepper overview',
|
||||||
component: StepperOverviewExample
|
component: StepperOverviewExample
|
||||||
},
|
},
|
||||||
|
'stepper-states' : {
|
||||||
|
title : 'Stepper with customized states',
|
||||||
|
component: StepperStatesExample
|
||||||
|
},
|
||||||
'stepper-vertical' : {
|
'stepper-vertical' : {
|
||||||
title : 'Stepper vertical',
|
title : 'Stepper vertical',
|
||||||
component: StepperVerticalExample
|
component: StepperVerticalExample
|
||||||
|
@ -981,6 +991,10 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Table with selection',
|
title : 'Table with selection',
|
||||||
component: TableSelectionExample
|
component: TableSelectionExample
|
||||||
},
|
},
|
||||||
|
'table-simple-column' : {
|
||||||
|
title : 'Table with a custom column component for easy column definition reuse',
|
||||||
|
component: TableSimpleColumnExample
|
||||||
|
},
|
||||||
'table-sorting' : {
|
'table-sorting' : {
|
||||||
title : 'Table with sorting',
|
title : 'Table with sorting',
|
||||||
component: TableSortingExample
|
component: TableSortingExample
|
||||||
|
@ -1005,6 +1019,14 @@ export const EXAMPLE_COMPONENTS = {
|
||||||
title : 'Table with sticky header',
|
title : 'Table with sticky header',
|
||||||
component: TableStickyHeaderExample
|
component: TableStickyHeaderExample
|
||||||
},
|
},
|
||||||
|
'table-wrapped' : {
|
||||||
|
title : 'Table example that shows how to wrap a table component for definition and behavior reuse',
|
||||||
|
component: TableWrappedExample
|
||||||
|
},
|
||||||
|
'tab-group-align' : {
|
||||||
|
title : 'Tab group with aligned labels',
|
||||||
|
component: TabGroupAlignExample
|
||||||
|
},
|
||||||
'tab-group-basic' : {
|
'tab-group-basic' : {
|
||||||
title : 'Basic use of the tab group',
|
title : 'Basic use of the tab group',
|
||||||
component: TabGroupBasicExample
|
component: TabGroupBasicExample
|
||||||
|
@ -1135,13 +1157,10 @@ export const EXAMPLE_LIST = [
|
||||||
ButtonOverviewExample,
|
ButtonOverviewExample,
|
||||||
ButtonToggleExclusiveExample,
|
ButtonToggleExclusiveExample,
|
||||||
ButtonToggleOverviewExample,
|
ButtonToggleOverviewExample,
|
||||||
|
ButtonToggleAppearanceExample,
|
||||||
ButtonTypesExample,
|
ButtonTypesExample,
|
||||||
CardFancyExample,
|
CardFancyExample,
|
||||||
CardOverviewExample,
|
CardOverviewExample,
|
||||||
CdkTableBasicExample,
|
|
||||||
CdkTableBasicFlexExample,
|
|
||||||
CdkTreeFlatExample,
|
|
||||||
CdkTreeNestedExample,
|
|
||||||
CheckboxConfigurableExample,
|
CheckboxConfigurableExample,
|
||||||
CheckboxOverviewExample,
|
CheckboxOverviewExample,
|
||||||
ChipsAutocompleteExample,
|
ChipsAutocompleteExample,
|
||||||
|
@ -1212,6 +1231,7 @@ export const EXAMPLE_LIST = [
|
||||||
ProgressSpinnerOverviewExample,
|
ProgressSpinnerOverviewExample,
|
||||||
RadioNgModelExample,
|
RadioNgModelExample,
|
||||||
RadioOverviewExample,
|
RadioOverviewExample,
|
||||||
|
RippleOverviewExample,
|
||||||
SelectCustomTriggerExample,
|
SelectCustomTriggerExample,
|
||||||
SelectDisabledExample,
|
SelectDisabledExample,
|
||||||
SelectErrorStateMatcherExample,
|
SelectErrorStateMatcherExample,
|
||||||
|
@ -1245,8 +1265,11 @@ export const EXAMPLE_LIST = [
|
||||||
SnackBarPositionExample,
|
SnackBarPositionExample,
|
||||||
SortOverviewExample,
|
SortOverviewExample,
|
||||||
StepperEditableExample,
|
StepperEditableExample,
|
||||||
|
StepperErrorsExample,
|
||||||
|
StepperLabelPositionBottomExample,
|
||||||
StepperOptionalExample,
|
StepperOptionalExample,
|
||||||
StepperOverviewExample,
|
StepperOverviewExample,
|
||||||
|
StepperStatesExample,
|
||||||
StepperVerticalExample,
|
StepperVerticalExample,
|
||||||
TableBasicExample,
|
TableBasicExample,
|
||||||
TableBasicFlexExample,
|
TableBasicFlexExample,
|
||||||
|
@ -1260,12 +1283,15 @@ export const EXAMPLE_LIST = [
|
||||||
TablePaginationExample,
|
TablePaginationExample,
|
||||||
TableRowContextExample,
|
TableRowContextExample,
|
||||||
TableSelectionExample,
|
TableSelectionExample,
|
||||||
|
TableSimpleColumnExample, SimpleColumn,
|
||||||
TableSortingExample,
|
TableSortingExample,
|
||||||
TableStickyColumnsExample,
|
TableStickyColumnsExample,
|
||||||
TableStickyComplexExample,
|
TableStickyComplexExample,
|
||||||
TableStickyComplexFlexExample,
|
TableStickyComplexFlexExample,
|
||||||
TableStickyFooterExample,
|
TableStickyFooterExample,
|
||||||
TableStickyHeaderExample,
|
TableStickyHeaderExample,
|
||||||
|
TableWrappedExample, WrapperTable,
|
||||||
|
TabGroupAlignExample,
|
||||||
TabGroupBasicExample,
|
TabGroupBasicExample,
|
||||||
TabGroupAsyncExample,
|
TabGroupAsyncExample,
|
||||||
TabGroupCustomLabelExample,
|
TabGroupCustomLabelExample,
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
import { NgModule } from '@angular/core';
|
import { NgModule } from '@angular/core';
|
||||||
|
|
||||||
import { CdkTableModule } from '@angular/cdk/table';
|
|
||||||
import { CdkTreeModule } from '@angular/cdk/tree';
|
|
||||||
import {
|
import {
|
||||||
MatAutocompleteModule, MatBadgeModule, MatBottomSheetModule, MatButtonModule,
|
MatAutocompleteModule, MatBadgeModule, MatBottomSheetModule, MatButtonModule,
|
||||||
MatButtonToggleModule, MatCardModule, MatCheckboxModule, MatChipsModule, MatDatepickerModule,
|
MatButtonToggleModule, MatCardModule, MatCheckboxModule, MatChipsModule, MatDatepickerModule,
|
||||||
|
@ -15,8 +13,6 @@ import { MatMomentDateModule } from '@angular/material-moment-adapter';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [
|
imports: [
|
||||||
CdkTableModule,
|
|
||||||
CdkTreeModule,
|
|
||||||
MatAutocompleteModule,
|
MatAutocompleteModule,
|
||||||
MatBadgeModule,
|
MatBadgeModule,
|
||||||
MatBottomSheetModule,
|
MatBottomSheetModule,
|
||||||
|
@ -55,8 +51,6 @@ import { MatMomentDateModule } from '@angular/material-moment-adapter';
|
||||||
MatTreeModule
|
MatTreeModule
|
||||||
],
|
],
|
||||||
exports: [
|
exports: [
|
||||||
CdkTableModule,
|
|
||||||
CdkTreeModule,
|
|
||||||
MatAutocompleteModule,
|
MatAutocompleteModule,
|
||||||
MatBadgeModule,
|
MatBadgeModule,
|
||||||
MatBottomSheetModule,
|
MatBottomSheetModule,
|
||||||
|
|
|
@ -63,9 +63,9 @@
|
||||||
<div class="new">
|
<div class="new">
|
||||||
<span class="title">New</span>
|
<span class="title">New</span>
|
||||||
<ul>
|
<ul>
|
||||||
<li>Updated Angular to 7.0.0-rc.0</li>
|
<li>Updated Angular to 7.0.0</li>
|
||||||
<li>Updated Angular Material to 7.0.0-rc.1</li>
|
<li>Updated Angular Material to 7.0.0</li>
|
||||||
<li>Updated Typescript to 3.1.1</li>
|
<li>Updated Typescript to 3.1.3</li>
|
||||||
<li>Updated various other libraries</li>
|
<li>Updated various other libraries</li>
|
||||||
<li>Modernization:
|
<li>Modernization:
|
||||||
<ul>
|
<ul>
|
||||||
|
|
|
@ -345,7 +345,7 @@ export const navigation: FuseNavigation[] = [
|
||||||
title: 'Modern',
|
title: 'Modern',
|
||||||
type : 'item',
|
type : 'item',
|
||||||
url : '/pages/search/modern'
|
url : '/pages/search/modern'
|
||||||
},
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -793,6 +793,12 @@ export const navigation: FuseNavigation[] = [
|
||||||
title: 'Progress bar',
|
title: 'Progress bar',
|
||||||
type : 'item',
|
type : 'item',
|
||||||
url : '/angular-material-elements/progress-bar'
|
url : '/angular-material-elements/progress-bar'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id : 'ripples',
|
||||||
|
title: 'Ripples',
|
||||||
|
type : 'item',
|
||||||
|
url : '/angular-material-elements/ripples'
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
|
@ -18,6 +18,7 @@ export const _filter = (opt: string[], value: string): string[] => {
|
||||||
* @title Option groups autocomplete
|
* @title Option groups autocomplete
|
||||||
*/
|
*/
|
||||||
@Component({
|
@Component({
|
||||||
|
selector: 'autocomplete-optgroup-example',
|
||||||
templateUrl: './autocomplete-optgroup-example.html',
|
templateUrl: './autocomplete-optgroup-example.html',
|
||||||
styleUrls: ['./autocomplete-optgroup-example.css'],
|
styleUrls: ['./autocomplete-optgroup-example.css'],
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
<p>You have receive a file called "cat-picture.jpeg".</p>
|
<p>You have received a file called "cat-picture.jpeg".</p>
|
||||||
|
|
||||||
<button mat-raised-button (click)="openBottomSheet()">Open file</button>
|
<button mat-raised-button (click)="openBottomSheet()">Open file</button>
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
/** No CSS for this example */
|
|
@ -0,0 +1,17 @@
|
||||||
|
<p>
|
||||||
|
Default appearance:
|
||||||
|
<mat-button-toggle-group name="fontStyle" aria-label="Font Style">
|
||||||
|
<mat-button-toggle value="bold">Bold</mat-button-toggle>
|
||||||
|
<mat-button-toggle value="italic">Italic</mat-button-toggle>
|
||||||
|
<mat-button-toggle value="underline">Underline</mat-button-toggle>
|
||||||
|
</mat-button-toggle-group>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Legacy appearance:
|
||||||
|
<mat-button-toggle-group appearance="legacy" name="fontStyle" aria-label="Font Style">
|
||||||
|
<mat-button-toggle value="bold">Bold</mat-button-toggle>
|
||||||
|
<mat-button-toggle value="italic">Italic</mat-button-toggle>
|
||||||
|
<mat-button-toggle value="underline">Underline</mat-button-toggle>
|
||||||
|
</mat-button-toggle-group>
|
||||||
|
</p>
|
|
@ -0,0 +1,11 @@
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Button toggle appearance
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'button-toggle-appearance-example',
|
||||||
|
templateUrl: 'button-toggle-appearance-example.html',
|
||||||
|
styleUrls: ['button-toggle-appearance-example.css'],
|
||||||
|
})
|
||||||
|
export class ButtonToggleAppearanceExample {}
|
|
@ -1,4 +1,4 @@
|
||||||
.button-row button,
|
.example-button-row button,
|
||||||
.button-row a {
|
.example-button-row a {
|
||||||
margin-right: 8px;
|
margin-right: 8px;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
<h3>Basic Buttons</h3>
|
<h3>Basic Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-button>Basic</button>
|
<button mat-button>Basic</button>
|
||||||
<button mat-button color="primary">Primary</button>
|
<button mat-button color="primary">Primary</button>
|
||||||
<button mat-button color="accent">Accent</button>
|
<button mat-button color="accent">Accent</button>
|
||||||
|
@ -9,7 +9,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h3>Raised Buttons</h3>
|
<h3>Raised Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-raised-button>Basic</button>
|
<button mat-raised-button>Basic</button>
|
||||||
<button mat-raised-button color="primary">Primary</button>
|
<button mat-raised-button color="primary">Primary</button>
|
||||||
<button mat-raised-button color="accent">Accent</button>
|
<button mat-raised-button color="accent">Accent</button>
|
||||||
|
@ -19,7 +19,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h3>Stroked Buttons</h3>
|
<h3>Stroked Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-stroked-button>Basic</button>
|
<button mat-stroked-button>Basic</button>
|
||||||
<button mat-stroked-button color="primary">Primary</button>
|
<button mat-stroked-button color="primary">Primary</button>
|
||||||
<button mat-stroked-button color="accent">Accent</button>
|
<button mat-stroked-button color="accent">Accent</button>
|
||||||
|
@ -29,7 +29,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h3>Flat Buttons</h3>
|
<h3>Flat Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-flat-button>Basic</button>
|
<button mat-flat-button>Basic</button>
|
||||||
<button mat-flat-button color="primary">Primary</button>
|
<button mat-flat-button color="primary">Primary</button>
|
||||||
<button mat-flat-button color="accent">Accent</button>
|
<button mat-flat-button color="accent">Accent</button>
|
||||||
|
@ -39,7 +39,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h3>Icon Buttons</h3>
|
<h3>Icon Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-icon-button>
|
<button mat-icon-button>
|
||||||
<mat-icon aria-label="Example icon-button with a heart icon">favorite</mat-icon>
|
<mat-icon aria-label="Example icon-button with a heart icon">favorite</mat-icon>
|
||||||
</button>
|
</button>
|
||||||
|
@ -58,7 +58,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h3>Fab Buttons</h3>
|
<h3>Fab Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-fab>Basic</button>
|
<button mat-fab>Basic</button>
|
||||||
<button mat-fab color="primary">Primary</button>
|
<button mat-fab color="primary">Primary</button>
|
||||||
<button mat-fab color="accent">Accent</button>
|
<button mat-fab color="accent">Accent</button>
|
||||||
|
@ -71,7 +71,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h3>Mini Fab Buttons</h3>
|
<h3>Mini Fab Buttons</h3>
|
||||||
<div class="button-row">
|
<div class="example-button-row">
|
||||||
<button mat-mini-fab>Basic</button>
|
<button mat-mini-fab>Basic</button>
|
||||||
<button mat-mini-fab color="primary">Primary</button>
|
<button mat-mini-fab color="primary">Primary</button>
|
||||||
<button mat-mini-fab color="accent">Accent</button>
|
<button mat-mini-fab color="accent">Accent</button>
|
||||||
|
|
|
@ -1,10 +0,0 @@
|
||||||
/**
|
|
||||||
* Add basic flex styling so that the cells evenly space themselves in the row.
|
|
||||||
*/
|
|
||||||
cdk-row, cdk-header-row, cdk-footer-row {
|
|
||||||
display: flex;
|
|
||||||
}
|
|
||||||
|
|
||||||
cdk-cell, cdk-header-cell, cdk-footer-cell {
|
|
||||||
flex: 1;
|
|
||||||
}
|
|
|
@ -1,28 +0,0 @@
|
||||||
<cdk-table [dataSource]="dataSource">
|
|
||||||
<!-- Position Column -->
|
|
||||||
<ng-container cdkColumnDef="position">
|
|
||||||
<cdk-header-cell *cdkHeaderCellDef> No. </cdk-header-cell>
|
|
||||||
<cdk-cell *cdkCellDef="let element"> {{element.position}} </cdk-cell>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Name Column -->
|
|
||||||
<ng-container cdkColumnDef="name">
|
|
||||||
<cdk-header-cell *cdkHeaderCellDef> Name </cdk-header-cell>
|
|
||||||
<cdk-cell *cdkCellDef="let element"> {{element.name}} </cdk-cell>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Weight Column -->
|
|
||||||
<ng-container cdkColumnDef="weight">
|
|
||||||
<cdk-header-cell *cdkHeaderCellDef> Weight </cdk-header-cell>
|
|
||||||
<cdk-cell *cdkCellDef="let element"> {{element.weight}} </cdk-cell>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Symbol Column -->
|
|
||||||
<ng-container cdkColumnDef="symbol">
|
|
||||||
<cdk-header-cell *cdkHeaderCellDef> Symbol </cdk-header-cell>
|
|
||||||
<cdk-cell *cdkCellDef="let element"> {{element.symbol}} </cdk-cell>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<cdk-header-row *cdkHeaderRowDef="displayedColumns"></cdk-header-row>
|
|
||||||
<cdk-row *cdkRowDef="let row; columns: displayedColumns;"></cdk-row>
|
|
||||||
</cdk-table>
|
|
|
@ -1,55 +0,0 @@
|
||||||
import {DataSource} from '@angular/cdk/collections';
|
|
||||||
import {Component} from '@angular/core';
|
|
||||||
import {BehaviorSubject, Observable} from 'rxjs';
|
|
||||||
|
|
||||||
export interface PeriodicElement {
|
|
||||||
name: string;
|
|
||||||
position: number;
|
|
||||||
symbol: string;
|
|
||||||
weight: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
const ELEMENT_DATA: PeriodicElement[] = [
|
|
||||||
{position: 1, name: 'Hydrogen', weight: 1.0079, symbol: 'H'},
|
|
||||||
{position: 2, name: 'Helium', weight: 4.0026, symbol: 'He'},
|
|
||||||
{position: 3, name: 'Lithium', weight: 6.941, symbol: 'Li'},
|
|
||||||
{position: 4, name: 'Beryllium', weight: 9.0122, symbol: 'Be'},
|
|
||||||
{position: 5, name: 'Boron', weight: 10.811, symbol: 'B'},
|
|
||||||
{position: 6, name: 'Carbon', weight: 12.0107, symbol: 'C'},
|
|
||||||
{position: 7, name: 'Nitrogen', weight: 14.0067, symbol: 'N'},
|
|
||||||
{position: 8, name: 'Oxygen', weight: 15.9994, symbol: 'O'},
|
|
||||||
{position: 9, name: 'Fluorine', weight: 18.9984, symbol: 'F'},
|
|
||||||
{position: 10, name: 'Neon', weight: 20.1797, symbol: 'Ne'},
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @title Basic use of `<cdk-table>` (uses display flex)
|
|
||||||
*/
|
|
||||||
@Component({
|
|
||||||
selector: 'cdk-table-basic-flex-example',
|
|
||||||
styleUrls: ['cdk-table-basic-flex-example.css'],
|
|
||||||
templateUrl: 'cdk-table-basic-flex-example.html',
|
|
||||||
})
|
|
||||||
export class CdkTableBasicFlexExample {
|
|
||||||
displayedColumns: string[] = ['position', 'name', 'weight', 'symbol'];
|
|
||||||
dataSource = new ExampleDataSource();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Data source to provide what data should be rendered in the table. Note that the data source
|
|
||||||
* can retrieve its data in any way. In this case, the data source is provided a reference
|
|
||||||
* to a common data base, ExampleDatabase. It is not the data source's responsibility to manage
|
|
||||||
* the underlying data. Instead, it only needs to take the data and send the table exactly what
|
|
||||||
* should be rendered.
|
|
||||||
*/
|
|
||||||
export class ExampleDataSource extends DataSource<PeriodicElement> {
|
|
||||||
/** Stream of data that is provided to the table. */
|
|
||||||
data = new BehaviorSubject<PeriodicElement[]>(ELEMENT_DATA);
|
|
||||||
|
|
||||||
/** Connect function called by the table to retrieve one stream containing the data to render. */
|
|
||||||
connect(): Observable<PeriodicElement[]> {
|
|
||||||
return this.data;
|
|
||||||
}
|
|
||||||
|
|
||||||
disconnect() {}
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
table {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
th {
|
|
||||||
text-align: left;
|
|
||||||
}
|
|
|
@ -1,28 +0,0 @@
|
||||||
<table cdk-table [dataSource]="dataSource">
|
|
||||||
<!-- Position Column -->
|
|
||||||
<ng-container cdkColumnDef="position">
|
|
||||||
<th cdk-header-cell *cdkHeaderCellDef> No. </th>
|
|
||||||
<td cdk-cell *cdkCellDef="let element"> {{element.position}} </td>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Name Column -->
|
|
||||||
<ng-container cdkColumnDef="name">
|
|
||||||
<th cdk-header-cell *cdkHeaderCellDef> Name </th>
|
|
||||||
<td cdk-cell *cdkCellDef="let element"> {{element.name}} </td>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Weight Column -->
|
|
||||||
<ng-container cdkColumnDef="weight">
|
|
||||||
<th cdk-header-cell *cdkHeaderCellDef> Weight </th>
|
|
||||||
<td cdk-cell *cdkCellDef="let element"> {{element.weight}} </td>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<!-- Symbol Column -->
|
|
||||||
<ng-container cdkColumnDef="symbol">
|
|
||||||
<th cdk-header-cell *cdkHeaderCellDef> Symbol </th>
|
|
||||||
<td cdk-cell *cdkCellDef="let element"> {{element.symbol}} </td>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<tr cdk-header-row *cdkHeaderRowDef="displayedColumns"></tr>
|
|
||||||
<tr cdk-row *cdkRowDef="let row; columns: displayedColumns;"></tr>
|
|
||||||
</table>
|
|
|
@ -1,55 +0,0 @@
|
||||||
import {DataSource} from '@angular/cdk/collections';
|
|
||||||
import {Component} from '@angular/core';
|
|
||||||
import {BehaviorSubject, Observable} from 'rxjs';
|
|
||||||
|
|
||||||
export interface PeriodicElement {
|
|
||||||
name: string;
|
|
||||||
position: number;
|
|
||||||
weight: number;
|
|
||||||
symbol: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const ELEMENT_DATA: PeriodicElement[] = [
|
|
||||||
{position: 1, name: 'Hydrogen', weight: 1.0079, symbol: 'H'},
|
|
||||||
{position: 2, name: 'Helium', weight: 4.0026, symbol: 'He'},
|
|
||||||
{position: 3, name: 'Lithium', weight: 6.941, symbol: 'Li'},
|
|
||||||
{position: 4, name: 'Beryllium', weight: 9.0122, symbol: 'Be'},
|
|
||||||
{position: 5, name: 'Boron', weight: 10.811, symbol: 'B'},
|
|
||||||
{position: 6, name: 'Carbon', weight: 12.0107, symbol: 'C'},
|
|
||||||
{position: 7, name: 'Nitrogen', weight: 14.0067, symbol: 'N'},
|
|
||||||
{position: 8, name: 'Oxygen', weight: 15.9994, symbol: 'O'},
|
|
||||||
{position: 9, name: 'Fluorine', weight: 18.9984, symbol: 'F'},
|
|
||||||
{position: 10, name: 'Neon', weight: 20.1797, symbol: 'Ne'},
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @title Basic CDK data-table
|
|
||||||
*/
|
|
||||||
@Component({
|
|
||||||
selector: 'cdk-table-basic-example',
|
|
||||||
styleUrls: ['cdk-table-basic-example.css'],
|
|
||||||
templateUrl: 'cdk-table-basic-example.html',
|
|
||||||
})
|
|
||||||
export class CdkTableBasicExample {
|
|
||||||
displayedColumns: string[] = ['position', 'name', 'weight', 'symbol'];
|
|
||||||
dataSource = new ExampleDataSource();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Data source to provide what data should be rendered in the table. Note that the data source
|
|
||||||
* can retrieve its data in any way. In this case, the data source is provided a reference
|
|
||||||
* to a common data base, ExampleDatabase. It is not the data source's responsibility to manage
|
|
||||||
* the underlying data. Instead, it only needs to take the data and send the table exactly what
|
|
||||||
* should be rendered.
|
|
||||||
*/
|
|
||||||
export class ExampleDataSource extends DataSource<PeriodicElement> {
|
|
||||||
/** Stream of data that is provided to the table. */
|
|
||||||
data = new BehaviorSubject<PeriodicElement[]>(ELEMENT_DATA);
|
|
||||||
|
|
||||||
/** Connect function called by the table to retrieve one stream containing the data to render. */
|
|
||||||
connect(): Observable<PeriodicElement[]> {
|
|
||||||
return this.data;
|
|
||||||
}
|
|
||||||
|
|
||||||
disconnect() {}
|
|
||||||
}
|
|
|
@ -1,4 +0,0 @@
|
||||||
.example-tree-node {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
|
@ -1,14 +0,0 @@
|
||||||
<cdk-tree [dataSource]="dataSource" [treeControl]="treeControl">
|
|
||||||
<cdk-tree-node *cdkTreeNodeDef="let node" cdkTreeNodePadding class="example-tree-node">
|
|
||||||
<button mat-icon-button disabled></button>
|
|
||||||
{{node.filename}}: {{node.type}}
|
|
||||||
</cdk-tree-node>
|
|
||||||
<cdk-tree-node *cdkTreeNodeDef="let node; when: hasChild" cdkTreeNodePadding class="example-tree-node">
|
|
||||||
<button mat-icon-button [attr.aria-label]="'toggle ' + node.filename" cdkTreeNodeToggle>
|
|
||||||
<mat-icon class="mat-icon-rtl-mirror">
|
|
||||||
{{treeControl.isExpanded(node) ? 'expand_more' : 'chevron_right'}}
|
|
||||||
</mat-icon>
|
|
||||||
</button>
|
|
||||||
{{node.filename}}: {{node.type}}
|
|
||||||
</cdk-tree-node>
|
|
||||||
</cdk-tree>
|
|
|
@ -1,149 +0,0 @@
|
||||||
import {FlatTreeControl} from '@angular/cdk/tree';
|
|
||||||
import {Component, Injectable} from '@angular/core';
|
|
||||||
import {MatTreeFlatDataSource, MatTreeFlattener} from '@angular/material/tree';
|
|
||||||
import {BehaviorSubject, Observable, of as observableOf} from 'rxjs';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* File node data with nested structure.
|
|
||||||
* Each node has a filename, and a type or a list of children.
|
|
||||||
*/
|
|
||||||
export class FileNode {
|
|
||||||
children: FileNode[];
|
|
||||||
filename: string;
|
|
||||||
type: any;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Flat node with expandable and level information */
|
|
||||||
export class FileFlatNode {
|
|
||||||
constructor(
|
|
||||||
public expandable: boolean, public filename: string, public level: number, public type: any) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The file structure tree data in string. The data could be parsed into a Json object
|
|
||||||
*/
|
|
||||||
const TREE_DATA = JSON.stringify({
|
|
||||||
Applications: {
|
|
||||||
Calendar: 'app',
|
|
||||||
Chrome: 'app',
|
|
||||||
Webstorm: 'app'
|
|
||||||
},
|
|
||||||
Documents: {
|
|
||||||
angular: {
|
|
||||||
src: {
|
|
||||||
compiler: 'ts',
|
|
||||||
core: 'ts'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
material2: {
|
|
||||||
src: {
|
|
||||||
button: 'ts',
|
|
||||||
checkbox: 'ts',
|
|
||||||
input: 'ts'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Downloads: {
|
|
||||||
October: 'pdf',
|
|
||||||
November: 'pdf',
|
|
||||||
Tutorial: 'html'
|
|
||||||
},
|
|
||||||
Pictures: {
|
|
||||||
'Photo Booth Library': {
|
|
||||||
Contents: 'dir',
|
|
||||||
Pictures: 'dir'
|
|
||||||
},
|
|
||||||
Sun: 'png',
|
|
||||||
Woods: 'jpg'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
/**
|
|
||||||
* File database, it can build a tree structured Json object from string.
|
|
||||||
* Each node in Json object represents a file or a directory. For a file, it has filename and type.
|
|
||||||
* For a directory, it has filename and children (a list of files or directories).
|
|
||||||
* The input will be a json object string, and the output is a list of `FileNode` with nested
|
|
||||||
* structure.
|
|
||||||
*/
|
|
||||||
@Injectable()
|
|
||||||
export class FileDatabase {
|
|
||||||
dataChange = new BehaviorSubject<FileNode[]>([]);
|
|
||||||
|
|
||||||
get data(): FileNode[] { return this.dataChange.value; }
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
this.initialize();
|
|
||||||
}
|
|
||||||
|
|
||||||
initialize() {
|
|
||||||
// Parse the string to json object.
|
|
||||||
const dataObject = JSON.parse(TREE_DATA);
|
|
||||||
|
|
||||||
// Build the tree nodes from Json object. The result is a list of `FileNode` with nested
|
|
||||||
// file node as children.
|
|
||||||
const data = this.buildFileTree(dataObject, 0);
|
|
||||||
|
|
||||||
// Notify the change.
|
|
||||||
this.dataChange.next(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
|
||||||
* The return value is the list of `FileNode`.
|
|
||||||
*/
|
|
||||||
buildFileTree(obj: object, level: number): FileNode[] {
|
|
||||||
return Object.keys(obj).reduce<FileNode[]>((accumulator, key) => {
|
|
||||||
const value = obj[key];
|
|
||||||
const node = new FileNode();
|
|
||||||
node.filename = key;
|
|
||||||
|
|
||||||
if (value != null) {
|
|
||||||
if (typeof value === 'object') {
|
|
||||||
node.children = this.buildFileTree(value, level + 1);
|
|
||||||
} else {
|
|
||||||
node.type = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return accumulator.concat(node);
|
|
||||||
}, []);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @title Tree with flat nodes
|
|
||||||
*/
|
|
||||||
@Component({
|
|
||||||
selector: 'cdk-tree-flat-example',
|
|
||||||
templateUrl: 'cdk-tree-flat-example.html',
|
|
||||||
styleUrls: ['cdk-tree-flat-example.css'],
|
|
||||||
providers: [FileDatabase]
|
|
||||||
})
|
|
||||||
export class CdkTreeFlatExample {
|
|
||||||
treeControl: FlatTreeControl<FileFlatNode>;
|
|
||||||
treeFlattener: MatTreeFlattener<FileNode, FileFlatNode>;
|
|
||||||
dataSource: MatTreeFlatDataSource<FileNode, FileFlatNode>;
|
|
||||||
|
|
||||||
constructor(database: FileDatabase) {
|
|
||||||
this.treeFlattener = new MatTreeFlattener(this.transformer, this._getLevel,
|
|
||||||
this._isExpandable, this._getChildren);
|
|
||||||
this.treeControl = new FlatTreeControl<FileFlatNode>(this._getLevel, this._isExpandable);
|
|
||||||
this.dataSource = new MatTreeFlatDataSource(this.treeControl, this.treeFlattener);
|
|
||||||
|
|
||||||
database.dataChange.subscribe(data => {
|
|
||||||
this.dataSource.data = data;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
hasChild = (_: number, _nodeData: FileFlatNode) => _nodeData.expandable;
|
|
||||||
|
|
||||||
transformer = (node: FileNode, level: number) => {
|
|
||||||
return new FileFlatNode(!!node.children, node.filename, level, node.type);
|
|
||||||
}
|
|
||||||
|
|
||||||
private _getLevel = (node: FileFlatNode) => node.level;
|
|
||||||
|
|
||||||
private _isExpandable = (node: FileFlatNode) => node.expandable;
|
|
||||||
|
|
||||||
private _getChildren = (node: FileNode): Observable<FileNode[]> => observableOf(node.children);
|
|
||||||
}
|
|
|
@ -1,15 +0,0 @@
|
||||||
.example-tree-invisible {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.example-tree ul,
|
|
||||||
.example-tree li {
|
|
||||||
margin-top: 0;
|
|
||||||
margin-bottom: 0;
|
|
||||||
list-style-type: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.example-tree-node {
|
|
||||||
display: block;
|
|
||||||
padding-left: 40px;
|
|
||||||
}
|
|
|
@ -1,17 +0,0 @@
|
||||||
<cdk-tree [dataSource]="nestedDataSource" [treeControl]="nestedTreeControl">
|
|
||||||
<cdk-nested-tree-node *cdkTreeNodeDef="let node" class="example-tree-node">
|
|
||||||
<button mat-icon-button disabled></button>
|
|
||||||
{{node.filename}}: {{node.type}}
|
|
||||||
</cdk-nested-tree-node>
|
|
||||||
<cdk-nested-tree-node *cdkTreeNodeDef="let node; when: hasNestedChild" class="example-tree-node">
|
|
||||||
<button mat-icon-button [attr.aria-label]="'toggle ' + node.filename" cdkTreeNodeToggle>
|
|
||||||
<mat-icon class="mat-icon-rtl-mirror">
|
|
||||||
{{nestedTreeControl.isExpanded(node) ? 'expand_more' : 'chevron_right'}}
|
|
||||||
</mat-icon>
|
|
||||||
</button>
|
|
||||||
{{node.filename}}: {{node.type}}
|
|
||||||
<div [class.example-tree-invisible]="!nestedTreeControl.isExpanded(node)">
|
|
||||||
<ng-container cdkTreeNodeOutlet></ng-container>
|
|
||||||
</div>
|
|
||||||
</cdk-nested-tree-node>
|
|
||||||
</cdk-tree>
|
|
|
@ -1,129 +0,0 @@
|
||||||
import {NestedTreeControl} from '@angular/cdk/tree';
|
|
||||||
import {Component, Injectable} from '@angular/core';
|
|
||||||
import {MatTreeNestedDataSource} from '@angular/material/tree';
|
|
||||||
import {BehaviorSubject, of as observableOf} from 'rxjs';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Json node data with nested structure. Each node has a filename and a value or a list of children
|
|
||||||
*/
|
|
||||||
export class FileNode {
|
|
||||||
children: FileNode[];
|
|
||||||
filename: string;
|
|
||||||
type: any;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The Json tree data in string. The data could be parsed into Json object
|
|
||||||
*/
|
|
||||||
const TREE_DATA = JSON.stringify({
|
|
||||||
Applications: {
|
|
||||||
Calendar: 'app',
|
|
||||||
Chrome: 'app',
|
|
||||||
Webstorm: 'app'
|
|
||||||
},
|
|
||||||
Documents: {
|
|
||||||
angular: {
|
|
||||||
src: {
|
|
||||||
compiler: 'ts',
|
|
||||||
core: 'ts'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
material2: {
|
|
||||||
src: {
|
|
||||||
button: 'ts',
|
|
||||||
checkbox: 'ts',
|
|
||||||
input: 'ts'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Downloads: {
|
|
||||||
October: 'pdf',
|
|
||||||
November: 'pdf',
|
|
||||||
Tutorial: 'html'
|
|
||||||
},
|
|
||||||
Pictures: {
|
|
||||||
'Photo Booth Library': {
|
|
||||||
Contents: 'dir',
|
|
||||||
Pictures: 'dir'
|
|
||||||
},
|
|
||||||
Sun: 'png',
|
|
||||||
Woods: 'jpg'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
/**
|
|
||||||
* File database, it can build a tree structured Json object from string.
|
|
||||||
* Each node in Json object represents a file or a directory. For a file, it has filename and type.
|
|
||||||
* For a directory, it has filename and children (a list of files or directories).
|
|
||||||
* The input will be a json object string, and the output is a list of `FileNode` with nested
|
|
||||||
* structure.
|
|
||||||
*/
|
|
||||||
@Injectable()
|
|
||||||
export class FileDatabase {
|
|
||||||
dataChange = new BehaviorSubject<FileNode[]>([]);
|
|
||||||
|
|
||||||
get data(): FileNode[] { return this.dataChange.value; }
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
this.initialize();
|
|
||||||
}
|
|
||||||
|
|
||||||
initialize() {
|
|
||||||
// Parse the string to json object.
|
|
||||||
const dataObject = JSON.parse(TREE_DATA);
|
|
||||||
|
|
||||||
// Build the tree nodes from Json object. The result is a list of `FileNode` with nested
|
|
||||||
// file node as children.
|
|
||||||
const data = this.buildFileTree(dataObject, 0);
|
|
||||||
|
|
||||||
// Notify the change.
|
|
||||||
this.dataChange.next(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
|
||||||
* The return value is the list of `FileNode`.
|
|
||||||
*/
|
|
||||||
buildFileTree(obj: object, level: number): FileNode[] {
|
|
||||||
return Object.keys(obj).reduce<FileNode[]>((accumulator, key) => {
|
|
||||||
const value = obj[key];
|
|
||||||
const node = new FileNode();
|
|
||||||
node.filename = key;
|
|
||||||
|
|
||||||
if (value != null) {
|
|
||||||
if (typeof value === 'object') {
|
|
||||||
node.children = this.buildFileTree(value, level + 1);
|
|
||||||
} else {
|
|
||||||
node.type = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return accumulator.concat(node);
|
|
||||||
}, []);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @title Tree with nested nodes
|
|
||||||
*/
|
|
||||||
@Component({
|
|
||||||
selector: 'cdk-tree-nested-example',
|
|
||||||
templateUrl: 'cdk-tree-nested-example.html',
|
|
||||||
styleUrls: ['cdk-tree-nested-example.css'],
|
|
||||||
providers: [FileDatabase]
|
|
||||||
})
|
|
||||||
export class CdkTreeNestedExample {
|
|
||||||
nestedTreeControl: NestedTreeControl<FileNode>;
|
|
||||||
nestedDataSource: MatTreeNestedDataSource<FileNode>;
|
|
||||||
|
|
||||||
constructor(database: FileDatabase) {
|
|
||||||
this.nestedTreeControl = new NestedTreeControl<FileNode>(this._getChildren);
|
|
||||||
this.nestedDataSource = new MatTreeNestedDataSource();
|
|
||||||
|
|
||||||
database.dataChange.subscribe(data => this.nestedDataSource.data = data);
|
|
||||||
}
|
|
||||||
|
|
||||||
hasNestedChild = (_: number, nodeData: FileNode) => !nodeData.type;
|
|
||||||
|
|
||||||
private _getChildren = (node: FileNode) => observableOf(node.children);
|
|
||||||
}
|
|
|
@ -1,7 +1,7 @@
|
||||||
import {COMMA, ENTER} from '@angular/cdk/keycodes';
|
import {COMMA, ENTER} from '@angular/cdk/keycodes';
|
||||||
import {Component, ElementRef, ViewChild} from '@angular/core';
|
import {Component, ElementRef, ViewChild} from '@angular/core';
|
||||||
import {FormControl} from '@angular/forms';
|
import {FormControl} from '@angular/forms';
|
||||||
import {MatAutocompleteSelectedEvent, MatChipInputEvent} from '@angular/material';
|
import {MatAutocompleteSelectedEvent, MatChipInputEvent, MatAutocomplete} from '@angular/material';
|
||||||
import {Observable} from 'rxjs';
|
import {Observable} from 'rxjs';
|
||||||
import {map, startWith} from 'rxjs/operators';
|
import {map, startWith} from 'rxjs/operators';
|
||||||
|
|
||||||
|
@ -17,14 +17,15 @@ export class ChipsAutocompleteExample {
|
||||||
visible = true;
|
visible = true;
|
||||||
selectable = true;
|
selectable = true;
|
||||||
removable = true;
|
removable = true;
|
||||||
addOnBlur = false;
|
addOnBlur = true;
|
||||||
separatorKeysCodes: number[] = [ENTER, COMMA];
|
separatorKeysCodes: number[] = [ENTER, COMMA];
|
||||||
fruitCtrl = new FormControl();
|
fruitCtrl = new FormControl();
|
||||||
filteredFruits: Observable<string[]>;
|
filteredFruits: Observable<string[]>;
|
||||||
fruits: string[] = ['Lemon'];
|
fruits: string[] = ['Lemon'];
|
||||||
allFruits: string[] = ['Apple', 'Lemon', 'Lime', 'Orange', 'Strawberry'];
|
allFruits: string[] = ['Apple', 'Lemon', 'Lime', 'Orange', 'Strawberry'];
|
||||||
|
|
||||||
@ViewChild('fruitInput') fruitInput: ElementRef;
|
@ViewChild('fruitInput') fruitInput: ElementRef<HTMLInputElement>;
|
||||||
|
@ViewChild('auto') matAutocomplete: MatAutocomplete;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.filteredFruits = this.fruitCtrl.valueChanges.pipe(
|
this.filteredFruits = this.fruitCtrl.valueChanges.pipe(
|
||||||
|
@ -33,6 +34,9 @@ export class ChipsAutocompleteExample {
|
||||||
}
|
}
|
||||||
|
|
||||||
add(event: MatChipInputEvent): void {
|
add(event: MatChipInputEvent): void {
|
||||||
|
// Add fruit only when MatAutocomplete is not open
|
||||||
|
// To make sure this does not conflict with OptionSelected Event
|
||||||
|
if (!this.matAutocomplete.isOpen) {
|
||||||
const input = event.input;
|
const input = event.input;
|
||||||
const value = event.value;
|
const value = event.value;
|
||||||
|
|
||||||
|
@ -48,6 +52,7 @@ export class ChipsAutocompleteExample {
|
||||||
|
|
||||||
this.fruitCtrl.setValue(null);
|
this.fruitCtrl.setValue(null);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
remove(fruit: string): void {
|
remove(fruit: string): void {
|
||||||
const index = this.fruits.indexOf(fruit);
|
const index = this.fruits.indexOf(fruit);
|
||||||
|
|
|
@ -1,16 +1 @@
|
||||||
.example-header {
|
/** No CSS for this example */
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 0.5em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.example-header-label {
|
|
||||||
flex: 1;
|
|
||||||
height: 1em;
|
|
||||||
font-weight: bold;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.example-double-arrow .mat-icon {
|
|
||||||
margin: -22%;
|
|
||||||
}
|
|
||||||
|
|
|
@ -15,7 +15,6 @@ import {takeUntil} from 'rxjs/operators';
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'datepicker-custom-header-example',
|
selector: 'datepicker-custom-header-example',
|
||||||
templateUrl: 'datepicker-custom-header-example.html',
|
templateUrl: 'datepicker-custom-header-example.html',
|
||||||
styleUrls: ['datepicker-custom-header-example.css'],
|
|
||||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||||
})
|
})
|
||||||
export class DatepickerCustomHeaderExample {
|
export class DatepickerCustomHeaderExample {
|
||||||
|
@ -25,6 +24,24 @@ export class DatepickerCustomHeaderExample {
|
||||||
/** Custom header component for datepicker. */
|
/** Custom header component for datepicker. */
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'example-header',
|
selector: 'example-header',
|
||||||
|
styles: [`
|
||||||
|
.example-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.example-header-label {
|
||||||
|
flex: 1;
|
||||||
|
height: 1em;
|
||||||
|
font-weight: 500;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.example-double-arrow .mat-icon {
|
||||||
|
margin: -22%;
|
||||||
|
}
|
||||||
|
`],
|
||||||
template: `
|
template: `
|
||||||
<div class="example-header">
|
<div class="example-header">
|
||||||
<button mat-icon-button class="example-double-arrow" (click)="previousClicked('year')">
|
<button mat-icon-button class="example-double-arrow" (click)="previousClicked('year')">
|
||||||
|
|
|
@ -16,7 +16,7 @@ import {
|
||||||
styleUrls: ['focus-monitor-focus-via-example.css']
|
styleUrls: ['focus-monitor-focus-via-example.css']
|
||||||
})
|
})
|
||||||
export class FocusMonitorFocusViaExample implements OnDestroy, OnInit {
|
export class FocusMonitorFocusViaExample implements OnDestroy, OnInit {
|
||||||
@ViewChild('monitored') monitoredEl: ElementRef;
|
@ViewChild('monitored') monitoredEl: ElementRef<HTMLElement>;
|
||||||
|
|
||||||
origin = this.formatOrigin(null);
|
origin = this.formatOrigin(null);
|
||||||
|
|
||||||
|
@ -25,7 +25,7 @@ export class FocusMonitorFocusViaExample implements OnDestroy, OnInit {
|
||||||
private ngZone: NgZone) {}
|
private ngZone: NgZone) {}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.focusMonitor.monitor(this.monitoredEl.nativeElement)
|
this.focusMonitor.monitor(this.monitoredEl)
|
||||||
.subscribe(origin => this.ngZone.run(() => {
|
.subscribe(origin => this.ngZone.run(() => {
|
||||||
this.origin = this.formatOrigin(origin);
|
this.origin = this.formatOrigin(origin);
|
||||||
this.cdr.markForCheck();
|
this.cdr.markForCheck();
|
||||||
|
@ -33,7 +33,7 @@ export class FocusMonitorFocusViaExample implements OnDestroy, OnInit {
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
this.focusMonitor.stopMonitoring(this.monitoredEl.nativeElement);
|
this.focusMonitor.stopMonitoring(this.monitoredEl);
|
||||||
}
|
}
|
||||||
|
|
||||||
formatOrigin(origin: FocusOrigin): string {
|
formatOrigin(origin: FocusOrigin): string {
|
||||||
|
|
|
@ -16,8 +16,8 @@ import {
|
||||||
styleUrls: ['focus-monitor-overview-example.css']
|
styleUrls: ['focus-monitor-overview-example.css']
|
||||||
})
|
})
|
||||||
export class FocusMonitorOverviewExample implements OnDestroy, OnInit {
|
export class FocusMonitorOverviewExample implements OnDestroy, OnInit {
|
||||||
@ViewChild('element') element: ElementRef;
|
@ViewChild('element') element: ElementRef<HTMLElement>;
|
||||||
@ViewChild('subtree') subtree: ElementRef;
|
@ViewChild('subtree') subtree: ElementRef<HTMLElement>;
|
||||||
|
|
||||||
elementOrigin = this.formatOrigin(null);
|
elementOrigin = this.formatOrigin(null);
|
||||||
subtreeOrigin = this.formatOrigin(null);
|
subtreeOrigin = this.formatOrigin(null);
|
||||||
|
@ -27,12 +27,12 @@ export class FocusMonitorOverviewExample implements OnDestroy, OnInit {
|
||||||
private ngZone: NgZone) {}
|
private ngZone: NgZone) {}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.focusMonitor.monitor(this.element.nativeElement)
|
this.focusMonitor.monitor(this.element)
|
||||||
.subscribe(origin => this.ngZone.run(() => {
|
.subscribe(origin => this.ngZone.run(() => {
|
||||||
this.elementOrigin = this.formatOrigin(origin);
|
this.elementOrigin = this.formatOrigin(origin);
|
||||||
this.cdr.markForCheck();
|
this.cdr.markForCheck();
|
||||||
}));
|
}));
|
||||||
this.focusMonitor.monitor(this.subtree.nativeElement, true)
|
this.focusMonitor.monitor(this.subtree, true)
|
||||||
.subscribe(origin => this.ngZone.run(() => {
|
.subscribe(origin => this.ngZone.run(() => {
|
||||||
this.subtreeOrigin = this.formatOrigin(origin);
|
this.subtreeOrigin = this.formatOrigin(origin);
|
||||||
this.cdr.markForCheck();
|
this.cdr.markForCheck();
|
||||||
|
@ -40,8 +40,8 @@ export class FocusMonitorOverviewExample implements OnDestroy, OnInit {
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
this.focusMonitor.stopMonitoring(this.element.nativeElement);
|
this.focusMonitor.stopMonitoring(this.element);
|
||||||
this.focusMonitor.stopMonitoring(this.subtree.nativeElement);
|
this.focusMonitor.stopMonitoring(this.subtree);
|
||||||
}
|
}
|
||||||
|
|
||||||
formatOrigin(origin: FocusOrigin): string {
|
formatOrigin(origin: FocusOrigin): string {
|
||||||
|
|
|
@ -0,0 +1,21 @@
|
||||||
|
.example-tel-input-container {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.example-tel-input-element {
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
padding: 0;
|
||||||
|
outline: none;
|
||||||
|
font: inherit;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.example-tel-input-spacer {
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 200ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host.example-floating .example-tel-input-spacer {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
<div [formGroup]="parts" class="example-tel-input-container">
|
||||||
|
<input class="example-tel-input-element" formControlName="area" size="3">
|
||||||
|
<span class="example-tel-input-spacer">–</span>
|
||||||
|
<input class="example-tel-input-element" formControlName="exchange" size="3">
|
||||||
|
<span class="example-tel-input-spacer">–</span>
|
||||||
|
<input class="example-tel-input-element" formControlName="subscriber" size="4">
|
||||||
|
</div>
|
|
@ -1,21 +1 @@
|
||||||
div {
|
/** No CSS for this example */
|
||||||
display: flex;
|
|
||||||
}
|
|
||||||
|
|
||||||
input {
|
|
||||||
border: none;
|
|
||||||
background: none;
|
|
||||||
padding: 0;
|
|
||||||
outline: none;
|
|
||||||
font: inherit;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
span {
|
|
||||||
opacity: 0;
|
|
||||||
transition: opacity 200ms;
|
|
||||||
}
|
|
||||||
|
|
||||||
:host.floating span {
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
<div [formGroup]="parts">
|
<mat-form-field>
|
||||||
<input class="area" formControlName="area" size="3">
|
<example-tel-input placeholder="Phone number" required></example-tel-input>
|
||||||
<span>–</span>
|
<mat-icon matSuffix>phone</mat-icon>
|
||||||
<input class="exchange" formControlName="exchange" size="3">
|
<mat-hint>Include area code</mat-hint>
|
||||||
<span>–</span>
|
</mat-form-field>
|
||||||
<input class="subscriber" formControlName="subscriber" size="4">
|
|
||||||
</div>
|
|
||||||
|
|
|
@ -5,20 +5,27 @@ import {FormBuilder, FormGroup} from '@angular/forms';
|
||||||
import {MatFormFieldControl} from '@angular/material';
|
import {MatFormFieldControl} from '@angular/material';
|
||||||
import {Subject} from 'rxjs';
|
import {Subject} from 'rxjs';
|
||||||
|
|
||||||
|
/** @title Form field with custom telephone number input control. */
|
||||||
|
@Component({
|
||||||
|
selector: 'form-field-custom-control-example',
|
||||||
|
templateUrl: 'form-field-custom-control-example.html',
|
||||||
|
styleUrls: ['form-field-custom-control-example.css'],
|
||||||
|
})
|
||||||
|
export class FormFieldCustomControlExample {}
|
||||||
|
|
||||||
/** Data structure for holding telephone number. */
|
/** Data structure for holding telephone number. */
|
||||||
export class MyTel {
|
export class MyTel {
|
||||||
constructor(public area: string, public exchange: string, public subscriber: string) {}
|
constructor(public area: string, public exchange: string, public subscriber: string) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/** Custom `MatFormFieldControl` for telephone number input. */
|
/** Custom `MatFormFieldControl` for telephone number input. */
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'my-tel-input',
|
selector: 'example-tel-input',
|
||||||
templateUrl: 'form-field-custom-control-example.html',
|
templateUrl: 'example-tel-input-example.html',
|
||||||
styleUrls: ['form-field-custom-control-example.css'],
|
styleUrls: ['example-tel-input-example.css'],
|
||||||
providers: [{provide: MatFormFieldControl, useExisting: MyTelInput}],
|
providers: [{provide: MatFormFieldControl, useExisting: MyTelInput}],
|
||||||
host: {
|
host: {
|
||||||
'[class.floating]': 'shouldLabelFloat',
|
'[class.example-floating]': 'shouldLabelFloat',
|
||||||
'[id]': 'id',
|
'[id]': 'id',
|
||||||
'[attr.aria-describedby]': 'describedBy',
|
'[attr.aria-describedby]': 'describedBy',
|
||||||
}
|
}
|
||||||
|
@ -31,8 +38,8 @@ export class MyTelInput implements MatFormFieldControl<MyTel>, OnDestroy {
|
||||||
focused = false;
|
focused = false;
|
||||||
ngControl = null;
|
ngControl = null;
|
||||||
errorState = false;
|
errorState = false;
|
||||||
controlType = 'my-tel-input';
|
controlType = 'example-tel-input';
|
||||||
id = `my-tel-input-${MyTelInput.nextId++}`;
|
id = `example-tel-input-${MyTelInput.nextId++}`;
|
||||||
describedBy = '';
|
describedBy = '';
|
||||||
|
|
||||||
get empty() {
|
get empty() {
|
||||||
|
@ -81,14 +88,14 @@ export class MyTelInput implements MatFormFieldControl<MyTel>, OnDestroy {
|
||||||
this.stateChanges.next();
|
this.stateChanges.next();
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor(fb: FormBuilder, private fm: FocusMonitor, private elRef: ElementRef) {
|
constructor(fb: FormBuilder, private fm: FocusMonitor, private elRef: ElementRef<HTMLElement>) {
|
||||||
this.parts = fb.group({
|
this.parts = fb.group({
|
||||||
area: '',
|
area: '',
|
||||||
exchange: '',
|
exchange: '',
|
||||||
subscriber: '',
|
subscriber: '',
|
||||||
});
|
});
|
||||||
|
|
||||||
fm.monitor(elRef.nativeElement, true).subscribe(origin => {
|
fm.monitor(elRef, true).subscribe(origin => {
|
||||||
this.focused = !!origin;
|
this.focused = !!origin;
|
||||||
this.stateChanges.next();
|
this.stateChanges.next();
|
||||||
});
|
});
|
||||||
|
@ -96,7 +103,7 @@ export class MyTelInput implements MatFormFieldControl<MyTel>, OnDestroy {
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
this.stateChanges.complete();
|
this.stateChanges.complete();
|
||||||
this.fm.stopMonitoring(this.elRef.nativeElement);
|
this.fm.stopMonitoring(this.elRef);
|
||||||
}
|
}
|
||||||
|
|
||||||
setDescribedByIds(ids: string[]) {
|
setDescribedByIds(ids: string[]) {
|
||||||
|
@ -105,21 +112,7 @@ export class MyTelInput implements MatFormFieldControl<MyTel>, OnDestroy {
|
||||||
|
|
||||||
onContainerClick(event: MouseEvent) {
|
onContainerClick(event: MouseEvent) {
|
||||||
if ((event.target as Element).tagName.toLowerCase() != 'input') {
|
if ((event.target as Element).tagName.toLowerCase() != 'input') {
|
||||||
this.elRef.nativeElement.querySelector('input').focus();
|
this.elRef.nativeElement.querySelector('input')!.focus();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/** @title Form field with custom telephone number input control. */
|
|
||||||
@Component({
|
|
||||||
selector: 'form-field-custom-control-example',
|
|
||||||
template: `
|
|
||||||
<mat-form-field>
|
|
||||||
<my-tel-input placeholder="Phone number" required></my-tel-input>
|
|
||||||
<mat-icon matSuffix>phone</mat-icon>
|
|
||||||
<mat-hint>Include area code</mat-hint>
|
|
||||||
</mat-form-field>
|
|
||||||
`
|
|
||||||
})
|
|
||||||
export class FormFieldCustomControlExample {}
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
<div class="example-container">
|
<div class="example-container">
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<input matInput placeholder="Enter your password" [type]="hide ? 'password' : 'text'">
|
<input matInput placeholder="Enter your password" [type]="hide ? 'password' : 'text'">
|
||||||
<mat-icon matSuffix (click)="hide = !hide">{{hide ? 'visibility' : 'visibility_off'}}</mat-icon>
|
<mat-icon matSuffix (click)="hide = !hide">{{hide ? 'visibility_off' : 'visibility'}}</mat-icon>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
.example-ripple-container {
|
||||||
|
cursor: pointer;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
width: 300px;
|
||||||
|
height: 300px;
|
||||||
|
line-height: 300px;
|
||||||
|
|
||||||
|
user-select: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
-ms-user-select: none;
|
||||||
|
|
||||||
|
-webkit-user-drag: none;
|
||||||
|
-webkit-tap-highlight-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Styles to make the demo look better. */
|
||||||
|
.example-ripple-checkbox {
|
||||||
|
margin: 6px 12px 6px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.example-ripple-form-field {
|
||||||
|
margin: 0 12px 0 0;
|
||||||
|
}
|
|
@ -0,0 +1,21 @@
|
||||||
|
<mat-checkbox [(ngModel)]="centered" class="example-ripple-checkbox">Centered</mat-checkbox>
|
||||||
|
<mat-checkbox [(ngModel)]="disabled" class="example-ripple-checkbox">Disabled</mat-checkbox>
|
||||||
|
<mat-checkbox [(ngModel)]="unbounded" class="example-ripple-checkbox">Unbounded</mat-checkbox>
|
||||||
|
|
||||||
|
<mat-form-field class="example-ripple-form-field">
|
||||||
|
<input matInput [(ngModel)]="radius" type="number" placeholder="Radius">
|
||||||
|
</mat-form-field>
|
||||||
|
<mat-form-field class="example-ripple-form-field">
|
||||||
|
<input matInput [(ngModel)]="color" type="text" placeholder="Color">
|
||||||
|
</mat-form-field>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="example-ripple-container mat-elevation-z4"
|
||||||
|
matRipple
|
||||||
|
[matRippleCentered]="centered"
|
||||||
|
[matRippleDisabled]="disabled"
|
||||||
|
[matRippleUnbounded]="unbounded"
|
||||||
|
[matRippleRadius]="radius"
|
||||||
|
[matRippleColor]="color">
|
||||||
|
Click me
|
||||||
|
</div>
|
|
@ -0,0 +1,18 @@
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title MatRipple basic usage
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'ripple-overview-example',
|
||||||
|
templateUrl: 'ripple-overview-example.html',
|
||||||
|
styleUrls: ['ripple-overview-example.css'],
|
||||||
|
})
|
||||||
|
export class RippleOverviewExample {
|
||||||
|
centered = false;
|
||||||
|
disabled = false;
|
||||||
|
unbounded = false;
|
||||||
|
|
||||||
|
radius: number;
|
||||||
|
color: string;
|
||||||
|
}
|
|
@ -1,7 +1,8 @@
|
||||||
<p>
|
<p>
|
||||||
<mat-checkbox [formControl]="disableSelect">Disable select</mat-checkbox>
|
<mat-checkbox [formControl]="disableSelect">Disable select</mat-checkbox>
|
||||||
</p>
|
</p>
|
||||||
<p>
|
|
||||||
|
<h4>mat-select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="Choose an option" [disabled]="disableSelect.value">
|
<mat-select placeholder="Choose an option" [disabled]="disableSelect.value">
|
||||||
<mat-option value="option1">Option 1</mat-option>
|
<mat-option value="option1">Option 1</mat-option>
|
||||||
|
@ -9,4 +10,15 @@
|
||||||
<mat-option value="option3">Option 3</mat-option>
|
<mat-option value="option3">Option 3</mat-option>
|
||||||
</mat-select>
|
</mat-select>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
</p>
|
|
||||||
|
<h4>native html select</h4>
|
||||||
|
<mat-form-field>
|
||||||
|
<select matNativeControl placeholder="Choose an option" [disabled]="disableSelect.value">
|
||||||
|
<option value="" disabled selected></option>
|
||||||
|
<option value="volvo">Volvo</option>
|
||||||
|
<option value="saab" disabled>Saab</option>
|
||||||
|
<option value="mercedes">Mercedes</option>
|
||||||
|
<option value="audi">Audi</option>
|
||||||
|
</select>
|
||||||
|
</mat-form-field>
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
<h4>mat-select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="Choose one" [formControl]="selected" [errorStateMatcher]="matcher">
|
<mat-select placeholder="Choose one" [formControl]="selected" [errorStateMatcher]="matcher">
|
||||||
<mat-option>Clear</mat-option>
|
<mat-option>Clear</mat-option>
|
||||||
|
@ -10,3 +11,16 @@
|
||||||
Your selection is invalid
|
Your selection is invalid
|
||||||
</mat-error>
|
</mat-error>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
|
||||||
|
<h4>native html select</h4>
|
||||||
|
<mat-form-field class="demo-full-width">
|
||||||
|
<select matNativeControl placeholder="Choose one" [formControl]="nativeSelectFormControl" [errorStateMatcher]="matcher">
|
||||||
|
<option value=""></option>
|
||||||
|
<option value="valid" selected>Valid option</option>
|
||||||
|
<option value="invalid">Invalid option</option>
|
||||||
|
</select>
|
||||||
|
<mat-error *ngIf="nativeSelectFormControl.hasError('required')">You must make a selection</mat-error>
|
||||||
|
<mat-error *ngIf="nativeSelectFormControl.hasError('pattern') && !nativeSelectFormControl.hasError('required')">
|
||||||
|
Your selection is invalid
|
||||||
|
</mat-error>
|
||||||
|
</mat-form-field>
|
||||||
|
|
|
@ -22,5 +22,15 @@ export class SelectErrorStateMatcherExample {
|
||||||
Validators.pattern('valid'),
|
Validators.pattern('valid'),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
selectFormControl = new FormControl('valid', [
|
||||||
|
Validators.required,
|
||||||
|
Validators.pattern('valid'),
|
||||||
|
]);
|
||||||
|
|
||||||
|
nativeSelectFormControl = new FormControl('valid', [
|
||||||
|
Validators.required,
|
||||||
|
Validators.pattern('valid'),
|
||||||
|
]);
|
||||||
|
|
||||||
matcher = new MyErrorStateMatcher();
|
matcher = new MyErrorStateMatcher();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
<form>
|
<form>
|
||||||
|
<h4>mat-select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="Favorite food" [(ngModel)]="selectedValue" name="food">
|
<mat-select placeholder="Favorite food" [(ngModel)]="selectedValue" name="food">
|
||||||
<mat-option *ngFor="let food of foods" [value]="food.value">
|
<mat-option *ngFor="let food of foods" [value]="food.value">
|
||||||
|
@ -6,6 +7,15 @@
|
||||||
</mat-option>
|
</mat-option>
|
||||||
</mat-select>
|
</mat-select>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
<p> Selected food: {{selectedValue}} </p>
|
||||||
<p> Selected value: {{selectedValue}} </p>
|
<h4>native html select</h4>
|
||||||
|
<mat-form-field>
|
||||||
|
<select matNativeControl placeholder="Favorite car" [(ngModel)]="selectedCar" name="car">
|
||||||
|
<option value="" disabled selected></option>
|
||||||
|
<option *ngFor="let car of cars" [value]="car.value">
|
||||||
|
{{car.viewValue}}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</mat-form-field>
|
||||||
|
<p> Selected car: {{selectedCar}} </p>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -5,6 +5,11 @@ export interface Food {
|
||||||
viewValue: string;
|
viewValue: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface Car {
|
||||||
|
value: string;
|
||||||
|
viewValue: string;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @title Select in a form
|
* @title Select in a form
|
||||||
*/
|
*/
|
||||||
|
@ -15,10 +20,17 @@ export interface Food {
|
||||||
})
|
})
|
||||||
export class SelectFormExample {
|
export class SelectFormExample {
|
||||||
selectedValue: string;
|
selectedValue: string;
|
||||||
|
selectedCar: string;
|
||||||
|
|
||||||
foods: Food[] = [
|
foods: Food[] = [
|
||||||
{value: 'steak-0', viewValue: 'Steak'},
|
{value: 'steak-0', viewValue: 'Steak'},
|
||||||
{value: 'pizza-1', viewValue: 'Pizza'},
|
{value: 'pizza-1', viewValue: 'Pizza'},
|
||||||
{value: 'tacos-2', viewValue: 'Tacos'}
|
{value: 'tacos-2', viewValue: 'Tacos'}
|
||||||
];
|
];
|
||||||
|
|
||||||
|
cars: Car[] = [
|
||||||
|
{value: 'volvo', viewValue: 'Volvo'},
|
||||||
|
{value: 'saab', viewValue: 'Saab'},
|
||||||
|
{value: 'mercedes', viewValue: 'Mercedes'}
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
<h4>mat select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="Favorite animal" [formControl]="animalControl" required>
|
<mat-select placeholder="Favorite animal" [formControl]="animalControl" required>
|
||||||
<mat-option>--</mat-option>
|
<mat-option>--</mat-option>
|
||||||
|
@ -8,3 +9,19 @@
|
||||||
<mat-error *ngIf="animalControl.hasError('required')">Please choose an animal</mat-error>
|
<mat-error *ngIf="animalControl.hasError('required')">Please choose an animal</mat-error>
|
||||||
<mat-hint>{{animalControl.value?.sound}}</mat-hint>
|
<mat-hint>{{animalControl.value?.sound}}</mat-hint>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
|
||||||
|
<h4>native html select</h4>
|
||||||
|
<mat-form-field>
|
||||||
|
<mat-label>Select your car (required)</mat-label>
|
||||||
|
<select matNativeControl required [formControl]="selectFormControl">
|
||||||
|
<option label="--select something --"></option>
|
||||||
|
<option value="saab">Saab</option>
|
||||||
|
<option value="mercedes">Mercedes</option>
|
||||||
|
<option value="audi">Audi</option>
|
||||||
|
</select>
|
||||||
|
<mat-error *ngIf="selectFormControl.hasError('required')">
|
||||||
|
This field is required
|
||||||
|
</mat-error>
|
||||||
|
<mat-hint>You can pick up your favorite car here</mat-hint>
|
||||||
|
</mat-form-field>
|
||||||
|
|
||||||
|
|
|
@ -14,6 +14,7 @@ export interface Animal {
|
||||||
})
|
})
|
||||||
export class SelectHintErrorExample {
|
export class SelectHintErrorExample {
|
||||||
animalControl = new FormControl('', [Validators.required]);
|
animalControl = new FormControl('', [Validators.required]);
|
||||||
|
selectFormControl = new FormControl('', Validators.required);
|
||||||
animals: Animal[] = [
|
animals: Animal[] = [
|
||||||
{name: 'Dog', sound: 'Woof!'},
|
{name: 'Dog', sound: 'Woof!'},
|
||||||
{name: 'Cat', sound: 'Meow!'},
|
{name: 'Cat', sound: 'Meow!'},
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
<h4>mat-select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="Pokemon" [formControl]="pokemonControl">
|
<mat-select placeholder="Pokemon" [formControl]="pokemonControl">
|
||||||
<mat-option>-- None --</mat-option>
|
<mat-option>-- None --</mat-option>
|
||||||
|
@ -9,3 +10,17 @@
|
||||||
</mat-optgroup>
|
</mat-optgroup>
|
||||||
</mat-select>
|
</mat-select>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
|
||||||
|
<h4>native html select</h4>
|
||||||
|
<mat-form-field>
|
||||||
|
<select matNativeControl>
|
||||||
|
<optgroup label="Swedish Cars">
|
||||||
|
<option value="volvo">volvo</option>
|
||||||
|
<option value="saab">Saab</option>
|
||||||
|
</optgroup>
|
||||||
|
<optgroup label="German Cars">
|
||||||
|
<option value="mercedes">Mercedes</option>
|
||||||
|
<option value="audi">Audi</option>
|
||||||
|
</optgroup>
|
||||||
|
</select>
|
||||||
|
</mat-form-field>
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
<h4>Basic mat-select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="Favorite food">
|
<mat-select placeholder="Favorite food">
|
||||||
<mat-option *ngFor="let food of foods" [value]="food.value">
|
<mat-option *ngFor="let food of foods" [value]="food.value">
|
||||||
|
@ -5,3 +6,13 @@
|
||||||
</mat-option>
|
</mat-option>
|
||||||
</mat-select>
|
</mat-select>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
|
||||||
|
<h4>Basic native select</h4>
|
||||||
|
<mat-form-field>
|
||||||
|
<select matNativeControl required>
|
||||||
|
<option value="volvo">Volvo</option>
|
||||||
|
<option value="saab">Saab</option>
|
||||||
|
<option value="mercedes">Mercedes</option>
|
||||||
|
<option value="audi">Audi</option>
|
||||||
|
</select>
|
||||||
|
</mat-form-field>
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
.example-panel-red .mat-select-content {
|
.example-panel-red .mat-select-panel {
|
||||||
background: rgba(255, 0, 0, 0.5);
|
background: rgba(255, 0, 0, 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
.example-panel-green .mat-select-content {
|
.example-panel-green .mat-select-panel {
|
||||||
background: rgba(0, 255, 0, 0.5);
|
background: rgba(0, 255, 0, 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
.example-panel-blue .mat-select-content {
|
.example-panel-blue .mat-select-panel {
|
||||||
background: rgba(0, 0, 255, 0.5);
|
background: rgba(0, 0, 255, 0.5);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,19 @@
|
||||||
|
<h4>mat-select</h4>
|
||||||
<mat-form-field>
|
<mat-form-field>
|
||||||
<mat-select placeholder="State">
|
<mat-select placeholder="State">
|
||||||
<mat-option>None</mat-option>
|
<mat-option>None</mat-option>
|
||||||
<mat-option *ngFor="let state of states" [value]="state">{{state}}</mat-option>
|
<mat-option *ngFor="let state of states" [value]="state">{{state}}</mat-option>
|
||||||
</mat-select>
|
</mat-select>
|
||||||
</mat-form-field>
|
</mat-form-field>
|
||||||
|
|
||||||
|
<h4>native html select</h4>
|
||||||
|
<mat-form-field>
|
||||||
|
<mat-label>Select your car</mat-label>
|
||||||
|
<select matNativeControl id="mySelectId">
|
||||||
|
<option value="" disabled selected></option>
|
||||||
|
<option value="volvo">Volvo</option>
|
||||||
|
<option value="saab">Saab</option>
|
||||||
|
<option value="mercedes">Mercedes</option>
|
||||||
|
<option value="audi">Audi</option>
|
||||||
|
</select>
|
||||||
|
</mat-form-field>
|
||||||
|
|
|
@ -53,6 +53,6 @@ export class SortOverviewExample {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function compare(a, b, isAsc) {
|
function compare(a: number | string, b: number | string, isAsc: boolean) {
|
||||||
return (a < b ? -1 : 1) * (isAsc ? 1 : -1);
|
return (a < b ? -1 : 1) * (isAsc ? 1 : -1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,33 @@
|
||||||
|
<mat-horizontal-stepper linear #stepper>
|
||||||
|
<mat-step [stepControl]="firstFormGroup" errorMessage="Name is required.">
|
||||||
|
<form [formGroup]="firstFormGroup">
|
||||||
|
<ng-template matStepLabel>Fill out your name</ng-template>
|
||||||
|
<mat-form-field>
|
||||||
|
<input matInput placeholder="Last name, First name" formControlName="firstCtrl" required>
|
||||||
|
</mat-form-field>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step [stepControl]="secondFormGroup" errorMessage="Address is required.">
|
||||||
|
<form [formGroup]="secondFormGroup">
|
||||||
|
<ng-template matStepLabel>Fill out your address</ng-template>
|
||||||
|
<mat-form-field>
|
||||||
|
<input matInput placeholder="Address" formControlName="secondCtrl" required>
|
||||||
|
</mat-form-field>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step>
|
||||||
|
<ng-template matStepLabel>Done</ng-template>
|
||||||
|
You are now done.
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button (click)="stepper.reset()">Reset</button>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
</mat-horizontal-stepper>
|
|
@ -0,0 +1,30 @@
|
||||||
|
import {Component, OnInit} from '@angular/core';
|
||||||
|
import {FormBuilder, FormGroup, Validators} from '@angular/forms';
|
||||||
|
import {MAT_STEPPER_GLOBAL_OPTIONS} from '@angular/cdk/stepper';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Stepper that displays errors in the steps
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'stepper-errors-example',
|
||||||
|
templateUrl: 'stepper-errors-example.html',
|
||||||
|
styleUrls: ['stepper-errors-example.css'],
|
||||||
|
providers: [{
|
||||||
|
provide: MAT_STEPPER_GLOBAL_OPTIONS, useValue: {showError: true}
|
||||||
|
}]
|
||||||
|
})
|
||||||
|
export class StepperErrorsExample implements OnInit {
|
||||||
|
firstFormGroup: FormGroup;
|
||||||
|
secondFormGroup: FormGroup;
|
||||||
|
|
||||||
|
constructor(private _formBuilder: FormBuilder) {}
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
this.firstFormGroup = this._formBuilder.group({
|
||||||
|
firstCtrl: ['', Validators.required]
|
||||||
|
});
|
||||||
|
this.secondFormGroup = this._formBuilder.group({
|
||||||
|
secondCtrl: ['', Validators.required]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1 @@
|
||||||
|
/** No CSS for this example */
|
|
@ -0,0 +1,33 @@
|
||||||
|
<mat-horizontal-stepper labelPosition="bottom" #stepper>
|
||||||
|
<mat-step [stepControl]="firstFormGroup">
|
||||||
|
<form [formGroup]="firstFormGroup">
|
||||||
|
<ng-template matStepLabel>Fill out your name</ng-template>
|
||||||
|
<mat-form-field>
|
||||||
|
<input matInput placeholder="Last name, First name" formControlName="firstCtrl" required>
|
||||||
|
</mat-form-field>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step [stepControl]="secondFormGroup" optional>
|
||||||
|
<form [formGroup]="secondFormGroup">
|
||||||
|
<ng-template matStepLabel>Fill out your address</ng-template>
|
||||||
|
<mat-form-field>
|
||||||
|
<input matInput placeholder="Address" formControlName="secondCtrl" required>
|
||||||
|
</mat-form-field>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step>
|
||||||
|
<ng-template matStepLabel>Done</ng-template>
|
||||||
|
You are now done.
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button (click)="stepper.reset()">Reset</button>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
</mat-horizontal-stepper>
|
|
@ -0,0 +1,26 @@
|
||||||
|
import {Component, OnInit} from '@angular/core';
|
||||||
|
import {FormBuilder, FormGroup, Validators} from '@angular/forms';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Stepper label bottom position
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'stepper-label-position-bottom-example',
|
||||||
|
templateUrl: 'stepper-label-position-bottom-example.html',
|
||||||
|
styleUrls: ['stepper-label-position-bottom-example.css'],
|
||||||
|
})
|
||||||
|
export class StepperLabelPositionBottomExample implements OnInit {
|
||||||
|
firstFormGroup: FormGroup;
|
||||||
|
secondFormGroup: FormGroup;
|
||||||
|
|
||||||
|
constructor(private _formBuilder: FormBuilder) {}
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
this.firstFormGroup = this._formBuilder.group({
|
||||||
|
firstCtrl: ['', Validators.required]
|
||||||
|
});
|
||||||
|
this.secondFormGroup = this._formBuilder.group({
|
||||||
|
secondCtrl: ['', Validators.required]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,60 @@
|
||||||
|
<mat-horizontal-stepper #stepper>
|
||||||
|
<mat-step [stepControl]="firstFormGroup">
|
||||||
|
<form [formGroup]="firstFormGroup">
|
||||||
|
<ng-template matStepLabel>Fill out your name</ng-template>
|
||||||
|
<mat-form-field>
|
||||||
|
<input matInput placeholder="Last name, First name" formControlName="firstCtrl" required>
|
||||||
|
</mat-form-field>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step [stepControl]="secondFormGroup">
|
||||||
|
<form [formGroup]="secondFormGroup">
|
||||||
|
<ng-template matStepLabel>Fill out your address</ng-template>
|
||||||
|
<mat-form-field>
|
||||||
|
<input matInput placeholder="Address" formControlName="secondCtrl" required>
|
||||||
|
</mat-form-field>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step>
|
||||||
|
<ng-template matStepLabel>Done</ng-template>
|
||||||
|
You are now done.
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button (click)="stepper.reset()">Reset</button>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
</mat-horizontal-stepper>
|
||||||
|
|
||||||
|
<mat-horizontal-stepper>
|
||||||
|
<mat-step label="Step 1" state="phone">
|
||||||
|
<p>Put down your phones.</p>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step label="Step 2" state="chat">
|
||||||
|
<p>Socialize with each other.</p>
|
||||||
|
<div>
|
||||||
|
<button mat-button matStepperPrevious>Back</button>
|
||||||
|
<button mat-button matStepperNext>Next</button>
|
||||||
|
</div>
|
||||||
|
</mat-step>
|
||||||
|
<mat-step label="Step 3">
|
||||||
|
<p>You're welcome.</p>
|
||||||
|
</mat-step>
|
||||||
|
|
||||||
|
<!-- Icon overrides. -->
|
||||||
|
<ng-template matStepperIcon="phone">
|
||||||
|
<mat-icon>call_end</mat-icon>
|
||||||
|
</ng-template>
|
||||||
|
<ng-template matStepperIcon="chat">
|
||||||
|
<mat-icon>forum</mat-icon>
|
||||||
|
</ng-template>
|
||||||
|
</mat-horizontal-stepper>
|
|
@ -0,0 +1,30 @@
|
||||||
|
import {Component, OnInit} from '@angular/core';
|
||||||
|
import {FormBuilder, FormGroup, Validators} from '@angular/forms';
|
||||||
|
import {MAT_STEPPER_GLOBAL_OPTIONS} from '@angular/cdk/stepper';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Stepper with customized states
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'stepper-states-example',
|
||||||
|
templateUrl: 'stepper-states-example.html',
|
||||||
|
styleUrls: ['stepper-states-example.css'],
|
||||||
|
providers: [{
|
||||||
|
provide: MAT_STEPPER_GLOBAL_OPTIONS, useValue: {displayDefaultIndicatorType: false}
|
||||||
|
}]
|
||||||
|
})
|
||||||
|
export class StepperStatesExample implements OnInit {
|
||||||
|
firstFormGroup: FormGroup;
|
||||||
|
secondFormGroup: FormGroup;
|
||||||
|
|
||||||
|
constructor(private _formBuilder: FormBuilder) {}
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
this.firstFormGroup = this._formBuilder.group({
|
||||||
|
firstCtrl: ['', Validators.required]
|
||||||
|
});
|
||||||
|
this.secondFormGroup = this._formBuilder.group({
|
||||||
|
secondCtrl: ['', Validators.required]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,7 +5,7 @@ import {FormBuilder, FormGroup, Validators} from '@angular/forms';
|
||||||
* @title Stepper vertical
|
* @title Stepper vertical
|
||||||
*/
|
*/
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'stepper-vertical',
|
selector: 'stepper-vertical-example',
|
||||||
templateUrl: 'stepper-vertical-example.html',
|
templateUrl: 'stepper-vertical-example.html',
|
||||||
styleUrls: ['stepper-vertical-example.css']
|
styleUrls: ['stepper-vertical-example.css']
|
||||||
})
|
})
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
.mat-tab-group {
|
||||||
|
margin-bottom: 48px;
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
<mat-tab-group mat-align-tabs="start">
|
||||||
|
<mat-tab label="First">Content 1</mat-tab>
|
||||||
|
<mat-tab label="Second">Content 2</mat-tab>
|
||||||
|
<mat-tab label="Third">Content 3</mat-tab>
|
||||||
|
</mat-tab-group>
|
||||||
|
|
||||||
|
<mat-tab-group mat-align-tabs="center">
|
||||||
|
<mat-tab label="First">Content 1</mat-tab>
|
||||||
|
<mat-tab label="Second">Content 2</mat-tab>
|
||||||
|
<mat-tab label="Third">Content 3</mat-tab>
|
||||||
|
</mat-tab-group>
|
||||||
|
|
||||||
|
<mat-tab-group mat-align-tabs="end">
|
||||||
|
<mat-tab label="First">Content 1</mat-tab>
|
||||||
|
<mat-tab label="Second">Content 2</mat-tab>
|
||||||
|
<mat-tab label="Third">Content 3</mat-tab>
|
||||||
|
</mat-tab-group>
|
|
@ -0,0 +1,11 @@
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Tab group with aligned labels
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'tab-group-align-example',
|
||||||
|
templateUrl: 'tab-group-align-example.html',
|
||||||
|
styleUrls: ['tab-group-align-example.css'],
|
||||||
|
})
|
||||||
|
export class TabGroupAlignExample {}
|
|
@ -0,0 +1,3 @@
|
||||||
|
table {
|
||||||
|
width: 100%;
|
||||||
|
}
|
|
@ -0,0 +1,15 @@
|
||||||
|
<table mat-table [dataSource]="dataSource" class="mat-elevation-z8"
|
||||||
|
matSort #sort="matSort">
|
||||||
|
<!-- Basic column: name is used for header label AND data property -->
|
||||||
|
<simple-column name="name" sortable></simple-column>
|
||||||
|
<simple-column name="position"></simple-column>
|
||||||
|
|
||||||
|
<!-- Name doesn't match the data property (or transform needed); define a custom data accessor -->
|
||||||
|
<simple-column name="weight" [dataAccessor]="getWeight"></simple-column>
|
||||||
|
|
||||||
|
<!-- Name doesn't match desired header text; define a custom label -->
|
||||||
|
<simple-column name="symbol" label="SYMBOL!"></simple-column>
|
||||||
|
|
||||||
|
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
||||||
|
<tr mat-row *matRowDef="let data; columns: displayedColumns;"></tr>
|
||||||
|
</table>
|
|
@ -0,0 +1,130 @@
|
||||||
|
import {Component, Input, OnDestroy, OnInit, Optional, ViewChild} from '@angular/core';
|
||||||
|
import {coerceBooleanProperty} from '@angular/cdk/coercion';
|
||||||
|
import {
|
||||||
|
MatColumnDef,
|
||||||
|
MatSort,
|
||||||
|
MatSortHeader,
|
||||||
|
MatTable,
|
||||||
|
MatTableDataSource
|
||||||
|
} from '@angular/material';
|
||||||
|
|
||||||
|
export interface PeriodicElement {
|
||||||
|
name: string;
|
||||||
|
position: number;
|
||||||
|
weight: number;
|
||||||
|
symbol: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ELEMENT_DATA: PeriodicElement[] = [
|
||||||
|
{position: 1, name: 'Hydrogen', weight: 1.0079, symbol: 'H'},
|
||||||
|
{position: 2, name: 'Helium', weight: 4.0026, symbol: 'He'},
|
||||||
|
{position: 3, name: 'Lithium', weight: 6.941, symbol: 'Li'},
|
||||||
|
{position: 4, name: 'Beryllium', weight: 9.0122, symbol: 'Be'},
|
||||||
|
{position: 5, name: 'Boron', weight: 10.811, symbol: 'B'},
|
||||||
|
{position: 6, name: 'Carbon', weight: 12.0107, symbol: 'C'},
|
||||||
|
{position: 7, name: 'Nitrogen', weight: 14.0067, symbol: 'N'},
|
||||||
|
{position: 8, name: 'Oxygen', weight: 15.9994, symbol: 'O'},
|
||||||
|
{position: 9, name: 'Fluorine', weight: 18.9984, symbol: 'F'},
|
||||||
|
{position: 10, name: 'Neon', weight: 20.1797, symbol: 'Ne'},
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Table with a custom column component for easy column definition reuse.
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'table-simple-column-example',
|
||||||
|
styleUrls: ['table-simple-column-example.css'],
|
||||||
|
templateUrl: 'table-simple-column-example.html',
|
||||||
|
})
|
||||||
|
export class TableSimpleColumnExample implements OnInit {
|
||||||
|
displayedColumns: string[] = ['position', 'name', 'weight', 'symbol'];
|
||||||
|
dataSource = new MatTableDataSource<PeriodicElement>(ELEMENT_DATA);
|
||||||
|
getWeight = (data: PeriodicElement) => '~' + data.weight;
|
||||||
|
|
||||||
|
@ViewChild('sort') sort: MatSort;
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
this.dataSource.sort = this.sort;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Column that shows simply shows text content for the header and row
|
||||||
|
* cells. By default, the name of this column will be assumed to be both the header
|
||||||
|
* text and data property used to access the data value to show in cells. To override
|
||||||
|
* the header text, provide a label text. To override the data cell values,
|
||||||
|
* provide a dataAccessor function that provides the string to display for each row's cell.
|
||||||
|
*
|
||||||
|
* Note that this component sets itself as visually hidden since it will show up in the `mat-table`
|
||||||
|
* DOM because it is an empty element with an ng-container (nothing rendered). It should not
|
||||||
|
* interfere with screen readers.
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'simple-column',
|
||||||
|
template: `
|
||||||
|
<ng-container matColumnDef>
|
||||||
|
<th mat-header-cell *matHeaderCellDef mat-sort-header> {{label || name}} </th>
|
||||||
|
<td mat-cell *matCellDef="let data"> {{getData(data)}}</td>
|
||||||
|
</ng-container>
|
||||||
|
`,
|
||||||
|
host: {
|
||||||
|
'class': 'simple-column cdk-visually-hidden',
|
||||||
|
'[attr.ariaHidden]': 'true',
|
||||||
|
}
|
||||||
|
})
|
||||||
|
export class SimpleColumn<T> implements OnDestroy, OnInit {
|
||||||
|
/** Column name that should be used to reference this column. */
|
||||||
|
@Input()
|
||||||
|
get name(): string { return this._name; }
|
||||||
|
set name(name: string) {
|
||||||
|
this._name = name;
|
||||||
|
this.columnDef.name = name;
|
||||||
|
}
|
||||||
|
_name: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text label that should be used for the column header. If this property is not
|
||||||
|
* set, the header text will default to the column name.
|
||||||
|
*/
|
||||||
|
@Input() label: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Accessor function to retrieve the data should be provided to the cell. If this
|
||||||
|
* property is not set, the data cells will assume that the column name is the same
|
||||||
|
* as the data property the cells should display.
|
||||||
|
*/
|
||||||
|
@Input() dataAccessor: ((data: T, name: string) => string);
|
||||||
|
|
||||||
|
/** Alignment of the cell values. */
|
||||||
|
@Input() align: 'before' | 'after' = 'before';
|
||||||
|
|
||||||
|
/** Whether the column is sortable */
|
||||||
|
@Input()
|
||||||
|
get sortable(): boolean { return this._sortable; }
|
||||||
|
set sortable(sortable: boolean) {
|
||||||
|
this._sortable = coerceBooleanProperty(sortable);
|
||||||
|
}
|
||||||
|
_sortable: boolean;
|
||||||
|
|
||||||
|
@ViewChild(MatColumnDef) columnDef: MatColumnDef;
|
||||||
|
|
||||||
|
@ViewChild(MatSortHeader) sortHeader: MatSortHeader;
|
||||||
|
|
||||||
|
constructor(@Optional() public table: MatTable<any>) { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
if (this.table) {
|
||||||
|
this.table.addColumnDef(this.columnDef);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ngOnDestroy() {
|
||||||
|
if (this.table) {
|
||||||
|
this.table.removeColumnDef(this.columnDef);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getData(data: T): any {
|
||||||
|
return this.dataAccessor ? this.dataAccessor(data, this.name) : (data as any)[this.name];
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,3 @@
|
||||||
|
table {
|
||||||
|
width: 100%;
|
||||||
|
}
|
|
@ -0,0 +1,12 @@
|
||||||
|
<wrapper-table [dataSource]="dataSource" [columns]="displayedColumns"
|
||||||
|
matSort #sort="matSort">
|
||||||
|
<!-- Custom column definition to be provided to the wrapper table. -->
|
||||||
|
<ng-container matColumnDef="name">
|
||||||
|
<th mat-header-cell *matHeaderCellDef> Name </th>
|
||||||
|
<td mat-cell *matCellDef="let element"> {{element.name}} </td>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
<!-- Custom row definitions to be provided to the wrapper table. -->
|
||||||
|
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
||||||
|
<tr mat-row *matRowDef="let row; columns: displayedColumns; "></tr>
|
||||||
|
</wrapper-table>
|
88
src/assets/angular-material-examples/table-wrapped/table-wrapped-example.ts
Executable file
88
src/assets/angular-material-examples/table-wrapped/table-wrapped-example.ts
Executable file
|
@ -0,0 +1,88 @@
|
||||||
|
import {
|
||||||
|
AfterContentInit,
|
||||||
|
Component,
|
||||||
|
ContentChildren,
|
||||||
|
Input,
|
||||||
|
OnInit,
|
||||||
|
QueryList,
|
||||||
|
ViewChild
|
||||||
|
} from '@angular/core';
|
||||||
|
import {
|
||||||
|
MatColumnDef,
|
||||||
|
MatHeaderRowDef,
|
||||||
|
MatRowDef,
|
||||||
|
MatSort,
|
||||||
|
MatTable,
|
||||||
|
MatTableDataSource
|
||||||
|
} from '@angular/material';
|
||||||
|
import {DataSource} from '@angular/cdk/collections';
|
||||||
|
|
||||||
|
export interface PeriodicElement {
|
||||||
|
name: string;
|
||||||
|
position: number;
|
||||||
|
weight: number;
|
||||||
|
symbol: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ELEMENT_DATA: PeriodicElement[] = [
|
||||||
|
{position: 1, name: 'Hydrogen', weight: 1.0079, symbol: 'H'},
|
||||||
|
{position: 2, name: 'Helium', weight: 4.0026, symbol: 'He'},
|
||||||
|
{position: 3, name: 'Lithium', weight: 6.941, symbol: 'Li'},
|
||||||
|
{position: 4, name: 'Beryllium', weight: 9.0122, symbol: 'Be'},
|
||||||
|
{position: 5, name: 'Boron', weight: 10.811, symbol: 'B'},
|
||||||
|
{position: 6, name: 'Carbon', weight: 12.0107, symbol: 'C'},
|
||||||
|
{position: 7, name: 'Nitrogen', weight: 14.0067, symbol: 'N'},
|
||||||
|
{position: 8, name: 'Oxygen', weight: 15.9994, symbol: 'O'},
|
||||||
|
{position: 9, name: 'Fluorine', weight: 18.9984, symbol: 'F'},
|
||||||
|
{position: 10, name: 'Neon', weight: 20.1797, symbol: 'Ne'},
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title Table example that shows how to wrap a table component for definition and behavior reuse.
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'table-wrapped-example',
|
||||||
|
styleUrls: ['table-wrapped-example.css'],
|
||||||
|
templateUrl: 'table-wrapped-example.html',
|
||||||
|
})
|
||||||
|
export class TableWrappedExample implements OnInit {
|
||||||
|
displayedColumns: string[] = ['position', 'name', 'weight', 'symbol'];
|
||||||
|
dataSource = new MatTableDataSource<PeriodicElement>(ELEMENT_DATA);
|
||||||
|
|
||||||
|
@ViewChild('sort') sort: MatSort;
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
this.dataSource.sort = this.sort;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Table component that accepts column and row definitions in its content to be registered to the
|
||||||
|
* table.
|
||||||
|
*/
|
||||||
|
@Component({
|
||||||
|
selector: 'wrapper-table',
|
||||||
|
templateUrl: 'wrapper-table.html',
|
||||||
|
styles: [`
|
||||||
|
table {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
`]
|
||||||
|
})
|
||||||
|
export class WrapperTable<T> implements AfterContentInit {
|
||||||
|
@ContentChildren(MatHeaderRowDef) headerRowDefs: QueryList<MatHeaderRowDef>;
|
||||||
|
@ContentChildren(MatRowDef) rowDefs: QueryList<MatRowDef<T>>;
|
||||||
|
@ContentChildren(MatColumnDef) columnDefs: QueryList<MatColumnDef>;
|
||||||
|
|
||||||
|
@ViewChild(MatTable) table: MatTable<T>;
|
||||||
|
|
||||||
|
@Input() columns: string[];
|
||||||
|
|
||||||
|
@Input() dataSource: DataSource<T>;
|
||||||
|
|
||||||
|
ngAfterContentInit() {
|
||||||
|
this.columnDefs.forEach(columnDef => this.table.addColumnDef(columnDef));
|
||||||
|
this.rowDefs.forEach(rowDef => this.table.addRowDef(rowDef));
|
||||||
|
this.headerRowDefs.forEach(headerRowDef => this.table.addHeaderRowDef(headerRowDef));
|
||||||
|
}
|
||||||
|
}
|
21
src/assets/angular-material-examples/table-wrapped/wrapper-table.html
Executable file
21
src/assets/angular-material-examples/table-wrapped/wrapper-table.html
Executable file
|
@ -0,0 +1,21 @@
|
||||||
|
<table mat-table [dataSource]="dataSource" class="mat-elevation-z8">
|
||||||
|
<ng-content></ng-content>
|
||||||
|
|
||||||
|
<!-- Position Column -->
|
||||||
|
<ng-container matColumnDef="position">
|
||||||
|
<th mat-header-cell *matHeaderCellDef mat-sort-header> No. </th>
|
||||||
|
<td mat-cell *matCellDef="let element"> {{element.position}} </td>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
<!-- Weight Column -->
|
||||||
|
<ng-container matColumnDef="weight">
|
||||||
|
<th mat-header-cell *matHeaderCellDef mat-sort-header> Weight </th>
|
||||||
|
<td mat-cell *matCellDef="let element"> {{element.weight}} </td>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
<!-- Color Column -->
|
||||||
|
<ng-container matColumnDef="symbol">
|
||||||
|
<th mat-header-cell *matHeaderCellDef> Symbol </th>
|
||||||
|
<td mat-cell *matCellDef="let element"> {{element.symbol}} </td>
|
||||||
|
</ng-container>
|
||||||
|
</table>
|
|
@ -8,22 +8,22 @@ import {Component, ElementRef, OnDestroy, OnInit, ViewChild} from '@angular/core
|
||||||
styleUrls: ['./text-field-autofill-monitor-example.css'],
|
styleUrls: ['./text-field-autofill-monitor-example.css'],
|
||||||
})
|
})
|
||||||
export class TextFieldAutofillMonitorExample implements OnDestroy, OnInit {
|
export class TextFieldAutofillMonitorExample implements OnDestroy, OnInit {
|
||||||
@ViewChild('first', {read: ElementRef}) firstName: ElementRef;
|
@ViewChild('first', {read: ElementRef}) firstName: ElementRef<HTMLElement>;
|
||||||
@ViewChild('last', {read: ElementRef}) lastName: ElementRef;
|
@ViewChild('last', {read: ElementRef}) lastName: ElementRef<HTMLElement>;
|
||||||
firstNameAutofilled: boolean;
|
firstNameAutofilled: boolean;
|
||||||
lastNameAutofilled: boolean;
|
lastNameAutofilled: boolean;
|
||||||
|
|
||||||
constructor(private autofill: AutofillMonitor) {}
|
constructor(private autofill: AutofillMonitor) {}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.autofill.monitor(this.firstName.nativeElement)
|
this.autofill.monitor(this.firstName)
|
||||||
.subscribe(e => this.firstNameAutofilled = e.isAutofilled);
|
.subscribe(e => this.firstNameAutofilled = e.isAutofilled);
|
||||||
this.autofill.monitor(this.lastName.nativeElement)
|
this.autofill.monitor(this.lastName)
|
||||||
.subscribe(e => this.lastNameAutofilled = e.isAutofilled);
|
.subscribe(e => this.lastNameAutofilled = e.isAutofilled);
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
this.autofill.stopMonitoring(this.firstName.nativeElement);
|
this.autofill.stopMonitoring(this.firstName);
|
||||||
this.autofill.stopMonitoring(this.lastName.nativeElement);
|
this.autofill.stopMonitoring(this.lastName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
.example-button {
|
.example-button {
|
||||||
display: block;
|
display: block;
|
||||||
width: 48px;
|
|
||||||
margin: 80px auto 400px;
|
margin: 80px auto 400px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<button mat-icon-button disabled></button>
|
<button mat-icon-button disabled></button>
|
||||||
<mat-checkbox class="checklist-leaf-node"
|
<mat-checkbox class="checklist-leaf-node"
|
||||||
[checked]="checklistSelection.isSelected(node)"
|
[checked]="checklistSelection.isSelected(node)"
|
||||||
(change)="checklistSelection.toggle(node);">{{node.item}}</mat-checkbox>
|
(change)="todoLeafItemSelectionToggle(node)">{{node.item}}</mat-checkbox>
|
||||||
</mat-tree-node>
|
</mat-tree-node>
|
||||||
|
|
||||||
<mat-tree-node *matTreeNodeDef="let node; when: hasNoContent" matTreeNodePadding>
|
<mat-tree-node *matTreeNodeDef="let node; when: hasNoContent" matTreeNodePadding>
|
||||||
|
|
|
@ -68,7 +68,7 @@ export class ChecklistDatabase {
|
||||||
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
||||||
* The return value is the list of `TodoItemNode`.
|
* The return value is the list of `TodoItemNode`.
|
||||||
*/
|
*/
|
||||||
buildFileTree(obj: object, level: number): TodoItemNode[] {
|
buildFileTree(obj: {[key: string]: any}, level: number): TodoItemNode[] {
|
||||||
return Object.keys(obj).reduce<TodoItemNode[]>((accumulator, key) => {
|
return Object.keys(obj).reduce<TodoItemNode[]>((accumulator, key) => {
|
||||||
const value = obj[key];
|
const value = obj[key];
|
||||||
const node = new TodoItemNode();
|
const node = new TodoItemNode();
|
||||||
|
@ -168,10 +168,13 @@ export class TreeChecklistExample {
|
||||||
return flatNode;
|
return flatNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether all the descendants of the node are selected */
|
/** Whether all the descendants of the node are selected. */
|
||||||
descendantsAllSelected(node: TodoItemFlatNode): boolean {
|
descendantsAllSelected(node: TodoItemFlatNode): boolean {
|
||||||
const descendants = this.treeControl.getDescendants(node);
|
const descendants = this.treeControl.getDescendants(node);
|
||||||
return descendants.every(child => this.checklistSelection.isSelected(child));
|
const descAllSelected = descendants.every(child =>
|
||||||
|
this.checklistSelection.isSelected(child)
|
||||||
|
);
|
||||||
|
return descAllSelected;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Whether part of the descendants are selected */
|
/** Whether part of the descendants are selected */
|
||||||
|
@ -188,6 +191,61 @@ export class TreeChecklistExample {
|
||||||
this.checklistSelection.isSelected(node)
|
this.checklistSelection.isSelected(node)
|
||||||
? this.checklistSelection.select(...descendants)
|
? this.checklistSelection.select(...descendants)
|
||||||
: this.checklistSelection.deselect(...descendants);
|
: this.checklistSelection.deselect(...descendants);
|
||||||
|
|
||||||
|
// Force update for the parent
|
||||||
|
descendants.every(child =>
|
||||||
|
this.checklistSelection.isSelected(child)
|
||||||
|
);
|
||||||
|
this.checkAllParentsSelection(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Toggle a leaf to-do item selection. Check all the parents to see if they changed */
|
||||||
|
todoLeafItemSelectionToggle(node: TodoItemFlatNode): void {
|
||||||
|
this.checklistSelection.toggle(node);
|
||||||
|
this.checkAllParentsSelection(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Checks all the parents when a leaf node is selected/unselected */
|
||||||
|
checkAllParentsSelection(node: TodoItemFlatNode): void {
|
||||||
|
let parent: TodoItemFlatNode | null = this.getParentNode(node);
|
||||||
|
while (parent) {
|
||||||
|
this.checkRootNodeSelection(parent);
|
||||||
|
parent = this.getParentNode(parent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Check root node checked state and change it accordingly */
|
||||||
|
checkRootNodeSelection(node: TodoItemFlatNode): void {
|
||||||
|
const nodeSelected = this.checklistSelection.isSelected(node);
|
||||||
|
const descendants = this.treeControl.getDescendants(node);
|
||||||
|
const descAllSelected = descendants.every(child =>
|
||||||
|
this.checklistSelection.isSelected(child)
|
||||||
|
);
|
||||||
|
if (nodeSelected && !descAllSelected) {
|
||||||
|
this.checklistSelection.deselect(node);
|
||||||
|
} else if (!nodeSelected && descAllSelected) {
|
||||||
|
this.checklistSelection.select(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get the parent node of a node */
|
||||||
|
getParentNode(node: TodoItemFlatNode): TodoItemFlatNode | null {
|
||||||
|
const currentLevel = this.getLevel(node);
|
||||||
|
|
||||||
|
if (currentLevel < 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const startIndex = this.treeControl.dataNodes.indexOf(node) - 1;
|
||||||
|
|
||||||
|
for (let i = startIndex; i >= 0; i--) {
|
||||||
|
const currentNode = this.treeControl.dataNodes[i];
|
||||||
|
|
||||||
|
if (this.getLevel(currentNode) < currentLevel) {
|
||||||
|
return currentNode;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Select the category so we can insert the new item. */
|
/** Select the category so we can insert the new item. */
|
||||||
|
|
|
@ -59,7 +59,7 @@ export class DynamicDataSource {
|
||||||
private database: DynamicDatabase) {}
|
private database: DynamicDatabase) {}
|
||||||
|
|
||||||
connect(collectionViewer: CollectionViewer): Observable<DynamicFlatNode[]> {
|
connect(collectionViewer: CollectionViewer): Observable<DynamicFlatNode[]> {
|
||||||
this.treeControl.expansionModel.onChange!.subscribe(change => {
|
this.treeControl.expansionModel.onChange.subscribe(change => {
|
||||||
if ((change as SelectionChange<DynamicFlatNode>).added ||
|
if ((change as SelectionChange<DynamicFlatNode>).added ||
|
||||||
(change as SelectionChange<DynamicFlatNode>).removed) {
|
(change as SelectionChange<DynamicFlatNode>).removed) {
|
||||||
this.handleTreeControl(change as SelectionChange<DynamicFlatNode>);
|
this.handleTreeControl(change as SelectionChange<DynamicFlatNode>);
|
||||||
|
|
|
@ -91,7 +91,7 @@ export class FileDatabase {
|
||||||
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
||||||
* The return value is the list of `FileNode`.
|
* The return value is the list of `FileNode`.
|
||||||
*/
|
*/
|
||||||
buildFileTree(obj: object, level: number): FileNode[] {
|
buildFileTree(obj: {[key: string]: any}, level: number): FileNode[] {
|
||||||
return Object.keys(obj).reduce<FileNode[]>((accumulator, key) => {
|
return Object.keys(obj).reduce<FileNode[]>((accumulator, key) => {
|
||||||
const value = obj[key];
|
const value = obj[key];
|
||||||
const node = new FileNode();
|
const node = new FileNode();
|
||||||
|
|
|
@ -84,7 +84,7 @@ export class FileDatabase {
|
||||||
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
* Build the file structure tree. The `value` is the Json object, or a sub-tree of a Json object.
|
||||||
* The return value is the list of `FileNode`.
|
* The return value is the list of `FileNode`.
|
||||||
*/
|
*/
|
||||||
buildFileTree(obj: object, level: number): FileNode[] {
|
buildFileTree(obj: {[key: string]: any}, level: number): FileNode[] {
|
||||||
return Object.keys(obj).reduce<FileNode[]>((accumulator, key) => {
|
return Object.keys(obj).reduce<FileNode[]>((accumulator, key) => {
|
||||||
const value = obj[key];
|
const value = obj[key];
|
||||||
const node = new FileNode();
|
const node = new FileNode();
|
||||||
|
|
Loading…
Reference in New Issue
Block a user