Compare commits

..

53 Commits
1.0.0 ... main

Author SHA1 Message Date
e0022b101e Merge pull request 'dev' (#62) from dev into main
Reviewed-on: #62
2025-06-08 11:57:27 +03:00
17df4a5812 Merge branch 'main' into dev 2025-06-08 11:56:31 +03:00
1957cbada0 Merge pull request 'release/v2.2.0' (#61) from release/v2.2.0 into dev
Reviewed-on: #61
2025-06-08 11:55:54 +03:00
c7214b04b4 fix release v2.2.0 prepare 2025-06-08 11:54:27 +03:00
409b520e69 feat(todo): release v2.2.0 (close #53)
- Обновил README с описанием новых возможностей и примерами использования
- Добавил детальные заметки о релизе в CHANGELOG
- Установил версию 2.2.0
2025-06-08 11:43:32 +03:00
fb5ede78ce Merge pull request 'refactor(cli): Вынес логику форматирования таблиц в utility-класс (close #52)' (#60) from feature/generic-utilities into dev
Reviewed-on: #60
2025-06-08 10:46:46 +03:00
a7eaa693bd refactor(cli): Вынес логику форматирования таблиц в utility-класс (close #52)
- Перенес метод formatWithTable из ListCommand в новый класс Formatter
- Обновил ListCommand и SearchCommand, чтобы они использовали Formatter.asTable
- Изменение улучшает переиспользование кода и разделение ответственности
2025-06-08 10:45:44 +03:00
8254f5292b Merge pull request 'refactor(cli): Обобщение CLI-команд для поддержки разных типов сущностей' (#59) from feature/update-commands into dev
Reviewed-on: #59
2025-06-08 08:38:10 +03:00
e64318df27 refactor(cli): Обобщение CLI-команд для поддержки разных типов сущностей
- Обновил движок CLI и команды для использования EntityService вместо TaskService
- Заменил TaskService на EntityService в реализациях команд
- Модифицировал логику команд для работы с любым типом сущности, расширяющим Identifiable
- Обновил сигнатуры типов и вызовы методов для соответствия новому интерфейсу
2025-06-08 08:37:31 +03:00
18acbd3b32 Merge pull request 'refactor(service): унификация операций с сущностями и обновление команды list (close #49)' (#58) from feature/refactor-taskservice into dev
Reviewed-on: #58
2025-06-07 16:27:18 +03:00
56038f62f7 refactor(service): унификация операций с сущностями и обновление команды list (close #49)
- Обновление ListCommand для использования метода getAll() вместо list()
- Рефакторинг EntityService для предоставления универсальных CRUD-операций
- Изменение TaskService для расширения EntityService и использования его методов
2025-06-07 16:26:41 +03:00
728cb4f84f Merge pull request 'feat(service): создан универсальный EntityService для операций CRUD (close #48)' (#57) from feature/entity-service into dev
Reviewed-on: #57
2025-06-07 15:31:24 +03:00
5140228f3e feat(service): создан универсальный EntityService для операций CRUD (close #48)
- Реализован универсальный класс EntityService для стандартных операций CRUD
- Определены методы для сохранения, получения и удаления сущностей
- Предоставлена основа для реализации специализированных сервисов
2025-06-07 15:21:55 +03:00
0174224089 Merge pull request 'refactor(repo): переименование и реструктуризация классов репозитория (close #47)' (#56) from feature/inmemory-repository into dev
Reviewed-on: #56
2025-06-06 20:34:20 +03:00
8c0cc0aa06 refactor(repo): переименование и реструктуризация классов репозитория (close #47)
- Переименован InMemoryTaskRepository в InMemoryRepository
- Обновлён TaskService для использования InMemoryRepository вместо TaskRepository
- Удалён интерфейс TaskRepository
- Обновлены импорты и ссылки на классы
2025-06-06 20:33:35 +03:00
da878ec81c Merge pull request 'refactor(repo): внедрение универсального интерфейса Repository и упрощение работы с моделью задач (close #46)' (#55) from feature/generic-repository into dev
Reviewed-on: #55
2025-06-06 19:31:56 +03:00
4cd38a9afe refactor(todo): внедрение универсального интерфейса Repository и упрощение работы с моделью задач (close #46)
- Добавлен интерфейс Repository для базовых CRUD-операций
- Обновлён TaskRepository для наследования от Repository<SimpleTask>
- Рефакторинг InMemoryTaskRepository для работы с SimpleTask
- Настройка ListCommand и TaskService для прямой работы с SimpleTask
- Удалено лишнее приведение типов и улучшена типобезопасность
2025-06-06 19:30:39 +03:00
2c7deaa9ae Merge pull request 'добавление интерфейса Identifiable и его реализация в Task (closes #45)' (#54) from feature/add-identifiable into dev
Reviewed-on: #54
2025-06-06 18:15:53 +03:00
8399ab0d58 добавление интерфейса Identifiable и его реализация в Task (closes #45)
- Создан интерфейс Identifiable с методом getId()
- Реализован Identifiable в классе Task
- Переопределён метод getId() в Task
- Сохранён существующий метод id() для обратной совместимости
2025-06-06 18:13:21 +03:00
f0b79f0db0 Merge pull request 'Рефакторинг класса ListCommand' (#44) from fix/cli/list-format-with-table into dev
Reviewed-on: #44
2025-06-04 16:46:41 +03:00
09cbf440ad Рефакторинг класса ListCommand
-Обновление версии проекта до 2.2.0-SNAPSHOT
-Рефакторинг класса ListCommand для объединения строк через класс StringBuilder
-Повышение читаемости и производительности метода formatWithTable
2025-06-04 16:45:37 +03:00
0f8f4037c9 Merge pull request 'dev' (#43) from dev into main
Reviewed-on: #43
2025-06-04 15:14:11 +03:00
65b3ddcc24 Merge branch 'main' into dev 2025-06-04 15:13:17 +03:00
20f9f7e8df Merge pull request 'Release v2.1.0' (#42) from release/2.1.0 into dev
Reviewed-on: #42
2025-06-04 15:08:40 +03:00
0f838ae4a0 Release v2.1.0 2025-06-04 15:02:31 +03:00
1f609abcb7 Merge pull request 'Добавлена функция поиска и улучшена команда вывода списка.' (#41) from feature/search into dev
Reviewed-on: #41
2025-06-04 14:06:02 +03:00
99e60947a6 Добавлена функция поиска и улучшена команда вывода списка.
- Добавлен класс SearchCommand в CliEngine
- Улучшен класс ListCommand, реализован новый метод formatWithTable
- Реализован метод поиска в сервисе задач (TaskService)
2025-06-04 13:25:39 +03:00
f380e12f67 Merge pull request 'dev' (#39) from dev into main
Reviewed-on: #39
2025-05-31 14:56:43 +03:00
28cdd22635 Merge pull request 'Pom.xml - добавлена конфигурация для сборки jar файла. Добавлена инструкция по сборке и запуску.' (#38) from fix/pom into dev
Reviewed-on: #38
2025-05-31 14:56:18 +03:00
0776e220b3 Pom.xml - добавлена конфигурация для сборки jar файла. Добавлена инструкция по сборке и запуску. 2025-05-31 14:55:28 +03:00
8700123455 Release version 2.0 2025-05-30 15:02:56 +03:00
54d80bfc32 Merge pull request 'Release version 2.0' (#35) from release/2.0 into dev
Reviewed-on: #35
2025-05-29 22:17:07 +03:00
9b98399b6e Release version 2.0 2025-05-29 22:16:25 +03:00
4b4ac7f0c9 Merge pull request 'Добавлена команда удаления задач в CliEngine.' (#34) from feature/cli/delete-command into dev
Reviewed-on: #34
2025-05-29 21:51:40 +03:00
3e89dc1613 Добавлена команда удаления задач в CliEngine. 2025-05-29 21:51:09 +03:00
5100e3791b Merge pull request 'feature/cli/complete-command' (#33) from feature/cli/complete-command into dev
Reviewed-on: #33
2025-05-29 21:42:40 +03:00
a35758bc6f Добавлена команда завершения задач в CliEngine. 2025-05-29 21:42:05 +03:00
adc7fe67f8 Добавлена проверка длины названия задачи в команде создания. Теперь, если название превышает 30 символов, выводится сообщение об ошибке. 2025-05-29 19:08:08 +03:00
518e2e55bc Добавлена команда для отображения списка задач в CLI. Изменён модификатор доступа для класса Data в SimpleTask на public. 2025-05-29 19:08:08 +03:00
644709a3f0 Merge pull request 'Добавлен метод has для проверки существования задачи в InMemoryTaskRepository и TaskService.' (#32) from feature/service/has-with-id into dev
Reviewed-on: #32
2025-05-29 18:58:21 +03:00
38ac198eb6 Добавлен метод has для проверки существования задачи в InMemoryTaskRepository и TaskService. 2025-05-29 18:57:31 +03:00
782325ecc8 Merge pull request 'Добавлена команда добавления задачи. Обновлены методы обработки команд в CLI: заменены example и description на usage, добавлено шаблонное сообщение о…' (#31) from feature/cli/create-command into dev
Reviewed-on: #31
2025-05-28 17:56:17 +03:00
ac7e84f133 Добавлена команда добавления задачи. Обновлены методы обработки команд в CLI: заменены example и description на usage, добавлено шаблонное сообщение об ошибке. Упрощен вывод доступных команд. 2025-05-28 17:54:47 +03:00
dd1ec57cd3 Merge pull request 'Теперь приложение запускается через CLI Engine, что улучшает взаимодействие с пользователем.' (#29) from feature/cli into dev
Reviewed-on: #29
2025-05-28 16:38:49 +03:00
abe0a08b2e Теперь приложение запускается через CLI Engine, что улучшает взаимодействие с пользователем. 2025-05-28 16:38:12 +03:00
f25725476b Merge pull request 'Реализованы методы создания, завершения и удаления задач через сервис.' (#28) from feature/task-service into dev
Reviewed-on: #28
2025-05-26 22:38:45 +03:00
f4483ff951 Реализованы методы создания, завершения и удаления задач через сервис. 2025-05-26 22:38:16 +03:00
af15988c62 Merge pull request 'feature/task-manager/init' (#27) from feature/task-manager/init into dev
Reviewed-on: #27
2025-05-26 21:44:19 +03:00
5b4e948846 Добавлена реализация InMemoryTaskRepository для управления задачами. 2025-05-26 21:40:02 +03:00
9312b760c7 Добавлена реализация InMemoryTaskRepository для управления задачами. Обновлён метод main в TodoApp для сохранения, поиска, завершения и удаления задач. Добавлен метод id() в класс Task для получения идентификатора задачи. 2025-05-26 21:34:59 +03:00
6fbf3064c5 Merge pull request 'init/2.0.0' (#23) from init/2.0.0 into dev
Reviewed-on: #23
2025-05-25 17:57:49 +03:00
7aa394ec08 Создана основа ооп для задач 2025-05-25 17:34:48 +03:00
768f7ddf37 Подготовлен пустой проект для разработке v2 с чистого листа, с новой архитектурой. 2025-05-21 18:47:43 +03:00
24 changed files with 671 additions and 377 deletions

1
.gitignore vendored
View File

@ -1,2 +1,3 @@
todo/target/
.vscode/
.lingma/

47
CHANGELOG.md Normal file
View 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
View 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

View File

@ -6,10 +6,40 @@
<groupId>ru.kamask.pet</groupId>
<artifactId>todo</artifactId>
<version>1.0-SNAPSHOT</version>
<version>2.2.0</version>
<properties>
<maven.compiler.release>24</maven.compiler.release>
<maven.compiler.target>24</maven.compiler.target>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</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>

View File

@ -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);
}
}
}

View File

@ -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;
}
}

View File

@ -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Напишите название дела (320 символов): ");
title = scanner.nextLine().trim();
if (title.length() >= 3 && title.length() <= 20)
break;
System.out.print("\nОшибка: Название должно содержать от 3 до 20 символов.\опробуйте снова: ");
} 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("Ошибка: используйте цифры.\овторите ввод:");
}
} while (true);
for (int i : allowedInts)
if (i == input)
return input;
System.out.print("Ошибка: укажите номер выбранного пункта.\овторите ввод:");
} 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]);
}
}
}

View 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();
}
}

View 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("\обро пожаловать в Список задач 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();
}
}

View 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();
}

View File

@ -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 - должен быть числом.");
}
}
}

View 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() + "азваниеадачи>", "Добавление новой задачи");
}
}

View 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 - должен быть числом.");
}
}
}

View 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(), "Список всех задач.");
}
}

View 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, "Не найдено задач, соответствующих запросу."));
}
}

View File

@ -0,0 +1,5 @@
package ru.kamask.pet.todo.model;
public interface DataBuilder<T> {
T data();
}

View File

@ -0,0 +1,5 @@
package ru.kamask.pet.todo.model;
public interface Identifiable {
int getId();
}

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

View 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();
}

View File

@ -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);
}
}

View 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);
}

View File

@ -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);
}
}

View File

@ -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();
}
}

View 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();
}
}