2023-12-12 23:30:03 +04:00
|
|
|
package imageutils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
"image/draw"
|
|
|
|
)
|
|
|
|
|
2024-05-01 16:50:36 +04:00
|
|
|
type Side int
|
|
|
|
|
|
|
|
const (
|
|
|
|
Left Side = iota
|
|
|
|
Right
|
|
|
|
Up
|
|
|
|
Down
|
|
|
|
)
|
|
|
|
|
2023-12-12 23:30:03 +04:00
|
|
|
type pair struct {
|
|
|
|
first, second image.Image
|
2024-05-01 16:50:36 +04:00
|
|
|
side Side
|
2023-12-12 23:30:03 +04:00
|
|
|
}
|
|
|
|
|
2024-05-01 16:50:36 +04:00
|
|
|
// Takes color.Model of the first Image.
|
2023-12-12 23:30:03 +04:00
|
|
|
func (p pair) ColorModel() color.Model { return p.first.ColorModel() }
|
|
|
|
|
|
|
|
func (p pair) Bounds() image.Rectangle {
|
|
|
|
var (
|
|
|
|
b1 = p.first.Bounds()
|
|
|
|
b2 = p.second.Bounds()
|
|
|
|
)
|
|
|
|
|
2024-05-01 16:50:36 +04:00
|
|
|
rect := image.Rectangle{}
|
|
|
|
switch p.side {
|
|
|
|
case Left:
|
|
|
|
fallthrough
|
|
|
|
case Right:
|
|
|
|
rect = image.Rectangle{
|
|
|
|
image.ZP,
|
|
|
|
image.Point{
|
|
|
|
X: b1.Dx() + b2.Dx(),
|
|
|
|
Y: max(b1.Dy(), b2.Dy()),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
case Up:
|
|
|
|
fallthrough
|
|
|
|
case Down:
|
|
|
|
rect = image.Rectangle{
|
|
|
|
image.ZP,
|
|
|
|
image.Point{
|
|
|
|
X: max(b1.Dx(), b2.Dx()),
|
|
|
|
Y: b1.Dy() + b2.Dy(),
|
|
|
|
},
|
|
|
|
}
|
2023-12-12 23:30:03 +04:00
|
|
|
}
|
2024-05-01 16:50:36 +04:00
|
|
|
return rect
|
2023-12-12 23:30:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p pair) At(x, y int) color.Color {
|
|
|
|
rgbaImg := image.NewRGBA(p.Bounds())
|
2024-05-01 16:50:36 +04:00
|
|
|
|
|
|
|
switch p.side {
|
|
|
|
case Left:
|
|
|
|
draw.Draw(
|
|
|
|
rgbaImg, rgbaImg.Bounds(),
|
|
|
|
p.second,
|
|
|
|
image.ZP, draw.Src,
|
|
|
|
)
|
|
|
|
draw.Draw(
|
2024-05-01 17:16:30 +04:00
|
|
|
rgbaImg, p.first.Bounds(),
|
2024-05-01 16:50:36 +04:00
|
|
|
p.first,
|
|
|
|
image.Point{
|
|
|
|
p.second.Bounds().Dx(),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
draw.Src,
|
|
|
|
)
|
|
|
|
case Right:
|
|
|
|
draw.Draw(
|
|
|
|
rgbaImg, rgbaImg.Bounds(),
|
|
|
|
p.first,
|
|
|
|
image.ZP, draw.Src,
|
|
|
|
)
|
|
|
|
draw.Draw(
|
2024-05-01 17:16:30 +04:00
|
|
|
rgbaImg, p.first.Bounds(),
|
2024-05-01 16:50:36 +04:00
|
|
|
p.second,
|
|
|
|
image.Point{
|
|
|
|
p.first.Bounds().Dx(),
|
|
|
|
0,
|
|
|
|
},
|
2024-05-01 17:08:31 +04:00
|
|
|
draw.Over,
|
2024-05-01 16:50:36 +04:00
|
|
|
)
|
|
|
|
case Up:
|
|
|
|
draw.Draw(
|
2024-05-01 17:05:20 +04:00
|
|
|
rgbaImg, rgbaImg.Bounds(),
|
2024-05-01 16:50:36 +04:00
|
|
|
p.first,
|
2024-05-01 17:05:20 +04:00
|
|
|
image.ZP, draw.Src,
|
2024-05-01 16:50:36 +04:00
|
|
|
)
|
|
|
|
draw.Draw(
|
|
|
|
rgbaImg, p.Bounds(),
|
|
|
|
p.second,
|
|
|
|
image.Point{
|
|
|
|
0,
|
2024-05-01 17:05:20 +04:00
|
|
|
p.first.Bounds().Dy(),
|
2024-05-01 16:50:36 +04:00
|
|
|
},
|
|
|
|
draw.Src,
|
|
|
|
)
|
|
|
|
case Down:
|
|
|
|
draw.Draw(
|
2024-05-01 17:05:20 +04:00
|
|
|
rgbaImg, rgbaImg.Bounds(),
|
2024-05-01 16:50:36 +04:00
|
|
|
p.second,
|
2024-05-01 17:05:20 +04:00
|
|
|
image.ZP, draw.Src,
|
2024-05-01 16:50:36 +04:00
|
|
|
)
|
|
|
|
draw.Draw(
|
|
|
|
rgbaImg, p.Bounds(),
|
|
|
|
p.first,
|
|
|
|
image.Point{
|
|
|
|
0,
|
2024-05-01 17:05:20 +04:00
|
|
|
p.second.Bounds().Dy(),
|
2024-05-01 16:50:36 +04:00
|
|
|
},
|
|
|
|
draw.Src,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return rgbaImg.At(x, y)
|
2023-12-12 23:30:03 +04:00
|
|
|
}
|
|
|
|
|
2024-05-01 16:50:36 +04:00
|
|
|
// Concat concatenates second image on a given side.
|
|
|
|
func Concat(i1, i2 image.Image, s Side) image.Image {
|
|
|
|
return pair{i1, i2, s}
|
2023-12-12 23:30:03 +04:00
|
|
|
}
|