29 Commits

Author SHA1 Message Date
3ebb1560ae Add snack messages after some actions. 2022-03-06 15:10:07 +01:00
e845bfb27f Change some style. 2022-03-06 14:57:01 +01:00
863caeccf3 Disable a button. 2022-03-06 14:45:34 +01:00
9b56dddab9 Use factorized style. 2022-03-06 14:42:24 +01:00
ba11f17531 Use factorized colors. 2022-03-06 14:35:49 +01:00
a08f92c84c Factorize colors. 2022-03-06 14:29:04 +01:00
c027357cfe Refactor header style. 2022-03-06 13:38:02 +01:00
0fbc8b1392 Implementation of task-lists deletion with dialog. 2022-03-06 13:30:50 +01:00
54108bc7e5 Change task-list adding dialog style. 2022-03-06 13:04:39 +01:00
6de8a4f0fa Change style of task-list renaming dialog. 2022-03-06 13:02:02 +01:00
7283e4f1aa Implementation of task-list renaming. 2022-03-06 12:57:31 +01:00
8d3b7fa1c4 Add selection actions bar. 2022-03-06 12:24:03 +01:00
b8e4d34456 Change taskLists selection display. 2022-03-06 12:19:48 +01:00
1037d74620 Minor visual changes 2022-03-05 12:54:19 +01:00
4cfed23613 Add description saving and silent save in service. 2022-03-05 12:36:59 +01:00
a5f4c18eb5 Add task deletion. 2022-03-05 12:11:41 +01:00
5ad34da8f7 Lot of visual improvements. 2022-03-05 00:10:02 +01:00
efa34e30be Lot of visual improvements 2022-03-04 23:47:52 +01:00
8bc8f6eee0 Reset new task input after creation; 2022-03-04 22:56:00 +01:00
7686057022 Change store storage system. 2022-03-04 22:54:52 +01:00
21bd3826e6 Add task displaying in active list component. 2022-03-04 22:08:13 +01:00
Florian THIERRY
86fcdd4d12 Some change styling. 2022-03-04 17:43:04 +01:00
Florian THIERRY
3819b859fe Change header display in selection mode. 2022-03-04 17:40:47 +01:00
Florian THIERRY
760e08a595 Add active-list-tasks component and navigation with home pane. 2022-03-04 17:27:13 +01:00
Florian THIERRY
9173d2220c Add task list selection system. 2022-03-04 16:53:08 +01:00
Florian THIERRY
8e582c96e3 Code cleaning. 2022-03-04 16:10:24 +01:00
Florian THIERRY
da80bc17c0 Refactor store observer mecanism to pilot edition actions. 2022-03-04 16:09:45 +01:00
Florian THIERRY
182cc0bb67 Add task-list creation. 2022-03-04 10:35:54 +01:00
Florian THIERRY
f4d0aa3b27 Add new task list model. 2022-03-04 09:21:08 +01:00
37 changed files with 1432 additions and 14 deletions

18
package-lock.json generated
View File

@@ -21,6 +21,7 @@
"ngx-cookie-service": "^12.0.3", "ngx-cookie-service": "^12.0.3",
"rxjs": "~6.6.0", "rxjs": "~6.6.0",
"tslib": "^2.3.0", "tslib": "^2.3.0",
"uuid": "^8.3.2",
"zone.js": "~0.11.4" "zone.js": "~0.11.4"
}, },
"devDependencies": { "devDependencies": {
@@ -29,6 +30,7 @@
"@angular/compiler-cli": "~12.2.0", "@angular/compiler-cli": "~12.2.0",
"@types/jasmine": "~3.8.0", "@types/jasmine": "~3.8.0",
"@types/node": "^12.11.1", "@types/node": "^12.11.1",
"@types/uuid": "^8.3.4",
"jasmine-core": "~3.8.0", "jasmine-core": "~3.8.0",
"karma": "~6.3.0", "karma": "~6.3.0",
"karma-chrome-launcher": "~3.1.0", "karma-chrome-launcher": "~3.1.0",
@@ -2573,6 +2575,12 @@
"integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==", "integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==",
"dev": true "dev": true
}, },
"node_modules/@types/uuid": {
"version": "8.3.4",
"resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.4.tgz",
"integrity": "sha512-c/I8ZRb51j+pYGAu5CrFMRxqZ2ke4y2grEBO5AUjgSkSk+qT2Ea+OdWElz/OiMf5MNpn2b17kuVBwZLQJXzihw==",
"dev": true
},
"node_modules/@types/webpack-sources": { "node_modules/@types/webpack-sources": {
"version": "0.1.9", "version": "0.1.9",
"resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-0.1.9.tgz", "resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-0.1.9.tgz",
@@ -14882,7 +14890,6 @@
"version": "8.3.2", "version": "8.3.2",
"resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
"integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
"dev": true,
"bin": { "bin": {
"uuid": "dist/bin/uuid" "uuid": "dist/bin/uuid"
} }
@@ -17916,6 +17923,12 @@
"integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==", "integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==",
"dev": true "dev": true
}, },
"@types/uuid": {
"version": "8.3.4",
"resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-8.3.4.tgz",
"integrity": "sha512-c/I8ZRb51j+pYGAu5CrFMRxqZ2ke4y2grEBO5AUjgSkSk+qT2Ea+OdWElz/OiMf5MNpn2b17kuVBwZLQJXzihw==",
"dev": true
},
"@types/webpack-sources": { "@types/webpack-sources": {
"version": "0.1.9", "version": "0.1.9",
"resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-0.1.9.tgz", "resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-0.1.9.tgz",
@@ -27377,8 +27390,7 @@
"uuid": { "uuid": {
"version": "8.3.2", "version": "8.3.2",
"resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
"integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg=="
"dev": true
}, },
"validate-npm-package-name": { "validate-npm-package-name": {
"version": "3.0.0", "version": "3.0.0",

View File

@@ -20,9 +20,9 @@
"@angular/platform-browser": "~12.2.0", "@angular/platform-browser": "~12.2.0",
"@angular/platform-browser-dynamic": "~12.2.0", "@angular/platform-browser-dynamic": "~12.2.0",
"@angular/router": "~12.2.0", "@angular/router": "~12.2.0",
"ngx-cookie-service": "^12.0.3",
"rxjs": "~6.6.0", "rxjs": "~6.6.0",
"tslib": "^2.3.0", "tslib": "^2.3.0",
"uuid": "^8.3.2",
"zone.js": "~0.11.4" "zone.js": "~0.11.4"
}, },
"devDependencies": { "devDependencies": {
@@ -31,6 +31,7 @@
"@angular/compiler-cli": "~12.2.0", "@angular/compiler-cli": "~12.2.0",
"@types/jasmine": "~3.8.0", "@types/jasmine": "~3.8.0",
"@types/node": "^12.11.1", "@types/node": "^12.11.1",
"@types/uuid": "^8.3.4",
"jasmine-core": "~3.8.0", "jasmine-core": "~3.8.0",
"karma": "~6.3.0", "karma": "~6.3.0",
"karma-chrome-launcher": "~3.1.0", "karma-chrome-launcher": "~3.1.0",
@@ -39,4 +40,4 @@
"karma-jasmine-html-reporter": "~1.7.0", "karma-jasmine-html-reporter": "~1.7.0",
"typescript": "~4.3.5" "typescript": "~4.3.5"
} }
} }

