mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-11 19:48:54 +01:00
parent
08e931782a
commit
2360b2930f
120
examples/triangle/main.go
Normal file
120
examples/triangle/main.go
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
// 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.
|
||||||
|
|
||||||
|
// +build example jsgo
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"image/color"
|
||||||
|
"log"
|
||||||
|
"math"
|
||||||
|
|
||||||
|
"github.com/hajimehoshi/ebiten"
|
||||||
|
"github.com/hajimehoshi/ebiten/ebitenutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
screenWidth = 640
|
||||||
|
screenHeight = 480
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
emptyImage, _ = ebiten.NewImage(16, 16, ebiten.FilterDefault)
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
emptyImage.Fill(color.White)
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
vertices []ebiten.Vertex
|
||||||
|
indices []uint16
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
const (
|
||||||
|
num = 120
|
||||||
|
centerX = screenWidth / 2
|
||||||
|
centerY = screenHeight / 2
|
||||||
|
r = 160
|
||||||
|
)
|
||||||
|
|
||||||
|
for i := 0; i < num; i++ {
|
||||||
|
theta := float64(i) / num * 2 * math.Pi
|
||||||
|
cr := float32(0)
|
||||||
|
cg := float32(0)
|
||||||
|
cb := float32(0)
|
||||||
|
if 0 <= i && i < 2*num/3 {
|
||||||
|
cr = 2 * float32(i) / float32(num/3)
|
||||||
|
}
|
||||||
|
if num/3 <= i && i < 2*num/3 {
|
||||||
|
cr = 2 - 2*float32(i-num/3)/float32(num/3)
|
||||||
|
}
|
||||||
|
if num/3 <= i && i < 2*num/3 {
|
||||||
|
cg = 2 * float32(i-num/3) / float32(num/3)
|
||||||
|
}
|
||||||
|
if 2*num/3 <= i && i < num {
|
||||||
|
cg = 2 - 2*float32(i-2*num/3)/float32(num/3)
|
||||||
|
}
|
||||||
|
if 2*num/3 <= i && i < num {
|
||||||
|
cb = 2 * float32(i-2*num/3) / float32(num/3)
|
||||||
|
}
|
||||||
|
if 0 <= i && i < num/3 {
|
||||||
|
cb = 2 - 2*float32(i)/float32(num/3)
|
||||||
|
}
|
||||||
|
vertices = append(vertices, ebiten.Vertex{
|
||||||
|
DstX: float32(r*math.Cos(theta)) + centerX,
|
||||||
|
DstY: float32(r*math.Sin(theta)) + centerY,
|
||||||
|
SrcX: 0,
|
||||||
|
SrcY: 0,
|
||||||
|
ColorR: cr,
|
||||||
|
ColorG: cg,
|
||||||
|
ColorB: cb,
|
||||||
|
ColorA: 1,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
vertices = append(vertices, ebiten.Vertex{
|
||||||
|
DstX: centerX,
|
||||||
|
DstY: centerY,
|
||||||
|
SrcX: 0,
|
||||||
|
SrcY: 0,
|
||||||
|
ColorR: 1,
|
||||||
|
ColorG: 1,
|
||||||
|
ColorB: 1,
|
||||||
|
ColorA: 1,
|
||||||
|
})
|
||||||
|
for i := 0; i < num; i++ {
|
||||||
|
indices = append(indices, uint16(i), uint16(i+1)%num, num)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func update(screen *ebiten.Image) error {
|
||||||
|
if ebiten.IsDrawingSkipped() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
op := &ebiten.DrawTrianglesOptions{}
|
||||||
|
screen.DrawTriangles(vertices, indices, emptyImage, op)
|
||||||
|
ebitenutil.DebugPrint(screen, fmt.Sprintf("TPS: %0.2f", ebiten.CurrentTPS()))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if err := ebiten.Run(update, screenWidth, screenHeight, 1, "Triangle (Ebiten Demo)"); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
78
image.go
78
image.go
@ -338,6 +338,84 @@ func (i *Image) drawImage(img *Image, options *DrawImageOptions) {
|
|||||||
i.disposeMipmaps()
|
i.disposeMipmaps()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Vertex represents a vertex passed to DrawTriangles.
|
||||||
|
//
|
||||||
|
// Note that this API is experimental.
|
||||||
|
type Vertex struct {
|
||||||
|
// DstX and DstY represents a point on a destination image.
|
||||||
|
DstX float32
|
||||||
|
DstY float32
|
||||||
|
|
||||||
|
// SrcX and SrcY represents a point on a source image.
|
||||||
|
SrcX float32
|
||||||
|
SrcY float32
|
||||||
|
|
||||||
|
// ColorR/ColorG/ColorB/ColorA represents color scaling values.
|
||||||
|
// 1 means the original source image color is used.
|
||||||
|
// 0 means a transparent color is used.
|
||||||
|
ColorR float32
|
||||||
|
ColorG float32
|
||||||
|
ColorB float32
|
||||||
|
ColorA float32
|
||||||
|
}
|
||||||
|
|
||||||
|
// DrawTrianglesOptions represents options to render triangles on an image.
|
||||||
|
//
|
||||||
|
// Note that this API is experimental.
|
||||||
|
type DrawTrianglesOptions struct {
|
||||||
|
// ColorM is a color matrix to draw.
|
||||||
|
// The default (zero) value is identity, which doesn't change any color.
|
||||||
|
// ColorM is applied before vertex color scale is applied.
|
||||||
|
ColorM ColorM
|
||||||
|
|
||||||
|
// CompositeMode is a composite mode to draw.
|
||||||
|
// The default (zero) value is regular alpha blending.
|
||||||
|
CompositeMode CompositeMode
|
||||||
|
|
||||||
|
// Filter is a type of texture filter.
|
||||||
|
// The default (zero) value is FilterDefault.
|
||||||
|
Filter Filter
|
||||||
|
}
|
||||||
|
|
||||||
|
// DrawTriangles draws a triangle with the specified vertices and their indices.
|
||||||
|
//
|
||||||
|
// If len(indices) is not multiple of 3, DrawTriangles panics.
|
||||||
|
//
|
||||||
|
// The rule in which DrawTriangles works effectively is same as DrawImage's.
|
||||||
|
//
|
||||||
|
// In contrast to DrawImage, DrawTriangles doesn't care source image edges.
|
||||||
|
// This means that you might need to add 1px gap on a source region when you render an image by DrawTriangles.
|
||||||
|
// Note that Ebiten creates texture atlases internally, so you still have to care this even when
|
||||||
|
// you render a single image.
|
||||||
|
//
|
||||||
|
// Note that this API is experimental.
|
||||||
|
func (i *Image) DrawTriangles(vertices []Vertex, indices []uint16, img *Image, options *DrawTrianglesOptions) {
|
||||||
|
if len(indices)%3 != 0 {
|
||||||
|
panic("ebiten: len(indices) % 3 must be 0")
|
||||||
|
}
|
||||||
|
// TODO: Check the maximum value of indices and len(vertices)?
|
||||||
|
|
||||||
|
if options == nil {
|
||||||
|
options = &DrawTrianglesOptions{}
|
||||||
|
}
|
||||||
|
|
||||||
|
mode := opengl.CompositeMode(options.CompositeMode)
|
||||||
|
|
||||||
|
filter := graphics.FilterNearest
|
||||||
|
if options.Filter != FilterDefault {
|
||||||
|
filter = graphics.Filter(options.Filter)
|
||||||
|
} else if img.filter != FilterDefault {
|
||||||
|
filter = graphics.Filter(img.filter)
|
||||||
|
}
|
||||||
|
|
||||||
|
vs := []float32{}
|
||||||
|
src := img.shareableImages[0]
|
||||||
|
for _, v := range vertices {
|
||||||
|
vs = append(vs, src.Vertex(float32(v.DstX), float32(v.DstY), v.SrcX, v.SrcY, v.ColorR, v.ColorG, v.ColorB, v.ColorA)...)
|
||||||
|
}
|
||||||
|
i.shareableImages[0].DrawImage(img.shareableImages[0], vs, indices, options.ColorM.impl, mode, filter)
|
||||||
|
}
|
||||||
|
|
||||||
// Bounds returns the bounds of the image.
|
// Bounds returns the bounds of the image.
|
||||||
func (i *Image) Bounds() image.Rectangle {
|
func (i *Image) Bounds() image.Rectangle {
|
||||||
w, h := i.Size()
|
w, h := i.Size()
|
||||||
|
@ -56,10 +56,19 @@ varying vec2 varying_tex_coord_min;
|
|||||||
varying vec2 varying_tex_coord_max;
|
varying vec2 varying_tex_coord_max;
|
||||||
varying vec4 varying_color_scale;
|
varying vec4 varying_color_scale;
|
||||||
|
|
||||||
|
bool isNaN(float x) {
|
||||||
|
return x != x;
|
||||||
|
}
|
||||||
|
|
||||||
void main(void) {
|
void main(void) {
|
||||||
varying_tex_coord = vec2(tex_coord[0], tex_coord[1]);
|
varying_tex_coord = vec2(tex_coord[0], tex_coord[1]);
|
||||||
|
if (!isNaN(tex_coord[2]) && !isNaN(tex_coord[3])) {
|
||||||
varying_tex_coord_min = vec2(min(tex_coord[0], tex_coord[2]), min(tex_coord[1], tex_coord[3]));
|
varying_tex_coord_min = vec2(min(tex_coord[0], tex_coord[2]), min(tex_coord[1], tex_coord[3]));
|
||||||
varying_tex_coord_max = vec2(max(tex_coord[0], tex_coord[2]), max(tex_coord[1], tex_coord[3]));
|
varying_tex_coord_max = vec2(max(tex_coord[0], tex_coord[2]), max(tex_coord[1], tex_coord[3]));
|
||||||
|
} else {
|
||||||
|
varying_tex_coord_min = vec2(0, 0);
|
||||||
|
varying_tex_coord_max = vec2(1, 1);
|
||||||
|
}
|
||||||
varying_color_scale = color_scale;
|
varying_color_scale = color_scale;
|
||||||
gl_Position = projection_matrix * vec4(vertex, 0, 1);
|
gl_Position = projection_matrix * vec4(vertex, 0, 1);
|
||||||
}
|
}
|
||||||
|
@ -15,6 +15,8 @@
|
|||||||
package graphicsutil
|
package graphicsutil
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"math"
|
||||||
|
|
||||||
"github.com/hajimehoshi/ebiten/internal/graphics"
|
"github.com/hajimehoshi/ebiten/internal/graphics"
|
||||||
"github.com/hajimehoshi/ebiten/internal/opengl"
|
"github.com/hajimehoshi/ebiten/internal/opengl"
|
||||||
)
|
)
|
||||||
@ -135,3 +137,35 @@ var (
|
|||||||
func QuadIndices() []uint16 {
|
func QuadIndices() []uint16 {
|
||||||
return quadIndices
|
return quadIndices
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
nan32 = float32(math.NaN())
|
||||||
|
)
|
||||||
|
|
||||||
|
func Vertex(width, height int, dx, dy, sx, sy float32, cr, cg, cb, ca float32) []float32 {
|
||||||
|
if !isPowerOf2(width) {
|
||||||
|
panic("not reached")
|
||||||
|
}
|
||||||
|
if !isPowerOf2(height) {
|
||||||
|
panic("not reached")
|
||||||
|
}
|
||||||
|
|
||||||
|
wf := float32(width)
|
||||||
|
hf := float32(height)
|
||||||
|
|
||||||
|
// Specifying a range explicitly here is redundant but this helps optimization
|
||||||
|
// to eliminate boundry checks.
|
||||||
|
vs := theVerticesBackend.sliceForOneQuad()[0:10]
|
||||||
|
vs[0] = dx
|
||||||
|
vs[1] = dy
|
||||||
|
vs[2] = sx / wf
|
||||||
|
vs[3] = sy / hf
|
||||||
|
vs[4] = nan32
|
||||||
|
vs[5] = nan32
|
||||||
|
vs[6] = cr
|
||||||
|
vs[7] = cg
|
||||||
|
vs[8] = cb
|
||||||
|
vs[9] = ca
|
||||||
|
|
||||||
|
return vs
|
||||||
|
}
|
||||||
|
@ -193,6 +193,15 @@ func (i *Image) QuadVertices(sx0, sy0, sx1, sy1 int, a, b, c, d, tx, ty float32,
|
|||||||
return graphicsutil.QuadVertices(w, h, sx0+ox, sy0+oy, sx1+ox, sy1+oy, a, b, c, d, tx, ty, cr, cg, cb, ca)
|
return graphicsutil.QuadVertices(w, h, sx0+ox, sy0+oy, sx1+ox, sy1+oy, a, b, c, d, tx, ty, cr, cg, cb, ca)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (i *Image) Vertex(dx, dy, sx, sy float32, cr, cg, cb, ca float32) []float32 {
|
||||||
|
if i.backend == nil {
|
||||||
|
i.allocate(true)
|
||||||
|
}
|
||||||
|
ox, oy, _, _ := i.region()
|
||||||
|
w, h := i.backend.restorable.SizePowerOf2()
|
||||||
|
return graphicsutil.Vertex(w, h, dx, dy, sx+float32(ox), sy+float32(oy), cr, cg, cb, ca)
|
||||||
|
}
|
||||||
|
|
||||||
const MaxCountForShare = 10
|
const MaxCountForShare = 10
|
||||||
|
|
||||||
func (i *Image) DrawImage(img *Image, vertices []float32, indices []uint16, colorm *affine.ColorM, mode opengl.CompositeMode, filter graphics.Filter) {
|
func (i *Image) DrawImage(img *Image, vertices []float32, indices []uint16, colorm *affine.ColorM, mode opengl.CompositeMode, filter graphics.Filter) {
|
||||||
|
Loading…
Reference in New Issue
Block a user