Skip to content

Commit

Permalink
Add Lab 07 assignment
Browse files Browse the repository at this point in the history
  • Loading branch information
100yo committed Nov 22, 2024
1 parent e06c9b5 commit 1fcdc12
Show file tree
Hide file tree
Showing 4 changed files with 193 additions and 0 deletions.
193 changes: 193 additions & 0 deletions 07-io-streams-and-files/lab/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
# Photo Edge Detector :framed_picture:

Тази седмица задачата ни е да разработим модули на библиотека за обработка на изображения. Библиотеката ще може да работи с изображения в различни графични формати (JPEG, PNG, BMP) и ще прилага различни трансформации върху тях. В първата версия, клиентът иска функционалност за конвертиране на цветно изображение в черно-бяло и възможност за откриване на ръбовете в изображение.

![Maserati Edge Detected](../lecture/images/07.12-maserati-edge-detected.png)

Библиотеката има два основни компонента:

- Алгоритми за обработка на изображения
- Мениджър за файловата система за зареждане и съхраняване на изображения.

## Основни интерфейси и класове

Ето интерфейсите и класовете, които трябва да реализирате. Някои класове и интерфейси са частично дефинирани, за да ви насочат.

### ImageAlgorithm

Интерфейсът `ImageAlgorithm` представлява алгоритъм за обработка на изображения.

```java
package bg.sofia.uni.fmi.mjt.imagekit.algorithm;

import java.awt.image.BufferedImage;

/**
* Represents an algorithm that processes images.
*/
public interface ImageAlgorithm {

/**
* Applies the image processing algorithm to the given image.
*
* @param image the image to be processed
* @return BufferedImage the processed image of type (TYPE_INT_RGB)
*/
BufferedImage process(BufferedImage image);
}
```

