Remove internal/conv

This sets font rendering back a little bit, but not much.
This commit is contained in:
2020-03-21 13:37:20 +00:00
parent 4c0355ac4f
commit be4229b8fe
8 changed files with 145 additions and 220 deletions

View File

@@ -2,7 +2,6 @@ package main
import ( import (
"flag" "flag"
"fmt"
"image" "image"
"log" "log"
"os" "os"
@@ -10,24 +9,23 @@ import (
"github.com/hajimehoshi/ebiten" "github.com/hajimehoshi/ebiten"
"code.ur.gs/lupine/ordoor/internal/conv" "code.ur.gs/lupine/ordoor/internal/assetstore"
"code.ur.gs/lupine/ordoor/internal/data" "code.ur.gs/lupine/ordoor/internal/data"
"code.ur.gs/lupine/ordoor/internal/fonts"
"code.ur.gs/lupine/ordoor/internal/menus" "code.ur.gs/lupine/ordoor/internal/menus"
"code.ur.gs/lupine/ordoor/internal/ui" "code.ur.gs/lupine/ordoor/internal/ui"
) )
var ( var (
gamePath = flag.String("game-path", "./orig", "Path to a WH40K: Chaos Gate installation") gamePath = flag.String("game-path", "./orig", "Path to a WH40K: Chaos Gate installation")
menuFile = flag.String("menu", "", "Path to a .mnu file, e.g. ./orig/Menu/MainGame.mnu") menuFile = flag.String("menu", "", "Name of a menu, e.g. Main")
) )
type env struct { type env struct {
menu *menus.Menu menu *menus.Menu
objects []*conv.Object objects []*assetstore.Object
fonts []*conv.Font // fonts []*assetstore.Font
fontObjs []*conv.Object // fontObjs []*assetstore.Object
step int step int
state state state state
@@ -39,41 +37,6 @@ type state struct {
winBounds image.Rectangle winBounds image.Rectangle
} }
func loadObjects(names ...string) ([]*conv.Object, error) {
objs := make([]*conv.Object, 0, len(names))
for _, name := range names {
objFile := filepath.Join(filepath.Dir(*menuFile), name)
rawObj, err := data.LoadObject(objFile)
if err != nil {
log.Fatalf("Failed to load %s: %v", name, err)
}
obj, err := conv.ConvertObject(rawObj, name)
if err != nil {
return nil, err
}
objs = append(objs, obj)
}
return objs, nil
}
func loadFonts(names ...string) ([]*conv.Font, error) {
var out []*conv.Font
for _, name := range names {
fnt, err := fonts.LoadFont(filepath.Join(*gamePath, "Fonts", name+".fnt"))
if err != nil {
return nil, fmt.Errorf("%v: %v", name, err)
}
out = append(out, conv.ConvertFont(fnt))
}
return out, nil
}
func main() { func main() {
flag.Parse() flag.Parse()
@@ -82,6 +45,11 @@ func main() {
os.Exit(1) os.Exit(1)
} }
assets, err := assetstore.New(*gamePath)
if err != nil {
log.Fatal(err)
}
menu, err := menus.LoadMenu(*menuFile) menu, err := menus.LoadMenu(*menuFile)
if err != nil { if err != nil {
log.Fatalf("Couldn't load menu file %s: %v", *menuFile, err) log.Fatalf("Couldn't load menu file %s: %v", *menuFile, err)
@@ -93,21 +61,26 @@ func main() {
menu.Internationalize(i18n) menu.Internationalize(i18n)
} }
loadedFonts, err := loadFonts(menu.FontNames...) // loadedFonts, err := loadFonts(menu.FontNames...)
if err != nil { // if err != nil {
log.Fatalf("Failed to load font: %v", err) // log.Fatalf("Failed to load font: %v", err)
} // }
menuObjs, err := loadObjects(menu.ObjectFiles...) var menuObjs []*assetstore.Object
if err != nil { for _, filename := range menu.ObjectFiles {
log.Fatalf("Failed to load objects: %v", err) obj, err := assets.ObjectByPath(filepath.Join(*gamePath, "Menu", filename))
if err != nil {
log.Fatalf("Failed to load %v: %v", filename, err)
}
menuObjs = append(menuObjs, obj)
} }
state := state{} state := state{}
env := &env{ env := &env{
menu: menu, menu: menu,
objects: menuObjs, objects: menuObjs,
fonts: loadedFonts, // fonts: loadedFonts,
state: state, state: state,
lastState: state, lastState: state,
} }
@@ -156,8 +129,6 @@ func (e *env) Draw(screen *ebiten.Image) error {
} }
func (e *env) drawRecord(record *menus.Record, screen *ebiten.Image, offset ebiten.GeoM) error { func (e *env) drawRecord(record *menus.Record, screen *ebiten.Image, offset ebiten.GeoM) error {
origOffset := offset
// Draw this record if it's valid to do so. FIXME: lots to learn // Draw this record if it's valid to do so. FIXME: lots to learn
if len(record.SpriteId) >= 0 { if len(record.SpriteId) >= 0 {
spriteId := record.SpriteId[0] spriteId := record.SpriteId[0]
@@ -187,8 +158,10 @@ func (e *env) drawRecord(record *menus.Record, screen *ebiten.Image, offset ebit
// FIXME: Need to handle multiple objects // FIXME: Need to handle multiple objects
obj := e.objects[0] obj := e.objects[0]
sprite := obj.Sprites[spriteId] sprite, err := obj.Sprite(spriteId)
if err != nil {
return err
}
x = x + float64(sprite.XOffset) x = x + float64(sprite.XOffset)
y = y + float64(sprite.YOffset) y = y + float64(sprite.YOffset)
@@ -205,9 +178,9 @@ func (e *env) drawRecord(record *menus.Record, screen *ebiten.Image, offset ebit
// FIXME: we probably shouldn't draw everything? // FIXME: we probably shouldn't draw everything?
// FIXME: handle multiple fonts // FIXME: handle multiple fonts
if len(e.fonts) > 0 && record.Desc != "" { // if len(e.fonts) > 0 && record.Desc != "" {
e.fonts[0].Output(screen, origOffset, record.Desc) // e.fonts[0].Output(screen, origOffset, record.Desc)
} // }
} }
out: out:
// Draw all children of this record // Draw all children of this record

View File

@@ -6,22 +6,20 @@ import (
"log" "log"
"math" "math"
"os" "os"
"path/filepath"
"github.com/hajimehoshi/ebiten" "github.com/hajimehoshi/ebiten"
"code.ur.gs/lupine/ordoor/internal/conv" "code.ur.gs/lupine/ordoor/internal/assetstore"
"code.ur.gs/lupine/ordoor/internal/data"
"code.ur.gs/lupine/ordoor/internal/ui" "code.ur.gs/lupine/ordoor/internal/ui"
) )
var ( var (
gamePath = flag.String("game-path", "./orig", "Path to a WH40K: Chaos Gate installation") gamePath = flag.String("game-path", "./orig", "Path to a WH40K: Chaos Gate installation")
objFile = flag.String("obj", "", "Path to a .obj file, e.g. ./orig/Obj/TZEENTCH.OBJ") objName = flag.String("obj", "", "Name of an .obj file, e.g. TZEENTCH")
) )
type env struct { type env struct {
obj *conv.Object obj *assetstore.Object
step int step int
state state state state
@@ -38,19 +36,19 @@ type state struct {
func main() { func main() {
flag.Parse() flag.Parse()
if *gamePath == "" || *objFile == "" { if *gamePath == "" || *objName == "" {
flag.Usage() flag.Usage()
os.Exit(1) os.Exit(1)
} }
rawObj, err := data.LoadObject(*objFile) assets, err := assetstore.New(*gamePath)
if err != nil { if err != nil {
log.Fatalf("Failed to load %s: %v", *objFile, err) log.Fatal("Failed to set up asset store: %v", err)
} }
obj, err := conv.ConvertObject(rawObj, filepath.Base(*objFile)) obj, err := assets.Object(*objName)
if err != nil { if err != nil {
log.Fatalf("Failed to convert %s: %v", *objFile, err) log.Fatalf("Failed to load %s: %v", *objName, err)
} }
state := state{ state := state{
@@ -64,7 +62,7 @@ func main() {
lastState: state, lastState: state,
} }
win, err := ui.NewWindow("View Object: " + *objFile) win, err := ui.NewWindow("View Object: " + *objName)
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
} }
@@ -86,9 +84,9 @@ func main() {
func (e *env) Update() error { func (e *env) Update() error {
if e.step == 0 || e.lastState != e.state { if e.step == 0 || e.lastState != e.state {
log.Printf( log.Printf(
"new state: numSprites=%d sprite=%d zoom=%.2f, origin=%+v", "new state: sprite=%d/%d zoom=%.2f, origin=%+v",
len(e.obj.Sprites),
e.state.spriteIdx, e.state.spriteIdx,
e.obj.NumSprites,
e.state.zoom, e.state.zoom,
e.state.origin, e.state.origin,
) )
@@ -102,7 +100,10 @@ func (e *env) Update() error {
} }
func (e *env) Draw(screen *ebiten.Image) error { func (e *env) Draw(screen *ebiten.Image) error {
sprite := e.obj.Sprites[e.state.spriteIdx] sprite, err := e.obj.Sprite(e.state.spriteIdx)
if err != nil {
return err
}
cam := ebiten.GeoM{} cam := ebiten.GeoM{}
cam.Translate(float64(e.state.origin.X), float64(e.state.origin.Y)) // Move to origin cam.Translate(float64(e.state.origin.X), float64(e.state.origin.Y)) // Move to origin
@@ -119,8 +120,8 @@ func (e *env) changeSprite(by int) func() {
e.state.spriteIdx = 0 e.state.spriteIdx = 0
} }
if e.state.spriteIdx > len(e.obj.Sprites)-1 { if e.state.spriteIdx > e.obj.NumSprites-1 {
e.state.spriteIdx = len(e.obj.Sprites) - 1 e.state.spriteIdx = e.obj.NumSprites - 1
} }
} }
} }

View File

@@ -6,25 +6,20 @@ import (
"log" "log"
"math" "math"
"os" "os"
"path/filepath"
"github.com/hajimehoshi/ebiten" "github.com/hajimehoshi/ebiten"
"code.ur.gs/lupine/ordoor/internal/conv" "code.ur.gs/lupine/ordoor/internal/assetstore"
"code.ur.gs/lupine/ordoor/internal/data"
"code.ur.gs/lupine/ordoor/internal/sets"
"code.ur.gs/lupine/ordoor/internal/ui" "code.ur.gs/lupine/ordoor/internal/ui"
) )
var ( var (
gamePath = flag.String("game-path", "./orig", "Path to a WH40K: Chaos Gate installation") gamePath = flag.String("game-path", "./orig", "Path to a WH40K: Chaos Gate installation")
setFile = flag.String("set", "", "Path to a .set file, e.g. ./orig/Sets/map01.set") setName = flag.String("set", "", "Name of a set, e.g., map01")
) )
type env struct { type env struct {
set *sets.MapSet set *assetstore.Set
objects map[string]*conv.Object
step int step int
state state state state
lastState state lastState state
@@ -41,47 +36,29 @@ type state struct {
func main() { func main() {
flag.Parse() flag.Parse()
if *gamePath == "" || *setFile == "" { if *gamePath == "" || *setName == "" {
flag.Usage() flag.Usage()
os.Exit(1) os.Exit(1)
} }
mapSet, err := sets.LoadSet(*setFile) assets, err := assetstore.New(*gamePath)
if err != nil { if err != nil {
log.Fatalf("Couldn't load set file %s: %v", *setFile, err) log.Fatal(err)
} }
rawObjs := make([]*data.Object, 0, len(mapSet.Palette)) set, err := assets.Set(*setName)
for _, name := range mapSet.Palette { if err != nil {
objFile := filepath.Join(*gamePath, "Obj", name+".obj") log.Fatalf("Couldn't load set %s: %v", *setName, err)
obj, err := data.LoadObject(objFile)
if err != nil {
log.Fatalf("Failed to load %s: %v", name, err)
}
obj.Name = name
rawObjs = append(rawObjs, obj)
} }
objs := make([]*conv.Object, 0, len(rawObjs)) win, err := ui.NewWindow("View Set: " + *setName)
for _, rawObj := range rawObjs {
obj, err := conv.ConvertObject(rawObj, rawObj.Name)
if err != nil {
log.Fatal(err)
}
objs = append(objs, obj)
}
win, err := ui.NewWindow("View Set: " + *setFile)
if err != nil { if err != nil {
log.Fatal("Couldn't create window: %v", err) log.Fatal("Couldn't create window: %v", err)
} }
state := state{zoom: 8.0} state := state{zoom: 8.0}
env := &env{ env := &env{
set: mapSet, set: set,
objects: conv.MapByName(objs),
state: state, state: state,
lastState: state, lastState: state,
} }
@@ -101,13 +78,18 @@ func main() {
} }
func (e *env) Update() error { func (e *env) Update() error {
curObj, err := e.curObject()
if err != nil {
return err
}
if e.step == 0 || e.lastState != e.state { if e.step == 0 || e.lastState != e.state {
log.Printf( log.Printf(
"new state: numObj=%d object=%d (%s) numFrames=%d sprite=%d zoom=%.2f", "new state: object=%d/%d (%s) numFrames=%d sprite=%d zoom=%.2f",
e.set.Count(),
e.state.objIdx, e.state.objIdx,
e.set.Palette[e.state.objIdx], // FIXME: palette is a confusing name e.set.NumObjects,
len(e.curObject().Sprites), curObj.Name,
curObj.NumSprites,
e.state.spriteIdx, e.state.spriteIdx,
e.state.zoom, e.state.zoom,
) )
@@ -120,8 +102,10 @@ func (e *env) Update() error {
} }
func (e *env) Draw(screen *ebiten.Image) error { func (e *env) Draw(screen *ebiten.Image) error {
obj := e.curObject() sprite, err := e.curSprite()
sprite := obj.Sprites[e.state.spriteIdx] if err != nil {
return err
}
cam := ebiten.GeoM{} cam := ebiten.GeoM{}
cam.Scale(e.state.zoom, e.state.zoom) // apply current zoom factor cam.Scale(e.state.zoom, e.state.zoom) // apply current zoom factor
@@ -140,8 +124,8 @@ func (e *env) changeObjIdx(by int) func() {
e.state.objIdx = 0 e.state.objIdx = 0
} }
if e.state.objIdx > e.set.Count()-1 { if e.state.objIdx > e.set.NumObjects-1 {
e.state.objIdx = e.set.Count() - 1 e.state.objIdx = e.set.NumObjects - 1
} }
// reset sprite index when object changes // reset sprite index when object changes
@@ -153,15 +137,19 @@ func (e *env) changeObjIdx(by int) func() {
func (e *env) changeSpriteIdx(by int) func() { func (e *env) changeSpriteIdx(by int) func() {
return func() { return func() {
e.state.spriteIdx += by obj, err := e.curObject()
if err != nil {
log.Printf("Encountered %v trying to change sprite index", err)
return
}
e.state.spriteIdx += by
if e.state.spriteIdx < 0 { if e.state.spriteIdx < 0 {
e.state.spriteIdx = 0 e.state.spriteIdx = 0
} }
count := len(e.curObject().Sprites) if e.state.spriteIdx > obj.NumSprites-1 {
if e.state.spriteIdx > count-1 { e.state.spriteIdx = obj.NumSprites - 1
e.state.spriteIdx = count - 1
} }
} }
} }
@@ -171,7 +159,15 @@ func (e *env) changeZoom(_, y float64) {
e.state.zoom *= math.Pow(1.2, y) e.state.zoom *= math.Pow(1.2, y)
} }
func (e *env) curObject() *conv.Object { func (e *env) curObject() (*assetstore.Object, error) {
name := e.set.Palette[e.state.objIdx] return e.set.Object(e.state.objIdx)
return e.objects[name] }
func (e *env) curSprite() (*assetstore.Sprite, error) {
obj, err := e.curObject()
if err != nil {
return nil, err
}
return obj.Sprite(e.state.spriteIdx)
} }

View File

@@ -3,6 +3,7 @@ package assetstore
import ( import (
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"log"
"os" "os"
"path/filepath" "path/filepath"
"strings" "strings"
@@ -91,7 +92,10 @@ func (a *AssetStore) lookup(name, ext string, dirs ...string) (string, error) {
for _, dir := range dirs { for _, dir := range dirs {
dir = canonical(dir) dir = canonical(dir)
log.Printf("Looking in canonical dir %s", dir)
if base, ok := a.entries[dir]; ok { if base, ok := a.entries[dir]; ok {
log.Printf(" base: %#+v", base)
log.Printf(" filename: %v", filename)
if file, ok := base[filename]; ok { if file, ok := base[filename]; ok {
actualDir := a.entries[RootDir][dir] actualDir := a.entries[RootDir][dir]
return filepath.Join(a.RootDir, actualDir, file), nil return filepath.Join(a.RootDir, actualDir, file), nil

View File

@@ -3,6 +3,8 @@ package assetstore
import ( import (
"fmt" "fmt"
"log" "log"
"path/filepath"
"strings"
"github.com/hajimehoshi/ebiten" "github.com/hajimehoshi/ebiten"
@@ -13,7 +15,9 @@ type Object struct {
assets *AssetStore assets *AssetStore
sprites []*Sprite sprites []*Sprite
raw *data.Object raw *data.Object
NumSprites int
Name string
} }
type Sprite struct { type Sprite struct {
@@ -41,23 +45,49 @@ func (a *AssetStore) Object(name string) (*Object, error) {
return nil, err return nil, err
} }
raw, err := data.LoadObjectLazily(filename) obj, err := a.ObjectByPath(filename)
if err != nil { if err != nil {
return nil, err return nil, err
} }
raw.Name = name
obj := &Object{
assets: a,
sprites: make([]*Sprite, int(raw.NumSprites)),
raw: raw,
}
a.objs[name] = obj a.objs[name] = obj
return obj, nil return obj, nil
} }
// Filled lazily // FIXME: Objects loaded by path are not cached
func (a *AssetStore) ObjectByPath(path string) (*Object, error) {
name := filepath.Base(path)
name = strings.Replace(name, filepath.Ext(name), "", -1)
name = canonical(name)
raw, err := data.LoadObjectLazily(path)
if err != nil {
return nil, err
}
obj := &Object{
assets: a,
sprites: make([]*Sprite, int(raw.NumSprites)),
raw: raw,
NumSprites: int(raw.NumSprites),
Name: canonical(name),
}
return obj, nil
}
// Loads all sprites in the object eagerly
func (o *Object) LoadSprites() error {
for i := 0; i < o.NumSprites; i++ {
if _, err := o.Sprite(i); err != nil {
return err
}
}
return nil
}
func (o *Object) Sprite(idx int) (*Sprite, error) { func (o *Object) Sprite(idx int) (*Sprite, error) {
if sprite := o.sprites[idx]; sprite != nil { if sprite := o.sprites[idx]; sprite != nil {
return sprite, nil return sprite, nil

View File

@@ -14,11 +14,12 @@ var (
type Set struct { type Set struct {
assets *AssetStore assets *AssetStore
raw *sets.MapSet raw *sets.MapSet
NumObjects int
} }
func (s *Set) Object(idx int) (*Object, error) { func (s *Set) Object(idx int) (*Object, error) {
if idx < 0 || idx >= len(s.raw.Palette) { if idx < 0 || idx >= s.NumObjects {
return nil, ErrOutOfBounds return nil, ErrOutOfBounds
} }
@@ -27,6 +28,7 @@ func (s *Set) Object(idx int) (*Object, error) {
func (a *AssetStore) Set(name string) (*Set, error) { func (a *AssetStore) Set(name string) (*Set, error) {
name = canonical(name) name = canonical(name)
if set, ok := a.sets[name]; ok { if set, ok := a.sets[name]; ok {
return set, nil return set, nil
} }
@@ -43,8 +45,9 @@ func (a *AssetStore) Set(name string) (*Set, error) {
} }
set := &Set{ set := &Set{
assets: a, assets: a,
raw: raw, raw: raw,
NumObjects: len(raw.Palette),
} }
a.sets[name] = set a.sets[name] = set

View File

@@ -1,23 +0,0 @@
package conv
import (
"fmt"
"code.ur.gs/lupine/ordoor/internal/fonts"
)
type Font struct {
Name string
}
func (f *Font) Output(to interface{}, m interface{}, format string, args ...interface{}) {
// FIXME: actually output some text onto screen
fmt.Printf(format+"\n", args...)
}
func ConvertFont(font *fonts.Font) *Font {
// FIXME: actually use the pixel data in font
return &Font{Name: font.Name}
}

View File

@@ -1,59 +0,0 @@
package conv
import (
"github.com/hajimehoshi/ebiten"
"code.ur.gs/lupine/ordoor/internal/data"
)
// Important conversions:
//
// * Width & height now stored using int
// * Colour data is now 32-bit rather than using a palette
type Sprite struct {
XOffset int
YOffset int
Width int
Height int
Image *ebiten.Image
}
type Object struct {
Name string
Sprites []*Sprite
}
func MapByName(objects []*Object) map[string]*Object {
out := make(map[string]*Object, len(objects))
for _, obj := range objects {
out[obj.Name] = obj
}
return out
}
func ConvertObject(rawObj *data.Object, name string) (*Object, error) {
out := &Object{
Name: name,
Sprites: make([]*Sprite, len(rawObj.Sprites)),
}
for i, rawSpr := range rawObj.Sprites {
ebitenImage, err := ebiten.NewImageFromImage(rawSpr.ToImage(), ebiten.FilterDefault)
if err != nil {
return nil, err
}
out.Sprites[i] = &Sprite{
XOffset: int(rawSpr.XOffset),
YOffset: int(rawSpr.YOffset),
Width: int(rawSpr.Width),
Height: int(rawSpr.Height),
Image: ebitenImage,
}
}
return out, nil
}