added CRUD functions

This commit is contained in:
imelilabourne 2020-09-15 11:06:17 +08:00
parent 4c9e822238
commit a66e2fa3f8
15 changed files with 2776 additions and 2235 deletions

View file

@ -0,0 +1,84 @@
.main{
width: 100%;
background-color: #9cb1a033;
}
.completed{
text-decoration-line: line-through;
}
.addInput{
max-width: 500px;
line-height: 2em;
margin: 10px;
}
ul > li{
display: grid;
grid-template-columns: .5fr 2.5fr .5fr .5fr;
}
li > *{
display: flex;
flex-wrap: wrap;
}
.listBtn{
max-width: 80px;
}
.main{
max-width: 50%;
box-shadow: 8px 4px#9cb1a0;
border-radius: 20px;
padding: 0;
}
.header, .inputDiv{
display: flex;
justify-content: center;
}
.nav{
background-color:#bfdbc546;
display: block;
/* color: white; */
line-height: 3em;
font-size: 16px;
padding-left: 1.5em;
margin-top: 1em;
}
.main-container{
margin: 4%;
}
.main{
margin-top: 5em;
}
li{
background-color: #b9ecc24d;
margin: 4px;
display: flex;
align-items: center;
padding-left: 10px;
font-size: 2rem;
line-height: 1.5em;
}
.info{
display: flex;
justify-content: flex-end;
}
input[type="checkbox"]{
margin: 0 10px;
}
.dateClass{
font-size: 1.5em;
float: right;
padding-right: 20px;
color: rgb(42, 59, 59);
}

View file

@ -0,0 +1,45 @@
<div class="container main">
<div class="nav">Hello! <span class="dateClass">{{today | date: 'short'}}</span></div>
<div class="main-container">
<div class="header">
<h1>TODO LIST</h1>
</div>
<div class="info"><input type="checkbox" (change)="selectAll()">Select All</div>
<div class="inputDiv">
<input type="text" class="form-control addInput" placeholder="Things you want to do today?" [(ngModel)]="taskTitle" (keyup.enter) = "addTask()">
</div>
<div class="content">
<ul>
<li *ngFor= "let task of tasks" @fade>
<input type="checkbox" [(ngModel)]="task.completed">
<!-- input to focus -->
<div *ngIf="!task.editing; else editingTask" (dblclick) = "toggleEdit(task)" [ngClass]="{completed: task.completed}">{{task.title}}</div>
<ng-template #editingTask>
<input type="text" [(ngModel)] = "task.title" (blur)="doneEditing(task)" (keyup.enter)="editTask(task)" (keyup.esc)="cancelEditing(task)" autofocus>
</ng-template>
<!-- buttons switching -->
<div *ngIf="!task.editing; else doneBtn" ><button class="btn btn-success listBtn"(click)="toggleEdit(task)" ><i class="fa fa-edit"> Edit</i></button></div>
<ng-template #doneBtn>
<div>
<button class="btn btn-success listBtn"(click)="editTask(task)" ><i class="fa fa-edit"> Done</i></button>
</div>
</ng-template>
<div>
<button class="btn btn-danger listBtn" (click)="deleteTask(task) && doneEditing(task)"><i class="fa fa-trash"> Del</i></button>
</div>
</li>
</ul>
<div class="info">
<div>{{ remaining()}} uncompleted tasks</div>
</div>
<div *ngIf="atleastOneCompleted()">
<button class="btn btn-warning btn-block" (click) ="clearCompleted() && deleteCompleted(task)">Clear Completed</button>
<div class="alert alert-warning">You've selected an item</div>
</div>
</div>
</div>
</div>

View file

