2022-10-31 13:51:28 -06:00
|
|
|
package stone
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Color represents all the different colors a cell can be.
|
2022-10-31 13:51:28 -06:00
|
|
|
type Color uint8
|
|
|
|
|
|
|
|
const (
|
|
|
|
ColorBackground Color = 0x0
|
|
|
|
ColorSelection Color = 0x1
|
|
|
|
ColorForeground Color = 0x2
|
|
|
|
ColorApplication Color = 0x3
|
|
|
|
)
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Style contains styling information about cells. These properties can be or'd
|
|
|
|
// together to combine them.
|
2022-10-31 13:51:28 -06:00
|
|
|
type Style uint8
|
|
|
|
|
|
|
|
const (
|
|
|
|
StyleNormal Style = iota
|
|
|
|
StyleBold Style = iota >> 1
|
|
|
|
StyleItalic
|
|
|
|
StyleBoldItalic Style = StyleBold | StyleItalic
|
|
|
|
)
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Cell is a grid-aligned rune in a buffer with associated styling and color
|
|
|
|
// informaiton.
|
2022-10-31 13:51:28 -06:00
|
|
|
type Cell struct {
|
|
|
|
color Color
|
|
|
|
style Style
|
|
|
|
content rune
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Color returns the cell's color.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (cell Cell) Color (color Color) {
|
|
|
|
color = cell.color
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Style returns the styling information associated with the cell
|
2022-10-31 13:51:28 -06:00
|
|
|
func (cell Cell) Style (style Style) {
|
|
|
|
style = cell.style
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Rune returns the rune in the cell
|
2022-11-02 16:51:33 -06:00
|
|
|
func (cell Cell) Rune () (content rune) {
|
2022-10-31 13:51:28 -06:00
|
|
|
content = cell.content
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Buffer is a basic grid of cells.
|
2022-10-31 13:51:28 -06:00
|
|
|
type Buffer struct {
|
|
|
|
content []Cell
|
|
|
|
width int
|
|
|
|
height int
|
2022-11-06 12:17:43 -07:00
|
|
|
Dot struct {
|
|
|
|
X int
|
|
|
|
Y int
|
|
|
|
}
|
2022-10-31 13:51:28 -06:00
|
|
|
}
|
|
|
|
|
2022-11-06 13:25:55 -07:00
|
|
|
func (buffer *Buffer) isOutOfBounds (x, y int) (outOfBounds bool) {
|
|
|
|
outOfBounds =
|
|
|
|
x < 0 ||
|
|
|
|
y < 0 ||
|
|
|
|
x >= buffer.width ||
|
|
|
|
y >= buffer.height
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Size returns the width and height of the buffer.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *Buffer) Size () (width, height int) {
|
|
|
|
width = buffer.width
|
|
|
|
height = buffer.height
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetSize sets the width and height of the buffer. This clears all data in the
|
|
|
|
// buffer. If the width or height is negative, this method does nothing.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *Buffer) SetSize (width, height int) {
|
2022-11-06 13:59:06 -07:00
|
|
|
if width < 0 || height < 0 { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.width = width
|
|
|
|
buffer.height = height
|
|
|
|
buffer.content = make([]Cell, width * height)
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Cell returns the cell at the specified x and y coordinates. If the
|
|
|
|
// coordinates are out of bounds, this method will return a blank cell.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *Buffer) Cell (x, y int) (cell Cell) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
cell = buffer.content[x + y * buffer.width]
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetColor sets the color of the cell at the specified x and y coordinates.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *Buffer) SetColor (x, y int, color Color) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.content[x + y * buffer.width].color = color
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetStyle sets the style of the cell at the specified x and y coordinates.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *Buffer) SetStyle (x, y int, style Style) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.content[x + y * buffer.width].style = style
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetRune sets the rune of the cell at the specified x and y coordinates.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *Buffer) SetRune (x, y int, content rune) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.content[x + y * buffer.width].content = content
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Write writes data stored in a byte slice to the buffer at the current dot
|
|
|
|
// position. This makes Buffer an io.Writer.
|
2022-11-06 12:17:43 -07:00
|
|
|
func (buffer *Buffer) Write (bytes []byte) (bytesWritten int, err error) {
|
|
|
|
text := string(bytes)
|
|
|
|
bytesWritten = len(bytes)
|
|
|
|
|
|
|
|
for _, character := range text {
|
|
|
|
buffer.SetRune(buffer.Dot.X, buffer.Dot.Y, character)
|
|
|
|
buffer.Dot.X ++
|
2022-11-06 13:26:50 -07:00
|
|
|
if buffer.Dot.X > buffer.width { break }
|
2022-11-06 12:17:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// ResetDot is a convenience method to reset the dot to the buffer origin point
|
|
|
|
// (0, 0).
|
2022-11-06 12:17:43 -07:00
|
|
|
func (buffer *Buffer) ResetDot () {
|
|
|
|
buffer.Dot.X = 0
|
|
|
|
buffer.Dot.Y = 0
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// DamageBuffer is a special buffer that keeps track of damage information.
|
|
|
|
// Cells are dirty by default, are only clean when marked as clean, and become
|
|
|
|
// dirty again when they are altered in some way.
|
2022-10-31 13:51:28 -06:00
|
|
|
type DamageBuffer struct {
|
|
|
|
Buffer
|
|
|
|
clean []bool
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetSize sets the width and height of the buffer. This clears all data in the
|
|
|
|
// buffer. If the width or height is negative, this method does nothing.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *DamageBuffer) SetSize (width, height int) {
|
2022-11-06 13:59:06 -07:00
|
|
|
if width < 0 || height < 0 { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.Buffer.SetSize(width, height)
|
|
|
|
buffer.clean = make([]bool, width * height)
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetColor sets the color of the cell at the specified x and y coordinates.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *DamageBuffer) SetColor (x, y int, color Color) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
|
|
|
index := x + y * buffer.width
|
|
|
|
buffer.clean[index] = buffer.content[index].color == color
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.Buffer.SetColor(x, y, color)
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetStyle sets the style of the cell at the specified x and y coordinates.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *DamageBuffer) SetStyle (x, y int, style Style) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
|
|
|
index := x + y * buffer.width
|
|
|
|
buffer.clean[index] = buffer.content[index].style == style
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.Buffer.SetStyle(x, y, style)
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// SetRune sets the rune of the cell at the specified x and y coordinates.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *DamageBuffer) SetRune (x, y int, content rune) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
|
|
|
index := x + y * buffer.width
|
|
|
|
buffer.clean[index] = buffer.content[index].content == content
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.Buffer.SetRune(x, y, content)
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Write writes data stored in a byte slice to the buffer at the current dot
|
|
|
|
// position. This makes DamageBuffer an io.Writer.
|
2022-11-06 12:17:43 -07:00
|
|
|
func (buffer *DamageBuffer) Write (bytes []byte) (bytesWritten int, err error) {
|
|
|
|
text := string(bytes)
|
|
|
|
bytesWritten = len(bytes)
|
|
|
|
|
|
|
|
for _, character := range text {
|
|
|
|
buffer.SetRune(buffer.Dot.X, buffer.Dot.Y, character)
|
|
|
|
buffer.Dot.X ++
|
2022-11-06 13:26:50 -07:00
|
|
|
if buffer.Dot.X > buffer.width { break }
|
2022-11-06 12:17:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// Clean returns whether or not the cell at the specified x and y coordinates is
|
|
|
|
// clean.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *DamageBuffer) Clean (x, y int) (clean bool) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
clean = buffer.clean[x + y * buffer.width]
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-11-14 22:22:01 -07:00
|
|
|
// MarkClean marks the cell at the specified x and y coordinates as clean.
|
2022-10-31 13:51:28 -06:00
|
|
|
func (buffer *DamageBuffer) MarkClean (x, y int) {
|
2022-11-06 13:25:55 -07:00
|
|
|
if buffer.isOutOfBounds(x, y) { return }
|
2022-10-31 13:51:28 -06:00
|
|
|
buffer.clean[x + y * buffer.width] = true
|
|
|
|
}
|