mirror of
https://github.com/vsariola/sointu.git
synced 2025-05-28 03:10:24 -04:00
Also includes a refactoring of the List: all methods that accepted or returned a [from, to] range now return a Range, which is non-inclusive i.e. [start,end). Also the assignUnitIds was slightly refactored & a new function called assignUnitIdsForPatch was added, to assign all unit IDs for an patch at once. Closes #157, #163.
512 lines
19 KiB
Go
512 lines
19 KiB
Go
package gioui
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"image"
|
|
"image/color"
|
|
"io"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"gioui.org/font"
|
|
"gioui.org/io/clipboard"
|
|
"gioui.org/io/key"
|
|
"gioui.org/layout"
|
|
"gioui.org/op"
|
|
"gioui.org/op/clip"
|
|
"gioui.org/text"
|
|
"gioui.org/unit"
|
|
"gioui.org/widget"
|
|
"github.com/vsariola/sointu"
|
|
"github.com/vsariola/sointu/tracker"
|
|
"golang.org/x/exp/shiny/materialdesign/icons"
|
|
)
|
|
|
|
type InstrumentEditor struct {
|
|
newInstrumentBtn *ActionClickable
|
|
enlargeBtn *BoolClickable
|
|
deleteInstrumentBtn *ActionClickable
|
|
linkInstrTrackBtn *BoolClickable
|
|
splitInstrumentBtn *ActionClickable
|
|
copyInstrumentBtn *TipClickable
|
|
saveInstrumentBtn *TipClickable
|
|
loadInstrumentBtn *TipClickable
|
|
addUnitBtn *ActionClickable
|
|
presetMenuBtn *TipClickable
|
|
commentExpandBtn *BoolClickable
|
|
soloBtn *BoolClickable
|
|
muteBtn *BoolClickable
|
|
commentEditor *Editor
|
|
commentString tracker.String
|
|
nameEditor *Editor
|
|
nameString tracker.String
|
|
searchEditor *Editor
|
|
instrumentDragList *DragList
|
|
unitDragList *DragList
|
|
unitEditor *UnitEditor
|
|
wasFocused bool
|
|
presetMenuItems []MenuItem
|
|
presetMenu Menu
|
|
|
|
enlargeHint, shrinkHint string
|
|
addInstrumentHint string
|
|
octaveHint string
|
|
expandCommentHint string
|
|
collapseCommentHint string
|
|
deleteInstrumentHint string
|
|
muteHint, unmuteHint string
|
|
soloHint, unsoloHint string
|
|
linkDisabledHint string
|
|
linkEnabledHint string
|
|
splitInstrumentHint string
|
|
}
|
|
|
|
func NewInstrumentEditor(model *tracker.Model) *InstrumentEditor {
|
|
ret := &InstrumentEditor{
|
|
newInstrumentBtn: NewActionClickable(model.AddInstrument()),
|
|
enlargeBtn: NewBoolClickable(model.InstrEnlarged().Bool()),
|
|
deleteInstrumentBtn: NewActionClickable(model.DeleteInstrument()),
|
|
linkInstrTrackBtn: NewBoolClickable(model.LinkInstrTrack().Bool()),
|
|
splitInstrumentBtn: NewActionClickable(model.SplitInstrument()),
|
|
copyInstrumentBtn: new(TipClickable),
|
|
saveInstrumentBtn: new(TipClickable),
|
|
loadInstrumentBtn: new(TipClickable),
|
|
addUnitBtn: NewActionClickable(model.AddUnit(false)),
|
|
commentExpandBtn: NewBoolClickable(model.CommentExpanded().Bool()),
|
|
presetMenuBtn: new(TipClickable),
|
|
soloBtn: NewBoolClickable(model.Solo().Bool()),
|
|
muteBtn: NewBoolClickable(model.Mute().Bool()),
|
|
commentEditor: NewEditor(widget.Editor{}),
|
|
nameEditor: NewEditor(widget.Editor{SingleLine: true, Submit: true, Alignment: text.Middle}),
|
|
searchEditor: NewEditor(widget.Editor{SingleLine: true, Submit: true, Alignment: text.Start}),
|
|
commentString: model.InstrumentComment().String(),
|
|
nameString: model.InstrumentName().String(),
|
|
instrumentDragList: NewDragList(model.Instruments().List(), layout.Horizontal),
|
|
unitDragList: NewDragList(model.Units().List(), layout.Vertical),
|
|
unitEditor: NewUnitEditor(model),
|
|
presetMenuItems: []MenuItem{},
|
|
}
|
|
model.IterateInstrumentPresets(func(index int, name string) bool {
|
|
ret.presetMenuItems = append(ret.presetMenuItems, MenuItem{Text: name, IconBytes: icons.ImageAudiotrack, Doer: model.LoadPreset(index)})
|
|
return true
|
|
})
|
|
ret.enlargeHint = makeHint("Enlarge", " (%s)", "InstrEnlargedToggle")
|
|
ret.shrinkHint = makeHint("Shrink", " (%s)", "InstrEnlargedToggle")
|
|
ret.addInstrumentHint = makeHint("Add\ninstrument", "\n(%s)", "AddInstrument")
|
|
ret.octaveHint = makeHint("Octave down", " (%s)", "OctaveNumberInputSubtract") + makeHint(" or up", " (%s)", "OctaveNumberInputAdd")
|
|
ret.expandCommentHint = makeHint("Expand comment", " (%s)", "CommentExpandedToggle")
|
|
ret.collapseCommentHint = makeHint("Collapse comment", " (%s)", "CommentExpandedToggle")
|
|
ret.deleteInstrumentHint = makeHint("Delete\ninstrument", "\n(%s)", "DeleteInstrument")
|
|
ret.muteHint = makeHint("Mute", " (%s)", "MuteToggle")
|
|
ret.unmuteHint = makeHint("Unmute", " (%s)", "MuteToggle")
|
|
ret.soloHint = makeHint("Solo", " (%s)", "SoloToggle")
|
|
ret.unsoloHint = makeHint("Unsolo", " (%s)", "SoloToggle")
|
|
ret.linkDisabledHint = makeHint("Instrument-Track\nlinking disabled", "\n(%s)", "LinkInstrTrackToggle")
|
|
ret.linkEnabledHint = makeHint("Instrument-Track\nlinking enabled", "\n(%s)", "LinkInstrTrackToggle")
|
|
ret.splitInstrumentHint = makeHint("Split instrument", " (%s)", "SplitInstrument")
|
|
return ret
|
|
}
|
|
|
|
func (ie *InstrumentEditor) Focus() {
|
|
ie.unitDragList.Focus()
|
|
}
|
|
|
|
func (ie *InstrumentEditor) Focused() bool {
|
|
return ie.unitDragList.focused
|
|
}
|
|
|
|
func (ie *InstrumentEditor) childFocused(gtx C) bool {
|
|
return ie.unitEditor.sliderList.Focused() ||
|
|
ie.instrumentDragList.Focused() || gtx.Source.Focused(ie.commentEditor) || gtx.Source.Focused(ie.nameEditor) || gtx.Source.Focused(ie.searchEditor) ||
|
|
gtx.Source.Focused(ie.addUnitBtn.Clickable) || gtx.Source.Focused(ie.commentExpandBtn.Clickable) || gtx.Source.Focused(ie.presetMenuBtn.Clickable) ||
|
|
gtx.Source.Focused(ie.deleteInstrumentBtn.Clickable) || gtx.Source.Focused(ie.copyInstrumentBtn.Clickable)
|
|
}
|
|
|
|
func (ie *InstrumentEditor) Layout(gtx C, t *Tracker) D {
|
|
ie.wasFocused = ie.Focused() || ie.childFocused(gtx)
|
|
fullscreenBtnStyle := ToggleIcon(gtx, t.Theme, ie.enlargeBtn, icons.NavigationFullscreen, icons.NavigationFullscreenExit, ie.enlargeHint, ie.shrinkHint)
|
|
linkBtnStyle := ToggleIcon(gtx, t.Theme, ie.linkInstrTrackBtn, icons.NotificationSyncDisabled, icons.NotificationSync, ie.linkDisabledHint, ie.linkEnabledHint)
|
|
|
|
octave := func(gtx C) D {
|
|
in := layout.UniformInset(unit.Dp(1))
|
|
numStyle := NumericUpDown(t.Theme, t.OctaveNumberInput, ie.octaveHint)
|
|
dims := in.Layout(gtx, numStyle.Layout)
|
|
return dims
|
|
}
|
|
|
|
newBtnStyle := ActionIcon(gtx, t.Theme, ie.newInstrumentBtn, icons.ContentAdd, ie.addInstrumentHint)
|
|
ret := layout.Flex{Axis: layout.Vertical}.Layout(gtx,
|
|
layout.Rigid(func(gtx C) D {
|
|
return layout.Flex{}.Layout(
|
|
gtx,
|
|
layout.Flexed(1, func(gtx C) D {
|
|
return ie.layoutInstrumentList(gtx, t)
|
|
}),
|
|
layout.Rigid(func(gtx C) D {
|
|
inset := layout.UniformInset(unit.Dp(6))
|
|
return inset.Layout(gtx, func(gtx C) D {
|
|
return layout.Flex{Axis: layout.Horizontal}.Layout(gtx,
|
|
layout.Rigid(Label("OCT:", white, t.Theme.Shaper)),
|
|
layout.Rigid(octave),
|
|
)
|
|
})
|
|
}),
|
|
layout.Rigid(func(gtx C) D {
|
|
return layout.E.Layout(gtx, linkBtnStyle.Layout)
|
|
}),
|
|
layout.Rigid(func(gtx C) D {
|
|
return layout.E.Layout(gtx, fullscreenBtnStyle.Layout)
|
|
}),
|
|
layout.Rigid(func(gtx C) D {
|
|
return layout.E.Layout(gtx, newBtnStyle.Layout)
|
|
}),
|
|
)
|
|
}),
|
|
layout.Rigid(func(gtx C) D {
|
|
return ie.layoutInstrumentHeader(gtx, t)
|
|
}),
|
|
layout.Flexed(1, func(gtx C) D {
|
|
return layout.Flex{Axis: layout.Horizontal}.Layout(gtx,
|
|
layout.Rigid(func(gtx C) D {
|
|
return ie.layoutUnitList(gtx, t)
|
|
}),
|
|
layout.Flexed(1, func(gtx C) D {
|
|
return ie.unitEditor.Layout(gtx, t)
|
|
}),
|
|
)
|
|
}))
|
|
return ret
|
|
}
|
|
|
|
func (ie *InstrumentEditor) layoutInstrumentHeader(gtx C, t *Tracker) D {
|
|
header := func(gtx C) D {
|
|
commentExpandBtnStyle := ToggleIcon(gtx, t.Theme, ie.commentExpandBtn, icons.NavigationExpandMore, icons.NavigationExpandLess, ie.expandCommentHint, ie.collapseCommentHint)
|
|
presetMenuBtnStyle := TipIcon(t.Theme, ie.presetMenuBtn, icons.NavigationMenu, "Load preset")
|
|
copyInstrumentBtnStyle := TipIcon(t.Theme, ie.copyInstrumentBtn, icons.ContentContentCopy, "Copy instrument")
|
|
saveInstrumentBtnStyle := TipIcon(t.Theme, ie.saveInstrumentBtn, icons.ContentSave, "Save instrument")
|
|
loadInstrumentBtnStyle := TipIcon(t.Theme, ie.loadInstrumentBtn, icons.FileFolderOpen, "Load instrument")
|
|
deleteInstrumentBtnStyle := ActionIcon(gtx, t.Theme, ie.deleteInstrumentBtn, icons.ActionDelete, ie.deleteInstrumentHint)
|
|
splitInstrumentBtnStyle := ActionIcon(gtx, t.Theme, ie.splitInstrumentBtn, icons.CommunicationCallSplit, ie.splitInstrumentHint)
|
|
soloBtnStyle := ToggleIcon(gtx, t.Theme, ie.soloBtn, icons.SocialGroup, icons.SocialPerson, ie.soloHint, ie.unsoloHint)
|
|
muteBtnStyle := ToggleIcon(gtx, t.Theme, ie.muteBtn, icons.AVVolumeUp, icons.AVVolumeOff, ie.muteHint, ie.unmuteHint)
|
|
|
|
m := PopupMenu(&ie.presetMenu, t.Theme.Shaper)
|
|
|
|
for ie.copyInstrumentBtn.Clickable.Clicked(gtx) {
|
|
if contents, ok := t.Instruments().List().CopyElements(); ok {
|
|
gtx.Execute(clipboard.WriteCmd{Type: "application/text", Data: io.NopCloser(bytes.NewReader(contents))})
|
|
t.Alerts().Add("Instrument copied to clipboard", tracker.Info)
|
|
}
|
|
}
|
|
|
|
for ie.saveInstrumentBtn.Clickable.Clicked(gtx) {
|
|
writer, err := t.Explorer.CreateFile(t.InstrumentName().Value() + ".yml")
|
|
if err != nil {
|
|
continue
|
|
}
|
|
t.SaveInstrument(writer)
|
|
}
|
|
|
|
for ie.loadInstrumentBtn.Clickable.Clicked(gtx) {
|
|
reader, err := t.Explorer.ChooseFile(".yml", ".json", ".4ki", ".4kp")
|
|
if err != nil {
|
|
continue
|
|
}
|
|
t.LoadInstrument(reader)
|
|
}
|
|
|
|
header := func(gtx C) D {
|
|
return layout.Flex{Axis: layout.Horizontal, Alignment: layout.Middle}.Layout(gtx,
|
|
layout.Rigid(Label("Voices: ", white, t.Theme.Shaper)),
|
|
layout.Rigid(func(gtx layout.Context) layout.Dimensions {
|
|
numStyle := NumericUpDown(t.Theme, t.InstrumentVoices, "Number of voices for this instrument")
|
|
dims := numStyle.Layout(gtx)
|
|
return dims
|
|
}),
|
|
layout.Rigid(splitInstrumentBtnStyle.Layout),
|
|
layout.Flexed(1, func(gtx C) D { return layout.Dimensions{Size: gtx.Constraints.Min} }),
|
|
layout.Rigid(commentExpandBtnStyle.Layout),
|
|
layout.Rigid(soloBtnStyle.Layout),
|
|
layout.Rigid(muteBtnStyle.Layout),
|
|
layout.Rigid(func(gtx C) D {
|
|
//defer op.Offset(image.Point{}).Push(gtx.Ops).Pop()
|
|
dims := presetMenuBtnStyle.Layout(gtx)
|
|
op.Offset(image.Pt(0, dims.Size.Y)).Add(gtx.Ops)
|
|
gtx.Constraints.Max.Y = gtx.Dp(unit.Dp(500))
|
|
gtx.Constraints.Max.X = gtx.Dp(unit.Dp(180))
|
|
m.Layout(gtx, ie.presetMenuItems...)
|
|
return dims
|
|
}),
|
|
layout.Rigid(saveInstrumentBtnStyle.Layout),
|
|
layout.Rigid(loadInstrumentBtnStyle.Layout),
|
|
layout.Rigid(copyInstrumentBtnStyle.Layout),
|
|
layout.Rigid(deleteInstrumentBtnStyle.Layout))
|
|
}
|
|
|
|
for ie.presetMenuBtn.Clickable.Clicked(gtx) {
|
|
ie.presetMenu.Visible = true
|
|
}
|
|
|
|
if ie.commentExpandBtn.Bool.Value() || gtx.Source.Focused(ie.commentEditor) { // we draw once the widget after it manages to lose focus
|
|
ret := layout.Flex{Axis: layout.Vertical}.Layout(gtx,
|
|
layout.Rigid(header),
|
|
layout.Rigid(func(gtx C) D {
|
|
defer clip.Rect(image.Rect(0, 0, gtx.Constraints.Max.X, gtx.Constraints.Max.Y)).Push(gtx.Ops).Pop()
|
|
ie.commentEditor.SetText(ie.commentString.Value())
|
|
for ie.commentEditor.Submitted(gtx) || ie.commentEditor.Cancelled(gtx) {
|
|
ie.instrumentDragList.Focus()
|
|
}
|
|
style := MaterialEditor(t.Theme, ie.commentEditor, "Comment")
|
|
style.Color = highEmphasisTextColor
|
|
ret := layout.UniformInset(unit.Dp(6)).Layout(gtx, style.Layout)
|
|
ie.commentString.Set(ie.commentEditor.Text())
|
|
return ret
|
|
}),
|
|
)
|
|
return ret
|
|
}
|
|
return header(gtx)
|
|
}
|
|
|
|
return Surface{Gray: 37, Focus: ie.wasFocused}.Layout(gtx, header)
|
|
}
|
|
|
|
func (ie *InstrumentEditor) layoutInstrumentList(gtx C, t *Tracker) D {
|
|
gtx.Constraints.Max.Y = gtx.Dp(unit.Dp(36))
|
|
element := func(gtx C, i int) D {
|
|
gtx.Constraints.Min.Y = gtx.Dp(unit.Dp(36))
|
|
gtx.Constraints.Min.X = gtx.Dp(unit.Dp(30))
|
|
grabhandle := LabelStyle{Text: strconv.Itoa(i + 1), ShadeColor: black, Color: mediumEmphasisTextColor, FontSize: unit.Sp(10), Alignment: layout.Center, Shaper: t.Theme.Shaper}
|
|
label := func(gtx C) D {
|
|
name, level, mute, ok := (*tracker.Instruments)(t.Model).Item(i)
|
|
if !ok {
|
|
labelStyle := LabelStyle{Text: "", ShadeColor: black, Color: white, FontSize: unit.Sp(12), Shaper: t.Theme.Shaper}
|
|
return layout.Center.Layout(gtx, labelStyle.Layout)
|
|
}
|
|
k := byte(255 - level*127)
|
|
color := color.NRGBA{R: 255, G: k, B: 255, A: 255}
|
|
if i == ie.instrumentDragList.TrackerList.Selected() {
|
|
ie.nameEditor.SetText(name)
|
|
for ie.nameEditor.Submitted(gtx) || ie.nameEditor.Cancelled(gtx) {
|
|
ie.instrumentDragList.Focus()
|
|
}
|
|
style := MaterialEditor(t.Theme, ie.nameEditor, "Instr")
|
|
style.Color = color
|
|
style.HintColor = instrumentNameHintColor
|
|
style.TextSize = unit.Sp(12)
|
|
style.Font = labelDefaultFont
|
|
if mute {
|
|
style.Color = disabledTextColor
|
|
style.Font.Style = font.Italic
|
|
}
|
|
dims := layout.Center.Layout(gtx, func(gtx C) D {
|
|
defer clip.Rect(image.Rect(0, 0, gtx.Constraints.Max.X, gtx.Constraints.Max.Y)).Push(gtx.Ops).Pop()
|
|
return style.Layout(gtx)
|
|
})
|
|
ie.nameString.Set(ie.nameEditor.Text())
|
|
return dims
|
|
}
|
|
if name == "" {
|
|
name = "Instr"
|
|
}
|
|
labelStyle := LabelStyle{Text: name, ShadeColor: black, Color: color, Font: labelDefaultFont, FontSize: unit.Sp(12), Shaper: t.Theme.Shaper}
|
|
if mute {
|
|
labelStyle.Color = disabledTextColor
|
|
labelStyle.Font.Style = font.Italic
|
|
}
|
|
return layout.Center.Layout(gtx, labelStyle.Layout)
|
|
}
|
|
return layout.Inset{Left: unit.Dp(6), Right: unit.Dp(6), Top: unit.Dp(4)}.Layout(gtx, func(gtx C) D {
|
|
return layout.Flex{Axis: layout.Vertical, Alignment: layout.Middle}.Layout(gtx,
|
|
layout.Rigid(grabhandle.Layout),
|
|
layout.Rigid(label),
|
|
)
|
|
})
|
|
}
|
|
|
|
color := inactiveLightSurfaceColor
|
|
if ie.wasFocused {
|
|
color = activeLightSurfaceColor
|
|
}
|
|
instrumentList := FilledDragList(t.Theme, ie.instrumentDragList, element, nil)
|
|
instrumentList.SelectedColor = color
|
|
instrumentList.HoverColor = instrumentHoverColor
|
|
instrumentList.ScrollBarWidth = unit.Dp(6)
|
|
|
|
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()
|
|
for {
|
|
event, ok := gtx.Event(
|
|
key.Filter{Focus: ie.instrumentDragList, Name: key.NameDownArrow},
|
|
key.Filter{Focus: ie.instrumentDragList, Name: key.NameReturn},
|
|
key.Filter{Focus: ie.instrumentDragList, Name: key.NameEnter},
|
|
)
|
|
if !ok {
|
|
break
|
|
}
|
|
switch e := event.(type) {
|
|
case key.Event:
|
|
switch e.State {
|
|
case key.Press:
|
|
switch e.Name {
|
|
case key.NameDownArrow:
|
|
ie.unitDragList.Focus()
|
|
case key.NameReturn, key.NameEnter:
|
|
gtx.Execute(key.FocusCmd{Tag: &ie.nameEditor.Editor})
|
|
l := len(ie.nameEditor.Editor.Text())
|
|
ie.nameEditor.Editor.SetCaret(l, l)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
dims := instrumentList.Layout(gtx)
|
|
gtx.Constraints = layout.Exact(dims.Size)
|
|
instrumentList.LayoutScrollBar(gtx)
|
|
return dims
|
|
}
|
|
|
|
func (ie *InstrumentEditor) layoutUnitList(gtx C, t *Tracker) D {
|
|
// TODO: how to ie.unitDragList.Focus()
|
|
addUnitBtnStyle := ActionIcon(gtx, t.Theme, ie.addUnitBtn, icons.ContentAdd, "Add unit (Enter)")
|
|
addUnitBtnStyle.IconButtonStyle.Color = t.Theme.ContrastFg
|
|
addUnitBtnStyle.IconButtonStyle.Background = t.Theme.Fg
|
|
addUnitBtnStyle.IconButtonStyle.Inset = layout.UniformInset(unit.Dp(4))
|
|
|
|
var units [256]tracker.UnitListItem
|
|
for i, item := range (*tracker.Units)(t.Model).Iterate {
|
|
if i >= 256 {
|
|
break
|
|
}
|
|
units[i] = item
|
|
}
|
|
count := min(ie.unitDragList.TrackerList.Count(), 256)
|
|
|
|
element := func(gtx C, i int) D {
|
|
gtx.Constraints.Max.Y = gtx.Dp(20)
|
|
gtx.Constraints.Min.Y = gtx.Constraints.Max.Y
|
|
if i < 0 || i > 255 {
|
|
return layout.Dimensions{Size: gtx.Constraints.Min}
|
|
}
|
|
u := units[i]
|
|
var color color.NRGBA = white
|
|
f := labelDefaultFont
|
|
|
|
stackText := strconv.FormatInt(int64(u.StackAfter), 10)
|
|
if u.StackNeed > u.StackBefore {
|
|
color = errorColor
|
|
(*tracker.Alerts)(t.Model).AddNamed("UnitNeedsInputs", fmt.Sprintf("%v needs at least %v input signals, got %v", u.Type, u.StackNeed, u.StackBefore), tracker.Error)
|
|
} else if i == count-1 && u.StackAfter != 0 {
|
|
color = warningColor
|
|
(*tracker.Alerts)(t.Model).AddNamed("InstrumentLeavesSignals", fmt.Sprintf("Instrument leaves %v signal(s) on the stack", u.StackAfter), tracker.Warning)
|
|
}
|
|
if u.Disabled {
|
|
color = disabledTextColor
|
|
f.Style = font.Italic
|
|
}
|
|
|
|
stackLabel := LabelStyle{Text: stackText, ShadeColor: black, Color: mediumEmphasisTextColor, Font: labelDefaultFont, FontSize: unit.Sp(12), Shaper: t.Theme.Shaper}
|
|
rightMargin := layout.Inset{Right: unit.Dp(10)}
|
|
return layout.Flex{Axis: layout.Horizontal}.Layout(gtx,
|
|
layout.Rigid(func(gtx C) D {
|
|
if i == ie.unitDragList.TrackerList.Selected() {
|
|
defer clip.Rect(image.Rect(0, 0, gtx.Constraints.Max.X, gtx.Constraints.Max.Y)).Push(gtx.Ops).Pop()
|
|
str := tracker.String{StringData: (*tracker.UnitSearch)(t.Model)}
|
|
ie.searchEditor.SetText(str.Value())
|
|
for ie.searchEditor.Submitted(gtx) {
|
|
ie.unitDragList.Focus()
|
|
if text := ie.searchEditor.Text(); text != "" {
|
|
for _, n := range sointu.UnitNames {
|
|
if strings.HasPrefix(n, ie.searchEditor.Text()) {
|
|
t.Units().SetSelectedType(n)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
t.UnitSearching().Bool().Set(false)
|
|
ie.searchEditor.SetText(str.Value())
|
|
}
|
|
for ie.searchEditor.Cancelled(gtx) {
|
|
t.UnitSearching().Bool().Set(false)
|
|
ie.searchEditor.SetText(str.Value())
|
|
ie.unitDragList.Focus()
|
|
}
|
|
style := MaterialEditor(t.Theme, ie.searchEditor, "---")
|
|
style.Color = color
|
|
style.HintColor = instrumentNameHintColor
|
|
style.TextSize = unit.Sp(12)
|
|
style.Font = f
|
|
ret := style.Layout(gtx)
|
|
str.Set(ie.searchEditor.Text())
|
|
return ret
|
|
} else {
|
|
unitNameLabel := LabelStyle{Text: u.Type, ShadeColor: black, Color: color, Font: f, FontSize: unit.Sp(12), Shaper: t.Theme.Shaper}
|
|
if unitNameLabel.Text == "" {
|
|
unitNameLabel.Text = "---"
|
|
}
|
|
return unitNameLabel.Layout(gtx)
|
|
}
|
|
}),
|
|
layout.Flexed(1, func(gtx C) D {
|
|
unitNameLabel := LabelStyle{Text: u.Comment, ShadeColor: black, Color: mediumEmphasisTextColor, Font: f, FontSize: unit.Sp(12), Shaper: t.Theme.Shaper}
|
|
inset := layout.Inset{Left: unit.Dp(5)}
|
|
return inset.Layout(gtx, unitNameLabel.Layout)
|
|
}),
|
|
layout.Rigid(func(gtx C) D {
|
|
return rightMargin.Layout(gtx, stackLabel.Layout)
|
|
}),
|
|
)
|
|
}
|
|
|
|
defer op.Offset(image.Point{}).Push(gtx.Ops).Pop()
|
|
unitList := FilledDragList(t.Theme, ie.unitDragList, element, nil)
|
|
for {
|
|
event, ok := gtx.Event(
|
|
key.Filter{Focus: ie.unitDragList, Name: key.NameRightArrow},
|
|
key.Filter{Focus: ie.unitDragList, Name: key.NameEnter, Optional: key.ModCtrl},
|
|
key.Filter{Focus: ie.unitDragList, Name: key.NameReturn, Optional: key.ModCtrl},
|
|
key.Filter{Focus: ie.unitDragList, Name: key.NameDeleteBackward},
|
|
key.Filter{Focus: ie.unitDragList, Name: key.NameEscape},
|
|
)
|
|
if !ok {
|
|
break
|
|
}
|
|
switch e := event.(type) {
|
|
case key.Event:
|
|
switch e.State {
|
|
case key.Press:
|
|
switch e.Name {
|
|
case key.NameEscape:
|
|
ie.instrumentDragList.Focus()
|
|
case key.NameRightArrow:
|
|
ie.unitEditor.sliderList.Focus()
|
|
case key.NameDeleteBackward:
|
|
t.Units().SetSelectedType("")
|
|
t.UnitSearching().Bool().Set(true)
|
|
gtx.Execute(key.FocusCmd{Tag: &ie.searchEditor.Editor})
|
|
case key.NameEnter, key.NameReturn:
|
|
t.Model.AddUnit(e.Modifiers.Contain(key.ModCtrl)).Do()
|
|
t.UnitSearching().Bool().Set(true)
|
|
gtx.Execute(key.FocusCmd{Tag: &ie.searchEditor.Editor})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return Surface{Gray: 30, Focus: ie.wasFocused}.Layout(gtx, func(gtx C) D {
|
|
return layout.Stack{Alignment: layout.SE}.Layout(gtx,
|
|
layout.Expanded(func(gtx C) D {
|
|
defer clip.Rect(image.Rect(0, 0, gtx.Constraints.Max.X, gtx.Constraints.Max.Y)).Push(gtx.Ops).Pop()
|
|
gtx.Constraints = layout.Exact(image.Pt(gtx.Dp(140), gtx.Constraints.Max.Y))
|
|
dims := unitList.Layout(gtx)
|
|
unitList.LayoutScrollBar(gtx)
|
|
return dims
|
|
}),
|
|
layout.Stacked(func(gtx C) D {
|
|
margin := layout.Inset{Right: unit.Dp(20), Bottom: unit.Dp(1)}
|
|
return margin.Layout(gtx, addUnitBtnStyle.Layout)
|
|
}),
|
|
)
|
|
})
|
|
}
|