Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/insightsoftwareconsortium/itkcolornormalization
Structure preserving color normalization on H&E images using a reference image
https://github.com/insightsoftwareconsortium/itkcolornormalization
color-normalization cpp image-processing insight-toolkit itk itk-module python
Last synced: 23 days ago
JSON representation
Structure preserving color normalization on H&E images using a reference image
- Host: GitHub
- URL: https://github.com/insightsoftwareconsortium/itkcolornormalization
- Owner: InsightSoftwareConsortium
- License: apache-2.0
- Created: 2020-05-13T14:08:00.000Z (over 4 years ago)
- Default Branch: master
- Last Pushed: 2024-09-07T20:43:25.000Z (4 months ago)
- Last Synced: 2024-10-29T17:38:46.754Z (2 months ago)
- Topics: color-normalization, cpp, image-processing, insight-toolkit, itk, itk-module, python
- Language: C++
- Homepage:
- Size: 151 KB
- Stars: 18
- Watchers: 18
- Forks: 5
- Open Issues: 6
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# ITKColorNormalization
[![Apache 2.0 License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://github.com/InsightSoftwareConsortium/ITKColorNormalization/blob/master/LICENSE) [![DOI](https://zenodo.org/badge/263648073.svg)](https://zenodo.org/badge/latestdoi/263648073) [![Build, test, package status](https://github.com/InsightSoftwareConsortium/ITKColorNormalization/workflows/Build,%20test,%20package/badge.svg)](https://github.com/InsightSoftwareConsortium/ITKColorNormalization/actions?query=workflow%3A%22Build%2C+test%2C+package%22)
## Overview
This [Insight Toolkit (ITK)](https://itk.org/) module performs Structure Preserving Color Normalization on an [H & E](https://en.wikipedia.org/wiki/H%26E_stain) image using a reference image. The module is in C++ and is also packaged for [Python](https://www.python.org/).
H & E ([hematoxylin](https://en.wikipedia.org/wiki/Haematoxylin) and [eosin](https://en.wikipedia.org/wiki/Eosin)) are stains used to color parts of cells in a histological image, often for medical diagnosis. Hematoxylin is a compound that stains cell nuclei a purple-blue color. Eosin is a compound that stains extracellular matrix and cytoplasm pink. However, the exact color of purple-blue or pink can vary from image to image, and this can make comparison of images difficult. This routine addresses the issue by re-coloring one image (the first image supplied to the routine) using the color scheme of a reference image (the second image supplied to the routine). The technique requires that the images have at least 3 colors, such as red, green, and blue (RGB).
Structure Preserving Color Normalization is a technique described in [Vahadane et al., 2016](https://doi.org/10.1109/TMI.2016.2529665) and modified in [Ramakrishnan et al., 2019](https://arxiv.org/abs/1901.03088). The idea is to model the color of an image pixel as something close to pure white, which is reduced in intensity in a color-specific way via an optical absorption model that depends upon the amounts of hematoxylin and eosin that are present. [Non-negative matrix factorization](https://en.wikipedia.org/wiki/Non-negative_matrix_factorization) is used on each analyzed image to simultaneously derive the amount of hematoxylin and eosin stain at each pixel and the image-wide effective colors of each stain.
The implementation in ITK accelerates the non-negative matrix factorization by choosing the initial estimate for the color absorption characteristics using a technique mimicking that presented in [Arora et al., 2013](https://proceedings.mlr.press/v28/arora13.html) and modified in [Newberg et al., 2018](https://doi.org/10.1371/journal.pone.0193067). This approach finds a good solution for a non-negative matrix factorization by first transforming it to the problem of finding a convex hull for a set of points in a cloud.
The lead developer of InsightSoftwareConsortium/ITKColorNormalization is [Lee Newberg](https://github.com/Leengit/).
## Try it online
We have set up a demonstration of this using [Binder](www.mybinder.org) that anyone can try. Click here: [![Binder](https://mybinder.org/badge_logo.svg)](https://mybinder.org/v2/gh/InsightSoftwareConsortium/ITKColorNormalization/master?filepath=examples%2FITKColorNormalization.ipynb)
## Installation for Python
[![PyPI Version](https://img.shields.io/pypi/v/itk-spcn.svg)](https://pypi.python.org/pypi/itk-spcn)
ITKColorNormalization and all its dependencies can be easily installed with [Python wheels](https://blog.kitware.com/itk-is-on-pypi-pip-install-itk-is-here/). Wheels have been generated for macOS, Linux, and Windows and several versions of Python, 3.6, 3.7, 3.8, and 3.9. If you do not want the installation to be to your current Python environment, you should first create and activate a [Python virtual environment (venv)](https://docs.python.org/3/tutorial/venv.html) to work in. Then, run the following from the command-line:
```shell-script
pip install itk-spcn
```Launch `python`, import the itk package, and set variable names for the input images
```python
import itkinput_image_filename = "path/to/image_to_be_normalized"
reference_image_filename = "path/to/image_to_be_used_as_color_reference"
```## Usage in Python
The following example transforms this input image
![Input image to be normalized](https://data.kitware.com/api/v1/file/57718cc48d777f1ecd8a883f/download)
using the color scheme of this reference image
![Reference image for normalization](https://data.kitware.com/api/v1/file/576ad39b8d777f1ecd6702f2/download)
to produce this output image
![Output of spcn_filter](https://data.kitware.com/api/v1/file/5ed685d89014a6d84e9bc6f0/download)
### Functional interface to ITK
You can use the functional, eager interface to ITK to choose when each step will be executed as follows. The `input_image` and `reference_image` are processed to produce `normalized_image`, which is the `input_image` with the color scheme of the `reference_image`. The `color_index_suppressed_by_hematoxylin` and `color_index_suppressed_by_eosin` arguments are optional if the `input_image` pixel type is RGB or RGBA. Here you are indicating that the color channel most suppressed by hematoxylin is 0 (which is red for RGB and RGBA pixels) and that the color most suppressed by eosin is 1 (which is green for RGB and RGBA pixels)\; these are the defaults for RGB and RGBA pixels.
```python
input_image = itk.imread(input_image_filename)
reference_image = itk.imread(reference_image_filename)eager_normalized_image = itk.structure_preserving_color_normalization_filter(
input_image,
reference_image,
color_index_suppressed_by_hematoxylin=0,
color_index_suppressed_by_eosin=1,
)itk.imwrite(eager_normalized_image, output_image_filename)
```### ITK pipeline interface
Alternatively, you can use the ITK pipeline infrastructure that waits until a call to `Update()` or `Write()` before executing the pipeline. The function `itk.StructurePreservingColorNormalizationFilter.New()` uses its argument to determine the pixel type for the filter\; the actual image is not used there but is supplied with the `spcn_filter.SetInput(0, input_reader.GetOutput())` call. As above, the calls to `SetColorIndexSuppressedByHematoxylin` and `SetColorIndexSuppressedByEosin` are optional if the pixel type is RGB or RGBA.
```python
input_reader = itk.ImageFileReader.New(FileName=input_image_filename)
reference_reader = itk.ImageFileReader.New(FileName=reference_image_filename)spcn_filter = itk.StructurePreservingColorNormalizationFilter.New(
Input=input_reader.GetOutput()
)
spcn_filter.SetColorIndexSuppressedByHematoxylin(0)
spcn_filter.SetColorIndexSuppressedByEosin(1)
spcn_filter.SetInput(0, input_reader.GetOutput())
spcn_filter.SetInput(1, reference_reader.GetOutput())output_writer = itk.ImageFileWriter.New(spcn_filter.GetOutput())
output_writer.SetInput(spcn_filter.GetOutput())
output_writer.SetFileName(output_image_filename)
output_writer.Write()
```Note that if `spcn_filter` is used again with a different `input_image`, for example from a different reader,
```python
spcn_filter.SetInput(0, input_reader2.GetOutput())
```but the `reference_image` is unchanged then the filter will use its cached analysis of the `reference_image`, which saves about half the processing time.