View File

@@ -0,0 +1,10 @@
<div class="container">
<app-task-display *ngFor="let task of activeTaskList?.tasks"
[task]="task" class="task"></app-task-display>
<div class="task new">
<mat-icon>add</mat-icon>
<input placeholder="Nouvelle tâche..."
(keydown)="onNewTaskKeyDown($event)"
[formControl]="newTaskControl"/>
</div>
</div>

View File

@@ -0,0 +1,31 @@
.container {
max-width: 50rem;
width: 90%;
margin: 0 auto;
padding: 2rem;
.task {
&.new {
position: relative;
width: 100%;
display: flex;
align-items: center;
margin-top: 1rem;
height: 2.5rem;
mat-icon {
position: absolute;
left: 1rem;
}
input {
width: 100%;
height: 100%;
padding: 0 3rem;
border-radius: .1rem;
border-style: none;
background-color: var(--secondary);
}
}
}
}

View File

@@ -0,0 +1,54 @@
import { Component, OnDestroy, OnInit } from '@angular/core';
import { FormControl, Validators } from '@angular/forms';
import { MatSnackBar } from '@angular/material/snack-bar';
import { Router } from '@angular/router';
import { Subscription } from 'rxjs';
import { Task } from '../core/entity/task';
import { TaskList } from '../core/entity/task-list';
import { TaskListService } from '../core/service/task-list.service';
@Component({
selector: 'app-active-list-tasks',
templateUrl: './active-list-tasks.component.html',
styleUrls: ['./active-list-tasks.component.scss']
})
export class ActiveListTasksComponent implements OnInit, OnDestroy {
private _storeSubscription?: Subscription;
activeTaskList?: TaskList;
newTaskControl: FormControl = new FormControl(undefined, Validators.required);
constructor(
private _router: Router,
private _taskListService: TaskListService,
private _snackBar: MatSnackBar
) {}
ngOnInit(): void {
this._storeSubscription = this._taskListService.store$.subscribe(store => {
if (store.activeTaskListId) {
this.activeTaskList = store.taskLists.find(taskList => taskList.id === store.activeTaskListId);
if (!this.activeTaskList) {
this._backToTaskListPane();
}
}
});
}
ngOnDestroy(): void {
this._storeSubscription?.unsubscribe();
}
private _backToTaskListPane(): void {
console.error('La task-list active n\'existe pas dans le store.', 'Fermer', {duration: 5000});
this._router.navigate(['/']);
}
onNewTaskKeyDown(event: KeyboardEvent): void {
if (event.key === 'Enter') {
if (this.newTaskControl.valid) {
this._taskListService.addTask(this.newTaskControl.value as string);
this.newTaskControl.reset();
}
}
}
}

View File

@@ -0,0 +1,37 @@
<div class="task">
<div class="header">
<mat-icon class="drag-n-drop">drag_handle</mat-icon>
<mat-checkbox class="example-margin"></mat-checkbox>
<div class="input-container">
<input type="text" #titleInput [formControl]="titleControl"/>
</div>
<button type="button" class="expand icon" (click)="expand()" matRipple>
<mat-icon *ngIf="!isExpanded">expand_more</mat-icon>
<mat-icon *ngIf="isExpanded">expand_less</mat-icon>
</button>
</div>
<div [ngClass]="getExpendedClass()">
<div class="container">
<div class="description-container">
<label for="description">
Description
</label>
<textarea id="description" [formControl]="descriptionControl"></textarea>
</div>
<div class="actions">
<div class="row">
<button class="stroked secondary" [disabled]="true" matRipple matTooltip="Définir une alerte dans X minutes">
<mat-icon>update</mat-icon>
Rappel
</button>
</div>
<div class="row">
<button class="stroked alert" [disabled]="!task" (click)="delete()">
<mat-icon>delete</mat-icon>
Supprimer
</button>
</div>
</div>
</div>
</div>
</div>

View File

