Compare commits
53 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e0022b101e | |||
| 17df4a5812 | |||
| 1957cbada0 | |||
| c7214b04b4 | |||
| 409b520e69 | |||
| fb5ede78ce | |||
| a7eaa693bd | |||
| 8254f5292b | |||
| e64318df27 | |||
| 18acbd3b32 | |||
| 56038f62f7 | |||
| 728cb4f84f | |||
| 5140228f3e | |||
| 0174224089 | |||
| 8c0cc0aa06 | |||
| da878ec81c | |||
| 4cd38a9afe | |||
| 2c7deaa9ae | |||
| 8399ab0d58 | |||
| f0b79f0db0 | |||
| 09cbf440ad | |||
| 0f8f4037c9 | |||
| 65b3ddcc24 | |||
| 20f9f7e8df | |||
| 0f838ae4a0 | |||
| 1f609abcb7 | |||
| 99e60947a6 | |||
| f380e12f67 | |||
| 28cdd22635 | |||
| 0776e220b3 | |||
| 8700123455 | |||
| 54d80bfc32 | |||
| 9b98399b6e | |||
| 4b4ac7f0c9 | |||
| 3e89dc1613 | |||
| 5100e3791b | |||
| a35758bc6f | |||
| adc7fe67f8 | |||
| 518e2e55bc | |||
| 644709a3f0 | |||
| 38ac198eb6 | |||
| 782325ecc8 | |||
| ac7e84f133 | |||
| dd1ec57cd3 | |||
| abe0a08b2e | |||
| f25725476b | |||
| f4483ff951 | |||
| af15988c62 | |||
| 5b4e948846 | |||
| 9312b760c7 | |||
| 6fbf3064c5 | |||
| 7aa394ec08 | |||
| 768f7ddf37 |
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,2 +1,3 @@
|
|||||||
todo/target/
|
todo/target/
|
||||||
.vscode/
|
.vscode/
|
||||||
|
.lingma/
|
||||||
47
CHANGELOG.md
Normal file
47
CHANGELOG.md
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
# Журнал изменений
|
||||||
|
|
||||||
|
Все заметные изменения проекта будут документироваться в этом файле.
|
||||||
|
Формат основан на [Keep a Changelog](https://keepachangelog.com/ru/1.0.0/),
|
||||||
|
и этот проект придерживается [Semantic Versioning](https://semver.org/spec/v2.0.0.html)
|
||||||
|
|
||||||
|
## [2.2.0] - 2025-06-08
|
||||||
|
|
||||||
|
### Добавлено
|
||||||
|
- Универсальный интерфейс `Identifiable` для всех сущностей с ID
|
||||||
|
- Универсальный класс `EntityService` для стандартных CRUD-операций
|
||||||
|
- Универсальный интерфейс `Repository<T>` для базовых операций с данными
|
||||||
|
- Утилитарный класс `Formatter` с методом `asTable()` для форматирования таблиц
|
||||||
|
|
||||||
|
### Изменено
|
||||||
|
- **[BREAKING]** Обобщение CLI-команд для поддержки разных типов сущностей через `EntityService`
|
||||||
|
- Рефакторинг `TaskService` для расширения `EntityService` вместо собственной реализации
|
||||||
|
- Переименование `InMemoryTaskRepository` → `InMemoryRepository` с поддержкой generics
|
||||||
|
- Унификация работы с моделью `Task` через интерфейс `Identifiable`
|
||||||
|
- Вынесение логики форматирования таблиц из `ListCommand` в отдельный utility-класс
|
||||||
|
- Обновление `ListCommand` и `SearchCommand` для использования `Formatter.asTable()`
|
||||||
|
- Замена метода `list()` на `getAll()` в сервисном слое
|
||||||
|
|
||||||
|
### Технические улучшения
|
||||||
|
- Улучшена типобезопасность через использование generics
|
||||||
|
- Повышена переиспользуемость кода за счёт вынесения общей логики
|
||||||
|
- Упрощена архитектура через унификацию интерфейсов
|
||||||
|
- Удалено дублирование кода форматирования между командами
|
||||||
|
|
||||||
|
## [2.1.0] - 2025-06-01
|
||||||
|
|
||||||
|
### Добавлено
|
||||||
|
- Команда `search <текст>` для поиска задач по тексту (реализован класс `SearchCommand`)
|
||||||
|
- Метод `search(String query)` в `TaskService` для поддержки поиска задач
|
||||||
|
- Рефакторинг: вынесение общего функционала отображения задач в отдельный метод `formatWithTable()` в классе `ListCommand`
|
||||||
|
|
||||||
|
### Изменено
|
||||||
|
- Обновлены внутренние зависимости между компонентами CLI (`SearchCommand`, `ListCommand`) и моделью данных (`Task`, `SimpleTask`)
|
||||||
|
- Добавлена обработка минимальной длины поискового запроса (3 символа) в `SearchCommand`
|
||||||
|
|
||||||
|
## [2.0.0] - 2025-05-29
|
||||||
|
|
||||||
|
### Добавлено
|
||||||
|
- Базовая функциональность для управления задачами
|
||||||
|
- Команды: create, list, delete, complete
|
||||||
|
- Консольный интерфейс для взаимодействия с приложением
|
||||||
|
- In-memory хранилище задач
|
||||||
92
README.md
Normal file
92
README.md
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
# TodoApp
|
||||||
|
|
||||||
|
## Описание
|
||||||
|
|
||||||
|
TodoApp - это консольное приложение для управления списком задач, разработанное на Java в качестве учебного проекта. Приложение построено на универсальной архитектуре с использованием generics, что позволяет легко расширять функционал для работы с различными типами сущностей.
|
||||||
|
|
||||||
|
|
||||||
|
### Особенности
|
||||||
|
- Консольный интерфейс с интуитивными командами
|
||||||
|
- Универсальная архитектура на базе `EntityService` и `Repository<T>`
|
||||||
|
- Поддержка поиска задач по тексту
|
||||||
|
- Красивое табличное отображение данных
|
||||||
|
- Типобезопасность через использование generics
|
||||||
|
|
||||||
|
|
||||||
|
## Требования
|
||||||
|
|
||||||
|
- Java 24 или выше
|
||||||
|
- Maven 3.8.x или выше
|
||||||
|
|
||||||
|
|
||||||
|
## Установка и запуск
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ./todo
|
||||||
|
mvn clean package
|
||||||
|
java -jar ./target/todo.jar
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
## Использование
|
||||||
|
|
||||||
|
### Доступные команды
|
||||||
|
- `create <название_задачи>` - Создать новую задачу
|
||||||
|
- `list` - Показать список всех задач в табличном виде
|
||||||
|
- `complete <id>` - Отметить задачу как выполненную
|
||||||
|
- `delete <id>` - Удалить задачу по ID
|
||||||
|
- `search <текст>` - Найти задачи по тексту (минимум 3 символа)
|
||||||
|
- `help` - Показать список всех команд
|
||||||
|
- `exit` - Выйти из приложения
|
||||||
|
|
||||||
|
|
||||||
|
## Пример использования
|
||||||
|
|
||||||
|
```shell
|
||||||
|
todo> create Изучить Java generics
|
||||||
|
Задача "Изучить Java generics" успешно добавлена!
|
||||||
|
|
||||||
|
todo> create Написать документацию
|
||||||
|
Задача "Написать документацию" успешно добавлена!
|
||||||
|
|
||||||
|
todo> list
|
||||||
|
|
||||||
|
ID │ Название задачи │ Статус
|
||||||
|
--------------------------------------------------
|
||||||
|
1 │ Изучить Java generics │ В процессе
|
||||||
|
2 │ Написать документацию │ В процессе
|
||||||
|
|
||||||
|
|
||||||
|
todo> complete 1
|
||||||
|
|
||||||
|
Задача ID-1 выполнена.
|
||||||
|
|
||||||
|
todo> search Java
|
||||||
|
|
||||||
|
ID │ Название задачи │ Статус
|
||||||
|
--------------------------------------------------
|
||||||
|
1 │ Изучить Java generics │ Выполнена
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
## Версии
|
||||||
|
|
||||||
|
Актуальная версия: 2.2.0
|
||||||
|
|
||||||
|
### История изменений
|
||||||
|
- 2.2.0 - Универсальная архитектура с generics, утилиты форматирования
|
||||||
|
- 2.1.0 - Добавлена команда поиска, улучшено отображение
|
||||||
|
- 2.0.0 - Базовый функционал управления задачами
|
||||||
|
|
||||||
|
Подробная история изменений доступна в CHANGELOG.md
|
||||||
|
|
||||||
|
### Разработка
|
||||||
|
Проект использует современные подходы Java-разработки:
|
||||||
|
- Generic programming для типобезопасности
|
||||||
|
- Разделение ответственности через слоистую архитектуру
|
||||||
|
- Command pattern для CLI-команд
|
||||||
|
- Repository pattern для работы с данными
|
||||||
|
|
||||||
|
## Лицензия
|
||||||
|
|
||||||
|
MIT
|
||||||
32
todo/pom.xml
32
todo/pom.xml
@ -6,10 +6,40 @@
|
|||||||
|
|
||||||
<groupId>ru.kamask.pet</groupId>
|
<groupId>ru.kamask.pet</groupId>
|
||||||
<artifactId>todo</artifactId>
|
<artifactId>todo</artifactId>
|
||||||
<version>1.0-SNAPSHOT</version>
|
<version>2.2.0</version>
|
||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<maven.compiler.release>24</maven.compiler.release>
|
<maven.compiler.release>24</maven.compiler.release>
|
||||||
|
<maven.compiler.target>24</maven.compiler.target>
|
||||||
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<finalName>todo</finalName>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-jar-plugin</artifactId>
|
||||||
|
<version>3.4.2</version>
|
||||||
|
<configuration>
|
||||||
|
<archive>
|
||||||
|
<manifest>
|
||||||
|
<mainClass>ru.kamask.pet.todo.TodoApp</mainClass>
|
||||||
|
</manifest>
|
||||||
|
</archive>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.apache.maven.plugins</groupId>
|
||||||
|
<artifactId>maven-compiler-plugin</artifactId>
|
||||||
|
<version>3.14.0</version>
|
||||||
|
<configuration>
|
||||||
|
<source>24</source>
|
||||||
|
<target>24</target>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
|
||||||
</project>
|
</project>
|
||||||
@ -1,85 +0,0 @@
|
|||||||
package ru.kamask.pet;
|
|
||||||
|
|
||||||
import java.time.LocalDate;
|
|
||||||
|
|
||||||
class Task {
|
|
||||||
static int taskCount = 0;
|
|
||||||
|
|
||||||
enum TaskIdGenerator {
|
|
||||||
INSTANCE;
|
|
||||||
|
|
||||||
private int currentId = 1;
|
|
||||||
|
|
||||||
int nextId() {
|
|
||||||
return currentId++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private int id;
|
|
||||||
private String title;
|
|
||||||
private String description;
|
|
||||||
private TaskStatus status;
|
|
||||||
private LocalDate createdAt;
|
|
||||||
|
|
||||||
Task(String title, String description) {
|
|
||||||
id = TaskIdGenerator.INSTANCE.nextId();
|
|
||||||
status = TaskStatus.NEW;
|
|
||||||
createdAt = LocalDate.now();
|
|
||||||
taskCount++;
|
|
||||||
|
|
||||||
this.title = title;
|
|
||||||
this.description = description;
|
|
||||||
};
|
|
||||||
|
|
||||||
static void printTotalTasksCreated() {
|
|
||||||
System.out.println("Колличество дел: " + taskCount);
|
|
||||||
}
|
|
||||||
|
|
||||||
int getId() {
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
|
|
||||||
String getTitle() {
|
|
||||||
return title;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean isCompleted() {
|
|
||||||
return status == TaskStatus.COMPLETED;
|
|
||||||
}
|
|
||||||
|
|
||||||
void markCompleted() {
|
|
||||||
status = TaskStatus.COMPLETED;
|
|
||||||
}
|
|
||||||
|
|
||||||
void markInProgress() {
|
|
||||||
status = TaskStatus.IN_PROGRESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
void markCanceled() {
|
|
||||||
status = TaskStatus.CANCELLED;
|
|
||||||
}
|
|
||||||
|
|
||||||
TaskStatus getStatus() {
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static class TaskPrinter {
|
|
||||||
static void print(Task task) {
|
|
||||||
String template = """
|
|
||||||
|
|
||||||
──────────────────────────────
|
|
||||||
Номер: %-3d
|
|
||||||
Дата создания: %s
|
|
||||||
Статус: %s
|
|
||||||
Название: %-20s
|
|
||||||
------------------------------
|
|
||||||
%s
|
|
||||||
|
|
||||||
""";
|
|
||||||
;
|
|
||||||
String stringCompleted = task.status.getColorCode() + task.status.getDescription() + "\u001B[0m";
|
|
||||||
System.out.printf(template, task.id, task.createdAt, stringCompleted, task.title, task.description);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@ -1,41 +0,0 @@
|
|||||||
package ru.kamask.pet;
|
|
||||||
|
|
||||||
public enum TaskStatus {
|
|
||||||
NEW("новое") {
|
|
||||||
@Override
|
|
||||||
String getColorCode() {
|
|
||||||
return "\u001B[34m";
|
|
||||||
}
|
|
||||||
},
|
|
||||||
IN_PROGRESS("в работе") {
|
|
||||||
@Override
|
|
||||||
String getColorCode() {
|
|
||||||
return "\u001B[35m";
|
|
||||||
}
|
|
||||||
},
|
|
||||||
COMPLETED("сделано") {
|
|
||||||
@Override
|
|
||||||
String getColorCode() {
|
|
||||||
return "\u001B[32m";
|
|
||||||
}
|
|
||||||
},
|
|
||||||
CANCELLED("отменено") {
|
|
||||||
@Override
|
|
||||||
String getColorCode() {
|
|
||||||
return "\u001B[31m";
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
abstract String getColorCode();
|
|
||||||
|
|
||||||
private final String description;
|
|
||||||
|
|
||||||
TaskStatus(String description) {
|
|
||||||
this.description = description;
|
|
||||||
}
|
|
||||||
|
|
||||||
String getDescription() {
|
|
||||||
return description;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@ -1,250 +0,0 @@
|
|||||||
package ru.kamask.pet;
|
|
||||||
|
|
||||||
import java.util.Scanner;
|
|
||||||
|
|
||||||
public class TodoApp {
|
|
||||||
|
|
||||||
private static TaskManager taskManager;
|
|
||||||
private static Task[] tasks = new Task[10];
|
|
||||||
private static int tasksCounter = 0;
|
|
||||||
private static Scanner scanner = new Scanner(System.in);
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
|
||||||
|
|
||||||
TodoApp app = new TodoApp();
|
|
||||||
taskManager = app.new TaskManager();
|
|
||||||
|
|
||||||
app.run();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private void run() {
|
|
||||||
while (true) {
|
|
||||||
Task.printTotalTasksCreated();
|
|
||||||
String menu = """
|
|
||||||
|
|
||||||
┌──────────────────────────┐
|
|
||||||
| [1] Добавить дело |
|
|
||||||
| [2] Список дел |
|
|
||||||
| |
|
|
||||||
| [0] Выйти из программы |
|
|
||||||
└──────────────────────────┘
|
|
||||||
|
|
||||||
Введите номер пункта меню:""";
|
|
||||||
|
|
||||||
switch (requestIntFromInput(menu, new int[] { 0, 1, 2 })) {
|
|
||||||
case 1 -> displayCreateTask();
|
|
||||||
case 2 -> displayTasks();
|
|
||||||
case 0 -> {
|
|
||||||
scanner.close();
|
|
||||||
System.exit(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void displayCreateTask() {
|
|
||||||
|
|
||||||
if (Task.taskCount >= 9) {
|
|
||||||
System.out.println("\nОшибка: Достигнут лимит в 10 дел.\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
String title;
|
|
||||||
String description;
|
|
||||||
|
|
||||||
do {
|
|
||||||
System.out.print("\nНапишите название дела (3–20 символов): ");
|
|
||||||
title = scanner.nextLine().trim();
|
|
||||||
if (title.length() >= 3 && title.length() <= 20)
|
|
||||||
break;
|
|
||||||
|
|
||||||
System.out.print("\nОшибка: Название должно содержать от 3 до 20 символов.\nПопробуйте снова: ");
|
|
||||||
} while (true);
|
|
||||||
|
|
||||||
System.out.print("\nНaпишите описание дела: ");
|
|
||||||
description = scanner.nextLine().trim();
|
|
||||||
|
|
||||||
taskManager.addTask(new Task(title, description));
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void displayTask(Task task) {
|
|
||||||
Task.TaskPrinter.print(task);
|
|
||||||
|
|
||||||
String firstOption = switch (task.getStatus()) {
|
|
||||||
case NEW, CANCELLED -> "Начать";
|
|
||||||
case IN_PROGRESS -> "Выполнено";
|
|
||||||
case COMPLETED -> "Доделать";
|
|
||||||
};
|
|
||||||
|
|
||||||
int input = requestIntFromInput("[1]" + firstOption + " [2]Отменить [0]Главное меню\n\nВвод:",
|
|
||||||
new int[] { 0, 1, 2 });
|
|
||||||
|
|
||||||
switch (input) {
|
|
||||||
case 0:
|
|
||||||
return;
|
|
||||||
case 1: {
|
|
||||||
switch (task.getStatus()) {
|
|
||||||
case NEW, CANCELLED, COMPLETED -> taskManager.markInProgress(task.getId());
|
|
||||||
case IN_PROGRESS -> taskManager.markCompleted(task.getId());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 2:
|
|
||||||
taskManager.markCanceled(task.getId());
|
|
||||||
}
|
|
||||||
|
|
||||||
displayTask(task);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void displayTasks() {
|
|
||||||
|
|
||||||
if (tasksCounter == 0) {
|
|
||||||
int input = requestIntFromInput("""
|
|
||||||
|
|
||||||
Список дел пуст.
|
|
||||||
|
|
||||||
[1] Добавить дело
|
|
||||||
[0] Выйти из программы
|
|
||||||
|
|
||||||
Введите номер пункта:""", new int[] { 0, 1 });
|
|
||||||
|
|
||||||
if (input == 1) {
|
|
||||||
displayCreateTask();
|
|
||||||
} else
|
|
||||||
System.exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
System.out.println("""
|
|
||||||
Список дел:
|
|
||||||
|
|
||||||
""");
|
|
||||||
|
|
||||||
taskManager.printTasks();
|
|
||||||
|
|
||||||
int[] variantsInput = new int[tasksCounter + 1];
|
|
||||||
for (int i = 0; i < tasksCounter; i++)
|
|
||||||
variantsInput[i] = tasks[i].getId();
|
|
||||||
variantsInput[tasksCounter] = 0;
|
|
||||||
|
|
||||||
int input = requestIntFromInput("""
|
|
||||||
|
|
||||||
Введите номер дела или 0 для возврата в меню:""", variantsInput);
|
|
||||||
|
|
||||||
if (input == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
displayTask(getTaskById(input));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Task getTaskById(int id) {
|
|
||||||
for (int i = 0; i < Task.taskCount; i++) {
|
|
||||||
if (tasks[i].getId() == id)
|
|
||||||
return tasks[i];
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static int requestIntFromInput(String template, int[] allowedInts) {
|
|
||||||
do {
|
|
||||||
System.out.print(template);
|
|
||||||
int input;
|
|
||||||
do {
|
|
||||||
if (scanner.hasNextInt()) {
|
|
||||||
input = scanner.nextInt();
|
|
||||||
scanner.nextLine();
|
|
||||||
break;
|
|
||||||
} else {
|
|
||||||
scanner.next();
|
|
||||||
System.out.print("Ошибка: используйте цифры.\nПовторите ввод:");
|
|
||||||
}
|
|
||||||
} while (true);
|
|
||||||
|
|
||||||
for (int i : allowedInts)
|
|
||||||
if (i == input)
|
|
||||||
return input;
|
|
||||||
System.out.print("Ошибка: укажите номер выбранного пункта.\nПовторите ввод:");
|
|
||||||
} while (true);
|
|
||||||
}
|
|
||||||
|
|
||||||
interface TaskAction {
|
|
||||||
default void execute(Task task) {
|
|
||||||
};
|
|
||||||
|
|
||||||
default void start(Task task) {
|
|
||||||
};
|
|
||||||
|
|
||||||
default void cancel(Task task) {
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
private class TaskManager {
|
|
||||||
void addTask(Task task) {
|
|
||||||
tasks[tasksCounter++] = task;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean markCompleted(int id) {
|
|
||||||
for (int i = 0; i < tasksCounter; i++)
|
|
||||||
if (tasks[i].getId() == id) {
|
|
||||||
TaskAction action = new TaskAction() {
|
|
||||||
@Override
|
|
||||||
public void execute(Task task) {
|
|
||||||
task.markCompleted();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
action.execute(tasks[i]);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean markInProgress(int id) {
|
|
||||||
for (int i = 0; i < tasksCounter; i++)
|
|
||||||
if (tasks[i].getId() == id) {
|
|
||||||
TaskAction action = new TaskAction() {
|
|
||||||
@Override
|
|
||||||
public void start(Task task) {
|
|
||||||
task.markInProgress();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
action.start(tasks[i]);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean markCanceled(int id) {
|
|
||||||
for (int i = 0; i < tasksCounter; i++)
|
|
||||||
if (tasks[i].getId() == id) {
|
|
||||||
TaskAction action = new TaskAction() {
|
|
||||||
@Override
|
|
||||||
public void cancel(Task task) {
|
|
||||||
task.markCanceled();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
action.cancel(tasks[i]);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void printTasks() {
|
|
||||||
class ShortTaskPrinter {
|
|
||||||
void print(Task task) {
|
|
||||||
int id = task.getId();
|
|
||||||
String title = task.getTitle();
|
|
||||||
TaskStatus status = task.getStatus();
|
|
||||||
|
|
||||||
String stringCompleted = status.getColorCode() + status.getDescription() + "\u001B[0m";
|
|
||||||
System.out.printf("%-3d | %-20s | %s\n", id, title, stringCompleted);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ShortTaskPrinter printer = new ShortTaskPrinter();
|
|
||||||
for (int i = 0; i < Task.taskCount; i++)
|
|
||||||
printer.print(tasks[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
18
todo/src/main/java/ru/kamask/pet/todo/TodoApp.java
Normal file
18
todo/src/main/java/ru/kamask/pet/todo/TodoApp.java
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
package ru.kamask.pet.todo;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.cli.CliEngine;
|
||||||
|
import ru.kamask.pet.todo.model.SimpleTask;
|
||||||
|
import ru.kamask.pet.todo.repo.InMemoryRepository;
|
||||||
|
import ru.kamask.pet.todo.service.TaskService;
|
||||||
|
|
||||||
|
public class TodoApp {
|
||||||
|
public static void main(String[] args) throws IOException {
|
||||||
|
var service = new TaskService(new InMemoryRepository<SimpleTask>());
|
||||||
|
var cli = new CliEngine(service);
|
||||||
|
|
||||||
|
cli.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
71
todo/src/main/java/ru/kamask/pet/todo/cli/CliEngine.java
Normal file
71
todo/src/main/java/ru/kamask/pet/todo/cli/CliEngine.java
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
|
||||||
|
public class CliEngine {
|
||||||
|
private HashMap<String, Command> registry = new HashMap<>();
|
||||||
|
private EntityService<? extends Identifiable> service;
|
||||||
|
private BufferedReader reader;
|
||||||
|
|
||||||
|
public CliEngine(EntityService<? extends Identifiable> service) {
|
||||||
|
this.service = service;
|
||||||
|
reader = new BufferedReader(new InputStreamReader(System.in));
|
||||||
|
initializeCommands();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void start() throws IOException {
|
||||||
|
System.out.println("\nДобро пожаловать в Список задач 2.2!");
|
||||||
|
System.out.println("Введите help для просмтора доступных команд или exit для выхода.");
|
||||||
|
while (true) {
|
||||||
|
System.out.print("\ntodo> ");
|
||||||
|
var input = reader.readLine().trim();
|
||||||
|
System.out.println();
|
||||||
|
switch (input) {
|
||||||
|
case "help" -> handleHelp();
|
||||||
|
case "exit" -> System.exit(0);
|
||||||
|
default -> handleCommand(input).ifPresentOrElse(System.out::println, () -> {
|
||||||
|
System.out.println("Не известная команда. Повторите ввод.\n");
|
||||||
|
handleHelp();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void registerCommand(Command command) {
|
||||||
|
registry.put(command.name(), command);
|
||||||
|
}
|
||||||
|
|
||||||
|
void initializeCommands() {
|
||||||
|
registerCommand(new CreateCommand());
|
||||||
|
registerCommand(new ListCommand());
|
||||||
|
registerCommand(new CompleteCommand());
|
||||||
|
registerCommand(new SearchCommand());
|
||||||
|
registerCommand(new DeleteCommand());
|
||||||
|
}
|
||||||
|
|
||||||
|
void handleHelp() {
|
||||||
|
registry.values().stream().map(handler -> handler.usage()).forEach(System.out::println);
|
||||||
|
System.out.printf(Command.templateUsage, "help", "Вот этот список команд\n");
|
||||||
|
System.out.printf(Command.templateUsage, "exit", "Выход, но лучше не надо)\n");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Optional<String> handleCommand(String input) {
|
||||||
|
var parts = input.split("\\s+");
|
||||||
|
var command = parts[0];
|
||||||
|
var args = Arrays.copyOfRange(parts, 1, parts.length);
|
||||||
|
|
||||||
|
var handlerOpt = Optional.ofNullable(registry.get(command));
|
||||||
|
if (handlerOpt.isPresent())
|
||||||
|
return handlerOpt.get().handle(args, service);
|
||||||
|
return Optional.empty();
|
||||||
|
}
|
||||||
|
}
|
||||||
17
todo/src/main/java/ru/kamask/pet/todo/cli/Command.java
Normal file
17
todo/src/main/java/ru/kamask/pet/todo/cli/Command.java
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
|
||||||
|
public interface Command {
|
||||||
|
String templateUsage = " %-30s // %s";
|
||||||
|
String errorMessage = "Не корректно введена команда. Введите help для спарвки.";
|
||||||
|
|
||||||
|
Optional<String> handle(String[] args, EntityService<? extends Identifiable> service);
|
||||||
|
|
||||||
|
String name();
|
||||||
|
|
||||||
|
String usage();
|
||||||
|
}
|
||||||
@ -0,0 +1,38 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
import ru.kamask.pet.todo.service.TaskService;
|
||||||
|
|
||||||
|
public class CompleteCommand implements Command {
|
||||||
|
@Override
|
||||||
|
public String name() {
|
||||||
|
return "complete";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String usage() {
|
||||||
|
return String.format(Command.templateUsage, name() + " <ID>", "Отметить задачу как \"выполнена\".");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
|
||||||
|
if (args.length != 1)
|
||||||
|
return Optional.of(Command.errorMessage);
|
||||||
|
|
||||||
|
TaskService taskService = (TaskService) service;
|
||||||
|
|
||||||
|
try {
|
||||||
|
int id = Integer.parseInt(args[0]);
|
||||||
|
|
||||||
|
return Optional.of(taskService.complete(id)
|
||||||
|
? "Задача ID-%d выполнена.".formatted(id)
|
||||||
|
: "Задача ID-%d не найдена.".formatted(id));
|
||||||
|
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
return Optional.of("ID - должен быть числом.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
32
todo/src/main/java/ru/kamask/pet/todo/cli/CreateCommand.java
Normal file
32
todo/src/main/java/ru/kamask/pet/todo/cli/CreateCommand.java
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
import ru.kamask.pet.todo.service.TaskService;
|
||||||
|
|
||||||
|
public class CreateCommand implements Command{
|
||||||
|
@Override
|
||||||
|
public String name() {
|
||||||
|
return "create";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
|
||||||
|
if (args.length > 0) {
|
||||||
|
var title = String.join(" ", args);
|
||||||
|
if (title.length() > 30)
|
||||||
|
return Optional.of("Ошибка: максимальная длинна названия задачи 30 символов.");
|
||||||
|
TaskService taskService = (TaskService) service;
|
||||||
|
taskService.create(title);
|
||||||
|
return Optional.of(String.format("Задача \"%s\" успешно добавлена!", title));
|
||||||
|
}
|
||||||
|
return Optional.of(Command.errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String usage() {
|
||||||
|
return String.format(templateUsage, name() + " <название_задачи>", "Добавление новой задачи");
|
||||||
|
}
|
||||||
|
}
|
||||||
36
todo/src/main/java/ru/kamask/pet/todo/cli/DeleteCommand.java
Normal file
36
todo/src/main/java/ru/kamask/pet/todo/cli/DeleteCommand.java
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
|
||||||
|
public class DeleteCommand implements Command{
|
||||||
|
@Override
|
||||||
|
public String name() {
|
||||||
|
return "delete";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String usage() {
|
||||||
|
return String.format(Command.templateUsage, name() + " <ID>", "Удалить задачу.");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
|
||||||
|
if (args.length != 1)
|
||||||
|
return Optional.of(Command.errorMessage);
|
||||||
|
|
||||||
|
try {
|
||||||
|
int id = Integer.parseInt(args[0]);
|
||||||
|
if (!service.has(id))
|
||||||
|
return Optional.of("Задача ID-%d не найдена.".formatted(id));
|
||||||
|
service.remove(id);
|
||||||
|
return Optional.of("Задача ID-%d удалена.".formatted(id));
|
||||||
|
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
return Optional.of("ID - должен быть числом.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
30
todo/src/main/java/ru/kamask/pet/todo/cli/ListCommand.java
Normal file
30
todo/src/main/java/ru/kamask/pet/todo/cli/ListCommand.java
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
import ru.kamask.pet.todo.service.TaskService;
|
||||||
|
import ru.kamask.pet.todo.util.Formatter;
|
||||||
|
|
||||||
|
public class ListCommand implements Command {
|
||||||
|
@Override
|
||||||
|
public String name() {
|
||||||
|
return "list";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
|
||||||
|
if (args.length > 0)
|
||||||
|
return Optional.of(Command.errorMessage);
|
||||||
|
|
||||||
|
TaskService taskService = (TaskService) service;
|
||||||
|
|
||||||
|
return Optional.of(Formatter.asTable(taskService.getAll()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String usage() {
|
||||||
|
return String.format(templateUsage, name(), "Список всех задач.");
|
||||||
|
}
|
||||||
|
}
|
||||||
34
todo/src/main/java/ru/kamask/pet/todo/cli/SearchCommand.java
Normal file
34
todo/src/main/java/ru/kamask/pet/todo/cli/SearchCommand.java
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
package ru.kamask.pet.todo.cli;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.service.EntityService;
|
||||||
|
import ru.kamask.pet.todo.service.TaskService;
|
||||||
|
import ru.kamask.pet.todo.util.Formatter;
|
||||||
|
|
||||||
|
public class SearchCommand implements Command {
|
||||||
|
@Override
|
||||||
|
public String name() {
|
||||||
|
return "search";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String usage() {
|
||||||
|
return String.format(templateUsage, name() + " <текст>", "Поиск задач по тексту (минимум 3 символа)");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
|
||||||
|
if (args.length != 1)
|
||||||
|
return Optional.of(Command.errorMessage);
|
||||||
|
if (args[0].length() < 3)
|
||||||
|
return Optional.of("Длина запроса должна быть не менее 3 символов.");
|
||||||
|
|
||||||
|
TaskService taskService = (TaskService) service;
|
||||||
|
|
||||||
|
var matchTask = taskService.search(args[0]);
|
||||||
|
|
||||||
|
return Optional.of(Formatter.asTable(matchTask, "Не найдено задач, соответствующих запросу."));
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,5 @@
|
|||||||
|
package ru.kamask.pet.todo.model;
|
||||||
|
|
||||||
|
public interface DataBuilder<T> {
|
||||||
|
T data();
|
||||||
|
}
|
||||||
@ -0,0 +1,5 @@
|
|||||||
|
package ru.kamask.pet.todo.model;
|
||||||
|
|
||||||
|
public interface Identifiable {
|
||||||
|
int getId();
|
||||||
|
}
|
||||||
35
todo/src/main/java/ru/kamask/pet/todo/model/SimpleTask.java
Normal file
35
todo/src/main/java/ru/kamask/pet/todo/model/SimpleTask.java
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
package ru.kamask.pet.todo.model;
|
||||||
|
|
||||||
|
public class SimpleTask extends Task implements DataBuilder<SimpleTask.Data> {
|
||||||
|
private boolean done = false;
|
||||||
|
|
||||||
|
public SimpleTask(String title) {
|
||||||
|
super(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void markAsCompleted() {
|
||||||
|
done();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isCompleted() {
|
||||||
|
return isDone();
|
||||||
|
}
|
||||||
|
|
||||||
|
void done() {
|
||||||
|
done = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean isDone() {
|
||||||
|
return done;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Data data() {
|
||||||
|
return new Data(id, title, done);
|
||||||
|
}
|
||||||
|
|
||||||
|
public record Data(int id, String title, boolean done) {
|
||||||
|
}
|
||||||
|
}
|
||||||
32
todo/src/main/java/ru/kamask/pet/todo/model/Task.java
Normal file
32
todo/src/main/java/ru/kamask/pet/todo/model/Task.java
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
package ru.kamask.pet.todo.model;
|
||||||
|
|
||||||
|
public abstract class Task implements Identifiable{
|
||||||
|
private static int nextId = 1;
|
||||||
|
|
||||||
|
protected int id;
|
||||||
|
protected String title;
|
||||||
|
|
||||||
|
Task(String title) {
|
||||||
|
this.id = nextId++;
|
||||||
|
this.title = title;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int id() {
|
||||||
|
return getId();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return String.format("Задача: id - %d, title: \"%s\"", id, title);
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract public void markAsCompleted();
|
||||||
|
|
||||||
|
abstract public boolean isCompleted();
|
||||||
|
|
||||||
|
}
|
||||||
@ -0,0 +1,37 @@
|
|||||||
|
package ru.kamask.pet.todo.repo;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
|
||||||
|
public class InMemoryRepository<T extends Identifiable> implements Repository<T> {
|
||||||
|
private Map<Integer, T> storage = new HashMap<>();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void save(T obj) {
|
||||||
|
storage.put(obj.getId(), obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Optional<T> findById(int id) {
|
||||||
|
return Optional.ofNullable(storage.get(id));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> findAll() {
|
||||||
|
return new ArrayList<>(storage.values());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void delete(int id) {
|
||||||
|
storage.remove(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean has(int id) {
|
||||||
|
return storage.containsKey(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
14
todo/src/main/java/ru/kamask/pet/todo/repo/Repository.java
Normal file
14
todo/src/main/java/ru/kamask/pet/todo/repo/Repository.java
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
package ru.kamask.pet.todo.repo;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
|
||||||
|
public interface Repository<T extends Identifiable> {
|
||||||
|
void save(T obj);
|
||||||
|
Optional<T> findById(int id);
|
||||||
|
List<T> findAll();
|
||||||
|
void delete(int id);
|
||||||
|
boolean has(int id);
|
||||||
|
}
|
||||||
@ -0,0 +1,35 @@
|
|||||||
|
package ru.kamask.pet.todo.service;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.Identifiable;
|
||||||
|
import ru.kamask.pet.todo.repo.Repository;
|
||||||
|
|
||||||
|
public class EntityService<T extends Identifiable> {
|
||||||
|
private final Repository<T> repo;
|
||||||
|
|
||||||
|
protected EntityService(Repository<T> repo) {
|
||||||
|
this.repo = repo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void save(T obj) {
|
||||||
|
repo.save(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Optional<T> getById(int id) {
|
||||||
|
return repo.findById(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<T> getAll() {
|
||||||
|
return repo.findAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void remove(int id) {
|
||||||
|
repo.delete(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean has(int id) {
|
||||||
|
return repo.has(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,34 @@
|
|||||||
|
package ru.kamask.pet.todo.service;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.SimpleTask;
|
||||||
|
import ru.kamask.pet.todo.repo.Repository;
|
||||||
|
|
||||||
|
public class TaskService extends EntityService<SimpleTask> {
|
||||||
|
|
||||||
|
public TaskService(Repository<SimpleTask> repo) {
|
||||||
|
super(repo);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void create(String title) {
|
||||||
|
super.save(new SimpleTask(title));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public boolean complete(int id) {
|
||||||
|
var taskOpt = super.getById(id);
|
||||||
|
if (taskOpt.isPresent()) {
|
||||||
|
taskOpt.get().markAsCompleted();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public List<SimpleTask> search(String query) {
|
||||||
|
return super.getAll().stream()
|
||||||
|
.filter(task -> task.data().title().contains(query))
|
||||||
|
.toList();
|
||||||
|
}
|
||||||
|
}
|
||||||
27
todo/src/main/java/ru/kamask/pet/todo/util/Formatter.java
Normal file
27
todo/src/main/java/ru/kamask/pet/todo/util/Formatter.java
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
package ru.kamask.pet.todo.util;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import ru.kamask.pet.todo.model.SimpleTask;
|
||||||
|
|
||||||
|
public class Formatter {
|
||||||
|
public static String asTable(List<SimpleTask> tasks){
|
||||||
|
return asTable(tasks, "Список задач пуст.");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String asTable(List<SimpleTask> tasks, String msgIfEmpty){
|
||||||
|
String template = "%-2s | %-30s | %s\n";
|
||||||
|
var res = new StringBuilder(String.format(template, "ID", "Название задачи", "Статус"));
|
||||||
|
res.append("-".repeat(50) + "\n");
|
||||||
|
|
||||||
|
if (tasks.size() == 0)
|
||||||
|
return res.append("\n" + msgIfEmpty).toString();
|
||||||
|
|
||||||
|
for (SimpleTask task : tasks) {
|
||||||
|
SimpleTask.Data data = task.data();
|
||||||
|
res.append(String.format(template, data.id(), data.title(), data.done() ? "выполнено" : "не выполнено"));
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user