mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-11 19:48:54 +01:00
examples/drag: Add touches (#506)
This commit is contained in:
parent
5677c8a916
commit
252f4430d8
@ -23,6 +23,7 @@ import (
|
||||
_ "image/png"
|
||||
"log"
|
||||
"math/rand"
|
||||
"time"
|
||||
|
||||
"github.com/hajimehoshi/ebiten"
|
||||
"github.com/hajimehoshi/ebiten/ebitenutil"
|
||||
@ -30,6 +31,10 @@ import (
|
||||
"github.com/hajimehoshi/ebiten/inpututil"
|
||||
)
|
||||
|
||||
func init() {
|
||||
rand.Seed(time.Now().UnixNano())
|
||||
}
|
||||
|
||||
const (
|
||||
screenWidth = 320
|
||||
screenHeight = 240
|
||||
@ -82,63 +87,104 @@ func (s *Sprite) Draw(screen *ebiten.Image, dx, dy int, alpha float64) {
|
||||
screen.DrawImage(s.image, op)
|
||||
}
|
||||
|
||||
type DragPhase int
|
||||
// StrokeSource represents a input device to provide strokes.
|
||||
type StrokeSource interface {
|
||||
Position() (int, int)
|
||||
IsJustReleased() bool
|
||||
}
|
||||
|
||||
const (
|
||||
DragPhaseNone DragPhase = iota
|
||||
DragPhaseStart
|
||||
DragPhaseDrag
|
||||
DragPhaseEnd
|
||||
)
|
||||
// MouseStrokeSource is a StrokeSource implementation of mouse.
|
||||
type MouseStrokeSource struct{}
|
||||
|
||||
// DragState manages the current drag state.
|
||||
type DragState struct {
|
||||
phase DragPhase
|
||||
func (m *MouseStrokeSource) Position() (int, int) {
|
||||
return ebiten.CursorPosition()
|
||||
}
|
||||
|
||||
func (m *MouseStrokeSource) IsJustReleased() bool {
|
||||
return inpututil.IsMouseButtonJustReleased(ebiten.MouseButtonLeft)
|
||||
}
|
||||
|
||||
// TouchStrokeSource is a StrokeSource implementation of touch.
|
||||
type TouchStrokeSource struct {
|
||||
ID int
|
||||
}
|
||||
|
||||
func (t *TouchStrokeSource) Position() (int, int) {
|
||||
return ebiten.TouchPosition(t.ID)
|
||||
}
|
||||
|
||||
func (t *TouchStrokeSource) IsJustReleased() bool {
|
||||
return inpututil.IsTouchJustReleased(t.ID)
|
||||
}
|
||||
|
||||
// Stroke manages the current drag state by mouse.
|
||||
type Stroke struct {
|
||||
source StrokeSource
|
||||
|
||||
// initX and initY represents the position when dragging starts.
|
||||
// initX and initY values don't make sense when phase is DragPhaseNone.
|
||||
initX int
|
||||
initY int
|
||||
|
||||
// currentX and currentY represents the current position
|
||||
// initX and initY values don't make sense when phase is DragPhaseNone.
|
||||
currentX int
|
||||
currentY int
|
||||
|
||||
released bool
|
||||
|
||||
// draggingObject represents a object (sprite in this case)
|
||||
// that is being dragged.
|
||||
draggingObject interface{}
|
||||
}
|
||||
|
||||
func (d *DragState) Update() {
|
||||
switch d.phase {
|
||||
case DragPhaseNone:
|
||||
if inpututil.IsMouseButtonJustPressed(ebiten.MouseButtonLeft) {
|
||||
cx, cy := ebiten.CursorPosition()
|
||||
d.phase = DragPhaseStart
|
||||
d.initX = cx
|
||||
d.initY = cy
|
||||
d.currentX = cx
|
||||
d.currentY = cy
|
||||
func NewStroke(source StrokeSource) *Stroke {
|
||||
cx, cy := source.Position()
|
||||
return &Stroke{
|
||||
source: source,
|
||||
initX: cx,
|
||||
initY: cy,
|
||||
currentX: cx,
|
||||
currentY: cy,
|
||||
}
|
||||
case DragPhaseStart:
|
||||
d.phase = DragPhaseDrag
|
||||
case DragPhaseDrag:
|
||||
x, y := ebiten.CursorPosition()
|
||||
d.currentX = x
|
||||
d.currentY = y
|
||||
if inpututil.IsMouseButtonJustReleased(ebiten.MouseButtonLeft) {
|
||||
d.phase = DragPhaseEnd
|
||||
}
|
||||
case DragPhaseEnd:
|
||||
d.phase = DragPhaseNone
|
||||
|
||||
func (s *Stroke) Update() {
|
||||
if s.released {
|
||||
return
|
||||
}
|
||||
if s.source.IsJustReleased() {
|
||||
s.released = true
|
||||
return
|
||||
}
|
||||
x, y := s.source.Position()
|
||||
s.currentX = x
|
||||
s.currentY = y
|
||||
}
|
||||
|
||||
func (s *Stroke) IsReleased() bool {
|
||||
return s.released
|
||||
}
|
||||
|
||||
func (s *Stroke) Position() (int, int) {
|
||||
return s.currentX, s.currentY
|
||||
}
|
||||
|
||||
func (s *Stroke) PositionDiff() (int, int) {
|
||||
dx := s.currentX - s.initX
|
||||
dy := s.currentY - s.initY
|
||||
return dx, dy
|
||||
}
|
||||
|
||||
func (s *Stroke) DraggingObject() interface{} {
|
||||
return s.draggingObject
|
||||
}
|
||||
|
||||
func (s *Stroke) SetDraggingObject(object interface{}) {
|
||||
s.draggingObject = object
|
||||
}
|
||||
|
||||
type Game struct {
|
||||
dragState DragState
|
||||
strokes map[*Stroke]struct{}
|
||||
sprites []*Sprite
|
||||
|
||||
// draggingSpriteIndex represents the index of the sprites
|
||||
// that is being dragged. If draggingSpriteIndex is -1,
|
||||
// there is not such sprite.
|
||||
draggingSpriteIndex int
|
||||
}
|
||||
|
||||
var theGame *Game
|
||||
@ -173,40 +219,67 @@ func init() {
|
||||
|
||||
// Initialize the game.
|
||||
theGame = &Game{
|
||||
strokes: map[*Stroke]struct{}{},
|
||||
sprites: sprites,
|
||||
draggingSpriteIndex: -1,
|
||||
}
|
||||
}
|
||||
|
||||
func (g *Game) update(screen *ebiten.Image) error {
|
||||
g.dragState.Update()
|
||||
switch g.dragState.phase {
|
||||
case DragPhaseStart:
|
||||
if g.draggingSpriteIndex == -1 {
|
||||
func (g *Game) spriteAt(x, y int) *Sprite {
|
||||
// As the sprites are ordered from back to front,
|
||||
// search the clicked/touched sprite in reverse order.
|
||||
for i := len(g.sprites) - 1; i >= 0; i-- {
|
||||
s := g.sprites[i]
|
||||
if s.In(g.dragState.initX, g.dragState.initY) {
|
||||
g.draggingSpriteIndex = i
|
||||
if s.In(x, y) {
|
||||
return s
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (g *Game) updateStroke(stroke *Stroke) {
|
||||
stroke.Update()
|
||||
if !stroke.IsReleased() {
|
||||
return
|
||||
}
|
||||
|
||||
s := stroke.DraggingObject().(*Sprite)
|
||||
if s == nil {
|
||||
return
|
||||
}
|
||||
|
||||
s.MoveBy(stroke.PositionDiff())
|
||||
|
||||
index := -1
|
||||
for i, ss := range g.sprites {
|
||||
if ss == s {
|
||||
index = i
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
case DragPhaseEnd:
|
||||
if g.draggingSpriteIndex != -1 {
|
||||
dx := g.dragState.currentX - g.dragState.initX
|
||||
dy := g.dragState.currentY - g.dragState.initY
|
||||
g.sprites[g.draggingSpriteIndex].MoveBy(dx, dy)
|
||||
|
||||
// Move the dragged sprite to the front.
|
||||
s := g.sprites[g.draggingSpriteIndex]
|
||||
g.sprites = append(
|
||||
g.sprites[:g.draggingSpriteIndex],
|
||||
g.sprites[g.draggingSpriteIndex+1:]...)
|
||||
g.sprites = append(g.sprites[:index], g.sprites[index+1:]...)
|
||||
g.sprites = append(g.sprites, s)
|
||||
|
||||
g.draggingSpriteIndex = -1
|
||||
stroke.SetDraggingObject(nil)
|
||||
}
|
||||
|
||||
func (g *Game) update(screen *ebiten.Image) error {
|
||||
if inpututil.IsMouseButtonJustPressed(ebiten.MouseButtonLeft) {
|
||||
s := NewStroke(&MouseStrokeSource{})
|
||||
s.SetDraggingObject(g.spriteAt(s.Position()))
|
||||
g.strokes[s] = struct{}{}
|
||||
}
|
||||
for _, id := range inpututil.JustPressedTouchIDs() {
|
||||
s := NewStroke(&TouchStrokeSource{id})
|
||||
s.SetDraggingObject(g.spriteAt(s.Position()))
|
||||
g.strokes[s] = struct{}{}
|
||||
}
|
||||
|
||||
for s := range g.strokes {
|
||||
g.updateStroke(s)
|
||||
if s.IsReleased() {
|
||||
delete(g.strokes, s)
|
||||
}
|
||||
}
|
||||
|
||||
@ -214,19 +287,19 @@ func (g *Game) update(screen *ebiten.Image) error {
|
||||
return nil
|
||||
}
|
||||
|
||||
for i, s := range g.sprites {
|
||||
if i == g.draggingSpriteIndex {
|
||||
s.Draw(screen, 0, 0, 0.5)
|
||||
ss := map[*Sprite]*Stroke{}
|
||||
for s := range g.strokes {
|
||||
ss[s.DraggingObject().(*Sprite)] = s
|
||||
}
|
||||
|
||||
for _, s := range g.sprites {
|
||||
if stroke, ok := ss[s]; ok {
|
||||
dx, dy := stroke.PositionDiff()
|
||||
s.Draw(screen, dx, dy, 0.5)
|
||||
} else {
|
||||
s.Draw(screen, 0, 0, 1)
|
||||
}
|
||||
}
|
||||
if g.draggingSpriteIndex != -1 {
|
||||
s := g.sprites[g.draggingSpriteIndex]
|
||||
dx := g.dragState.currentX - g.dragState.initX
|
||||
dy := g.dragState.currentY - g.dragState.initY
|
||||
s.Draw(screen, dx, dy, 1)
|
||||
}
|
||||
|
||||
ebitenutil.DebugPrint(screen, "Drag & Drop the sprites!")
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user