354 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			354 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
| package main
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"github.com/jung-kurt/gofpdf"
 | |
| 	"golang.org/x/text/language"
 | |
| 	"golang.org/x/text/message"
 | |
| 	"gopkg.in/yaml.v2"
 | |
| 	"io/ioutil"
 | |
| 	"log"
 | |
| 	"math"
 | |
| 	"os"
 | |
| 	"os/exec"
 | |
| 	"path/filepath"
 | |
| 	"strings"
 | |
| )
 | |
| 
 | |
| // Metadata type
 | |
| type Metadata struct {
 | |
| 	InvoiceNr   string  `yaml:"invoice_nr"`
 | |
| 	InvoiceInfo string  `yaml:"invoice_info"`
 | |
| 	InvoiceDate string  `yaml:"invoice_date"`
 | |
| 	Vat         float64 `yaml:"vat"`
 | |
| 	Account     string  `yaml:"account"`
 | |
| 	VatNumber   string  `yaml:"vat_number"`
 | |
| }
 | |
| 
 | |
| // Address type
 | |
| type Address struct {
 | |
| 	Name   string `yaml:"name"`
 | |
| 	Street string `yaml:"street"`
 | |
| 	Zip    string `yaml:"zip"`
 | |
| 	City   string `yaml:"city"`
 | |
| 	TelNo  string `yaml:"tel_no"`
 | |
| 	Email  string `yaml:"email"`
 | |
| }
 | |
| 
 | |
| // InvoiceItem type
 | |
| type InvoiceItem struct {
 | |
| 	Text         string  `yaml:"text"`
 | |
| 	Quantity     float64 `yaml:"quantity"`
 | |
| 	PricePerUnit float64 `yaml:"price_per_unit"`
 | |
| }
 | |
| 
 | |
| // InvoiceData type
 | |
| type InvoiceData struct {
 | |
| 	Metadata       Metadata      `yaml:"metadata"`
 | |
| 	SenderAddress  Address       `yaml:"sender_address"`
 | |
| 	BillingAddress Address       `yaml:"billing_address"`
 | |
| 	InvoiceItems   []InvoiceItem `yaml:"invoice_items"`
 | |
| }
 | |
| 
 | |
| /* global variable declaration */
 | |
| var pdf *gofpdf.Fpdf
 | |
| var yPos float64
 | |
| var invoiceData InvoiceData
 | |
| var progDir string
 | |
| var currentPage int
 | |
| var totalPages int
 | |
| var totalItemLines int
 | |
| var totalNetAmount float64
 | |
| var totalInvoiceAmount float64
 | |
| 
 | |
| const defaultFontSize = 9
 | |
| const smallFontSize = 7
 | |
| const marginTop = 7
 | |
| const logoTop = 6
 | |
| const logoHeight = 20
 | |
| const lineSpacing = 5
 | |
| const lineSpacingSmall = 4.5
 | |
| const addressTop = 50
 | |
| const metadataTopFirstPage = 70
 | |
| const metadataTopNotFirstPage = 30
 | |
| const tabstopLeft = 20
 | |
| const tabstopLeftAlt = 40
 | |
| const tabstopMetadata = 60
 | |
| const tabstopAddress = 120
 | |
| const tabstopLogo = 155
 | |
| const tabstopRight = 200
 | |
| const widthItemText = 96
 | |
| const widthQuantity = 28
 | |
| const widthPricePerUnit = 28
 | |
| const widthPrice = 28
 | |
| const tabstopQuantity = tabstopRight - widthPrice - widthPricePerUnit - widthQuantity
 | |
| const tabstopPricePerUnit = tabstopRight - widthPrice - widthPricePerUnit
 | |
| const tabstopPrice = tabstopRight - widthPrice
 | |
| const itemsTopFirstPage = 105
 | |
| const itemsTopNotFirstPage = 50
 | |
| const totalsTop = 165
 | |
| const maxYPos = 265
 | |
| 
 | |
| func round5rappen(f float64) float64 {
 | |
| 	return (math.Round(f*20) / 20)
 | |
| }
 | |
| 
 | |
