2019-02-23 17:20:03 +01:00
|
|
|
// Copyright 2019 The Ebiten Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"container/list"
|
|
|
|
"fmt"
|
|
|
|
"image"
|
|
|
|
"image/color"
|
|
|
|
_ "image/png"
|
|
|
|
"log"
|
|
|
|
"math"
|
2024-09-11 18:03:53 +02:00
|
|
|
"math/rand/v2"
|
2019-02-23 17:20:03 +01:00
|
|
|
|
2020-10-03 19:35:13 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/ebitenutil"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/examples/resources/images"
|
2019-02-23 17:20:03 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
screenWidth = 640
|
|
|
|
screenHeight = 480
|
|
|
|
)
|
|
|
|
|
|
|
|
var smokeImage *ebiten.Image
|
|
|
|
|
|
|
|
func init() {
|
2021-09-20 08:22:46 +02:00
|
|
|
// Decode an image from the image file's byte slice.
|
2019-02-23 17:20:03 +01:00
|
|
|
img, _, err := image.Decode(bytes.NewReader(images.Smoke_png))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2020-10-05 18:03:30 +02:00
|
|
|
smokeImage = ebiten.NewImageFromImage(img)
|
2019-02-23 17:20:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type sprite struct {
|
|
|
|
count int
|
|
|
|
maxCount int
|
2019-02-23 18:09:38 +01:00
|
|
|
dir float64
|
2019-02-23 17:20:03 +01:00
|
|
|
|
|
|
|
img *ebiten.Image
|
|
|
|
scale float64
|
2019-02-23 18:09:38 +01:00
|
|
|
angle float64
|
2022-11-01 18:43:42 +01:00
|
|
|
alpha float32
|
2019-02-23 17:20:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *sprite) update() {
|
|
|
|
if s.count == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.count--
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *sprite) terminated() bool {
|
|
|
|
return s.count == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *sprite) draw(screen *ebiten.Image) {
|
|
|
|
if s.count == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
ox = screenWidth / 2
|
|
|
|
oy = screenHeight / 2
|
|
|
|
)
|
2019-02-23 18:09:38 +01:00
|
|
|
x := math.Cos(s.dir) * float64(s.maxCount-s.count)
|
|
|
|
y := math.Sin(s.dir) * float64(s.maxCount-s.count)
|
2019-02-23 17:20:03 +01:00
|
|
|
|
|
|
|
op := &ebiten.DrawImageOptions{}
|
|
|
|
|
2023-01-19 15:42:35 +01:00
|
|
|
sx, sy := s.img.Bounds().Dx(), s.img.Bounds().Dy()
|
2019-02-23 17:20:03 +01:00
|
|
|
op.GeoM.Translate(-float64(sx)/2, -float64(sy)/2)
|
2019-02-23 18:09:38 +01:00
|
|
|
op.GeoM.Rotate(s.angle)
|
2019-02-23 17:20:03 +01:00
|
|
|
op.GeoM.Scale(s.scale, s.scale)
|
|
|
|
op.GeoM.Translate(x, y)
|
|
|
|
op.GeoM.Translate(ox, oy)
|
|
|
|
|
2022-11-01 18:43:42 +01:00
|
|
|
rate := float32(s.count) / float32(s.maxCount)
|
|
|
|
var alpha float32
|
2019-02-23 17:20:03 +01:00
|
|
|
if rate < 0.2 {
|
|
|
|
alpha = rate / 0.2
|
|
|
|
} else if rate > 0.8 {
|
|
|
|
alpha = (1 - rate) / 0.2
|
|
|
|
} else {
|
|
|
|
alpha = 1
|
|
|
|
}
|
|
|
|
alpha *= s.alpha
|
2022-11-01 18:43:42 +01:00
|
|
|
op.ColorScale.ScaleAlpha(alpha)
|
2019-02-23 17:20:03 +01:00
|
|
|
|
|
|
|
screen.DrawImage(s.img, op)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newSprite(img *ebiten.Image) *sprite {
|
2024-09-11 18:03:53 +02:00
|
|
|
c := rand.IntN(50) + 300
|
2019-02-23 18:09:38 +01:00
|
|
|
dir := rand.Float64() * 2 * math.Pi
|
2019-02-23 17:20:03 +01:00
|
|
|
a := rand.Float64() * 2 * math.Pi
|
|
|
|
s := rand.Float64()*0.1 + 0.4
|
|
|
|
return &sprite{
|
|
|
|
img: img,
|
|
|
|
|
|
|
|
maxCount: c,
|
|
|
|
count: c,
|
2019-02-23 18:09:38 +01:00
|
|
|
dir: dir,
|
|
|
|
|
|
|
|
angle: a,
|
|
|
|
scale: s,
|
|
|
|
alpha: 0.5,
|
2019-02-23 17:20:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-14 20:51:40 +02:00
|
|
|
type Game struct {
|
|
|
|
sprites *list.List
|
|
|
|
}
|
|
|
|
|
2020-10-04 10:42:54 +02:00
|
|
|
func (g *Game) Update() error {
|
2020-05-14 20:51:40 +02:00
|
|
|
if g.sprites == nil {
|
|
|
|
g.sprites = list.New()
|
|
|
|
}
|
2019-02-23 17:20:03 +01:00
|
|
|
|
2024-09-11 18:03:53 +02:00
|
|
|
if g.sprites.Len() < 500 && rand.IntN(4) < 3 {
|
2019-02-23 17:20:03 +01:00
|
|
|
// Emit
|
2020-05-14 20:51:40 +02:00
|
|
|
g.sprites.PushBack(newSprite(smokeImage))
|
2019-02-23 17:20:03 +01:00
|
|
|
}
|
|
|
|
|
2020-05-14 20:51:40 +02:00
|
|
|
for e := g.sprites.Front(); e != nil; e = e.Next() {
|
2019-02-23 17:20:03 +01:00
|
|
|
s := e.Value.(*sprite)
|
|
|
|
s.update()
|
|
|
|
if s.terminated() {
|
2020-05-14 20:51:40 +02:00
|
|
|
defer g.sprites.Remove(e)
|
2019-02-23 17:20:03 +01:00
|
|
|
}
|
|
|
|
}
|
2020-05-14 20:51:40 +02:00
|
|
|
return nil
|
|
|
|
}
|
2019-02-23 17:20:03 +01:00
|
|
|
|
2020-05-14 20:51:40 +02:00
|
|
|
func (g *Game) Draw(screen *ebiten.Image) {
|
2019-02-23 17:20:03 +01:00
|
|
|
screen.Fill(color.RGBA{0x99, 0xcc, 0xff, 0xff})
|
2020-05-14 20:51:40 +02:00
|
|
|
for e := g.sprites.Front(); e != nil; e = e.Next() {
|
2019-02-23 17:20:03 +01:00
|
|
|
s := e.Value.(*sprite)
|
|
|
|
s.draw(screen)
|
|
|
|
}
|
|
|
|
|
2022-07-17 04:25:45 +02:00
|
|
|
ebitenutil.DebugPrint(screen, fmt.Sprintf("TPS: %0.2f\nSprites: %d", ebiten.ActualTPS(), g.sprites.Len()))
|
2020-05-14 20:51:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Game) Layout(outsideWidth, outsideHeight int) (int, int) {
|
|
|
|
return screenWidth, screenHeight
|
2019-02-23 17:20:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
2020-05-14 20:51:40 +02:00
|
|
|
ebiten.SetWindowSize(screenWidth, screenHeight)
|
2022-08-29 04:16:39 +02:00
|
|
|
ebiten.SetWindowTitle("Particles (Ebitengine Demo)")
|
2020-05-14 20:51:40 +02:00
|
|
|
if err := ebiten.RunGame(&Game{}); err != nil {
|
2019-02-23 17:20:03 +01:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|