2022-03-19 18:00:44 +01:00
|
|
|
// Copyright 2022 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.
|
|
|
|
|
|
|
|
package ui
|
|
|
|
|
|
|
|
import (
|
2022-08-26 10:03:17 +02:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
2022-03-19 18:00:44 +01:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/mipmap"
|
2022-04-03 19:15:33 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/shaderir"
|
2022-03-19 18:00:44 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Shader struct {
|
2022-03-21 09:48:47 +01:00
|
|
|
shader *mipmap.Shader
|
2022-08-26 10:03:17 +02:00
|
|
|
|
|
|
|
uniformNames []string
|
|
|
|
uniformTypes []shaderir.Type
|
|
|
|
uniformNameToIndex map[string]int
|
|
|
|
uniformNameToType map[string]shaderir.Type
|
2022-03-19 18:00:44 +01:00
|
|
|
}
|
|
|
|
|
2022-04-03 19:15:33 +02:00
|
|
|
func NewShader(ir *shaderir.Program) *Shader {
|
2022-03-19 18:00:44 +01:00
|
|
|
return &Shader{
|
2022-08-26 10:03:17 +02:00
|
|
|
shader: mipmap.NewShader(ir),
|
|
|
|
uniformNames: ir.UniformNames,
|
|
|
|
uniformTypes: ir.Uniforms,
|
2022-03-19 18:00:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Shader) MarkDisposed() {
|
|
|
|
s.shader.MarkDisposed()
|
|
|
|
s.shader = nil
|
|
|
|
}
|
2022-08-26 10:03:17 +02:00
|
|
|
|
|
|
|
func (s *Shader) ConvertUniforms(uniforms map[string]interface{}) [][]float32 {
|
|
|
|
nameToF32s := map[string][]float32{}
|
|
|
|
for name, v := range uniforms {
|
|
|
|
switch v := v.(type) {
|
|
|
|
case float32:
|
|
|
|
nameToF32s[name] = []float32{v}
|
|
|
|
case []float32:
|
|
|
|
nameToF32s[name] = v
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("ebiten: unexpected uniform value type: %s, %T", name, v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.uniformNameToIndex == nil {
|
|
|
|
s.uniformNameToIndex = map[string]int{}
|
|
|
|
s.uniformNameToType = map[string]shaderir.Type{}
|
|
|
|
|
|
|
|
var idx int
|
|
|
|
for i, n := range s.uniformNames {
|
|
|
|
if strings.HasPrefix(n, "__") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
s.uniformNameToIndex[n] = idx
|
|
|
|
s.uniformNameToType[n] = s.uniformTypes[i]
|
|
|
|
idx++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
us := make([][]float32, len(s.uniformNameToIndex))
|
|
|
|
for name, idx := range s.uniformNameToIndex {
|
|
|
|
if v, ok := nameToF32s[name]; ok {
|
|
|
|
us[idx] = v
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
t := s.uniformNameToType[name]
|
|
|
|
us[idx] = make([]float32, t.FloatCount())
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Panic if uniforms include an invalid name
|
|
|
|
|
|
|
|
return us
|
|
|
|
}
|