mirror of
https://github.com/OpenDiablo2/OpenDiablo2
synced 2024-11-02 09:17:19 -04:00
6e31cfb52a
* migrate to ebiten v2.0 API * fixed lint errors
537 lines
14 KiB
Go
537 lines
14 KiB
Go
package d2ui
|
|
|
|
import (
|
|
"fmt"
|
|
"image"
|
|
"log"
|
|
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2common/d2enum"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2common/d2interface"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2common/d2resource"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2common/d2util"
|
|
"github.com/OpenDiablo2/OpenDiablo2/d2core/d2gui"
|
|
)
|
|
|
|
// ButtonType defines the type of button
|
|
type ButtonType int
|
|
|
|
// ButtonType constants
|
|
const (
|
|
ButtonTypeWide ButtonType = 1
|
|
ButtonTypeMedium ButtonType = 2
|
|
ButtonTypeNarrow ButtonType = 3
|
|
ButtonTypeCancel ButtonType = 4
|
|
ButtonTypeTall ButtonType = 5
|
|
ButtonTypeShort ButtonType = 6
|
|
ButtonTypeOkCancel ButtonType = 7
|
|
|
|
// Game UI
|
|
|
|
ButtonTypeSkill ButtonType = 7
|
|
ButtonTypeRun ButtonType = 8
|
|
ButtonTypeMenu ButtonType = 9
|
|
ButtonTypeGoldCoin ButtonType = 10
|
|
ButtonTypeClose ButtonType = 11
|
|
ButtonTypeSecondaryInvHand ButtonType = 12
|
|
ButtonTypeMinipanelCharacter ButtonType = 13
|
|
ButtonTypeMinipanelInventory ButtonType = 14
|
|
ButtonTypeMinipanelSkill ButtonType = 15
|
|
ButtonTypeMinipanelAutomap ButtonType = 16
|
|
ButtonTypeMinipanelMessage ButtonType = 17
|
|
ButtonTypeMinipanelQuest ButtonType = 18
|
|
ButtonTypeMinipanelMen ButtonType = 19
|
|
ButtonTypeSquareClose ButtonType = 20
|
|
ButtonTypeSkillTreeTab ButtonType = 21
|
|
|
|
ButtonNoFixedWidth int = -1
|
|
ButtonNoFixedHeight int = -1
|
|
)
|
|
|
|
const (
|
|
buttonStatePressed = iota + 1
|
|
buttonStateToggled
|
|
buttonStatePressedToggled
|
|
)
|
|
|
|
const (
|
|
closeButtonBaseFrame = 10 // base frame offset of the "close" button dc6
|
|
)
|
|
|
|
const (
|
|
greyAlpha100 = 0x646464ff
|
|
lightGreyAlpha75 = 0x808080c3
|
|
whiteAlpha100 = 0xffffffff
|
|
)
|
|
|
|
// ButtonLayout defines the type of buttons
|
|
type ButtonLayout struct {
|
|
ResourceName string
|
|
PaletteName string
|
|
FontPath string
|
|
ClickableRect *image.Rectangle
|
|
XSegments int
|
|
YSegments int
|
|
BaseFrame int
|
|
DisabledFrame int
|
|
TextOffset int
|
|
FixedWidth int
|
|
FixedHeight int
|
|
LabelColor uint32
|
|
Toggleable bool
|
|
AllowFrameChange bool
|
|
HasImage bool
|
|
}
|
|
|
|
const (
|
|
buttonWideSegmentsX = 2
|
|
buttonWideSegmentsY = 1
|
|
buttonWideDisabledFrame = -1
|
|
buttonWideTextOffset = 1
|
|
|
|
buttonShortSegmentsX = 1
|
|
buttonShortSegmentsY = 1
|
|
buttonShortDisabledFrame = -1
|
|
buttonShortTextOffset = -1
|
|
|
|
buttonMediumSegmentsX = 1
|
|
buttonMediumSegmentsY = 1
|
|
|
|
buttonTallSegmentsX = 1
|
|
buttonTallSegmentsY = 1
|
|
buttonTallTextOffset = 5
|
|
|
|
buttonOkCancelSegmentsX = 1
|
|
buttonOkCancelSegmentsY = 1
|
|
buttonOkCancelDisabledFrame = -1
|
|
|
|
buttonBuySellSegmentsX = 1
|
|
buttonBuySellSegmentsY = 1
|
|
buttonBuySellDisabledFrame = 1
|
|
|
|
buttonSkillTreeTabXSegments = 1
|
|
buttonSkillTreeTabYSegments = 1
|
|
buttonSkillTreeTabDisabledFrame = 7
|
|
buttonSkillTreeTabBaseFrame = 7
|
|
buttonSkillTreeTabFixedWidth = 93
|
|
buttonSkillTreeTabFixedHeight = 107
|
|
|
|
buttonRunSegmentsX = 1
|
|
buttonRunSegmentsY = 1
|
|
buttonRunDisabledFrame = -1
|
|
|
|
pressedButtonOffset = 2
|
|
)
|
|
|
|
// nolint:funlen // cant reduce
|
|
func getButtonLayouts() map[ButtonType]ButtonLayout {
|
|
return map[ButtonType]ButtonLayout{
|
|
ButtonTypeWide: {
|
|
XSegments: buttonWideSegmentsX,
|
|
YSegments: buttonWideSegmentsY,
|
|
DisabledFrame: buttonWideDisabledFrame,
|
|
TextOffset: buttonWideTextOffset,
|
|
ResourceName: d2resource.WideButtonBlank,
|
|
PaletteName: d2resource.PaletteUnits,
|
|
FontPath: d2resource.FontExocet10,
|
|
AllowFrameChange: true,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeShort: {
|
|
XSegments: buttonShortSegmentsX,
|
|
YSegments: buttonShortSegmentsY,
|
|
DisabledFrame: buttonShortDisabledFrame,
|
|
TextOffset: buttonShortTextOffset,
|
|
ResourceName: d2resource.ShortButtonBlank,
|
|
PaletteName: d2resource.PaletteUnits,
|
|
FontPath: d2resource.FontRediculous,
|
|
AllowFrameChange: true,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeMedium: {
|
|
XSegments: buttonMediumSegmentsX,
|
|
YSegments: buttonMediumSegmentsY,
|
|
ResourceName: d2resource.MediumButtonBlank,
|
|
PaletteName: d2resource.PaletteUnits,
|
|
FontPath: d2resource.FontExocet10,
|
|
AllowFrameChange: true,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeTall: {
|
|
XSegments: buttonTallSegmentsX,
|
|
YSegments: buttonTallSegmentsY,
|
|
TextOffset: buttonTallTextOffset,
|
|
ResourceName: d2resource.TallButtonBlank,
|
|
PaletteName: d2resource.PaletteUnits,
|
|
FontPath: d2resource.FontExocet10,
|
|
AllowFrameChange: true,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeOkCancel: {
|
|
XSegments: buttonOkCancelSegmentsX,
|
|
YSegments: buttonOkCancelSegmentsY,
|
|
DisabledFrame: buttonOkCancelDisabledFrame,
|
|
ResourceName: d2resource.CancelButton,
|
|
PaletteName: d2resource.PaletteUnits,
|
|
FontPath: d2resource.FontRediculous,
|
|
AllowFrameChange: true,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeRun: {
|
|
XSegments: buttonRunSegmentsX,
|
|
YSegments: buttonRunSegmentsY,
|
|
DisabledFrame: buttonRunDisabledFrame,
|
|
ResourceName: d2resource.RunButton,
|
|
PaletteName: d2resource.PaletteSky,
|
|
Toggleable: true,
|
|
FontPath: d2resource.FontRediculous,
|
|
AllowFrameChange: true,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeSquareClose: {
|
|
XSegments: buttonBuySellSegmentsX,
|
|
YSegments: buttonBuySellSegmentsY,
|
|
DisabledFrame: buttonBuySellDisabledFrame,
|
|
ResourceName: d2resource.BuySellButton,
|
|
PaletteName: d2resource.PaletteUnits,
|
|
Toggleable: true,
|
|
FontPath: d2resource.Font30,
|
|
AllowFrameChange: true,
|
|
BaseFrame: closeButtonBaseFrame,
|
|
HasImage: true,
|
|
FixedWidth: ButtonNoFixedWidth,
|
|
FixedHeight: ButtonNoFixedHeight,
|
|
LabelColor: greyAlpha100,
|
|
},
|
|
ButtonTypeSkillTreeTab: {
|
|
XSegments: buttonSkillTreeTabXSegments,
|
|
YSegments: buttonSkillTreeTabYSegments,
|
|
DisabledFrame: buttonSkillTreeTabDisabledFrame,
|
|
BaseFrame: buttonSkillTreeTabBaseFrame,
|
|
ResourceName: d2resource.SkillsPanelAmazon,
|
|
PaletteName: d2resource.PaletteSky,
|
|
Toggleable: false,
|
|
FontPath: d2resource.Font16,
|
|
AllowFrameChange: false,
|
|
HasImage: false,
|
|
FixedWidth: buttonSkillTreeTabFixedWidth,
|
|
FixedHeight: buttonSkillTreeTabFixedHeight,
|
|
LabelColor: whiteAlpha100,
|
|
},
|
|
}
|
|
}
|
|
|
|
var _ Widget = &Button{} // static check to ensure button implements widget
|
|
|
|
// Button defines a standard wide UI button
|
|
type Button struct {
|
|
manager *UIManager
|
|
buttonLayout ButtonLayout
|
|
normalSurface d2interface.Surface
|
|
pressedSurface d2interface.Surface
|
|
toggledSurface d2interface.Surface
|
|
pressedToggledSurface d2interface.Surface
|
|
disabledSurface d2interface.Surface
|
|
x int
|
|
y int
|
|
width int
|
|
height int
|
|
onClick func()
|
|
enabled bool
|
|
visible bool
|
|
pressed bool
|
|
toggled bool
|
|
}
|
|
|
|
// NewButton creates an instance of Button
|
|
func (ui *UIManager) NewButton(buttonType ButtonType, text string) *Button {
|
|
btn := &Button{
|
|
width: 0,
|
|
height: 0,
|
|
visible: true,
|
|
enabled: true,
|
|
pressed: false,
|
|
}
|
|
|
|
buttonLayout := getButtonLayouts()[buttonType]
|
|
btn.buttonLayout = buttonLayout
|
|
lbl := ui.NewLabel(buttonLayout.FontPath, d2resource.PaletteUnits)
|
|
|
|
lbl.SetText(text)
|
|
lbl.Color[0] = d2util.Color(buttonLayout.LabelColor)
|
|
lbl.Alignment = d2gui.HorizontalAlignCenter
|
|
|
|
buttonSprite, err := ui.NewSprite(buttonLayout.ResourceName, buttonLayout.PaletteName)
|
|
if err != nil {
|
|
log.Print(err)
|
|
return nil
|
|
}
|
|
|
|
if buttonLayout.FixedWidth > 0 {
|
|
btn.width = buttonLayout.FixedWidth
|
|
} else {
|
|
for i := 0; i < buttonLayout.XSegments; i++ {
|
|
w, _, frameSizeErr := buttonSprite.GetFrameSize(i)
|
|
if frameSizeErr != nil {
|
|
log.Print(frameSizeErr)
|
|
return nil
|
|
}
|
|
|
|
btn.width += w
|
|
}
|
|
}
|
|
|
|
if buttonLayout.FixedHeight > 0 {
|
|
btn.height = buttonLayout.FixedHeight
|
|
} else {
|
|
for i := 0; i < buttonLayout.YSegments; i++ {
|
|
_, h, frameSizeErr := buttonSprite.GetFrameSize(i * buttonLayout.YSegments)
|
|
if frameSizeErr != nil {
|
|
log.Print(frameSizeErr)
|
|
return nil
|
|
}
|
|
|
|
btn.height += h
|
|
}
|
|
}
|
|
|
|
btn.normalSurface = ui.renderer.NewSurface(btn.width, btn.height)
|
|
|
|
buttonSprite.SetPosition(0, 0)
|
|
buttonSprite.SetEffect(d2enum.DrawEffectModulate)
|
|
|
|
ui.addWidget(btn) // important that this comes before prerenderStates!
|
|
|
|
btn.prerenderStates(buttonSprite, &buttonLayout, lbl)
|
|
|
|
return btn
|
|
}
|
|
|
|
type buttonStateDescriptor struct {
|
|
baseFrame int
|
|
offsetX, offsetY int
|
|
prerenderdestination *d2interface.Surface
|
|
fmtErr string
|
|
}
|
|
|
|
func (v *Button) prerenderStates(btnSprite *Sprite, btnLayout *ButtonLayout, label *Label) {
|
|
numButtonStates := btnSprite.GetFrameCount() / (btnLayout.XSegments * btnLayout.YSegments)
|
|
|
|
// buttons always have a base image
|
|
if v.buttonLayout.HasImage {
|
|
err := btnSprite.RenderSegmented(v.normalSurface, btnLayout.XSegments,
|
|
btnLayout.YSegments, btnLayout.BaseFrame)
|
|
if err != nil {
|
|
fmt.Printf("failed to render button normalSurface, err: %v\n", err)
|
|
}
|
|
}
|
|
|
|
_, labelHeight := label.GetSize()
|
|
textY := half(v.height - labelHeight)
|
|
xOffset := half(v.width)
|
|
|
|
label.SetPosition(xOffset, textY)
|
|
label.Render(v.normalSurface)
|
|
|
|
if !btnLayout.HasImage || !btnLayout.AllowFrameChange {
|
|
return
|
|
}
|
|
|
|
xSeg, ySeg, baseFrame := btnLayout.XSegments, btnLayout.YSegments, btnLayout.BaseFrame
|
|
|
|
buttonStateConfigs := make([]*buttonStateDescriptor, 0)
|
|
|
|
// pressed button
|
|
if numButtonStates > buttonStatePressed {
|
|
state := &buttonStateDescriptor{
|
|
baseFrame + buttonStatePressed,
|
|
xOffset - pressedButtonOffset, textY + pressedButtonOffset,
|
|
&v.pressedSurface,
|
|
"failed to render button pressedSurface, err: %v\n",
|
|
}
|
|
|
|
buttonStateConfigs = append(buttonStateConfigs, state)
|
|
}
|
|
|
|
// toggle button
|
|
if numButtonStates > buttonStateToggled {
|
|
buttonStateConfigs = append(buttonStateConfigs, &buttonStateDescriptor{
|
|
baseFrame + buttonStateToggled,
|
|
xOffset, textY,
|
|
&v.toggledSurface,
|
|
"failed to render button toggledSurface, err: %v\n",
|
|
})
|
|
}
|
|
|
|
// pressed+toggled
|
|
if numButtonStates > buttonStatePressedToggled {
|
|
buttonStateConfigs = append(buttonStateConfigs, &buttonStateDescriptor{
|
|
baseFrame + buttonStatePressedToggled,
|
|
xOffset, textY,
|
|
&v.pressedToggledSurface,
|
|
"failed to render button pressedToggledSurface, err: %v\n",
|
|
})
|
|
}
|
|
|
|
// disabled button
|
|
if btnLayout.DisabledFrame != -1 {
|
|
disabledState := &buttonStateDescriptor{
|
|
btnLayout.DisabledFrame,
|
|
xOffset, textY,
|
|
&v.disabledSurface,
|
|
"failed to render button disabledSurface, err: %v\n",
|
|
}
|
|
|
|
buttonStateConfigs = append(buttonStateConfigs, disabledState)
|
|
}
|
|
|
|
for stateIdx, w, h := 0, v.width, v.height; stateIdx < len(buttonStateConfigs); stateIdx++ {
|
|
state := buttonStateConfigs[stateIdx]
|
|
|
|
if stateIdx > 1 && btnLayout.ResourceName == d2resource.BuySellButton {
|
|
// Without returning early, the button UI gets all subsequent (unrelated) frames
|
|
// stacked on top. Only 2 frames from this sprite are applicable to the button
|
|
// in question. The presentation is incorrect without this hack!
|
|
continue
|
|
}
|
|
|
|
surface := v.manager.renderer.NewSurface(w, h)
|
|
|
|
*state.prerenderdestination = surface
|
|
|
|
err := btnSprite.RenderSegmented(*state.prerenderdestination, xSeg, ySeg, state.baseFrame)
|
|
if err != nil {
|
|
fmt.Printf(state.fmtErr, err)
|
|
}
|
|
|
|
label.SetPosition(state.offsetX, state.offsetY)
|
|
label.Render(*state.prerenderdestination)
|
|
}
|
|
}
|
|
|
|
// bindManager binds the button to the UI manager
|
|
func (v *Button) bindManager(manager *UIManager) {
|
|
v.manager = manager
|
|
}
|
|
|
|
// OnActivated defines the callback handler for the activate event
|
|
func (v *Button) OnActivated(callback func()) {
|
|
v.onClick = callback
|
|
}
|
|
|
|
// Activate calls the on activated callback handler, if any
|
|
func (v *Button) Activate() {
|
|
if v.onClick == nil {
|
|
return
|
|
}
|
|
|
|
v.onClick()
|
|
}
|
|
|
|
// Render renders the button
|
|
func (v *Button) Render(target d2interface.Surface) error {
|
|
target.PushFilter(d2enum.FilterNearest)
|
|
defer target.Pop()
|
|
|
|
target.PushTranslation(v.x, v.y)
|
|
defer target.Pop()
|
|
|
|
switch {
|
|
case !v.enabled:
|
|
target.PushColor(d2util.Color(lightGreyAlpha75))
|
|
defer target.Pop()
|
|
target.Render(v.disabledSurface)
|
|
case v.toggled && v.pressed:
|
|
target.Render(v.pressedToggledSurface)
|
|
case v.pressed:
|
|
if v.buttonLayout.AllowFrameChange {
|
|
target.Render(v.pressedSurface)
|
|
} else {
|
|
target.Render(v.normalSurface)
|
|
}
|
|
case v.toggled:
|
|
target.Render(v.toggledSurface)
|
|
default:
|
|
target.Render(v.normalSurface)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Toggle negates the toggled state of the button
|
|
func (v *Button) Toggle() {
|
|
v.toggled = !v.toggled
|
|
}
|
|
|
|
// Advance advances the button state
|
|
func (v *Button) Advance(_ float64) error {
|
|
return nil
|
|
}
|
|
|
|
// GetEnabled returns the enabled state
|
|
func (v *Button) GetEnabled() bool {
|
|
return v.enabled
|
|
}
|
|
|
|
// SetEnabled sets the enabled state
|
|
func (v *Button) SetEnabled(enabled bool) {
|
|
v.enabled = enabled
|
|
}
|
|
|
|
// GetSize returns the size of the button
|
|
func (v *Button) GetSize() (width, height int) {
|
|
return v.width, v.height
|
|
}
|
|
|
|
// SetPosition moves the button
|
|
func (v *Button) SetPosition(x, y int) {
|
|
v.x = x
|
|
v.y = y
|
|
}
|
|
|
|
// GetPosition returns the location of the button
|
|
func (v *Button) GetPosition() (x, y int) {
|
|
return v.x, v.y
|
|
}
|
|
|
|
// GetVisible returns the visibility of the button
|
|
func (v *Button) GetVisible() bool {
|
|
return v.visible
|
|
}
|
|
|
|
// SetVisible sets the visibility of the button
|
|
func (v *Button) SetVisible(visible bool) {
|
|
v.visible = visible
|
|
}
|
|
|
|
// GetPressed returns the pressed state of the button
|
|
func (v *Button) GetPressed() bool {
|
|
return v.pressed
|
|
}
|
|
|
|
// SetPressed sets the pressed state of the button
|
|
func (v *Button) SetPressed(pressed bool) {
|
|
v.pressed = pressed
|
|
}
|
|
|
|
func half(n int) int {
|
|
return n / 2
|
|
}
|