Compare commits
15 Commits
release-v0
...
master
Author | SHA1 | Date | |
---|---|---|---|
965b7701bc | |||
6d65d6b783 | |||
4ad0995076 | |||
c6b9153782 | |||
7ed2ca2ca0 | |||
0f232e3c92 | |||
a51d02daf6 | |||
69977a3096 | |||
373f5800ea | |||
ea62e2dca4 | |||
90daf028e1 | |||
311a15e3e0 | |||
a0de8fd135 | |||
3720389fef | |||
b2e8ec4f3d |
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2018 Zolfite
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
BIN
background.png
Normal file
BIN
background.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 17 KiB |
BIN
hiking.png
BIN
hiking.png
Binary file not shown.
Before Width: | Height: | Size: 68 KiB |
158
main.go
158
main.go
@ -1,28 +1,25 @@
|
|||||||
// main
|
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"image"
|
"image"
|
||||||
_ "image/png"
|
"math"
|
||||||
"os"
|
"os"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
_ "image/png"
|
||||||
|
|
||||||
"github.com/faiface/pixel"
|
"github.com/faiface/pixel"
|
||||||
"github.com/faiface/pixel/pixelgl"
|
"github.com/faiface/pixel/pixelgl"
|
||||||
"golang.org/x/image/colornames"
|
"golang.org/x/image/colornames"
|
||||||
)
|
)
|
||||||
|
|
||||||
//open file from the system
|
|
||||||
func loadPicture(path string) (pixel.Picture, error) {
|
func loadPicture(path string) (pixel.Picture, error) {
|
||||||
file, err := os.Open(path)
|
file, err := os.Open(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
//user 'defer' to close the file later on
|
|
||||||
defer file.Close()
|
defer file.Close()
|
||||||
|
|
||||||
//decode the file to find it's type
|
|
||||||
img, _, err := image.Decode(file)
|
img, _, err := image.Decode(file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -30,59 +27,144 @@ func loadPicture(path string) (pixel.Picture, error) {
|
|||||||
return pixel.PictureDataFromImage(img), nil
|
return pixel.PictureDataFromImage(img), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
//Pixel's main function (because GO forced them to do it this way)
|
|
||||||
func run() {
|
func run() {
|
||||||
//set the Window parameters
|
|
||||||
cfg := pixelgl.WindowConfig{
|
cfg := pixelgl.WindowConfig{
|
||||||
Title: "Pixel Rocks!",
|
Title: "Project RPG",
|
||||||
Bounds: pixel.R(0, 0, 1024, 768),
|
Bounds: pixel.R(0, 0, 1024, 768),
|
||||||
VSync: true,
|
VSync: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
//create a window and pass the parameters
|
|
||||||
win, err := pixelgl.NewWindow(cfg)
|
win, err := pixelgl.NewWindow(cfg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
//smooth the edges of the image (antialiasing)
|
spritesheet, err := loadPicture("sprites.png")
|
||||||
win.SetSmooth(true)
|
|
||||||
|
|
||||||
//load a specific image and catch any errors
|
|
||||||
pic, err := loadPicture("hiking.png")
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
//batch := pixel.NewBatch(&pixel.TrianglesData{}, spritesheet)
|
||||||
|
|
||||||
//create a sprite using the loaded image
|
background, err := loadPicture("background.png")
|
||||||
sprite := pixel.NewSprite(pic, pic.Bounds())
|
|
||||||
|
|
||||||
//set the angle of the image
|
if err != nil {
|
||||||
angle := 0.0
|
panic(err)
|
||||||
|
}
|
||||||
|
bg := pixel.NewSprite(background, background.Bounds())
|
||||||
|
//batch := pixel.NewBatch(&pixel.TrianglesData{}, spritesheet)
|
||||||
|
|
||||||
//set the time to compare later
|
/*
|
||||||
last := time.Now()
|
When creating the sprite, you need to know big each frame is inside the spritesheet.
|
||||||
|
If you use aseprite or some other pixel art program, you can find the number of pixels
|
||||||
|
within each frame. For our sprite sheet, each frame is 96 px wide and 96 px high. So
|
||||||
|
we increment it by 96 for both x and y so that it can get the position of the frame
|
||||||
|
at the bottom right corner.
|
||||||
|
*/
|
||||||
|
var spritesFrames []pixel.Rect
|
||||||
|
for x := spritesheet.Bounds().Min.X; x < spritesheet.Bounds().Max.X; x += 96 {
|
||||||
|
for y := spritesheet.Bounds().Min.Y; y < spritesheet.Bounds().Max.Y; y += 96 {
|
||||||
|
spritesFrames = append(spritesFrames, pixel.R(x, y, x+96, y+96)) // (x, y, width, height) of frame
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Sprite := pixel.NewSprite(spritesheet, spritesFrames[3])
|
||||||
|
|
||||||
//loop until window is closed by "X" button
|
var (
|
||||||
|
camPos = pixel.ZV
|
||||||
|
//camSpeed = 500.0
|
||||||
|
camZoom = 0.3
|
||||||
|
camZoomSpeed = 1.2
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
frames = 0
|
||||||
|
second = time.Tick(time.Second)
|
||||||
|
)
|
||||||
|
playerX := win.Bounds().Max.X / 2
|
||||||
|
playerY := win.Bounds().Max.Y / 2
|
||||||
|
|
||||||
|
var (
|
||||||
|
playerXY = pixel.Vec{
|
||||||
|
playerX,
|
||||||
|
playerY,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
var fps int64 = 60
|
||||||
|
timePerFrame := 1000000000 / fps
|
||||||
|
var lastTime int64 = time.Now().UnixNano()
|
||||||
|
var now int64
|
||||||
|
var dt int64
|
||||||
|
|
||||||
|
// Game Loop
|
||||||
for !win.Closed() {
|
for !win.Closed() {
|
||||||
//compare the last time to delta time
|
now = time.Now().UnixNano()
|
||||||
dt := time.Since(last).Seconds()
|
dt += (now - lastTime)
|
||||||
last = time.Now()
|
lastTime = time.Now().UnixNano()
|
||||||
|
|
||||||
//set the angle according to delta time
|
//Execute a frame.
|
||||||
angle += 3 * dt
|
if dt >= timePerFrame {
|
||||||
|
|
||||||
//refresh screen and set background color
|
// *** Update begins *** //
|
||||||
win.Clear(colornames.Firebrick)
|
|
||||||
|
|
||||||
//set the matrix and location of spirte in window
|
playerXY = pixel.Vec{
|
||||||
mat := pixel.IM
|
playerX,
|
||||||
mat = mat.Rotated(pixel.ZV, angle)
|
playerY,
|
||||||
mat = mat.Moved(win.Bounds().Center())
|
}
|
||||||
sprite.Draw(win, mat)
|
cam := pixel.IM.Scaled(camPos, camZoom).Moved(win.Bounds().Center().Sub(camPos))
|
||||||
|
|
||||||
//update the window
|
win.SetMatrix(cam)
|
||||||
win.Update()
|
/*
|
||||||
|
This is where Sprite and bg was originally. Creating them over and over inside
|
||||||
|
the main loop is not very good resource management. When i added my own dt, the game was
|
||||||
|
only going about 22 fps, when it should've been going 60. I moved them
|
||||||
|
out of the loop and created them up top. It fixed it and now runs faster.
|
||||||
|
|
||||||
|
the tutorial only has it in the loop with the exception that it creates it only
|
||||||
|
if you click the button to make a tree. It only created it once and added it.
|
||||||
|
*/
|
||||||
|
|
||||||
|
//mouse := cam.Unproject(win.MousePosition())
|
||||||
|
|
||||||
|
speed := 10.0
|
||||||
|
if win.Pressed(pixelgl.KeyA) { //left
|
||||||
|
playerX -= speed
|
||||||
|
Sprite.Set(spritesheet, spritesFrames[2])
|
||||||
|
}
|
||||||
|
if win.Pressed(pixelgl.KeyD) { //Right
|
||||||
|
playerX += speed
|
||||||
|
Sprite.Set(spritesheet, spritesFrames[1])
|
||||||
|
}
|
||||||
|
if win.Pressed(pixelgl.KeyS) { //Down
|
||||||
|
playerY -= speed
|
||||||
|
Sprite.Set(spritesheet, spritesFrames[3])
|
||||||
|
}
|
||||||
|
if win.Pressed(pixelgl.KeyW) { //up
|
||||||
|
playerY += speed
|
||||||
|
Sprite.Set(spritesheet, spritesFrames[0])
|
||||||
|
}
|
||||||
|
camPos.X = playerX
|
||||||
|
camPos.Y = playerY
|
||||||
|
camZoom *= math.Pow(camZoomSpeed, win.MouseScroll().Y)
|
||||||
|
|
||||||
|
// *** Update Ends *** //
|
||||||
|
|
||||||
|
// *** Render begin *** //
|
||||||
|
win.Clear(colornames.Forestgreen)
|
||||||
|
bg.Draw(win, pixel.IM.Moved(win.Bounds().Center()))
|
||||||
|
Sprite.Draw(win, pixel.IM.Scaled(pixel.ZV, 4).Moved(playerXY))
|
||||||
|
|
||||||
|
//batch.Draw(win)
|
||||||
|
win.Update()
|
||||||
|
|
||||||
|
// *** Render End *** //
|
||||||
|
|
||||||
|
frames++
|
||||||
|
select {
|
||||||
|
case <-second:
|
||||||
|
win.SetTitle(fmt.Sprintf("%s | FPS: %d", cfg.Title, frames))
|
||||||
|
frames = 0
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BIN
sprites.png
Normal file
BIN
sprites.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 62 KiB |
Reference in New Issue
Block a user