ebiten/ui.go

207 lines
4.3 KiB
Go
Raw Normal View History

// Copyright 2014 Hajime Hoshi
//
// 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.
2014-12-09 15:16:04 +01:00
2014-12-14 08:53:32 +01:00
package ebiten
2014-12-05 18:26:02 +01:00
import (
"fmt"
2014-12-05 18:26:02 +01:00
glfw "github.com/go-gl/glfw3"
"github.com/hajimehoshi/ebiten/internal/graphics"
2014-12-30 19:04:52 +01:00
"github.com/hajimehoshi/ebiten/internal/opengl"
2014-12-14 10:57:29 +01:00
"image"
"runtime"
2014-12-05 18:26:02 +01:00
)
var currentUI *ui
func init() {
runtime.LockOSThread()
2014-12-05 18:26:02 +01:00
2014-12-17 09:10:38 +01:00
glfw.SetErrorCallback(func(err glfw.ErrorCode, desc string) {
panic(fmt.Sprintf("%v: %v\n", err, desc))
})
2014-12-05 18:26:02 +01:00
if !glfw.Init() {
panic("glfw.Init() fails")
2014-12-05 18:26:02 +01:00
}
glfw.WindowHint(glfw.Visible, glfw.False)
2014-12-05 18:26:02 +01:00
glfw.WindowHint(glfw.Resizable, glfw.False)
window, err := glfw.CreateWindow(16, 16, "", nil, nil)
2014-12-07 20:22:50 +01:00
if err != nil {
panic(err)
2014-12-07 20:22:50 +01:00
}
currentUI = &ui{
2014-12-17 09:10:38 +01:00
window: window,
funcs: make(chan func()),
2014-12-07 20:22:50 +01:00
}
currentUI.run()
currentUI.use(func() {
2014-12-30 19:04:52 +01:00
currentUI.glContext = opengl.NewContext()
glfw.SwapInterval(1)
})
}
type ui struct {
window *glfw.Window
scale int
2014-12-30 19:04:52 +01:00
glContext *opengl.Context
graphicsContext *graphicsContext
input input
funcs chan func()
}
func startUI(width, height, scale int, title string) error {
monitor, err := glfw.GetPrimaryMonitor()
if err != nil {
return err
}
videoMode, err := monitor.GetVideoMode()
if err != nil {
return err
}
x := (videoMode.Width - width*scale) / 2
y := (videoMode.Height - height*scale) / 3
ch := make(chan struct{})
window := currentUI.window
window.SetFramebufferSizeCallback(func(w *glfw.Window, width, height int) {
close(ch)
})
window.SetSize(width*scale, height*scale)
window.SetTitle(title)
window.SetPosition(x, y)
window.Show()
for {
done := false
glfw.PollEvents()
select {
case <-ch:
done = true
default:
}
if done {
break
}
}
ui := currentUI
ui.scale = scale
2014-12-14 10:57:29 +01:00
// For retina displays, recalculate the scale with the framebuffer size.
windowWidth, _ := window.GetFramebufferSize()
realScale := windowWidth / width
ui.use(func() {
2014-12-31 06:57:51 +01:00
ui.graphicsContext, err = newGraphicsContext(ui.glContext, width, height, realScale)
2014-12-14 10:57:29 +01:00
})
return err
2014-12-05 18:26:02 +01:00
}
2014-12-14 10:34:47 +01:00
func (u *ui) doEvents() {
2014-12-06 14:56:57 +01:00
glfw.PollEvents()
2014-12-14 10:57:29 +01:00
u.update()
2014-12-05 18:26:02 +01:00
}
2014-12-14 10:34:47 +01:00
func (u *ui) terminate() {
2014-12-05 18:26:02 +01:00
glfw.Terminate()
}
2014-12-10 02:42:47 +01:00
2014-12-14 10:34:47 +01:00
func (u *ui) isClosed() bool {
2014-12-14 10:57:29 +01:00
return u.window.ShouldClose()
2014-12-10 02:42:47 +01:00
}
2014-12-14 13:04:26 +01:00
func (u *ui) Sync(f func()) {
u.use(f)
}
func (u *ui) draw(f func(*Image) error) (err error) {
2014-12-14 10:57:29 +01:00
u.use(func() {
err = u.graphicsContext.preUpdate()
2014-12-14 10:57:29 +01:00
})
if err != nil {
return
}
err = f(&Image{syncer: u, inner: u.graphicsContext.screen})
if err != nil {
2014-12-14 10:57:29 +01:00
return
}
u.use(func() {
err = u.graphicsContext.postUpdate()
if err != nil {
return
}
2014-12-14 10:57:29 +01:00
u.window.SwapBuffers()
})
return
}
2014-12-22 20:32:36 +01:00
func (u *ui) newImageFromImage(img image.Image, filter Filter) (*Image, error) {
var innerImage *innerImage
2014-12-14 10:57:29 +01:00
var err error
u.use(func() {
var texture *graphics.Texture
2014-12-31 06:57:51 +01:00
texture, err = graphics.NewTextureFromImage(u.glContext, img, glFilter(u.glContext, filter))
2014-12-17 14:12:32 +01:00
if err != nil {
return
}
2014-12-31 07:22:15 +01:00
innerImage, err = newInnerImage(u.glContext, texture)
2014-12-14 10:57:29 +01:00
})
if err != nil {
return nil, err
}
return &Image{syncer: u, inner: innerImage}, nil
2014-12-14 10:57:29 +01:00
}
2014-12-22 20:32:36 +01:00
func (u *ui) newImage(width, height int, filter Filter) (*Image, error) {
var innerImage *innerImage
2014-12-14 10:57:29 +01:00
var err error
u.use(func() {
var texture *graphics.Texture
2014-12-31 06:57:51 +01:00
texture, err = graphics.NewTexture(u.glContext, width, height, glFilter(u.glContext, filter))
if err != nil {
return
}
2014-12-31 07:22:15 +01:00
innerImage, err = newInnerImage(u.glContext, texture)
2014-12-31 10:23:18 +01:00
innerImage.Clear(u.glContext)
2014-12-14 10:57:29 +01:00
})
if err != nil {
return nil, err
}
return &Image{syncer: u, inner: innerImage}, nil
2014-12-14 10:57:29 +01:00
}
func (u *ui) run() {
2014-12-14 10:57:29 +01:00
go func() {
runtime.LockOSThread()
u.window.MakeContextCurrent()
2014-12-17 09:10:38 +01:00
for f := range u.funcs {
f()
2014-12-14 10:57:29 +01:00
}
}()
}
func (u *ui) use(f func()) {
2014-12-17 09:10:38 +01:00
ch := make(chan struct{})
u.funcs <- func() {
defer close(ch)
2014-12-17 09:10:38 +01:00
f()
}
<-ch
2014-12-14 10:57:29 +01:00
}
func (u *ui) update() {
u.input.update(u.window, u.scale)
}