Compare commits

...

27 Commits
2.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
20 changed files with 281 additions and 95 deletions

1
.gitignore vendored
View File

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

View File

@ -1,14 +1,47 @@
# Журнал изменений # Журнал изменений
Все заметные изменения проекта будут документироваться в этом файле. Все заметные изменения проекта будут документироваться в этом файле.
Формат основан на [Keep a Changelog](https://keepachangelog.com/ru/1.0.0/), Формат основан на [Keep a Changelog](https://keepachangelog.com/ru/1.0.0/),
и этот проект придерживается [Semantic Versioning](https://semver.org/spec/v2.0.0.html). и этот проект придерживается [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 ## [2.0.0] - 2025-05-29
### Добавлено ### Добавлено
- Базовая функциональность для управления задачами - Базовая функциональность для управления задачами
- Команды: create, list, delete - Команды: create, list, delete, complete
- Консольный интерфейс для взаимодействия с приложением - Консольный интерфейс для взаимодействия с приложением
- In-memory хранилище задач - In-memory хранилище задач

View File

@ -1,34 +1,92 @@
# TodoApp # TodoApp
## Описание ## Описание
TodoApp - это консольное приложение для управления списком задач, разработанное на Java в качестве учебного проекта. Приложение позволяет создавать, просматривать, отмечать как выполненные и удалять задачи через интерфейс командной строки.
TodoApp - это консольное приложение для управления списком задач, разработанное на Java в качестве учебного проекта. Приложение построено на универсальной архитектуре с использованием generics, что позволяет легко расширять функционал для работы с различными типами сущностей.
### Особенности
- Консольный интерфейс с интуитивными командами
- Универсальная архитектура на базе `EntityService` и `Repository<T>`
- Поддержка поиска задач по тексту
- Красивое табличное отображение данных
- Типобезопасность через использование generics
## Требования ## Требования
- Java 24 или выше - Java 24 или выше
- Maven 3.8.x или выше - Maven 3.8.x или выше
## Установка и запуск ## Установка и запуск
```bash ```bash
cd ./todo cd ./todo
mvn clean package mvn clean package
java -jar ./target/todo.jar java -jar ./target/todo.jar
``` ```
## Использование
### Доступные команды ### Доступные команды
- `create <названиеадачи>` - Создать новую задачу - `create <названиеадачи>` - Создать новую задачу
- `list` - Показать список всех задач - `list` - Показать список всех задач в табличном виде
- `complete <id>` - Отметить задачу как выполненную - `complete <id>` - Отметить задачу как выполненную
- `delete <id>` - Удалить задачу - `delete <id>` - Удалить задачу по ID
- `search <текст>` - Найти задачи по тексту (минимум 3 символа)
- `help` - Показать список всех команд
- `exit` - Выйти из приложения - `exit` - Выйти из приложения
## Структура проекта
- `model` - Модели данных (Task, SimpleTask) ## Пример использования
- `repo` - Репозитории для хранения задач
- `service` - Бизнес-логика ```shell
- `cli` - Компоненты интерфейса командной строки 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.0
Актуальная версия: 2.2.0
### История изменений
- 2.2.0 - Универсальная архитектура с generics, утилиты форматирования
- 2.1.0 - Добавлена команда поиска, улучшено отображение
- 2.0.0 - Базовый функционал управления задачами
Подробная история изменений доступна в CHANGELOG.md
### Разработка
Проект использует современные подходы Java-разработки:
- Generic programming для типобезопасности
- Разделение ответственности через слоистую архитектуру
- Command pattern для CLI-команд
- Repository pattern для работы с данными
## Лицензия ## Лицензия
[MIT](LICENSE)
MIT

View File

@ -6,7 +6,7 @@
<groupId>ru.kamask.pet</groupId> <groupId>ru.kamask.pet</groupId>
<artifactId>todo</artifactId> <artifactId>todo</artifactId>
<version>2.0-SNAPSHOT</version> <version>2.2.0</version>
<properties> <properties>
<maven.compiler.release>24</maven.compiler.release> <maven.compiler.release>24</maven.compiler.release>

View File

@ -3,12 +3,13 @@ package ru.kamask.pet.todo;
import java.io.IOException; import java.io.IOException;
import ru.kamask.pet.todo.cli.CliEngine; import ru.kamask.pet.todo.cli.CliEngine;
import ru.kamask.pet.todo.repo.InMemoryTaskRepository; import ru.kamask.pet.todo.model.SimpleTask;
import ru.kamask.pet.todo.repo.InMemoryRepository;
import ru.kamask.pet.todo.service.TaskService; import ru.kamask.pet.todo.service.TaskService;
public class TodoApp { public class TodoApp {
public static void main(String[] args) throws IOException { public static void main(String[] args) throws IOException {
var service = new TaskService(new InMemoryTaskRepository()); var service = new TaskService(new InMemoryRepository<SimpleTask>());
var cli = new CliEngine(service); var cli = new CliEngine(service);
cli.start(); cli.start();

View File

@ -7,21 +7,22 @@ import java.util.Arrays;
import java.util.HashMap; import java.util.HashMap;
import java.util.Optional; import java.util.Optional;
import ru.kamask.pet.todo.service.TaskService; import ru.kamask.pet.todo.model.Identifiable;
import ru.kamask.pet.todo.service.EntityService;
public class CliEngine { public class CliEngine {
private HashMap<String, Command> registry = new HashMap<>(); private HashMap<String, Command> registry = new HashMap<>();
private TaskService service; private EntityService<? extends Identifiable> service;
private BufferedReader reader; private BufferedReader reader;
public CliEngine(TaskService service) { public CliEngine(EntityService<? extends Identifiable> service) {
this.service = service; this.service = service;
reader = new BufferedReader(new InputStreamReader(System.in)); reader = new BufferedReader(new InputStreamReader(System.in));
initializeCommands(); initializeCommands();
} }
public void start() throws IOException { public void start() throws IOException {
System.out.println("\обро пожаловать в Список задач 2.0!"); System.out.println("\обро пожаловать в Список задач 2.2!");
System.out.println("Введите help для просмтора доступных команд или exit для выхода."); System.out.println("Введите help для просмтора доступных команд или exit для выхода.");
while (true) { while (true) {
System.out.print("\ntodo> "); System.out.print("\ntodo> ");
@ -46,6 +47,7 @@ public class CliEngine {
registerCommand(new CreateCommand()); registerCommand(new CreateCommand());
registerCommand(new ListCommand()); registerCommand(new ListCommand());
registerCommand(new CompleteCommand()); registerCommand(new CompleteCommand());
registerCommand(new SearchCommand());
registerCommand(new DeleteCommand()); registerCommand(new DeleteCommand());
} }

View File

@ -2,13 +2,14 @@ package ru.kamask.pet.todo.cli;
import java.util.Optional; import java.util.Optional;
import ru.kamask.pet.todo.service.TaskService; import ru.kamask.pet.todo.model.Identifiable;
import ru.kamask.pet.todo.service.EntityService;
public interface Command { public interface Command {
String templateUsage = " %-30s // %s"; String templateUsage = " %-30s // %s";
String errorMessage = "Не корректно введена команда. Введите help для спарвки."; String errorMessage = "Не корректно введена команда. Введите help для спарвки.";
Optional<String> handle(String[] args, TaskService service); Optional<String> handle(String[] args, EntityService<? extends Identifiable> service);
String name(); String name();

View File

@ -2,6 +2,8 @@ package ru.kamask.pet.todo.cli;
import java.util.Optional; 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.service.TaskService;
public class CompleteCommand implements Command { public class CompleteCommand implements Command {
@ -16,14 +18,16 @@ public class CompleteCommand implements Command {
} }
@Override @Override
public Optional<String> handle(String[] args, TaskService service) { public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
if (args.length != 1) if (args.length != 1)
return Optional.of(Command.errorMessage); return Optional.of(Command.errorMessage);
TaskService taskService = (TaskService) service;
try { try {
int id = Integer.parseInt(args[0]); int id = Integer.parseInt(args[0]);
return Optional.of(service.complete(id) return Optional.of(taskService.complete(id)
? "Задача ID-%d выполнена.".formatted(id) ? "Задача ID-%d выполнена.".formatted(id)
: "Задача ID-%d не найдена.".formatted(id)); : "Задача ID-%d не найдена.".formatted(id));

View File

@ -2,21 +2,24 @@ package ru.kamask.pet.todo.cli;
import java.util.Optional; 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.service.TaskService;
public class CreateCommand implements Command { public class CreateCommand implements Command{
@Override @Override
public String name() { public String name() {
return "create"; return "create";
} }
@Override @Override
public Optional<String> handle(String[] args, TaskService service) { public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
if (args.length > 0) { if (args.length > 0) {
var title = String.join(" ", args); var title = String.join(" ", args);
if (title.length() > 30) if (title.length() > 30)
return Optional.of("Ошибка: максимальная длинна названия задачи 30 символов."); return Optional.of("Ошибка: максимальная длинна названия задачи 30 символов.");
service.create(title); TaskService taskService = (TaskService) service;
taskService.create(title);
return Optional.of(String.format("Задача \"%s\" успешно добавлена!", title)); return Optional.of(String.format("Задача \"%s\" успешно добавлена!", title));
} }
return Optional.of(Command.errorMessage); return Optional.of(Command.errorMessage);

View File

@ -2,9 +2,10 @@ package ru.kamask.pet.todo.cli;
import java.util.Optional; import java.util.Optional;
import ru.kamask.pet.todo.service.TaskService; import ru.kamask.pet.todo.model.Identifiable;
import ru.kamask.pet.todo.service.EntityService;
public class DeleteCommand implements Command { public class DeleteCommand implements Command{
@Override @Override
public String name() { public String name() {
return "delete"; return "delete";
@ -16,7 +17,7 @@ public class DeleteCommand implements Command {
} }
@Override @Override
public Optional<String> handle(String[] args, TaskService service) { public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
if (args.length != 1) if (args.length != 1)
return Optional.of(Command.errorMessage); return Optional.of(Command.errorMessage);

View File

@ -2,9 +2,10 @@ package ru.kamask.pet.todo.cli;
import java.util.Optional; import java.util.Optional;
import ru.kamask.pet.todo.model.SimpleTask; import ru.kamask.pet.todo.model.Identifiable;
import ru.kamask.pet.todo.model.Task; import ru.kamask.pet.todo.service.EntityService;
import ru.kamask.pet.todo.service.TaskService; import ru.kamask.pet.todo.service.TaskService;
import ru.kamask.pet.todo.util.Formatter;
public class ListCommand implements Command { public class ListCommand implements Command {
@Override @Override
@ -13,23 +14,13 @@ public class ListCommand implements Command {
} }
@Override @Override
public Optional<String> handle(String[] args, TaskService service) { public Optional<String> handle(String[] args, EntityService<? extends Identifiable> service) {
if (args.length > 0) if (args.length > 0)
return Optional.of(Command.errorMessage); return Optional.of(Command.errorMessage);
String template = "%-2s | %-30s | %s\n"; TaskService taskService = (TaskService) service;
String res = "";
res += String.format(template, "ID", "Название задачи", "Статус"); return Optional.of(Formatter.asTable(taskService.getAll()));
res += "-".repeat(50) + "\n";
if (service.list().size() == 0)
return Optional.of(res + "\nСписок задач пуст.");
for (Task task : service.list()) {
SimpleTask.Data data = ((SimpleTask) task).data();
res += String.format(template, data.id(), data.title(), data.done() ? "выполнено" : "не выполнено");
}
return Optional.of(res);
} }
@Override @Override

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 Identifiable {
int getId();
}

View File

@ -1,6 +1,6 @@
package ru.kamask.pet.todo.model; package ru.kamask.pet.todo.model;
public abstract class Task { public abstract class Task implements Identifiable{
private static int nextId = 1; private static int nextId = 1;
protected int id; protected int id;
@ -11,10 +11,15 @@ public abstract class Task {
this.title = title; this.title = title;
} }
public int id() { @Override
public int getId() {
return id; return id;
} }
public int id() {
return getId();
}
@Override @Override
public String toString() { public String toString() {
return String.format("Задача: id - %d, title: \"%s\"", id, title); return String.format("Задача: id - %d, title: \"%s\"", id, title);

View File

@ -6,23 +6,23 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Optional; import java.util.Optional;
import ru.kamask.pet.todo.model.Task; import ru.kamask.pet.todo.model.Identifiable;
public class InMemoryTaskRepository implements TaskRepository { public class InMemoryRepository<T extends Identifiable> implements Repository<T> {
private Map<Integer, Task> storage = new HashMap<>(); private Map<Integer, T> storage = new HashMap<>();
@Override @Override
public void save(Task task) { public void save(T obj) {
storage.put(task.id(), task); storage.put(obj.getId(), obj);
} }
@Override @Override
public Optional<Task> findById(int id) { public Optional<T> findById(int id) {
return Optional.ofNullable(storage.get(id)); return Optional.ofNullable(storage.get(id));
} }
@Override @Override
public List<Task> findAll() { public List<T> findAll() {
return new ArrayList<>(storage.values()); return new ArrayList<>(storage.values());
} }

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

@ -1,18 +0,0 @@
package ru.kamask.pet.todo.repo;
import java.util.List;
import java.util.Optional;
import ru.kamask.pet.todo.model.Task;
public interface TaskRepository {
void save(Task task);
Optional<Task> findById(int id);
List<Task> 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

@ -1,29 +1,23 @@
package ru.kamask.pet.todo.service; package ru.kamask.pet.todo.service;
import java.util.List; import java.util.List;
import java.util.Optional;
import ru.kamask.pet.todo.model.Task;
import ru.kamask.pet.todo.model.SimpleTask; import ru.kamask.pet.todo.model.SimpleTask;
import ru.kamask.pet.todo.repo.TaskRepository; import ru.kamask.pet.todo.repo.Repository;
public class TaskService { public class TaskService extends EntityService<SimpleTask> {
private final TaskRepository repo;
public TaskService(TaskRepository repo) { public TaskService(Repository<SimpleTask> repo) {
this.repo = repo; super(repo);
} }
public void create(String title) { public void create(String title) {
repo.save(new SimpleTask(title)); super.save(new SimpleTask(title));
} }
public Optional<Task> getById(int id) {
return repo.findById(id);
}
public boolean complete(int id) { public boolean complete(int id) {
var taskOpt = repo.findById(id); var taskOpt = super.getById(id);
if (taskOpt.isPresent()) { if (taskOpt.isPresent()) {
taskOpt.get().markAsCompleted(); taskOpt.get().markAsCompleted();
return true; return true;
@ -31,15 +25,10 @@ public class TaskService {
return false; return false;
} }
public List<Task> list() {
return repo.findAll();
}
public void remove(int id) { public List<SimpleTask> search(String query) {
repo.delete(id); return super.getAll().stream()
} .filter(task -> task.data().title().contains(query))
.toList();
public boolean has(int id) {
return repo.has(id);
} }
} }

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