2018-09-30 11:53:38 +02: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.
|
|
|
|
|
2021-06-24 14:49:37 +02:00
|
|
|
//go:build example
|
2020-10-06 17:45:54 +02:00
|
|
|
// +build example
|
2018-09-30 11:53:38 +02:00
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"image"
|
|
|
|
_ "image/jpeg"
|
|
|
|
"log"
|
|
|
|
|
2020-10-03 19:35:13 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/examples/resources/images"
|
2018-09-30 11:53:38 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
screenWidth = 640
|
|
|
|
screenHeight = 480
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
gophersImage *ebiten.Image
|
|
|
|
)
|
|
|
|
|
2020-04-12 11:18:45 +02:00
|
|
|
type Game struct{}
|
|
|
|
|
2020-10-04 10:42:54 +02:00
|
|
|
func (g *Game) Update() error {
|
2020-04-12 11:18:45 +02:00
|
|
|
return nil
|
|
|
|
}
|
2018-09-30 11:53:38 +02:00
|
|
|
|
2020-04-12 11:18:45 +02:00
|
|
|
func (g *Game) Draw(screen *ebiten.Image) {
|
2018-09-30 11:53:38 +02:00
|
|
|
op := &ebiten.DrawImageOptions{}
|
|
|
|
op.GeoM.Translate(0, 0)
|
|
|
|
screen.DrawImage(gophersImage, op)
|
|
|
|
|
2018-09-30 19:37:53 +02:00
|
|
|
// Box blur (7x7)
|
|
|
|
// https://en.wikipedia.org/wiki/Box_blur
|
2022-02-23 19:15:52 +01:00
|
|
|
//
|
|
|
|
// Note that this is a fixed function implementation of a box blur - more
|
|
|
|
// efficiency can be gained by using a separable blur
|
|
|
|
// (blurring horizontally and vertically separately, or for large blurs,
|
|
|
|
// even multiple horizontal or vertical passes), ideally combined with
|
|
|
|
// doing the summing up in a fragment shader (Kage can be used here).
|
|
|
|
//
|
|
|
|
// So this implementation only serves to demonstrate use of alpha blending.
|
|
|
|
layers := 0
|
2018-09-30 11:53:38 +02:00
|
|
|
for j := -3; j <= 3; j++ {
|
|
|
|
for i := -3; i <= 3; i++ {
|
|
|
|
op := &ebiten.DrawImageOptions{}
|
|
|
|
op.GeoM.Translate(float64(i), 244+float64(j))
|
2022-10-16 13:02:42 +02:00
|
|
|
// This is a blur based on the source-over blend mode,
|
2022-02-23 19:15:52 +01:00
|
|
|
// which is basically (GL_ONE, GL_ONE_MINUS_SRC_ALPHA). ColorM acts
|
2022-08-29 04:16:39 +02:00
|
|
|
// on unpremultiplied colors, but all Ebitengine internal colors are
|
2022-02-23 19:15:52 +01:00
|
|
|
// premultiplied, meaning this mode is regular alpha blending,
|
|
|
|
// computing each destination pixel as srcPix * alpha + dstPix * (1 - alpha).
|
|
|
|
//
|
2022-10-16 13:02:42 +02:00
|
|
|
// This means that the final color is affected by the destination color when BlendSourceOver is used.
|
|
|
|
// This blend mode is the default mode. See how this is calculated at the doc:
|
|
|
|
// https://pkg.go.dev/github.com/hajimehoshi/ebiten/v2#Blend
|
2019-01-06 11:14:44 +01:00
|
|
|
//
|
2022-02-23 19:15:52 +01:00
|
|
|
// So if using the same alpha every time, the end result will sure be biased towards the last layer.
|
|
|
|
//
|
|
|
|
// Correct averaging works based on
|
|
|
|
// Let A_n := (a_1 + ... + a_n) / n
|
|
|
|
// A_{n+1} = (a_1 + ... + a_{n+1}) / (n + 1)
|
|
|
|
// A_{n+1} = (n * A_n + a_{n+1)) / (n + 1)
|
|
|
|
// A_{n+1} = A_n * (1 - 1/(n+1)) + a_{n+1} * 1/(n+1)
|
|
|
|
// which is precisely what an alpha blend with alpha 1/(n+1) does.
|
|
|
|
layers++
|
|
|
|
op.ColorM.Scale(1, 1, 1, 1.0/float64(layers))
|
2018-09-30 11:53:38 +02:00
|
|
|
screen.DrawImage(gophersImage, op)
|
|
|
|
}
|
|
|
|
}
|
2020-04-12 11:18:45 +02:00
|
|
|
}
|
2018-09-30 11:53:38 +02:00
|
|
|
|
2020-04-12 11:18:45 +02:00
|
|
|
func (g *Game) Layout(outsideWidth, outsideHeight int) (int, int) {
|
|
|
|
return screenWidth, screenHeight
|
2018-09-30 11:53:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
2021-09-20 08:22:46 +02:00
|
|
|
// Decode an image from the image file's byte slice.
|
2018-09-30 11:53:38 +02:00
|
|
|
img, _, err := image.Decode(bytes.NewReader(images.FiveYears_jpg))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2020-10-05 18:03:30 +02:00
|
|
|
gophersImage = ebiten.NewImageFromImage(img)
|
2018-09-30 11:53:38 +02:00
|
|
|
|
2020-04-12 11:18:45 +02:00
|
|
|
ebiten.SetWindowSize(screenWidth, screenHeight)
|
2022-08-29 04:16:39 +02:00
|
|
|
ebiten.SetWindowTitle("Blur (Ebitengine Demo)")
|
2020-04-12 11:18:45 +02:00
|
|
|
if err := ebiten.RunGame(&Game{}); err != nil {
|
2018-09-30 11:53:38 +02:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|