mirror of
https://github.com/vsariola/sointu.git
synced 2025-05-28 03:10:24 -04:00
161 lines
4.8 KiB
Go
161 lines
4.8 KiB
Go
package gioui
|
|
|
|
import (
|
|
"fmt"
|
|
"image"
|
|
"math"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"gioui.org/f32"
|
|
"gioui.org/io/key"
|
|
"gioui.org/layout"
|
|
"gioui.org/op"
|
|
"gioui.org/op/clip"
|
|
"gioui.org/op/paint"
|
|
"gioui.org/text"
|
|
"gioui.org/unit"
|
|
"gioui.org/widget"
|
|
"github.com/vsariola/sointu/tracker"
|
|
)
|
|
|
|
const patternCellHeight = 16
|
|
const patternCellWidth = 16
|
|
const patternRowMarkerWidth = 30
|
|
const orderTitleHeight = unit.Dp(52)
|
|
|
|
type OrderEditor struct {
|
|
scrollTable *ScrollTable
|
|
tag struct{}
|
|
}
|
|
|
|
var patternIndexStrings [36]string
|
|
|
|
func init() {
|
|
for i := 0; i < 10; i++ {
|
|
patternIndexStrings[i] = string('0' + byte(i))
|
|
}
|
|
for i := 10; i < 36; i++ {
|
|
patternIndexStrings[i] = string('A' + byte(i-10))
|
|
}
|
|
}
|
|
|
|
func NewOrderEditor(m *tracker.Model) *OrderEditor {
|
|
return &OrderEditor{
|
|
scrollTable: NewScrollTable(
|
|
m.Order().Table(),
|
|
m.Tracks().List(),
|
|
m.OrderRows().List(),
|
|
),
|
|
}
|
|
}
|
|
|
|
func (oe *OrderEditor) Layout(gtx C, t *Tracker) D {
|
|
if oe.scrollTable.CursorMoved() {
|
|
cursor := t.TrackEditor.scrollTable.Table.Cursor()
|
|
t.TrackEditor.scrollTable.ColTitleList.CenterOn(cursor.X)
|
|
t.TrackEditor.scrollTable.RowTitleList.CenterOn(cursor.Y)
|
|
}
|
|
|
|
for _, e := range gtx.Events(&oe.tag) {
|
|
switch e := e.(type) {
|
|
case key.Event:
|
|
if e.State != key.Press {
|
|
continue
|
|
}
|
|
oe.command(gtx, t, e)
|
|
}
|
|
}
|
|
defer op.Offset(image.Point{}).Push(gtx.Ops).Pop()
|
|
defer clip.Rect(image.Rect(0, 0, gtx.Constraints.Max.X, gtx.Constraints.Max.Y)).Push(gtx.Ops).Pop()
|
|
key.InputOp{Tag: &oe.tag, Keys: "Ctrl-⌫|Ctrl-⌦|⏎|Ctrl-⏎|0|1|2|3|4|5|6|7|8|9|A|B|C|D|E|F|G|H|I|J|K|L|M|N|O|P|Q|R|S|T|U|V|W|X|Y|Z"}.Add(gtx.Ops)
|
|
|
|
colTitle := func(gtx C, i int) D {
|
|
h := gtx.Dp(orderTitleHeight)
|
|
defer op.Offset(image.Pt(0, -2)).Push(gtx.Ops).Pop()
|
|
defer op.Affine(f32.Affine2D{}.Rotate(f32.Pt(0, 0), -90*math.Pi/180).Offset(f32.Point{X: 0, Y: float32(h)})).Push(gtx.Ops).Pop()
|
|
gtx.Constraints = layout.Exact(image.Pt(1e6, 1e6))
|
|
title := t.Model.Order().Title(i)
|
|
LabelStyle{Alignment: layout.NW, Text: title, FontSize: unit.Sp(12), Color: mediumEmphasisTextColor, Shaper: t.Theme.Shaper}.Layout(gtx)
|
|
return D{Size: image.Pt(patternCellWidth, h)}
|
|
}
|
|
|
|
rowTitle := func(gtx C, j int) D {
|
|
w := gtx.Dp(unit.Dp(30))
|
|
if playPos := t.PlayPosition(); t.SongPanel.PlayingBtn.Bool.Value() && j == playPos.OrderRow {
|
|
paint.FillShape(gtx.Ops, patternPlayColor, clip.Rect{Max: image.Pt(gtx.Constraints.Max.X, patternCellHeight)}.Op())
|
|
}
|
|
color := rowMarkerPatternTextColor
|
|
if l := t.Loop(); j >= l.Start && j < l.Start+l.Length {
|
|
color = loopMarkerColor
|
|
}
|
|
paint.ColorOp{Color: color}.Add(gtx.Ops)
|
|
defer op.Offset(image.Pt(0, -2)).Push(gtx.Ops).Pop()
|
|
widget.Label{}.Layout(gtx, t.Theme.Shaper, trackerFont, trackerFontSize, strings.ToUpper(fmt.Sprintf("%02x", j)), op.CallOp{})
|
|
return D{Size: image.Pt(w, patternCellHeight)}
|
|
}
|
|
|
|
selection := oe.scrollTable.Table.Range()
|
|
|
|
cell := func(gtx C, x, y int) D {
|
|
val := patternIndexToString(t.Model.Order().Value(tracker.Point{X: x, Y: y}))
|
|
color := patternCellColor
|
|
point := tracker.Point{X: x, Y: y}
|
|
if selection.Contains(point) {
|
|
color = inactiveSelectionColor
|
|
if oe.scrollTable.Focused() {
|
|
color = selectionColor
|
|
if point == oe.scrollTable.Table.Cursor() {
|
|
color = cursorColor
|
|
}
|
|
}
|
|
}
|
|
paint.FillShape(gtx.Ops, color, clip.Rect{Min: image.Pt(1, 1), Max: image.Pt(gtx.Constraints.Min.X-1, gtx.Constraints.Min.X-1)}.Op())
|
|
paint.ColorOp{Color: patternTextColor}.Add(gtx.Ops)
|
|
defer op.Offset(image.Pt(0, -2)).Push(gtx.Ops).Pop()
|
|
widget.Label{Alignment: text.Middle}.Layout(gtx, t.Theme.Shaper, trackerFont, trackerFontSize, val, op.CallOp{})
|
|
return D{Size: image.Pt(patternCellWidth, patternCellHeight)}
|
|
}
|
|
|
|
table := FilledScrollTable(t.Theme, oe.scrollTable, cell, colTitle, rowTitle, nil, nil)
|
|
table.ColumnTitleHeight = orderTitleHeight
|
|
|
|
return table.Layout(gtx)
|
|
}
|
|
|
|
func (oe *OrderEditor) command(gtx C, t *Tracker, e key.Event) {
|
|
switch e.Name {
|
|
case key.NameDeleteBackward:
|
|
if e.Modifiers.Contain(key.ModShortcut) {
|
|
t.Model.DeleteOrderRow(true).Do()
|
|
}
|
|
case key.NameDeleteForward:
|
|
if e.Modifiers.Contain(key.ModShortcut) {
|
|
t.Model.DeleteOrderRow(false).Do()
|
|
}
|
|
case key.NameReturn:
|
|
if e.Modifiers.Contain(key.ModShortcut) {
|
|
oe.scrollTable.Table.MoveCursor(0, -1)
|
|
oe.scrollTable.Table.SetCursor2(oe.scrollTable.Table.Cursor())
|
|
}
|
|
t.Model.AddOrderRow(!e.Modifiers.Contain(key.ModShortcut)).Do()
|
|
}
|
|
if iv, err := strconv.Atoi(e.Name); err == nil {
|
|
t.Model.Order().SetValue(oe.scrollTable.Table.Cursor(), iv)
|
|
oe.scrollTable.EnsureCursorVisible()
|
|
}
|
|
if b := int(e.Name[0]) - 'A'; len(e.Name) == 1 && b >= 0 && b < 26 {
|
|
t.Model.Order().SetValue(oe.scrollTable.Table.Cursor(), b+10)
|
|
oe.scrollTable.EnsureCursorVisible()
|
|
}
|
|
}
|
|
|
|
func patternIndexToString(index int) string {
|
|
if index < 0 {
|
|
return ""
|
|
} else if index < len(patternIndexStrings) {
|
|
return patternIndexStrings[index]
|
|
}
|
|
return "?"
|
|
}
|