web-dev-qa-db-ja.com

Goの画像に簡単なテキストラベルを追加する方法は?

与えられたimage.RGBA、座標、およびテキストの行、単純な固定フォントで簡単なラベルを追加するにはどうすればよいですか?例えば。 Face7x13 from font/basicfont

package main

import (
    "image"
    "image/color"
    "image/png"
    "os"
)

func main() {
    img := image.NewRGBA(image.Rect(0, 0, 320, 240))
    x, y := 100, 100
    addLabel(img, x, y, "Test123")
    png.Encode(os.Stdout, img)
}

func addLabel(img *image.RGBA, x, y int, label string) {
     col := color.Black
     // now what?
}

整列は実際には重要ではありませんが、座標で始まる線の上にラベルを書くことができれば最高です。

そして、フォントのような外部のロード可能な依存関係を避けたいです。

29
sanmai

golang.org/x/image/font パッケージは、フォントフェイスと画像上のテキストの描画のためのインターフェイスを定義するだけです。

FreetypeフォントラスタライザのGo実装を使用できます: github.com/golang/freetype

キータイプは freetype.Context で、必要なすべてのメソッドがあります。

完全な例については、このファイルを確認してください: example/freetype/main.go 。この例では、フォントファイルを読み込み、freetype.Contextを作成および構成し、画像にテキストを描画して、結果の画像をファイルに保存します。

フォントファイルが既にロードされ、cコンテキストが構成されていると仮定しましょう(その方法の例を参照してください)。 addLabel()関数は次のようになります。

func addLabel(img *image.RGBA, x, y int, label string) {
    c.SetDst(img)
    size := 12.0 // font size in pixels
    pt := freetype.Pt(x, y+int(c.PointToFixed(size)>>6))

    if _, err := c.DrawString(label, pt); err != nil {
        // handle error
    }
}

freetypeパッケージと外部フォントファイルに煩わされたくない場合、 font/basicfont パッケージには、グラフィカルデータが完全に自己完結しているFace7x13という名前の基本フォントが含まれます。これはあなたがそれを使用する方法です:

import (
    "golang.org/x/image/font"
    "golang.org/x/image/font/basicfont"
    "golang.org/x/image/math/fixed"
    "image"
    "image/color"
)

func addLabel(img *image.RGBA, x, y int, label string) {
    col := color.RGBA{200, 100, 0, 255}
    point := fixed.Point26_6{fixed.Int26_6(x * 64), fixed.Int26_6(y * 64)}

    d := &font.Drawer{
        Dst:  img,
        Src:  image.NewUniform(col),
        Face: basicfont.Face7x13,
        Dot:  point,
    }
    d.DrawString(label)
}

これが、このaddLabel()関数の使用方法です。次のコードは、新しい画像を作成し、その上に"Hello Go"テキストを描画し、hello-go.pngという名前のファイルに保存します。

func main() {
    img := image.NewRGBA(image.Rect(0, 0, 300, 100))
    addLabel(img, 20, 30, "Hello Go")

    f, err := os.Create("hello-go.png")
    if err != nil {
        panic(err)
    }
    defer f.Close()
    if err := png.Encode(f, img); err != nil {
        panic(err)
    }
}

上記のコードには "image/png" パッケージのインポートも必要です。

また、指定されたy座標がテキストの最下行になることに注意してください。したがって、左上隅に線を描画する場合は、x = 0およびy = 13を使用する必要があります(13はこのFace7x13フォントの高さです)。必要に応じて、y座標から13を引くことにより、これをaddLabel()関数に組み込むことができます。これにより、渡されたy座標がテキストの最上位座標になります。描画されます。

golang.org/x/image/font/inconsolata パッケージには、通常の太字スタイルの追加の内蔵フォントもあります。それらを使用するには、異なる Face を指定するだけです。 addLabel()

import "golang.org/x/image/font/inconsolata"

        // To use regular Inconsolata font family:
        Face: inconsolata.Regular8x16,

        // To use bold Inconsolata font family:
        Face: inconsolata.Bold8x16,
42
icza

これは、すでにsrc.jpgまたは任意の画像があり、その上にテキストを書き込むggライブラリを使用したサンプルコードです。それに応じてキャンバスサイズを調整できます。これは単なる例です。機能しない場合はお知らせください。

package main

import (
    "github.com/fogleman/gg"
    "log"
)

func main() {
    const S = 1024
    im, err := gg.LoadImage("src.jpg")
    if err != nil {
        log.Fatal(err)
    }

    dc := gg.NewContext(S, S)
    dc.SetRGB(1, 1, 1)
    dc.Clear()
    dc.SetRGB(0, 0, 0)
    if err := dc.LoadFontFace("/Library/Fonts/Arial.ttf", 96); err != nil {
        panic(err)
    }
    dc.DrawStringAnchored("Hello, world!", S/2, S/2, 0.5, 0.5)

    dc.DrawRoundedRectangle(0, 0, 512, 512, 0)
    dc.DrawImage(im, 0, 0)
    dc.DrawStringAnchored("Hello, world!", S/2, S/2, 0.5, 0.5)
    dc.Clip()
    dc.SavePNG("out.png")
}
7
Yatender Singh