Bootstrap

Angular 任务列表页

新建一个任务Module

$ ng g m task

功能:项目列表显示,增加,修改,删除项目。邀请其它人员加入项目。

单一性原则:希望搭建多个组件,每个组件负责自己的功能。

一、task相关组件 

$ ng g c task/task-home

$ ng g c task/task-list

$ ng g c task/task-item

$ ng g c task/task-header

$ ng g c task/new-task

$ ng g c task/copy-task

$ ng g c task/new-task-list

需要把NewTaskComponent和CopyTaskComponent放在entryComponents中。

@NgModule({
  declarations: [
    TaskHomeComponent,
    TaskListComponent,
    TaskItemComponent,
    TaskHeaderComponent,
    NewTaskComponent,
    CopyTaskComponent
  ],
  imports: [SharedModule, TaskRoutingModule],
  entryComponents:[NewTaskComponent,CopyTaskComponent]
})

 

三、TaskHome

<div class="task-list">
  <app-task-list *ngFor="let list of lists" class="list-container">
    <app-task-header [header]="list.name"> </app-task-header>
    <app-task-item *ngFor="let task of list.tasks"> </app-task-item>
  </app-task-list>
</div>

<button
  class="ab-buttonmad-fab fab-button"
  mat-fab
  type="button"
  (click)="openNewProjectDialog()"
>
  <mat-icon>add</mat-icon>
</button>

TaskHome中处理新建任务,修改任务,移动任务,删除任务列表:

<app-task-header [header]="list.name"
(newTask)="lauchNewTaskDialog()"
(moveAll)="lauchCopyTaskDialog()"
(deleteList)="lauchConfirmDialog()"
> </app-task-header>

  lauchNewTaskDialog() {
    // this.dialog.open(NewTaskComponent);
    const dialogRef = this.dialog.open(NewTaskComponent,{data:{title:'新建任务'}});
  }
  lauchCopyTaskDialog(){
    const dialogRef = this.dialog.open(CopyTaskComponent,{data:{lists:this.lists}});
  }

  launchUpdateTaskDialog(task){
    const dialogRef = this.dialog.open(NewTaskComponent,{data:{title:'修改任务',task:task}});
  }

  lauchConfirmDialog(){
    const dialogRef = this.dialog.open(ConfirmDialogComponent,{data:{title:'删除任务列表',content:'您确定要删除该任务列表吗?'}});
  }

 

 

 

四、TaskList

app-task-list组件里面是可以放内容的。

<mat-list>
  <ng-content>
  </ng-content>
</mat-list>

list里面内容不去管。

五、Header

<div mat-subheader class="fill">
  <div class="header-container">
    <h3>{{header}}</h3>
  </div>
  <div>
    <div class="fill">
      <button mat-button>
        <mat-icon>add_circle_outine</mat-icon>
        <span>新任务</span>
      </button>
    </div>
  </div>

  <div>
    <button mat-icon-button [matMenuTriggerFor]="menu">
      <mat-icon>keyboard_arrow_down</mat-icon>
    </button>
  </div>
</div>

<mat-menu #menu="matMenu">
  <button mat-menu-item>
    <mat-icon>
      mode_edit
    </mat-icon>
    <span>
      修改列表名称
    </span>
  </button>
  <button mat-menu-item>
    <mat-icon svgIcon="move">
    </mat-icon>
    <span>
      移动列表所有内容
    </span>
  </button>
  <button mat-menu-item>
    <mat-icon>
      delete_forever
    </mat-icon>
    <span>
      删除列表
    </span>
  </button>
</mat-menu>
View Code

六、ListItem

1、通过优先级不同让任务有一个不同颜色的边框。

  .priority-normal {
    border-left: 3px solid #A6A6A6;
  }
  .priority-important {
    border-left: 3px solid #FFAF38;
  }
  .priority-emergency {
    border-left: 3px solid red;
  }
<mat-list-item class="container" 
[ngClass]="{
  'priority-normal':item.priority===3,
  'priority-important':item.priority===2,
  'priority-emergency':item.priority===1
}">