| func floatToString(f float64) string {
 | |
| 	p := message.NewPrinter(language.English)
 | |
| 	s := strings.ReplaceAll(p.Sprintf("%.2f", f), ",", "'")
 | |
| 	//fmt.Printf("--- s: @%s@\n", s)
 | |
| 	return s
 | |
| }
 | |
| 
 | |
| func readInvoiceData(filename string) {
 | |
| 	data, err := ioutil.ReadFile(filename)
 | |
| 	if err != nil {
 | |
| 		log.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	//fmt.Printf("File contents: %s", data)
 | |
| 
 | |
| 	err = yaml.Unmarshal([]byte(data), &invoiceData)
 | |
| 	if err != nil {
 | |
| 		log.Fatalf("error: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	//fmt.Printf("--- t:\n%v\n\n", invoiceData)
 | |
| }
 | |
| 
 | |
| func writeText(x float64, y float64, w float64, text string, alignStr ...string) {
 | |
| 	tr := pdf.UnicodeTranslatorFromDescriptor("")
 | |
| 	align := "LT"
 | |
| 	if len(alignStr) > 0 {
 | |
| 		align = alignStr[0]
 | |
| 	}
 | |
| 	pdf.SetXY(x, y)
 | |
| 	pdf.CellFormat(w, 11, tr(text), "", 0, align, false, 0, "")
 | |
| }
 | |
| 
 | |
| func setupInvoice() {
 | |
| 	pdf = gofpdf.New("P", "mm", "A4", "")
 | |
| 	pdf.SetMargins(0, 0, 0)
 | |
| 	pdf.SetFontLocation("fonts")
 | |
| 	pdf.AddFont("Dejavusans", "", "DejaVuSans.json")
 | |
| 	pdf.AddFont("Dejavusans-Bold", "", "DejaVuSans-Bold.json")
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	currentPage = 0
 | |
| }
 | |
| 
 | |
| func printPageHeader(firstPage bool) {
 | |
| 	var opt gofpdf.ImageOptions
 | |
| 
 | |
| 	currentPage = currentPage + 1
 | |
| 	pdf.AddPage()
 | |
| 	yPos = marginTop
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, invoiceData.SenderAddress.Name)
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	writeText(tabstopLeft, yPos, 0, invoiceData.SenderAddress.Street)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	writeText(tabstopLeft, yPos, 0, invoiceData.SenderAddress.Zip+" "+invoiceData.SenderAddress.City)
 | |
| 	if firstPage {
 | |
| 		yPos = yPos + lineSpacing
 | |
| 		yPos = yPos + lineSpacing
 | |
| 		writeText(tabstopLeft, yPos, 0, "Telefon")
 | |
| 		writeText(tabstopLeftAlt, yPos, 0, invoiceData.SenderAddress.TelNo)
 | |
| 		yPos = yPos + lineSpacing
 | |
| 		writeText(tabstopLeft, yPos, 0, "E-Mail")
 | |
| 		writeText(tabstopLeftAlt, yPos, 0, invoiceData.SenderAddress.Email)
 | |
| 		yPos = yPos + lineSpacing
 | |
| 		writeText(tabstopLeft, yPos, 0, "MwSt. Nr.")
 | |
| 		writeText(tabstopLeftAlt, yPos, 0, invoiceData.Metadata.VatNumber)
 | |
| 	}
 | |
| 
 | |
| 	opt.ImageType = "png"
 | |
| 	opt.ReadDpi = true
 | |
| 	pdf.ImageOptions("logos/nbit-logo-40x20mm-400dpi.png", tabstopLogo, logoTop, 0, logoHeight, false, opt, 0, "")
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| }
 | |
| 
 | |
| func printAddress() {
 | |
| 	yPos = addressTop
 | |
| 	writeText(tabstopAddress, yPos, 0, invoiceData.BillingAddress.Name)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	writeText(tabstopAddress, yPos, 0, invoiceData.BillingAddress.Street)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	writeText(tabstopAddress, yPos, 0, invoiceData.BillingAddress.Zip+" "+invoiceData.BillingAddress.City)
 | |
| }
 | |
| 
 | |
| func printMetadata(firstPage bool) {
 | |
| 	yPos = metadataTopNotFirstPage
 | |
| 	if firstPage {
 | |
| 		yPos = metadataTopFirstPage
 | |
| 		pdf.SetFont("Dejavusans-Bold", "", 16)
 | |
| 		writeText(tabstopLeft, yPos, 0, "Rechnung")
 | |
| 		yPos = yPos + lineSpacing
 | |
| 		yPos = yPos + lineSpacing
 | |
| 	}
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, "Rechnungsnummer:")
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	writeText(tabstopMetadata, yPos, 0, invoiceData.Metadata.InvoiceNr)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	if firstPage {
 | |
| 		pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 		writeText(tabstopLeft, yPos, 0, "Rechnungsdatum:")
 | |
| 		pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 		writeText(tabstopMetadata, yPos, 0, invoiceData.Metadata.InvoiceDate)
 | |
| 		yPos = yPos + lineSpacing
 | |
| 	} 
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, "Seite:")
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	writeText(tabstopMetadata, yPos, 0, fmt.Sprintf("%d von %d", currentPage, totalPages))
 | |
| }
 | |
