2018-03-01 17:30:18 +01:00
|
|
|
// Copyright 2018 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.
|
|
|
|
|
2019-05-01 10:54:08 +02:00
|
|
|
// +build js
|
2018-03-14 17:26:21 +01:00
|
|
|
// +build example jsgo
|
2018-03-01 17:30:18 +01:00
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2018-03-14 17:16:41 +01:00
|
|
|
"bytes"
|
2018-03-01 17:30:18 +01:00
|
|
|
"fmt"
|
2018-03-14 17:16:41 +01:00
|
|
|
"image"
|
2018-03-01 17:30:18 +01:00
|
|
|
_ "image/jpeg"
|
|
|
|
"log"
|
|
|
|
"math"
|
2019-04-30 19:15:28 +02:00
|
|
|
"syscall/js"
|
2019-01-29 17:44:57 +01:00
|
|
|
"time"
|
2018-03-01 17:30:18 +01:00
|
|
|
|
|
|
|
"github.com/hajimehoshi/ebiten"
|
|
|
|
"github.com/hajimehoshi/ebiten/ebitenutil"
|
2018-03-14 17:16:41 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/examples/resources/images"
|
2018-03-01 17:30:18 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/inpututil"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
screenWidth = 320
|
|
|
|
screenHeight = 240
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
gophersImage *ebiten.Image
|
2018-03-18 19:16:52 +01:00
|
|
|
extraImages []*ebiten.Image
|
2018-03-01 17:30:18 +01:00
|
|
|
)
|
|
|
|
|
2020-04-12 12:12:57 +02:00
|
|
|
type Game struct {
|
|
|
|
count int
|
|
|
|
lost bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Game) loseAndRestoreContext(context js.Value) {
|
|
|
|
if g.lost {
|
2019-01-29 17:44:57 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Edge might not support the extension. See
|
|
|
|
// https://developer.mozilla.org/en-US/docs/Web/API/WEBGL_lose_context
|
|
|
|
ext := context.Call("getExtension", "WEBGL_lose_context")
|
2020-08-16 16:05:33 +02:00
|
|
|
if !ext.Truthy() {
|
2019-01-29 17:44:57 +01:00
|
|
|
fmt.Println("Fail to force context lost. Edge might not support the extension yet.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ext.Call("loseContext")
|
|
|
|
fmt.Println("Lost the context!")
|
|
|
|
fmt.Println("The context is automatically restored after 3 seconds.")
|
2020-04-12 12:12:57 +02:00
|
|
|
g.lost = true
|
2019-01-29 17:44:57 +01:00
|
|
|
|
|
|
|
// If and only if the context is lost by loseContext, you need to call restoreContext. Note that in usual
|
|
|
|
// case of context lost, you cannot call restoreContext but the context should be restored automatically.
|
|
|
|
//
|
|
|
|
// After the context is lost, update will not be called. Instead, fire the goroutine to restore the context.
|
|
|
|
go func() {
|
|
|
|
time.Sleep(3 * time.Second)
|
|
|
|
ext.Call("restoreContext")
|
|
|
|
fmt.Println("Restored the context!")
|
2020-04-12 12:12:57 +02:00
|
|
|
g.lost = false
|
2019-01-29 17:44:57 +01:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-04-12 12:12:57 +02:00
|
|
|
func (g *Game) Update(screen *ebiten.Image) error {
|
2019-05-01 10:54:08 +02:00
|
|
|
if inpututil.IsKeyJustPressed(ebiten.KeySpace) {
|
2018-06-29 17:02:15 +02:00
|
|
|
doc := js.Global().Get("document")
|
2018-03-01 17:30:18 +01:00
|
|
|
canvas := doc.Call("getElementsByTagName", "canvas").Index(0)
|
2020-03-28 18:03:14 +01:00
|
|
|
context := canvas.Call("getContext", "webgl2")
|
2020-08-16 16:05:33 +02:00
|
|
|
if !context.Truthy() {
|
2020-03-28 18:03:14 +01:00
|
|
|
context = canvas.Call("getContext", "webgl")
|
2020-08-16 16:05:33 +02:00
|
|
|
if !context.Truthy() {
|
2020-03-28 18:03:14 +01:00
|
|
|
context = canvas.Call("getContext", "experimental-webgl")
|
|
|
|
}
|
2018-05-13 17:35:40 +02:00
|
|
|
}
|
2020-04-12 12:12:57 +02:00
|
|
|
g.loseAndRestoreContext(context)
|
2018-03-01 17:30:18 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-18 18:23:27 +02:00
|
|
|
if inpututil.IsKeyJustPressed(ebiten.KeyS) {
|
|
|
|
ebiten.SetClearingScreenSkipped(!ebiten.IsClearingScreenSkipped())
|
|
|
|
}
|
|
|
|
|
2020-04-12 12:12:57 +02:00
|
|
|
g.count++
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Game) Draw(screen *ebiten.Image) {
|
2020-08-19 14:15:47 +02:00
|
|
|
if g.lost {
|
|
|
|
// When the context is lost, skip rendering. Usually this logic should not be required, but when the
|
|
|
|
// context lost happens by the API explicitly, Draw can be called even after the data in GPU is
|
|
|
|
// disappered.
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-03-01 17:30:18 +01:00
|
|
|
w, h := gophersImage.Size()
|
|
|
|
op := &ebiten.DrawImageOptions{}
|
|
|
|
|
|
|
|
// For the details, see examples/rotate.
|
|
|
|
op.GeoM.Translate(-float64(w)/2, -float64(h)/2)
|
2020-04-12 12:12:57 +02:00
|
|
|
op.GeoM.Rotate(float64(g.count%360) * 2 * math.Pi / 360)
|
2018-03-01 17:30:18 +01:00
|
|
|
op.GeoM.Translate(screenWidth/2, screenHeight/2)
|
|
|
|
screen.DrawImage(gophersImage, op)
|
|
|
|
|
2020-08-18 18:23:27 +02:00
|
|
|
msg := `Press Space to force to lose/restore the GL context!
|
|
|
|
(Browser only)
|
|
|
|
|
|
|
|
Press S to switch clearing the screen
|
|
|
|
at the beginning of each frame.`
|
|
|
|
ebitenutil.DebugPrint(screen, msg)
|
2020-04-12 12:12:57 +02:00
|
|
|
}
|
2018-03-01 17:30:18 +01:00
|
|
|
|
2020-04-12 12:12:57 +02:00
|
|
|
func (g *Game) Layout(outsideWidth, outsideHeight int) (int, int) {
|
|
|
|
return screenWidth, screenHeight
|
2018-03-01 17:30:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
2018-03-16 04:05:53 +01:00
|
|
|
// Decode image from a byte slice instead of a file so that
|
|
|
|
// this example works in any working directory.
|
|
|
|
// If you want to use a file, there are some options:
|
|
|
|
// 1) Use os.Open and pass the file to the image decoder.
|
|
|
|
// This is a very regular way, but doesn't work on browsers.
|
|
|
|
// 2) Use ebitenutil.OpenFile and pass the file to the image decoder.
|
|
|
|
// This works even on browsers.
|
|
|
|
// 3) Use ebitenutil.NewImageFromFile to create an ebiten.Image directly from a file.
|
|
|
|
// This also works on browsers.
|
2018-03-14 17:16:41 +01:00
|
|
|
img, _, err := image.Decode(bytes.NewReader(images.Gophers_jpg))
|
2018-03-01 17:30:18 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2018-03-14 17:16:41 +01:00
|
|
|
gophersImage, _ = ebiten.NewImageFromImage(img, ebiten.FilterDefault)
|
|
|
|
|
2018-03-18 19:16:52 +01:00
|
|
|
// Extend the shared backend GL texture on purpose.
|
|
|
|
for i := 0; i < 20; i++ {
|
|
|
|
eimg, _ := ebiten.NewImageFromImage(img, ebiten.FilterDefault)
|
|
|
|
extraImages = append(extraImages, eimg)
|
|
|
|
}
|
|
|
|
|
2020-04-12 12:12:57 +02:00
|
|
|
ebiten.SetWindowSize(screenWidth*2, screenHeight*2)
|
|
|
|
ebiten.SetWindowTitle("Context Lost (Ebiten Demo)")
|
|
|
|
if err := ebiten.RunGame(&Game{}); err != nil {
|
2018-03-01 17:30:18 +01:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|