2、list超出后显示...,鼠标移上去后给出提示。

    <div class="content" mat-line [ngClass]="{'completed':item.completed}">
      {{item.desc}}
    </div>

改为:

    <div class="content" mat-line [ngClass]="{'completed':item.completed}">
      <span [matTooltip]="item.desc">{{item.desc}}</span>
    </div>

全部布局

<mat-list-item class="container" [ngClass]="{
  'priority-normal':item.priority===3,
  'priority-important':item.priority===2,
  'priority-emergency':item.priority===1
}">
  <mat-checkbox [checked]="item.completed" class="status"> </mat-checkbox>
  <div class="content" mat-line [ngClass]="{'completed':item.completed}">
    <span [matTooltip]="item.desc">{{item.desc}}</span>
  </div>
  <div class="bottom-bar" mat-line>
    <span class="due-date" *ngIf="item.dueDate">
      {{item.dueDate | date:"yy-MM-dd"}}
    </span>
    <mat-icon *ngIf="item.reminder">
      alarm
    </mat-icon>
  </div>
  <mat-icon [svgIcon]="avatar" mat-list-avatar class="avatar">

  </mat-icon>
</mat-list-item>
View Code
mat-icon.avatar {
  overflow: hidden;
  width: 64px;
  height: 64px;
  border-radius: 50%;
  margin: 12px;
  order: 3;
}

.completed {
  opacity: 0.64;
  color: #d9d9d9;
  text-decoration: line-through;
}

.priority-normal {
  border-left: 3px solid #a6a6a6;
}

.priority-important {
  border-left: 3px solid #ffaf38;
}

.priority-emergency {
  border-left: 3px solid red;
}

.checkbox-section {
  border: 0 solid #a6a6a6;
}

.duedate {
  background-color: #ff4f3e;
  color: #fff;
}

.alarm {
  font-size: 18px;
}

.bottom-bar {
  margin-top: 3px;
  margin-bottom: 2px;
  font-size: 10px;
  width: 100%;
  order: 1;
}

.status {
  order: -1;
}

.content {
  order: 1;
  width: 100%;
  padding: 5px;
}

.container {
  width: 100%;
  border-radius: 3px;
  box-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);
}

.drag-start {
  opacity: 0.5;
  border: #ff525b dashed 2px;
}

:host {
  width: 100%;
}
View Code

七、NewTask

1、任务优先级:

<mat-radio-group>
      <mat-radio-button *ngFor="let priority of priorities" [value]="priority.value">
        {{priority.label}}
      </mat-radio-button>
</mat-radio-group>
export class NewTaskComponent implements OnInit {
  priorities = [
    {
      label: '紧急',
      value: 1
    },
    {
      label: '重要',
      value: 2
    },
    {
      label: '普通',
      value: 3
    }
  ];
  constructor() { }

  ngOnInit() {
  }
}

2、任务截止日期

    <mat-form-field class="full-width">
      <input type="text" [matDatepicker]="dueDatepicker" matInput [matDatepicker]="" placeholder="任务截止日期">
      <mat-datepicker-toggle matSuffix [for]="dueDatepicker"></mat-datepicker-toggle>
    </mat-form-field>
    <mat-datepicker #dueDatepicker></mat-datepicker>

 3、调起NewTask组件

在header中把新建任务的事件发射出来

<button mat-button (click)="onNewTaskClick()">
        <mat-icon>add_circle_outine</mat-icon>
        <span>新任务</span>
</button>

  @Output() newTask= new EventEmitter<void>() ;
  onNewTaskClick(){
    this.newTask.emit();
  }

在TaskHome中监听

<app-task-header [header]="list.name" (newTask)="lauchNewTaskDialog()"> </app-task-header>

  lauchNewTaskDialog() {
    this.dialog.open(NewTaskComponent);
  }

4、修改任务

taskHome中去监听taskItem的click事件

    <app-task-item *ngFor="let task of list.tasks" [item]="task" (taskClick)="launchUpdateTaskDialog(task)"> </app-task-item>

  launchUpdateTaskDialog(task){
    const dialogRef = this.dialog.open(NewTaskComponent,{data:{title:'修改任务',task:task}});
  }

