Compare commits

...

3 Commits

Author SHA1 Message Date
e1eac9de0f
displayStatsMenu 2023-04-02 00:36:12 -07:00
7371cddab3
Add more dividers 2023-04-02 00:28:34 -07:00
aecd8683b2
Restructure loop, basic display functions 2023-04-02 00:26:34 -07:00
3 changed files with 93 additions and 75 deletions

View File

@ -5,9 +5,8 @@ package cardsim
type Card[C StatsCollection] interface { type Card[C StatsCollection] interface {
// Title is the short name of the card displayed in the hand // Title is the short name of the card displayed in the hand
// and at the top of the card output. It receives the current // and at the top of the card output. It receives the current
// player as an argument. If it returns an error that is not // player as an argument.
// a warning, the game crashes. Title(p *Player[C]) Message
Title(p *Player[C]) (Message, error)
// Urgent reports whether the card is considered urgent. If // Urgent reports whether the card is considered urgent. If
// the player has any urgent cards in hand, they cannot choose to act // the player has any urgent cards in hand, they cannot choose to act
@ -68,8 +67,8 @@ type BasicCard[C StatsCollection] struct {
} }
// Title implements Card. // Title implements Card.
func (b *BasicCard[C]) Title(_ *Player[C]) (Message, error) { func (b *BasicCard[C]) Title(_ *Player[C]) Message {
return b.CardTitle, nil return b.CardTitle
} }
// Urgent implements Card. // Urgent implements Card.

View File

@ -11,7 +11,7 @@ import (
type InfoPanel[C StatsCollection] interface { type InfoPanel[C StatsCollection] interface {
// Title returns the title of this InfoPanel, which is also used as the // Title returns the title of this InfoPanel, which is also used as the
// label presented to the player to access this panel. // label presented to the player to access this panel.
Title(p *Player[C]) (Message, error) Title(p *Player[C]) Message
// Info returns the displayable contents of this InfoPanel. A nil Message // Info returns the displayable contents of this InfoPanel. A nil Message
// in the output is interpreted as a paragraph break. // in the output is interpreted as a paragraph break.
@ -44,8 +44,8 @@ func VisibleOrDebug[C StatsCollection](p *Player[C], s Stat) bool {
} }
// Title implements `InfoPanel[C]` by returning b's `Name`. // Title implements `InfoPanel[C]` by returning b's `Name`.
func (b *BasicStatsPanel[C]) Title(p *Player[C]) (Message, error) { func (b *BasicStatsPanel[C]) Title(p *Player[C]) Message {
return b.Name, nil return b.Name
} }
// Info implements `InfoPanel[C]` by dumpiing p.Stats, showing those items for // Info implements `InfoPanel[C]` by dumpiing p.Stats, showing those items for

View File

@ -27,7 +27,8 @@ func RunSimpleTerminalUI[C StatsCollection](p *Player[C]) error {
if p.DebugLevel < 1 && IsSeriousError(err) { if p.DebugLevel < 1 && IsSeriousError(err) {
return err return err
} }
displayAndWait(msg) display(msg)
wait()
} }
review(p) review(p)
@ -43,12 +44,11 @@ func RunSimpleTerminalUI[C StatsCollection](p *Player[C]) error {
return nil return nil
} }
func displayAndWait(m Message) { func display(m Message) {
if m == nil { if m == nil {
return return
} }
fmt.Println(m.String()) fmt.Println(m.String())
wait()
} }
func wait() { func wait() {
@ -58,13 +58,18 @@ func wait() {
} }
func pickNextAction[C StatsCollection](p *Player[C]) (isCard bool, cardIdx int, choiceIdx int) { func pickNextAction[C StatsCollection](p *Player[C]) (isCard bool, cardIdx int, choiceIdx int) {
for {
cls() cls()
needsDivider := displayMessageSection(p) needsDivider := displayMessageSection(p)
if needsDivider { if needsDivider {
divider() divider()
} }
displayOnePanel(p, p.Prompt) displayOnePanel(p, p.Prompt)
divider()
actionsOffset := displayStatsMenu(p) actionsOffset := displayStatsMenu(p)
if actionsOffset > 0 {
divider()
}
handOffset := displayPermanentActionsMenu(p, actionsOffset) handOffset := displayPermanentActionsMenu(p, actionsOffset)
max := displayHandMenu(p, handOffset) max := displayHandMenu(p, handOffset)
@ -101,7 +106,8 @@ func pickNextAction[C StatsCollection](p *Player[C]) (isCard bool, cardIdx int,
i -= 1 i -= 1
if i < actionsOffset { if i < actionsOffset {
cls() cls()
DisplayOnePanelAndWait(p, i) displayOnePanel(p, p.InfoPanels[i])
wait()
} else if i < handOffset { } else if i < handOffset {
cls() cls()
i -= actionsOffset i -= actionsOffset
@ -118,7 +124,7 @@ func pickNextAction[C StatsCollection](p *Player[C]) (isCard bool, cardIdx int,
} }
} }
} }
return pickNextAction(p) }
} }
func cls() { func cls() {
@ -145,23 +151,36 @@ func confirmQuit() {
} }
func displayOnePanel[C StatsCollection](p *Player[C], panel InfoPanel[C]) error { func displayOnePanel[C StatsCollection](p *Player[C], panel InfoPanel[C]) error {
var errs ErrorCollector ts := panel.Title(p).String()
t, err := panel.Title(p)
if IsSeriousError(err) {
return err
}
errs.Add(err)
ts := t.String()
if len(ts) > 0 { if len(ts) > 0 {
fmt.Println(ts) fmt.Println(ts)
fmt.Println(strings.Repeat("-", len(ts))) fmt.Println(strings.Repeat("-", len(ts)))
fmt.Println() fmt.Println()
} }
m, err := panel.Info(p) m, err := panel.Info(p)
errs.Add(err)
if IsSeriousError(err) { if IsSeriousError(err) {
return errs.Emit() return err
} }
displayAndWait(MultiMessage(m)) display(MultiMessage(m))
return errs.Emit() return err
}
func displayMessageSection[C StatsCollection](p *Player[C]) bool {
if len(p.TemporaryMessages) == 0 {
return false
}
display(MultiMessage(p.TemporaryMessages))
return true
}
func displayStatsMenu[C StatsCollection](p *Player[C]) int {
if len(p.InfoPanels) == 0 {
return 0
}
fmt.Println("Info Panels")
fmt.Println("-----------")
for i, s := range p.InfoPanels {
fmt.Printf("[%2d]: %s", i+1, s.Title(p).String())
}
return len(p.InfoPanels)
} }