forked from makiuchi-d/gozxing
-
Notifications
You must be signed in to change notification settings - Fork 1
/
rgb_luminance_source.go
113 lines (98 loc) · 3.15 KB
/
rgb_luminance_source.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
package gozxing
import (
errors "golang.org/x/xerrors"
)
type RGBLuminanceSource struct {
LuminanceSourceBase
luminances []byte
dataWidth int
dataHeight int
left int
top int
}
func NewRGBLuminanceSource(width, height int, pixels []int) LuminanceSource {
dataWidth := width
dataHeight := height
left := 0
top := 0
// In order to measure pure decoding speed, we convert the entire image to a greyscale array
// up front, which is the same as the Y channel of the YUVLuminanceSource in the real app.
//
// Total number of pixels suffices, can ignore shape
size := width * height
luminances := make([]byte, size)
for offset := 0; offset < size; offset++ {
pixel := pixels[offset]
r := (pixel >> 16) & 0xff // red
g2 := (pixel >> 7) & 0x1fe // 2 * green
b := pixel & 0xff // blue
// Calculate green-favouring average cheaply
luminances[offset] = byte((r + g2 + b) / 4)
}
return &RGBLuminanceSource{
LuminanceSourceBase{width, height},
luminances,
dataWidth,
dataHeight,
left,
top,
}
}
func (this *RGBLuminanceSource) GetRow(y int, row []byte) ([]byte, error) {
if y < 0 || y >= this.GetHeight() {
return row, errors.Errorf("IllegalArgumentException: Requested row is outside the image: %d", y)
}
width := this.GetWidth()
if row == nil || len(row) < width {
row = make([]byte, width)
}
offset := (y+this.top)*this.dataWidth + this.left
copy(row, this.luminances[offset:offset+width])
return row, nil
}
func (this *RGBLuminanceSource) GetMatrix() []byte {
width := this.GetWidth()
height := this.GetHeight()
// If the caller asks for the entire underlying image, save the copy and give them the
// original data. The docs specifically warn that result.length must be ignored.
if width == this.dataWidth && height == this.dataHeight {
return this.luminances
}
area := width * height
matrix := make([]byte, area)
inputOffset := this.top*this.dataWidth + this.left
// If the width matches the full width of the underlying data, perform a single copy.
if width == this.dataWidth {
copy(matrix, this.luminances[inputOffset:inputOffset+area])
return matrix
}
// Otherwise copy one cropped row at a time.
for y := 0; y < height; y++ {
outputOffset := y * width
copy(matrix[outputOffset:outputOffset+width], this.luminances[inputOffset:inputOffset+width])
inputOffset += this.dataWidth
}
return matrix
}
func (this *RGBLuminanceSource) IsCropSupported() bool {
return true
}
func (this *RGBLuminanceSource) Crop(left, top, width, height int) (LuminanceSource, error) {
if left+width > this.dataWidth || top+height > this.dataHeight {
return nil, errors.New("IllegalArgumentException: Crop rectangle does not fit within image data")
}
return &RGBLuminanceSource{
LuminanceSourceBase: LuminanceSourceBase{width, height},
luminances: this.luminances,
dataWidth: this.dataWidth,
dataHeight: this.dataHeight,
left: this.left + left,
top: this.top + top,
}, nil
}
func (this *RGBLuminanceSource) Invert() LuminanceSource {
return LuminanceSourceInvert(this)
}
func (this *RGBLuminanceSource) String() string {
return LuminanceSourceString(this)
}