修改NewTask组件,首先需要有一个title

title:'';
constructor(@Inject(MAT_DIALOG_DATA) private data: any) {
    this.title = this.data.title;
    console.log(JSON.stringify(this.data.task));
}

  <h2 md-dialog-title>{{title}}</h2>

在新建任何和修改任务时候都会传入title,新建任务时候不会传入task

  lauchNewTaskDialog() {
    // this.dialog.open(NewTaskComponent);
    const dialogRef = this.dialog.open(NewTaskComponent,{data:{title:'新建任务'}});
  }

  launchUpdateTaskDialog(task){
    const dialogRef = this.dialog.open(NewTaskComponent,{data:{title:'修改任务',task:task}});
  }

问题:点击checkbox的时候也会弹出修改任务对话框。

解决:

  <mat-checkbox [checked]="item.completed" class="status" (click)="onCheckBoxClick($event)"> </mat-checkbox>

  onCheckBoxClick(event: Event): void {
    event.stopPropagation();
  }

 

八、移动内容 

<mat-select placeholder="请选择目标列表">
      <mat-option *ngFor="let list of lists" [value]="list">{{list.name}}</mat-option>
</mat-select>

export class CopyTaskComponent implements OnInit {
  lists: any[];
  constructor(@Inject(MAT_DIALOG_DATA) private data: any,
  public dialogRef: MatDialogRef<CopyTaskComponent>) { }

  ngOnInit() {
    this.lists = this.data.lists;
  }

}

九、新建,修改任务列表 

 新建任务列表和改名字用的同一个Component

<form>
  <h2 md-dialog-title>新建项目列表</h2>
    <div mat-dialog-content>
      <mat-form-field class="full-width">
        <input type="text" matInput placeholder="列表名称">
      </mat-form-field>
    </div>
    <div mat-dialog-actions>
      <button type="button" mat-raised-button color="primary" (click)="onSave()">保存</button>
      <button type="button" mat-button mat-dialog-close>关闭</button>
    </div>
