From 9fc02d6218fe02e7beb214788ea037634f9c04a2 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 31 Aug 2021 03:02:55 -0500 Subject: [PATCH] Fix and test clustering MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This has one breaking change: while the docs advertise that "All the segmentation algorithms (except Fuzzy C-means) return a struct SegmentedImage," that's not been true for `kmeans`. This changes the output so that `kmeans` does return a `SegmentedImage`. This is a breaking change, because the types are not interchangeable. This became apparent when trying to add tests for `kmeans`, which have been lacking. Before releasing this, we may want to make a second breaking change: currently, ImageSegmentation provides a new meaning for `Matrix{Gray{T}}` than Clustering.jl provides for `Matrix{T}`. This breaks our abstraction that `Gray ≈ Number`. A way to fix that would be to have `ImageSegmentation.kmeans` be a different function from `Clustering.kmeans`, and obviously have the one in ImageSegmentation call the one in Clustering. --- src/ImageSegmentation.jl | 3 ++- src/clustering.jl | 10 ++++------ src/core.jl | 7 +++++++ test/clustering.jl | 17 +++++++++++++++++ test/runtests.jl | 3 ++- 5 files changed, 32 insertions(+), 8 deletions(-) create mode 100644 test/clustering.jl diff --git a/src/ImageSegmentation.jl b/src/ImageSegmentation.jl index 67115ff..b398205 100644 --- a/src/ImageSegmentation.jl +++ b/src/ImageSegmentation.jl @@ -4,6 +4,7 @@ import Base: show using LinearAlgebra, Statistics using DataStructures, StaticArrays, ImageCore, ImageFiltering, ImageMorphology, LightGraphs, SimpleWeightedGraphs, RegionTrees, Distances, StaticArrays, Clustering, MetaGraphs +using ImageCore.MappedArrays: of_eltype import Clustering: kmeans, fuzzy_cmeans include("compat.jl") @@ -41,7 +42,7 @@ export kmeans, fuzzy_cmeans, merge_segments, - + # types SegmentedImage, ImageEdge diff --git a/src/clustering.jl b/src/clustering.jl index 7133f7d..1ad5532 100644 --- a/src/clustering.jl +++ b/src/clustering.jl @@ -1,13 +1,11 @@ function img_to_data(img::AbstractArray{T,N}) where T<:Colorant where N - AT = accum_type(T) - aimg = AT.(img) - pimg = parent(aimg) - ch = channelview(pimg) - data = reshape(ch, :, *(size(pimg)...)) + aimg = of_eltype(accum_type(T), img) + ch = channelview(aimg) + return reshape(ch, :, *(size(img)...)) end kmeans(img::AbstractArray{T,N}, args...; kwargs...) where {T<:Colorant,N} = - kmeans(img_to_data(img), args...; kwargs...) + SegmentedImage(kmeans(img_to_data(img), args...; kwargs...), img) fuzzy_cmeans(img::AbstractArray{T,N}, args...; kwargs...) where {T<:Colorant,N} = fuzzy_cmeans(img_to_data(img), args...; kwargs...) diff --git a/src/core.jl b/src/core.jl index 166f1fa..2932638 100644 --- a/src/core.jl +++ b/src/core.jl @@ -14,6 +14,13 @@ struct SegmentedImage{T<:AbstractArray,U<:Union{Colorant,Real}} segment_pixel_count::Dict{Int,Int} end +SegmentedImage(r::Clustering.KmeansResult, img) = + SegmentedImage(reshape(r.assignments, axes(img)), + collect(1:length(r.counts)), + Dict(zip(1:length(r.counts), colorview(ImageCore.ColorTypes.base_colorant_type(eltype(img)), r.centers))), + Dict(zip(1:length(r.counts), r.counts)) + ) + """ edge = ImageEdge(index1, index2, weight) diff --git a/test/clustering.jl b/test/clustering.jl new file mode 100644 index 0000000..5a2f246 --- /dev/null +++ b/test/clustering.jl @@ -0,0 +1,17 @@ +@testset "Clustering" begin + # Test the example in the docs + path = download("https://github.com/JuliaImages/juliaimages.github.io/raw/source/docs/src/pkgs/segmentation/assets/flower.jpg") + img = load(path) + r = fuzzy_cmeans(img, 3, 2) + @test size(r.centers) == (3,3) + @test size(r.weights, 1) == length(img) + @test all(≈(1), sum(r.weights; dims=2)) + cmin, cmax = extrema(sum(r.weights; dims=1)) + @test cmax < 3*cmin + + # Also with kmeans + r = kmeans(img, 3) + nc = last.(sort([pr for pr in segment_pixel_count(r)]; by=first)) + @test sum(nc) == length(img) + @test 3*minimum(nc) > maximum(nc) +end diff --git a/test/runtests.jl b/test/runtests.jl index a8c257c..fbb4a7c 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -1,5 +1,5 @@ using ImageSegmentation, Images, Test, SimpleWeightedGraphs, LightGraphs, StaticArrays, DataStructures -using RegionTrees: isleaf, Cell, split! +using RegionTrees: isleaf, Cell, split! using MetaGraphs: MetaGraph, clear_props!, get_prop, has_prop, set_prop!, props, vertices using Documenter @@ -12,4 +12,5 @@ include("fast_scanning.jl") include("watershed.jl") include("region_merging.jl") include("meanshift.jl") +include("clustering.jl") include("merge_segments.jl")