2018-04-15 06:06:57 +08:00
|
|
|
// Copyright 2018 Google Inc.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2018-04-06 10:56:36 +08:00
|
|
|
package container
|
|
|
|
|
|
|
|
// draw.go contains logic to draw containers and the contained widgets.
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2018-04-09 10:17:40 +08:00
|
|
|
"fmt"
|
|
|
|
"image"
|
2018-04-06 10:56:36 +08:00
|
|
|
|
2018-05-07 19:40:40 +08:00
|
|
|
"github.com/mum4k/termdash/align"
|
2018-04-06 10:56:36 +08:00
|
|
|
"github.com/mum4k/termdash/area"
|
|
|
|
"github.com/mum4k/termdash/canvas"
|
|
|
|
"github.com/mum4k/termdash/cell"
|
|
|
|
"github.com/mum4k/termdash/draw"
|
|
|
|
)
|
|
|
|
|
|
|
|
// drawTree draws this container and all of its sub containers.
|
|
|
|
func drawTree(c *Container) error {
|
|
|
|
var errStr string
|
2018-04-23 07:45:17 +08:00
|
|
|
|
|
|
|
root := rootCont(c)
|
|
|
|
size := root.term.Size()
|
|
|
|
root.area = image.Rect(0, 0, size.X, size.Y)
|
|
|
|
|
|
|
|
preOrder(root, &errStr, visitFunc(func(c *Container) error {
|
|
|
|
first, second := c.split()
|
|
|
|
if c.first != nil {
|
|
|
|
c.first.area = first
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.second != nil {
|
|
|
|
c.second.area = second
|
|
|
|
}
|
2018-04-06 10:56:36 +08:00
|
|
|
return drawCont(c)
|
|
|
|
}))
|
|
|
|
if errStr != "" {
|
|
|
|
return errors.New(errStr)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-04-09 10:17:40 +08:00
|
|
|
// drawBorder draws the border around the container if requested.
|
|
|
|
func drawBorder(c *Container) error {
|
|
|
|
if !c.hasBorder() {
|
2018-04-06 10:56:36 +08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cvs, err := canvas.New(c.area)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-04-09 10:17:40 +08:00
|
|
|
ar, err := area.FromSize(cvs.Size())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-04-06 10:56:36 +08:00
|
|
|
|
2018-05-07 20:33:18 +08:00
|
|
|
var cOpts []cell.Option
|
2018-04-09 10:17:40 +08:00
|
|
|
if c.focusTracker.isActive(c) {
|
2018-05-07 20:33:18 +08:00
|
|
|
cOpts = append(cOpts, cell.FgColor(c.opts.inherited.focusedColor))
|
2018-04-09 10:17:40 +08:00
|
|
|
} else {
|
2018-05-07 20:33:18 +08:00
|
|
|
cOpts = append(cOpts, cell.FgColor(c.opts.inherited.borderColor))
|
2018-04-09 10:17:40 +08:00
|
|
|
}
|
2018-05-07 20:33:18 +08:00
|
|
|
|
|
|
|
if err := draw.Border(cvs, ar, draw.BorderLineStyle(c.opts.border), draw.BorderCellOpts(cOpts...)); err != nil {
|
2018-04-09 10:17:40 +08:00
|
|
|
return err
|
2018-04-06 10:56:36 +08:00
|
|
|
}
|
|
|
|
return cvs.Apply(c.term)
|
|
|
|
}
|
2018-04-09 10:17:40 +08:00
|
|
|
|
|
|
|
// hAlignWidget adjusts the given widget area within the containers area
|
|
|
|
// based on the requested horizontal alignment.
|
|
|
|
func hAlignWidget(c *Container, wArea image.Rectangle) image.Rectangle {
|
|
|
|
gap := c.usable().Dx() - wArea.Dx()
|
|
|
|
switch c.opts.hAlign {
|
2018-05-07 19:40:40 +08:00
|
|
|
case align.HorizontalRight:
|
2018-04-09 10:17:40 +08:00
|
|
|
// Use gap from above.
|
2018-05-07 19:40:40 +08:00
|
|
|
case align.HorizontalCenter:
|
2018-04-09 10:17:40 +08:00
|
|
|
gap /= 2
|
|
|
|
default:
|
|
|
|
// Left or unknown.
|
|
|
|
gap = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return image.Rect(
|
|
|
|
wArea.Min.X+gap,
|
|
|
|
wArea.Min.Y,
|
|
|
|
wArea.Max.X+gap,
|
|
|
|
wArea.Max.Y,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// vAlignWidget adjusts the given widget area within the containers area
|
|
|
|
// based on the requested vertical alignment.
|
|
|
|
func vAlignWidget(c *Container, wArea image.Rectangle) image.Rectangle {
|
|
|
|
gap := c.usable().Dy() - wArea.Dy()
|
|
|
|
switch c.opts.vAlign {
|
2018-05-07 19:40:40 +08:00
|
|
|
case align.VerticalBottom:
|
2018-04-09 10:17:40 +08:00
|
|
|
// Use gap from above.
|
2018-05-07 19:40:40 +08:00
|
|
|
case align.VerticalMiddle:
|
2018-04-09 10:17:40 +08:00
|
|
|
gap /= 2
|
|
|
|
default:
|
|
|
|
// Top or unknown.
|
|
|
|
gap = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
return image.Rect(
|
|
|
|
wArea.Min.X,
|
|
|
|
wArea.Min.Y+gap,
|
|
|
|
wArea.Max.X,
|
|
|
|
wArea.Max.Y+gap,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// drawWidget requests the widget to draw on the canvas.
|
|
|
|
func drawWidget(c *Container) error {
|
2018-04-15 05:00:25 +08:00
|
|
|
widgetArea := c.widgetArea()
|
|
|
|
if widgetArea == image.ZR {
|
2018-04-09 10:17:40 +08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-04-15 05:00:25 +08:00
|
|
|
if !c.hasWidget() {
|
|
|
|
return nil
|
2018-04-09 10:17:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
needSize := image.Point{1, 1}
|
2018-04-15 05:00:25 +08:00
|
|
|
wOpts := c.opts.widget.Options()
|
2018-04-09 10:17:40 +08:00
|
|
|
if wOpts.MinimumSize.X > 0 && wOpts.MinimumSize.Y > 0 {
|
|
|
|
needSize = wOpts.MinimumSize
|
|
|
|
}
|
|
|
|
|
2018-04-15 05:00:25 +08:00
|
|
|
if widgetArea.Dx() < needSize.X || widgetArea.Dy() < needSize.Y {
|
2018-04-09 10:17:40 +08:00
|
|
|
return drawResize(c, c.usable())
|
|
|
|
}
|
|
|
|
|
2018-04-15 05:00:25 +08:00
|
|
|
cvs, err := canvas.New(widgetArea)
|
2018-04-09 10:17:40 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.opts.widget.Draw(cvs); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return cvs.Apply(c.term)
|
|
|
|
}
|
|
|
|
|
|
|
|
// drawResize draws an unicode character indicating that the size is too small to draw this container.
|
|
|
|
// Does nothing if the size is smaller than one cell, leaving no space for the character.
|
|
|
|
func drawResize(c *Container, area image.Rectangle) error {
|
|
|
|
if area.Dx() < 1 || area.Dy() < 1 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cvs, err := canvas.New(area)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-05-07 23:50:27 +08:00
|
|
|
if err := draw.Text(cvs, "⇄", image.Point{0, 0}); err != nil {
|
2018-04-09 10:17:40 +08:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return cvs.Apply(c.term)
|
|
|
|
}
|
|
|
|
|
|
|
|
// drawCont draws the container and its widget.
|
|
|
|
func drawCont(c *Container) error {
|
|
|
|
if us := c.usable(); us.Dx() <= 0 || us.Dy() <= 0 {
|
|
|
|
return drawResize(c, c.area)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := drawBorder(c); err != nil {
|
|
|
|
return fmt.Errorf("unable to draw container border: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := drawWidget(c); err != nil {
|
|
|
|
return fmt.Errorf("unable to draw widget %T: %v", c.opts.widget, err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|