angular - 如何在运行时更改 Angular Material Datepicker 格式

标签 angular datepicker angular-material momentjs

我正在使用 Material 设计开发一个 Angular 应用程序,我正在使用 Moment.js 来解析和格式化日期。

在我的一个页面中,我有一个 Material's Datepicker .我已按照 Material 的指导方针使 Datepicker 与 moment 对象一起工作 - 而不是在 native Date 对象上工作。

我还设法以某种格式在日期选择器中显示日期。但是这种格式硬编码在组件的提供者中。 如何根据用户偏好在运行时更改格式?

这是我到目前为止所拥有的:

我已经安装了 npm 的包:

$ npm i moment -S
$ npm i @angular/material-moment-adapter -S
// app.module.ts
import { MatMomentDateModule } from '@angular/material-moment-adapter'
...
@NgModule({
  imports: [
    MatMomentDateModule,
...
// demo.component.ts
import * as moment from 'moment';
import { MAT_DATE_FORMATS } from '@angular/material/core';
const MY_FORMATS = {
  parse: {
    dateInput: 'LL',
  },
  display: {
    dateInput: 'LL',
    monthYearLabel: 'MMM YYYY',
    dateA11yLabel: 'LL',
    monthYearA11yLabel: 'MMMM YYYY',
  },
};
@Component({
  selector: 'app-demo',
  templateUrl: './demo.component.html',
  styleUrls: ['./demo.component.scss'],
  providers: [
    { provide: MAT_DATE_FORMATS, useValue: MY_FORMATS },
  ],
})
export class DemoComponent {
   public dateVal = moment();
}
// demo.component.html
<mat-form-field>
  <input matInput [matDatepicker]="myDatePicker" [(ngModel)]="dateVal">
  <mat-datepicker-toggle matSuffix [for]="myDatePicker"></mat-datepicker-toggle>
  <mat-datepicker #myDatePicker></mat-datepicker>
</mat-form-field>

如您所见,日期格式是在组件的提供程序中定义的。如何在运行时更改它?

可以在这里找到一个工作示例:
https://stackblitz.com/angular/mgaargebgpd?file=app%2Fdatepicker-formats-example.ts

最佳答案

好的,所以我终于找到了一种在运行时更改 mat-date-picker 格式的方法(文档根本没有帮助)。

第 1 步 - 创建将提供格式设置的服务。

您可能已经有了这样的服务,如果没有,您应该创建一个,这样您就可以在一个地方控制日期格式。

// date-time.service
import { Injectable } from '@angular/core';
import * as moment from 'moment';

@Injectable({ providedIn: 'root' })
export class DateTimeService
{
  public getFormat(): string
  {
    return "DD-MM-YYYY"; // add you own logic here
  }
  public getLocale(): string
  {
    return "he-IL"; // add you own logic here
  }  
}

第 2 步 - 创建一个 CustomDateAdapter,它将负责在运行时解析日期

// customDateAdapter.ts
import { Injectable } from '@angular/core';
import { MomentDateAdapter } from '@angular/material-moment-adapter';
import * as moment from 'moment';
import { DateTimeService } from './date-time.service';

@Injectable()
export class CustomDateAdapter extends MomentDateAdapter
{
  constructor(private _dateTimeService: DateTimeService)
  {
    super('en-US'); // set default locale
  }

  public format(date: moment.Moment, displayFormat: string): string
  {
    const locale = this._dateTimeService.getLocale();
    const format = this._dateTimeService.getFormat();

    return date.locale(locale).format(format);
  }
}

请注意:“CustomDateAdapter”是一个常规类,而不是一个组件。虽然我们正在向这个类注入(inject)一个服务。为此,我们需要将 @Injectable() 装饰器添加到“CustomDateAdapter”,并在 app.module.ts 中进行一些细微的更改。

第 3 步 - 修改 app.module.ts 以支持自定义格式并允许对 CustomDateAdapter 进行依赖注入(inject)。

// app.module.ts
import { DateAdapter, MatNativeDateModule } from '@angular/material';
import { MatMomentDateModule } from '@angular/material-moment-adapter'
import { CustomDateAdapter } from './<some-path>/customDateAdapter';

@NgModule({
  imports:
  [
    MatNativeDateModule,
    MatMomentDateModule
  ],
  providers:
  [
    CustomDateAdapter, // so we could inject services to 'CustomDateAdapter'
    { provide: DateAdapter, useClass: CustomDateAdapter }, // Parse MatDatePicker Format
  ]
})
export class AppModule { /* ... */ }

附言
请注意问题中的代码(来自“demo.component.ts”)不再相关。

Demo at Stackblitz

关于angular - 如何在运行时更改 Angular Material Datepicker 格式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51634726/

相关文章:

html - 选择依赖于另一个选择 - Material Angular

python - 如何在 html 中单击 zebra datepicker 中的日期?

jquery - 如何在年份的日期选择器下拉列表中显示从 1920 年到当前年份的正确年份

javascript - 动态组件列表上带有 ng-template 的 cdkDropList 不起作用

css - Angular:如何使用查询参数过滤数据表

javascript - 如何根据对象的属性加载动态组件?

javascript - 使用 ion-item 在 NTH 元素后添加 Item

angular - 将 angular.json 配置拆分为多个文件

android - 如何在 DatePickerDialog Android 中禁用今天日期之前的日期

css - 正确对齐的 div 内的 Angular Material 卡必须具有相同的高度