@@ -0,0 +1,118 @@
.task {
position: relative;
width: 100%;
display: flex;
align-items: center;
margin-bottom: .5rem;
flex-direction: column;
background-color: var(--secondary);
border-radius: .2rem;
.header {
height: 2.5rem;
width: 100%;
display: flex;
align-items: center;
.drag-n-drop {
padding: 0 1rem;
&:hover {
cursor: grab;
}
}
.input-container {
display: flex;
flex-grow: 1;
input {
border-style: none;
height: 1.8rem;
padding: 0 .8rem;
margin: 0 .8rem;
background-color: inherit;
border-style: solid;
border-width: .1rem;
border-color: rgba(0,0,0, 0);
border-radius: .2rem;
width: 100%;
transition: background-color .2s ease-out,
border-color .2s ease-out;
&:hover {
border-color: var(--secondary-border);
}
}
}
.expand {
margin-right: .5rem;
}
}
.body {
height: 0;
visibility: hidden;
transition: height .1s ease-in-out;
display: flex;
border-top: .1rem solid var(--primary-border);
&.expanded {
height: 20rem;
visibility: visible;
width: 100%;
}
.container {
flex-grow: 1;
padding: 1rem;
display: flex;
flex-direction: row;
.description-container {
width: 80%;
height: 100%;
display: flex;
flex-direction: column;
textarea {
flex-grow: 1;
border: 1px solid var(--secondary-border);
border-radius: .2rem;
background-color: #ddd;
resize: none;
padding: .5rem;
}
}
.actions {
display: flex;
flex-direction: column;
flex-grow: 1;
height: 100%;
max-width: 20%;
.row {
display: flex;
flex-direction: row;
flex-wrap: wrap;
justify-content: space-around;
align-items: flex-start;
padding: .5rem 0 .5rem 1rem;
button {
display: flex;
justify-content: left;
flex-grow: 1;
mat-icon {
margin-right: .5rem;
}
}
}
}
}
}
}

View File

@@ -0,0 +1,103 @@
import { AfterViewInit, Component, ElementRef, EventEmitter, Input, OnDestroy, OnInit, Output, ViewChild } from '@angular/core';
import { FormControl } from '@angular/forms';
import { MatDialog } from '@angular/material/dialog';
import { Subscription } from 'rxjs';
import { debounceTime, distinctUntilChanged } from 'rxjs/operators';
import { ConfirmDialogComponent, ConfirmDialogModel } from 'src/app/core/components/confirm-dialog/confirm-dialog.component';
import { Task } from 'src/app/core/entity/task';
import { TaskListService } from 'src/app/core/service/task-list.service';
@Component({
selector: 'app-task-display',
templateUrl: './task-display.component.html',
styleUrls: ['./task-display.component.scss']
})
export class TaskDisplayComponent implements AfterViewInit, OnDestroy {
@Input() task?: Task;
@ViewChild('titleInput', {static: true}) titleInput?: ElementRef<HTMLInputElement>;
@ViewChild('descriptionInput', {static: true}) descriptionInput?: ElementRef<HTMLTextAreaElement>;
titleControl = new FormControl();
descriptionControl = new FormControl();
isExpanded = false;
private _subscriptions: Subscription[] = [];
constructor(
private _dialog: MatDialog,
private _taskListService: TaskListService
) {}
ngAfterViewInit(): void {
this.titleControl.setValue(this?.task?.title);
this.descriptionControl.setValue(this?.task?.description);
const titleControlSubscription = this.titleControl.valueChanges
.pipe(
distinctUntilChanged(),
debounceTime(500)
)
.subscribe(newTitle => {
if (this.task) {
this.task.title = newTitle;
this._taskListService.updateTask(this.task);
}
});
this._subscriptions.push(titleControlSubscription);
const descriptionControlSubscription = this.descriptionControl.valueChanges
.pipe(
distinctUntilChanged(),
debounceTime(500)
)
.subscribe(description => {
if (this.task) {
this.task.description = description;
this._taskListService.updateTask(this.task);
}
});
this._subscriptions.push(descriptionControlSubscription);
}
ngOnDestroy(): void {
this._subscriptions.forEach(subscription => subscription.unsubscribe());
}
expand(): void {
this.isExpanded = !this.isExpanded;
}
getExpendedClass(): string {
let result = 'body';
if (this.isExpanded) {
result += ' expanded';
}
return result;
}
delete(): void {
if (this.task) {
const confirmData = {
title: `Supprimer la tâche ${this.task.title} ?`,
description: 'Une fois supprimé, sa description sera perdue définitivement.',
confirmButtonLabel: 'Supprimer la tâche',
confirmButtonType: 'alert'
} as ConfirmDialogModel;
const dialogRef = this._dialog.open(
ConfirmDialogComponent,
{
width: '30rem',
data: confirmData
}
);
const afterDialogCloseSubscription = dialogRef.afterClosed().subscribe(result => {
if (result && this.task) {
this._taskListService.delete(this.task);
}
})
this._subscriptions.push(afterDialogCloseSubscription);
}
}
}

View File

