This commit is contained in:
Florian Sundermann 2013-12-16 14:10:24 +01:00
parent a4167688a0
commit 9741f0ff8d
12 changed files with 143 additions and 270 deletions

View File

@ -1,36 +0,0 @@
package codabar
import (
"github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
"image"
"image/color"
)
type codabarcode struct {
*utils.BitList
content string
}
func (c *codabarcode) Content() string {
return c.content
}
func (c *codabarcode) Metadata() barcode.Metadata {
return barcode.Metadata{"Codabar", 1}
}
func (c *codabarcode) ColorModel() color.Model {
return color.Gray16Model
}
func (c *codabarcode) Bounds() image.Rectangle {
return image.Rect(0, 0, c.Len(), 1)
}
func (c *codabarcode) At(x, y int) color.Color {
if c.GetBit(x) {
return color.Black
}
return color.White
}

View File

@ -44,9 +44,7 @@ func Encode(content string) (barcode.Barcode, error) {
if !ok { if !ok {
return nil, fmt.Errorf("can not encode \"%s\"", content) return nil, fmt.Errorf("can not encode \"%s\"", content)
} }
for _, bit := range bits { resBits.AddBit(bits...)
resBits.AddBit(bit)
} }
} return utils.New1DCode("Codabar", content, resBits), nil
return &codabarcode{resBits, content}, nil
} }

View File

@ -1,21 +0,0 @@
package code128
import (
"github.com/boombuler/barcode/utils"
"strings"
)
const bTable = " !\"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~"
func encodeBTable(content []rune) *utils.BitList {
result := new(utils.BitList)
result.AddByte(startBSymbol)
for _, r := range content {
idx := strings.IndexRune(bTable, r)
if idx < 0 {
return nil
}
result.AddByte(byte(idx))
}
return result
}

View File

@ -1,36 +0,0 @@
package code128
import (
"github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
"image"
"image/color"
)
type code struct {
*utils.BitList
content string
}
func (c *code) Content() string {
return c.content
}
func (c *code) Metadata() barcode.Metadata {
return barcode.Metadata{"Code 128", 1}
}
func (c *code) ColorModel() color.Model {
return color.Gray16Model
}
func (c *code) Bounds() image.Rectangle {
return image.Rect(0, 0, c.Len(), 1)
}
func (c *code) At(x, y int) color.Color {
if c.GetBit(x) {
return color.Black
}
return color.White
}

View File

