2015-01-11 17:54:18 +01:00
|
|
|
// Copyright 2015 Hajime Hoshi
|
|
|
|
//
|
|
|
|
// 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 main
|
|
|
|
|
|
|
|
import (
|
2015-01-12 05:33:21 +01:00
|
|
|
"fmt"
|
2015-01-11 17:54:18 +01:00
|
|
|
"log"
|
2020-09-21 10:30:28 +02:00
|
|
|
"sort"
|
2015-01-11 17:54:18 +01:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2022-07-09 09:58:54 +02:00
|
|
|
"time"
|
2016-02-15 17:13:04 +01:00
|
|
|
|
2020-10-03 19:35:13 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/v2"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/ebitenutil"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/inpututil"
|
2015-01-11 17:54:18 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2017-10-25 20:44:49 +02:00
|
|
|
screenWidth = 640
|
|
|
|
screenHeight = 480
|
2015-01-11 17:54:18 +01:00
|
|
|
)
|
|
|
|
|
2020-04-19 18:36:09 +02:00
|
|
|
type Game struct {
|
2021-07-10 15:32:15 +02:00
|
|
|
gamepadIDsBuf []ebiten.GamepadID
|
2020-10-07 18:08:30 +02:00
|
|
|
gamepadIDs map[ebiten.GamepadID]struct{}
|
|
|
|
axes map[ebiten.GamepadID][]string
|
|
|
|
pressedButtons map[ebiten.GamepadID][]string
|
2020-04-19 18:36:09 +02:00
|
|
|
}
|
|
|
|
|
2020-10-04 10:42:54 +02:00
|
|
|
func (g *Game) Update() error {
|
2020-04-19 18:36:09 +02:00
|
|
|
if g.gamepadIDs == nil {
|
2020-10-07 18:08:30 +02:00
|
|
|
g.gamepadIDs = map[ebiten.GamepadID]struct{}{}
|
2020-04-19 18:36:09 +02:00
|
|
|
}
|
2018-04-29 21:34:15 +02:00
|
|
|
|
|
|
|
// Log the gamepad connection events.
|
2021-07-10 15:32:15 +02:00
|
|
|
g.gamepadIDsBuf = inpututil.AppendJustConnectedGamepadIDs(g.gamepadIDsBuf[:0])
|
|
|
|
for _, id := range g.gamepadIDsBuf {
|
2021-07-18 08:41:20 +02:00
|
|
|
log.Printf("gamepad connected: id: %d, SDL ID: %s", id, ebiten.GamepadSDLID(id))
|
2020-04-19 18:36:09 +02:00
|
|
|
g.gamepadIDs[id] = struct{}{}
|
2018-04-29 19:51:38 +02:00
|
|
|
}
|
2020-04-19 18:36:09 +02:00
|
|
|
for id := range g.gamepadIDs {
|
2018-04-29 21:34:15 +02:00
|
|
|
if inpututil.IsGamepadJustDisconnected(id) {
|
2022-01-22 13:27:42 +01:00
|
|
|
log.Printf("gamepad disconnected: id: %d", id)
|
2020-04-19 18:36:09 +02:00
|
|
|
delete(g.gamepadIDs, id)
|
2018-04-29 21:34:15 +02:00
|
|
|
}
|
2018-04-29 19:51:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-07 18:08:30 +02:00
|
|
|
g.axes = map[ebiten.GamepadID][]string{}
|
|
|
|
g.pressedButtons = map[ebiten.GamepadID][]string{}
|
2020-04-19 18:36:09 +02:00
|
|
|
for id := range g.gamepadIDs {
|
2023-12-24 17:50:20 +01:00
|
|
|
maxAxis := ebiten.GamepadAxisType(ebiten.GamepadAxisCount(id))
|
|
|
|
for a := ebiten.GamepadAxisType(0); a < maxAxis; a++ {
|
2021-07-18 18:31:17 +02:00
|
|
|
v := ebiten.GamepadAxisValue(id, a)
|
2021-07-18 19:19:42 +02:00
|
|
|
g.axes[id] = append(g.axes[id], fmt.Sprintf("%d:%+0.2f", a, v))
|
2017-10-25 20:44:49 +02:00
|
|
|
}
|
2021-07-19 19:25:52 +02:00
|
|
|
|
2022-07-12 18:27:14 +02:00
|
|
|
maxButton := ebiten.GamepadButton(ebiten.GamepadButtonCount(id))
|
2023-08-24 09:27:35 +02:00
|
|
|
for b := ebiten.GamepadButton(0); b < maxButton; b++ {
|
2017-10-26 16:59:09 +02:00
|
|
|
if ebiten.IsGamepadButtonPressed(id, b) {
|
2020-04-19 18:36:09 +02:00
|
|
|
g.pressedButtons[id] = append(g.pressedButtons[id], strconv.Itoa(int(b)))
|
2017-10-25 20:44:49 +02:00
|
|
|
}
|
2018-02-16 19:50:43 +01:00
|
|
|
|
2018-04-29 21:34:15 +02:00
|
|
|
// Log button events.
|
2018-02-16 19:50:43 +01:00
|
|
|
if inpututil.IsGamepadButtonJustPressed(id, b) {
|
2018-04-29 19:51:38 +02:00
|
|
|
log.Printf("button pressed: id: %d, button: %d", id, b)
|
2018-02-16 19:50:43 +01:00
|
|
|
}
|
|
|
|
if inpututil.IsGamepadButtonJustReleased(id, b) {
|
2018-04-29 19:51:38 +02:00
|
|
|
log.Printf("button released: id: %d, button: %d", id, b)
|
2018-02-16 19:50:43 +01:00
|
|
|
}
|
2015-01-11 17:54:18 +01:00
|
|
|
}
|
2021-07-19 19:25:52 +02:00
|
|
|
|
2021-07-19 19:46:09 +02:00
|
|
|
if ebiten.IsStandardGamepadLayoutAvailable(id) {
|
2021-07-19 19:25:52 +02:00
|
|
|
for b := ebiten.StandardGamepadButton(0); b <= ebiten.StandardGamepadButtonMax; b++ {
|
|
|
|
// Log button events.
|
|
|
|
if inpututil.IsStandardGamepadButtonJustPressed(id, b) {
|
2022-07-09 09:58:54 +02:00
|
|
|
var strong float64
|
|
|
|
var weak float64
|
|
|
|
switch b {
|
|
|
|
case ebiten.StandardGamepadButtonLeftTop,
|
|
|
|
ebiten.StandardGamepadButtonLeftLeft,
|
|
|
|
ebiten.StandardGamepadButtonLeftRight,
|
|
|
|
ebiten.StandardGamepadButtonLeftBottom:
|
|
|
|
weak = 0.5
|
|
|
|
case ebiten.StandardGamepadButtonRightTop,
|
|
|
|
ebiten.StandardGamepadButtonRightLeft,
|
|
|
|
ebiten.StandardGamepadButtonRightRight,
|
|
|
|
ebiten.StandardGamepadButtonRightBottom:
|
|
|
|
strong = 0.5
|
|
|
|
}
|
|
|
|
if strong > 0 || weak > 0 {
|
|
|
|
op := &ebiten.VibrateGamepadOptions{
|
|
|
|
Duration: 200 * time.Millisecond,
|
|
|
|
StrongMagnitude: strong,
|
|
|
|
WeakMagnitude: weak,
|
|
|
|
}
|
|
|
|
ebiten.VibrateGamepad(id, op)
|
|
|
|
}
|
2021-07-19 19:25:52 +02:00
|
|
|
log.Printf("standard button pressed: id: %d, button: %d", id, b)
|
|
|
|
}
|
|
|
|
if inpututil.IsStandardGamepadButtonJustReleased(id, b) {
|
|
|
|
log.Printf("standard button released: id: %d, button: %d", id, b)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-11 17:54:18 +01:00
|
|
|
}
|
2020-04-19 18:36:09 +02:00
|
|
|
return nil
|
|
|
|
}
|
2017-10-25 20:44:49 +02:00
|
|
|
|
2021-09-12 13:13:23 +02:00
|
|
|
var standardButtonToString = map[ebiten.StandardGamepadButton]string{
|
|
|
|
ebiten.StandardGamepadButtonRightBottom: "RB",
|
|
|
|
ebiten.StandardGamepadButtonRightRight: "RR",
|
|
|
|
ebiten.StandardGamepadButtonRightLeft: "RL",
|
|
|
|
ebiten.StandardGamepadButtonRightTop: "RT",
|
|
|
|
ebiten.StandardGamepadButtonFrontTopLeft: "FTL",
|
|
|
|
ebiten.StandardGamepadButtonFrontTopRight: "FTR",
|
|
|
|
ebiten.StandardGamepadButtonFrontBottomLeft: "FBL",
|
|
|
|
ebiten.StandardGamepadButtonFrontBottomRight: "FBR",
|
|
|
|
ebiten.StandardGamepadButtonCenterLeft: "CL",
|
|
|
|
ebiten.StandardGamepadButtonCenterRight: "CR",
|
|
|
|
ebiten.StandardGamepadButtonLeftStick: "LS",
|
|
|
|
ebiten.StandardGamepadButtonRightStick: "RS",
|
|
|
|
ebiten.StandardGamepadButtonLeftBottom: "LB",
|
|
|
|
ebiten.StandardGamepadButtonLeftRight: "LR",
|
|
|
|
ebiten.StandardGamepadButtonLeftLeft: "LL",
|
|
|
|
ebiten.StandardGamepadButtonLeftTop: "LT",
|
|
|
|
ebiten.StandardGamepadButtonCenterCenter: "CC",
|
|
|
|
}
|
|
|
|
|
2021-07-18 16:28:26 +02:00
|
|
|
func standardMap(id ebiten.GamepadID) string {
|
2021-09-12 13:13:23 +02:00
|
|
|
m := ` [FBL ] [FBR ]
|
|
|
|
[FTL ] [FTR ]
|
2021-07-18 16:28:26 +02:00
|
|
|
|
2021-09-12 13:13:23 +02:00
|
|
|
[LT ] [CC ] [RT ]
|
|
|
|
[LL ][LR ] [CL ][CR ] [RL ][RR ]
|
|
|
|
[LB ] [RB ]
|
|
|
|
[LS ] [RS ]
|
2021-07-18 16:28:26 +02:00
|
|
|
`
|
2021-09-12 13:13:23 +02:00
|
|
|
|
|
|
|
for b, str := range standardButtonToString {
|
|
|
|
placeholder := "[" + str + strings.Repeat(" ", 4-len(str)) + "]"
|
|
|
|
v := ebiten.StandardGamepadButtonValue(id, b)
|
2022-08-11 05:35:20 +02:00
|
|
|
switch {
|
|
|
|
case !ebiten.IsStandardGamepadButtonAvailable(id, b):
|
|
|
|
m = strings.Replace(m, placeholder, " -- ", 1)
|
|
|
|
case ebiten.IsStandardGamepadButtonPressed(id, b):
|
2021-09-12 13:13:23 +02:00
|
|
|
m = strings.Replace(m, placeholder, fmt.Sprintf("[%0.2f]", v), 1)
|
2022-08-11 05:35:20 +02:00
|
|
|
default:
|
2021-09-12 13:13:23 +02:00
|
|
|
m = strings.Replace(m, placeholder, fmt.Sprintf(" %0.2f ", v), 1)
|
|
|
|
}
|
2021-07-18 16:28:26 +02:00
|
|
|
}
|
|
|
|
|
2022-08-11 05:35:20 +02:00
|
|
|
// TODO: Use ebiten.IsStandardGamepadAxisAvailable
|
2021-07-18 16:28:26 +02:00
|
|
|
m += fmt.Sprintf(" Left Stick: X: %+0.2f, Y: %+0.2f\n Right Stick: X: %+0.2f, Y: %+0.2f",
|
|
|
|
ebiten.StandardGamepadAxisValue(id, ebiten.StandardGamepadAxisLeftStickHorizontal),
|
|
|
|
ebiten.StandardGamepadAxisValue(id, ebiten.StandardGamepadAxisLeftStickVertical),
|
|
|
|
ebiten.StandardGamepadAxisValue(id, ebiten.StandardGamepadAxisRightStickHorizontal),
|
|
|
|
ebiten.StandardGamepadAxisValue(id, ebiten.StandardGamepadAxisRightStickVertical))
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2020-04-19 18:36:09 +02:00
|
|
|
func (g *Game) Draw(screen *ebiten.Image) {
|
2018-01-28 12:10:50 +01:00
|
|
|
// Draw the current gamepad status.
|
2017-10-25 20:52:23 +02:00
|
|
|
str := ""
|
2020-04-19 18:36:09 +02:00
|
|
|
if len(g.gamepadIDs) > 0 {
|
2020-10-07 18:08:30 +02:00
|
|
|
ids := make([]ebiten.GamepadID, 0, len(g.gamepadIDs))
|
2020-04-19 18:36:09 +02:00
|
|
|
for id := range g.gamepadIDs {
|
2020-09-21 10:30:28 +02:00
|
|
|
ids = append(ids, id)
|
|
|
|
}
|
2020-10-07 18:08:30 +02:00
|
|
|
sort.Slice(ids, func(a, b int) bool {
|
|
|
|
return ids[a] < ids[b]
|
|
|
|
})
|
2020-10-07 18:13:51 +02:00
|
|
|
for _, id := range ids {
|
2021-07-18 16:28:26 +02:00
|
|
|
var standard string
|
2021-07-19 19:46:09 +02:00
|
|
|
if ebiten.IsStandardGamepadLayoutAvailable(id) {
|
2021-07-18 16:28:26 +02:00
|
|
|
standard = " (Standard Layout)"
|
|
|
|
}
|
|
|
|
str += fmt.Sprintf("Gamepad (ID: %d, SDL ID: %s)%s:\n", id, ebiten.GamepadSDLID(id), standard)
|
2022-01-25 13:13:59 +01:00
|
|
|
str += fmt.Sprintf(" Name: %s\n", ebiten.GamepadName(id))
|
2020-04-19 18:36:09 +02:00
|
|
|
str += fmt.Sprintf(" Axes: %s\n", strings.Join(g.axes[id], ", "))
|
|
|
|
str += fmt.Sprintf(" Buttons: %s\n", strings.Join(g.pressedButtons[id], ", "))
|
2021-07-19 19:46:09 +02:00
|
|
|
if ebiten.IsStandardGamepadLayoutAvailable(id) {
|
2021-07-18 16:28:26 +02:00
|
|
|
str += "\n"
|
2021-09-12 13:13:23 +02:00
|
|
|
str += standardMap(id) + "\n"
|
2024-03-16 07:35:25 +01:00
|
|
|
|
|
|
|
// Print unmapped buttons.
|
|
|
|
buttonIDs := map[ebiten.GamepadButton]struct{}{}
|
|
|
|
for i := 0; i < ebiten.GamepadButtonCount(id); i++ {
|
|
|
|
buttonIDs[ebiten.GamepadButton(i)] = struct{}{}
|
|
|
|
}
|
|
|
|
for _, m := range ebiten.GamepadMapping(id) {
|
|
|
|
if m.PhysicalType == ebiten.GamepadMappingTypeButton {
|
|
|
|
delete(buttonIDs, ebiten.GamepadButton(m.PhysicalIndex))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(buttonIDs) > 0 {
|
|
|
|
unmappedButtonIDs := make([]ebiten.GamepadButton, 0, len(buttonIDs))
|
|
|
|
for id := range buttonIDs {
|
|
|
|
unmappedButtonIDs = append(unmappedButtonIDs, id)
|
|
|
|
}
|
|
|
|
sort.Slice(unmappedButtonIDs, func(i, j int) bool {
|
|
|
|
return unmappedButtonIDs[i] < unmappedButtonIDs[j]
|
|
|
|
})
|
|
|
|
str += " Unmapped Buttons: "
|
|
|
|
for i, id := range unmappedButtonIDs {
|
|
|
|
str += fmt.Sprintf("%d", id)
|
|
|
|
if i < len(unmappedButtonIDs)-1 {
|
|
|
|
str += ", "
|
|
|
|
}
|
|
|
|
}
|
|
|
|
str += "\n"
|
|
|
|
}
|
2021-07-18 16:28:26 +02:00
|
|
|
}
|
2017-10-26 16:59:09 +02:00
|
|
|
str += "\n"
|
2017-10-25 20:44:49 +02:00
|
|
|
}
|
2017-10-26 16:59:09 +02:00
|
|
|
} else {
|
|
|
|
str = "Please connect your gamepad."
|
2017-10-25 20:44:49 +02:00
|
|
|
}
|
2017-03-04 15:00:19 +01:00
|
|
|
ebitenutil.DebugPrint(screen, str)
|
2020-04-19 18:36:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Game) Layout(outsideWidth, outsideHeight int) (int, int) {
|
|
|
|
return screenWidth, screenHeight
|
2015-01-11 17:54:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
2020-04-19 18:36:09 +02:00
|
|
|
ebiten.SetWindowSize(screenWidth, screenHeight)
|
2022-08-29 04:16:39 +02:00
|
|
|
ebiten.SetWindowTitle("Gamepad (Ebitengine Demo)")
|
2020-04-19 18:36:09 +02:00
|
|
|
if err := ebiten.RunGame(&Game{}); err != nil {
|
2015-01-11 17:54:18 +01:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|