internal/uidriver/glfw: Bug fix: Enable to specify ebiten.Image to SetWindowIcon

Closes #1468
This commit is contained in:
Hajime Hoshi 2021-02-07 21:16:09 +09:00
parent acb8bcae38
commit d999b4dc8e
3 changed files with 50 additions and 25 deletions

View File

@ -66,18 +66,23 @@ var (
func createRandomIconImage() image.Image { func createRandomIconImage() image.Image {
const size = 32 const size = 32
r := byte(rand.Intn(0x100)) rf := float64(rand.Intn(0x100))
g := byte(rand.Intn(0x100)) gf := float64(rand.Intn(0x100))
b := byte(rand.Intn(0x100)) bf := float64(rand.Intn(0x100))
img := image.NewNRGBA(image.Rect(0, 0, size, size)) img := ebiten.NewImage(size, size)
pix := make([]byte, 4*size*size)
for j := 0; j < size; j++ { for j := 0; j < size; j++ {
for i := 0; i < size; i++ { for i := 0; i < size; i++ {
img.Pix[j*img.Stride+4*i] = r af := float64(i+j) / float64(2*size)
img.Pix[j*img.Stride+4*i+1] = g if af > 0 {
img.Pix[j*img.Stride+4*i+2] = b pix[4*(j*size+i)] = byte(rf * af)
img.Pix[j*img.Stride+4*i+3] = byte(float64(i+j) / float64(2*size) * 0xff) pix[4*(j*size+i)+1] = byte(gf * af)
pix[4*(j*size+i)+2] = byte(bf * af)
pix[4*(j*size+i)+3] = byte(af * 0xff)
} }
} }
}
img.ReplacePixels(pix)
return img return img
} }

View File

@ -68,8 +68,8 @@ type UserInterface struct {
initWindowFloating bool initWindowFloating bool
initWindowMaximized bool initWindowMaximized bool
initScreenTransparent bool initScreenTransparent bool
initIconImages []image.Image
initFocused bool initFocused bool
iconImages []image.Image
vsyncInited bool vsyncInited bool
@ -308,16 +308,16 @@ func (u *UserInterface) setInitScreenTransparent(transparent bool) {
u.m.RUnlock() u.m.RUnlock()
} }
func (u *UserInterface) getInitIconImages() []image.Image { func (u *UserInterface) getIconImages() []image.Image {
u.m.RLock() u.m.RLock()
i := u.initIconImages i := u.iconImages
u.m.RUnlock() u.m.RUnlock()
return i return i
} }
func (u *UserInterface) setInitIconImages(iconImages []image.Image) { func (u *UserInterface) setIconImages(iconImages []image.Image) {
u.m.Lock() u.m.Lock()
u.initIconImages = iconImages u.iconImages = iconImages
u.m.Unlock() u.m.Unlock()
} }
@ -685,10 +685,6 @@ func (u *UserInterface) init() error {
return err return err
} }
if i := u.getInitIconImages(); i != nil {
u.window.SetIcon(i)
}
setPosition := func() { setPosition := func() {
u.iwindow.setPosition(u.getInitWindowPosition()) u.iwindow.setPosition(u.getInitWindowPosition())
} }
@ -807,6 +803,7 @@ func (u *UserInterface) loop() error {
return nil return nil
}) })
}() }()
for { for {
var unfocused bool var unfocused bool
@ -840,6 +837,35 @@ func (u *UserInterface) loop() error {
return err return err
} }
if imgs := u.getIconImages(); imgs != nil {
u.setIconImages(nil)
// Convert the icons in the different goroutine, as (*ebiten.Image).At cannot be invoked
// from this goroutine. At works only in between BeginFrame and EndFrame.
go func() {
newImgs := make([]image.Image, len(imgs))
for i, img := range imgs {
// TODO: If img is not *ebiten.Image, this converting is not necessary.
// However, this package cannot refer *ebiten.Image due to the package
// dependencies.
b := img.Bounds()
rgba := image.NewRGBA(b)
for j := b.Min.Y; j < b.Max.Y; j++ {
for i := b.Min.X; i < b.Max.X; i++ {
rgba.Set(i, j, img.At(i, j))
}
}
newImgs[i] = rgba
}
_ = u.t.Call(func() error {
u.window.SetIcon(newImgs)
return nil
})
}()
}
// swapBuffers also checks IsGL, so this condition is redundant. // swapBuffers also checks IsGL, so this condition is redundant.
// However, (*thread).Call is not good for performance due to channels. // However, (*thread).Call is not good for performance due to channels.
// Let's avoid this whenever possible (#1367). // Let's avoid this whenever possible (#1367).

View File

@ -249,14 +249,8 @@ func (w *window) SetSize(width, height int) {
} }
func (w *window) SetIcon(iconImages []image.Image) { func (w *window) SetIcon(iconImages []image.Image) {
if !w.ui.isRunning() { // The icons are actually set at (*UserInterface).loop.
w.ui.setInitIconImages(iconImages) w.ui.setIconImages(iconImages)
return
}
_ = w.ui.t.Call(func() error {
w.ui.window.SetIcon(iconImages)
return nil
})
} }
func (w *window) SetTitle(title string) { func (w *window) SetTitle(title string) {