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

@ -23,9 +23,12 @@
"src/assets"
],
"styles": [
"node_modules/bootstrap/dist/css/bootstrap.min.css",
"node_modules/font-awesome/css/font-awesome.css",
"src/styles.css"
],
"scripts": []
"scripts": [
]
},
"configurations": {
"production": {

16
db.json Normal file
View File

@ -0,0 +1,16 @@
{
"tasks": [
{
"title": "Study Angular",
"completed": false,
"editing": false,
"id": 1
},
{
"title": "Workout ",
"completed": false,
"editing": false,
"id": 2
}
]
}

4586
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -11,6 +11,7 @@
},
"private": true,
"dependencies": {
"@angular-devkit/build-angular": "^0.13.0",
"@angular/animations": "^6.1.0",
"@angular/common": "^6.1.0",
"@angular/compiler": "^6.1.0",
@ -20,12 +21,16 @@
"@angular/platform-browser": "^6.1.0",
"@angular/platform-browser-dynamic": "^6.1.0",
"@angular/router": "^6.1.0",
"angular-autofocus-fix": "^0.1.2",
"bootstrap": "^3.4.1",
"core-js": "^2.5.4",
"font-awesome": "^4.7.0",
"json-server": "^0.16.1",
"rxjs": "~6.2.0",
"rxjs-compat": "^6.6.3",
"zone.js": "~0.8.26"
},
"devDependencies": {
"@angular-devkit/build-angular": "~0.8.0",
"@angular/cli": "~6.2.9",
"@angular/compiler-cli": "^6.1.0",
"@angular/language-service": "^6.1.0",

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 { }

BIN
src/favicon.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.1 KiB

View File

@ -2,11 +2,13 @@
<html lang="en">
<head>
<meta charset="utf-8">
<title>AngularAppseptten</title>
<title>Todo List App</title>
<base href="/">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="icon" type="image/x-icon" href="favicon.ico">
<!-- <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> -->
</head>
<body>
<app-root></app-root>

51
src/to-do-list.svg Normal file
View File

@ -0,0 +1,51 @@
<?xml version="1.0" encoding="iso-8859-1"?>
<!-- Generator: Adobe Illustrator 18.1.1, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
<svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
viewBox="0 0 19.118 19.118" style="enable-background:new 0 0 19.118 19.118;" xml:space="preserve">
<g>
<g>
<path style="fill:#030104;" d="M16.981,0H2.137C1.731,0,1.401,0.33,1.401,0.736v17.646c0,0.408,0.33,0.736,0.736,0.736h14.843
c0.406,0,0.736-0.328,0.736-0.736V0.736C17.717,0.329,17.386,0,16.981,0z M16.245,17.646H2.873V1.473h13.371V17.646z"/>
<path style="fill:#030104;" d="M7.64,4.668H4.946v2.693h2.693V4.668H7.64z M7.312,5.562L6.116,6.758
c-0.031,0.032-0.084,0.032-0.115,0L5.272,6.026c-0.031-0.032-0.031-0.084,0-0.117l0.175-0.172c0.031-0.033,0.083-0.033,0.116,0
L6.06,6.236l0.963-0.963c0.033-0.032,0.084-0.032,0.117,0l0.173,0.174C7.345,5.478,7.345,5.53,7.312,5.562z"/>
<rect x="8.202" y="5.274" style="fill:#030104;" width="6.161" height="1.481"/>
<path style="fill:#030104;" d="M7.64,8.18H4.946v2.692h2.693V8.18H7.64z M7.312,9.073l-1.196,1.196
c-0.031,0.032-0.084,0.032-0.115,0L5.272,9.537c-0.031-0.032-0.031-0.084,0-0.116l0.175-0.173c0.031-0.032,0.083-0.032,0.116,0
L6.06,9.747l0.963-0.963c0.033-0.032,0.084-0.032,0.117,0l0.173,0.173C7.345,8.989,7.345,9.041,7.312,9.073z"/>
<rect x="8.202" y="8.785" style="fill:#030104;" width="6.161" height="1.481"/>
<rect x="4.947" y="11.769" style="fill:#030104;" width="2.693" height="2.693"/>
<rect x="8.202" y="12.376" style="fill:#030104;" width="6.161" height="1.48"/>
</g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
<g>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.7 KiB