Added new func: CustomSimilar

master
Vitali Fedulov 2024-01-28 22:50:59 +01:00
parent c075bc302b
commit 53fdf151da
4 changed files with 137 additions and 0 deletions

65
custom.go Normal file
View File

@ -0,0 +1,65 @@
package images4
// Threshold multiplication coefficients for func CustomSimilar.
// When all values equal 1.0 func CustomSimilar is equivalent
// to func Similar. By setting those values less than 1, similarity
// comparison becomes stricter (more precise). Values larger than 1
// will generalize more and show more false positives. When uncertain,
// setting all coefficients to 1.0 is the safe starting point.
type CustomCoefficients struct {
Y float64 // Luma (grayscale information).
Cb float64 // Chrominance b (color information).
Cr float64 // Chrominance r (color information).
Prop float64 // Proportion tolerance (how similar are image borders).
}
// CustomSimilar is like Similar, except it allows changing default
// thresholds by multiplying them. The practically useful range of
// the coefficients is [0, 1.0), but can be equal or larger than 1
// if necessary. All coefficients set to 0 correspond to identical images,
// for example an image file copy. All coefficients equal to 1 make func
// CustomSimilar equivalent to func Similar.
func CustomSimilar(iconA, iconB IconT, coeff CustomCoefficients) bool {
if !customPropSimilar(iconA, iconB, coeff) {
return false
}
if !customEucSimilar(iconA, iconB, coeff) {
return false
}
return true
}
func customPropSimilar(iconA, iconB IconT, coeff CustomCoefficients) bool {
return PropMetric(iconA, iconB) <= thProp*coeff.Prop
}
func customEucSimilar(iconA, iconB IconT, coeff CustomCoefficients) bool {
m1, m2, m3 := EucMetric(iconA, iconB)
return m1 <= thY*coeff.Y &&
m2 <= thCbCr*coeff.Cb &&
m3 <= thCbCr*coeff.Cr
}
// Similar90270 works like Similar, but also considers rotations of ±90°.
// Those are rotations users might reasonably often do.
func CustomSimilar90270(iconA, iconB IconT, coeff CustomCoefficients) bool {
if CustomSimilar(iconA, iconB, coeff) {
return true
}
// iconB rotated 90 degrees.
if CustomSimilar(iconA, Rotate90(iconB), coeff) {
return true
}
// As if iconB was rotated 270 degrees.
if CustomSimilar(Rotate90(iconA), iconB, coeff) {
return true
}
return false
}

72
custom_test.go Normal file
View File

@ -0,0 +1,72 @@
package images4
import (
"path"
"testing"
)
func TestCustomSimilar(t *testing.T) {
// Proportions test.
i1, _ := Open(path.Join("testdata", "euclidean", "distorted.jpg"))
i2, _ := Open(path.Join("testdata", "euclidean", "large.jpg"))
icon1 := Icon(i1)
icon2 := Icon(i2)
if Similar(icon1, icon2) {
t.Errorf("distorted.jpg is NOT similar to large.jpg")
}
if !CustomSimilar(icon1, icon2, CustomCoefficients{1, 1, 1, 10}) {
t.Errorf("distorted.jpg IS similar to large.jpg, assuming proportion differences are widely tolerated.")
}
// Euclidean tests.
i1, _ = Open(path.Join("testdata", "custom", "1.jpg"))
i2, _ = Open(path.Join("testdata", "custom", "2.jpg"))
icon1 = Icon(i1)
icon2 = Icon(i2)
if !Similar(icon1, icon2) {
t.Errorf("1.jpg is GENERALLY similar to 2.jpg")
}
// Luma.
if CustomSimilar(icon1, icon2, CustomCoefficients{0, 1, 1, 1}) {
t.Errorf("1.jpg is NOT IDENTICAL to 2.jpg")
}
// Luma.
if CustomSimilar(icon1, icon2, CustomCoefficients{0.4, 1, 1, 1}) {
t.Errorf("1.jpg is similar to 2.jpg, BUT NOT VERY SIMILAR")
}
// Chrominance b.
if CustomSimilar(icon1, icon2, CustomCoefficients{1, 0.1, 1, 1}) {
t.Errorf("1.jpg is similar to 2.jpg, BUT NOT VERY SIMILAR")
}
// Chrominance c.
if CustomSimilar(icon1, icon2, CustomCoefficients{1, 1, 0.1, 1}) {
t.Errorf("1.jpg is similar to 2.jpg, BUT NOT VERY SIMILAR")
}
// Image comparison to itself (or its own copy).
if !CustomSimilar(icon1, icon1, CustomCoefficients{0, 0, 0, 0}) {
t.Errorf("1.jpg IS IDENTICAL to itself")
}
if !CustomSimilar(icon1, icon1, CustomCoefficients{0.5, 0.5, 0.5, 0.5}) {
t.Errorf("1.jpg IS IDENTICAL to itself")
}
if !CustomSimilar(icon1, icon1, CustomCoefficients{1, 1, 1, 1}) {
t.Errorf("1.jpg IS IDENTICAL to itself")
}
}

BIN
testdata/custom/1.jpg vendored Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.8 KiB

BIN
testdata/custom/2.jpg vendored Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.2 KiB