@ -4,6 +4,7 @@ import (
"fmt" "fmt"
"github.com/boombuler/barcode" "github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils" "github.com/boombuler/barcode/utils"
"strings"
"unicode/utf8" "unicode/utf8"
) )
@ -17,15 +18,65 @@ func strToRunes(str string) []rune {
return result return result
} }
func shouldUseCTable(nextRunes []rune, curEncoding byte) bool {
requiredDigits := 4
if curEncoding == startCSymbol {
requiredDigits = 2
}
if len(nextRunes) < requiredDigits {
return false
}
for i := 0; i < requiredDigits; i++ {
if nextRunes[i] < '0' || nextRunes[i] > '9' {
return false
}
}
return true
}
func getCodeIndexList(content []rune) *utils.BitList {
result := new(utils.BitList)
curEncoding := byte(0)
for i := 0; i < len(content); i++ {
if shouldUseCTable(content[i:], curEncoding) {
if curEncoding != startCSymbol {
result.AddByte(startCSymbol)
curEncoding = startCSymbol
}
idx := (content[i] - '0') * 10
i++
idx = idx + (content[i] - '0')
result.AddByte(byte(idx))
} else {
if curEncoding != startBSymbol {
result.AddByte(startBSymbol)
curEncoding = startBSymbol
}
idx := strings.IndexRune(bTable, content[i])
if idx < 0 {
return nil
}
result.AddByte(byte(idx))
}
}
fmt.Println(result.GetBytes())
return result
}
func Encode(content string) (barcode.Barcode, error) { func Encode(content string) (barcode.Barcode, error) {
contentRunes := strToRunes(content) contentRunes := strToRunes(content)
idxList := encodeBTable(contentRunes) if len(contentRunes) < 0 || len(contentRunes) > 80 {
return nil, fmt.Errorf("content length should be between 1 and 80 runes but got %d", len(contentRunes))
}
idxList := getCodeIndexList(contentRunes)
if idxList == nil { if idxList == nil {
return nil, fmt.Errorf("\"%s\" could not be encoded", content) return nil, fmt.Errorf("\"%s\" could not be encoded", content)
} }
result := &code{new(utils.BitList), content} result := new(utils.BitList)
sum := 0 sum := 0
for i, idx := range idxList.GetBytes() { for i, idx := range idxList.GetBytes() {
if i == 0 { if i == 0 {
@ -33,9 +84,9 @@ func Encode(content string) (barcode.Barcode, error) {
} else { } else {
sum += i * int(idx) sum += i * int(idx)
} }
result.AddRange(encodingTable[idx]) result.AddBit(encodingTable[idx]...)
} }
result.AddRange(encodingTable[sum%103]) result.AddBit(encodingTable[sum%103]...)
result.AddRange(encodingTable[stopSymbol]) result.AddBit(encodingTable[stopSymbol]...)
return result, nil return utils.New1DCode("Code 128", content, result), nil
} }

View File

@ -114,3 +114,5 @@ const startASymbol byte = 103
const startBSymbol byte = 104 const startBSymbol byte = 104
const startCSymbol byte = 105 const startCSymbol byte = 105
const stopSymbol byte = 106 const stopSymbol byte = 106
const bTable = " !\"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~"

View File

@ -1,40 +0,0 @@
package code39
import (
"github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
"image"
"image/color"
)
type code struct {
*utils.BitList
content string
}
func newCode() *code {
return &code{new(utils.BitList), ""}
}
func (c *code) Content() string {
return c.content
}
func (c *code) Metadata() barcode.Metadata {
return barcode.Metadata{"Code39", 1}
}
func (c *code) ColorModel() color.Model {
return color.Gray16Model
}
func (c *code) Bounds() image.Rectangle {
return image.Rect(0, 0, c.Len(), 1)
}
func (c *code) At(x, y int) color.Color {
if c.GetBit(x) {
return color.Black
}
return color.White
}

View File

@ -3,6 +3,7 @@ package code39
import ( import (
"errors" "errors"
"github.com/boombuler/barcode" "github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
"strings" "strings"
) )
@ -91,22 +92,19 @@ func Encode(content string, includeChecksum bool) (barcode.Barcode, error) {
} }
data += "*" data += "*"
cd := newCode() result := new(utils.BitList)
cd.content = content
for i, r := range data { for i, r := range data {
if i != 0 { if i != 0 {
cd.AddBit(false) result.AddBit(false)
} }
info, ok := encodeTable[r] info, ok := encodeTable[r]
if !ok { if !ok {
return nil, errors.New("invalid data") return nil, errors.New("invalid data")
} }
for _, bit := range info.data { result.AddBit(info.data...)
cd.AddBit(bit)
}
} }
return cd, nil return utils.New1DCode("Code 39", content, result), nil
} }

View File

@ -1,47 +0,0 @@
package ean
import (
"github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
"image"
"image/color"
)
type eancode struct {
*utils.BitList
content string
}
func newEANCode(isEAN8 bool) *eancode {
capacity := 95
if isEAN8 {
capacity = 67
}
return &eancode{utils.NewBitList(capacity), ""}
}
func (c *eancode) Content() string {
return c.content
}
func (c *eancode) Metadata() barcode.Metadata {
if c.Len() == 67 {
return barcode.Metadata{"EAN 8", 1}
}
return barcode.Metadata{"EAN 13", 1}
}
func (c *eancode) ColorModel() color.Model {
return color.Gray16Model
}
func (c *eancode) Bounds() image.Rectangle {
return image.Rect(0, 0, c.Len(), 1)
}
func (c *eancode) At(x, y int) color.Color {
if c.GetBit(x) {
return color.Black
}
return color.White
}

View File