@ -0,0 +1,127 @@
import { animate, style, transition, trigger } from '@angular/animations';
import { Component } from '@angular/core';
import { Tasks } from 'src/app/Todo-List-App/models/todo-interface';
import { TodoService } from '../../services/todo.service';
@Component({
selector: '<todo-list></todo-list>',
templateUrl: 'todo-list.component.html',
styleUrls: ['todo-list.component.css'],
animations: [
trigger('fade', [
transition(':enter', [
style({opacity: 0, transform: 'translateX(-40px)'}),
animate(500, style({opacity:1, transform: 'translateY(0px)'}))
]),
transition(':leave', [
style({opacity: 0, transform: 'translateX(0px)'}),
animate(1000, style({opacity:1, transform: 'translateX(-50px)'}))
])
])
]
})
export class TodoList{
taskId: number;
taskTitle: string;
editing: boolean = false;
tasks:Tasks[];
beforeEditing: string;
today: number = Date.now();
constructor(private todoService: TodoService){
}
ngOnInit(){
this.beforeEditing = '';
this.taskId;
this.taskTitle = '';
this.todoService
.getTodo()
.subscribe(data => this.tasks = data);
}
addTask(){
//restricting empty string
if(this.taskTitle.trim().length === 0){
return;
}
this.todoService.addTask({id:this.taskId,
title: this.taskTitle,
completed: false,
editing: false})
.subscribe((data: Tasks) => {
console.log(data);
})
//display
this.tasks.push({
id:this.taskId,
title: this.taskTitle,
completed: false,
editing: false
})
this.taskTitle ='';
// this.taskId++;
}
//delete function
deleteTask(tasks: Tasks){
this.tasks = this.tasks.filter(task => task.id !== tasks.id);
this.todoService.deleteTask(tasks)
.subscribe(data => this.tasks.filter(task => {
return task.id !== data;
}))
}
toggleEdit(event: Tasks){
event.editing = !event.editing;
}
editTask(event: Tasks): void{
this.beforeEditing = event.title;
event.editing = !event.editing;
this.todoService.editTodo(event)
.subscribe(data => this.tasks = this.tasks.map((task: Tasks )=>{
if (task.title === event.title){
task = Object.assign({}, task, event);
}
return task;
}));
}
doneEditing(task: Tasks):void{
if(task.title.trim().length === 0){
task.title = this.beforeEditing;
}
task.editing = false;
}
cancelEditing(task: Tasks){
task.title = this.beforeEditing;
task.editing = false;
}
remaining(): number{
return this.tasks.filter(task => !task.completed).length;
}
atleastOneCompleted(): boolean{
return this.tasks.filter(task => task.completed).length > 0;
}
clearCompleted(): void{
this.tasks = this.tasks.filter(task => !task.completed);
}
selectAll():void{
this.tasks.forEach(task => task.completed =
(<HTMLInputElement>event.target).checked);
}
}

View file

@ -0,0 +1,6 @@
export interface Tasks{
id: number,
title: string,
completed: boolean,
editing:boolean
}

View file

@ -0,0 +1,33 @@
import { HttpClient, HttpHeaders} from "@angular/common/http";
import { Injectable } from "@angular/core";
import { Observable } from "rxjs";
import { Tasks } from "../models/todo-interface";
import { map } from "rxjs/operators";
const URL = 'http://localhost:3000/tasks';
@Injectable()
export class TodoService{
constructor(private http: HttpClient){}
getTodo():Observable<Tasks[]>{
return this.http.get<Tasks[]>(URL)
}
addTask(task: Tasks):Observable<Tasks>{
return this.http.post<Tasks>(URL, task, {
headers: new HttpHeaders({
'Content-Type': 'application/json'
})
})
}
editTodo(task: Tasks):Observable<Tasks[]>{
return this.http
.put(URL + '/' + task.id, task)
.pipe(map((response: any) => response.json()));
}
deleteTask(task: Tasks){
return this.http.delete(URL + '/' + task.id);
}
}

View file

@ -0,0 +1,8 @@
@import url('https://fonts.googleapis.com/css2?family=Ruluko&display=swap');
*{
font-family: 'Ruluko', sans-serif;
margin: 0;
padding: 0;
}

View file

@ -1,20 +1 @@
<!--The content below is only a placeholder and can be replaced.-->
<div style="text-align:center">
<h1>
Welcome to {{ title }}!
</h1>
<img width="300" alt="Angular Logo" src="">
</div>
<h2>Here are some links to help you start: </h2>
<ul>
<li>
<h2><a target="_blank" rel="noopener" href="https://angular.io/tutorial">Tour of Heroes</a></h2>
</li>
<li>
<h2><a target="_blank" rel="noopener" href="https://github.com/angular/angular-cli/wiki">CLI Documentation</a></h2>
</li>
<li>
<h2><a target="_blank" rel="noopener" href="https://blog.angular.io/">Angular blog</a></h2>
</li>
</ul>
<todo-list></todo-list>

View file

@ -1,16 +1,26 @@
import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { FormsModule } from '@angular/forms';
import { AutofocusModule } from 'angular-autofocus-fix';
import { BrowserAnimationsModule } from '@angular/platform-browser/animations'
import { AppComponent } from './app.component';
import { TodoList } from './Todo-List-App/containers/TodoList/Todo-list.component';
import { TodoService } from './Todo-List-App/services/todo.service';
import { HttpClientModule } from '@angular/common/http';
@NgModule({
declarations: [
AppComponent
AppComponent,
TodoList
],
imports: [
BrowserModule
BrowserModule,
FormsModule,
AutofocusModule,
BrowserAnimationsModule,
HttpClientModule
],
providers: [],
providers: [TodoService],
bootstrap: [AppComponent]
})
export class AppModule { }