/* * Copyright (c) 2013-2015 Kurt Jung (Gmail: kurt.w.jung) * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ package gofpdf_test import ( "bufio" "bytes" "fmt" "io" "io/ioutil" "math" "math/rand" "net/http" "os" "path/filepath" "strconv" "strings" "testing" "time" "github.com/jung-kurt/gofpdf" "github.com/jung-kurt/gofpdf/internal/example" "github.com/jung-kurt/gofpdf/internal/files" ) func init() { cleanup() } func cleanup() { filepath.Walk(example.PdfDir(), func(path string, info os.FileInfo, err error) (reterr error) { if info.Mode().IsRegular() { dir, _ := filepath.Split(path) if "reference" != filepath.Base(dir) { if len(path) > 3 { if path[len(path)-4:] == ".pdf" { os.Remove(path) } } } } return }) } func TestFpdfImplementPdf(t *testing.T) { // this will not compile if Fpdf and Tpl // do not implement Pdf var _ gofpdf.Pdf = (*gofpdf.Fpdf)(nil) var _ gofpdf.Pdf = (*gofpdf.Tpl)(nil) } // TestPagedTemplate ensures new paged templates work func TestPagedTemplate(t *testing.T) { pdf := gofpdf.New("P", "mm", "A4", "") tpl := pdf.CreateTemplate(func(t *gofpdf.Tpl) { // this will be the second page, as a page is already // created by default t.AddPage() t.AddPage() t.AddPage() }) if tpl.NumPages() != 4 { t.Fatalf("The template does not have the correct number of pages %d", tpl.NumPages()) } tplPages := tpl.FromPages() for x := 0; x < len(tplPages); x++ { pdf.AddPage() pdf.UseTemplate(tplPages[x]) } // get the last template tpl2, err := tpl.FromPage(tpl.NumPages()) if err != nil { t.Fatal(err) } // the objects should be the exact same, as the // template will represent the last page by default // therefore no new id should be set, and the object // should be the same object if fmt.Sprintf("%p", tpl2) != fmt.Sprintf("%p", tpl) { t.Fatal("Template no longer respecting initial template object") } } // TestIssue0116 addresses issue 116 in which library silently fails after // calling CellFormat when no font has been set. func TestIssue0116(t *testing.T) { var pdf *gofpdf.Fpdf pdf = gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "B", 16) pdf.Cell(40, 10, "OK") if pdf.Error() != nil { t.Fatalf("not expecting error when rendering text") } pdf = gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.Cell(40, 10, "Not OK") // Font not set if pdf.Error() == nil { t.Fatalf("expecting error when rendering text without having set font") } } // TestIssue0193 addresses issue 193 in which the error io.EOF is incorrectly // assigned to the FPDF instance error. func TestIssue0193(t *testing.T) { var png []byte var pdf *gofpdf.Fpdf var err error var rdr *bytes.Reader png, err = ioutil.ReadFile(example.ImageFile("sweden.png")) if err == nil { rdr = bytes.NewReader(png) pdf = gofpdf.New("P", "mm", "A4", "") pdf.AddPage() _ = pdf.RegisterImageOptionsReader("sweden", gofpdf.ImageOptions{ImageType: "png", ReadDpi: true}, rdr) err = pdf.Error() } if err != nil { t.Fatalf("issue 193 error: %s", err) } } // TestIssue0209SplitLinesEqualMultiCell addresses issue 209 // make SplitLines and MultiCell split at the same place func TestIssue0209SplitLinesEqualMultiCell(t *testing.T) { var pdf *gofpdf.Fpdf pdf = gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 8) // this sentence should not be splited str := "Guochin Amandine" lines := pdf.SplitLines([]byte(str), 26) _, FontSize := pdf.GetFontSize() y_start := pdf.GetY() pdf.MultiCell(26, FontSize, str, "", "L", false) y_end := pdf.GetY() if len(lines) != 1 { t.Fatalf("expect SplitLines split in one line") } if int(y_end-y_start) != int(FontSize) { t.Fatalf("expect MultiCell split in one line %.2f != %.2f", y_end-y_start, FontSize) } // this sentence should be splited in two lines str = "Guiochini Amandine" lines = pdf.SplitLines([]byte(str), 26) y_start = pdf.GetY() pdf.MultiCell(26, FontSize, str, "", "L", false) y_end = pdf.GetY() if len(lines) != 2 { t.Fatalf("expect SplitLines split in two lines") } if int(y_end-y_start) != int(FontSize*2) { t.Fatalf("expect MultiCell split in two lines %.2f != %.2f", y_end-y_start, FontSize*2) } } // TestFooterFuncLpi tests to make sure the footer is not call twice and SetFooterFuncLpi can work // without SetFooterFunc. func TestFooterFuncLpi(t *testing.T) { pdf := gofpdf.New("P", "mm", "A4", "") var ( oldFooterFnc = "oldFooterFnc" bothPages = "bothPages" firstPageOnly = "firstPageOnly" lastPageOnly = "lastPageOnly" ) // This set just for testing, only set SetFooterFuncLpi. pdf.SetFooterFunc(func() { pdf.SetY(-15) pdf.SetFont("Arial", "I", 8) pdf.CellFormat(0, 10, oldFooterFnc, "", 0, "C", false, 0, "") }) pdf.SetFooterFuncLpi(func(lastPage bool) { pdf.SetY(-15) pdf.SetFont("Arial", "I", 8) pdf.CellFormat(0, 10, bothPages, "", 0, "L", false, 0, "") if !lastPage { pdf.CellFormat(0, 10, firstPageOnly, "", 0, "C", false, 0, "") } else { pdf.CellFormat(0, 10, lastPageOnly, "", 0, "C", false, 0, "") } }) pdf.AddPage() pdf.SetFont("Arial", "B", 16) for j := 1; j <= 40; j++ { pdf.CellFormat(0, 10, fmt.Sprintf("Printing line number %d", j), "", 1, "", false, 0, "") } if pdf.Error() != nil { t.Fatalf("not expecting error when rendering text") } w := &bytes.Buffer{} if err := pdf.Output(w); err != nil { t.Errorf("unexpected err: %s", err) } b := w.Bytes() if bytes.Contains(b, []byte(oldFooterFnc)) { t.Errorf("not expecting %s render on pdf when FooterFncLpi is set", oldFooterFnc) } got := bytes.Count(b, []byte("bothPages")) if got != 2 { t.Errorf("footer %s should render on two page got:%d", bothPages, got) } got = bytes.Count(b, []byte(firstPageOnly)) if got != 1 { t.Errorf("footer %s should render only on first page got: %d", firstPageOnly, got) } got = bytes.Count(b, []byte(lastPageOnly)) if got != 1 { t.Errorf("footer %s should render only on first page got: %d", lastPageOnly, got) } f := bytes.Index(b, []byte(firstPageOnly)) l := bytes.Index(b, []byte(lastPageOnly)) if f > l { t.Errorf("index %d (%s) should less than index %d (%s)", f, firstPageOnly, l, lastPageOnly) } } type fontResourceType struct { } func (f fontResourceType) Open(name string) (rdr io.Reader, err error) { var buf []byte buf, err = ioutil.ReadFile(example.FontFile(name)) if err == nil { rdr = bytes.NewReader(buf) fmt.Printf("Generalized font loader reading %s\n", name) } return } // strDelimit converts 'ABCDEFG' to, for example, 'A,BCD,EFG' func strDelimit(str string, sepstr string, sepcount int) string { pos := len(str) - sepcount for pos > 0 { str = str[:pos] + sepstr + str[pos:] pos = pos - sepcount } return str } func loremList() []string { return []string{ "Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod " + "tempor incididunt ut labore et dolore magna aliqua.", "Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut " + "aliquip ex ea commodo consequat.", "Duis aute irure dolor in reprehenderit in voluptate velit esse cillum " + "dolore eu fugiat nulla pariatur.", "Excepteur sint occaecat cupidatat non proident, sunt in culpa qui " + "officia deserunt mollit anim id est laborum.", } } func lorem() string { return strings.Join(loremList(), " ") } // Example demonstrates the generation of a simple PDF document. Note that // since only core fonts are used (in this case Arial, a synonym for // Helvetica), an empty string can be specified for the font directory in the // call to New(). Note also that the example.Filename() and example.Summary() // functions belong to a separate, internal package and are not part of the // gofpdf library. If an error occurs at some point during the construction of // the document, subsequent method calls exit immediately and the error is // finally retrieved with the output call where it can be handled by the // application. func Example() { pdf := gofpdf.New(gofpdf.OrientationPortrait, "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "B", 16) pdf.Cell(40, 10, "Hello World!") fileStr := example.Filename("basic") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/basic.pdf } // ExampleFpdf_AddPage demonsrates the generation of headers, footers and page breaks. func ExampleFpdf_AddPage() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetTopMargin(30) pdf.SetHeaderFuncMode(func() { pdf.Image(example.ImageFile("logo.png"), 10, 6, 30, 0, false, "", 0, "") pdf.SetY(5) pdf.SetFont("Arial", "B", 15) pdf.Cell(80, 0, "") pdf.CellFormat(30, 10, "Title", "1", 0, "C", false, 0, "") pdf.Ln(20) }, true) pdf.SetFooterFunc(func() { pdf.SetY(-15) pdf.SetFont("Arial", "I", 8) pdf.CellFormat(0, 10, fmt.Sprintf("Page %d/{nb}", pdf.PageNo()), "", 0, "C", false, 0, "") }) pdf.AliasNbPages("") pdf.AddPage() pdf.SetFont("Times", "", 12) for j := 1; j <= 40; j++ { pdf.CellFormat(0, 10, fmt.Sprintf("Printing line number %d", j), "", 1, "", false, 0, "") } fileStr := example.Filename("Fpdf_AddPage") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_AddPage.pdf } // ExampleFpdf_MultiCell demonstrates word-wrapping, line justification and // page-breaking. func ExampleFpdf_MultiCell() { pdf := gofpdf.New("P", "mm", "A4", "") titleStr := "20000 Leagues Under the Seas" pdf.SetTitle(titleStr, false) pdf.SetAuthor("Jules Verne", false) pdf.SetHeaderFunc(func() { // Arial bold 15 pdf.SetFont("Arial", "B", 15) // Calculate width of title and position wd := pdf.GetStringWidth(titleStr) + 6 pdf.SetX((210 - wd) / 2) // Colors of frame, background and text pdf.SetDrawColor(0, 80, 180) pdf.SetFillColor(230, 230, 0) pdf.SetTextColor(220, 50, 50) // Thickness of frame (1 mm) pdf.SetLineWidth(1) // Title pdf.CellFormat(wd, 9, titleStr, "1", 1, "C", true, 0, "") // Line break pdf.Ln(10) }) pdf.SetFooterFunc(func() { // Position at 1.5 cm from bottom pdf.SetY(-15) // Arial italic 8 pdf.SetFont("Arial", "I", 8) // Text color in gray pdf.SetTextColor(128, 128, 128) // Page number pdf.CellFormat(0, 10, fmt.Sprintf("Page %d", pdf.PageNo()), "", 0, "C", false, 0, "") }) chapterTitle := func(chapNum int, titleStr string) { // // Arial 12 pdf.SetFont("Arial", "", 12) // Background color pdf.SetFillColor(200, 220, 255) // Title pdf.CellFormat(0, 6, fmt.Sprintf("Chapter %d : %s", chapNum, titleStr), "", 1, "L", true, 0, "") // Line break pdf.Ln(4) } chapterBody := func(fileStr string) { // Read text file txtStr, err := ioutil.ReadFile(fileStr) if err != nil { pdf.SetError(err) } // Times 12 pdf.SetFont("Times", "", 12) // Output justified text pdf.MultiCell(0, 5, string(txtStr), "", "", false) // Line break pdf.Ln(-1) // Mention in italics pdf.SetFont("", "I", 0) pdf.Cell(0, 5, "(end of excerpt)") } printChapter := func(chapNum int, titleStr, fileStr string) { pdf.AddPage() chapterTitle(chapNum, titleStr) chapterBody(fileStr) } printChapter(1, "A RUNAWAY REEF", example.TextFile("20k_c1.txt")) printChapter(2, "THE PROS AND CONS", example.TextFile("20k_c2.txt")) fileStr := example.Filename("Fpdf_MultiCell") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_MultiCell.pdf } // ExampleFpdf_SetLeftMargin demonstrates the generation of a PDF document that has multiple // columns. This is accomplished with the SetLeftMargin() and Cell() methods. func ExampleFpdf_SetLeftMargin() { var y0 float64 var crrntCol int pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetDisplayMode("fullpage", "TwoColumnLeft") titleStr := "20000 Leagues Under the Seas" pdf.SetTitle(titleStr, false) pdf.SetAuthor("Jules Verne", false) setCol := func(col int) { // Set position at a given column crrntCol = col x := 10.0 + float64(col)*65.0 pdf.SetLeftMargin(x) pdf.SetX(x) } chapterTitle := func(chapNum int, titleStr string) { // Arial 12 pdf.SetFont("Arial", "", 12) // Background color pdf.SetFillColor(200, 220, 255) // Title pdf.CellFormat(0, 6, fmt.Sprintf("Chapter %d : %s", chapNum, titleStr), "", 1, "L", true, 0, "") // Line break pdf.Ln(4) y0 = pdf.GetY() } chapterBody := func(fileStr string) { // Read text file txtStr, err := ioutil.ReadFile(fileStr) if err != nil { pdf.SetError(err) } // Font pdf.SetFont("Times", "", 12) // Output text in a 6 cm width column pdf.MultiCell(60, 5, string(txtStr), "", "", false) pdf.Ln(-1) // Mention pdf.SetFont("", "I", 0) pdf.Cell(0, 5, "(end of excerpt)") // Go back to first column setCol(0) } printChapter := func(num int, titleStr, fileStr string) { // Add chapter pdf.AddPage() chapterTitle(num, titleStr) chapterBody(fileStr) } pdf.SetAcceptPageBreakFunc(func() bool { // Method accepting or not automatic page break if crrntCol < 2 { // Go to next column setCol(crrntCol + 1) // Set ordinate to top pdf.SetY(y0) // Keep on page return false } // Go back to first column setCol(0) // Page break return true }) pdf.SetHeaderFunc(func() { // Arial bold 15 pdf.SetFont("Arial", "B", 15) // Calculate width of title and position wd := pdf.GetStringWidth(titleStr) + 6 pdf.SetX((210 - wd) / 2) // Colors of frame, background and text pdf.SetDrawColor(0, 80, 180) pdf.SetFillColor(230, 230, 0) pdf.SetTextColor(220, 50, 50) // Thickness of frame (1 mm) pdf.SetLineWidth(1) // Title pdf.CellFormat(wd, 9, titleStr, "1", 1, "C", true, 0, "") // Line break pdf.Ln(10) // Save ordinate y0 = pdf.GetY() }) pdf.SetFooterFunc(func() { // Position at 1.5 cm from bottom pdf.SetY(-15) // Arial italic 8 pdf.SetFont("Arial", "I", 8) // Text color in gray pdf.SetTextColor(128, 128, 128) // Page number pdf.CellFormat(0, 10, fmt.Sprintf("Page %d", pdf.PageNo()), "", 0, "C", false, 0, "") }) printChapter(1, "A RUNAWAY REEF", example.TextFile("20k_c1.txt")) printChapter(2, "THE PROS AND CONS", example.TextFile("20k_c2.txt")) fileStr := example.Filename("Fpdf_SetLeftMargin_multicolumn") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetLeftMargin_multicolumn.pdf } // ExampleFpdf_SplitLines_tables demonstrates word-wrapped table cells func ExampleFpdf_SplitLines_tables() { const ( colCount = 3 colWd = 60.0 marginH = 15.0 lineHt = 5.5 cellGap = 2.0 ) // var colStrList [colCount]string type cellType struct { str string list [][]byte ht float64 } var ( cellList [colCount]cellType cell cellType ) pdf := gofpdf.New("P", "mm", "A4", "") // 210 x 297 header := [colCount]string{"Column A", "Column B", "Column C"} alignList := [colCount]string{"L", "C", "R"} strList := loremList() pdf.SetMargins(marginH, 15, marginH) pdf.SetFont("Arial", "", 14) pdf.AddPage() // Headers pdf.SetTextColor(224, 224, 224) pdf.SetFillColor(64, 64, 64) for colJ := 0; colJ < colCount; colJ++ { pdf.CellFormat(colWd, 10, header[colJ], "1", 0, "CM", true, 0, "") } pdf.Ln(-1) pdf.SetTextColor(24, 24, 24) pdf.SetFillColor(255, 255, 255) // Rows y := pdf.GetY() count := 0 for rowJ := 0; rowJ < 2; rowJ++ { maxHt := lineHt // Cell height calculation loop for colJ := 0; colJ < colCount; colJ++ { count++ if count > len(strList) { count = 1 } cell.str = strings.Join(strList[0:count], " ") cell.list = pdf.SplitLines([]byte(cell.str), colWd-cellGap-cellGap) cell.ht = float64(len(cell.list)) * lineHt if cell.ht > maxHt { maxHt = cell.ht } cellList[colJ] = cell } // Cell render loop x := marginH for colJ := 0; colJ < colCount; colJ++ { pdf.Rect(x, y, colWd, maxHt+cellGap+cellGap, "D") cell = cellList[colJ] cellY := y + cellGap + (maxHt-cell.ht)/2 for splitJ := 0; splitJ < len(cell.list); splitJ++ { pdf.SetXY(x+cellGap, cellY) pdf.CellFormat(colWd-cellGap-cellGap, lineHt, string(cell.list[splitJ]), "", 0, alignList[colJ], false, 0, "") cellY += lineHt } x += colWd } y += maxHt + cellGap + cellGap } fileStr := example.Filename("Fpdf_SplitLines_tables") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SplitLines_tables.pdf } // ExampleFpdf_CellFormat_tables demonstrates various table styles. func ExampleFpdf_CellFormat_tables() { pdf := gofpdf.New("P", "mm", "A4", "") type countryType struct { nameStr, capitalStr, areaStr, popStr string } countryList := make([]countryType, 0, 8) header := []string{"Country", "Capital", "Area (sq km)", "Pop. (thousands)"} loadData := func(fileStr string) { fl, err := os.Open(fileStr) if err == nil { scanner := bufio.NewScanner(fl) var c countryType for scanner.Scan() { // Austria;Vienna;83859;8075 lineStr := scanner.Text() list := strings.Split(lineStr, ";") if len(list) == 4 { c.nameStr = list[0] c.capitalStr = list[1] c.areaStr = list[2] c.popStr = list[3] countryList = append(countryList, c) } else { err = fmt.Errorf("error tokenizing %s", lineStr) } } fl.Close() if len(countryList) == 0 { err = fmt.Errorf("error loading data from %s", fileStr) } } if err != nil { pdf.SetError(err) } } // Simple table basicTable := func() { left := (210.0 - 4*40) / 2 pdf.SetX(left) for _, str := range header { pdf.CellFormat(40, 7, str, "1", 0, "", false, 0, "") } pdf.Ln(-1) for _, c := range countryList { pdf.SetX(left) pdf.CellFormat(40, 6, c.nameStr, "1", 0, "", false, 0, "") pdf.CellFormat(40, 6, c.capitalStr, "1", 0, "", false, 0, "") pdf.CellFormat(40, 6, c.areaStr, "1", 0, "", false, 0, "") pdf.CellFormat(40, 6, c.popStr, "1", 0, "", false, 0, "") pdf.Ln(-1) } } // Better table improvedTable := func() { // Column widths w := []float64{40.0, 35.0, 40.0, 45.0} wSum := 0.0 for _, v := range w { wSum += v } left := (210 - wSum) / 2 // Header pdf.SetX(left) for j, str := range header { pdf.CellFormat(w[j], 7, str, "1", 0, "C", false, 0, "") } pdf.Ln(-1) // Data for _, c := range countryList { pdf.SetX(left) pdf.CellFormat(w[0], 6, c.nameStr, "LR", 0, "", false, 0, "") pdf.CellFormat(w[1], 6, c.capitalStr, "LR", 0, "", false, 0, "") pdf.CellFormat(w[2], 6, strDelimit(c.areaStr, ",", 3), "LR", 0, "R", false, 0, "") pdf.CellFormat(w[3], 6, strDelimit(c.popStr, ",", 3), "LR", 0, "R", false, 0, "") pdf.Ln(-1) } pdf.SetX(left) pdf.CellFormat(wSum, 0, "", "T", 0, "", false, 0, "") } // Colored table fancyTable := func() { // Colors, line width and bold font pdf.SetFillColor(255, 0, 0) pdf.SetTextColor(255, 255, 255) pdf.SetDrawColor(128, 0, 0) pdf.SetLineWidth(.3) pdf.SetFont("", "B", 0) // Header w := []float64{40, 35, 40, 45} wSum := 0.0 for _, v := range w { wSum += v } left := (210 - wSum) / 2 pdf.SetX(left) for j, str := range header { pdf.CellFormat(w[j], 7, str, "1", 0, "C", true, 0, "") } pdf.Ln(-1) // Color and font restoration pdf.SetFillColor(224, 235, 255) pdf.SetTextColor(0, 0, 0) pdf.SetFont("", "", 0) // Data fill := false for _, c := range countryList { pdf.SetX(left) pdf.CellFormat(w[0], 6, c.nameStr, "LR", 0, "", fill, 0, "") pdf.CellFormat(w[1], 6, c.capitalStr, "LR", 0, "", fill, 0, "") pdf.CellFormat(w[2], 6, strDelimit(c.areaStr, ",", 3), "LR", 0, "R", fill, 0, "") pdf.CellFormat(w[3], 6, strDelimit(c.popStr, ",", 3), "LR", 0, "R", fill, 0, "") pdf.Ln(-1) fill = !fill } pdf.SetX(left) pdf.CellFormat(wSum, 0, "", "T", 0, "", false, 0, "") } loadData(example.TextFile("countries.txt")) pdf.SetFont("Arial", "", 14) pdf.AddPage() basicTable() pdf.AddPage() improvedTable() pdf.AddPage() fancyTable() fileStr := example.Filename("Fpdf_CellFormat_tables") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_CellFormat_tables.pdf } // ExampleFpdf_HTMLBasicNew demonstrates internal and external links with and without basic // HTML. func ExampleFpdf_HTMLBasicNew() { pdf := gofpdf.New("P", "mm", "A4", "") // First page: manual local link pdf.AddPage() pdf.SetFont("Helvetica", "", 20) _, lineHt := pdf.GetFontSize() pdf.Write(lineHt, "To find out what's new in this tutorial, click ") pdf.SetFont("", "U", 0) link := pdf.AddLink() pdf.WriteLinkID(lineHt, "here", link) pdf.SetFont("", "", 0) // Second page: image link and basic HTML with link pdf.AddPage() pdf.SetLink(link, 0, -1) pdf.Image(example.ImageFile("logo.png"), 10, 12, 30, 0, false, "", 0, "http://www.fpdf.org") pdf.SetLeftMargin(45) pdf.SetFontSize(14) _, lineHt = pdf.GetFontSize() htmlStr := `You can now easily print text mixing different styles: bold, ` + `italic, underlined, or all at once!

` + `
You can also center text.
` + `Or align it to the right.` + `You can also insert links on text, such as ` + `www.fpdf.org, or on an image: click on the logo.` html := pdf.HTMLBasicNew() html.Write(lineHt, htmlStr) fileStr := example.Filename("Fpdf_HTMLBasicNew") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_HTMLBasicNew.pdf } // ExampleFpdf_AddFont demonstrates the use of a non-standard font. func ExampleFpdf_AddFont() { pdf := gofpdf.New("P", "mm", "A4", example.FontDir()) pdf.AddFont("Calligrapher", "", "calligra.json") pdf.AddPage() pdf.SetFont("Calligrapher", "", 35) pdf.Cell(0, 10, "Enjoy new fonts with FPDF!") fileStr := example.Filename("Fpdf_AddFont") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_AddFont.pdf } // ExampleFpdf_WriteAligned demonstrates how to align text with the Write function. func ExampleFpdf_WriteAligned() { pdf := gofpdf.New("P", "mm", "A4", example.FontDir()) pdf.SetLeftMargin(50.0) pdf.SetRightMargin(50.0) pdf.AddPage() pdf.SetFont("Helvetica", "", 12) pdf.WriteAligned(0, 35, "This text is the default alignment, Left", "") pdf.Ln(35) pdf.WriteAligned(0, 35, "This text is aligned Left", "L") pdf.Ln(35) pdf.WriteAligned(0, 35, "This text is aligned Center", "C") pdf.Ln(35) pdf.WriteAligned(0, 35, "This text is aligned Right", "R") pdf.Ln(35) line := "This can by used to write justified text" leftMargin, _, rightMargin, _ := pdf.GetMargins() pageWidth, _ := pdf.GetPageSize() pageWidth -= leftMargin + rightMargin pdf.SetWordSpacing((pageWidth - pdf.GetStringWidth(line)) / float64(strings.Count(line, " "))) pdf.WriteAligned(pageWidth, 35, line, "L") fileStr := example.Filename("Fpdf_WriteAligned") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_WriteAligned.pdf } // ExampleFpdf_Image demonstrates how images are included in documents. func ExampleFpdf_Image() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 11) pdf.Image(example.ImageFile("logo.png"), 10, 10, 30, 0, false, "", 0, "") pdf.Text(50, 20, "logo.png") pdf.Image(example.ImageFile("logo.gif"), 10, 40, 30, 0, false, "", 0, "") pdf.Text(50, 50, "logo.gif") pdf.Image(example.ImageFile("logo-gray.png"), 10, 70, 30, 0, false, "", 0, "") pdf.Text(50, 80, "logo-gray.png") pdf.Image(example.ImageFile("logo-rgb.png"), 10, 100, 30, 0, false, "", 0, "") pdf.Text(50, 110, "logo-rgb.png") pdf.Image(example.ImageFile("logo.jpg"), 10, 130, 30, 0, false, "", 0, "") pdf.Text(50, 140, "logo.jpg") fileStr := example.Filename("Fpdf_Image") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Image.pdf } // ExampleFpdf_ImageOptions demonstrates how the AllowNegativePosition field of the // ImageOption struct can be used to affect horizontal image placement. func ExampleFpdf_ImageOptions() { var opt gofpdf.ImageOptions pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 11) pdf.SetX(60) opt.ImageType = "png" pdf.ImageOptions(example.ImageFile("logo.png"), -10, 10, 30, 0, false, opt, 0, "") opt.AllowNegativePosition = true pdf.ImageOptions(example.ImageFile("logo.png"), -10, 50, 30, 0, false, opt, 0, "") fileStr := example.Filename("Fpdf_ImageOptions") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_ImageOptions.pdf } // ExampleFpdf_RegisterImageOptionsReader demonstrates how to load an image // from a io.Reader (in this case, a file) and register it with options. func ExampleFpdf_RegisterImageOptionsReader() { var ( opt gofpdf.ImageOptions pdfStr string fl *os.File err error ) pdfStr = example.Filename("Fpdf_RegisterImageOptionsReader") pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 11) fl, err = os.Open(example.ImageFile("logo.png")) if err == nil { opt.ImageType = "png" opt.AllowNegativePosition = true _ = pdf.RegisterImageOptionsReader("logo", opt, fl) fl.Close() for x := -20.0; x <= 40.0; x += 5 { pdf.ImageOptions("logo", x, x+30, 0, 0, false, opt, 0, "") } err = pdf.OutputFileAndClose(pdfStr) } example.Summary(err, pdfStr) // Output: // Successfully generated pdf/Fpdf_RegisterImageOptionsReader.pdf } // This example demonstrates Landscape mode with images. func ExampleFpdf_SetAcceptPageBreakFunc() { var y0 float64 var crrntCol int loremStr := lorem() pdf := gofpdf.New("L", "mm", "A4", "") const ( pageWd = 297.0 // A4 210.0 x 297.0 margin = 10.0 gutter = 4 colNum = 3 colWd = (pageWd - 2*margin - (colNum-1)*gutter) / colNum ) setCol := func(col int) { crrntCol = col x := margin + float64(col)*(colWd+gutter) pdf.SetLeftMargin(x) pdf.SetX(x) } pdf.SetHeaderFunc(func() { titleStr := "gofpdf" pdf.SetFont("Helvetica", "B", 48) wd := pdf.GetStringWidth(titleStr) + 6 pdf.SetX((pageWd - wd) / 2) pdf.SetTextColor(128, 128, 160) pdf.Write(12, titleStr[:2]) pdf.SetTextColor(128, 128, 128) pdf.Write(12, titleStr[2:]) pdf.Ln(20) y0 = pdf.GetY() }) pdf.SetAcceptPageBreakFunc(func() bool { if crrntCol < colNum-1 { setCol(crrntCol + 1) pdf.SetY(y0) // Start new column, not new page return false } setCol(0) return true }) pdf.AddPage() pdf.SetFont("Times", "", 12) for j := 0; j < 20; j++ { if j == 1 { pdf.Image(example.ImageFile("fpdf.png"), -1, 0, colWd, 0, true, "", 0, "") } else if j == 5 { pdf.Image(example.ImageFile("golang-gopher.png"), -1, 0, colWd, 0, true, "", 0, "") } pdf.MultiCell(colWd, 5, loremStr, "", "", false) pdf.Ln(-1) } fileStr := example.Filename("Fpdf_SetAcceptPageBreakFunc_landscape") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetAcceptPageBreakFunc_landscape.pdf } // This example tests corner cases as reported by the gocov tool. func ExampleFpdf_SetKeywords() { var err error fileStr := example.Filename("Fpdf_SetKeywords") err = gofpdf.MakeFont(example.FontFile("CalligrapherRegular.pfb"), example.FontFile("cp1252.map"), example.FontDir(), nil, true) if err == nil { pdf := gofpdf.New("", "", "", "") pdf.SetFontLocation(example.FontDir()) pdf.SetTitle("世界", true) pdf.SetAuthor("世界", true) pdf.SetSubject("世界", true) pdf.SetCreator("世界", true) pdf.SetKeywords("世界", true) pdf.AddFont("Calligrapher", "", "CalligrapherRegular.json") pdf.AddPage() pdf.SetFont("Calligrapher", "", 16) pdf.Writef(5, "\x95 %s \x95", pdf) err = pdf.OutputFileAndClose(fileStr) } example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetKeywords.pdf } // ExampleFpdf_Circle demonstrates the construction of various geometric figures, func ExampleFpdf_Circle() { const ( thin = 0.2 thick = 3.0 ) pdf := gofpdf.New("", "", "", "") pdf.SetFont("Helvetica", "", 12) pdf.SetFillColor(200, 200, 220) pdf.AddPage() y := 15.0 pdf.Text(10, y, "Circles") pdf.SetFillColor(200, 200, 220) pdf.SetLineWidth(thin) pdf.Circle(20, y+15, 10, "D") pdf.Circle(45, y+15, 10, "F") pdf.Circle(70, y+15, 10, "FD") pdf.SetLineWidth(thick) pdf.Circle(95, y+15, 10, "FD") pdf.SetLineWidth(thin) y += 40.0 pdf.Text(10, y, "Ellipses") pdf.SetFillColor(220, 200, 200) pdf.Ellipse(30, y+15, 20, 10, 0, "D") pdf.Ellipse(75, y+15, 20, 10, 0, "F") pdf.Ellipse(120, y+15, 20, 10, 0, "FD") pdf.SetLineWidth(thick) pdf.Ellipse(165, y+15, 20, 10, 0, "FD") pdf.SetLineWidth(thin) y += 40.0 pdf.Text(10, y, "Curves (quadratic)") pdf.SetFillColor(220, 220, 200) pdf.Curve(10, y+30, 15, y-20, 40, y+30, "D") pdf.Curve(45, y+30, 50, y-20, 75, y+30, "F") pdf.Curve(80, y+30, 85, y-20, 110, y+30, "FD") pdf.SetLineWidth(thick) pdf.Curve(115, y+30, 120, y-20, 145, y+30, "FD") pdf.SetLineCapStyle("round") pdf.Curve(150, y+30, 155, y-20, 180, y+30, "FD") pdf.SetLineWidth(thin) pdf.SetLineCapStyle("butt") y += 40.0 pdf.Text(10, y, "Curves (cubic)") pdf.SetFillColor(220, 200, 220) pdf.CurveBezierCubic(10, y+30, 15, y-20, 10, y+30, 40, y+30, "D") pdf.CurveBezierCubic(45, y+30, 50, y-20, 45, y+30, 75, y+30, "F") pdf.CurveBezierCubic(80, y+30, 85, y-20, 80, y+30, 110, y+30, "FD") pdf.SetLineWidth(thick) pdf.CurveBezierCubic(115, y+30, 120, y-20, 115, y+30, 145, y+30, "FD") pdf.SetLineCapStyle("round") pdf.CurveBezierCubic(150, y+30, 155, y-20, 150, y+30, 180, y+30, "FD") pdf.SetLineWidth(thin) pdf.SetLineCapStyle("butt") y += 40.0 pdf.Text(10, y, "Arcs") pdf.SetFillColor(200, 220, 220) pdf.SetLineWidth(thick) pdf.Arc(45, y+35, 20, 10, 0, 0, 180, "FD") pdf.SetLineWidth(thin) pdf.Arc(45, y+35, 25, 15, 0, 90, 270, "D") pdf.SetLineWidth(thick) pdf.Arc(45, y+35, 30, 20, 0, 0, 360, "D") pdf.SetLineCapStyle("round") pdf.Arc(135, y+35, 20, 10, 135, 0, 180, "FD") pdf.SetLineWidth(thin) pdf.Arc(135, y+35, 25, 15, 135, 90, 270, "D") pdf.SetLineWidth(thick) pdf.Arc(135, y+35, 30, 20, 135, 0, 360, "D") pdf.SetLineWidth(thin) pdf.SetLineCapStyle("butt") fileStr := example.Filename("Fpdf_Circle_figures") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Circle_figures.pdf } // ExampleFpdf_SetAlpha demonstrates alpha transparency. func ExampleFpdf_SetAlpha() { const ( gapX = 10.0 gapY = 9.0 rectW = 40.0 rectH = 58.0 pageW = 210 pageH = 297 ) modeList := []string{"Normal", "Multiply", "Screen", "Overlay", "Darken", "Lighten", "ColorDodge", "ColorBurn", "HardLight", "SoftLight", "Difference", "Exclusion", "Hue", "Saturation", "Color", "Luminosity"} pdf := gofpdf.New("", "", "", "") pdf.SetLineWidth(2) pdf.SetAutoPageBreak(false, 0) pdf.AddPage() pdf.SetFont("Helvetica", "", 18) pdf.SetXY(0, gapY) pdf.SetTextColor(0, 0, 0) pdf.CellFormat(pageW, gapY, "Alpha Blending Modes", "", 0, "C", false, 0, "") j := 0 y := 3 * gapY for col := 0; col < 4; col++ { x := gapX for row := 0; row < 4; row++ { pdf.Rect(x, y, rectW, rectH, "D") pdf.SetFont("Helvetica", "B", 12) pdf.SetFillColor(0, 0, 0) pdf.SetTextColor(250, 250, 230) pdf.SetXY(x, y+rectH-4) pdf.CellFormat(rectW, 5, modeList[j], "", 0, "C", true, 0, "") pdf.SetFont("Helvetica", "I", 150) pdf.SetTextColor(80, 80, 120) pdf.SetXY(x, y+2) pdf.CellFormat(rectW, rectH, "A", "", 0, "C", false, 0, "") pdf.SetAlpha(0.5, modeList[j]) pdf.Image(example.ImageFile("golang-gopher.png"), x-gapX, y, rectW+2*gapX, 0, false, "", 0, "") pdf.SetAlpha(1.0, "Normal") x += rectW + gapX j++ } y += rectH + gapY } fileStr := example.Filename("Fpdf_SetAlpha_transparency") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetAlpha_transparency.pdf } // ExampleFpdf_LinearGradient deomstrates various gradients. func ExampleFpdf_LinearGradient() { pdf := gofpdf.New("", "", "", "") pdf.SetFont("Helvetica", "", 12) pdf.AddPage() pdf.LinearGradient(0, 0, 210, 100, 250, 250, 255, 220, 220, 225, 0, 0, 0, .5) pdf.LinearGradient(20, 25, 75, 75, 220, 220, 250, 80, 80, 220, 0, .2, 0, .8) pdf.Rect(20, 25, 75, 75, "D") pdf.LinearGradient(115, 25, 75, 75, 220, 220, 250, 80, 80, 220, 0, 0, 1, 1) pdf.Rect(115, 25, 75, 75, "D") pdf.RadialGradient(20, 120, 75, 75, 220, 220, 250, 80, 80, 220, 0.25, 0.75, 0.25, 0.75, 1) pdf.Rect(20, 120, 75, 75, "D") pdf.RadialGradient(115, 120, 75, 75, 220, 220, 250, 80, 80, 220, 0.25, 0.75, 0.75, 0.75, 0.75) pdf.Rect(115, 120, 75, 75, "D") fileStr := example.Filename("Fpdf_LinearGradient_gradient") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_LinearGradient_gradient.pdf } // ExampleFpdf_ClipText demonstrates clipping. func ExampleFpdf_ClipText() { pdf := gofpdf.New("", "", "", "") y := 10.0 pdf.AddPage() pdf.SetFont("Helvetica", "", 24) pdf.SetXY(0, y) pdf.ClipText(10, y+12, "Clipping examples", false) pdf.RadialGradient(10, y, 100, 20, 128, 128, 160, 32, 32, 48, 0.25, 0.5, 0.25, 0.5, 0.2) pdf.ClipEnd() y += 12 pdf.SetFont("Helvetica", "B", 120) pdf.SetDrawColor(64, 80, 80) pdf.SetLineWidth(.5) pdf.ClipText(10, y+40, pdf.String(), true) pdf.RadialGradient(10, y, 200, 50, 220, 220, 250, 80, 80, 220, 0.25, 0.5, 0.25, 0.5, 1) pdf.ClipEnd() y += 55 pdf.ClipRect(10, y, 105, 20, true) pdf.SetFillColor(255, 255, 255) pdf.Rect(10, y, 105, 20, "F") pdf.ClipCircle(40, y+10, 15, false) pdf.RadialGradient(25, y, 30, 30, 220, 250, 220, 40, 60, 40, 0.3, 0.85, 0.3, 0.85, 0.5) pdf.ClipEnd() pdf.ClipEllipse(80, y+10, 20, 15, false) pdf.RadialGradient(60, y, 40, 30, 250, 220, 220, 60, 40, 40, 0.3, 0.85, 0.3, 0.85, 0.5) pdf.ClipEnd() pdf.ClipEnd() y += 28 pdf.ClipEllipse(26, y+10, 16, 10, true) pdf.Image(example.ImageFile("logo.jpg"), 10, y, 32, 0, false, "JPG", 0, "") pdf.ClipEnd() pdf.ClipCircle(60, y+10, 10, true) pdf.RadialGradient(50, y, 20, 20, 220, 220, 250, 40, 40, 60, 0.3, 0.7, 0.3, 0.7, 0.5) pdf.ClipEnd() pdf.ClipPolygon([]gofpdf.PointType{{X: 80, Y: y + 20}, {X: 90, Y: y}, {X: 100, Y: y + 20}}, true) pdf.LinearGradient(80, y, 20, 20, 250, 220, 250, 60, 40, 60, 0.5, 1, 0.5, 0.5) pdf.ClipEnd() y += 30 pdf.SetLineWidth(.1) pdf.SetDrawColor(180, 180, 180) pdf.ClipRoundedRect(10, y, 120, 20, 5, true) pdf.RadialGradient(10, y, 120, 20, 255, 255, 255, 240, 240, 220, 0.25, 0.75, 0.25, 0.75, 0.5) pdf.SetXY(5, y-5) pdf.SetFont("Times", "", 12) pdf.MultiCell(130, 5, lorem(), "", "", false) pdf.ClipEnd() y += 30 pdf.SetDrawColor(180, 100, 180) pdf.ClipRoundedRectExt(10, y, 120, 20, 5, 10, 5, 10, true) pdf.RadialGradient(10, y, 120, 20, 255, 255, 255, 240, 240, 220, 0.25, 0.75, 0.25, 0.75, 0.5) pdf.SetXY(5, y-5) pdf.SetFont("Times", "", 12) pdf.MultiCell(130, 5, lorem(), "", "", false) pdf.ClipEnd() fileStr := example.Filename("Fpdf_ClipText") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_ClipText.pdf } // ExampleFpdf_PageSize generates a PDF document with various page sizes. func ExampleFpdf_PageSize() { pdf := gofpdf.NewCustom(&gofpdf.InitType{ UnitStr: "in", Size: gofpdf.SizeType{Wd: 6, Ht: 6}, FontDirStr: example.FontDir(), }) pdf.SetMargins(0.5, 1, 0.5) pdf.SetFont("Times", "", 14) pdf.AddPageFormat("L", gofpdf.SizeType{Wd: 3, Ht: 12}) pdf.SetXY(0.5, 1.5) pdf.CellFormat(11, 0.2, "12 in x 3 in", "", 0, "C", false, 0, "") pdf.AddPage() // Default size established in NewCustom() pdf.SetXY(0.5, 3) pdf.CellFormat(5, 0.2, "6 in x 6 in", "", 0, "C", false, 0, "") pdf.AddPageFormat("P", gofpdf.SizeType{Wd: 3, Ht: 12}) pdf.SetXY(0.5, 6) pdf.CellFormat(2, 0.2, "3 in x 12 in", "", 0, "C", false, 0, "") for j := 0; j <= 3; j++ { wd, ht, u := pdf.PageSize(j) fmt.Printf("%d: %6.2f %s, %6.2f %s\n", j, wd, u, ht, u) } fileStr := example.Filename("Fpdf_PageSize") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // 0: 6.00 in, 6.00 in // 1: 12.00 in, 3.00 in // 2: 6.00 in, 6.00 in // 3: 3.00 in, 12.00 in // Successfully generated pdf/Fpdf_PageSize.pdf } // ExampleFpdf_Bookmark demonstrates the Bookmark method. func ExampleFpdf_Bookmark() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 15) pdf.Bookmark("Page 1", 0, 0) pdf.Bookmark("Paragraph 1", 1, -1) pdf.Cell(0, 6, "Paragraph 1") pdf.Ln(50) pdf.Bookmark("Paragraph 2", 1, -1) pdf.Cell(0, 6, "Paragraph 2") pdf.AddPage() pdf.Bookmark("Page 2", 0, 0) pdf.Bookmark("Paragraph 3", 1, -1) pdf.Cell(0, 6, "Paragraph 3") fileStr := example.Filename("Fpdf_Bookmark") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Bookmark.pdf } // ExampleFpdf_TransformBegin demonstrates various transformations. It is adapted from an // example script by Moritz Wagner and Andreas Würmser. func ExampleFpdf_TransformBegin() { const ( light = 200 dark = 0 ) var refX, refY float64 var refStr string pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() color := func(val int) { pdf.SetDrawColor(val, val, val) pdf.SetTextColor(val, val, val) } reference := func(str string, x, y float64, val int) { color(val) pdf.Rect(x, y, 40, 10, "D") pdf.Text(x, y-1, str) } refDraw := func(str string, x, y float64) { refStr = str refX = x refY = y reference(str, x, y, light) } refDupe := func() { reference(refStr, refX, refY, dark) } titleStr := "Transformations" titlePt := 36.0 titleHt := pdf.PointConvert(titlePt) pdf.SetFont("Helvetica", "", titlePt) titleWd := pdf.GetStringWidth(titleStr) titleX := (210 - titleWd) / 2 pdf.Text(titleX, 10+titleHt, titleStr) pdf.TransformBegin() pdf.TransformMirrorVertical(10 + titleHt + 0.5) pdf.ClipText(titleX, 10+titleHt, titleStr, false) // Remember that the transform will mirror the gradient box too pdf.LinearGradient(titleX, 10, titleWd, titleHt+4, 120, 120, 120, 255, 255, 255, 0, 0, 0, 0.6) pdf.ClipEnd() pdf.TransformEnd() pdf.SetFont("Helvetica", "", 12) // Scale by 150% centered by lower left corner of the rectangle refDraw("Scale", 50, 60) pdf.TransformBegin() pdf.TransformScaleXY(150, 50, 70) refDupe() pdf.TransformEnd() // Translate 7 to the right, 5 to the bottom refDraw("Translate", 125, 60) pdf.TransformBegin() pdf.TransformTranslate(7, 5) refDupe() pdf.TransformEnd() // Rotate 20 degrees counter-clockwise centered by the lower left corner of // the rectangle refDraw("Rotate", 50, 110) pdf.TransformBegin() pdf.TransformRotate(20, 50, 120) refDupe() pdf.TransformEnd() // Skew 30 degrees along the x-axis centered by the lower left corner of the // rectangle refDraw("Skew", 125, 110) pdf.TransformBegin() pdf.TransformSkewX(30, 125, 110) refDupe() pdf.TransformEnd() // Mirror horizontally with axis of reflection at left side of the rectangle refDraw("Mirror horizontal", 50, 160) pdf.TransformBegin() pdf.TransformMirrorHorizontal(50) refDupe() pdf.TransformEnd() // Mirror vertically with axis of reflection at bottom side of the rectangle refDraw("Mirror vertical", 125, 160) pdf.TransformBegin() pdf.TransformMirrorVertical(170) refDupe() pdf.TransformEnd() // Reflect against a point at the lower left point of rectangle refDraw("Mirror point", 50, 210) pdf.TransformBegin() pdf.TransformMirrorPoint(50, 220) refDupe() pdf.TransformEnd() // Mirror against a straight line described by a point and an angle angle := -20.0 px := 120.0 py := 220.0 refDraw("Mirror line", 125, 210) pdf.TransformBegin() pdf.TransformRotate(angle, px, py) pdf.Line(px-1, py-1, px+1, py+1) pdf.Line(px-1, py+1, px+1, py-1) pdf.Line(px-5, py, px+60, py) pdf.TransformEnd() pdf.TransformBegin() pdf.TransformMirrorLine(angle, px, py) refDupe() pdf.TransformEnd() fileStr := example.Filename("Fpdf_TransformBegin") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_TransformBegin.pdf } // ExampleFpdf_RegisterImage demonstrates Lawrence Kesteloot's image registration code. func ExampleFpdf_RegisterImage() { const ( margin = 10 wd = 210 ht = 297 ) fileList := []string{ "logo-gray.png", "logo.jpg", "logo.png", "logo-rgb.png", "logo-progressive.jpg", } var infoPtr *gofpdf.ImageInfoType var imageFileStr string var imgWd, imgHt, lf, tp float64 pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetMargins(10, 10, 10) pdf.SetFont("Helvetica", "", 15) for j, str := range fileList { imageFileStr = example.ImageFile(str) infoPtr = pdf.RegisterImage(imageFileStr, "") imgWd, imgHt = infoPtr.Extent() switch j { case 0: lf = margin tp = margin case 1: lf = wd - margin - imgWd tp = margin case 2: lf = (wd - imgWd) / 2.0 tp = (ht - imgHt) / 2.0 case 3: lf = margin tp = ht - imgHt - margin case 4: lf = wd - imgWd - margin tp = ht - imgHt - margin } pdf.Image(imageFileStr, lf, tp, imgWd, imgHt, false, "", 0, "") } fileStr := example.Filename("Fpdf_RegisterImage") // Test the image information retrieval method infoShow := func(imageStr string) { imageStr = example.ImageFile(imageStr) info := pdf.GetImageInfo(imageStr) if info != nil { if info.Width() > 0.0 { fmt.Printf("Image %s is registered\n", filepath.ToSlash(imageStr)) } else { fmt.Printf("Incorrect information for image %s\n", filepath.ToSlash(imageStr)) } } else { fmt.Printf("Image %s is not registered\n", filepath.ToSlash(imageStr)) } } infoShow(fileList[0]) infoShow("foo.png") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Image image/logo-gray.png is registered // Image image/foo.png is not registered // Successfully generated pdf/Fpdf_RegisterImage.pdf } // ExampleFpdf_SplitLines demonstrates Bruno Michel's line splitting function. func ExampleFpdf_SplitLines() { const ( fontPtSize = 18.0 wd = 100.0 ) pdf := gofpdf.New("P", "mm", "A4", "") // A4 210.0 x 297.0 pdf.SetFont("Times", "", fontPtSize) _, lineHt := pdf.GetFontSize() pdf.AddPage() pdf.SetMargins(10, 10, 10) lines := pdf.SplitLines([]byte(lorem()), wd) ht := float64(len(lines)) * lineHt y := (297.0 - ht) / 2.0 pdf.SetDrawColor(128, 128, 128) pdf.SetFillColor(255, 255, 210) x := (210.0 - (wd + 40.0)) / 2.0 pdf.Rect(x, y-20.0, wd+40.0, ht+40.0, "FD") pdf.SetY(y) for _, line := range lines { pdf.CellFormat(190.0, lineHt, string(line), "", 1, "C", false, 0, "") } fileStr := example.Filename("Fpdf_Splitlines") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Splitlines.pdf } // ExampleFpdf_SVGBasicWrite demonstrates how to render a simple path-only SVG image of the // type generated by the jSignature web control. func ExampleFpdf_SVGBasicWrite() { const ( fontPtSize = 16.0 wd = 100.0 sigFileStr = "signature.svg" ) var ( sig gofpdf.SVGBasicType err error ) pdf := gofpdf.New("P", "mm", "A4", "") // A4 210.0 x 297.0 pdf.SetFont("Times", "", fontPtSize) lineHt := pdf.PointConvert(fontPtSize) pdf.AddPage() pdf.SetMargins(10, 10, 10) htmlStr := `This example renders a simple ` + `SVG (scalable vector graphics) ` + `image that contains only basic path commands without any styling, ` + `color fill, reflection or endpoint closures. In particular, the ` + `type of vector graphic returned from a ` + `jSignature ` + `web control is supported and is used in this example.` html := pdf.HTMLBasicNew() html.Write(lineHt, htmlStr) sig, err = gofpdf.SVGBasicFileParse(example.ImageFile(sigFileStr)) if err == nil { scale := 100 / sig.Wd scaleY := 30 / sig.Ht if scale > scaleY { scale = scaleY } pdf.SetLineCapStyle("round") pdf.SetLineWidth(0.25) pdf.SetDrawColor(0, 0, 128) pdf.SetXY((210.0-scale*sig.Wd)/2.0, pdf.GetY()+10) pdf.SVGBasicWrite(&sig, scale) } else { pdf.SetError(err) } fileStr := example.Filename("Fpdf_SVGBasicWrite") err = pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SVGBasicWrite.pdf } // ExampleFpdf_CellFormat_align demonstrates Stefan Schroeder's code to control vertical // alignment. func ExampleFpdf_CellFormat_align() { type recType struct { align, txt string } recList := []recType{ {"TL", "top left"}, {"TC", "top center"}, {"TR", "top right"}, {"LM", "middle left"}, {"CM", "middle center"}, {"RM", "middle right"}, {"BL", "bottom left"}, {"BC", "bottom center"}, {"BR", "bottom right"}, } recListBaseline := []recType{ {"AL", "baseline left"}, {"AC", "baseline center"}, {"AR", "baseline right"}, } var formatRect = func(pdf *gofpdf.Fpdf, recList []recType) { linkStr := "" for pageJ := 0; pageJ < 2; pageJ++ { pdf.AddPage() pdf.SetMargins(10, 10, 10) pdf.SetAutoPageBreak(false, 0) borderStr := "1" for _, rec := range recList { pdf.SetXY(20, 20) pdf.CellFormat(170, 257, rec.txt, borderStr, 0, rec.align, false, 0, linkStr) borderStr = "" } linkStr = "https://github.com/jung-kurt/gofpdf" } } pdf := gofpdf.New("P", "mm", "A4", "") // A4 210.0 x 297.0 pdf.SetFont("Helvetica", "", 16) formatRect(pdf, recList) formatRect(pdf, recListBaseline) var fr fontResourceType pdf.SetFontLoader(fr) pdf.AddFont("Calligrapher", "", "calligra.json") pdf.SetFont("Calligrapher", "", 16) formatRect(pdf, recListBaseline) fileStr := example.Filename("Fpdf_CellFormat_align") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Generalized font loader reading calligra.json // Generalized font loader reading calligra.z // Successfully generated pdf/Fpdf_CellFormat_align.pdf } // ExampleFpdf_CellFormat_codepageescape demonstrates the use of characters in the high range of the // Windows-1252 code page (gofdpf default). See the example for CellFormat (4) // for a way to do this automatically. func ExampleFpdf_CellFormat_codepageescape() { pdf := gofpdf.New("P", "mm", "A4", "") // A4 210.0 x 297.0 fontSize := 16.0 pdf.SetFont("Helvetica", "", fontSize) ht := pdf.PointConvert(fontSize) write := func(str string) { pdf.CellFormat(190, ht, str, "", 1, "C", false, 0, "") pdf.Ln(ht) } pdf.AddPage() htmlStr := `Until gofpdf supports UTF-8 encoded source text, source text needs ` + `to be specified with all special characters escaped to match the code page ` + `layout of the currently selected font. By default, gofdpf uses code page 1252.` + ` See Wikipedia for ` + `a table of this layout.` html := pdf.HTMLBasicNew() html.Write(ht, htmlStr) pdf.Ln(2 * ht) write("Voix ambigu\xeb d'un c\x9cur qui au z\xe9phyr pr\xe9f\xe8re les jattes de kiwi.") write("Falsches \xdcben von Xylophonmusik qu\xe4lt jeden gr\xf6\xdferen Zwerg.") write("Heiz\xf6lr\xfccksto\xdfabd\xe4mpfung") write("For\xe5rsj\xe6vnd\xf8gn / Efter\xe5rsj\xe6vnd\xf8gn") fileStr := example.Filename("Fpdf_CellFormat_codepageescape") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_CellFormat_codepageescape.pdf } // ExampleFpdf_CellFormat_codepage demonstrates the automatic conversion of UTF-8 strings to an // 8-bit font encoding. func ExampleFpdf_CellFormat_codepage() { pdf := gofpdf.New("P", "mm", "A4", example.FontDir()) // A4 210.0 x 297.0 // See documentation for details on how to generate fonts pdf.AddFont("Helvetica-1251", "", "helvetica_1251.json") pdf.AddFont("Helvetica-1253", "", "helvetica_1253.json") fontSize := 16.0 pdf.SetFont("Helvetica", "", fontSize) ht := pdf.PointConvert(fontSize) tr := pdf.UnicodeTranslatorFromDescriptor("") // "" defaults to "cp1252" write := func(str string) { // pdf.CellFormat(190, ht, tr(str), "", 1, "C", false, 0, "") pdf.MultiCell(190, ht, tr(str), "", "C", false) pdf.Ln(ht) } pdf.AddPage() str := `Gofpdf provides a translator that will convert any UTF-8 code point ` + `that is present in the specified code page.` pdf.MultiCell(190, ht, str, "", "L", false) pdf.Ln(2 * ht) write("Voix ambiguë d'un cœur qui au zéphyr préfère les jattes de kiwi.") write("Falsches Üben von Xylophonmusik quält jeden größeren Zwerg.") write("Heizölrückstoßabdämpfung") write("Forårsjævndøgn / Efterårsjævndøgn") write("À noite, vovô Kowalsky vê o ímã cair no pé do pingüim queixoso e vovó" + "põe açúcar no chá de tâmaras do jabuti feliz.") pdf.SetFont("Helvetica-1251", "", fontSize) // Name matches one specified in AddFont() tr = pdf.UnicodeTranslatorFromDescriptor("cp1251") write("Съешь же ещё этих мягких французских булок, да выпей чаю.") pdf.SetFont("Helvetica-1253", "", fontSize) tr = pdf.UnicodeTranslatorFromDescriptor("cp1253") write("Θέλει αρετή και τόλμη η ελευθερία. (Ανδρέας Κάλβος)") fileStr := example.Filename("Fpdf_CellFormat_codepage") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_CellFormat_codepage.pdf } // ExampleFpdf_SetProtection demonstrates password protection for documents. func ExampleFpdf_SetProtection() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetProtection(gofpdf.CnProtectPrint, "123", "abc") pdf.AddPage() pdf.SetFont("Arial", "", 12) pdf.Write(10, "Password-protected.") fileStr := example.Filename("Fpdf_SetProtection") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetProtection.pdf } // ExampleFpdf_Polygon displays equilateral polygons in a demonstration of the Polygon // function. func ExampleFpdf_Polygon() { const rowCount = 5 const colCount = 4 const ptSize = 36 var x, y, radius, gap, advance float64 var rgVal int var pts []gofpdf.PointType vertices := func(count int) (res []gofpdf.PointType) { var pt gofpdf.PointType res = make([]gofpdf.PointType, 0, count) mlt := 2.0 * math.Pi / float64(count) for j := 0; j < count; j++ { pt.Y, pt.X = math.Sincos(float64(j) * mlt) res = append(res, gofpdf.PointType{ X: x + radius*pt.X, Y: y + radius*pt.Y}) } return } pdf := gofpdf.New("P", "mm", "A4", "") // A4 210.0 x 297.0 pdf.AddPage() pdf.SetFont("Helvetica", "", ptSize) pdf.SetDrawColor(0, 80, 180) gap = 12.0 pdf.SetY(gap) pdf.CellFormat(190.0, gap, "Equilateral polygons", "", 1, "C", false, 0, "") radius = (210.0 - float64(colCount+1)*gap) / (2.0 * float64(colCount)) advance = gap + 2.0*radius y = 2*gap + pdf.PointConvert(ptSize) + radius rgVal = 230 for row := 0; row < rowCount; row++ { pdf.SetFillColor(rgVal, rgVal, 0) rgVal -= 12 x = gap + radius for col := 0; col < colCount; col++ { pts = vertices(row*colCount + col + 3) pdf.Polygon(pts, "FD") x += advance } y += advance } fileStr := example.Filename("Fpdf_Polygon") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Polygon.pdf } // ExampleFpdf_AddLayer demonstrates document layers. The initial visibility of a layer // is specified with the second parameter to AddLayer(). The layer list // displayed by the document reader allows layer visibility to be controlled // interactively. func ExampleFpdf_AddLayer() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 15) pdf.Write(8, "This line doesn't belong to any layer.\n") // Define layers l1 := pdf.AddLayer("Layer 1", true) l2 := pdf.AddLayer("Layer 2", true) // Open layer pane in PDF viewer pdf.OpenLayerPane() // First layer pdf.BeginLayer(l1) pdf.Write(8, "This line belongs to layer 1.\n") pdf.EndLayer() // Second layer pdf.BeginLayer(l2) pdf.Write(8, "This line belongs to layer 2.\n") pdf.EndLayer() // First layer again pdf.BeginLayer(l1) pdf.Write(8, "This line belongs to layer 1 again.\n") pdf.EndLayer() fileStr := example.Filename("Fpdf_AddLayer") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_AddLayer.pdf } // ExampleFpdf_RegisterImageReader demonstrates the use of an image that is retrieved from a web // server. func ExampleFpdf_RegisterImageReader() { const ( margin = 10 wd = 210 ht = 297 fontSize = 15 urlStr = "https://github.com/jung-kurt/gofpdf/blob/master/image/gofpdf.png?raw=true" msgStr = `Images from the web can be easily embedded when a PDF document is generated.` ) var ( rsp *http.Response err error tp string ) pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Helvetica", "", fontSize) ln := pdf.PointConvert(fontSize) pdf.MultiCell(wd-margin-margin, ln, msgStr, "", "L", false) rsp, err = http.Get(urlStr) if err == nil { tp = pdf.ImageTypeFromMime(rsp.Header["Content-Type"][0]) infoPtr := pdf.RegisterImageReader(urlStr, tp, rsp.Body) if pdf.Ok() { imgWd, imgHt := infoPtr.Extent() pdf.Image(urlStr, (wd-imgWd)/2.0, pdf.GetY()+ln, imgWd, imgHt, false, tp, 0, "") } } else { pdf.SetError(err) } fileStr := example.Filename("Fpdf_RegisterImageReader_url") err = pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_RegisterImageReader_url.pdf } // ExampleFpdf_Beziergon demonstrates the Beziergon function. func ExampleFpdf_Beziergon() { const ( margin = 10 wd = 210 unit = (wd - 2*margin) / 6 ht = 297 fontSize = 15 msgStr = `Demonstration of Beziergon function` coefficient = 0.6 delta = coefficient * unit ln = fontSize * 25.4 / 72 offsetX = (wd - 4*unit) / 2.0 offsetY = offsetX + 2*ln ) srcList := []gofpdf.PointType{ {X: 0, Y: 0}, {X: 1, Y: 0}, {X: 1, Y: 1}, {X: 2, Y: 1}, {X: 2, Y: 2}, {X: 3, Y: 2}, {X: 3, Y: 3}, {X: 4, Y: 3}, {X: 4, Y: 4}, {X: 1, Y: 4}, {X: 1, Y: 3}, {X: 0, Y: 3}, } ctrlList := []gofpdf.PointType{ {X: 1, Y: -1}, {X: 1, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 1}, {X: 1, Y: 1}, {X: -1, Y: 1}, {X: -1, Y: -1}, {X: -1, Y: -1}, {X: -1, Y: -1}, } pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Helvetica", "", fontSize) for j, src := range srcList { srcList[j].X = offsetX + src.X*unit srcList[j].Y = offsetY + src.Y*unit } for j, ctrl := range ctrlList { ctrlList[j].X = ctrl.X * delta ctrlList[j].Y = ctrl.Y * delta } jPrev := len(srcList) - 1 srcPrev := srcList[jPrev] curveList := []gofpdf.PointType{srcPrev} // point [, control 0, control 1, point]* control := func(x, y float64) { curveList = append(curveList, gofpdf.PointType{X: x, Y: y}) } for j, src := range srcList { ctrl := ctrlList[jPrev] control(srcPrev.X+ctrl.X, srcPrev.Y+ctrl.Y) // Control 0 ctrl = ctrlList[j] control(src.X-ctrl.X, src.Y-ctrl.Y) // Control 1 curveList = append(curveList, src) // Destination jPrev = j srcPrev = src } pdf.MultiCell(wd-margin-margin, ln, msgStr, "", "C", false) pdf.SetDashPattern([]float64{0.8, 0.8}, 0) pdf.SetDrawColor(160, 160, 160) pdf.Polygon(srcList, "D") pdf.SetDashPattern([]float64{}, 0) pdf.SetDrawColor(64, 64, 128) pdf.SetLineWidth(pdf.GetLineWidth() * 3) pdf.Beziergon(curveList, "D") fileStr := example.Filename("Fpdf_Beziergon") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Beziergon.pdf } // ExampleFpdf_SetFontLoader demonstrates loading a non-standard font using a generalized // font loader. fontResourceType implements the FontLoader interface and is // defined locally in the test source code. func ExampleFpdf_SetFontLoader() { var fr fontResourceType pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetFontLoader(fr) pdf.AddFont("Calligrapher", "", "calligra.json") pdf.AddPage() pdf.SetFont("Calligrapher", "", 35) pdf.Cell(0, 10, "Load fonts from any source") fileStr := example.Filename("Fpdf_SetFontLoader") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Generalized font loader reading calligra.json // Generalized font loader reading calligra.z // Successfully generated pdf/Fpdf_SetFontLoader.pdf } // ExampleFpdf_MoveTo demonstrates the Path Drawing functions, such as: MoveTo, // LineTo, CurveTo, ..., ClosePath and DrawPath. func ExampleFpdf_MoveTo() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.MoveTo(20, 20) pdf.LineTo(170, 20) pdf.ArcTo(170, 40, 20, 20, 0, 90, 0) pdf.CurveTo(190, 100, 105, 100) pdf.CurveBezierCubicTo(20, 100, 105, 200, 20, 200) pdf.ClosePath() pdf.SetFillColor(200, 200, 200) pdf.SetLineWidth(3) pdf.DrawPath("DF") fileStr := example.Filename("Fpdf_MoveTo_path") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_MoveTo_path.pdf } // ExampleFpdf_SetLineJoinStyle demonstrates various line cap and line join styles. func ExampleFpdf_SetLineJoinStyle() { const offset = 75.0 pdf := gofpdf.New("L", "mm", "A4", "") pdf.AddPage() var draw = func(cap, join string, x0, y0, x1, y1 float64) { // transform begin & end needed to isolate caps and joins pdf.SetLineCapStyle(cap) pdf.SetLineJoinStyle(join) // Draw thick line pdf.SetDrawColor(0x33, 0x33, 0x33) pdf.SetLineWidth(30.0) pdf.MoveTo(x0, y0) pdf.LineTo((x0+x1)/2+offset, (y0+y1)/2) pdf.LineTo(x1, y1) pdf.DrawPath("D") // Draw thin helping line pdf.SetDrawColor(0xFF, 0x33, 0x33) pdf.SetLineWidth(2.56) pdf.MoveTo(x0, y0) pdf.LineTo((x0+x1)/2+offset, (y0+y1)/2) pdf.LineTo(x1, y1) pdf.DrawPath("D") } x := 35.0 caps := []string{"butt", "square", "round"} joins := []string{"bevel", "miter", "round"} for i := range caps { draw(caps[i], joins[i], x, 50, x, 160) x += offset } fileStr := example.Filename("Fpdf_SetLineJoinStyle_caps") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetLineJoinStyle_caps.pdf } // ExampleFpdf_DrawPath demonstrates various fill modes. func ExampleFpdf_DrawPath() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetDrawColor(0xff, 0x00, 0x00) pdf.SetFillColor(0x99, 0x99, 0x99) pdf.SetFont("Helvetica", "", 15) pdf.AddPage() pdf.SetAlpha(1, "Multiply") var ( polygon = func(cx, cy, r, n, dir float64) { da := 2 * math.Pi / n pdf.MoveTo(cx+r, cy) pdf.Text(cx+r, cy, "0") i := 1 for a := da; a < 2*math.Pi; a += da { x, y := cx+r*math.Cos(dir*a), cy+r*math.Sin(dir*a) pdf.LineTo(x, y) pdf.Text(x, y, strconv.Itoa(i)) i++ } pdf.ClosePath() } polygons = func(cx, cy, r, n, dir float64) { d := 1.0 for rf := r; rf > 0; rf -= 10 { polygon(cx, cy, rf, n, d) d *= dir } } star = func(cx, cy, r, n float64) { da := 4 * math.Pi / n pdf.MoveTo(cx+r, cy) for a := da; a < 4*math.Pi+da; a += da { x, y := cx+r*math.Cos(a), cy+r*math.Sin(a) pdf.LineTo(x, y) } pdf.ClosePath() } ) // triangle polygons(55, 45, 40, 3, 1) pdf.DrawPath("B") pdf.Text(15, 95, "B (same direction, non zero winding)") // square polygons(155, 45, 40, 4, 1) pdf.DrawPath("B*") pdf.Text(115, 95, "B* (same direction, even odd)") // pentagon polygons(55, 145, 40, 5, -1) pdf.DrawPath("B") pdf.Text(15, 195, "B (different direction, non zero winding)") // hexagon polygons(155, 145, 40, 6, -1) pdf.DrawPath("B*") pdf.Text(115, 195, "B* (different direction, even odd)") // star star(55, 245, 40, 5) pdf.DrawPath("B") pdf.Text(15, 290, "B (non zero winding)") // star star(155, 245, 40, 5) pdf.DrawPath("B*") pdf.Text(115, 290, "B* (even odd)") fileStr := example.Filename("Fpdf_DrawPath_fill") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_DrawPath_fill.pdf } // ExampleFpdf_CreateTemplate demonstrates creating and using templates func ExampleFpdf_CreateTemplate() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetCompression(false) // pdf.SetFont("Times", "", 12) template := pdf.CreateTemplate(func(tpl *gofpdf.Tpl) { tpl.Image(example.ImageFile("logo.png"), 6, 6, 30, 0, false, "", 0, "") tpl.SetFont("Arial", "B", 16) tpl.Text(40, 20, "Template says hello") tpl.SetDrawColor(0, 100, 200) tpl.SetLineWidth(2.5) tpl.Line(95, 12, 105, 22) }) _, tplSize := template.Size() // fmt.Println("Size:", tplSize) // fmt.Println("Scaled:", tplSize.ScaleBy(1.5)) template2 := pdf.CreateTemplate(func(tpl *gofpdf.Tpl) { tpl.UseTemplate(template) subtemplate := tpl.CreateTemplate(func(tpl2 *gofpdf.Tpl) { tpl2.Image(example.ImageFile("logo.png"), 6, 86, 30, 0, false, "", 0, "") tpl2.SetFont("Arial", "B", 16) tpl2.Text(40, 100, "Subtemplate says hello") tpl2.SetDrawColor(0, 200, 100) tpl2.SetLineWidth(2.5) tpl2.Line(102, 92, 112, 102) }) tpl.UseTemplate(subtemplate) }) pdf.SetDrawColor(200, 100, 0) pdf.SetLineWidth(2.5) pdf.SetFont("Arial", "B", 16) // serialize and deserialize template b, _ := template2.Serialize() template3, _ := gofpdf.DeserializeTemplate(b) pdf.AddPage() pdf.UseTemplate(template3) pdf.UseTemplateScaled(template3, gofpdf.PointType{X: 0, Y: 30}, tplSize) pdf.Line(40, 210, 60, 210) pdf.Text(40, 200, "Template example page 1") pdf.AddPage() pdf.UseTemplate(template2) pdf.UseTemplateScaled(template3, gofpdf.PointType{X: 0, Y: 30}, tplSize.ScaleBy(1.4)) pdf.Line(60, 210, 80, 210) pdf.Text(40, 200, "Template example page 2") fileStr := example.Filename("Fpdf_CreateTemplate") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_CreateTemplate.pdf } // ExampleFpdf_AddFontFromBytes demonstrate how to use embedded fonts from byte array func ExampleFpdf_AddFontFromBytes() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.AddFontFromBytes("calligra", "", files.CalligraJson, files.CalligraZ) pdf.SetFont("calligra", "", 16) pdf.Cell(40, 10, "Hello World With Embedded Font!") fileStr := example.Filename("Fpdf_EmbeddedFont") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_EmbeddedFont.pdf } // This example demonstrate Clipped table cells func ExampleFpdf_ClipRect() { marginCell := 2. // margin of top/bottom of cell pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetFont("Arial", "", 12) pdf.AddPage() pagew, pageh := pdf.GetPageSize() mleft, mright, _, mbottom := pdf.GetMargins() cols := []float64{60, 100, pagew - mleft - mright - 100 - 60} rows := [][]string{} for i := 1; i <= 50; i++ { word := fmt.Sprintf("%d:%s", i, strings.Repeat("A", i%100)) rows = append(rows, []string{word, word, word}) } for _, row := range rows { _, lineHt := pdf.GetFontSize() height := lineHt + marginCell x, y := pdf.GetXY() // add a new page if the height of the row doesn't fit on the page if y+height >= pageh-mbottom { pdf.AddPage() x, y = pdf.GetXY() } for i, txt := range row { width := cols[i] pdf.Rect(x, y, width, height, "") pdf.ClipRect(x, y, width, height, false) pdf.Cell(width, height, txt) pdf.ClipEnd() x += width } pdf.Ln(-1) } fileStr := example.Filename("Fpdf_ClippedTableCells") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_ClippedTableCells.pdf } // This example demonstrate wrapped table cells func ExampleFpdf_Rect() { marginCell := 2. // margin of top/bottom of cell pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetFont("Arial", "", 12) pdf.AddPage() pagew, pageh := pdf.GetPageSize() mleft, mright, _, mbottom := pdf.GetMargins() cols := []float64{60, 100, pagew - mleft - mright - 100 - 60} rows := [][]string{} for i := 1; i <= 30; i++ { word := fmt.Sprintf("%d:%s", i, strings.Repeat("A", i%100)) rows = append(rows, []string{word, word, word}) } for _, row := range rows { curx, y := pdf.GetXY() x := curx height := 0. _, lineHt := pdf.GetFontSize() for i, txt := range row { lines := pdf.SplitLines([]byte(txt), cols[i]) h := float64(len(lines))*lineHt + marginCell*float64(len(lines)) if h > height { height = h } } // add a new page if the height of the row doesn't fit on the page if pdf.GetY()+height > pageh-mbottom { pdf.AddPage() y = pdf.GetY() } for i, txt := range row { width := cols[i] pdf.Rect(x, y, width, height, "") pdf.MultiCell(width, lineHt+marginCell, txt, "", "", false) x += width pdf.SetXY(x, y) } pdf.SetXY(curx, y+height) } fileStr := example.Filename("Fpdf_WrappedTableCells") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_WrappedTableCells.pdf } // ExampleFpdf_SetJavascript demonstrates including JavaScript in the document. func ExampleFpdf_SetJavascript() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetJavascript("print(true);") pdf.AddPage() pdf.SetFont("Arial", "", 12) pdf.Write(10, "Auto-print.") fileStr := example.Filename("Fpdf_SetJavascript") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetJavascript.pdf } // ExampleFpdf_AddSpotColor demonstrates spot color use func ExampleFpdf_AddSpotColor() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddSpotColor("PANTONE 145 CVC", 0, 42, 100, 25) pdf.AddPage() pdf.SetFillSpotColor("PANTONE 145 CVC", 90) pdf.Rect(80, 40, 50, 50, "F") fileStr := example.Filename("Fpdf_AddSpotColor") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_AddSpotColor.pdf } // ExampleFpdf_RegisterAlias demonstrates how to use `RegisterAlias` to create a table of // contents. func ExampleFpdf_RegisterAlias() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetFont("Arial", "", 12) pdf.AliasNbPages("") pdf.AddPage() // Write the table of contents. We use aliases instead of the page number // because we don't know which page the section will begin on. numSections := 3 for i := 1; i <= numSections; i++ { pdf.Cell(0, 10, fmt.Sprintf("Section %d begins on page {mark %d}", i, i)) pdf.Ln(10) } // Write the sections. Before we start writing, we use `RegisterAlias` to // ensure that the alias written in the table of contents will be replaced // by the current page number. for i := 1; i <= numSections; i++ { pdf.AddPage() pdf.RegisterAlias(fmt.Sprintf("{mark %d}", i), fmt.Sprintf("%d", pdf.PageNo())) pdf.Write(10, fmt.Sprintf("Section %d, page %d of {nb}", i, pdf.PageNo())) } fileStr := example.Filename("Fpdf_RegisterAlias") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_RegisterAlias.pdf } // ExampleFpdf_RegisterAlias_utf8 demonstrates how to use `RegisterAlias` to // create a table of contents. This particular example demonstrates the use of // UTF-8 aliases. func ExampleFpdf_RegisterAlias_utf8() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddUTF8Font("dejavu", "", example.FontFile("DejaVuSansCondensed.ttf")) pdf.SetFont("dejavu", "", 12) pdf.AliasNbPages("{entute}") pdf.AddPage() // Write the table of contents. We use aliases instead of the page number // because we don't know which page the section will begin on. numSections := 3 for i := 1; i <= numSections; i++ { pdf.Cell(0, 10, fmt.Sprintf("Sekcio %d komenciĝas ĉe paĝo {ĉi tiu marko %d}", i, i)) pdf.Ln(10) } // Write the sections. Before we start writing, we use `RegisterAlias` to // ensure that the alias written in the table of contents will be replaced // by the current page number. for i := 1; i <= numSections; i++ { pdf.AddPage() pdf.RegisterAlias(fmt.Sprintf("{ĉi tiu marko %d}", i), fmt.Sprintf("%d", pdf.PageNo())) pdf.Write(10, fmt.Sprintf("Sekcio %d, paĝo %d de {entute}", i, pdf.PageNo())) } fileStr := example.Filename("Fpdf_RegisterAliasUTF8") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_RegisterAliasUTF8.pdf } // ExampleNewGrid demonstrates the generation of graph grids. func ExampleNewGrid() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetFont("Arial", "", 12) pdf.AddPage() gr := gofpdf.NewGrid(13, 10, 187, 130) gr.TickmarksExtentX(0, 10, 4) gr.TickmarksExtentY(0, 10, 3) gr.Grid(pdf) gr = gofpdf.NewGrid(13, 154, 187, 128) gr.XLabelRotate = true gr.TickmarksExtentX(0, 1, 12) gr.XDiv = 5 gr.TickmarksContainY(0, 1.1) gr.YDiv = 20 // Replace X label formatter with month abbreviation gr.XTickStr = func(val float64, precision int) string { return time.Month(math.Mod(val, 12) + 1).String()[0:3] } gr.Grid(pdf) dot := func(x, y float64) { pdf.Circle(gr.X(x), gr.Y(y), 0.5, "F") } pts := []float64{0.39, 0.457, 0.612, 0.84, 0.998, 1.037, 1.015, 0.918, 0.772, 0.659, 0.593, 0.164} for month, val := range pts { dot(float64(month)+0.5, val) } pdf.SetDrawColor(255, 64, 64) pdf.SetAlpha(0.5, "Normal") pdf.SetLineWidth(1.2) gr.Plot(pdf, 0.5, 11.5, 50, func(x float64) float64 { // http://www.xuru.org/rt/PR.asp return 0.227 * math.Exp(-0.0373*x*x+0.471*x) }) pdf.SetAlpha(1.0, "Normal") pdf.SetXY(gr.X(0.5), gr.Y(1.35)) pdf.SetFontSize(14) pdf.Write(0, "Solar energy (MWh) per month, 2016") pdf.AddPage() gr = gofpdf.NewGrid(13, 10, 187, 274) gr.TickmarksContainX(2.3, 3.4) gr.TickmarksContainY(10.4, 56.8) gr.Grid(pdf) fileStr := example.Filename("Fpdf_Grid") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Grid.pdf } // ExampleFpdf_SetPageBox demonstrates the use of a page box func ExampleFpdf_SetPageBox() { // pdfinfo (from http://www.xpdfreader.com) reports the following for this example: // ~ pdfinfo -box pdf/Fpdf_PageBox.pdf // Producer: FPDF 1.7 // CreationDate: Sat Jan 1 00:00:00 2000 // ModDate: Sat Jan 1 00:00:00 2000 // Tagged: no // Form: none // Pages: 1 // Encrypted: no // Page size: 493.23 x 739.85 pts (rotated 0 degrees) // MediaBox: 0.00 0.00 595.28 841.89 // CropBox: 51.02 51.02 544.25 790.87 // BleedBox: 51.02 51.02 544.25 790.87 // TrimBox: 51.02 51.02 544.25 790.87 // ArtBox: 51.02 51.02 544.25 790.87 // File size: 1053 bytes // Optimized: no // PDF version: 1.3 const ( wd = 210 ht = 297 fontsize = 6 boxmargin = 3 * fontsize ) pdf := gofpdf.New("P", "mm", "A4", "") // 210mm x 297mm pdf.SetPageBox("crop", boxmargin, boxmargin, wd-2*boxmargin, ht-2*boxmargin) pdf.SetFont("Arial", "", pdf.UnitToPointConvert(fontsize)) pdf.AddPage() pdf.MoveTo(fontsize, fontsize) pdf.Write(fontsize, "This will be cropped from printed output") pdf.MoveTo(boxmargin+fontsize, boxmargin+fontsize) pdf.Write(fontsize, "This will be displayed in cropped output") fileStr := example.Filename("Fpdf_PageBox") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_PageBox.pdf } // ExampleFpdf_SubWrite demonstrates subscripted and superscripted text // Adapted from http://www.fpdf.org/en/script/script61.php func ExampleFpdf_SubWrite() { const ( fontSize = 12 halfX = 105 ) pdf := gofpdf.New("P", "mm", "A4", "") // 210mm x 297mm pdf.AddPage() pdf.SetFont("Arial", "", fontSize) _, lineHt := pdf.GetFontSize() pdf.Write(lineHt, "Hello World!") pdf.SetX(halfX) pdf.Write(lineHt, "This is standard text.\n") pdf.Ln(lineHt * 2) pdf.SubWrite(10, "H", 33, 0, 0, "") pdf.Write(10, "ello World!") pdf.SetX(halfX) pdf.Write(10, "This is text with a capital first letter.\n") pdf.Ln(lineHt * 2) pdf.SubWrite(lineHt, "Y", 6, 0, 0, "") pdf.Write(lineHt, "ou can also begin the sentence with a small letter. And word wrap also works if the line is too long, like this one is.") pdf.SetX(halfX) pdf.Write(lineHt, "This is text with a small first letter.\n") pdf.Ln(lineHt * 2) pdf.Write(lineHt, "The world has a lot of km") pdf.SubWrite(lineHt, "2", 6, 4, 0, "") pdf.SetX(halfX) pdf.Write(lineHt, "This is text with a superscripted letter.\n") pdf.Ln(lineHt * 2) pdf.Write(lineHt, "The world has a lot of H") pdf.SubWrite(lineHt, "2", 6, -3, 0, "") pdf.Write(lineHt, "O") pdf.SetX(halfX) pdf.Write(lineHt, "This is text with a subscripted letter.\n") fileStr := example.Filename("Fpdf_SubWrite") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SubWrite.pdf } // ExampleFpdf_SetPage demomstrates the SetPage() method, allowing content // generation to be deferred until all pages have been added. func ExampleFpdf_SetPage() { rnd := rand.New(rand.NewSource(0)) // Make reproducible documents pdf := gofpdf.New("L", "cm", "A4", "") pdf.SetFont("Times", "", 12) var time []float64 temperaturesFromSensors := make([][]float64, 5) maxs := []float64{25, 41, 89, 62, 11} for i := range temperaturesFromSensors { temperaturesFromSensors[i] = make([]float64, 0) } for i := 0.0; i < 100; i += 0.5 { time = append(time, i) for j, sensor := range temperaturesFromSensors { dataValue := rnd.Float64() * maxs[j] sensor = append(sensor, dataValue) temperaturesFromSensors[j] = sensor } } var graphs []gofpdf.GridType var pageNums []int xMax := time[len(time)-1] for i := range temperaturesFromSensors { //Create a new page and graph for each sensor we want to graph. pdf.AddPage() pdf.Ln(1) //Custom label per sensor pdf.WriteAligned(0, 0, "Temperature Sensor "+strconv.Itoa(i+1)+" (C) vs Time (min)", "C") pdf.Ln(0.5) graph := gofpdf.NewGrid(pdf.GetX(), pdf.GetY(), 20, 10) graph.TickmarksContainX(0, xMax) //Custom Y axis graph.TickmarksContainY(0, maxs[i]) graph.Grid(pdf) //Save references and locations. graphs = append(graphs, graph) pageNums = append(pageNums, pdf.PageNo()) } // For each X, graph the Y in each sensor. for i, currTime := range time { for j, sensor := range temperaturesFromSensors { pdf.SetPage(pageNums[j]) graph := graphs[j] temperature := sensor[i] pdf.Circle(graph.X(currTime), graph.Y(temperature), 0.04, "D") } } fileStr := example.Filename("Fpdf_SetPage") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetPage.pdf } // ExampleFpdf_SetFillColor demonstrates how graphic attributes are properly // assigned within multiple transformations. See issue #234. func ExampleFpdf_SetFillColor() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.SetFont("Arial", "", 8) draw := func(trX, trY float64) { pdf.TransformBegin() pdf.TransformTranslateX(trX) pdf.TransformTranslateY(trY) pdf.SetLineJoinStyle("round") pdf.SetLineWidth(0.5) pdf.SetDrawColor(128, 64, 0) pdf.SetFillColor(255, 127, 0) pdf.SetAlpha(0.5, "Normal") pdf.SetDashPattern([]float64{5, 10}, 0) pdf.Rect(0, 0, 40, 40, "FD") pdf.SetFontSize(12) pdf.SetXY(5, 5) pdf.Write(0, "Test") pdf.TransformEnd() } draw(5, 5) draw(50, 50) fileStr := example.Filename("Fpdf_SetFillColor") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetFillColor.pdf } // ExampleFpdf_TransformRotate demonstrates how to rotate text within a header // to make a watermark that appears on each page. func ExampleFpdf_TransformRotate() { loremStr := lorem() + "\n\n" pdf := gofpdf.New("P", "mm", "A4", "") margin := 25.0 pdf.SetMargins(margin, margin, margin) fontHt := 13.0 lineHt := pdf.PointToUnitConvert(fontHt) markFontHt := 50.0 markLineHt := pdf.PointToUnitConvert(markFontHt) markY := (297.0 - markLineHt) / 2.0 ctrX := 210.0 / 2.0 ctrY := 297.0 / 2.0 pdf.SetHeaderFunc(func() { pdf.SetFont("Arial", "B", markFontHt) pdf.SetTextColor(206, 216, 232) pdf.SetXY(margin, markY) pdf.TransformBegin() pdf.TransformRotate(45, ctrX, ctrY) pdf.CellFormat(0, markLineHt, "W A T E R M A R K D E M O", "", 0, "C", false, 0, "") pdf.TransformEnd() pdf.SetXY(margin, margin) }) pdf.AddPage() pdf.SetFont("Arial", "", 8) for j := 0; j < 25; j++ { pdf.MultiCell(0, lineHt, loremStr, "", "L", false) } fileStr := example.Filename("Fpdf_RotateText") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_RotateText.pdf } // ExampleFpdf_AddUTF8Font demonstrates how use the font // with utf-8 mode func ExampleFpdf_AddUTF8Font() { var fileStr string var txtStr []byte var err error pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() pdf.AddUTF8Font("dejavu", "", example.FontFile("DejaVuSansCondensed.ttf")) pdf.AddUTF8Font("dejavu", "B", example.FontFile("DejaVuSansCondensed-Bold.ttf")) pdf.AddUTF8Font("dejavu", "I", example.FontFile("DejaVuSansCondensed-Oblique.ttf")) pdf.AddUTF8Font("dejavu", "BI", example.FontFile("DejaVuSansCondensed-BoldOblique.ttf")) fileStr = example.Filename("Fpdf_AddUTF8Font") txtStr, err = ioutil.ReadFile(example.TextFile("utf-8test.txt")) if err == nil { pdf.SetFont("dejavu", "B", 17) pdf.MultiCell(100, 8, "Text in different languages :", "", "C", false) pdf.SetFont("dejavu", "", 14) pdf.MultiCell(100, 5, string(txtStr), "", "C", false) pdf.Ln(15) txtStr, err = ioutil.ReadFile(example.TextFile("utf-8test2.txt")) if err == nil { pdf.SetFont("dejavu", "BI", 17) pdf.MultiCell(100, 8, "Greek text with alignStr = \"J\":", "", "C", false) pdf.SetFont("dejavu", "I", 14) pdf.MultiCell(100, 5, string(txtStr), "", "J", false) err = pdf.OutputFileAndClose(fileStr) } } example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_AddUTF8Font.pdf } // ExampleUTF8CutFont demonstrates how generate a TrueType font subset. func ExampleUTF8CutFont() { var pdfFileStr, fullFontFileStr, subFontFileStr string var subFont, fullFont []byte var err error pdfFileStr = example.Filename("Fpdf_UTF8CutFont") fullFontFileStr = example.FontFile("calligra.ttf") fullFont, err = ioutil.ReadFile(fullFontFileStr) if err == nil { subFontFileStr = "calligra_abcde.ttf" subFont = gofpdf.UTF8CutFont(fullFont, "abcde") err = ioutil.WriteFile(subFontFileStr, subFont, 0600) if err == nil { y := 24.0 pdf := gofpdf.New("P", "mm", "A4", "") fontHt := 17.0 lineHt := pdf.PointConvert(fontHt) write := func(format string, args ...interface{}) { pdf.SetXY(24.0, y) pdf.Cell(200.0, lineHt, fmt.Sprintf(format, args...)) y += lineHt } writeSize := func(fileStr string) { var info os.FileInfo var err error info, err = os.Stat(fileStr) if err == nil { write("%6d: size of %s", info.Size(), fileStr) } } pdf.AddPage() pdf.AddUTF8Font("calligra", "", subFontFileStr) pdf.SetFont("calligra", "", fontHt) write("cabbed") write("vwxyz") pdf.SetFont("courier", "", fontHt) writeSize(fullFontFileStr) writeSize(subFontFileStr) err = pdf.OutputFileAndClose(pdfFileStr) os.Remove(subFontFileStr) } } example.Summary(err, pdfFileStr) // Output: // Successfully generated pdf/Fpdf_UTF8CutFont.pdf } func ExampleFpdf_RoundedRect() { const ( wd = 40.0 hgap = 10.0 radius = 10.0 ht = 60.0 vgap = 10.0 ) corner := func(b1, b2, b3, b4 bool) (cstr string) { if b1 { cstr = "1" } if b2 { cstr += "2" } if b3 { cstr += "3" } if b4 { cstr += "4" } return } pdf := gofpdf.New("P", "mm", "A4", "") // 210 x 297 pdf.AddPage() pdf.SetLineWidth(0.5) y := vgap r := 40 g := 30 b := 20 for row := 0; row < 4; row++ { x := hgap for col := 0; col < 4; col++ { pdf.SetFillColor(r, g, b) pdf.RoundedRect(x, y, wd, ht, radius, corner(row&1 == 1, row&2 == 2, col&1 == 1, col&2 == 2), "FD") r += 8 g += 10 b += 12 x += wd + hgap } y += ht + vgap } pdf.AddPage() pdf.RoundedRectExt(10, 20, 40, 80, 4., 0., 20, 0., "FD") fileStr := example.Filename("Fpdf_RoundedRect") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_RoundedRect.pdf } // ExampleFpdf_SetUnderlineThickness demonstrates how to adjust the text // underline thickness. func ExampleFpdf_SetUnderlineThickness() { pdf := gofpdf.New("P", "mm", "A4", "") // 210mm x 297mm pdf.AddPage() pdf.SetFont("Arial", "U", 12) pdf.SetUnderlineThickness(0.5) pdf.CellFormat(0, 10, "Thin underline", "", 1, "", false, 0, "") pdf.SetUnderlineThickness(1) pdf.CellFormat(0, 10, "Normal underline", "", 1, "", false, 0, "") pdf.SetUnderlineThickness(2) pdf.CellFormat(0, 10, "Thicker underline", "", 1, "", false, 0, "") fileStr := example.Filename("Fpdf_UnderlineThickness") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_UnderlineThickness.pdf } // ExampleFpdf_Cell_strikeout demonstrates striked-out text func ExampleFpdf_Cell_strikeout() { pdf := gofpdf.New("P", "mm", "A4", "") // 210mm x 297mm pdf.AddPage() for fontSize := 4; fontSize < 40; fontSize += 10 { pdf.SetFont("Arial", "S", float64(fontSize)) pdf.SetXY(0, float64(fontSize)) pdf.Cell(40, 10, "Hello World") } fileStr := example.Filename("Fpdf_Cell_strikeout") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_Cell_strikeout.pdf } // ExampleFpdf_SetTextRenderingMode demonstrates rendering modes in PDFs. func ExampleFpdf_SetTextRenderingMode() { pdf := gofpdf.New("P", "mm", "A4", "") // 210mm x 297mm pdf.AddPage() fontSz := float64(16) lineSz := pdf.PointToUnitConvert(fontSz) pdf.SetFont("Times", "", fontSz) pdf.Write(lineSz, "This document demonstrates various modes of text rendering. Search for \"Mode 3\" "+ "to locate text that has been rendered invisibly. This selection can be copied "+ "into the clipboard as usual and is useful for overlaying onto non-textual elements such "+ "as images to make them searchable.\n\n") fontSz = float64(125) lineSz = pdf.PointToUnitConvert(fontSz) pdf.SetFontSize(fontSz) pdf.SetTextColor(170, 170, 190) pdf.SetDrawColor(50, 60, 90) write := func(mode int) { pdf.SetTextRenderingMode(mode) pdf.CellFormat(210, lineSz, fmt.Sprintf("Mode %d", mode), "", 1, "", false, 0, "") } for mode := 0; mode < 4; mode++ { write(mode) } write(0) fileStr := example.Filename("Fpdf_TextRenderingMode") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_TextRenderingMode.pdf } // TestIssue0316 addresses issue 316 in which AddUTF8FromBytes modifies its argument // utf8bytes resulting in a panic if you generate two PDFs with the "same" font bytes. func TestIssue0316(t *testing.T) { pdf := gofpdf.New(gofpdf.OrientationPortrait, "mm", "A4", "") pdf.AddPage() fontBytes, _ := ioutil.ReadFile(example.FontFile("DejaVuSansCondensed.ttf")) ofontBytes := append([]byte{}, fontBytes...) pdf.AddUTF8FontFromBytes("dejavu", "", fontBytes) pdf.SetFont("dejavu", "", 16) pdf.Cell(40, 10, "Hello World!") fileStr := example.Filename("TestIssue0316") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) pdf.AddPage() if !bytes.Equal(fontBytes, ofontBytes) { t.Fatal("Font data changed during pdf generation") } } func TestMultiCellUnsupportedChar(t *testing.T) { pdf := gofpdf.New("P", "mm", "A4", "") pdf.AddPage() fontBytes, _ := ioutil.ReadFile(example.FontFile("DejaVuSansCondensed.ttf")) pdf.AddUTF8FontFromBytes("dejavu", "", fontBytes) pdf.SetFont("dejavu", "", 16) defer func() { if r := recover(); r != nil { t.Errorf("unexpected panic: %v", r) } }() pdf.MultiCell(0, 5, "😀", "", "", false) fileStr := example.Filename("TestMultiCellUnsupportedChar") pdf.OutputFileAndClose(fileStr) } // ExampleFpdf_SetTextRenderingMode demonstrates embedding files in PDFs, // at the top-level. func ExampleFpdf_SetAttachments() { pdf := gofpdf.New("P", "mm", "A4", "") // Global attachments file, err := ioutil.ReadFile("grid.go") if err != nil { pdf.SetError(err) } a1 := gofpdf.Attachment{Content: file, Filename: "grid.go"} file, err = ioutil.ReadFile("LICENSE") if err != nil { pdf.SetError(err) } a2 := gofpdf.Attachment{Content: file, Filename: "License"} pdf.SetAttachments([]gofpdf.Attachment{a1, a2}) fileStr := example.Filename("Fpdf_EmbeddedFiles") err = pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_EmbeddedFiles.pdf } func ExampleFpdf_AddAttachmentAnnotation() { pdf := gofpdf.New("P", "mm", "A4", "") pdf.SetFont("Arial", "", 12) pdf.AddPage() // Per page attachment file, err := ioutil.ReadFile("grid.go") if err != nil { pdf.SetError(err) } a := gofpdf.Attachment{Content: file, Filename: "grid.go", Description: "Some amazing code !"} pdf.SetXY(5, 10) pdf.Rect(2, 10, 50, 15, "D") pdf.AddAttachmentAnnotation(&a, 2, 10, 50, 15) pdf.Cell(50, 15, "A first link") pdf.SetXY(5, 80) pdf.Rect(2, 80, 50, 15, "D") pdf.AddAttachmentAnnotation(&a, 2, 80, 50, 15) pdf.Cell(50, 15, "A second link (no copy)") fileStr := example.Filename("Fpdf_FileAnnotations") err = pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_FileAnnotations.pdf } func ExampleFpdf_SetModificationDate() { // pdfinfo (from http://www.xpdfreader.com) reports the following for this example : // ~ pdfinfo -box pdf/Fpdf_PageBox.pdf // Producer: FPDF 1.7 // CreationDate: Sat Jan 1 00:00:00 2000 // ModDate: Sun Jan 2 10:22:30 2000 pdf := gofpdf.New("", "", "", "") pdf.AddPage() pdf.SetModificationDate(time.Date(2000, 1, 2, 10, 22, 30, 0, time.UTC)) fileStr := example.Filename("Fpdf_SetModificationDate") err := pdf.OutputFileAndClose(fileStr) example.Summary(err, fileStr) // Output: // Successfully generated pdf/Fpdf_SetModificationDate.pdf }