mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-23 17:32:02 +01:00
160 lines
4.1 KiB
Go
160 lines
4.1 KiB
Go
// Copyright 2020 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.
|
|
|
|
// +build js
|
|
|
|
package monogame
|
|
|
|
import (
|
|
"reflect"
|
|
"runtime"
|
|
"syscall/js"
|
|
"unsafe"
|
|
|
|
"github.com/hajimehoshi/ebiten/internal/affine"
|
|
"github.com/hajimehoshi/ebiten/internal/driver"
|
|
)
|
|
|
|
// TODO: This implementation depends on some C# files that are not uploaded yet.
|
|
// Create 'ebitenmonogame' command to generate C# project for the MonoGame.
|
|
|
|
// namespace is C# namespace.
|
|
//
|
|
// This is overwritten by -ldflags='-X github.com/hajimehoshi/ebiten/internal/monogame.namespace=NAMESPACE'.
|
|
var namespace = "Go2DotNet.Example.Ebiten"
|
|
|
|
type UpdateDrawer interface {
|
|
Update() error
|
|
Draw() error
|
|
}
|
|
|
|
type Game struct {
|
|
binding js.Value
|
|
update js.Func
|
|
draw js.Func
|
|
}
|
|
|
|
func NewGame(ud UpdateDrawer) *Game {
|
|
update := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
|
return ud.Update()
|
|
})
|
|
|
|
draw := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
|
return ud.Draw()
|
|
})
|
|
|
|
v := js.Global().Get(".net").Get(namespace+".GameGoBinding").New(update, draw)
|
|
g := &Game{
|
|
binding: v,
|
|
update: update,
|
|
draw: draw,
|
|
}
|
|
runtime.SetFinalizer(g, (*Game).Dispose)
|
|
return g
|
|
}
|
|
|
|
func (g *Game) Dispose() {
|
|
runtime.SetFinalizer(g, nil)
|
|
g.update.Release()
|
|
g.draw.Release()
|
|
}
|
|
|
|
func (g *Game) Run() {
|
|
g.binding.Call("Run")
|
|
}
|
|
|
|
func (g *Game) NewRenderTarget2D(width, height int) *RenderTarget2D {
|
|
v := g.binding.Call("NewRenderTarget2D", width, height)
|
|
r := &RenderTarget2D{
|
|
v: v,
|
|
binding: g.binding,
|
|
}
|
|
runtime.SetFinalizer(r, (*RenderTarget2D).Dispose)
|
|
return r
|
|
}
|
|
|
|
func (g *Game) SetVertices(vertices []float32, indices []uint16) {
|
|
var vs, is js.Value
|
|
{
|
|
h := (*reflect.SliceHeader)(unsafe.Pointer(&vertices))
|
|
h.Len *= 4
|
|
h.Cap *= 4
|
|
bs := *(*[]byte)(unsafe.Pointer(h))
|
|
runtime.KeepAlive(vertices)
|
|
vs = js.Global().Get("Uint8Array").New(len(bs))
|
|
js.CopyBytesToJS(vs, bs)
|
|
}
|
|
{
|
|
h := (*reflect.SliceHeader)(unsafe.Pointer(&indices))
|
|
h.Len *= 2
|
|
h.Cap *= 2
|
|
bs := *(*[]byte)(unsafe.Pointer(h))
|
|
runtime.KeepAlive(indices)
|
|
is = js.Global().Get("Uint8Array").New(len(bs))
|
|
js.CopyBytesToJS(is, bs)
|
|
}
|
|
g.binding.Call("SetVertices", vs, is)
|
|
}
|
|
|
|
func (g *Game) Draw(indexLen int, indexOffset int, mode driver.CompositeMode, colorM *affine.ColorM, filter driver.Filter, address driver.Address) {
|
|
src, dst := mode.Operations()
|
|
g.binding.Call("Draw", indexLen, indexOffset, int(src), int(dst))
|
|
}
|
|
|
|
func (g *Game) ResetDestination(viewportWidth, viewportHeight int) {
|
|
g.binding.Call("SetDestination", nil, viewportWidth, viewportHeight)
|
|
}
|
|
|
|
func (g *Game) IsKeyPressed(key driver.Key) bool {
|
|
// Pass a string of the key since both driver.Key value and XNA's key value are not reliable.
|
|
return g.binding.Call("IsKeyPressed", key.String()).Bool()
|
|
}
|
|
|
|
type RenderTarget2D struct {
|
|
v js.Value
|
|
binding js.Value
|
|
}
|
|
|
|
func (r *RenderTarget2D) Dispose() {
|
|
runtime.SetFinalizer(r, nil)
|
|
r.binding.Call("Dispose", r.v)
|
|
}
|
|
|
|
func (r *RenderTarget2D) Pixels(width, height int) ([]byte, error) {
|
|
v := r.binding.Call("Pixels", r.v, width, height)
|
|
bs := make([]byte, v.Length())
|
|
js.CopyBytesToGo(bs, v)
|
|
return bs, nil
|
|
}
|
|
|
|
func (r *RenderTarget2D) ReplacePixels(args []*driver.ReplacePixelsArgs) {
|
|
for _, a := range args {
|
|
arr := js.Global().Get("Uint8Array").New(len(a.Pixels))
|
|
js.CopyBytesToJS(arr, a.Pixels)
|
|
r.binding.Call("ReplacePixels", r.v, arr, a.X, a.Y, a.Width, a.Height)
|
|
}
|
|
}
|
|
|
|
func (r *RenderTarget2D) SetAsDestination(viewportWidth, viewportHeight int) {
|
|
r.binding.Call("SetDestination", r.v, viewportWidth, viewportHeight)
|
|
}
|
|
|
|
func (r *RenderTarget2D) SetAsSource() {
|
|
r.binding.Call("SetSource", r.v)
|
|
}
|
|
|
|
func (r *RenderTarget2D) IsScreen() bool {
|
|
return false
|
|
}
|