Compare commits
4 Commits
4dade55387
...
32ad36a287
Author | SHA1 | Date |
---|---|---|
Přemysl Eric Janouch | 32ad36a287 | |
Přemysl Eric Janouch | 412efcb1ae | |
Přemysl Eric Janouch | 39e2c5b76d | |
Přemysl Eric Janouch | ea45784554 |
|
@ -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&scale={{.Scale}}&text={{.Text}}'>
|
<img border=1 src='?img&scale={{.Scale}}&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 ×
|
{{ .Printer.LastStatus.MediaWidthMM }} mm ×
|
||||||
{{ .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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
21
sklad/db.go
21
sklad/db.go
|
@ -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 {
|
||||||
|
|
|
@ -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 }}
|
|
@ -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, ¶ms)
|
executeTemplate("search.tmpl", w, ¶ms)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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, ¶ms)
|
executeTemplate("label.tmpl", w, ¶ms)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue