Improve and optimization

This commit is contained in:
2024-07-18 22:59:03 +08:00
parent 1610584ce4
commit 1b3a9833eb
9 changed files with 110 additions and 53 deletions

View File

@@ -1,7 +1,15 @@
package land
type UserEventType = int
const (
UserEventMouseDown = UserEventType(iota)
UserEventMouseUp
UserEventMouseMove
)
type InteractableObject interface {
GetPosition() Vector2D
GetSize() Vector2D
OnClick(mousePos Vector2D)
OnEvent(event UserEventType, args ...any)
}

View File

@@ -2,20 +2,20 @@ package land
import "github.com/veandco/go-sdl2/sdl"
type BaseUIObject struct {
type BaseWidget struct {
BaseObject
Position Vector2D
Size Vector2D
}
func (p *BaseUIObject) Draw(pen *sdl.Renderer) {
func (p *BaseWidget) Draw(pen *sdl.Renderer) {
}
func (p *BaseUIObject) GetPosition() Vector2D {
func (p *BaseWidget) GetPosition() Vector2D {
return p.Position
}
func (p *BaseUIObject) GetSize() Vector2D {
func (p *BaseWidget) GetSize() Vector2D {
return p.Size
}

View File

@@ -0,0 +1,10 @@
package renderer
import (
"github.com/veandco/go-sdl2/gfx"
"github.com/veandco/go-sdl2/sdl"
)
func FillRoundedRect(pen *sdl.Renderer, rect *sdl.Rect, color sdl.Color, radius int32) {
gfx.RoundedRectangleColor(pen, rect.X, rect.Y, rect.X+rect.W, rect.Y+rect.H, radius, color)
}

View File

@@ -51,18 +51,28 @@ func (p *RootObject) ForEachChildren(cb func(child Object)) {
func (p *RootObject) HandleUserEvent(event sdl.Event) {
switch event := event.(type) {
case *sdl.MouseButtonEvent:
if event.Type == sdl.MOUSEBUTTONDOWN {
x, y := float64(event.X), float64(event.Y)
vec := Vector2D{x, y}
p.ForEachChildren(func(child Object) {
if interChild, ok := child.(InteractableObject); ok {
isOverlap := IsOverlapWithPoint(child.(PositionedObject), vec)
if isOverlap {
interChild.OnClick(vec)
x, y := float64(event.X), float64(event.Y)
vec := Vector2D{x, y}
p.ForEachChildren(func(child Object) {
if interChild, ok := child.(InteractableObject); ok {
if IsOverlapWithPoint(child.(PositionedObject), vec) {
if event.Type == sdl.MOUSEBUTTONDOWN {
interChild.OnEvent(UserEventMouseDown, vec)
} else if event.Type == sdl.MOUSEBUTTONUP {
interChild.OnEvent(UserEventMouseUp, vec)
}
}
})
}
}
})
case *sdl.MouseMotionEvent:
x, y := float64(event.X), float64(event.Y)
vec := Vector2D{x, y}
p.ForEachChildren(func(child Object) {
if interChild, ok := child.(InteractableObject); ok {
isOverlap := IsOverlapWithPoint(child.(PositionedObject), vec)
interChild.OnEvent(UserEventMouseMove, vec, isOverlap)
}
})
}
}

View File

@@ -2,36 +2,51 @@ package ui
import (
"git.solsynth.dev/highland/codingland/pkg/internal/land"
"git.solsynth.dev/highland/codingland/pkg/internal/land/renderer"
"github.com/veandco/go-sdl2/sdl"
"github.com/veandco/go-sdl2/ttf"
"log"
)
type ButtonObject struct {
land.BaseUIObject
type ButtonWidget struct {
land.BaseWidget
Label string
Font *ttf.Font
OnClickEvent func(mousePos land.Vector2D)
Label string
Font *ttf.Font
OnClick func(mousePos land.Vector2D)
isHovering bool
isClicking bool
}
func (p *ButtonObject) Draw(pen *sdl.Renderer) {
func (p *ButtonWidget) Draw(pen *sdl.Renderer) {
var borderColor, bgColor sdl.Color
if p.isClicking {
borderColor = sdl.Color{R: 200, G: 0, B: 0, A: 255}
bgColor = sdl.Color{R: 255, G: 0, B: 0, A: 255}
} else if p.isHovering {
borderColor = sdl.Color{R: 0, G: 200, B: 0, A: 255}
bgColor = sdl.Color{R: 0, G: 255, B: 0, A: 255}
} else {
borderColor = sdl.Color{R: 255, G: 255, B: 255, A: 255}
bgColor = sdl.Color{R: 0, G: 0, B: 0, A: 255}
}
var borderWidth int32 = 2
pen.SetDrawColor(255, 255, 255, 255)
pen.FillRect(&sdl.Rect{
pen.SetDrawColor(borderColor.R, borderColor.G, borderColor.B, borderColor.A)
renderer.FillRoundedRect(pen, &sdl.Rect{
X: int32(p.Position.X),
Y: int32(p.Position.Y),
W: int32(p.Size.X),
H: int32(p.Size.Y),
})
pen.SetDrawColor(0, 0, 0, 255)
pen.FillRect(&sdl.Rect{
}, borderColor, 16)
renderer.FillRoundedRect(pen, &sdl.Rect{
X: int32(p.Position.X) + borderWidth/2,
Y: int32(p.Position.Y) + borderWidth/2,
W: int32(p.Size.X) - borderWidth,
H: int32(p.Size.Y) - borderWidth,
})
}, bgColor, 16)
surface, err := p.Font.RenderUTF8Blended(p.Label, sdl.Color{R: 255, G: 255, B: 255, A: 255})
if err != nil {
@@ -47,15 +62,22 @@ func (p *ButtonObject) Draw(pen *sdl.Renderer) {
}
defer texture.Destroy()
textRect := sdl.Rect{
X: int32(p.BaseUIObject.Position.X),
Y: int32(p.BaseUIObject.Position.Y),
W: surface.W,
H: surface.H,
}
textWidth := surface.W
textHeight := surface.H
textX := int32(p.BaseWidget.Position.X) + (int32(p.BaseWidget.Size.X)-textWidth)/2
textY := int32(p.BaseWidget.Position.Y) + (int32(p.BaseWidget.Size.Y)-textHeight)/2
textRect := sdl.Rect{X: textX, Y: textY, W: textWidth, H: textHeight}
pen.Copy(texture, nil, &textRect)
}
func (p *ButtonObject) OnClick(mousePos land.Vector2D) {
p.OnClickEvent(mousePos)
func (p *ButtonWidget) OnEvent(event land.UserEventType, args ...any) {
switch event {
case land.UserEventMouseDown:
p.isClicking = true
p.OnClick(args[0].(land.Vector2D))
case land.UserEventMouseUp:
p.isClicking = false
case land.UserEventMouseMove:
p.isHovering = args[1].(bool)
}
}

View File

@@ -1,6 +1,8 @@
package land
import "fmt"
import (
"fmt"
)
type Vector2D struct {
X, Y float64