@ -4,6 +4,7 @@ package ean
import ( import (
"errors" "errors"
"github.com/boombuler/barcode" "github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
) )
type encodedNumber struct { type encodedNumber struct {
@ -146,21 +147,14 @@ func calcCheckNum(code string) rune {
return intToRune((10 - (sum % 10)) % 10) return intToRune((10 - (sum % 10)) % 10)
} }
func encodeEAN8(code string, result *eancode) bool { func encodeEAN8(code string) *utils.BitList {
pos := 0 result := new(utils.BitList)
appendBit := func(b bool) { result.AddBit(true, false, true)
result.SetBit(pos, b)
pos++
}
appendBit(true)
appendBit(false)
appendBit(true)
for cpos, r := range code { for cpos, r := range code {
num, ok := encoderTable[r] num, ok := encoderTable[r]
if !ok { if !ok {
return false return nil
} }
var data []bool var data []bool
if cpos < 4 { if cpos < 4 {
@ -170,39 +164,24 @@ func encodeEAN8(code string, result *eancode) bool {
} }
if cpos == 4 { if cpos == 4 {
appendBit(false) result.AddBit(false, true, false, true, false)
appendBit(true)
appendBit(false)
appendBit(true)
appendBit(false)
} }
for _, bit := range data { result.AddBit(data...)
appendBit(bit)
} }
result.AddBit(true, false, true)
return result
} }
appendBit(true) func encodeEAN13(code string) *utils.BitList {
appendBit(false) result := new(utils.BitList)
appendBit(true) result.AddBit(true, false, true)
return true
}
func encodeEAN13(code string, result *eancode) bool {
pos := 0
appendBit := func(b bool) {
result.SetBit(pos, b)
pos++
}
appendBit(true)
appendBit(false)
appendBit(true)
var firstNum []bool var firstNum []bool
for cpos, r := range code { for cpos, r := range code {
num, ok := encoderTable[r] num, ok := encoderTable[r]
if !ok { if !ok {
return false return nil
} }
if cpos == 0 { if cpos == 0 {
firstNum = num.CheckSum firstNum = num.CheckSum
@ -221,21 +200,12 @@ func encodeEAN13(code string, result *eancode) bool {
} }
if cpos == 7 { if cpos == 7 {
appendBit(false) result.AddBit(false, true, false, true, false)
appendBit(true)
appendBit(false)
appendBit(true)
appendBit(false)
} }
result.AddBit(data...)
for _, bit := range data {
appendBit(bit)
} }
} result.AddByte(true, false, true)
appendBit(true) return result
appendBit(false)
appendBit(true)
return true
} }
// encodes the given EAN 8 or EAN 13 number to a barcode image // encodes the given EAN 8 or EAN 13 number to a barcode image
@ -249,18 +219,17 @@ func Encode(code string) (barcode.Barcode, error) {
return nil, errors.New("checksum missmatch!") return nil, errors.New("checksum missmatch!")
} }
} }
ean8 := false var result *utils.BitList = nil
if len(code) == 8 { if len(code) == 8 {
ean8 = true result := encodeEAN8(code)
if result != nil {
} else if len(code) != 13 { return utils.New1DCode("EAN 8", code, result), nil
}
} else if len(code) == 13 {
result := encodeEAN13(code)
if result != nil {
return utils.New1DCode("EAN 13", code, result), nil
}
}
return nil, errors.New("invalid ean code data") return nil, errors.New("invalid ean code data")
} }
result := newEANCode(ean8)
if (ean8 && encodeEAN8(code, result)) || (!ean8 && encodeEAN13(code, result)) {
result.content = code
return result, nil
}
return nil, errors.New("ean code contains invalid characters")
}

40
utils/base1dcode.go Normal file
View File

@ -0,0 +1,40 @@
package utils
import (
"github.com/boombuler/barcode"
"image"
"image/color"
)
type base1DCode struct {
*BitList
kind string
content string
}
func (c *base1DCode) Content() string {
return c.content
}
func (c *base1DCode) Metadata() barcode.Metadata {
return barcode.Metadata{c.kind, 1}
}
func (c *base1DCode) ColorModel() color.Model {
return color.Gray16Model
}
func (c *base1DCode) Bounds() image.Rectangle {
return image.Rect(0, 0, c.Len(), 1)
}
func (c *base1DCode) At(x, y int) color.Color {
if c.GetBit(x) {
return color.Black
}
return color.White
}
func New1DCode(codeKind, content string, bars *BitList) barcode.Barcode {
return &base1DCode{bars, codeKind, content}
}

View File

@ -37,8 +37,9 @@ func (bl *BitList) grow() {
bl.data = nd bl.data = nd
} }
// appends the given bit to the end of the list // appends the given bits to the end of the list
func (bl *BitList) AddBit(bit bool) { func (bl *BitList) AddBit(bits ...bool) {
for _, bit := range bits {
itmIndex := bl.count / 32 itmIndex := bl.count / 32
for itmIndex >= len(bl.data) { for itmIndex >= len(bl.data) {
bl.grow() bl.grow()
@ -46,6 +47,7 @@ func (bl *BitList) AddBit(bit bool) {
bl.SetBit(bl.count, bit) bl.SetBit(bl.count, bit)
bl.count++ bl.count++
} }
}
// sets the bit at the given index to the given value // sets the bit at the given index to the given value
func (bl *BitList) SetBit(index int, value bool) { func (bl *BitList) SetBit(index int, value bool) {
@ -79,13 +81,6 @@ func (bl *BitList) AddBits(b int, count byte) {
} }
} }
// appends all bits in the bool-slice to the end of the list
func (bl *BitList) AddRange(bits []bool) {
for _, b := range bits {
bl.AddBit(b)
}
}
// returns all bits of the BitList as a []byte // returns all bits of the BitList as a []byte
func (bl *BitList) GetBytes() []byte { func (bl *BitList) GetBytes() []byte {
len := bl.count >> 3 len := bl.count >> 3