Moved terminal stuff to a separate repository
This commit is contained in:
parent
43a664009c
commit
dc077a02ab
38
ansi/c0.go
38
ansi/c0.go
@ -1,38 +0,0 @@
|
|||||||
package ansi
|
|
||||||
|
|
||||||
// C0 represents a list of C0 control codes.
|
|
||||||
// https://en.wikipedia.org/wiki/C0_and_C1_control_codes
|
|
||||||
type C0 byte; const (
|
|
||||||
C0_Null C0 = iota
|
|
||||||
C0_StartOfHeading
|
|
||||||
C0_StartOfText
|
|
||||||
C0_EndOfText
|
|
||||||
C0_EndOfTransmission
|
|
||||||
C0_Enquiry
|
|
||||||
C0_Acknowledge
|
|
||||||
C0_Bell
|
|
||||||
C0_Backspace
|
|
||||||
C0_CharacterTab
|
|
||||||
C0_LineFeed
|
|
||||||
C0_LineTab
|
|
||||||
C0_FormFeed
|
|
||||||
C0_CarriageReturn
|
|
||||||
C0_ShiftOut
|
|
||||||
C0_ShiftIn
|
|
||||||
C0_DataLinkEscape
|
|
||||||
C0_DeviceControlOne
|
|
||||||
C0_DeviceControlTwo
|
|
||||||
C0_DeviceControlThree
|
|
||||||
C0_DeviceControlFour
|
|
||||||
C0_NegativeAcknowledge
|
|
||||||
C0_SynchronousIdle
|
|
||||||
C0_EndOfTransmissionBlock
|
|
||||||
C0_Cancel
|
|
||||||
C0_EndOfMedium
|
|
||||||
C0_Substitute
|
|
||||||
C0_Escape
|
|
||||||
C0_FileSeparator
|
|
||||||
C0_GroupSeparator
|
|
||||||
C0_RecordSeparator
|
|
||||||
C0_UnitSeparator
|
|
||||||
)
|
|
48
ansi/c1.go
48
ansi/c1.go
@ -1,48 +0,0 @@
|
|||||||
package ansi
|
|
||||||
|
|
||||||
// C1 represents a list of C1 control codes.
|
|
||||||
// https://en.wikipedia.org/wiki/C0_and_C1_control_codes
|
|
||||||
type C1 byte; const (
|
|
||||||
C1_PaddingCharacter C1 = iota + 128
|
|
||||||
C1_HighOctetPreset
|
|
||||||
C1_BreakPermittedHere
|
|
||||||
C1_NoBreakHere
|
|
||||||
C1_Index
|
|
||||||
C1_NextLine
|
|
||||||
C1_StartOfSelectedArea
|
|
||||||
C1_EndOfSelectedArea
|
|
||||||
C1_CharacterTabSet
|
|
||||||
C1_CharacterTabWithJustification
|
|
||||||
C1_LineTabSet
|
|
||||||
C1_PartialLineForward
|
|
||||||
C1_PartialLineBackward
|
|
||||||
C1_ReverseLineFeed
|
|
||||||
C1_SingleShift2
|
|
||||||
C1_SingleShift3
|
|
||||||
C1_DeviceControlString
|
|
||||||
C1_PrivateUse1
|
|
||||||
C1_PrivateUse2
|
|
||||||
C1_SetTransmitState
|
|
||||||
C1_CancelCharacter
|
|
||||||
C1_MessageWaiting
|
|
||||||
C1_StartOfProtectedArea
|
|
||||||
C1_EndOfProtectedArea
|
|
||||||
C1_StartOfString
|
|
||||||
C1_SingleGraphicCharacterIntroducer
|
|
||||||
C1_SingleCharacterIntroducer
|
|
||||||
C1_ControlSequenceIntroducer
|
|
||||||
C1_StringTerminator
|
|
||||||
C1_OperatingSystemCommand
|
|
||||||
C1_PrivacyMessage
|
|
||||||
C1_ApplicationProgramCommand
|
|
||||||
)
|
|
||||||
|
|
||||||
// Is checks if a byte is equal to a C0 code.
|
|
||||||
func (code C0) Is (test byte) bool {
|
|
||||||
return test == byte(code)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Is checks if a byte is equal to a C1 code.
|
|
||||||
func (code C1) Is (test byte) bool {
|
|
||||||
return byte(code) == test
|
|
||||||
}
|
|
@ -1,90 +0,0 @@
|
|||||||
package ansi
|
|
||||||
|
|
||||||
import "image/color"
|
|
||||||
|
|
||||||
var _ color.Color = Color(0)
|
|
||||||
|
|
||||||
// Color represents a 3, 4, or 8-Bit ansi color.
|
|
||||||
type Color byte; const (
|
|
||||||
// Dim/standard colors
|
|
||||||
ColorBlack Color = iota
|
|
||||||
ColorRed
|
|
||||||
ColorGreen
|
|
||||||
ColorYellow
|
|
||||||
ColorBlue
|
|
||||||
ColorMagenta
|
|
||||||
ColorCyan
|
|
||||||
ColorWhite
|
|
||||||
|
|
||||||
// Bright colors
|
|
||||||
ColorBrightBlack
|
|
||||||
ColorBrightRed
|
|
||||||
ColorBrightGreen
|
|
||||||
ColorBrightYellow
|
|
||||||
ColorBrightBlue
|
|
||||||
ColorBrightMagenta
|
|
||||||
ColorBrightCyan
|
|
||||||
ColorBrightWhite
|
|
||||||
|
|
||||||
// 216 cube colors (16 - 231)
|
|
||||||
// 24 grayscale colors (232 - 255)
|
|
||||||
)
|
|
||||||
|
|
||||||
// Is16 returns whether the color is a dim or bright color, and can be assigned
|
|
||||||
// to a theme palette.
|
|
||||||
func (c Color) Is16 () bool {
|
|
||||||
return c.IsDim() || c.IsBright()
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsDim returns whether the color is dim.
|
|
||||||
func (c Color) IsDim () bool {
|
|
||||||
return c < 8
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsBright returns whether the color is bright.
|
|
||||||
func (c Color) IsBright () bool {
|
|
||||||
return c >= 8 && c < 16
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsCube returns whether the color is part of the 6x6x6 cube.
|
|
||||||
func (c Color) IsCube () bool {
|
|
||||||
return c >= 16 && c < 232
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsGrayscale returns whether the color grayscale.
|
|
||||||
func (c Color) IsGrayscale () bool {
|
|
||||||
return c >= 232 && c <= 255
|
|
||||||
}
|
|
||||||
|
|
||||||
// RGB returns the 8 bit RGB values of the color as a color.RGBA value.
|
|
||||||
func (c Color) RGB () (out color.RGBA) {
|
|
||||||
switch {
|
|
||||||
case c.Is16():
|
|
||||||
// each bit is a color component
|
|
||||||
out.R = 0xFF * uint8((c & 0x1) >> 0)
|
|
||||||
out.G = 0xFF * uint8((c & 0x2) >> 1)
|
|
||||||
out.B = 0xFF * uint8((c & 0x4) >> 3)
|
|
||||||
// dim if color is in the dim range
|
|
||||||
if c & 0x8 > 0 { out.R >>= 1; out.G >>= 1; out.B >>= 1 }
|
|
||||||
|
|
||||||
case c.IsCube():
|
|
||||||
index := int(c - 16)
|
|
||||||
out.R = uint8((((index / 36) % 6) * 255) / 5)
|
|
||||||
out.G = uint8((((index / 6) % 6) * 255) / 5)
|
|
||||||
out.B = uint8((((index ) % 6) * 255) / 5)
|
|
||||||
|
|
||||||
case c.IsGrayscale():
|
|
||||||
out.R = uint8(((int(c) - 232) * 255) / 23)
|
|
||||||
out.G = out.R
|
|
||||||
out.B = out.R
|
|
||||||
}
|
|
||||||
|
|
||||||
out.A = 0xFF
|
|
||||||
return
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// RGBA fulfills the color.Color interface.
|
|
||||||
func (c Color) RGBA () (r, g, b, a uint32) {
|
|
||||||
return c.RGB().RGBA()
|
|
||||||
}
|
|
17
ansi/csi.go
17
ansi/csi.go
@ -1,17 +0,0 @@
|
|||||||
package ansi
|
|
||||||
|
|
||||||
// CSI represents a list of CSI sequences that have no parameters.
|
|
||||||
// FIXME: some of these do indeed have parameters
|
|
||||||
type CSI int; const (
|
|
||||||
CSI_DeviceStatusReport CSI = iota
|
|
||||||
CSI_SaveCursorPosition
|
|
||||||
CSI_RestoreCursorPosition
|
|
||||||
CSI_ShowCursor
|
|
||||||
CSI_HideCursor
|
|
||||||
CSI_EnableReportingFocus
|
|
||||||
CSI_DisableReportingFocus
|
|
||||||
CSI_EnableAlternativeBuffer
|
|
||||||
CSI_DisableAlternativeBuffer
|
|
||||||
CSI_EnableBracketedPasteMode
|
|
||||||
CSI_DisableBracketedPasteMode
|
|
||||||
)
|
|
360
ansi/decoder.go
360
ansi/decoder.go
@ -1,360 +0,0 @@
|
|||||||
package ansi
|
|
||||||
|
|
||||||
import "strconv"
|
|
||||||
import "strings"
|
|
||||||
import "image/color"
|
|
||||||
|
|
||||||
// Useful: https://en.wikipedia.org/wiki/ANSI_escape_code
|
|
||||||
|
|
||||||
// Decoder is a state machine capable of decoding text contianing various escape
|
|
||||||
// codes and sequences. It satisfies io.Writer. It has no constructor and its
|
|
||||||
// zero value can be used safely.
|
|
||||||
type Decoder struct {
|
|
||||||
// OnText is called when a segment of text is processed.
|
|
||||||
OnText func (string)
|
|
||||||
|
|
||||||
// OnC0 is called when a C0 control code is processed that isn't a
|
|
||||||
// whitespace character.
|
|
||||||
OnC0 func (C0)
|
|
||||||
|
|
||||||
// OnC1 is called when a C1 escape sequence is processed.
|
|
||||||
OnC1 func (C1)
|
|
||||||
|
|
||||||
// OnDCS is called when a device control string is processed.
|
|
||||||
OnDCS func (string)
|
|
||||||
|
|
||||||
// OnCSI is called when a non-SGR CSI escape sequence with no parameters
|
|
||||||
// is processed.
|
|
||||||
OnCSI func (CSI)
|
|
||||||
|
|
||||||
// OnSGR is called when a CSI SGR escape sequence with no parameters is
|
|
||||||
// processed.
|
|
||||||
OnSGR func (SGR)
|
|
||||||
|
|
||||||
// OnPM is called when a privacy message is processed.
|
|
||||||
OnPM func (string)
|
|
||||||
|
|
||||||
// OnAPC is called when an application program command is processed.
|
|
||||||
OnAPC func (string)
|
|
||||||
|
|
||||||
// Non-SGR CSI sequences with parameters:
|
|
||||||
OnCursorUp func (distance int)
|
|
||||||
OnCursorDown func (distance int)
|
|
||||||
OnCursorForward func (distance int)
|
|
||||||
OnCursorBack func (distance int)
|
|
||||||
OnCursorNextLine func (distance int)
|
|
||||||
OnCursorPreviousLine func (distance int)
|
|
||||||
OnCursorHorizontalAbsolute func (column int)
|
|
||||||
OnCursorPosition func (column, row int)
|
|
||||||
OnEraseInDisplay func (mode int)
|
|
||||||
OnEraseInLine func (mode int)
|
|
||||||
OnScrollUp func (distance int)
|
|
||||||
OnScrollDown func (distance int)
|
|
||||||
OnHorizontalVerticalPosition func (column, row int)
|
|
||||||
|
|
||||||
// SGR CSI sequences with parameters:
|
|
||||||
OnForegroundColor func (Color)
|
|
||||||
OnForegroundColorTrue func (color.RGBA)
|
|
||||||
OnBackgroundColor func (Color)
|
|
||||||
OnBackgroundColorTrue func (color.RGBA)
|
|
||||||
OnUnderlineColor func (Color)
|
|
||||||
OnUnderlineColorTrue func (color.RGBA)
|
|
||||||
|
|
||||||
// OSC sequences from XTerm:
|
|
||||||
OnWindowTitle func (title string)
|
|
||||||
OnIconName func (name string)
|
|
||||||
OnIconFile func (path string)
|
|
||||||
OnXProperty func (property, value string)
|
|
||||||
OnSelectionPut func (selection, text string)
|
|
||||||
OnSelectionGet func (selection string)
|
|
||||||
OnQueryAllowed func ()
|
|
||||||
OnQueryDisallowed func ()
|
|
||||||
|
|
||||||
// OSC sequences from iTerm2:
|
|
||||||
OnCursorShape func (shape int)
|
|
||||||
OnHyperlink func (params map[string] string, text, link string)
|
|
||||||
OnBackgroundImage func (path string)
|
|
||||||
|
|
||||||
state decodeState
|
|
||||||
expectingST bool
|
|
||||||
gathered []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
type decodeState int; const (
|
|
||||||
decodeStateText decodeState = iota
|
|
||||||
decodeStateAwaitC1
|
|
||||||
|
|
||||||
decodeStateGatherDCS
|
|
||||||
decodeStateGatherSOS
|
|
||||||
decodeStateGatherCSI
|
|
||||||
decodeStateGatherOSC
|
|
||||||
decodeStateGatherPM
|
|
||||||
decodeStateGatherAPC
|
|
||||||
)
|
|
||||||
|
|
||||||
func (decoder *Decoder) Write (buffer []byte) (wrote int, err error) {
|
|
||||||
wrote = len(buffer)
|
|
||||||
|
|
||||||
for len(buffer) > 0 {
|
|
||||||
switch decoder.state {
|
|
||||||
case decodeStateText:
|
|
||||||
if C0_Escape.Is(buffer[0]) {
|
|
||||||
// begin C1 control code
|
|
||||||
decoder.state = decodeStateAwaitC1
|
|
||||||
buffer = buffer[1:]
|
|
||||||
|
|
||||||
} else if buffer[0] < ' ' {
|
|
||||||
// process C0 control code
|
|
||||||
if decoder.OnC0 != nil {
|
|
||||||
decoder.OnC0(C0(buffer[0]))
|
|
||||||
}
|
|
||||||
buffer = buffer[1:]
|
|
||||||
|
|
||||||
} else {
|
|
||||||
// process as much plain text as we can
|
|
||||||
buffer = decoder.processString(buffer)
|
|
||||||
}
|
|
||||||
|
|
||||||
case decodeStateAwaitC1:
|
|
||||||
if buffer[0] < 128 {
|
|
||||||
// false alarm, this is just a C0 escape
|
|
||||||
if decoder.OnC0 != nil {
|
|
||||||
decoder.OnC0(C0_Escape)
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
switch C1(buffer[0]) {
|
|
||||||
case C1_DeviceControlString:
|
|
||||||
decoder.state = decodeStateGatherDCS
|
|
||||||
case C1_StartOfString:
|
|
||||||
decoder.state = decodeStateGatherSOS
|
|
||||||
case C1_ControlSequenceIntroducer:
|
|
||||||
decoder.state = decodeStateGatherCSI
|
|
||||||
case C1_OperatingSystemCommand:
|
|
||||||
decoder.state = decodeStateGatherOSC
|
|
||||||
case C1_PrivacyMessage:
|
|
||||||
decoder.state = decodeStateGatherPM
|
|
||||||
case C1_ApplicationProgramCommand:
|
|
||||||
decoder.state = decodeStateGatherAPC
|
|
||||||
default:
|
|
||||||
// process C1 control code
|
|
||||||
if decoder.OnC1 != nil {
|
|
||||||
decoder.OnC1(C1(buffer[0]))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
buffer = buffer[1:]
|
|
||||||
|
|
||||||
case
|
|
||||||
decodeStateGatherDCS,
|
|
||||||
decodeStateGatherSOS,
|
|
||||||
decodeStateGatherOSC,
|
|
||||||
decodeStateGatherPM,
|
|
||||||
decodeStateGatherAPC:
|
|
||||||
|
|
||||||
if decoder.expectingST && C1_StringTerminator.Is(buffer[0]) {
|
|
||||||
// remove the trailing ESC
|
|
||||||
decoder.gathered = decoder.gathered [
|
|
||||||
:len(decoder.gathered) - 1]
|
|
||||||
|
|
||||||
if decoder.state == decodeStateGatherOSC {
|
|
||||||
// we understand some OSC codes so we
|
|
||||||
// handle them differently
|
|
||||||
decoder.processOSC()
|
|
||||||
} else {
|
|
||||||
// generic handler for uncommon stuff
|
|
||||||
decoder.processGeneric()
|
|
||||||
}
|
|
||||||
decoder.state = decodeStateText
|
|
||||||
}
|
|
||||||
if C0_Escape.Is(buffer[0]) {
|
|
||||||
decoder.expectingST = true
|
|
||||||
}
|
|
||||||
|
|
||||||
case decodeStateGatherCSI:
|
|
||||||
decoder.gather(buffer[0])
|
|
||||||
if buffer[0] < 0x30 || buffer[0] > 0x3F {
|
|
||||||
decoder.processCSI()
|
|
||||||
decoder.state = decodeStateText
|
|
||||||
}
|
|
||||||
buffer = buffer[1:]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (decoder *Decoder) gather (character byte) {
|
|
||||||
decoder.gathered = append(decoder.gathered, character)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (decoder *Decoder) processString (buffer []byte) []byte {
|
|
||||||
for index, char := range buffer {
|
|
||||||
if C0_Escape.Is(char) {
|
|
||||||
if decoder.OnText != nil {
|
|
||||||
decoder.OnText(string(buffer[:index]))
|
|
||||||
}
|
|
||||||
return buffer[:index]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return buffer
|
|
||||||
}
|
|
||||||
|
|
||||||
func (decoder *Decoder) processGeneric () {
|
|
||||||
parameter := string(decoder.gathered)
|
|
||||||
switch decoder.state {
|
|
||||||
case decodeStateGatherDCS:
|
|
||||||
if decoder.OnDCS != nil { decoder.OnDCS(parameter) }
|
|
||||||
case decodeStateGatherSOS:
|
|
||||||
if decoder.OnText != nil { decoder.OnText(parameter) }
|
|
||||||
case decodeStateGatherPM:
|
|
||||||
if decoder.OnPM != nil { decoder.OnPM(parameter) }
|
|
||||||
case decodeStateGatherAPC:
|
|
||||||
if decoder.OnAPC != nil { decoder.OnAPC(parameter) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (decoder *Decoder) processOSC () {
|
|
||||||
// TODO: analyze OSC
|
|
||||||
}
|
|
||||||
|
|
||||||
func (decoder *Decoder) processCSI () {
|
|
||||||
if len(decoder.gathered) < 2 { return }
|
|
||||||
parameters := ParameterInts(decoder.gathered)
|
|
||||||
|
|
||||||
var p0, p1, p2, p3 int
|
|
||||||
if len(parameters) > 0 { p0 = parameters[0] }
|
|
||||||
if len(parameters) > 1 { p1 = parameters[1] }
|
|
||||||
if len(parameters) > 2 { p2 = parameters[2] }
|
|
||||||
if len(parameters) > 3 { p3 = parameters[3] }
|
|
||||||
|
|
||||||
switch Last(decoder.gathered) {
|
|
||||||
case 'A': if decoder.OnCursorUp != nil { decoder.OnCursorUp(clampOne(p0)) }
|
|
||||||
case 'B': if decoder.OnCursorDown != nil { decoder.OnCursorDown(clampOne(p0)) }
|
|
||||||
case 'C': if decoder.OnCursorForward != nil { decoder.OnCursorForward(clampOne(p0)) }
|
|
||||||
case 'D': if decoder.OnCursorBack != nil { decoder.OnCursorBack(clampOne(p0)) }
|
|
||||||
case 'E': if decoder.OnCursorNextLine != nil { decoder.OnCursorNextLine(clampOne(p0)) }
|
|
||||||
case 'F': if decoder.OnCursorPreviousLine != nil { decoder.OnCursorPreviousLine(clampOne(p0)) }
|
|
||||||
case 'G': if decoder.OnCursorHorizontalAbsolute != nil { decoder.OnCursorHorizontalAbsolute(clampOne(p0)) }
|
|
||||||
case 'H', 'f': if decoder.OnCursorPosition != nil { decoder.OnCursorPosition(clampOne(p0), clampOne(p1)) }
|
|
||||||
|
|
||||||
case 'J': if decoder.OnEraseInDisplay != nil { decoder.OnEraseInDisplay(p0) }
|
|
||||||
case 'K': if decoder.OnEraseInLine != nil { decoder.OnEraseInLine(p0) }
|
|
||||||
case 'S': if decoder.OnScrollUp != nil { decoder.OnScrollUp(clampOne(p0)) }
|
|
||||||
case 'T': if decoder.OnScrollDown != nil { decoder.OnScrollDown(clampOne(p0)) }
|
|
||||||
|
|
||||||
case 'm':
|
|
||||||
p0 := SGR(p0)
|
|
||||||
switch {
|
|
||||||
case
|
|
||||||
p0 >= SGR_ForegroundColorBlack &&
|
|
||||||
p0 <= SGR_ForegroundColorWhite &&
|
|
||||||
decoder.OnForegroundColor != nil :
|
|
||||||
decoder.OnForegroundColor(Color(p0 - SGR_ForegroundColorBlack))
|
|
||||||
case
|
|
||||||
p0 >= SGR_ForegroundColorBrightBlack &&
|
|
||||||
p0 <= SGR_ForegroundColorBrightWhite &&
|
|
||||||
decoder.OnForegroundColor != nil :
|
|
||||||
decoder.OnForegroundColor(Color(p0 - SGR_ForegroundColorBrightBlack + 8))
|
|
||||||
|
|
||||||
case p0 == SGR_ForegroundColor:
|
|
||||||
switch p1 {
|
|
||||||
case 2:
|
|
||||||
if decoder.OnForegroundColor == nil { break }
|
|
||||||
decoder.OnForegroundColor(Color(p1))
|
|
||||||
case 5:
|
|
||||||
if decoder.OnForegroundColorTrue == nil { break }
|
|
||||||
decoder.OnForegroundColorTrue (color.RGBA {
|
|
||||||
R: uint8(p1),
|
|
||||||
G: uint8(p2),
|
|
||||||
B: uint8(p3),
|
|
||||||
A: 0xFF,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
case
|
|
||||||
p0 >= SGR_BackgroundColorBlack &&
|
|
||||||
p0 <= SGR_BackgroundColorWhite &&
|
|
||||||
decoder.OnBackgroundColor != nil :
|
|
||||||
decoder.OnBackgroundColor(Color(p0 - SGR_BackgroundColorBlack))
|
|
||||||
case
|
|
||||||
p0 >= SGR_BackgroundColorBrightBlack &&
|
|
||||||
p0 <= SGR_BackgroundColorBrightWhite &&
|
|
||||||
decoder.OnBackgroundColor != nil :
|
|
||||||
decoder.OnBackgroundColor(Color(p0 - SGR_BackgroundColorBrightBlack + 8))
|
|
||||||
|
|
||||||
case p0 == SGR_BackgroundColor:
|
|
||||||
switch p1 {
|
|
||||||
case 2:
|
|
||||||
if decoder.OnBackgroundColor == nil { break }
|
|
||||||
decoder.OnBackgroundColor(Color(p1))
|
|
||||||
case 5:
|
|
||||||
if decoder.OnBackgroundColorTrue == nil { break }
|
|
||||||
decoder.OnBackgroundColorTrue (color.RGBA {
|
|
||||||
R: uint8(p1),
|
|
||||||
G: uint8(p2),
|
|
||||||
B: uint8(p3),
|
|
||||||
A: 0xFF,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
case p0 == SGR_UnderlineColor:
|
|
||||||
switch p1 {
|
|
||||||
case 2:
|
|
||||||
if decoder.OnUnderlineColor == nil { break }
|
|
||||||
decoder.OnUnderlineColor(Color(p1))
|
|
||||||
case 5:
|
|
||||||
if decoder.OnUnderlineColorTrue == nil { break }
|
|
||||||
decoder.OnUnderlineColorTrue (color.RGBA {
|
|
||||||
R: uint8(p1),
|
|
||||||
G: uint8(p2),
|
|
||||||
B: uint8(p3),
|
|
||||||
A: 0xFF,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
default: if decoder.OnSGR != nil { decoder.OnSGR(SGR(p0)) }
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO
|
|
||||||
case 'n':
|
|
||||||
case 's':
|
|
||||||
case 'u':
|
|
||||||
case 'h':
|
|
||||||
case 'l':
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func clampOne (number int) int {
|
|
||||||
if number < 1 {
|
|
||||||
return 1
|
|
||||||
} else {
|
|
||||||
return number
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Last returns the last item of a slice.
|
|
||||||
func Last[T any] (source []T) T {
|
|
||||||
return source[len(source) - 1]
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParameterStrings separates a byte slice by semicolons into a list of strings.
|
|
||||||
func ParameterStrings (source []byte) (parameters []string) {
|
|
||||||
parameters = strings.Split(string(source), ";")
|
|
||||||
for index := range parameters {
|
|
||||||
parameters[index] = strings.TrimSpace(parameters[index])
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParameterInts is like ParameterStrings, but returns integers instead of
|
|
||||||
// strings. If a parameter is empty or cannot be converted into an integer, that
|
|
||||||
// parameter will be zero.
|
|
||||||
func ParameterInts (source []byte) (parameters []int) {
|
|
||||||
stringParameters := ParameterStrings(source)
|
|
||||||
parameters = make([]int, len(stringParameters))
|
|
||||||
for index, parameter := range stringParameters {
|
|
||||||
parameters[index], _ = strconv.Atoi(parameter)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
97
ansi/sgr.go
97
ansi/sgr.go
@ -1,97 +0,0 @@
|
|||||||
package ansi
|
|
||||||
|
|
||||||
// SGR represents a list of Select Graphic Rendition parameters.
|
|
||||||
type SGR int; const (
|
|
||||||
SGR_Normal SGR = iota
|
|
||||||
SGR_Bold
|
|
||||||
SGR_Dim
|
|
||||||
SGR_Italic
|
|
||||||
SGR_Underline
|
|
||||||
SGR_SlowBlink
|
|
||||||
SGR_RapidBlink
|
|
||||||
SGR_Reverse
|
|
||||||
SGR_Conceal
|
|
||||||
SGR_Strike
|
|
||||||
|
|
||||||
SGR_FontPrimary
|
|
||||||
SGR_Font1
|
|
||||||
SGR_Font2
|
|
||||||
SGR_Font3
|
|
||||||
SGR_Font4
|
|
||||||
SGR_Font5
|
|
||||||
SGR_Font6
|
|
||||||
SGR_Font7
|
|
||||||
SGR_Font8
|
|
||||||
SGR_Font9
|
|
||||||
SGR_FontFraktur
|
|
||||||
|
|
||||||
SGR_DoubleUnderline
|
|
||||||
SGR_NormalIntensity
|
|
||||||
SGR_NeitherItalicNorBlackletter
|
|
||||||
SGR_NotUnderlined
|
|
||||||
SGR_NotBlinking
|
|
||||||
SGR_PorportionalSpacing
|
|
||||||
SGR_NotReversed
|
|
||||||
SGR_NotCrossedOut
|
|
||||||
|
|
||||||
SGR_ForegroundColorBlack
|
|
||||||
SGR_ForegroundColorRed
|
|
||||||
SGR_ForegroundColorGreen
|
|
||||||
SGR_ForegroundColorYellow
|
|
||||||
SGR_ForegroundColorBlue
|
|
||||||
SGR_ForegroundColorMagenta
|
|
||||||
SGR_ForegroundColorCyan
|
|
||||||
SGR_ForegroundColorWhite
|
|
||||||
SGR_ForegroundColor
|
|
||||||
SGR_ForegroundColorDefault
|
|
||||||
|
|
||||||
SGR_BackgroundColorBlack
|
|
||||||
SGR_BackgroundColorRed
|
|
||||||
SGR_BackgroundColorGreen
|
|
||||||
SGR_BackgroundColorYellow
|
|
||||||
SGR_BackgroundColorBlue
|
|
||||||
SGR_BackgroundColorMagenta
|
|
||||||
SGR_BackgroundColorCyan
|
|
||||||
SGR_BackgroundColorWhite
|
|
||||||
SGR_BackgroundColor
|
|
||||||
SGR_BackgroundColorDefault
|
|
||||||
|
|
||||||
SGR_DisablePorportionalSpacing
|
|
||||||
SGR_Framed
|
|
||||||
SGR_Encircled
|
|
||||||
SGR_Overlined
|
|
||||||
SGR_NeitherFramedNorEncircled
|
|
||||||
SGR_NotOverlined
|
|
||||||
|
|
||||||
SGR_UnderlineColor
|
|
||||||
SGR_UnderlineColorDefault
|
|
||||||
|
|
||||||
SGR_IdeogramUnderline
|
|
||||||
SGR_IdeogramDoubleUnderline
|
|
||||||
SGR_IdeogramOverline
|
|
||||||
SGR_IdeogramDoubleOverline
|
|
||||||
SGR_IdeogramStressMarking
|
|
||||||
SGR_NoIdeogramAttributes
|
|
||||||
|
|
||||||
SGR_Superscript
|
|
||||||
SGR_Subscript
|
|
||||||
SGR_NeitherSuperscriptNorSubscript
|
|
||||||
|
|
||||||
SGR_ForegroundColorBrightBlack
|
|
||||||
SGR_ForegroundColorBrightRed
|
|
||||||
SGR_ForegroundColorBrightGreen
|
|
||||||
SGR_ForegroundColorBrightYellow
|
|
||||||
SGR_ForegroundColorBrightBlue
|
|
||||||
SGR_ForegroundColorBrightMagenta
|
|
||||||
SGR_ForegroundColorBrightCyan
|
|
||||||
SGR_ForegroundColorBrightWhite
|
|
||||||
|
|
||||||
SGR_BackgroundColorBrightBlack
|
|
||||||
SGR_BackgroundColorBrightRed
|
|
||||||
SGR_BackgroundColorBrightGreen
|
|
||||||
SGR_BackgroundColorBrightYellow
|
|
||||||
SGR_BackgroundColorBrightBlue
|
|
||||||
SGR_BackgroundColorBrightMagenta
|
|
||||||
SGR_BackgroundColorBrightCyan
|
|
||||||
SGR_BackgroundColorBrightWhite
|
|
||||||
)
|
|
167
elements/grid.go
167
elements/grid.go
@ -1,167 +0,0 @@
|
|||||||
package elements
|
|
||||||
|
|
||||||
import "image"
|
|
||||||
import "golang.org/x/image/font"
|
|
||||||
import "git.tebibyte.media/sashakoshka/tomo"
|
|
||||||
import "git.tebibyte.media/sashakoshka/tomo/input"
|
|
||||||
import "git.tebibyte.media/sashakoshka/tomo/default/theme"
|
|
||||||
import "git.tebibyte.media/sashakoshka/tomo/default/config"
|
|
||||||
// import "git.tebibyte.media/sashakoshka/tomo/textdraw"
|
|
||||||
import "git.tebibyte.media/sashakoshka/tomo/elements/core"
|
|
||||||
|
|
||||||
type gridCell struct {
|
|
||||||
rune
|
|
||||||
tomo.FontStyle
|
|
||||||
background tomo.Color
|
|
||||||
foreground tomo.Color
|
|
||||||
clean bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cell *gridCell) initColor () {
|
|
||||||
cell.background = tomo.ColorBackground
|
|
||||||
cell.foreground = tomo.ColorForeground
|
|
||||||
}
|
|
||||||
|
|
||||||
type gridBuffer struct {
|
|
||||||
cells []gridCell
|
|
||||||
stride int
|
|
||||||
}
|
|
||||||
|
|
||||||
// Grid is an array of monospaced character cells. Each one has a foreground and
|
|
||||||
// background color. It satisfies io.Writer and can be fed text with ANSI escape
|
|
||||||
// codes.
|
|
||||||
type Grid struct {
|
|
||||||
*core.Core
|
|
||||||
*core.FocusableCore
|
|
||||||
core core.CoreControl
|
|
||||||
|
|
||||||
cells []gridCell
|
|
||||||
stride int
|
|
||||||
cellWidth int
|
|
||||||
cellHeight int
|
|
||||||
|
|
||||||
cursor image.Point
|
|
||||||
|
|
||||||
face font.Face
|
|
||||||
config config.Wrapped
|
|
||||||
theme theme.Wrapped
|
|
||||||
|
|
||||||
onResize func ()
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewGrid () (element *Grid) {
|
|
||||||
element = &Grid { }
|
|
||||||
element.theme.Case = tomo.C("tomo", "grid")
|
|
||||||
element.Core, element.core = core.NewCore(element, element.drawAndPush)
|
|
||||||
element.updateFont()
|
|
||||||
element.updateMinimumSize()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) OnResize (callback func ()) {
|
|
||||||
element.onResize = callback
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) Write (data []byte) (wrote int, err error) {
|
|
||||||
// TODO process ansi escape codes etx
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) HandleMouseDown (x, y int, button input.Button) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) HandleMouseUp (x, y int, button input.Button) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) HandleKeyDown (key input.Key, modifiers input.Modifiers) {
|
|
||||||
// TODO we need to grab shift ctrl c for copying text
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) HandleKeyUp(key input.Key, modifiers input.Modifiers) { }
|
|
||||||
|
|
||||||
// SetTheme sets the element's theme.
|
|
||||||
func (element *Grid) SetTheme (new tomo.Theme) {
|
|
||||||
if new == element.theme.Theme { return }
|
|
||||||
element.theme.Theme = new
|
|
||||||
element.updateFont()
|
|
||||||
element.updateMinimumSize()
|
|
||||||
element.drawAndPush()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetConfig sets the element's configuration.
|
|
||||||
func (element *Grid) SetConfig (new tomo.Config) {
|
|
||||||
if new == element.config.Config { return }
|
|
||||||
element.config.Config = new
|
|
||||||
element.updateMinimumSize()
|
|
||||||
element.drawAndPush()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) alloc () bool {
|
|
||||||
bounds := element.Bounds()
|
|
||||||
width := bounds.Dx() / element.cellWidth
|
|
||||||
height := bounds.Dy() / element.cellHeight
|
|
||||||
unchanged :=
|
|
||||||
width == element.stride &&
|
|
||||||
height == len(element.cells) / element.stride
|
|
||||||
if unchanged { return false }
|
|
||||||
|
|
||||||
oldCells := element.cells
|
|
||||||
oldWidth := element.stride
|
|
||||||
oldHeight := len(element.cells) / element.stride
|
|
||||||
heightLarger := height < oldHeight
|
|
||||||
|
|
||||||
element.stride = width
|
|
||||||
element.cells = make([]gridCell, width * height)
|
|
||||||
|
|
||||||
// TODO: attempt to wrap text?
|
|
||||||
|
|
||||||
if heightLarger {
|
|
||||||
for index := range element.cells[oldHeight * width:] {
|
|
||||||
element.cells[index].initColor()
|
|
||||||
}}
|
|
||||||
|
|
||||||
commonHeight := height
|
|
||||||
if heightLarger { commonHeight = oldHeight }
|
|
||||||
for index := range element.cells[:commonHeight * width] {
|
|
||||||
x := index % width
|
|
||||||
if x < oldWidth {
|
|
||||||
element.cells[index] = oldCells[x + index / oldWidth]
|
|
||||||
} else {
|
|
||||||
element.cells[index].initColor()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if element.onResize != nil { element.onResize() }
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) updateFont () {
|
|
||||||
element.face = element.theme.FontFace (
|
|
||||||
tomo.FontStyleMonospace,
|
|
||||||
tomo.FontSizeNormal)
|
|
||||||
emSpace, _ := element.face.GlyphAdvance('M')
|
|
||||||
metrics := element.face.Metrics()
|
|
||||||
element.cellWidth = emSpace.Round()
|
|
||||||
element.cellHeight = metrics.Height.Round()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) updateMinimumSize () {
|
|
||||||
element.core.SetMinimumSize(element.cellWidth, element.cellHeight)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) state () tomo.State {
|
|
||||||
return tomo.State {
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) drawAndPush () {
|
|
||||||
if element.core.HasImage () {
|
|
||||||
element.core.DamageRegion(element.draw(true))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (element *Grid) draw (force bool) image.Rectangle {
|
|
||||||
|
|
||||||
}
|
|
Reference in New Issue
Block a user