diff options
author | Přemysl Janouch <p@janouch.name> | 2019-04-06 21:31:11 +0200 |
---|---|---|
committer | Přemysl Janouch <p@janouch.name> | 2019-04-07 23:08:27 +0200 |
commit | ab253ce768c410f6c996fd3df98292df0554c2f3 (patch) | |
tree | 7b09a24e60689308620f00ff3d32bcdd40e92a83 | |
download | sklad-ab253ce768c410f6c996fd3df98292df0554c2f3.tar.gz sklad-ab253ce768c410f6c996fd3df98292df0554c2f3.tar.xz sklad-ab253ce768c410f6c996fd3df98292df0554c2f3.zip |
Initial commit
-rw-r--r-- | bdf-preview/main.go | 93 | ||||
-rw-r--r-- | bdf-sample/main.go | 41 | ||||
-rw-r--r-- | bdf/bdf.go | 353 | ||||
-rw-r--r-- | brother-info/main.go | 207 | ||||
-rw-r--r-- | go.mod | 5 | ||||
-rw-r--r-- | go.sum | 2 | ||||
-rw-r--r-- | label-exp/main.go | 402 |
7 files changed, 1103 insertions, 0 deletions
diff --git a/bdf-preview/main.go b/bdf-preview/main.go new file mode 100644 index 0000000..b1f2045 --- /dev/null +++ b/bdf-preview/main.go @@ -0,0 +1,93 @@ +package main + +import ( + "html/template" + "image" + "image/draw" + "image/png" + "log" + "net/http" + "os" + + "janouch.name/sklad/bdf" +) + +type fontItem struct { + Font *bdf.Font + Preview image.Image +} + +var fonts = map[string]fontItem{} + +var tmpl = template.Must(template.New("list").Parse(` + <!DOCTYPE html> + <html><body> + <table border='1' cellpadding='3' style='border-collapse: collapse'> + <tr> + <th>Name</th> + <th>Preview</th> + <tr> +{{range $k, $v := . }} + <tr> + <td>{{$k}}</td> + <td><img src='?name={{$k}}'></td> + </tr> +{{end}} + </table> + </body></html> +`)) + +func handle(w http.ResponseWriter, r *http.Request) { + if err := r.ParseForm(); err != nil { + http.Error(w, err.Error(), 500) + return + } + + name := r.FormValue("name") + if name == "" { + w.Header().Set("Content-Type", "text/html") + tmpl.Execute(w, fonts) + return + } + + item, ok := fonts[name] + if !ok { + http.Error(w, "No such font.", 400) + return + } + + w.Header().Set("Content-Type", "image/png") + if err := png.Encode(w, item.Preview); err != nil { + http.Error(w, err.Error(), 500) + return + } +} + +func main() { + for _, filename := range os.Args[1:] { + fi, err := os.Open(filename) + if err != nil { + log.Fatalln(err) + } + font, err := bdf.NewFromBDF(fi) + if err != nil { + log.Fatalf("%s: %s\n", filename, err) + } + if err := fi.Close(); err != nil { + log.Fatalln(err) + } + + r, _ := font.BoundString(font.Name) + super := r.Inset(-3) + + img := image.NewRGBA(super) + draw.Draw(img, super, image.White, image.ZP, draw.Src) + font.DrawString(img, image.ZP, font.Name) + + fonts[filename] = fontItem{Font: font, Preview: img} + } + + log.Println("Starting server") + http.HandleFunc("/", handle) + log.Fatal(http.ListenAndServe(":8080", nil)) +} diff --git a/bdf-sample/main.go b/bdf-sample/main.go new file mode 100644 index 0000000..ae9072f --- /dev/null +++ b/bdf-sample/main.go @@ -0,0 +1,41 @@ +package main + +import ( + "image" + "image/draw" + "image/png" + "janouch.name/sklad/bdf" + "log" + "os" +) + +func main() { + fi, err := os.Open(os.Args[1]) + if err != nil { + log.Fatalln(err) + } + defer fi.Close() + font, err := bdf.NewFromBDF(fi) + if err != nil { + log.Fatalln(err) + } + + r, _ := font.BoundString(font.Name) + super := r.Inset(-20) + + img := image.NewRGBA(super) + draw.Draw(img, super, image.White, image.ZP, draw.Src) + font.DrawString(img, image.ZP, font.Name) + + fo, err := os.Create("out.png") + if err != nil { + log.Fatalln(err) + } + if err := png.Encode(fo, img); err != nil { + fo.Close() + log.Fatal(err) + } + if err := fo.Close(); err != nil { + log.Fatal(err) + } +} diff --git a/bdf/bdf.go b/bdf/bdf.go new file mode 100644 index 0000000..c02e31e --- /dev/null +++ b/bdf/bdf.go @@ -0,0 +1,353 @@ +package bdf + +import ( + "bufio" + "encoding/hex" + "fmt" + "image" + "image/color" + "image/draw" + "io" + "strconv" +) + +// glyph is a singular bitmap glyph to be used as a mask, assumed to directly +// correspond to a rune. A zero value is also valid and drawable. +type glyph struct { + // Coordinates are relative to the origin, on the baseline. + // The ascent is thus negative, unlike the usual model. + bounds image.Rectangle + bitmap []byte + advance int +} + +// ColorModel implements image.Image. +func (g *glyph) ColorModel() color.Model { return color.Alpha16Model } + +// Bounds implements image.Image. +func (g *glyph) Bounds() image.Rectangle { return g.bounds } + +// At implements image.Image. This is going to be somewhat slow. +func (g *glyph) At(x, y int) color.Color { + x -= g.bounds.Min.X + y -= g.bounds.Min.Y + + dx, dy := g.bounds.Dx(), g.bounds.Dy() + if x < 0 || y < 0 || x >= dx || y >= dy { + return color.Transparent + } + + stride, offset, bit := (dx+7)/8, x/8, byte(1<<uint(7-x%8)) + if g.bitmap[y*stride+offset]&bit == 0 { + return color.Transparent + } + return color.Opaque +} + +// ----------------------------------------------------------------------------- + +// Font represents a particular bitmap font. +type Font struct { + Name string + glyphs map[rune]glyph + fallback glyph +} + +// FindGlyph returns the best glyph to use for the given rune. +// The returned boolean indicates whether a fallback has been used. +func (f *Font) FindGlyph(r rune) (glyph, bool) { + if g, ok := f.glyphs[r]; ok { + return g, true + } + return f.fallback, false +} + +// DrawString draws the specified text string onto dst horizontally along +// the baseline starting at dp, using black color. +func (f *Font) DrawString(dst draw.Image, dp image.Point, s string) { + for _, r := range s { + g, _ := f.FindGlyph(r) + draw.DrawMask(dst, g.bounds.Add(dp), + image.Black, image.ZP, &g, g.bounds.Min, draw.Over) + dp.X += g.advance + } +} + +// BoundString measures the text's bounds when drawn along the X axis +// for the baseline. Also returns the total advance. +func (f *Font) BoundString(s string) (image.Rectangle, int) { + var ( + bounds image.Rectangle + dot image.Point + ) + for _, r := range s { + g, _ := f.FindGlyph(r) + bounds = bounds.Union(g.bounds.Add(dot)) + dot.X += g.advance + } + return bounds, dot.X +} + +// ----------------------------------------------------------------------------- + +func latin1ToUTF8(latin1 []byte) string { + buf := make([]rune, len(latin1)) + for i, b := range latin1 { + buf[i] = rune(b) + } + return string(buf) +} + +// tokenize splits a BDF line into tokens. Quoted strings may start anywhere +// on the line. We only enforce that they must end somewhere. +func tokenize(s string) (tokens []string, err error) { + token, quotes, escape := []rune{}, false, false + for _, r := range s { + switch { + case escape: + switch r { + case '"': + escape = false + token = append(token, r) + case ' ', '\t': + quotes, escape = false, false + tokens = append(tokens, string(token)) + token = nil + default: + quotes, escape = false, false + token = append(token, r) + } + case quotes: + switch r { + case '"': + escape = true + default: + token = append(token, r) + } + default: + switch r { + case '"': + // We could also enable quote processing on demand, + // so that it is only turned on in properties. + if len(tokens) < 1 || tokens[0] != "COMMENT" { + quotes = true + } else { + token = append(token, r) + } + case ' ', '\t': + if len(token) > 0 { + tokens = append(tokens, string(token)) + token = nil + } + default: + token = append(token, r) + } + } + } + if quotes && !escape { + return nil, fmt.Errorf("strings may not contain newlines") + } + if quotes || len(token) > 0 { + tokens = append(tokens, string(token)) + } + return tokens, nil +} + +// ----------------------------------------------------------------------------- + +// bdfParser is a basic and rather lenient parser of +// Bitmap Distribution Format (BDF) files. +type bdfParser struct { + scanner *bufio.Scanner // input reader + line int // current line number + tokens []string // tokens on the current line + font *Font // glyph storage + + defaultBounds image.Rectangle + defaultAdvance int + defaultChar int +} + +// readLine reads the next line and splits it into tokens. +// Panics on error, returns false if the end of file has been reached normally. +func (p *bdfParser) readLine() bool { + p.line++ + if !p.scanner.Scan() { + if err := p.scanner.Err(); err != nil { + panic(err) + } + p.line-- + return false + } + + var err error + if p.tokens, err = tokenize(latin1ToUTF8(p.scanner.Bytes())); err != nil { + panic(err) + } + + // Eh, it would be nicer iteratively, this may overrun the stack. + if len(p.tokens) == 0 { + return p.readLine() + } + return true +} + +func (p *bdfParser) readCharEncoding() int { + if len(p.tokens) < 2 { + panic("insufficient arguments") + } + if i, err := strconv.Atoi(p.tokens[1]); err != nil { + panic(err) + } else { + return i // Some fonts even use -1 for things outside the encoding. + } +} + +func (p *bdfParser) parseProperties() { + // The wording in the specification suggests that the argument + // with the number of properties to follow isn't reliable. + for p.readLine() && p.tokens[0] != "ENDPROPERTIES" { + switch p.tokens[0] { + case "DEFAULT_CHAR": + p.defaultChar = p.readCharEncoding() + } + } +} + +// XXX: Ignoring vertical advance since we only expect purely horizontal fonts. +func (p *bdfParser) readDwidth() int { + if len(p.tokens) < 2 { + panic("insufficient arguments") + } + if i, err := strconv.Atoi(p.tokens[1]); err != nil { + panic(err) + } else { + return i + } +} + +func (p *bdfParser) readBBX() image.Rectangle { + if len(p.tokens) < 5 { + panic("insufficient arguments") + } + w, e1 := strconv.Atoi(p.tokens[1]) + h, e2 := strconv.Atoi(p.tokens[2]) + x, e3 := strconv.Atoi(p.tokens[3]) + y, e4 := strconv.Atoi(p.tokens[4]) + if e1 != nil || e2 != nil || e3 != nil || e4 != nil { + panic("invalid arguments") + } + if w < 0 || h < 0 { + panic("bounding boxes may not have negative dimensions") + } + return image.Rectangle{ + Min: image.Point{x, -(y + h)}, + Max: image.Point{x + w, -y}, + } +} + +func (p *bdfParser) parseChar() { + g := glyph{bounds: p.defaultBounds, advance: p.defaultAdvance} + bitmap, rows, encoding := false, 0, -1 + for p.readLine() && p.tokens[0] != "ENDCHAR" { + if bitmap { + b, err := hex.DecodeString(p.tokens[0]) + if err != nil { + panic(err) + } + if len(b) != (g.bounds.Dx()+7)/8 { + panic("invalid bitmap data, width mismatch") + } + g.bitmap = append(g.bitmap, b...) + rows++ + } else { + switch p.tokens[0] { + case "ENCODING": + encoding = p.readCharEncoding() + case "DWIDTH": + g.advance = p.readDwidth() + case "BBX": + g.bounds = p.readBBX() + case "BITMAP": + bitmap = true + } + } + } + if rows != g.bounds.Dy() { + panic("invalid bitmap data, height mismatch") + } + + // XXX: We don't try to convert encodings, since we'd need x/text/encoding + // for the conversion tables, though most fonts are at least going to use + // supersets of ASCII. Use ISO10646-1 X11 fonts for proper Unicode support. + if encoding >= 0 { + p.font.glyphs[rune(encoding)] = g + } + if encoding == p.defaultChar { + p.font.fallback = g + } +} + +// https://en.wikipedia.org/wiki/Glyph_Bitmap_Distribution_Format +// https://www.adobe.com/content/dam/acom/en/devnet/font/pdfs/5005.BDF_Spec.pdf +func (p *bdfParser) parse() { + if !p.readLine() || len(p.tokens) != 2 || p.tokens[0] != "STARTFONT" { + panic("invalid header") + } + if p.tokens[1] != "2.1" && p.tokens[1] != "2.2" { + panic("unsupported version number") + } + for p.readLine() && p.tokens[0] != "ENDFONT" { + switch p.tokens[0] { + case "FONT": + if len(p.tokens) < 2 { + panic("insufficient arguments") + } + p.font.Name = p.tokens[1] + case "FONTBOUNDINGBOX": + // There's no guarantee that this includes all BBXs. + p.defaultBounds = p.readBBX() + case "METRICSSET": + if len(p.tokens) < 2 { + panic("insufficient arguments") + } + if p.tokens[1] == "1" { + panic("purely vertical fonts are unsupported") + } + case "DWIDTH": + p.defaultAdvance = p.readDwidth() + case "STARTPROPERTIES": + p.parseProperties() + case "STARTCHAR": + p.parseChar() + } + } + if p.font.Name == "" { + panic("the font file doesn't contain the font's name") + } + if len(p.font.glyphs) == 0 { + panic("the font file doesn't seem to contain any glyphs") + } +} + +func NewFromBDF(r io.Reader) (f *Font, err error) { + p := bdfParser{ + scanner: bufio.NewScanner(r), + font: &Font{glyphs: make(map[rune]glyph)}, + defaultChar: -1, + } + defer func() { + if r := recover(); r != nil { + var ok bool + if err, ok = r.(error); !ok { + err = fmt.Errorf("%v", r) + } + } + if err != nil { + err = fmt.Errorf("line %d: %s", p.line, err) + } + }() + + p.parse() + return p.font, nil +} diff --git a/brother-info/main.go b/brother-info/main.go new file mode 100644 index 0000000..b65b1dc --- /dev/null +++ b/brother-info/main.go @@ -0,0 +1,207 @@ +package main + +import ( + "io" + "log" + "os" + "time" +) + +func decodeBitfieldErrors(b byte, errors [8]string) []string { + var result []string + for i := uint(0); i < 8; i++ { + if b&(1<<i) != 0 { + result = append(result, errors[i]) + } + } + return result +} + +// ----------------------------------------------------------------------------- + +type brotherStatus struct { + errors []string +} + +// TODO: What exactly do we need? Probably extend as needed. +func decodeStatusInformation(d []byte) brotherStatus { + var status brotherStatus + status.errors = append(status.errors, decodeBitfieldErrors(d[8], [8]string{ + "no media", "end of media", "cutter jam", "?", "printer in use", + "printer turned off", "high-voltage adapter", "fan motor error"})...) + status.errors = append(status.errors, decodeBitfieldErrors(d[9], [8]string{ + "replace media", "expansion buffer full", "communication error", + "communication buffer full", "cover open", "cancel key", + "media cannot be fed", "system error"})...) + return status +} + +// ----------------------------------------------------------------------------- + +func printStatusInformation(d []byte) { + if d[0] != 0x80 || d[1] != 0x20 || d[2] != 0x42 || d[3] != 0x34 { + log.Println("unexpected status fixed bytes") + } + + // Model code. + switch b := d[4]; b { + case 0x38: + log.Println("model: QL-800") + case 0x39: + log.Println("model: QL-810W") + case 0x41: + log.Println("model: QL-820NWB") + default: + log.Println("model:", b) + } + + // d[6] seems to be 0x00 in a real-world QL-800. + if d[5] != 0x30 || d[6] != 0x30 || d[7] != 0x00 { + log.Println("unexpected status fixed bytes") + } + + // Error information 1. + for _, e := range decodeBitfieldErrors(d[8], [8]string{ + "no media", "end of media", "cutter jam", "?", "printer in use", + "printer turned off", "high-voltage adapter", "fan motor error"}) { + log.Println("error:", e) + } + + // Error information 2. + for _, e := range decodeBitfieldErrors(d[9], [8]string{ + "replace media", "expansion buffer full", "communication error", + "communication buffer full", "cover open", "cancel key", + "media cannot be fed", "system error"}) { + log.Println("error:", e) + } + + // Media width. + log.Println("media width:", d[10], "mm") + + // Media type. + switch b := d[11]; b { + case 0x00: + log.Println("media: no media") + case 0x4a: + log.Println("media: continuous length tape") + case 0x4b: + log.Println("media: die-cut labels") + default: + log.Println("media:", b) + } + + // d[14] seems to be 0x14 in a real-world QL-800. + if d[12] != 0x00 || d[13] != 0x00 || d[14] != 0x3f { + log.Println("unexpected status fixed bytes") + } + + // Mode. + log.Println("mode:", d[15]) + + if d[16] != 0x00 { + log.Println("unexpected status fixed bytes") + } + + // Media length. + log.Println("media width:", d[17], "mm") + + // Status type. + switch b := d[18]; b { + case 0x00: + log.Println("status type: reply to status request") + case 0x01: + log.Println("status type: printing completed") + case 0x02: + log.Println("status type: error occurred") + case 0x04: + log.Println("status type: turned off") + case 0x05: + log.Println("status type: notification") + case 0x06: + log.Println("status type: phase change") + default: + log.Println("status type:", b) + } + + // Phase type. + switch b := d[19]; b { + case 0x00: + log.Println("phase state: receiving state") + case 0x01: + log.Println("phase state: printing state") + default: + log.Println("phase state:", b) + } + + // Phase number. + log.Println("phase number:", int(d[20])*256+int(d[21])) + + // Notification number. + switch b := d[22]; b { + case 0x00: + log.Println("notification number: not available") + case 0x03: + log.Println("notification number: cooling (started)") + case 0x04: + log.Println("notification number: cooling (finished)") + default: + log.Println("notification number:", b) + } + + // d[25] seems to be 0x01 in a real-world QL-800. + if d[23] != 0x00 || d[24] != 0x00 || d[25] != 0x00 || d[26] != 0x00 || + d[27] != 0x00 || d[28] != 0x00 || d[29] != 0x00 || d[30] != 0x00 || + d[31] != 0x00 { + log.Println("unexpected status fixed bytes") + } +} + +func main() { + // Linux usblp module, located in /drivers/usb/class/usblp.c + // (at least that's where the trails go, I don't understand the code) + f, err := os.OpenFile("/dev/usb/lp0", os.O_RDWR, 0) + if err != nil { + log.Fatalln(err) + } + defer f.Close() + + // Flush any former responses in the printer's queue. + for { + dummy := make([]byte, 32) + if _, err := f.Read(dummy); err == io.EOF { + break + } + } + + // Clear the print buffer. + invalidate := make([]byte, 400) + if _, err := f.Write(invalidate); err != nil { + log.Fatalln(err) + } + + // Initialize. + if _, err := f.WriteString("\x1b\x40"); err != nil { + log.Fatalln(err) + } + + // Request status information. + if _, err := f.WriteString("\x1b\x69\x53"); err != nil { + log.Fatalln(err) + } + + // We need to poll the device. + status := make([]byte, 32) + for { + if n, err := f.Read(status); err == io.EOF { + time.Sleep(10 * time.Millisecond) + } else if err != nil { + log.Fatalln(err) + } else if n < 32 { + log.Fatalln("invalid read") + } else { + break + } + } + + printStatusInformation(status) +} @@ -0,0 +1,5 @@ +module janouch.name/sklad + +go 1.12 + +require github.com/boombuler/barcode v1.0.0 @@ -0,0 +1,2 @@ +github.com/boombuler/barcode v1.0.0 h1:s1TvRnXwL2xJRaccrdcBQMZxq6X7DvsMogtmJeHDdrc= +github.com/boombuler/barcode v1.0.0/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= diff --git a/label-exp/main.go b/label-exp/main.go new file mode 100644 index 0000000..442c021 --- /dev/null +++ b/label-exp/main.go @@ -0,0 +1,402 @@ +package main + +import ( + "errors" + "html/template" + "image" + "image/color" + "image/draw" + "image/png" + "io" + "log" + "net/http" + "os" + "strconv" + "time" + + "github.com/boombuler/barcode" + "github.com/boombuler/barcode/qr" + + "janouch.name/sklad/bdf" +) + +// scaler is a scaling image.Image wrapper. +type scaler struct { + image image.Image + scale int +} + +// ColorModel implements image.Image. +func (s *scaler) ColorModel() color.Model { + return s.image.ColorModel() +} + +// Bounds implements image.Image. +func (s *scaler) Bounds() image.Rectangle { + r := s.image.Bounds() + return image.Rect(r.Min.X*s.scale, r.Min.Y*s.scale, + r.Max.X*s.scale, r.Max.Y*s.scale) +} + +// At implements image.Image. +func (s *scaler) At(x, y int) color.Color { + if x < 0 { + x = x - s.scale + 1 + } + if y < 0 { + y = y - s.scale + 1 + } + return s.image.At(x/s.scale, y/s.scale) +} + +// ----------------------------------------------------------------------------- + +func decodeBitfieldErrors(b byte, errors [8]string) []string { + var result []string + for i := uint(0); i < 8; i++ { + if b&(1<<i) != 0 { + result = append(result, errors[i]) + } + } + return result +} + +func printStatusInformation(d []byte) { + log.Println("-- status") + + // Error information 1. + for _, e := range decodeBitfieldErrors(d[8], [8]string{ + "no media", "end of media", "cutter jam", "?", "printer in use", + "printer turned off", "high-voltage adapter", "fan motor error"}) { + log.Println("error:", e) + } + + // Error information 2. + for _, e := range decodeBitfieldErrors(d[9], [8]string{ + "replace media", "expansion buffer full", "communication error", + "communication buffer full", "cover open", "cancel key", + "media cannot be fed", "system error"}) { + log.Println("error:", e) + } + + // Media width. + log.Println("media width:", d[10], "mm") + + // Media type. + switch b := d[11]; b { + case 0x00: + log.Println("media: no media") + case 0x4a: + log.Println("media: continuous length tape") + case 0x4b: + log.Println("media: die-cut labels") + default: + log.Println("media:", b) + } + + // Mode. + log.Println("mode:", d[15]) + + // Media length. + log.Println("media width:", d[17], "mm") + + // Status type. + switch b := d[18]; b { + case 0x00: + log.Println("status type: reply to status request") + case 0x01: + log.Println("status type: printing completed") + case 0x02: + log.Println("status type: error occurred") + case 0x04: + log.Println("status type: turned off") + case 0x05: + log.Println("status type: notification") + case 0x06: + log.Println("status type: phase change") + default: + log.Println("status type:", b) + } + + // Phase type. + switch b := d[19]; b { + case 0x00: + log.Println("phase state: receiving state") + case 0x01: + log.Println("phase state: printing state") + default: + log.Println("phase state:", b) + } + + // Phase number. + log.Println("phase number:", int(d[20])*256+int(d[21])) + + // Notification number. + switch b := d[22]; b { + case 0x00: + log.Println("notification number: not available") + case 0x03: + log.Println("notification number: cooling (started)") + case 0x04: + log.Println("notification number: cooling (finished)") + default: + log.Println("notification number:", b) + } +} + +// genLabelData converts an image to the printer's raster format. +func genLabelData(src image.Image, offset int) (data []byte) { + // TODO: Margins? For 29mm, it's 6 pins from the start, 306 printing pins. + bounds := src.Bounds() + pixels := make([]bool, 720) + for y := bounds.Min.Y; y < bounds.Max.Y; y++ { + off := offset + for x := bounds.Max.X - 1; x >= bounds.Min.X; x-- { + // TODO: Anything to do with the ColorModel? + r, g, b, a := src.At(x, y).RGBA() + pixels[off] = r == 0 && g == 0 && b == 0 && a != 0 + off++ + } + + data = append(data, 'g', 0x00, 90) + for i := 0; i < 90; i++ { + var b byte + for j := 0; j < 8; j++ { + b <<= 1 + if pixels[i*8+j] { + b |= 1 + } + } + data = append(data, b) + } + } + return +} + +func printLabel(src image.Image) error { + data := []byte(nil) + + // Raster mode. + // Should be the only supported mode for QL-800. + data = append(data, 0x1b, 0x69, 0x61, 0x01) + + // Automatic status mode (though it's the default). + data = append(data, 0x1b, 0x69, 0x21, 0x00) + + // Print information command. + dy := src.Bounds().Dy() + data = append(data, 0x1b, 0x69, 0x7a, 0x02|0x04|0x40|0x80, 0x0a, 29, 0, + byte(dy), byte(dy>>8), byte(dy>>16), byte(dy>>24), 0, 0x00) + + // Auto cut, each 1 label. + data = append(data, 0x1b, 0x69, 0x4d, 0x40) + data = append(data, 0x1b, 0x69, 0x41, 0x01) + + // Cut at end (though it's the default). + // Not sure what it means, doesn't seem to have any effect to turn it off. + data = append(data, 0x1b, 0x69, 0x4b, 0x08) + + // 3mm margins along the direction of feed. 0x23 = 35 dots, the minimum. + data = append(data, 0x1b, 0x69, 0x64, 0x23, 0x00) + + // Compression mode: no compression. + // Should be the only supported mode for QL-800. + data = append(data, 0x4d, 0x00) + + // The graphics data itself. + data = append(data, genLabelData(src, 6)...) + + // Print command with feeding. + data = append(data, 0x1a) + + // --- + + // Linux usblp module, located in /drivers/usb/class/usblp.c + // (at least that's where the trails go, I don't understand the code) + f, err := os.OpenFile("/dev/usb/lp0", os.O_RDWR, 0) + if err != nil { + return err + } + defer f.Close() + + // Flush any former responses in the printer's queue. + for { + dummy := make([]byte, 32) + if _, err := f.Read(dummy); err == io.EOF { + break + } + } + + // Clear the print buffer. + invalidate := make([]byte, 400) + if _, err := f.Write(invalidate); err != nil { + return err + } + + // Initialize. + if _, err := f.WriteString("\x1b\x40"); err != nil { + return err + } + + // Request status information. + if _, err := f.WriteString("\x1b\x69\x53"); err != nil { + return err + } + + // We need to poll the device. + status := make([]byte, 32) + for { + if n, err := f.Read(status); err == io.EOF { + time.Sleep(10 * time.Millisecond) + } else if err != nil { + return err + } else if n < 32 { + return errors.New("invalid read") + } else { + break + } + } + printStatusInformation(status) + + // Print the prepared data. + if _, err := f.Write(data); err != nil { + return err + } + + // TODO: We specifically need to wait for a transition to the receiving + // state, and try to figure out something from the statuses. + // We may also receive an error status instead of the transition to + // the printing state. Or even after it. + start := time.Now() + for { + if time.Now().Sub(start) > 3*time.Second { + break + } + if n, err := f.Read(status); err == io.EOF { + time.Sleep(100 * time.Millisecond) + } else if err != nil { + return err + } else if n < 32 { + return errors.New("invalid read") + } else { + printStatusInformation(status) + } + } + return nil +} + +// ----------------------------------------------------------------------------- + +var font *bdf.Font + +// TODO: By rotating the label we can make it smaller, as an alternate mode. +func genLabel(text string, width int) image.Image { + // Create a scaled bitmap of the QR code. + qrImg, _ := qr.Encode(text, qr.H, qr.Auto) + qrImg, _ = barcode.Scale(qrImg, 306, 306) + qrRect := qrImg.Bounds() + + // Create a scaled bitmap of the text label. + textRect, _ := font.BoundString(text) + textImg := image.NewRGBA(textRect) + draw.Draw(textImg, textRect, image.White, image.ZP, draw.Src) + font.DrawString(textImg, image.ZP, text) + + // TODO: We can scale as needed to make the text fit. + scaledTextImg := scaler{image: textImg, scale: 3} + scaledTextRect := scaledTextImg.Bounds() + + // Combine. + combinedRect := qrRect + combinedRect.Max.Y += scaledTextRect.Dy() + 20 + + combinedImg := image.NewRGBA(combinedRect) + draw.Draw(combinedImg, combinedRect, image.White, image.ZP, draw.Src) + draw.Draw(combinedImg, combinedRect, qrImg, image.ZP, draw.Src) + + target := image.Rect( + (width-scaledTextRect.Dx())/2, qrRect.Dy()+10, + combinedRect.Max.X, combinedRect.Max.Y) + draw.Draw(combinedImg, target, &scaledTextImg, scaledTextRect.Min, draw.Src) + return combinedImg +} + +var tmpl = template.Must(template.New("form").Parse(` + <!DOCTYPE html> + <html><body> + <table><tr> + <td valign=top> + <img border=1 src='?img&width={{.Width}}&text={{.Text}}'> + </td> + <td valign=top> + <form><fieldset> + <p><label for=width>Tape width in pt:</label> + <input id=width name=width value='{{.Width}}'> + <p><label for=text>Text:</label> + <input id=text name=text value='{{.Text}}'> + <p><input type=submit value='Update'> + <input type=checkbox id=print name=print> + <label for=print>Print label</label> + </fieldset></form> + </td> + </tr></table> + </body></html> +`)) + +func handle(w http.ResponseWriter, r *http.Request) { + if err := r.ParseForm(); err != nil { + http.Error(w, err.Error(), 500) + return + } + + var params = struct { + Width int + Text string + }{ + Text: r.FormValue("text"), + } + + var err error + params.Width, err = strconv.Atoi(r.FormValue("width")) + if err != nil { + params.Width = 306 // Default to 29mm tape. + } + + label := genLabel(params.Text, params.Width) + if r.FormValue("print") != "" { + if err := printLabel(label); err != nil { + log.Println("print error:", err) + } + } + + if _, ok := r.Form["img"]; !ok { + w.Header().Set("Content-Type", "text/html") + tmpl.Execute(w, ¶ms) + return + } + + w.Header().Set("Content-Type", "image/png") + if err := png.Encode(w, label); err != nil { + http.Error(w, err.Error(), 500) + return + } +} + +func main() { + var err error + fi, err := os.Open("../../ucs-fonts-75dpi100dpi/100dpi/luBS24.bdf") + if err != nil { + log.Fatalln(err) + } + font, err = bdf.NewFromBDF(fi) + if err != nil { + log.Fatalln(err) + } + if err := fi.Close(); err != nil { + log.Fatalln(err) + } + + log.Println("Starting server") + http.HandleFunc("/", handle) + log.Fatal(http.ListenAndServe(":8080", nil)) +} |