@@ -1,9 +1,11 @@
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router'; import { RouterModule, Routes } from '@angular/router';
import { ActiveListTasksComponent } from './active-list-tasks/active-list-tasks.component';
import { MainPageComponent } from './main-page/main-page.component'; import { MainPageComponent } from './main-page/main-page.component';
const routes: Routes = [ const routes: Routes = [
{ path: '', component: MainPageComponent} { path: '', component: MainPageComponent},
{ path: 'task-lists/active', component: ActiveListTasksComponent }
]; ];
@NgModule({ @NgModule({

View File

@@ -1 +1,2 @@
<router-outlet></router-outlet> <app-header></app-header>
<router-outlet></router-outlet>

View File

@@ -1,4 +1,4 @@
import { NgModule } from '@angular/core'; import { APP_INITIALIZER, NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser'; import { BrowserModule } from '@angular/platform-browser';
import { CookieService } from 'ngx-cookie-service'; import { CookieService } from 'ngx-cookie-service';
@@ -10,23 +10,59 @@ import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
import {MatIconModule} from '@angular/material/icon'; import {MatIconModule} from '@angular/material/icon';
import {MatInputModule} from '@angular/material/input'; import {MatInputModule} from '@angular/material/input';
import { DisplayTaskComponent } from './display-task/display-task.component'; import { DisplayTaskComponent } from './display-task/display-task.component';
import { TaskListsComponent } from './task-lists/task-lists.component';
import { AddTaskListComponent } from './task-lists/add-task-list/add-task-list.component';
import {MatDialogModule} from '@angular/material/dialog';
import {MatButtonModule} from '@angular/material/button';
import {ReactiveFormsModule} from '@angular/forms';
import {MatSnackBarModule} from '@angular/material/snack-bar';
import { HeaderComponent } from './core/components/header/header.component';
import { ActiveListTasksComponent } from './active-list-tasks/active-list-tasks.component';
import { TaskDisplayComponent } from './active-list-tasks/task-display/task-display.component';
import {MatCheckboxModule} from '@angular/material/checkbox';
import { TaskListService } from './core/service/task-list.service';
import { StorePersistenceService } from './core/service/store-persistence.service';
import {MatTooltipModule} from '@angular/material/tooltip';
import {MatRippleModule} from '@angular/material/core';
import { RenameTaskListComponent } from './task-lists/rename-task-list/rename-task-list.component';
import { ConfirmDialogComponent } from './core/components/confirm-dialog/confirm-dialog.component';
@NgModule({ @NgModule({
declarations: [ declarations: [
AppComponent, AppComponent,
MainPageComponent, MainPageComponent,
AddTaskComponent, AddTaskComponent,
DisplayTaskComponent DisplayTaskComponent,
TaskListsComponent,
AddTaskListComponent,
HeaderComponent,
ActiveListTasksComponent,
TaskDisplayComponent,
RenameTaskListComponent,
ConfirmDialogComponent
], ],
imports: [ imports: [
BrowserModule, BrowserModule,
AppRoutingModule, AppRoutingModule,
BrowserAnimationsModule, BrowserAnimationsModule,
MatIconModule, MatIconModule,
MatInputModule MatInputModule,
MatDialogModule,
MatButtonModule,
ReactiveFormsModule,
MatSnackBarModule,
MatCheckboxModule,
MatTooltipModule,
MatRippleModule
], ],
providers: [ providers: [
CookieService, CookieService,
{
provide: APP_INITIALIZER,
useFactory: (taskListService: TaskListService) => () => taskListService.removeActiveTaskList(),
deps: [TaskListService, StorePersistenceService],
multi: true
}
], ],
bootstrap: [AppComponent] bootstrap: [AppComponent]
}) })

View File

@@ -0,0 +1,17 @@
<div class="container">
<div class="body">
<mat-icon>help_outline</mat-icon>
<div class="message">
<div class="title">{{data.title}}</div>
<div class="description">{{data.description}}</div>
</div>
</div>
<div class="dialog-actions">
<button class="stroked" (click)="cancel()">Annuler</button>
<button class="stroked"
(click)="confirm()"
[ngClass]="data.confirmButtonType">
{{data.confirmButtonLabel}}
</button>
</div>
</div>

View File

@@ -0,0 +1,35 @@
.container {
.body {
display: flex;
flex-direction: row;
flex-grow: 1;
align-items: center;
mat-icon {
$iconSize: 4rem;
font-size: $iconSize;
width: $iconSize;
height: $iconSize;
margin: 0 1rem;
}
.message {
.title {
display: flex;
flex-grow: 1;
justify-content: center;
text-align: center;
font-size: 1.5rem;
margin: .5rem;
}
.description {
display: flex;
flex-grow: 1;
justify-content: center;
text-align: center;
font-size: 1.2rem;
}
}
}
}

View File

@@ -0,0 +1,25 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { ConfirmDialogComponent } from './confirm-dialog.component';
describe('ConfirmDialogComponent', () => {
let component: ConfirmDialogComponent;
let fixture: ComponentFixture<ConfirmDialogComponent>;
beforeEach(async () => {
await TestBed.configureTestingModule({
declarations: [ ConfirmDialogComponent ]
})
.compileComponents();
});
beforeEach(() => {
fixture = TestBed.createComponent(ConfirmDialogComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
});

View File

@@ -0,0 +1,31 @@
import { Component, Inject } from '@angular/core';
import { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';
export interface ConfirmDialogModel {
title: string;
description: string;
confirmButtonLabel: string;
confirmButtonType: '' | 'alert';
}
@Component({
selector: 'app-confirm-dialog',
templateUrl: './confirm-dialog.component.html',
styleUrls: ['./confirm-dialog.component.scss']
})
export class ConfirmDialogComponent {
constructor(
@Inject(MAT_DIALOG_DATA) public data: ConfirmDialogModel,
private _dialogRef: MatDialogRef<ConfirmDialogModel>
) {}
cancel(): void {
this._dialogRef.close();
}
confirm(): void {
this._dialogRef.close(true);
}
}

View File

@@ -0,0 +1,42 @@
<nav *ngIf="!selectionMode">
<div class="left">
<span class="title">
<img src="../../assets/images/to-do.png" />
To Do
</span>
<button *ngIf="!activeTaskList"
(click)="openNewListForm()"
class="stroked primary"
matRipple>
Nouvelle liste
</button>
<button *ngIf="activeTaskList"
(click)="goTaskListsPane()"
class="icon stroked primary"
matRipple
matTooltip="Retourner aux task-lists">
<mat-icon>chevron_left</mat-icon>
</button>
</div>
<div class="middle" *ngIf="activeTaskList">
{{activeTaskList.name}}
</div>
<div class="right">
<button class="icon stroked primary"
(click)="enableSelectionMode()"
[disabled]="isNoAnyTaskList()"
matTooltip="Activer la sélection des task-lists"
matTooltipPosition="left">
<mat-icon>checklist_rtl</mat-icon>
</button>
</div>
</nav>
<nav *ngIf="selectionMode" class="selectionMode">
<div></div>
<div>
Cliquez sur une liste pour la sélectionner
</div>
<div class="actions">
<button class="stroked primary" (click)="disableSelectionMode()">Annuler</button>
</div>
</nav>

View File

@@ -0,0 +1,49 @@
nav {
position: fixed;
top: 0;
left: 0;
right: 0;
display: flex;
align-items: center;
height: 3.2rem;
background-color: #eee;
.left {
position: absolute;
left: 1rem;
display: flex;
flex-direction: row;
align-items: center;
.title {
font-size: 1.5rem;
display: flex;
align-items: center;
margin-right: 1rem;
}
}
.middle {
display: flex;
margin: auto;
}
.right {
position: absolute;
right: 1rem;
}
&.selectionMode {
font-weight: bold;
background-color: var(--selection);
color: white;
display: flex;
justify-content: center;
.actions {
position: absolute;
right: 0;
margin: 0 1rem;
}
}
}

View File

@@ -0,0 +1,61 @@
import { Component, OnDestroy, OnInit } from '@angular/core';
import { MatDialog } from '@angular/material/dialog';
import { Router } from '@angular/router';
import { Subscription } from 'rxjs';
import { AddTaskListComponent } from 'src/app/task-lists/add-task-list/add-task-list.component';
import { TaskList } from '../../entity/task-list';
import { TaskListService } from '../../service/task-list.service';
@Component({
selector: 'app-header',
templateUrl: './header.component.html',
styleUrls: ['./header.component.scss']
})
export class HeaderComponent implements OnInit, OnDestroy {
private _storeSubscription?: Subscription;
selectionMode = false;
constructor(
private _dialog: MatDialog,
private _taskListService: TaskListService
) {}
ngOnInit(): void {
this._storeSubscription = this._taskListService.store$.subscribe(store => {
this.selectionMode = store.selectionMode;
});
}
ngOnDestroy(): void {
this._storeSubscription?.unsubscribe();
}
openNewListForm(): void {
this._dialog.open(
AddTaskListComponent,
{
width: '20rem'
}
);
}
goTaskListsPane(): void {
this._taskListService.removeActiveTaskList();
}
enableSelectionMode(): void {
this._taskListService.enableSelectionMode();
}
disableSelectionMode(): void {
this._taskListService.disableSelectionMode();
}
isNoAnyTaskList(): boolean {
return this._taskListService.isNoAnyTaskList();
}
get activeTaskList(): TaskList | undefined {
return this._taskListService.activeTaskList;
}
}

View File

@@ -0,0 +1,8 @@
import { TaskList } from "./task-list";
export interface Store {
activeTaskListId: string | undefined;
taskLists: TaskList[];
selectedTaskLists: TaskList[];
selectionMode: boolean;
}

View File

@@ -0,0 +1,7 @@
import { Task } from "./task";
export interface TaskList {
id: string;
name: string;
tasks: Task[];
}

View File

@@ -1,4 +1,5 @@
export interface Task { export interface Task {
id: string;
title: string; title: string;
creationDate: Date; creationDate: Date;
description: string; description: string;

View File

@@ -0,0 +1,33 @@
import { Injectable } from "@angular/core";
import { Store } from "../entity/store";
const STORE_NAME = 'todo-store';
@Injectable({
providedIn: 'root'
})
export class StorePersistenceService {
save(store: Store): void {
const serializedStore = JSON.stringify(store);
localStorage.setItem(STORE_NAME, serializedStore);
}
load(): Store {
const serializedStore = localStorage.getItem(STORE_NAME);
if (serializedStore?.length && serializedStore !== 'undefined') {
try {
return JSON.parse(serializedStore);
} catch (jsonParseError) {
throw new Error(`JsonSerializationException: Invalid format for store in cookie "${STORE_NAME}".`);
}
} else {
return {
activeTaskListId: undefined,
taskLists: [],
selectedTaskLists: [],
selectionMode: false
} as Store;
}
}
}

View File

@@ -0,0 +1,223 @@
import { Injectable } from "@angular/core";
import { BehaviorSubject, Observable } from "rxjs";
import { Task } from "../entity/task";
import { TaskList } from "../entity/task-list";
import { StorePersistenceService } from "./store-persistence.service";
import { v4 as uuidv4 } from 'uuid';
import { filter } from 'rxjs/operators';
import { Store } from "../entity/store";
import { Router } from "@angular/router";
import { MatSnackBar } from "@angular/material/snack-bar";
@Injectable({
providedIn: 'root'
})
export class TaskListService {
private _store: BehaviorSubject<Store> = new BehaviorSubject<Store>(undefined as unknown as Store);
constructor(
private _router: Router,
private _snackBar: MatSnackBar,
private _storePersistenceService: StorePersistenceService
) {
this.store$.subscribe(store => {
this.saveStore(store, true);
});
}
get store$(): Observable<Store> {
return this._store.asObservable()
.pipe(filter(store => !!store));
}
private get store(): Store {
let result = this._store.value;
if (!result) {
result = this._storePersistenceService.load();
}
return result;
}
private saveStore(store: Store, silent: boolean = false): void {
if (silent) {
this._storePersistenceService.save(store);
} else {
// We send the store into the subject because there is an observable on in, that saves the store at every single value.
this._store.next(store);
}
}
addTask(taskTitle: string): void {
const store = this.store;
const activeTaskList = store.taskLists.find(taskList => taskList.id === store.activeTaskListId);
if (!activeTaskList) {
throw new Error("No active tasklist");
}
if (!activeTaskList.tasks) {
activeTaskList.tasks = [];
}
const newTask = {
id: uuidv4(),
title: taskTitle,
creationDate: new Date(),
description: undefined as unknown as string
} as Task;
activeTaskList?.tasks.push(newTask);
this.saveStore(store);
this._snackBar.open('Tâche ajoutée.', 'Fermer', {duration: 2000});
}
updateTask(taskToUpdate: Task) {
const store = this.store;
const activeTaskList = store.taskLists.find(taskList => taskList.id === store.activeTaskListId);
if (!activeTaskList) {
throw new Error("No active tasklist");
}
const task = activeTaskList?.tasks.find(task => task.id === taskToUpdate.id);
if (!task) {
throw new Error('Unknown task to update');
}
task.title = taskToUpdate.title;
task.description = taskToUpdate.description;
// If the store is saved loudly, all views will be refreshed and the user will lose the focus of its input,
// so we save the store silently here.
this.saveStore(store, true);
}
delete(taskToDelete: Task) {
const store = this.store;
const activeTaskList = store.taskLists.find(taskList => taskList.id === store.activeTaskListId);
if (!activeTaskList) {
throw new Error("No active tasklist");
}
const taskIndex = activeTaskList?.tasks.findIndex(task => task.id === taskToDelete.id);
if (!taskIndex && taskIndex !== 0) {
throw new Error('Unknown task to delete');
}
activeTaskList?.tasks.splice(taskIndex, 1);
this.saveStore(store);
this._snackBar.open('Tâche supprimée.', 'Fermer', {duration: 2000});
}
createTaskList(taskListName: string): void {
const newTaskList = {
id: uuidv4(),
name: taskListName,
tasks: []
} as TaskList;
const store = this.store;
store.taskLists.push(newTaskList);
this.saveStore(store);
}
updateTaskListName(taskListToUpdate: TaskList): void {
const store = this.store;
const matchingTaskList = store.taskLists.find(taskList => taskList.id === taskListToUpdate.id);
if (matchingTaskList) {
matchingTaskList.name = taskListToUpdate.name;
this.saveStore(store);
this.disableSelectionMode();
}
}
deleteSelectedTaskLists(): void {
const store = this.store;
const nonSelectedTaskLists = store.taskLists.filter(taskList => !store.selectedTaskLists.some(selectedTaskList => selectedTaskList.id === taskList.id));
store.taskLists = nonSelectedTaskLists;
this.saveStore(store);
this.disableSelectionMode();
}
getAll(): TaskList[] {
return this.store.taskLists ?? [];
}
setActive(taskList: TaskList): void {
const store = this.store;
store.activeTaskListId = taskList.id;
this.saveStore(store);
this._router.navigate(['/task-lists/active']);
}
removeActiveTaskList(): void {
const store = this.store;
delete store.activeTaskListId;
this.saveStore(store);
this._router.navigate(['/']);
}
selectTaskList(taskList: TaskList): void {
this.enableSelectionMode();
const store = this.store;
if (!store.selectedTaskLists) {
store.selectedTaskLists = [];
}
if (store.selectedTaskLists.some(tl => tl.id === taskList.id)) {
const selectedTaskListIndex = store.selectedTaskLists.findIndex(tl => tl.id === taskList.id);
store.selectedTaskLists.splice(selectedTaskListIndex, 1);
if (!store.selectedTaskLists.length) {
store.selectionMode = false;
}
this.saveStore(store);
} else {
store.selectedTaskLists.push(taskList);
this.saveStore(store);
}
}
isSelected(taskList: TaskList): boolean {
const store = this.store;
return store.selectedTaskLists && store.selectedTaskLists.some(tl => tl.id === taskList.id);
}
isSelectionModeEnabled(): boolean {
return this.store.selectionMode;
}
enableSelectionMode(): void {
const store = this.store;
store.selectionMode = true;
this.saveStore(store);
}
disableSelectionMode(): void {
const store = this.store;
store.selectionMode = false;
store.selectedTaskLists = [];
this.saveStore(store);
}
isThereMultipleTaskListsSelected(): boolean {
const store = this.store;
return (store.selectedTaskLists?.length ?? 0) > 1;
}
get selectedTaskList(): TaskList {
return this._store.value?.selectedTaskLists[0];
}
isNoAnyTaskList(): boolean {
return !this.store?.taskLists?.length;
}
get activeTaskList(): TaskList | undefined {
const store = this.store;
return store.taskLists.find(taskList => taskList.id === store.activeTaskListId);
}
}

View File

@@ -1,8 +1,10 @@
<div class="component"> <div class="component">
<h1>Todo List</h1> <!-- <h1>Todo List</h1>
<app-display-task *ngFor="let task of tasks" [task]="task"></app-display-task> <app-display-task *ngFor="let task of tasks" [task]="task"></app-display-task>
<div id="add-task-btn" *ngIf="!isAddingATask" (click)="createTask()"> <div id="add-task-btn" *ngIf="!isAddingATask" (click)="createTask()">
<mat-icon>add</mat-icon> <mat-icon>add</mat-icon>
<span>Ajouter une nouvelle tâche...</span> <span>Ajouter une nouvelle tâche...</span>
</div> </div> -->
<app-task-lists></app-task-lists>
</div> </div>

View File

@@ -0,0 +1,16 @@
<h2>Création d'une task-list</h2>
<form [formGroup]="addTaskListFormGroup" (submit)="onSubmit()" ngNativeValidate>
<p>
<mat-form-field>
<mat-label>Nom de la liste</mat-label>
<input matInput autofocus id="task-name-input" name="task-name-input" formControlName="name" />
<mat-error *ngIf="form.name.invalid">
Veuillez saisir le nom de la task-list.
</mat-error>
</mat-form-field>
</p>
<div class="dialog-actions">
<button class="stroked primary" type="button" (click)="close()">Annuler</button>
<button class="stroked primary" type="submit">Créer une liste</button>
</div>
</form>

View File

@@ -0,0 +1,7 @@
form {
p {
mat-form-field {
width: 100%;
}
}
}

View File

@@ -0,0 +1,42 @@
import { Component, OnInit } from '@angular/core';
import { FormBuilder, FormGroup, Validators } from '@angular/forms';
import { MatDialogRef } from '@angular/material/dialog';
import { MatSnackBar } from '@angular/material/snack-bar';
import { TaskList } from 'src/app/core/entity/task-list';
import { TaskListService } from 'src/app/core/service/task-list.service';
@Component({
selector: 'app-add-task-list',
templateUrl: './add-task-list.component.html',
styleUrls: ['./add-task-list.component.scss']
})
export class AddTaskListComponent {
addTaskListFormGroup: FormGroup = this._formBuilder.group({
name: [undefined, Validators.required]
});
constructor(
private _dialogRef: MatDialogRef<AddTaskListComponent>,
private _formBuilder: FormBuilder,
private _snackBar: MatSnackBar,
private _taskListService: TaskListService
) {}
get form(): any {
return this.addTaskListFormGroup.controls;
}
onSubmit(): void {
if (this.addTaskListFormGroup.valid) {
this._taskListService.createTaskList(this.addTaskListFormGroup.controls.name.value);
this._snackBar.open('La task-list a été créée.', 'Fermer', {duration: 5000});
this.close();
} else {
this._snackBar.open('Veuillez vérifier les informations saisies.', 'Fermer', {duration: 5000});
}
}
close(): void {
this._dialogRef.close();
}
}

View File

@@ -0,0 +1,16 @@
<h2>Renommage d'une task-list</h2>
<form [formGroup]="renameTaskListFormGroup" (submit)="onSubmit()" ngNativeValidate>
<p>
<mat-form-field>
<mat-label>Nom de la liste</mat-label>
<input matInput autofocus id="task-name-input" name="task-name-input" formControlName="name" />
<mat-error *ngIf="form.name.invalid">
Veuillez saisir le nom de la task-list.
</mat-error>
</mat-form-field>
</p>
<div class="dialog-actions">
<button class="stroked" type="button" (click)="close()">Annuler</button>
<button class="stroked" type="submit">Renommer la liste</button>
</div>
</form>

View File

@@ -0,0 +1,7 @@
form {
p {
mat-form-field {
width: 100%;
}
}
}

View File

@@ -0,0 +1,59 @@
import { Component, OnInit } from '@angular/core';
import { FormBuilder, FormGroup, Validators } from '@angular/forms';
import { MatDialogRef } from '@angular/material/dialog';
import { MatSnackBar } from '@angular/material/snack-bar';
import { TaskList } from 'src/app/core/entity/task-list';
import { TaskListService } from 'src/app/core/service/task-list.service';
@Component({
selector: 'app-rename-task-list',
templateUrl: './rename-task-list.component.html',
styleUrls: ['./rename-task-list.component.scss']
})
export class RenameTaskListComponent implements OnInit {
renameTaskListFormGroup: FormGroup = this._formBuilder.group({
name: [undefined, Validators.required]
});
selectedTaskList?: TaskList;
constructor(
private _dialogRef: MatDialogRef<RenameTaskListComponent>,
private _formBuilder: FormBuilder,
private _snackBar: MatSnackBar,
private _taskListService: TaskListService
) {}
ngOnInit(): void {
this.selectedTaskList = this._taskListService.selectedTaskList;
if (this.selectedTaskList) {
this.renameTaskListFormGroup.controls.name.setValue(this.selectedTaskList.name);
} else {
this._snackBar.open('Impossible de renommer la task-list : aucune task-list n\'est sélectionnée.', 'Fermer', {duration: 5000});
this._dialogRef.close();
}
}
get form(): any {
return this.renameTaskListFormGroup.controls;
}
onSubmit(): void {
if (this.selectedTaskList) {
if (this.renameTaskListFormGroup.valid) {
this.selectedTaskList.name = this.renameTaskListFormGroup.controls.name.value;
this._taskListService.updateTaskListName(this.selectedTaskList);
this._snackBar.open('La task-list a été renommée.', 'Fermer', {duration: 5000});
this.close();
} else {
this._snackBar.open('Veuillez vérifier les informations saisies.', 'Fermer', {duration: 5000});
}
}
}
close(): void {
this._dialogRef.close();
}
}

View File

@@ -0,0 +1,35 @@
<div class="task-lists">
<div *ngIf="!taskLists?.length" class="no-task-list">
Aucune task-list.
</div>
<div *ngFor="let taskList of taskLists" class="task-list-container">
<div class="task-list shadowed"
(click)="selectActiveTaskList(taskList)"
(contextmenu)="$event.preventDefault(); onRightClick(taskList)"
matRipple>
<ng-container [ngPlural]="taskList.tasks?.length ?? 0">
<ng-template ngPluralCase="0">
Aucune tâche
</ng-template>
<ng-template ngPluralCase="1">
{{taskList.tasks?.length}} tâche
</ng-template>
<ng-template ngPluralCase="other">
{{taskList.tasks?.length}} tâches
</ng-template>
</ng-container>
<mat-icon *ngIf="isSelectionModeEnabled() && isSelected(taskList)" class="selection-icon">done</mat-icon>
</div>
{{taskList.name}}
</div>
</div>
<div class="selection-actions" *ngIf="isSelectionModeEnabled()">
<button class="stroked"
[disabled]="isThereMultipleTaskListsSelected()"
(click)="renameSelectedTaskList()">
Renommer
</button>
<button class="stroked alert"
(click)="deleteSelectedTaskLists()">Supprimer</button>
</div>

View File

@@ -0,0 +1,55 @@
.no-task-list {
display: flex;
margin: auto;
margin-top: 3rem;
font-size: 1.5rem;
}
.task-lists {
display: flex;
flex-direction: row;
flex-wrap: wrap;
margin: auto;
max-width: 80%;
.task-list-container {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
margin: 1rem;
.task-list {
width: 150px;
height: 150px;
display: flex;
justify-content: center;
align-items: center;
border-radius: .5rem;
background-color: aliceblue;
margin-bottom: .5rem;
position: relative;
.selection-icon {
position: absolute;
bottom: 0;
right: 0;
font-size: 3rem;
width: 3rem;
height: 3rem;
}
}
}
}
.selection-actions {
padding: .5rem;
position: fixed;
bottom: 0;
left: 0;
right: 0;
display: flex;
flex-grow: 1;
justify-content: space-between;
border-top: 1px solid #444;
}

View File

@@ -0,0 +1,93 @@
import { Component, NgZone, OnDestroy, OnInit } from '@angular/core';
import { MatDialog } from '@angular/material/dialog';
import { Subscription } from 'rxjs';
import { ConfirmDialogComponent, ConfirmDialogModel } from '../core/components/confirm-dialog/confirm-dialog.component';
import { TaskList } from '../core/entity/task-list';
import { TaskListService } from '../core/service/task-list.service';
import { AddTaskListComponent } from './add-task-list/add-task-list.component';
import { RenameTaskListComponent } from './rename-task-list/rename-task-list.component';
@Component({
selector: 'app-task-lists',
templateUrl: './task-lists.component.html',
styleUrls: ['./task-lists.component.scss']
})
export class TaskListsComponent implements OnInit, OnDestroy {
taskLists: TaskList[] = [];
private _storeSubscription?: Subscription;
private _subscriptions: Subscription[] = [];
constructor(
private _dialog: MatDialog,
private _taskListService: TaskListService
) {}
ngOnInit(): void {
this.taskLists = this._taskListService.getAll();
const storeSubscription = this._taskListService.store$.subscribe(store => {
this.taskLists = store.taskLists;
});
this._subscriptions.push(storeSubscription);
}
ngOnDestroy(): void {
this._subscriptions.forEach(subscription => subscription.unsubscribe());
}
selectActiveTaskList(taskList: TaskList): void {
if (this.isSelectionModeEnabled()) {
this._taskListService.selectTaskList(taskList);
} else {
this._taskListService.setActive(taskList);
}
}
onRightClick(taskList: TaskList): void {
this._taskListService.selectTaskList(taskList);
}
isSelected(taskList: TaskList): boolean {
return this._taskListService.isSelected(taskList);
}
isSelectionModeEnabled(): boolean {
return this._taskListService.isSelectionModeEnabled();
}
isThereMultipleTaskListsSelected(): boolean {
return this._taskListService.isThereMultipleTaskListsSelected();
}
renameSelectedTaskList(): void {
this._dialog.open(
RenameTaskListComponent,
{
width: '25rem'
}
);
}
deleteSelectedTaskLists(): void {
const confirmData = {
title: 'Supprimer les task-lists sélectionnées ?',
description: 'Une fois supprimées, les task-lists seront perdues définitivement.',
confirmButtonLabel: 'Supprimer les task-lists',
confirmButtonType: 'alert'
} as ConfirmDialogModel;
const dialogRef = this._dialog.open(
ConfirmDialogComponent,
{
width: '30rem',
data: confirmData
}
);
const afterDialogCloseSubscription = dialogRef.afterClosed().subscribe(result => {
if (result) {
this._taskListService.deleteSelectedTaskLists();
}
})
this._subscriptions.push(afterDialogCloseSubscription);
}
}

BIN
src/assets/images/to-do.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 948 B

After

Width:  |  Height:  |  Size: 1.0 KiB

View File

@@ -1,4 +1,139 @@
/* You can add global styles to this file, and also import other style files */ /* You can add global styles to this file, and also import other style files */
:root {
--primary: #fff;
--primary-text: #444444;
--primary-border: #ccc;
--primary-hover: #eee;
--secondary: #eee;
--secondary-hover: #c9c9c9;
--secondary-border: #bbb;
html, body { height: 100%; } --disabled: #eee;
body { margin: 0; font-family: Roboto, "Helvetica Neue", sans-serif; } --disabled-text: #aaa;
--transparent: rgba(0,0,0, 0);
--shadow-1: rgba(0, 0, 0, 0.2);
--shadow-2: rgba(0, 0, 0, 0.14);
--shadow-3: rgba(0, 0, 0, 0.12);
--shadow-hover: #777;
--alert: #eb1d3f;
--alert-text: #fff;
--alert-hover: #c20d2b;
--alert-border: #b91b35;
--selection: #185eb4;
}
html {
height: 100%;
}
body {
margin: 0;
font-family: Roboto, "Helvetica Neue", sans-serif;
padding-top: 3.2rem;
background-color: var(--primary);
}
.shadowed {
box-shadow: 0px 2px 1px -1px var(--shadow-1),
0px 1px 1px 0px var(--shadow-2),
0px 1px 3px 0px var(--shadow-3);
transition: box-shadow .2s ease-out;
&:hover {
box-shadow: 0 .2em .5em var(--shadow-hover);
}
}
a.no-style {
color: inherit;
text-decoration: none;
}
button {
border-style: solid;
border-width: .1rem;
border-color: var(--transparent);
border-radius: .2rem;
display: flex;
justify-content: center;
align-items: center;
padding: .5rem;
background-color: var(--transparent);
color: var(--primary-text);
transition: background-color .2s ease-out,
border-color .2s ease-out;
&:hover {
background-color: var(--primary-hover);
border-color: var(--primary-border);
}
&:disabled {
background-color: var(--disabled);
color: var(--disabled-text);
cursor: not-allowed;
}
&.icon {
width: 2rem;
height: 2rem;
min-width: 1rem;
min-height: 1rem;
padding: 0;
}
&.raised {
background-color: var(--primary); //#4a4a4a;
box-shadow: 0px 2px 1px -1px var(--shadow-1),
0px 1px 1px 0px var(--shadow-2),
0px 1px 3px 0px var(--shadow-3);
}
&.stroked {
border-color: var(--primary-border);
&.primary {
background-color: var(--primary);
&:hover {
background-color: var(--primary-hover);
}
}
&.secondary {
background-color: var(--secondary);
&:hover {
background-color: var(--secondary-hover)
}
}
}
&.alert {
background-color: var(--alert);
color: var(--alert-text);
&:hover {
background-color: var(--alert-hover);
border-color: var(--alert-border);
}
}
}
.dialog-actions {
margin-top: 1rem;
display: flex;
justify-content: space-between;
> * {
flex: 1 1 45%;
max-width: 45%;
}
}

14
todo.md Normal file
View File

@@ -0,0 +1,14 @@
# Fonctionnalités
- Ajouter une tâche
- Renommer la tâche
- Terminer une tâche
- Supprimer une tâche
- Saisir la note d'une tâche
- Ajouter des sous tâches à une autre
- Renommer des sous tâches d'une autre
- Terminer des sous tâches d'une autre
- Supprimer des sous tâches d'une autre
- J'aimerai "tagger" une tâche comme "asynchrone/bloquante" (où je dois attendre la fin, genre un build jenkins) et donc être notifié toutes les 5mins d'aller vérifier si la tâche est terminée.
- Quand je termine une sous tâche, je veux qu'elle soit archivée pour pouvoir la consulter à postériori
- J'aimerai définir un e API sur laquelle je pourrais sauverader mes tâches, à la manière de git