</form>
import { Component, OnInit, Inject } from '@angular/core';
import { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material';

@Component({
  selector: 'app-new-task-list',
  templateUrl: './new-task-list.component.html',
  styleUrls: ['./new-task-list.component.scss']
})
export class NewTaskListComponent implements OnInit {
  title='';

  constructor(@Inject(MAT_DIALOG_DATA) private data,
  private dialogRef: MatDialogRef<NewTaskListComponent>) { }

  ngOnInit() {
  }

  onSave(){
    this.dialogRef.close(this.title);
  }
}
View Code

在TaskHome组件中去处理事件

  launchEditListDialog() {
    const dialogRef = this.dialog.open(NewTaskListComponent, {
      data: { title: "更改列表名称" }
    });
    dialogRef.afterClosed().subscribe(result => console.log(result));
  }
  launchNewListDialog() {
    const dialogRef = this.dialog.open(NewTaskListComponent, {
      data: { title: "新建列表名称" }
    });
    dialogRef.afterClosed().subscribe(result => console.log(result));
  }

 

十、数据结构的演变

1、

lists=[
    {
      "id":1,
      "name": "待办",
      "tasks" :[
        {
          id:1,
          desc: '任务一: 去星巴克买咖啡',
          owner:{
            id:1,
            name:'张三',
            avatar:'avatars:svg-11'
          },
          dueDate: new Date()
        },
        {
          id:2,
          desc: '任务一: 完成老板布置的PPT作业',
          owner:{
            id:2,
            name:'李四',
            avatar:'avatars:svg-12'
          },
          dueDate: new Date()
        }
      ]
    },
    {
      "id":2,
      "name": "进行中",
      "tasks" :[
        {
          id:1,
          desc: '任务三: 项目代码评审',
          owner:{
            id:1,
            name:'王五',
            avatar:'avatars:svg-13'
          },
          dueDate: new Date()
        },
        {
          id:2,
          desc: '任务一: 制定项目计划',
          owner:{
            id:2,
            name:'李四',
            avatar:'avatars:svg-12'
          },
          dueDate: new Date()
        }
      ]
    }
  ]

 2、每一个任务Item都加一个完成状态completed

lists = [
    {
      id: 1,
      name: "待办",
      tasks: [
        {
          id: 1,
          desc: "任务一: 去星巴克买咖啡",
          completed: true,
          owner: {
            id: 1,
            name: "张三",
            avatar: "avatars:svg-11"
          },
          dueDate: new Date()
        },
        {
          id: 2,
          desc: "任务一: 完成老板布置的PPT作业",
          completed: false,
          owner: {
            id: 2,
            name: "李四",
            avatar: "avatars:svg-12"
          },
          dueDate: new Date()
        }
      ]
    },
    {
      id: 2,
      name: "进行中",
      tasks: [
        {
          id: 1,
          desc: "任务三: 项目代码评审",
          completed: false,
          owner: {
            id: 1,
            name: "王五",
            avatar: "avatars:svg-13"
          },
          dueDate: new Date()
        },
        {
          id: 2,
          desc: "任务一: 制定项目计划",
          completed: false,
          owner: {
            id: 2,
            name: "李四",
            avatar: "avatars:svg-12"
          },
          dueDate: new Date()
        }
      ]
    }
  ];
View Code

3、有些任务加reminder

import { Component, OnInit } from "@angular/core";

@Component({
  selector: "app-task-home",
  templateUrl: "./task-home.component.html",
  styleUrls: ["./task-home.component.scss"]
})
export class TaskHomeComponent implements OnInit {
  constructor() {}

  ngOnInit() {}

  lists = [
    {
      id: 1,
      name: "待办",
      tasks: [
        {
          id: 1,
          desc: "任务一: 去星巴克买咖啡",
          completed: true,
          owner: {
            id: 1,
            name: "张三",
            avatar: "avatars:svg-11"
          },
          dueDate: new Date(),
          reminder: new Date()
        },
        {
          id: 2,
          desc: "任务一: 完成老板布置的PPT作业",
          completed: false,
          owner: {
            id: 2,
            name: "李四",
            avatar: "avatars:svg-12"
          },
          dueDate: new Date()
        }
      ]
    },
    {
      id: 2,
      name: "进行中",
      tasks: [
        {
          id: 1,
          desc: "任务三: 项目代码评审",
          completed: false,
          owner: {
            id: 1,
            name: "王五",
            avatar: "avatars:svg-13"
          },
          dueDate: new Date()
        },
        {
          id: 2,
          desc: "任务一: 制定项目计划",
          completed: false,
          owner: {
            id: 2,
            name: "李四",
            avatar: "avatars:svg-12"
          },
          dueDate: new Date()
        }
      ]
    }
  ];
}
View Code

4、为每个任务添加优先级priority

lists = [
    {
      id: 1,
      name: "待办",
      tasks: [
        {
          id: 1,
          desc: "任务一: 去星巴克买咖啡",
          completed: true,
          priority: 3,
          owner: {
            id: 1,
            name: "张三",
            avatar: "avatars:svg-11"
          },
          dueDate: new Date(),
          reminder: new Date()
        },
        {
          id: 2,
          desc: "任务一: 完成老板布置的PPT作业",
          completed: false,
          priority: 2,
          owner: {
            id: 2,
            name: "李四",
            avatar: "avatars:svg-12"
          },
          dueDate: new Date()
        }
      ]
    },
    {
      id: 2,
      name: "进行中",
      tasks: [
        {
          id: 1,
          desc: "任务三: 项目代码评审",
          completed: false,
          priority: 1,
          owner: {
            id: 1,
            name: "王五",
            avatar: "avatars:svg-13"
          },
          dueDate: new Date()
        },
        {
          id: 2,
          desc: "任务一: 制定项目计划",
          completed: false,
          priority: 2,
          owner: {
            id: 2,
            name: "李四",
            avatar: "avatars:svg-12"
          },
          dueDate: new Date()
        }
      ]
    }
  ];
View Code

 

;