2012-09-21 20:02:25 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) 2012, Jan Schlicht <jan.schlicht@gmail.com>
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any purpose
|
|
|
|
with or without fee is hereby granted, provided that the above copyright notice
|
|
|
|
and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
|
|
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
|
|
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
|
|
|
|
OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
|
|
TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
|
|
|
|
THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package resize
|
|
|
|
|
|
|
|
import (
|
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
)
|
|
|
|
|
|
|
|
type colorArray [4]float32
|
|
|
|
|
2012-12-10 19:27:21 +01:00
|
|
|
func replicateBorder1d(x, min, max int) int {
|
2012-12-10 19:52:41 +01:00
|
|
|
if x < min {
|
2012-12-10 19:27:21 +01:00
|
|
|
x = min
|
2012-12-10 19:52:41 +01:00
|
|
|
} else if x >= max {
|
|
|
|
x = max - 1
|
2012-12-10 19:27:21 +01:00
|
|
|
}
|
2012-12-10 19:52:41 +01:00
|
|
|
|
2012-12-10 19:27:21 +01:00
|
|
|
return x
|
|
|
|
}
|
|
|
|
|
|
|
|
func replicateBorder(x, y int, rect image.Rectangle) (xx, yy int) {
|
|
|
|
xx = replicateBorder1d(x, rect.Min.X, rect.Max.X)
|
|
|
|
yy = replicateBorder1d(y, rect.Min.Y, rect.Max.Y)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2012-12-11 20:35:02 +01:00
|
|
|
// converter allows to retrieve a colorArray for points of an image.
|
|
|
|
// the idea is to speed up computation by providing optimized implementations
|
|
|
|
// for different image types instead of relying on image.Image.At().
|
2012-12-11 19:57:34 +01:00
|
|
|
type converter interface {
|
2014-01-17 22:54:15 +01:00
|
|
|
at(x, y int, color *colorArray)
|
2012-12-11 19:57:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type genericConverter struct {
|
|
|
|
src image.Image
|
|
|
|
}
|
|
|
|
|
2014-01-17 22:54:15 +01:00
|
|
|
func (c *genericConverter) at(x, y int, result *colorArray) {
|
2012-12-10 21:26:34 +01:00
|
|
|
r, g, b, a := c.src.At(replicateBorder(x, y, c.src.Bounds())).RGBA()
|
2014-01-17 22:54:15 +01:00
|
|
|
result[0] = float32(r)
|
|
|
|
result[1] = float32(g)
|
|
|
|
result[2] = float32(b)
|
|
|
|
result[3] = float32(a)
|
|
|
|
return
|
2012-09-21 20:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type rgbaConverter struct {
|
|
|
|
src *image.RGBA
|
|
|
|
}
|
|
|
|
|
2014-01-17 22:54:15 +01:00
|
|
|
func (c *rgbaConverter) at(x, y int, result *colorArray) {
|
2012-12-10 19:27:21 +01:00
|
|
|
i := c.src.PixOffset(replicateBorder(x, y, c.src.Rect))
|
2014-01-17 22:54:15 +01:00
|
|
|
result[0] = float32(uint16(c.src.Pix[i+0])<<8 | uint16(c.src.Pix[i+0]))
|
|
|
|
result[1] = float32(uint16(c.src.Pix[i+1])<<8 | uint16(c.src.Pix[i+1]))
|
|
|
|
result[2] = float32(uint16(c.src.Pix[i+2])<<8 | uint16(c.src.Pix[i+2]))
|
|
|
|
result[3] = float32(uint16(c.src.Pix[i+3])<<8 | uint16(c.src.Pix[i+3]))
|
|
|
|
return
|
2012-09-21 20:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type rgba64Converter struct {
|
|
|
|
src *image.RGBA64
|
|
|
|
}
|
|
|
|
|
2014-01-17 22:54:15 +01:00
|
|
|
func (c *rgba64Converter) at(x, y int, result *colorArray) {
|
2012-12-10 19:27:21 +01:00
|
|
|
i := c.src.PixOffset(replicateBorder(x, y, c.src.Rect))
|
2014-01-17 22:54:15 +01:00
|
|
|
result[0] = float32(uint16(c.src.Pix[i+0])<<8 | uint16(c.src.Pix[i+1]))
|
|
|
|
result[1] = float32(uint16(c.src.Pix[i+2])<<8 | uint16(c.src.Pix[i+3]))
|
|
|
|
result[2] = float32(uint16(c.src.Pix[i+4])<<8 | uint16(c.src.Pix[i+5]))
|
|
|
|
result[3] = float32(uint16(c.src.Pix[i+6])<<8 | uint16(c.src.Pix[i+7]))
|
|
|
|
return
|
2012-09-21 20:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type grayConverter struct {
|
|
|
|
src *image.Gray
|
|
|
|
}
|
|
|
|
|
2014-01-17 22:54:15 +01:00
|
|
|
func (c *grayConverter) at(x, y int, result *colorArray) {
|
2012-12-10 19:27:21 +01:00
|
|
|
i := c.src.PixOffset(replicateBorder(x, y, c.src.Rect))
|
2012-09-21 20:02:25 +02:00
|
|
|
g := float32(uint16(c.src.Pix[i])<<8 | uint16(c.src.Pix[i]))
|
2014-01-17 22:54:15 +01:00
|
|
|
result[0] = g
|
|
|
|
result[1] = g
|
|
|
|
result[2] = g
|
|
|
|
result[3] = float32(0xffff)
|
|
|
|
return
|
2012-09-21 20:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type gray16Converter struct {
|
|
|
|
src *image.Gray16
|
|
|
|
}
|
|
|
|
|
2014-01-17 22:54:15 +01:00
|
|
|
func (c *gray16Converter) at(x, y int, result *colorArray) {
|
2012-12-10 19:27:21 +01:00
|
|
|
i := c.src.PixOffset(replicateBorder(x, y, c.src.Rect))
|
2012-09-21 20:02:25 +02:00
|
|
|
g := float32(uint16(c.src.Pix[i+0])<<8 | uint16(c.src.Pix[i+1]))
|
2014-01-17 22:54:15 +01:00
|
|
|
result[0] = g
|
|
|
|
result[1] = g
|
|
|
|
result[2] = g
|
|
|
|
result[3] = float32(0xffff)
|
|
|
|
return
|
2012-09-21 20:02:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type ycbcrConverter struct {
|
|
|
|
src *image.YCbCr
|
|
|
|
}
|
|
|
|
|
2014-01-17 22:54:15 +01:00
|
|
|
func (c *ycbcrConverter) at(x, y int, result *colorArray) {
|
2012-12-10 19:27:21 +01:00
|
|
|
xx, yy := replicateBorder(x, y, c.src.Rect)
|
|
|
|
yi := c.src.YOffset(xx, yy)
|
|
|
|
ci := c.src.COffset(xx, yy)
|
2012-09-21 20:02:25 +02:00
|
|
|
r, g, b := color.YCbCrToRGB(c.src.Y[yi], c.src.Cb[ci], c.src.Cr[ci])
|
2014-01-17 22:54:15 +01:00
|
|
|
result[0] = float32(uint16(r) * 0x101)
|
|
|
|
result[1] = float32(uint16(g) * 0x101)
|
|
|
|
result[2] = float32(uint16(b) * 0x101)
|
|
|
|
result[3] = float32(0xffff)
|
|
|
|
return
|
2012-09-21 20:02:25 +02:00
|
|
|
}
|