barcode/code128/encode.go

213 lines
4.9 KiB
Go
Raw Normal View History

2014-08-11 09:11:24 +00:00
// Package code128 can create Code128 barcodes
2013-12-15 20:56:38 +00:00
package code128
import (
"fmt"
2013-12-16 13:10:24 +00:00
"strings"
2013-12-15 20:56:38 +00:00
"unicode/utf8"
2014-08-11 09:11:24 +00:00
"github.com/boombuler/barcode"
"github.com/boombuler/barcode/utils"
2013-12-15 20:56:38 +00:00
)
func strToRunes(str string) []rune {
result := make([]rune, utf8.RuneCountInString(str))
i := 0
for _, r := range str {
result[i] = r
i++
}
return result
}
2013-12-16 13:10:24 +00:00
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 i%2 == 0 && nextRunes[i] == FNC1 {
requiredDigits++
if len(nextRunes) < requiredDigits {
return false
}
continue
}
2013-12-16 13:10:24 +00:00
if nextRunes[i] < '0' || nextRunes[i] > '9' {
return false
}
}
return true
}
2017-04-03 18:31:06 +00:00
func tableContainsRune(table string, r rune) bool {
return strings.ContainsRune(table, r) || r == FNC1 || r == FNC2 || r == FNC3 || r == FNC4
}
func shouldUseATable(nextRunes []rune, curEncoding byte) bool {
nextRune := nextRunes[0]
if !tableContainsRune(bTable, nextRune) || curEncoding == startASymbol {
return tableContainsRune(aTable, nextRune)
}
if curEncoding == 0 {
for _, r := range nextRunes {
if tableContainsRune(abTable, r) {
continue
}
if strings.ContainsRune(aOnlyTable, r) {
return true
}
break
}
}
return false
}
2013-12-16 13:10:24 +00:00
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 {
2014-08-12 11:53:58 +00:00
if curEncoding == byte(0) {
result.AddByte(startCSymbol)
} else {
result.AddByte(codeCSymbol)
}
2013-12-16 13:10:24 +00:00
curEncoding = startCSymbol
}
if content[i] == FNC1 {
result.AddByte(102)
} else {
idx := (content[i] - '0') * 10
i++
idx = idx + (content[i] - '0')
result.AddByte(byte(idx))
}
2017-04-03 18:31:06 +00:00
} else if shouldUseATable(content[i:], curEncoding) {
if curEncoding != startASymbol {
if curEncoding == byte(0) {
result.AddByte(startASymbol)
} else {
result.AddByte(codeASymbol)
}
curEncoding = startASymbol
}
var idx int
switch content[i] {
case FNC1:
idx = 102
break
case FNC2:
idx = 97
break
case FNC3:
idx = 96
break
case FNC4:
idx = 101
break
default:
idx = strings.IndexRune(aTable, content[i])
break
}
if idx < 0 {
return nil
}
result.AddByte(byte(idx))
2013-12-16 13:10:24 +00:00
} else {
if curEncoding != startBSymbol {
2014-08-12 11:53:58 +00:00
if curEncoding == byte(0) {
result.AddByte(startBSymbol)
} else {
result.AddByte(codeBSymbol)
}
2013-12-16 13:10:24 +00:00
curEncoding = startBSymbol
}
2014-08-12 11:53:58 +00:00
var idx int
switch content[i] {
case FNC1:
idx = 102
break
case FNC2:
idx = 97
break
case FNC3:
idx = 96
break
case FNC4:
idx = 100
break
default:
idx = strings.IndexRune(bTable, content[i])
break
}
2013-12-16 13:10:24 +00:00
if idx < 0 {
return nil
}
result.AddByte(byte(idx))
}
}
return result
}
// Encode creates a Code 128 barcode for the given content and color scheme
func EncodeWithColor(content string, color barcode.ColorScheme) (barcode.BarcodeIntCS, error) {
2013-12-15 20:56:38 +00:00
contentRunes := strToRunes(content)
2014-08-12 11:53:58 +00:00
if len(contentRunes) <= 0 || len(contentRunes) > 80 {
2013-12-16 13:10:24 +00:00
return nil, fmt.Errorf("content length should be between 1 and 80 runes but got %d", len(contentRunes))
}
idxList := getCodeIndexList(contentRunes)
2013-12-15 20:56:38 +00:00
if idxList == nil {
return nil, fmt.Errorf("\"%s\" could not be encoded", content)
}
2013-12-16 13:10:24 +00:00
result := new(utils.BitList)
2013-12-15 20:56:38 +00:00
sum := 0
for i, idx := range idxList.GetBytes() {
if i == 0 {
sum = int(idx)
} else {
sum += i * int(idx)
}
2013-12-16 13:10:24 +00:00
result.AddBit(encodingTable[idx]...)
2013-12-15 20:56:38 +00:00
}
sum = sum % 103
result.AddBit(encodingTable[sum]...)
2013-12-16 13:10:24 +00:00
result.AddBit(encodingTable[stopSymbol]...)
return utils.New1DCodeIntCheckSumWithColor(barcode.TypeCode128, content, result, sum, color), nil
}
// Encode creates a Code 128 barcode for the given content
func Encode(content string) (barcode.BarcodeIntCS, error) {
return EncodeWithColor(content, barcode.ColorScheme16)
2013-12-15 20:56:38 +00:00
}
func EncodeWithoutChecksum(content string) (barcode.Barcode, error) {
2024-08-02 15:30:54 +00:00
return EncodeWithoutChecksumWithColor(content, barcode.ColorScheme16)
}
func EncodeWithoutChecksumWithColor(content string, color barcode.ColorScheme) (barcode.Barcode, error) {
contentRunes := strToRunes(content)
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 {
return nil, fmt.Errorf("\"%s\" could not be encoded", content)
}
result := new(utils.BitList)
for _, idx := range idxList.GetBytes() {
result.AddBit(encodingTable[idx]...)
}
result.AddBit(encodingTable[stopSymbol]...)
2024-08-02 15:30:54 +00:00
return utils.New1DCodeWithColor(barcode.TypeCode128, content, result, color), nil
}