Compare commits

...

4 Commits

6 changed files with 149 additions and 109 deletions

View File

@ -4,114 +4,74 @@ import (
"errors" "errors"
"html/template" "html/template"
"image" "image"
"image/draw"
"image/png" "image/png"
"log" "log"
"net/http" "net/http"
"os" "os"
"strconv" "strconv"
"github.com/boombuler/barcode"
"github.com/boombuler/barcode/qr"
"janouch.name/sklad/bdf" "janouch.name/sklad/bdf"
"janouch.name/sklad/imgutil" "janouch.name/sklad/imgutil"
"janouch.name/sklad/label"
"janouch.name/sklad/ql" "janouch.name/sklad/ql"
) )
var font *bdf.Font var font *bdf.Font
func genLabelForHeight(text string, height, scale int) image.Image {
// 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)
scaledTextImg := imgutil.Scale{Image: textImg, Scale: scale}
scaledTextRect := scaledTextImg.Bounds()
remains := height - scaledTextRect.Dy() - 20
width := scaledTextRect.Dx()
if remains > width {
width = remains
}
// Create a scaled bitmap of the QR code.
qrImg, _ := qr.Encode(text, qr.H, qr.Auto)
qrImg, _ = barcode.Scale(qrImg, remains, remains)
qrRect := qrImg.Bounds()
// Combine.
combinedRect := image.Rect(0, 0, width, height)
combinedImg := image.NewRGBA(combinedRect)
draw.Draw(combinedImg, combinedRect, image.White, image.ZP, draw.Src)
draw.Draw(combinedImg,
combinedRect.Add(image.Point{X: (width - qrRect.Dx()) / 2, Y: 0}),
qrImg, image.ZP, draw.Src)
target := image.Rect(
(width-scaledTextRect.Dx())/2, qrRect.Dy()+20,
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(` var tmpl = template.Must(template.New("form").Parse(`
<!DOCTYPE html> <!DOCTYPE html>
<html><body> <html><body>
<h1>PT-CBP label printing tool</h1> <h1>PT-CBP label printing tool</h1>
<table><tr> <table><tr>
<td valign=top> <td valign=top>
<img border=1 src='?img&amp;scale={{.Scale}}&amp;text={{.Text}}'> <img border=1 src='?img&amp;scale={{.Scale}}&amp;text={{.Text}}'>
</td> </td>
<td valign=top> <td valign=top>
<fieldset> <fieldset>
{{ if .Printer }} {{ if .Printer }}
<p>Printer: {{ .Printer.Manufacturer }} {{ .Printer.Model }} <p>Printer: {{ .Printer.Manufacturer }} {{ .Printer.Model }}
<p>Tape: <p>Tape:
{{ if .Printer.LastStatus }} {{ if .Printer.LastStatus }}
{{ .Printer.LastStatus.MediaWidthMM }} mm &times; {{ .Printer.LastStatus.MediaWidthMM }} mm &times;
{{ .Printer.LastStatus.MediaLengthMM }} mm {{ .Printer.LastStatus.MediaLengthMM }} mm
{{ if .MediaInfo }} {{ if .MediaInfo }}
(offset: {{ .MediaInfo.SideMarginPins }} pt, (offset: {{ .MediaInfo.SideMarginPins }} pt,
print area: {{ .MediaInfo.PrintAreaPins }} pt) print area: {{ .MediaInfo.PrintAreaPins }} pt)
{{ else }} {{ else }}
(unknown media) (unknown media)
{{ end }} {{ end }}
{{ if .Printer.LastStatus.Errors }} {{ if .Printer.LastStatus.Errors }}
{{ range .Printer.LastStatus.Errors }} {{ range .Printer.LastStatus.Errors }}
<p>Error: {{ . }} <p>Error: {{ . }}
{{ end }} {{ end }}
{{ end }} {{ end }}
{{ end }} {{ end }}
{{ if .InitErr }} {{ if .InitErr }}
{{ .InitErr }} {{ .InitErr }}
{{ end }} {{ end }}
{{ else }} {{ else }}
<p>Error: {{ .PrinterErr }} <p>Error: {{ .PrinterErr }}
{{ end }} {{ end }}
</fieldset> </fieldset>
<fieldset> <fieldset>
<p>Font: {{ .Font.Name }} <p>Font: {{ .Font.Name }}
</fieldset> </fieldset>
<form><fieldset> <form><fieldset>
<p><label for=text>Text:</label> <p><label for=text>Text:</label>
<input id=text name=text value='{{.Text}}'> <input id=text name=text value='{{.Text}}'>
<label for=scale>Scale:</label> <label for=scale>Scale:</label>
<input id=scale name=scale value='{{.Scale}}' size=1> <input id=scale name=scale value='{{.Scale}}' size=1>
<p><input type=submit value='Update'> <p><input type=submit value='Update'>
<input type=submit name=print value='Update and Print'> <input type=submit name=print value='Update and Print'>
</fieldset></form> </fieldset></form>
</td> </td>
</tr></table> </tr></table>
</body></html> </body></html>
`)) `))
func getPrinter() (*ql.Printer, error) { func getPrinter() (*ql.Printer, error) {
@ -183,12 +143,12 @@ func handle(w http.ResponseWriter, r *http.Request) {
params.Scale = 3 params.Scale = 3
} }
var label image.Image var img image.Image
if mediaInfo != nil { if mediaInfo != nil {
label = &imgutil.LeftRotate{Image: genLabelForHeight( img = &imgutil.LeftRotate{Image: label.GenLabelForHeight(
params.Text, mediaInfo.PrintAreaPins, params.Scale)} font, params.Text, mediaInfo.PrintAreaPins, params.Scale)}
if r.FormValue("print") != "" { if r.FormValue("print") != "" {
if err := printer.Print(label); err != nil { if err := printer.Print(img); err != nil {
log.Println("print error:", err) log.Println("print error:", err)
} }
} }
@ -206,7 +166,7 @@ func handle(w http.ResponseWriter, r *http.Request) {
} }
w.Header().Set("Content-Type", "image/png") w.Header().Set("Content-Type", "image/png")
if err := png.Encode(w, label); err != nil { if err := png.Encode(w, img); err != nil {
http.Error(w, err.Error(), 500) http.Error(w, err.Error(), 500)
return return
} }

View File

@ -108,16 +108,14 @@ func (p *Printer) Initialize() error {
// Flush any former responses in the printer's queue. // Flush any former responses in the printer's queue.
// //
// I'm not sure if this is necessary, or rather whether the kernel driver // I haven't checked if this is the kernel driver or the printer doing
// does any buffering that could cause data to be returned at this point. // the buffering that causes data to be returned at this point.
/* var dummy [32]byte
var dummy [32]byte for {
for { if _, err := p.File.Read(dummy[:]); err == io.EOF {
if _, err := f.Read(dummy[:]); err == io.EOF { break
break
}
} }
*/ }
return nil return nil
} }

View File

@ -7,6 +7,8 @@ import (
"os" "os"
"strings" "strings"
"time" "time"
"janouch.name/sklad/bdf"
) )
type Series struct { type Series struct {
@ -46,6 +48,9 @@ type Database struct {
Prefix string // prefix for all container IDs Prefix string // prefix for all container IDs
Series []*Series // all known series Series []*Series // all known series
Containers []*Container // all known containers Containers []*Container // all known containers
BDFPath string // path to bitmap font file
BDFScale int // integer scaling for the bitmap font
} }
var ( var (
@ -57,6 +62,8 @@ var (
indexSeries = map[string]*Series{} indexSeries = map[string]*Series{}
indexContainer = map[ContainerId]*Container{} indexContainer = map[ContainerId]*Container{}
indexChildren = map[ContainerId][]*Container{} indexChildren = map[ContainerId][]*Container{}
labelFont *bdf.Font
) )
// TODO: Some functions to add, remove and change things in the database. // TODO: Some functions to add, remove and change things in the database.
@ -192,6 +199,20 @@ func loadDatabase() error {
} }
} }
// Prepare label printing.
if db.BDFScale <= 0 {
db.BDFScale = 1
}
if f, err := os.Open(db.BDFPath); err != nil {
return fmt.Errorf("cannot load label font: %s", err)
} else {
defer f.Close()
if labelFont, err = bdf.NewFromBDF(f); err != nil {
return fmt.Errorf("cannot load label font: %s", err)
}
}
// Open database log file for appending. // Open database log file for appending.
if dbLog, err = os.OpenFile(dbPath+".log", if dbLog, err = os.OpenFile(dbPath+".log",
os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644); err != nil { os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644); err != nil {

13
sklad/label.tmpl Normal file
View File

@ -0,0 +1,13 @@
{{ define "Title" }}Tisk štítku{{ end }}
{{ define "Content" }}
<h2>Tisk štítku pro {{ .Id }}</h2>
{{ if .UnknownId }}
<p>Neznámý obal.
{{ else if .Error }}
<p>Tisk selhal: {{ .Error }}
{{ else }}
<p>Tisk proběhl úspěšně.
{{ end }}
{{ end }}

View File

@ -1,6 +1,7 @@
package main package main
import ( import (
"errors"
"html/template" "html/template"
"io" "io"
"log" "log"
@ -9,6 +10,10 @@ import (
"os" "os"
"path/filepath" "path/filepath"
"time" "time"
"janouch.name/sklad/imgutil"
"janouch.name/sklad/label"
"janouch.name/sklad/ql"
) )
var templates = map[string]*template.Template{} var templates = map[string]*template.Template{}
@ -170,18 +175,61 @@ func handleSearch(w http.ResponseWriter, r *http.Request) {
executeTemplate("search.tmpl", w, &params) executeTemplate("search.tmpl", w, &params)
} }
func printLabel(id string) error {
printer, err := ql.Open()
if err != nil {
return err
}
if printer == nil {
return errors.New("no suitable printer found")
}
defer printer.Close()
/*
printer.StatusNotify = func(status *ql.Status) {
log.Printf("\x1b[1mreceived status\x1b[m\n%+v\n%s",
status[:], status)
}
*/
if err := printer.Initialize(); err != nil {
return err
}
if err := printer.UpdateStatus(); err != nil {
return err
}
mediaInfo := ql.GetMediaInfo(
printer.LastStatus.MediaWidthMM(),
printer.LastStatus.MediaLengthMM(),
)
if mediaInfo == nil {
return errors.New("unknown media")
}
return printer.Print(&imgutil.LeftRotate{Image: label.GenLabelForHeight(
labelFont, id, mediaInfo.PrintAreaPins, db.BDFScale)})
}
func handleLabel(w http.ResponseWriter, r *http.Request) { func handleLabel(w http.ResponseWriter, r *http.Request) {
if r.Method != http.MethodPost { if r.Method != http.MethodPost {
w.WriteHeader(http.StatusMethodNotAllowed) w.WriteHeader(http.StatusMethodNotAllowed)
return return
} }
id := r.FormValue("id") params := struct {
_ = id Id string
UnknownId bool
Error error
}{
Id: r.FormValue("id"),
}
// TODO: See if such a container exists, print a label on the printer. if c := indexContainer[ContainerId(params.Id)]; c == nil {
params.UnknownId = true
params := struct{}{} } else {
params.Error = printLabel(params.Id)
}
executeTemplate("label.tmpl", w, &params) executeTemplate("label.tmpl", w, &params)
} }

View File

@ -51,7 +51,7 @@ func sessionWrap(inner func(http.ResponseWriter, *http.Request)) func(
w.Header().Set("Cache-Control", "no-store") w.Header().Set("Cache-Control", "no-store")
redirect := "/login" redirect := "/login"
if r.RequestURI != "/" { if r.RequestURI != "/" && r.Method == http.MethodGet {
redirect += "?redirect=" + url.QueryEscape(r.RequestURI) redirect += "?redirect=" + url.QueryEscape(r.RequestURI)
} }