tabs: make fields private

The Tabs object exposes an array of Tab objects and the current selected
index in that array. The these two fields are sometimes modified in
goroutines, which can lead to data races causing fatal out of bounds
accesses on the tab array.

Hide these fields as private API. Expose only what needs to be seen from
the outside. This will prepare for protecting concurrent access with
a lock in the next commit.

Signed-off-by: Robin Jarry <robin@jarry.cc>
Acked-by: Koni Marti <koni.marti@gmail.com>
This commit is contained in:
Robin Jarry 2022-07-14 18:29:56 +02:00
parent c841f36513
commit 171fefd209
6 changed files with 145 additions and 126 deletions

View file

@ -35,9 +35,13 @@ func (Postpone) Execute(aerc *widgets.Aerc, args []string) error {
if acct == nil { if acct == nil {
return errors.New("No account selected") return errors.New("No account selected")
} }
composer, _ := aerc.SelectedTabContent().(*widgets.Composer) tab := aerc.SelectedTab()
if tab == nil {
return errors.New("No tab selected")
}
composer, _ := tab.Content.(*widgets.Composer)
config := composer.Config() config := composer.Config()
tabName := aerc.TabNames()[aerc.SelectedTabIndex()] tabName := tab.Name
if config.Postpone == "" { if config.Postpone == "" {
return errors.New("No Postpone location configured") return errors.New("No Postpone location configured")

View file

@ -43,8 +43,12 @@ func (Send) Execute(aerc *widgets.Aerc, args []string) error {
if len(args) > 1 { if len(args) > 1 {
return errors.New("Usage: send") return errors.New("Usage: send")
} }
composer, _ := aerc.SelectedTabContent().(*widgets.Composer) tab := aerc.SelectedTab()
tabName := aerc.TabNames()[aerc.SelectedTabIndex()] if tab == nil {
return errors.New("No selected tab")
}
composer, _ := tab.Content.(*widgets.Composer)
tabName := tab.Name
config := composer.Config() config := composer.Config()
if config.Outgoing == "" { if config.Outgoing == "" {

View file

@ -34,18 +34,11 @@ func (MoveTab) Execute(aerc *widgets.Aerc, args []string) error {
return fmt.Errorf("failed to parse index argument: %v", err) return fmt.Errorf("failed to parse index argument: %v", err)
} }
i := aerc.SelectedTabIndex() var relative bool
l := aerc.NumTabs() if strings.HasPrefix(joinedArgs, "+") || strings.HasPrefix(joinedArgs, "-") {
relative = true
if strings.HasPrefix(joinedArgs, "+") {
i = (i + n) % l
} else if strings.HasPrefix(joinedArgs, "-") {
i = (((i + n) % l) + l) % l
} else {
i = n
} }
aerc.MoveTab(n, relative)
aerc.MoveTab(i)
return nil return nil
} }

View file

@ -10,10 +10,10 @@ import (
) )
type Tabs struct { type Tabs struct {
Tabs []*Tab tabs []*Tab
TabStrip *TabStrip TabStrip *TabStrip
TabContent *TabContent TabContent *TabContent
Selected int curIndex int
history []int history []int
uiConfig *config.UIConfig uiConfig *config.UIConfig
@ -54,18 +54,26 @@ func (tabs *Tabs) Add(content Drawable, name string, uiConf *config.UIConfig) *T
Name: name, Name: name,
uiConf: uiConf, uiConf: uiConf,
} }
tabs.Tabs = append(tabs.Tabs, tab) tabs.tabs = append(tabs.tabs, tab)
tabs.TabStrip.Invalidate() tabs.Select(len(tabs.tabs) - 1)
content.OnInvalidate(tabs.invalidateChild) content.OnInvalidate(tabs.invalidateChild)
return tab return tab
} }
func (tabs *Tabs) Names() []string {
var names []string
for _, tab := range tabs.tabs {
names = append(names, tab.Name)
}
return names
}
func (tabs *Tabs) invalidateChild(d Drawable) { func (tabs *Tabs) invalidateChild(d Drawable) {
if tabs.Selected >= len(tabs.Tabs) { if tabs.curIndex >= len(tabs.tabs) {
return return
} }
if tabs.Tabs[tabs.Selected].Content == d { if tabs.tabs[tabs.curIndex].Content == d {
if tabs.onInvalidateContent != nil { if tabs.onInvalidateContent != nil {
tabs.onInvalidateContent(tabs.TabContent) tabs.onInvalidateContent(tabs.TabContent)
} }
@ -74,9 +82,9 @@ func (tabs *Tabs) invalidateChild(d Drawable) {
func (tabs *Tabs) Remove(content Drawable) { func (tabs *Tabs) Remove(content Drawable) {
indexToRemove := -1 indexToRemove := -1
for i, tab := range tabs.Tabs { for i, tab := range tabs.tabs {
if tab.Content == content { if tab.Content == content {
tabs.Tabs = append(tabs.Tabs[:i], tabs.Tabs[i+1:]...) tabs.tabs = append(tabs.tabs[:i], tabs.tabs[i+1:]...)
tabs.removeHistory(i) tabs.removeHistory(i)
indexToRemove = i indexToRemove = i
break break
@ -86,30 +94,29 @@ func (tabs *Tabs) Remove(content Drawable) {
return return
} }
// only pop the tab history if the closing tab is selected // only pop the tab history if the closing tab is selected
if indexToRemove == tabs.Selected { if indexToRemove == tabs.curIndex {
index, ok := tabs.popHistory() index, ok := tabs.popHistory()
if ok { if ok {
tabs.Select(index) tabs.selectPriv(index)
interactive, ok := tabs.Tabs[tabs.Selected].Content.(Interactive) }
} else if indexToRemove < tabs.curIndex {
// selected tab is now one to the left of where it was
tabs.Select(tabs.curIndex - 1)
}
interactive, ok := tabs.tabs[tabs.curIndex].Content.(Interactive)
if ok { if ok {
interactive.Focus(true) interactive.Focus(true)
} }
} }
} else if indexToRemove < tabs.Selected {
// selected tab is now one to the left of where it was
tabs.Selected--
}
tabs.TabStrip.Invalidate()
}
func (tabs *Tabs) Replace(contentSrc Drawable, contentTarget Drawable, name string) { func (tabs *Tabs) Replace(contentSrc Drawable, contentTarget Drawable, name string) {
replaceTab := &Tab{ replaceTab := &Tab{
Content: contentTarget, Content: contentTarget,
Name: name, Name: name,
} }
for i, tab := range tabs.Tabs { for i, tab := range tabs.tabs {
if tab.Content == contentSrc { if tab.Content == contentSrc {
tabs.Tabs[i] = replaceTab tabs.tabs[i] = replaceTab
tabs.Select(i) tabs.Select(i)
if c, ok := contentSrc.(io.Closer); ok { if c, ok := contentSrc.(io.Closer); ok {
c.Close() c.Close()
@ -121,20 +128,44 @@ func (tabs *Tabs) Replace(contentSrc Drawable, contentTarget Drawable, name stri
contentTarget.OnInvalidate(tabs.invalidateChild) contentTarget.OnInvalidate(tabs.invalidateChild)
} }
func (tabs *Tabs) Select(index int) { func (tabs *Tabs) Get(index int) *Tab {
if index >= len(tabs.Tabs) { if index < 0 || index >= len(tabs.tabs) {
index = len(tabs.Tabs) - 1 return nil
}
return tabs.tabs[index]
} }
if tabs.Selected != index { func (tabs *Tabs) Selected() *Tab {
// only push valid tabs onto the history if tabs.curIndex < 0 || tabs.curIndex >= len(tabs.tabs) {
if tabs.Selected < len(tabs.Tabs) { return nil
tabs.pushHistory(tabs.Selected)
} }
tabs.Selected = index return tabs.tabs[tabs.curIndex]
}
func (tabs *Tabs) Select(index int) bool {
if index < 0 || index >= len(tabs.tabs) {
return false
}
if tabs.curIndex != index {
// only push valid tabs onto the history
if tabs.curIndex < len(tabs.tabs) {
tabs.pushHistory(tabs.curIndex)
}
tabs.curIndex = index
tabs.TabStrip.Invalidate() tabs.TabStrip.Invalidate()
tabs.TabContent.Invalidate() tabs.TabContent.Invalidate()
} }
return true
}
func (tabs *Tabs) SelectName(name string) bool {
for i, tab := range tabs.tabs {
if tab.Name == name {
return tabs.Select(i)
}
}
return false
} }
func (tabs *Tabs) SelectPrevious() bool { func (tabs *Tabs) SelectPrevious() bool {
@ -142,24 +173,25 @@ func (tabs *Tabs) SelectPrevious() bool {
if !ok { if !ok {
return false return false
} }
tabs.Select(index) return tabs.Select(index)
return true
} }
func (tabs *Tabs) MoveTab(to int) { func (tabs *Tabs) MoveTab(to int, relative bool) {
from := tabs.Selected from := tabs.curIndex
if relative {
to = from + to
}
if to < 0 { if to < 0 {
to = 0 to = 0
} }
if to >= len(tabs.tabs) {
if to >= len(tabs.Tabs) { to = len(tabs.tabs) - 1
to = len(tabs.Tabs) - 1
} }
tab := tabs.Tabs[from] tab := tabs.tabs[from]
if to > from { if to > from {
copy(tabs.Tabs[from:to], tabs.Tabs[from+1:to+1]) copy(tabs.tabs[from:to], tabs.tabs[from+1:to+1])
for i, h := range tabs.history { for i, h := range tabs.history {
if h == from { if h == from {
tabs.history[i] = to tabs.history[i] = to
@ -169,7 +201,7 @@ func (tabs *Tabs) MoveTab(to int) {
} }
} }
} else if from > to { } else if from > to {
copy(tabs.Tabs[to+1:from+1], tabs.Tabs[to:from]) copy(tabs.tabs[to+1:from+1], tabs.tabs[to:from])
for i, h := range tabs.history { for i, h := range tabs.history {
if h == from { if h == from {
tabs.history[i] = to tabs.history[i] = to
@ -182,44 +214,44 @@ func (tabs *Tabs) MoveTab(to int) {
return return
} }
tabs.Tabs[to] = tab tabs.tabs[to] = tab
tabs.Selected = to tabs.curIndex = to
tabs.TabStrip.Invalidate() tabs.TabStrip.Invalidate()
} }
func (tabs *Tabs) PinTab() { func (tabs *Tabs) PinTab() {
if tabs.Tabs[tabs.Selected].pinned { if tabs.tabs[tabs.curIndex].pinned {
return return
} }
pinEnd := len(tabs.Tabs) pinEnd := len(tabs.tabs)
for i, t := range tabs.Tabs { for i, t := range tabs.tabs {
if !t.pinned { if !t.pinned {
pinEnd = i pinEnd = i
break break
} }
} }
for _, t := range tabs.Tabs { for _, t := range tabs.tabs {
if t.pinned && t.indexBeforePin > tabs.Selected-pinEnd { if t.pinned && t.indexBeforePin > tabs.curIndex-pinEnd {
t.indexBeforePin -= 1 t.indexBeforePin -= 1
} }
} }
tabs.Tabs[tabs.Selected].pinned = true tabs.tabs[tabs.curIndex].pinned = true
tabs.Tabs[tabs.Selected].indexBeforePin = tabs.Selected - pinEnd tabs.tabs[tabs.curIndex].indexBeforePin = tabs.curIndex - pinEnd
tabs.MoveTab(pinEnd) tabs.MoveTab(pinEnd, false)
} }
func (tabs *Tabs) UnpinTab() { func (tabs *Tabs) UnpinTab() {
if !tabs.Tabs[tabs.Selected].pinned { if !tabs.tabs[tabs.curIndex].pinned {
return return
} }
pinEnd := len(tabs.Tabs) pinEnd := len(tabs.tabs)
for i, t := range tabs.Tabs { for i, t := range tabs.tabs {
if i != tabs.Selected && t.pinned && t.indexBeforePin > tabs.Tabs[tabs.Selected].indexBeforePin { if i != tabs.curIndex && t.pinned && t.indexBeforePin > tabs.tabs[tabs.curIndex].indexBeforePin {
t.indexBeforePin += 1 t.indexBeforePin += 1
} }
if !t.pinned { if !t.pinned {
@ -228,23 +260,23 @@ func (tabs *Tabs) UnpinTab() {
} }
} }
tabs.Tabs[tabs.Selected].pinned = false tabs.tabs[tabs.curIndex].pinned = false
tabs.MoveTab(tabs.Tabs[tabs.Selected].indexBeforePin + pinEnd - 1) tabs.MoveTab(tabs.tabs[tabs.curIndex].indexBeforePin+pinEnd-1, false)
} }
func (tabs *Tabs) NextTab() { func (tabs *Tabs) NextTab() {
next := tabs.Selected + 1 next := tabs.curIndex + 1
if next >= len(tabs.Tabs) { if next >= len(tabs.tabs) {
next = 0 next = 0
} }
tabs.Select(next) tabs.Select(next)
} }
func (tabs *Tabs) PrevTab() { func (tabs *Tabs) PrevTab() {
next := tabs.Selected - 1 next := tabs.curIndex - 1
if next < 0 { if next < 0 {
next = len(tabs.Tabs) - 1 next = len(tabs.tabs) - 1
} }
tabs.Select(next) tabs.Select(next)
} }
@ -284,13 +316,13 @@ func (tabs *Tabs) removeHistory(index int) {
// TODO: Color repository // TODO: Color repository
func (strip *TabStrip) Draw(ctx *Context) { func (strip *TabStrip) Draw(ctx *Context) {
x := 0 x := 0
for i, tab := range strip.Tabs { for i, tab := range strip.tabs {
uiConfig := strip.uiConfig uiConfig := strip.uiConfig
if tab.uiConf != nil { if tab.uiConf != nil {
uiConfig = tab.uiConf uiConfig = tab.uiConf
} }
style := uiConfig.GetStyle(config.STYLE_TAB) style := uiConfig.GetStyle(config.STYLE_TAB)
if strip.Selected == i { if strip.curIndex == i {
style = uiConfig.GetStyleSelected(config.STYLE_TAB) style = uiConfig.GetStyleSelected(config.STYLE_TAB)
} }
tabWidth := 32 tabWidth := 32
@ -319,7 +351,7 @@ func (strip *TabStrip) Invalidate() {
func (strip *TabStrip) MouseEvent(localX int, localY int, event tcell.Event) { func (strip *TabStrip) MouseEvent(localX int, localY int, event tcell.Event) {
changeFocus := func(focus bool) { changeFocus := func(focus bool) {
interactive, ok := strip.parent.Tabs[strip.parent.Selected].Content.(Interactive) interactive, ok := strip.parent.tabs[strip.parent.curIndex].Content.(Interactive)
if ok { if ok {
interactive.Focus(focus) interactive.Focus(focus)
} }
@ -330,8 +362,8 @@ func (strip *TabStrip) MouseEvent(localX int, localY int, event tcell.Event) {
case *tcell.EventMouse: case *tcell.EventMouse:
switch event.Buttons() { switch event.Buttons() {
case tcell.Button1: case tcell.Button1:
selectedTab, ok := strip.Clicked(localX, localY) selectedTab, ok := strip.clicked(localX, localY)
if !ok || selectedTab == strip.parent.Selected { if !ok || selectedTab == strip.parent.curIndex {
return return
} }
unfocus() unfocus()
@ -346,18 +378,12 @@ func (strip *TabStrip) MouseEvent(localX int, localY int, event tcell.Event) {
strip.parent.PrevTab() strip.parent.PrevTab()
refocus() refocus()
case tcell.Button3: case tcell.Button3:
selectedTab, ok := strip.Clicked(localX, localY) selectedTab, ok := strip.clicked(localX, localY)
if !ok { if !ok {
return return
} }
unfocus() unfocus()
if selectedTab == strip.parent.Selected {
strip.parent.CloseTab(selectedTab) strip.parent.CloseTab(selectedTab)
} else {
current := strip.parent.Selected
strip.parent.CloseTab(selectedTab)
strip.parent.Select(current)
}
refocus() refocus()
} }
} }
@ -367,9 +393,9 @@ func (strip *TabStrip) OnInvalidate(onInvalidate func(d Drawable)) {
strip.onInvalidateStrip = onInvalidate strip.onInvalidateStrip = onInvalidate
} }
func (strip *TabStrip) Clicked(mouseX int, mouseY int) (int, bool) { func (strip *TabStrip) clicked(mouseX int, mouseY int) (int, bool) {
x := 0 x := 0
for i, tab := range strip.Tabs { for i, tab := range strip.tabs {
trunc := runewidth.Truncate(tab.Name, 32, "…") trunc := runewidth.Truncate(tab.Name, 32, "…")
length := len(trunc) + 2 length := len(trunc) + 2
if x <= mouseX && mouseX < x+length { if x <= mouseX && mouseX < x+length {
@ -381,27 +407,27 @@ func (strip *TabStrip) Clicked(mouseX int, mouseY int) (int, bool) {
} }
func (content *TabContent) Children() []Drawable { func (content *TabContent) Children() []Drawable {
children := make([]Drawable, len(content.Tabs)) children := make([]Drawable, len(content.tabs))
for i, tab := range content.Tabs { for i, tab := range content.tabs {
children[i] = tab.Content children[i] = tab.Content
} }
return children return children
} }
func (content *TabContent) Draw(ctx *Context) { func (content *TabContent) Draw(ctx *Context) {
if content.Selected >= len(content.Tabs) { if content.curIndex >= len(content.tabs) {
width := ctx.Width() width := ctx.Width()
height := ctx.Height() height := ctx.Height()
ctx.Fill(0, 0, width, height, ' ', ctx.Fill(0, 0, width, height, ' ',
content.uiConfig.GetStyle(config.STYLE_TAB)) content.uiConfig.GetStyle(config.STYLE_TAB))
} }
tab := content.Tabs[content.Selected] tab := content.tabs[content.curIndex]
tab.Content.Draw(ctx) tab.Content.Draw(ctx)
} }
func (content *TabContent) MouseEvent(localX int, localY int, event tcell.Event) { func (content *TabContent) MouseEvent(localX int, localY int, event tcell.Event) {
tab := content.Tabs[content.Selected] tab := content.tabs[content.curIndex]
switch tabContent := tab.Content.(type) { switch tabContent := tab.Content.(type) {
case Mouseable: case Mouseable:
tabContent.MouseEvent(localX, localY, event) tabContent.MouseEvent(localX, localY, event)
@ -412,7 +438,7 @@ func (content *TabContent) Invalidate() {
if content.onInvalidateContent != nil { if content.onInvalidateContent != nil {
content.onInvalidateContent(content) content.onInvalidateContent(content)
} }
tab := content.Tabs[content.Selected] tab := content.tabs[content.curIndex]
tab.Content.Invalidate() tab.Content.Invalidate()
} }

View file

@ -234,7 +234,7 @@ func (acct *AccountView) SelectedMessagePart() *PartInfo {
} }
func (acct *AccountView) isSelected() bool { func (acct *AccountView) isSelected() bool {
return acct.aerc.NumTabs() > 0 && acct == acct.aerc.SelectedAccount() return acct == acct.aerc.SelectedAccount()
} }
func (acct *AccountView) onMessage(msg types.WorkerMessage) { func (acct *AccountView) onMessage(msg types.WorkerMessage) {

View file

@ -105,8 +105,14 @@ func NewAerc(conf *config.AercConfig, logger *log.Logger,
aerc.NewTab(wizard, "New account") aerc.NewTab(wizard, "New account")
} }
tabs.Select(0)
tabs.CloseTab = func(index int) { tabs.CloseTab = func(index int) {
switch content := aerc.tabs.Tabs[index].Content.(type) { tab := aerc.tabs.Get(index)
if tab == nil {
return
}
switch content := tab.Content.(type) {
case *AccountView: case *AccountView:
return return
case *AccountWizard: case *AccountWizard:
@ -284,7 +290,7 @@ func (aerc *Aerc) Event(event tcell.Event) bool {
aerc.BeginExCommand("") aerc.BeginExCommand("")
return true return true
} }
interactive, ok := aerc.tabs.Tabs[aerc.tabs.Selected].Content.(ui.Interactive) interactive, ok := aerc.SelectedTabContent().(ui.Interactive)
if ok { if ok {
return interactive.Event(event) return interactive.Event(event)
} }
@ -334,18 +340,15 @@ func (aerc *Aerc) SelectedAccountUiConfig() *config.UIConfig {
} }
func (aerc *Aerc) SelectedTabContent() ui.Drawable { func (aerc *Aerc) SelectedTabContent() ui.Drawable {
if aerc.NumTabs() == 0 || aerc.SelectedTabIndex() >= aerc.NumTabs() { tab := aerc.tabs.Selected()
if tab == nil {
return nil return nil
} }
return aerc.tabs.Tabs[aerc.SelectedTabIndex()].Content return tab.Content
} }
func (aerc *Aerc) SelectedTabIndex() int { func (aerc *Aerc) SelectedTab() *ui.Tab {
return aerc.tabs.Selected return aerc.tabs.Selected()
}
func (aerc *Aerc) NumTabs() int {
return len(aerc.tabs.Tabs)
} }
func (aerc *Aerc) NewTab(clickable ui.Drawable, name string) *ui.Tab { func (aerc *Aerc) NewTab(clickable ui.Drawable, name string) *ui.Tab {
@ -355,7 +358,6 @@ func (aerc *Aerc) NewTab(clickable ui.Drawable, name string) *ui.Tab {
uiConf = conf uiConf = conf
} }
tab := aerc.tabs.Add(clickable, name, uiConf) tab := aerc.tabs.Add(clickable, name, uiConf)
aerc.tabs.Select(len(aerc.tabs.Tabs) - 1)
aerc.UpdateStatus() aerc.UpdateStatus()
return tab return tab
} }
@ -369,8 +371,8 @@ func (aerc *Aerc) ReplaceTab(tabSrc ui.Drawable, tabTarget ui.Drawable, name str
aerc.tabs.Replace(tabSrc, tabTarget, name) aerc.tabs.Replace(tabSrc, tabTarget, name)
} }
func (aerc *Aerc) MoveTab(i int) { func (aerc *Aerc) MoveTab(i int, relative bool) {
aerc.tabs.MoveTab(i) aerc.tabs.MoveTab(i, relative)
} }
func (aerc *Aerc) PinTab() { func (aerc *Aerc) PinTab() {
@ -390,33 +392,23 @@ func (aerc *Aerc) PrevTab() {
} }
func (aerc *Aerc) SelectTab(name string) bool { func (aerc *Aerc) SelectTab(name string) bool {
for i, tab := range aerc.tabs.Tabs { ok := aerc.tabs.SelectName(name)
if tab.Name == name { if ok {
aerc.tabs.Select(i)
aerc.UpdateStatus() aerc.UpdateStatus()
return true
} }
} return ok
return false
} }
func (aerc *Aerc) SelectTabIndex(index int) bool { func (aerc *Aerc) SelectTabIndex(index int) bool {
for i := range aerc.tabs.Tabs { ok := aerc.tabs.Select(index)
if i == index { if ok {
aerc.tabs.Select(i)
aerc.UpdateStatus() aerc.UpdateStatus()
return true
} }
} return ok
return false
} }
func (aerc *Aerc) TabNames() []string { func (aerc *Aerc) TabNames() []string {
var names []string return aerc.tabs.Names()
for _, tab := range aerc.tabs.Tabs {
names = append(names, tab.Name)
}
return names
} }
func (aerc *Aerc) SelectPreviousTab() bool { func (aerc *Aerc) SelectPreviousTab() bool {
@ -463,7 +455,7 @@ func (aerc *Aerc) focus(item ui.Interactive) {
aerc.focused.Focus(false) aerc.focused.Focus(false)
} }
aerc.focused = item aerc.focused = item
interactive, ok := aerc.tabs.Tabs[aerc.tabs.Selected].Content.(ui.Interactive) interactive, ok := aerc.SelectedTabContent().(ui.Interactive)
if item != nil { if item != nil {
item.Focus(true) item.Focus(true)
if ok { if ok {