Разгледате документацията на класа [BufferedImage](https://docs.oracle.com/en/java/javase/23/docs/api/java.desktop/java/awt/image/BufferedImage.html), защото ще ви потрябва на няколко места на създавате негови инстанции и да ползвате API-то му. Когато създавате негова инстанция, ползвайте конструктора `BufferedImage(int width, int height, int imageType)`, а като трети аргумент подавайте винаги `BufferedImage.TYPE_INT_RGB`.

### GrayscaleAlgorithm

Интерфейсът `GrayscaleAlgorithm` е маркерен интерфейс за алгоритми за конвертиране в черно-бяло изображение.

```java
package bg.sofia.uni.fmi.mjt.imagekit.algorithm.grayscale;

import bg.sofia.uni.fmi.mjt.imagekit.algorithm.ImageAlgorithm;

public interface GrayscaleAlgorithm extends ImageAlgorithm {
}
```

### EdgeDetectionAlgorithm

Интерфейсът `EdgeDetectionAlgorithm` е друг маркерен интерфейс, този път за алгоритми за откриване на ръбове.

```java
package bg.sofia.uni.fmi.mjt.imagekit.algorithm.detection;

import bg.sofia.uni.fmi.mjt.imagekit.algorithm.ImageAlgorithm;

public interface EdgeDetectionAlgorithm extends ImageAlgorithm {
}
```

### FileSystemImageManager

Интерфейсът `FileSystemImageManager` управлява зареждането и съхраняването на изображения от файловата система.

```java
package bg.sofia.uni.fmi.mjt.imagekit.filesystem;

import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.List;

/**
* An interface for loading images from the file system.
* The supported image formats are JPEG, PNG, and BMP.
*/
public interface FileSystemImageManager {

/**
* Loads a single image from the given file path.
*
* @param imageFile the file containing the image.
* @return the loaded BufferedImage.
* @throws IllegalArgumentException if the file is null
* @throws IOException if the file does not exist, is not a regular file,
* or is not in one of the supported formats.
*/
BufferedImage loadImage(File imageFile) throws IOException;

/**
* Loads all images from the specified directory.
*
* @param imagesDirectory the directory containing the images.
* @return A list of BufferedImages representing the loaded images.
* @throws IllegalArgumentException if the directory is null.
* @throws IOException if the directory does not exist, is not a directory,
* or contains files that are not in one of the supported formats.
*/
List<BufferedImage> loadImagesFromDirectory(File imagesDirectory) throws IOException;

/**
* Saves the given image to the specified file path.
*
* @param image the image to save.
* @param imageFile the file to save the image to.
* @throws IllegalArgumentException if the image or file is null.
* @throws IOException if the file already exists or the parent directory does not exist.
*/
void saveImage(BufferedImage image, File imageFile) throws IOException;
}
```

### LocalFileSystemImageManager

Класът `LocalFileSystemImageManager` има публичен конструктор по подразбиране, имлпементира интерфейса `FileSystemImageManager` и предоставя методи за зареждане и съхраняване на изображения.

:point_right: Подсказка: в имплементацията ще ви влезе в употреба класът [javax.imageio.ImageIO](https://docs.oracle.com/en/java/javase/23/docs/api/java.desktop/javax/imageio/ImageIO.html)

### LuminosityGrayscale

Класът `LuminosityGrayscale` също има публичен конструктор по подразбиране, имплементира интерфейса `GrayscaleAlgorithm` и прилага черно-бяло конвертиране, използвайки [*Mетода на осветеност (Luminocity method)*](https://www.johndcook.com/blog/2009/08/24/algorithms-convert-color-grayscale/).
В литературата се срещат и други стойности на коефициентите във формулата, но ние ще използваме класическите, които са посочени и в линкнатата по-горе статия: `0.21 R + 0.72 G + 0.07 B`.

### SobelEdgeDetection

Класът `SobelEdgeDetection` има публичен конструктор `SobelEdgeDetection(ImageAlgorithm grayscaleAlgorithm)`, имплементира интерфейса `EdgeDetectionAlgorithm` и прилага един от класическите алгоритми за откриване на ръбове в изображения, [*Оператор на Sobel*](https://en.wikipedia.org/wiki/Sobel_operator), известен също като *Оператор на Sobel–Feldman* и *Филтър на Sobel*.

За да се запознаете с алгоритъма, може да разгледате [тази статия](https://cse442-17f.github.io/Sobel-Laplacian-and-Canny-Edge-Detection-Algorithms/), която съдържа и интерактивна визуализация, която ще ви помогне да схванете как работи той.

### Пример за използване

Ето един прост пример, как може да се използва библиотеката:

```java
FileSystemImageManager fsImageManager = new LocalFileSystemImageManager();

BufferedImage image = fsImageManager.loadImage(new File("kitten.png"));

ImageAlgorithm grayscaleAlgorithm = new LuminosityGrayscale();
BufferedImage grayscaleImage = grayscaleAlgorithm.process(image);

ImageAlgorithm sobelEdgeDetection = new SobelEdgeDetection(grayscaleAlgorithm);
BufferedImage edgeDetectedImage = sobelEdgeDetection.process(image);

fsImageManager.saveImage(grayscaleImage, new File("kitten-grayscale.png"));
fsImageManager.saveImage(edgeDetectedImage, new File("kitten-edge-detected.png"));
```

### Тестване

Тази задача събужда изследователя във вас, затова първо тествайте библиотеката си локално, с някоe от [нашите изображения](resources/), или с произволна ваша картинка. После помислете, как може да я тествате автоматично и създайте подходящи unit тестове. Не се фиксирайте върху code coverage-a, а как да изтествате базовата функционалност на библиотеката.

## Пакети

Спазвайте имената на пакетите на всички по-горе описани класове, тъй като в противен случай решението ви няма да може да бъде тествано от грейдъра.

```
src
└── bg.sofia.uni.fmi.mjt.imagekit
├── algorithm
│ ├── detection
│ │ ├── EdgeDetectionAlgorithm.java
│ │ └── SobelEdgeDetection.java
│ ├── grayscale
│ │ ├── GrayscaleAlgorithm.java
│ │ └── LuminosityGrayscale.java
│ ├── ImageAlgorithm.java
│ └── (...)
├── filesystem
│ ├── FileSystemImageManager.java
│ ├── LocalFileSystemImageManager.java
│ └── (...)
└── (...)
test
└── bg.sofia.uni.fmi.mjt.imagekit
└── (...)
```

## Забележки

- В грейдъра качете директориите src и test като ги селектирате и двете. Друг вариант е да ги сложите в общ .zip архив и да качите него
- Не качвайте jar-ките на JUnit и Mockito библиотеките. На грейдъра ги има, няма смисъл решението ви да набъбва излишно.
- Не качвайте и евентуални изображения, с които си тествате локално библиотеката

Успех в първия ви изследователския поход из интересната и необятна област на [computer vision](https://en.wikipedia.org/wiki/Computer_vision), и приятно кодене! :rocket:
Binary file added 07-io-streams-and-files/lab/resources/car.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added 07-io-streams-and-files/lab/resources/kitten.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.

0 comments on commit 1fcdc12

Please sign in to comment.