| 
 | |
| func printItemsHeader() {
 | |
| 	if currentPage == 1 {
 | |
| 		yPos = itemsTopFirstPage
 | |
| 	} else {
 | |
| 		yPos = itemsTopNotFirstPage
 | |
| 	}
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, "Bezeichnung")
 | |
| 	writeText(tabstopQuantity, yPos, widthQuantity, "Menge", "TR")
 | |
| 	writeText(tabstopPricePerUnit, yPos, widthPricePerUnit, "Einheitspreis", "TR")
 | |
| 	writeText(tabstopPrice, yPos, widthPrice, "Preis", "TR")
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	pdf.Line(tabstopLeft, yPos, tabstopRight, yPos)
 | |
| 	yPos = yPos + lineSpacing
 | |
| }
 | |
| 
 | |
| func maybeNewPage() {
 | |
| 	if yPos > maxYPos {
 | |
| 		printPageHeader(false)
 | |
| 		printMetadata(false)
 | |
| 		printItemsHeader()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func printItems() {
 | |
| 	for _, i := range invoiceData.InvoiceItems {
 | |
| 		if i.Quantity != 0 {
 | |
| 			writeText(tabstopQuantity, yPos, widthQuantity, fmt.Sprintf("%.1f", i.Quantity), "TR")
 | |
| 			writeText(tabstopPricePerUnit, yPos, widthPricePerUnit, fmt.Sprintf("%.2f", i.PricePerUnit), "TR")
 | |
| 			itemNetAmount := round5rappen(i.Quantity * i.PricePerUnit)
 | |
| 			totalNetAmount = totalNetAmount + itemNetAmount
 | |
| 			writeText(tabstopPrice, yPos, widthPrice, floatToString(itemNetAmount), "TR")
 | |
| 		}
 | |
| 		if i.Text != "" {
 | |
| 			lines := pdf.SplitText(i.Text, widthItemText)
 | |
| 			for _, il := range lines {
 | |
| 				totalItemLines = totalItemLines + 1
 | |
| 				writeText(tabstopLeft, yPos, 0, strings.ReplaceAll(il, "_", " "))
 | |
| 				yPos = yPos + lineSpacing
 | |
| 				maybeNewPage()
 | |
| 			}
 | |
| 		} else {
 | |
| 			totalItemLines = totalItemLines + 1
 | |
| 			yPos = yPos + lineSpacing
 | |
| 			maybeNewPage()
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func printTotals() {
 | |
| 	yPos = totalsTop
 | |
| 	pdf.Line(tabstopRight-widthPrice, yPos + (lineSpacing / 2), tabstopRight, yPos + (lineSpacing / 2))
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	writeText(tabstopLeft, yPos, 0, "Summe")
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	writeText(tabstopPrice, yPos, widthPrice, floatToString(totalNetAmount), "TR")
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, fmt.Sprintf("MwSt. %.1f%%", invoiceData.Metadata.Vat))
 | |
| 	mwstAmount := round5rappen(totalNetAmount * invoiceData.Metadata.Vat / 100)
 | |
| 	pdf.SetFont("Dejavusans", "", defaultFontSize)
 | |
| 	writeText(tabstopPrice, yPos, widthPrice, floatToString(mwstAmount), "TR")
 | |
| yPos = yPos + lineSpacing
 | |
| 	yPos = yPos + lineSpacing
 | |
| 	pdf.SetFont("Dejavusans-Bold", "", defaultFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, "Rechnungbetrag in CHF")
 | |
| 	totalInvoiceAmount = totalNetAmount + mwstAmount
 | |
| 	writeText(tabstopPrice, yPos, widthPrice, floatToString(totalNetAmount+mwstAmount), "TR")
 | |
| 	yPos = yPos + lineSpacingSmall
 | |
| 	pdf.SetFont("Dejavusans", "", smallFontSize)
 | |
| 	writeText(tabstopLeft, yPos, 0, "30 Tage netto")
 | |
| }
 | |
| 
 | |
| func printQR() {
 | |
| 	var opt gofpdf.ImageOptions
 | |
| 
 | |
| 	cmd := exec.Command(filepath.Join(progDir, "qrbill.sh"),
 | |
| 		"--account", invoiceData.Metadata.Account,
 | |
| 		"--amount", floatToString(totalInvoiceAmount),
 | |
| 		"--creditor-name", invoiceData.SenderAddress.Name,
 | |
| 		"--creditor-street", invoiceData.SenderAddress.Street,
 | |
| 		"--creditor-postalcode", invoiceData.SenderAddress.Zip,
 | |
| 		"--creditor-city", invoiceData.SenderAddress.City,
 | |
| 		"--extra-infos", invoiceData.Metadata.InvoiceInfo,
 | |
| 		"--debtor-name", invoiceData.BillingAddress.Name,
 | |
| 		"--debtor-street", invoiceData.BillingAddress.Street,
 | |
| 		"--debtor-postalcode", invoiceData.BillingAddress.Zip,
 | |
| 		"--debtor-city", invoiceData.BillingAddress.City,
 | |
| 		"--language", "de")
 | |
| 	cmd.Env = append(os.Environ(),
 | |
| 		"INVNO="+invoiceData.Metadata.InvoiceNr,
 | |
| 	)
 | |
| 	stdoutStderr, err := cmd.CombinedOutput()
 | |
| 	fmt.Printf("%s\n", stdoutStderr)
 | |
| 	if err != nil {
 | |
| 		log.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	opt.ImageType = "png"
 | |
| 	opt.ReadDpi = true
 | |
| 	pdf.ImageOptions("qr-images/"+invoiceData.Metadata.InvoiceNr+".png", 0, 200, -1, -1, false, opt, 0, "")
 | |
| }
 | |
| 
 | |
| func CreateInvoice() {
 | |
| 	totalNetAmount = 0
 | |
| 	totalItemLines = 0
 | |
| 	readInvoiceData(os.Args[1])
 | |
| 	setupInvoice()
 | |
| 	printPageHeader(true)
 | |
| 	printAddress()
 | |
| 	printMetadata(true)
 | |
| 	printItemsHeader()
 | |
| 	printItems()
 | |
| 	printTotals()
 | |
| 	printQR()
 | |
| }
 | |
| 
 | |
| func main() {
 | |
| 	dir, err := filepath.Abs(filepath.Dir(os.Args[0]))
 | |
| 	if err != nil {
 | |
| 		log.Fatal(err)
 | |
| 	}
 | |
| 	progDir = dir
 | |
| 
 | |
| 	if len(os.Args) != 2 {
 | |
| 		fmt.Printf("usage: mkinvoice <yaml-file>\n")
 | |
| 		os.Exit(1)
 | |
| 	}
 | |
| 
 | |
| 	// First Run to get total number of pages
 | |
| 	CreateInvoice()
 | |
| 	totalPages = pdf.PageNo()
 | |
| 	fmt.Printf("Total Pages is: %d\n", totalPages)
 | |
| 	fmt.Printf("Total Item Lines is: %d\n", totalItemLines)
 | |
| 
 | |
| 	// Second Run
 | |
| 	CreateInvoice()
 | |
| 
 | |
| 	err = pdf.OutputFileAndClose(filepath.Join(progDir, "output", invoiceData.Metadata.InvoiceNr+".pdf"))
 | |
| 	if err == nil {
 | |
| 		fmt.Printf("Successfully created invoice\n")
 | |
| 	} else {
 | |
| 		fmt.Printf("Error: %v\n", err)
 | |
| 	}
 | |
| }
 |