2018-07-26 02:38:54 +00:00
|
|
|
package doodle
|
|
|
|
|
|
|
|
import (
|
2020-01-02 01:50:15 +00:00
|
|
|
"fmt"
|
|
|
|
|
2019-04-10 00:35:44 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/balance"
|
2019-06-24 00:52:48 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/branding"
|
2019-06-25 21:57:11 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/level"
|
|
|
|
"git.kirsle.net/apps/doodle/pkg/log"
|
2020-01-02 01:50:15 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/native"
|
2019-06-28 05:54:46 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/scripting"
|
2021-04-01 02:16:33 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/shmem"
|
2019-06-25 21:57:11 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/uix"
|
2020-01-02 01:50:15 +00:00
|
|
|
"git.kirsle.net/apps/doodle/pkg/updater"
|
2019-12-28 03:16:34 +00:00
|
|
|
"git.kirsle.net/go/render"
|
|
|
|
"git.kirsle.net/go/render/event"
|
|
|
|
"git.kirsle.net/go/ui"
|
2018-07-26 02:38:54 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// MainScene implements the main menu of Doodle.
|
|
|
|
type MainScene struct {
|
2018-07-26 03:25:02 +00:00
|
|
|
Supervisor *ui.Supervisor
|
2019-06-25 21:57:11 +00:00
|
|
|
|
|
|
|
// Background wallpaper canvas.
|
2019-06-28 05:54:46 +00:00
|
|
|
scripting *scripting.Supervisor
|
|
|
|
canvas *uix.Canvas
|
2020-01-02 01:50:15 +00:00
|
|
|
|
|
|
|
// UI components.
|
|
|
|
labelTitle *ui.Label
|
|
|
|
labelVersion *ui.Label
|
2021-04-01 02:16:33 +00:00
|
|
|
labelHint *ui.Label
|
2020-01-02 01:50:15 +00:00
|
|
|
frame *ui.Frame // Main button frame
|
|
|
|
|
|
|
|
// Update check variables.
|
|
|
|
updateButton *ui.Button
|
|
|
|
updateInfo updater.VersionInfo
|
2018-07-26 02:38:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Name of the scene.
|
|
|
|
func (s *MainScene) Name() string {
|
|
|
|
return "Main"
|
|
|
|
}
|
|
|
|
|
|
|
|
// Setup the scene.
|
|
|
|
func (s *MainScene) Setup(d *Doodle) error {
|
2018-07-26 03:25:02 +00:00
|
|
|
s.Supervisor = ui.NewSupervisor()
|
|
|
|
|
2019-06-28 05:54:46 +00:00
|
|
|
if err := s.SetupDemoLevel(d); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-06-25 21:57:11 +00:00
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
// Main title label
|
|
|
|
s.labelTitle = ui.NewLabel(ui.Label{
|
|
|
|
Text: branding.AppName,
|
2020-04-22 06:50:45 +00:00
|
|
|
Font: balance.TitleScreenFont,
|
2020-01-02 01:50:15 +00:00
|
|
|
})
|
|
|
|
s.labelTitle.Compute(d.Engine)
|
|
|
|
|
|
|
|
// Version label.
|
|
|
|
var shareware string
|
|
|
|
if balance.FreeVersion {
|
|
|
|
shareware = " (shareware)"
|
|
|
|
}
|
|
|
|
ver := ui.NewLabel(ui.Label{
|
|
|
|
Text: fmt.Sprintf("v%s%s", branding.Version, shareware),
|
|
|
|
Font: render.Text{
|
|
|
|
Size: 18,
|
|
|
|
Color: render.Grey,
|
|
|
|
Shadow: render.Black,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
ver.Compute(d.Engine)
|
|
|
|
s.labelVersion = ver
|
|
|
|
|
2021-04-01 02:16:33 +00:00
|
|
|
// Arrow Keys hint label (scroll the demo level).
|
|
|
|
s.labelHint = ui.NewLabel(ui.Label{
|
|
|
|
Text: "Hint: press the Arrow keys",
|
|
|
|
Font: render.Text{
|
|
|
|
Size: 16,
|
|
|
|
Color: render.Grey,
|
|
|
|
Shadow: render.Purple,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
s.labelHint.Compute(d.Engine)
|
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
// "Update Available" button.
|
|
|
|
s.updateButton = ui.NewButton("Update Button", ui.NewLabel(ui.Label{
|
|
|
|
Text: "An update is available!",
|
|
|
|
Font: render.Text{
|
|
|
|
FontFilename: "DejaVuSans-Bold.ttf",
|
|
|
|
Size: 16,
|
|
|
|
Color: render.Blue,
|
|
|
|
Padding: 4,
|
|
|
|
},
|
|
|
|
}))
|
2020-04-07 06:21:17 +00:00
|
|
|
s.updateButton.Handle(ui.Click, func(ed ui.EventData) error {
|
2020-01-02 01:50:15 +00:00
|
|
|
native.OpenURL(s.updateInfo.DownloadURL)
|
2020-04-07 06:21:17 +00:00
|
|
|
return nil
|
2020-01-02 01:50:15 +00:00
|
|
|
})
|
|
|
|
s.updateButton.Compute(d.Engine)
|
|
|
|
s.updateButton.Hide()
|
|
|
|
s.Supervisor.Add(s.updateButton)
|
|
|
|
|
2019-06-25 21:57:11 +00:00
|
|
|
// Main UI button frame.
|
2018-08-02 01:52:52 +00:00
|
|
|
frame := ui.NewFrame("frame")
|
2018-08-01 00:18:13 +00:00
|
|
|
s.frame = frame
|
|
|
|
|
Add Switches, Fire/Water Collision and Play Menu
* New doodads: Switches.
* They come in four varieties: wall switch (background element, with
"ON/OFF" text) and three side-profile switches for the floor, left
or right walls.
* On collision with the player, they flip their state from "OFF" to
"ON" or vice versa. If the player walks away and then collides
again, the switch flips again.
* Can be used to open/close Electric Doors when turned on/off. Their
default state is "off"
* If a switch receives a power signal from another linked switch, it
sets its own state to match. So, two "on/off" switches that are
connected to a door AND to each other will both flip on/off when one
of them flips.
* Update the Level Collision logic to support Decoration, Fire and Water
pixel collisions.
* Previously, ALL pixels in the level were acting as though solid.
* Non-solid pixels don't count for collision detection, but their
attributes (fire and water) are collected and returned.
* Updated the MenuScene to support loading a map file in Play Mode
instead of Edit Mode. Updated the title screen menu to add a button
for playing levels instead of editing them.
* Wrote some documentation.
2019-07-07 01:30:03 +00:00
|
|
|
var buttons = []struct {
|
|
|
|
Name string
|
|
|
|
Func func()
|
|
|
|
}{
|
2020-04-22 06:50:45 +00:00
|
|
|
// {
|
|
|
|
// Name: "Story Mode",
|
|
|
|
// Func: d.GotoStoryMenu,
|
|
|
|
// },
|
Add Switches, Fire/Water Collision and Play Menu
* New doodads: Switches.
* They come in four varieties: wall switch (background element, with
"ON/OFF" text) and three side-profile switches for the floor, left
or right walls.
* On collision with the player, they flip their state from "OFF" to
"ON" or vice versa. If the player walks away and then collides
again, the switch flips again.
* Can be used to open/close Electric Doors when turned on/off. Their
default state is "off"
* If a switch receives a power signal from another linked switch, it
sets its own state to match. So, two "on/off" switches that are
connected to a door AND to each other will both flip on/off when one
of them flips.
* Update the Level Collision logic to support Decoration, Fire and Water
pixel collisions.
* Previously, ALL pixels in the level were acting as though solid.
* Non-solid pixels don't count for collision detection, but their
attributes (fire and water) are collected and returned.
* Updated the MenuScene to support loading a map file in Play Mode
instead of Edit Mode. Updated the title screen menu to add a button
for playing levels instead of editing them.
* Wrote some documentation.
2019-07-07 01:30:03 +00:00
|
|
|
{
|
|
|
|
Name: "Play a Level",
|
|
|
|
Func: d.GotoPlayMenu,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "Create a New Level",
|
|
|
|
Func: d.GotoNewMenu,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "Edit a Level",
|
|
|
|
Func: d.GotoLoadMenu,
|
|
|
|
},
|
2020-09-19 05:52:05 +00:00
|
|
|
// {
|
|
|
|
// Name: "Settings",
|
|
|
|
// Func: d.GotoSettingsMenu,
|
|
|
|
// },
|
Add Switches, Fire/Water Collision and Play Menu
* New doodads: Switches.
* They come in four varieties: wall switch (background element, with
"ON/OFF" text) and three side-profile switches for the floor, left
or right walls.
* On collision with the player, they flip their state from "OFF" to
"ON" or vice versa. If the player walks away and then collides
again, the switch flips again.
* Can be used to open/close Electric Doors when turned on/off. Their
default state is "off"
* If a switch receives a power signal from another linked switch, it
sets its own state to match. So, two "on/off" switches that are
connected to a door AND to each other will both flip on/off when one
of them flips.
* Update the Level Collision logic to support Decoration, Fire and Water
pixel collisions.
* Previously, ALL pixels in the level were acting as though solid.
* Non-solid pixels don't count for collision detection, but their
attributes (fire and water) are collected and returned.
* Updated the MenuScene to support loading a map file in Play Mode
instead of Edit Mode. Updated the title screen menu to add a button
for playing levels instead of editing them.
* Wrote some documentation.
2019-07-07 01:30:03 +00:00
|
|
|
}
|
|
|
|
for _, button := range buttons {
|
|
|
|
button := button
|
|
|
|
btn := ui.NewButton(button.Name, ui.NewLabel(ui.Label{
|
|
|
|
Text: button.Name,
|
|
|
|
Font: balance.StatusFont,
|
|
|
|
}))
|
2020-04-07 06:21:17 +00:00
|
|
|
btn.Handle(ui.Click, func(ed ui.EventData) error {
|
Add Switches, Fire/Water Collision and Play Menu
* New doodads: Switches.
* They come in four varieties: wall switch (background element, with
"ON/OFF" text) and three side-profile switches for the floor, left
or right walls.
* On collision with the player, they flip their state from "OFF" to
"ON" or vice versa. If the player walks away and then collides
again, the switch flips again.
* Can be used to open/close Electric Doors when turned on/off. Their
default state is "off"
* If a switch receives a power signal from another linked switch, it
sets its own state to match. So, two "on/off" switches that are
connected to a door AND to each other will both flip on/off when one
of them flips.
* Update the Level Collision logic to support Decoration, Fire and Water
pixel collisions.
* Previously, ALL pixels in the level were acting as though solid.
* Non-solid pixels don't count for collision detection, but their
attributes (fire and water) are collected and returned.
* Updated the MenuScene to support loading a map file in Play Mode
instead of Edit Mode. Updated the title screen menu to add a button
for playing levels instead of editing them.
* Wrote some documentation.
2019-07-07 01:30:03 +00:00
|
|
|
button.Func()
|
2020-04-07 06:21:17 +00:00
|
|
|
return nil
|
Add Switches, Fire/Water Collision and Play Menu
* New doodads: Switches.
* They come in four varieties: wall switch (background element, with
"ON/OFF" text) and three side-profile switches for the floor, left
or right walls.
* On collision with the player, they flip their state from "OFF" to
"ON" or vice versa. If the player walks away and then collides
again, the switch flips again.
* Can be used to open/close Electric Doors when turned on/off. Their
default state is "off"
* If a switch receives a power signal from another linked switch, it
sets its own state to match. So, two "on/off" switches that are
connected to a door AND to each other will both flip on/off when one
of them flips.
* Update the Level Collision logic to support Decoration, Fire and Water
pixel collisions.
* Previously, ALL pixels in the level were acting as though solid.
* Non-solid pixels don't count for collision detection, but their
attributes (fire and water) are collected and returned.
* Updated the MenuScene to support loading a map file in Play Mode
instead of Edit Mode. Updated the title screen menu to add a button
for playing levels instead of editing them.
* Wrote some documentation.
2019-07-07 01:30:03 +00:00
|
|
|
})
|
|
|
|
s.Supervisor.Add(btn)
|
|
|
|
frame.Pack(btn, ui.Pack{
|
2019-12-29 05:48:49 +00:00
|
|
|
Side: ui.N,
|
2019-12-31 02:13:28 +00:00
|
|
|
PadY: 8,
|
Add Switches, Fire/Water Collision and Play Menu
* New doodads: Switches.
* They come in four varieties: wall switch (background element, with
"ON/OFF" text) and three side-profile switches for the floor, left
or right walls.
* On collision with the player, they flip their state from "OFF" to
"ON" or vice versa. If the player walks away and then collides
again, the switch flips again.
* Can be used to open/close Electric Doors when turned on/off. Their
default state is "off"
* If a switch receives a power signal from another linked switch, it
sets its own state to match. So, two "on/off" switches that are
connected to a door AND to each other will both flip on/off when one
of them flips.
* Update the Level Collision logic to support Decoration, Fire and Water
pixel collisions.
* Previously, ALL pixels in the level were acting as though solid.
* Non-solid pixels don't count for collision detection, but their
attributes (fire and water) are collected and returned.
* Updated the MenuScene to support loading a map file in Play Mode
instead of Edit Mode. Updated the title screen menu to add a button
for playing levels instead of editing them.
* Wrote some documentation.
2019-07-07 01:30:03 +00:00
|
|
|
// Fill: true,
|
|
|
|
FillX: true,
|
|
|
|
})
|
|
|
|
}
|
2018-07-26 03:25:02 +00:00
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
// Check for update in the background.
|
|
|
|
go s.checkUpdate()
|
|
|
|
|
2018-07-26 02:38:54 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
// checkUpdate checks for a version update and shows the button.
|
|
|
|
func (s *MainScene) checkUpdate() {
|
2021-04-01 02:16:33 +00:00
|
|
|
if shmem.OfflineMode {
|
|
|
|
log.Info("OfflineMode: skip updates check")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
info, err := updater.Check()
|
|
|
|
if err != nil {
|
|
|
|
log.Error(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if info.LatestVersion != branding.Version {
|
|
|
|
s.updateInfo = info
|
|
|
|
s.updateButton.Show()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-28 05:54:46 +00:00
|
|
|
// SetupDemoLevel configures the wallpaper behind the New screen,
|
|
|
|
// which demos a title screen demo level.
|
|
|
|
func (s *MainScene) SetupDemoLevel(d *Doodle) error {
|
|
|
|
// Set up the background wallpaper canvas.
|
|
|
|
s.canvas = uix.NewCanvas(100, false)
|
|
|
|
s.canvas.Scrollable = true
|
|
|
|
s.canvas.Resize(render.Rect{
|
2019-12-28 03:16:34 +00:00
|
|
|
W: d.width,
|
|
|
|
H: d.height,
|
2019-06-28 05:54:46 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
s.scripting = scripting.NewSupervisor()
|
|
|
|
s.canvas.SetScriptSupervisor(s.scripting)
|
|
|
|
|
2019-07-07 03:31:50 +00:00
|
|
|
// Title screen level to load.
|
2020-09-19 05:35:43 +00:00
|
|
|
if lvl, err := level.LoadFile(balance.DemoLevelName); err == nil {
|
2019-07-07 03:31:50 +00:00
|
|
|
s.canvas.LoadLevel(d.Engine, lvl)
|
|
|
|
s.canvas.InstallActors(lvl.Actors)
|
|
|
|
|
|
|
|
// Load all actor scripts.
|
|
|
|
if err := s.scripting.InstallScripts(lvl); err != nil {
|
|
|
|
log.Error("Error with title screen level scripts: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run all actors scripts main function to start them off.
|
|
|
|
if err := s.canvas.InstallScripts(); err != nil {
|
|
|
|
log.Error("Error running actor main() functions: %s", err)
|
|
|
|
}
|
|
|
|
} else {
|
2020-11-15 23:20:15 +00:00
|
|
|
log.Error("Error loading demo level %s: %s", balance.DemoLevelName, err)
|
2019-06-28 05:54:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:38:54 +00:00
|
|
|
// Loop the editor scene.
|
2019-12-22 22:11:01 +00:00
|
|
|
func (s *MainScene) Loop(d *Doodle, ev *event.State) error {
|
2018-07-26 03:25:02 +00:00
|
|
|
s.Supervisor.Loop(ev)
|
2019-06-25 21:57:11 +00:00
|
|
|
|
2019-06-28 05:54:46 +00:00
|
|
|
if err := s.scripting.Loop(); err != nil {
|
|
|
|
log.Error("MainScene.Loop: scripting.Loop: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.canvas.Loop(ev)
|
|
|
|
|
2019-12-22 22:11:01 +00:00
|
|
|
if ev.WindowResized {
|
2019-06-25 21:57:11 +00:00
|
|
|
w, h := d.Engine.WindowSize()
|
|
|
|
d.width = w
|
|
|
|
d.height = h
|
|
|
|
log.Info("Resized to %dx%d", d.width, d.height)
|
|
|
|
s.canvas.Resize(render.Rect{
|
2019-12-28 03:16:34 +00:00
|
|
|
W: d.width,
|
|
|
|
H: d.height,
|
2019-06-25 21:57:11 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-07-26 02:38:54 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw the pixels on this frame.
|
|
|
|
func (s *MainScene) Draw(d *Doodle) error {
|
|
|
|
// Clear the canvas and fill it with white.
|
|
|
|
d.Engine.Clear(render.White)
|
|
|
|
|
2019-06-25 21:57:11 +00:00
|
|
|
s.canvas.Present(d.Engine, render.Origin)
|
|
|
|
|
2019-06-28 05:54:46 +00:00
|
|
|
// Draw a sheen over the level for clarity.
|
|
|
|
d.Engine.DrawBox(render.RGBA(255, 255, 254, 128), render.Rect{
|
|
|
|
X: 0,
|
|
|
|
Y: 0,
|
2019-12-28 03:16:34 +00:00
|
|
|
W: d.width,
|
|
|
|
H: d.height,
|
2019-06-28 05:54:46 +00:00
|
|
|
})
|
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
// App title label.
|
|
|
|
s.labelTitle.MoveTo(render.Point{
|
|
|
|
X: (d.width / 2) - (s.labelTitle.Size().W / 2),
|
2018-07-26 02:38:54 +00:00
|
|
|
Y: 120,
|
|
|
|
})
|
2020-01-02 01:50:15 +00:00
|
|
|
s.labelTitle.Present(d.Engine, s.labelTitle.Point())
|
|
|
|
|
|
|
|
// Version label
|
|
|
|
s.labelVersion.MoveTo(render.Point{
|
|
|
|
X: (d.width / 2) - (s.labelVersion.Size().W / 2),
|
|
|
|
Y: s.labelTitle.Point().Y + s.labelTitle.Size().H + 8,
|
|
|
|
})
|
|
|
|
s.labelVersion.Present(d.Engine, s.labelVersion.Point())
|
|
|
|
|
2021-04-01 02:16:33 +00:00
|
|
|
// Hint label.
|
|
|
|
s.labelHint.MoveTo(render.Point{
|
|
|
|
X: d.width - s.labelHint.Size().W - 32,
|
|
|
|
Y: d.height - s.labelHint.Size().H - 32,
|
|
|
|
})
|
|
|
|
s.labelHint.Present(d.Engine, s.labelHint.Point())
|
|
|
|
|
2020-01-02 01:50:15 +00:00
|
|
|
// Update button.
|
|
|
|
s.updateButton.MoveTo(render.Point{
|
|
|
|
X: 24,
|
|
|
|
Y: d.height - s.updateButton.Size().H - 24,
|
|
|
|
})
|
|
|
|
s.updateButton.Present(d.Engine, s.updateButton.Point())
|
2018-07-26 02:38:54 +00:00
|
|
|
|
2018-08-01 00:18:13 +00:00
|
|
|
s.frame.Compute(d.Engine)
|
|
|
|
s.frame.MoveTo(render.Point{
|
2019-12-28 03:16:34 +00:00
|
|
|
X: (d.width / 2) - (s.frame.Size().W / 2),
|
2019-06-27 01:36:54 +00:00
|
|
|
Y: 260,
|
2018-08-01 00:18:13 +00:00
|
|
|
})
|
2018-08-05 19:54:57 +00:00
|
|
|
s.frame.Present(d.Engine, s.frame.Point())
|
2018-07-26 02:38:54 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroy the scene.
|
|
|
|
func (s *MainScene) Destroy() error {
|
|
|
|
return nil
|
|
|
|
}
|