Implement basic text drawing in svg context
fonts, bounds and string paths not yet implemented
This commit is contained in:
parent
215a761ccb
commit
90f962641f
3 changed files with 109 additions and 83 deletions
156
draw2dsvg/gc.go
156
draw2dsvg/gc.go
|
@ -46,9 +46,6 @@ func NewGraphicContext(svg *Svg) *GraphicContext {
|
||||||
// Clear fills the current canvas with a default transparent color
|
// Clear fills the current canvas with a default transparent color
|
||||||
func (gc *GraphicContext) Clear() {
|
func (gc *GraphicContext) Clear() {
|
||||||
gc.svg.Groups = nil
|
gc.svg.Groups = nil
|
||||||
gc.svg.Groups = append(gc.svg.Groups, Group{
|
|
||||||
// TODO add background color?
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stroke strokes the paths with the color specified by SetStrokeColor
|
// Stroke strokes the paths with the color specified by SetStrokeColor
|
||||||
|
@ -69,47 +66,102 @@ func (gc *GraphicContext) FillStroke(paths ...*draw2d.Path) {
|
||||||
gc.Current.Path.Clear()
|
gc.Current.Path.Clear()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FillString draws the text at point (0, 0)
|
||||||
|
func (gc *GraphicContext) FillString(text string) (cursor float64) {
|
||||||
|
return gc.FillStringAt(text, 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FillStringAt draws the text at the specified point (x, y)
|
||||||
|
func (gc *GraphicContext) FillStringAt(text string, x, y float64) (cursor float64) {
|
||||||
|
return gc.drawString(text, filled, x, y)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StrokeString draws the contour of the text at point (0, 0)
|
||||||
|
func (gc *GraphicContext) StrokeString(text string) (cursor float64) {
|
||||||
|
return gc.StrokeStringAt(text, 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StrokeStringAt draws the contour of the text at point (x, y)
|
||||||
|
func (gc *GraphicContext) StrokeStringAt(text string, x, y float64) (cursor float64) {
|
||||||
|
return gc.drawString(text, stroked, x, y)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save the context and push it to the context stack
|
||||||
|
func (gc *GraphicContext) Save() {
|
||||||
|
gc.StackGraphicContext.Save()
|
||||||
|
// TODO use common transformation group for multiple elements
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restore remove the current context and restore the last one
|
||||||
|
func (gc *GraphicContext) Restore() {
|
||||||
|
gc.StackGraphicContext.Restore()
|
||||||
|
// TODO use common transformation group for multiple elements
|
||||||
|
}
|
||||||
|
|
||||||
|
// private funcitons
|
||||||
|
|
||||||
func (gc *GraphicContext) drawPaths(drawType drawType, paths ...*draw2d.Path) {
|
func (gc *GraphicContext) drawPaths(drawType drawType, paths ...*draw2d.Path) {
|
||||||
// create elements
|
// create elements
|
||||||
svgPath := Path{}
|
svgPath := Path{}
|
||||||
|
group := gc.newGroup(drawType)
|
||||||
|
|
||||||
|
// set attrs to path element
|
||||||
|
paths = append(paths, gc.Current.Path)
|
||||||
|
svgPathsDesc := make([]string, len(paths))
|
||||||
|
// multiple pathes has to be joined to single svg path description
|
||||||
|
// because fill-rule wont work for whole group as excepted
|
||||||
|
for i, path := range paths {
|
||||||
|
svgPathsDesc[i] = toSvgPathDesc(path)
|
||||||
|
}
|
||||||
|
svgPath.Desc = strings.Join(svgPathsDesc, " ")
|
||||||
|
|
||||||
|
// link to group
|
||||||
|
group.Paths = []*Path{&svgPath}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gc *GraphicContext) drawString(text string, drawType drawType, x, y float64) float64 {
|
||||||
|
// create elements
|
||||||
|
svgText := Text{}
|
||||||
|
group := gc.newGroup(drawType)
|
||||||
|
|
||||||
|
// set attrs to text element
|
||||||
|
svgText.Text = text
|
||||||
|
svgText.X = x
|
||||||
|
svgText.Y = y
|
||||||
|
// TODO set font
|
||||||
|
|
||||||
|
// link to group
|
||||||
|
(*group).Texts = []*Text{&svgText}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Creates new group from current context
|
||||||
|
// append it to svg and return
|
||||||
|
func (gc *GraphicContext) newGroup(drawType drawType) *Group {
|
||||||
group := Group{}
|
group := Group{}
|
||||||
|
|
||||||
// set attrs to path
|
|
||||||
{
|
|
||||||
paths = append(paths, gc.Current.Path)
|
|
||||||
svgPathsDesc := make([]string, len(paths))
|
|
||||||
// multiple pathes has to be joined to single svg path description
|
|
||||||
// because fill-rule wont work for whole group as excepted
|
|
||||||
for i, path := range paths {
|
|
||||||
svgPathsDesc[i] = toSvgPathDesc(path)
|
|
||||||
}
|
|
||||||
svgPath.Desc = strings.Join(svgPathsDesc, " ")
|
|
||||||
}
|
|
||||||
|
|
||||||
// set attrs to group
|
// set attrs to group
|
||||||
{
|
if drawType&stroked == stroked {
|
||||||
if drawType&stroked == stroked {
|
group.Stroke = toSvgRGBA(gc.Current.StrokeColor)
|
||||||
group.Stroke = toSvgRGBA(gc.Current.StrokeColor)
|
group.StrokeWidth = toSvgLength(gc.Current.LineWidth)
|
||||||
group.StrokeWidth = toSvgLength(gc.Current.LineWidth)
|
group.StrokeLinecap = gc.Current.Cap.String()
|
||||||
group.StrokeLinecap = gc.Current.Cap.String()
|
group.StrokeLinejoin = gc.Current.Join.String()
|
||||||
group.StrokeLinejoin = gc.Current.Join.String()
|
if len(gc.Current.Dash) > 0 {
|
||||||
if len(gc.Current.Dash) > 0 {
|
group.StrokeDasharray = toSvgArray(gc.Current.Dash)
|
||||||
group.StrokeDasharray = toSvgArray(gc.Current.Dash)
|
group.StrokeDashoffset = toSvgLength(gc.Current.DashOffset)
|
||||||
group.StrokeDashoffset = toSvgLength(gc.Current.DashOffset)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if drawType&filled == filled {
|
|
||||||
group.Fill = toSvgRGBA(gc.Current.FillColor)
|
|
||||||
group.FillRule = toSvgFillRule(gc.Current.FillRule)
|
|
||||||
}
|
|
||||||
|
|
||||||
group.Transform = toSvgTransform(gc.Current.Tr)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// link elements
|
if drawType&filled == filled {
|
||||||
group.Paths = []Path{svgPath}
|
group.Fill = toSvgRGBA(gc.Current.FillColor)
|
||||||
gc.svg.Groups = append(gc.svg.Groups, group)
|
group.FillRule = toSvgFillRule(gc.Current.FillRule)
|
||||||
|
}
|
||||||
|
|
||||||
|
group.Transform = toSvgTransform(gc.Current.Tr)
|
||||||
|
|
||||||
|
// link
|
||||||
|
gc.svg.Groups = append(gc.svg.Groups, &group)
|
||||||
|
|
||||||
|
return &group
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////
|
///////////////////////////////////////
|
||||||
|
@ -135,18 +187,6 @@ func (gc *GraphicContext) DrawImage(image image.Image) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Save the context and push it to the context stack
|
|
||||||
func (gc *GraphicContext) Save() {
|
|
||||||
gc.StackGraphicContext.Save()
|
|
||||||
// TODO use common transformation group for multiple elements
|
|
||||||
}
|
|
||||||
|
|
||||||
// Restore remove the current context and restore the last one
|
|
||||||
func (gc *GraphicContext) Restore() {
|
|
||||||
gc.StackGraphicContext.Restore()
|
|
||||||
// TODO use common transformation group for multiple elements
|
|
||||||
}
|
|
||||||
|
|
||||||
// ClearRect fills the specified rectangle with a default transparent color
|
// ClearRect fills the specified rectangle with a default transparent color
|
||||||
func (gc *GraphicContext) ClearRect(x1, y1, x2, y2 int) {
|
func (gc *GraphicContext) ClearRect(x1, y1, x2, y2 int) {
|
||||||
|
|
||||||
|
@ -171,23 +211,3 @@ func (gc *GraphicContext) GetStringBounds(s string) (left, top, right, bottom fl
|
||||||
func (gc *GraphicContext) CreateStringPath(text string, x, y float64) (cursor float64) {
|
func (gc *GraphicContext) CreateStringPath(text string, x, y float64) (cursor float64) {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
// FillString draws the text at point (0, 0)
|
|
||||||
func (gc *GraphicContext) FillString(text string) (cursor float64) {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// FillStringAt draws the text at the specified point (x, y)
|
|
||||||
func (gc *GraphicContext) FillStringAt(text string, x, y float64) (cursor float64) {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// StrokeString draws the contour of the text at point (0, 0)
|
|
||||||
func (gc *GraphicContext) StrokeString(text string) (cursor float64) {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// StrokeStringAt draws the contour of the text at point (x, y)
|
|
||||||
func (gc *GraphicContext) StrokeStringAt(text string, x, y float64) (cursor float64) {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
|
@ -12,16 +12,16 @@ import (
|
||||||
type Svg struct {
|
type Svg struct {
|
||||||
XMLName xml.Name `xml:"svg"`
|
XMLName xml.Name `xml:"svg"`
|
||||||
Xmlns string `xml:"xmlns,attr"`
|
Xmlns string `xml:"xmlns,attr"`
|
||||||
Groups []Group `xml:"g"`
|
Groups []*Group `xml:"g"`
|
||||||
FillStroke
|
FillStroke
|
||||||
}
|
}
|
||||||
|
|
||||||
type Group struct {
|
type Group struct {
|
||||||
FillStroke
|
FillStroke
|
||||||
Transform string `xml:"transform,attr,omitempty"`
|
Transform string `xml:"transform,attr,omitempty"`
|
||||||
Groups []Group `xml:"g"`
|
Groups []*Group `xml:"g"`
|
||||||
Paths []Path `xml:"path"`
|
Paths []*Path `xml:"path"`
|
||||||
Texts []Text `xml:"text"`
|
Texts []*Text `xml:"text"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Path struct {
|
type Path struct {
|
||||||
|
@ -31,12 +31,18 @@ type Path struct {
|
||||||
|
|
||||||
type Text struct {
|
type Text struct {
|
||||||
FillStroke
|
FillStroke
|
||||||
|
Position
|
||||||
Text string `xml:",innerxml"`
|
Text string `xml:",innerxml"`
|
||||||
Style string `xml:"style,attr,omitempty"`
|
Style string `xml:"style,attr,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
/* shared attrs */
|
/* shared attrs */
|
||||||
|
|
||||||
|
type Position struct {
|
||||||
|
X float64 `xml:"x,attr,omitempty"`
|
||||||
|
Y float64 `xml:"y,attr,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
type FillStroke struct {
|
type FillStroke struct {
|
||||||
Fill string `xml:"fill,attr,omitempty"`
|
Fill string `xml:"fill,attr,omitempty"`
|
||||||
FillRule string `xml:"fill-rule,attr,omitempty"`
|
FillRule string `xml:"fill-rule,attr,omitempty"`
|
||||||
|
|
|
@ -15,18 +15,18 @@ import (
|
||||||
func TestXml(t *testing.T) {
|
func TestXml(t *testing.T) {
|
||||||
|
|
||||||
svg := NewSvg()
|
svg := NewSvg()
|
||||||
svg.Groups = []Group{Group{
|
svg.Groups = []*Group{&Group{
|
||||||
Groups: []Group{
|
Groups: []*Group{
|
||||||
Group{}, // nested groups
|
&Group{}, // nested groups
|
||||||
Group{},
|
&Group{},
|
||||||
},
|
},
|
||||||
Texts: []Text{
|
Texts: []*Text{
|
||||||
Text{Text: "Hello"}, // text
|
&Text{Text: "Hello"}, // text
|
||||||
Text{Text: "world", Style: "opacity: 0.5"}, // text with style
|
&Text{Text: "world", Style: "opacity: 0.5"}, // text with style
|
||||||
},
|
},
|
||||||
Paths: []Path{
|
Paths: []*Path{
|
||||||
Path{Desc: "M100,200 C100,100 250,100 250,200 S400,300 400,200"}, // simple path
|
&Path{Desc: "M100,200 C100,100 250,100 250,200 S400,300 400,200"}, // simple path
|
||||||
Path{}, // empty path
|
&Path{}, // empty path
|
||||||
},
|
},
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue