mirror of
https://github.com/onyx-and-iris/gobs-cli.git
synced 2026-04-18 07:03:37 +00:00
Compare commits
10 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 0a2c622645 | |||
| cb973c09f5 | |||
| 4fa32bfb42 | |||
| 8616f3b486 | |||
| 05f13ab87a | |||
| 71300a416b | |||
| 2fc2000b11 | |||
| 7692de752b | |||
| 107d1bca38 | |||
| 035b467a14 |
@@ -5,6 +5,12 @@ All notable changes to this project will be documented in this file.
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||
|
||||
# [0.2.0] - 2025-04-27
|
||||
|
||||
### Added
|
||||
|
||||
- sceneitem transform, see *transform* under [SceneItemCmd](https://github.com/onyx-and-iris/gobs-cli?tab=readme-ov-file#sceneitemcmd)
|
||||
|
||||
# [0.1.0] - 2025-04-24
|
||||
|
||||
### Added
|
||||
|
||||
21
LICENSE
Normal file
21
LICENSE
Normal file
@@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2025 Onyx and Iris
|
||||
|
||||
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.
|
||||
43
README.md
43
README.md
@@ -4,6 +4,12 @@ A command line interface for OBS Websocket v5
|
||||
|
||||
For an outline of past/future changes refer to: [CHANGELOG](CHANGELOG.md)
|
||||
|
||||
## Installation
|
||||
|
||||
```console
|
||||
go install github.com/onyx-and-iris/gobs-cli@latest
|
||||
```
|
||||
|
||||
## Configuration
|
||||
|
||||
#### Flags
|
||||
@@ -121,6 +127,35 @@ gobs-cli sceneitem toggle --parent=test_group START "Colour Source 3"
|
||||
gobs-cli sceneitem visible --parent=test_group START "Colour Source 4"
|
||||
```
|
||||
|
||||
- transform: Transform scene item.
|
||||
- flags:
|
||||
*optional*
|
||||
- --parent: Parent group name.
|
||||
|
||||
- --alignment: Alignment of the scene item.
|
||||
- --bounds-alignment: Bounds alignment of the scene item.
|
||||
- --bounds-height: Bounds height of the scene item.
|
||||
- --bounds-type: Bounds type of the scene item.
|
||||
- --bounds-width: Bounds width of the scene item.
|
||||
- --crop-to-bounds: Whether to crop the scene item to bounds.
|
||||
- --crop-bottom: Crop bottom value of the scene item.
|
||||
- --crop-left: Crop left value of the scene item.
|
||||
- --crop-right: Crop right value of the scene item.
|
||||
- --crop-top: Crop top value of the scene item.
|
||||
- --position-x: X position of the scene item.
|
||||
- --position-y: Y position of the scene item.
|
||||
- --rotation: Rotation of the scene item.
|
||||
- --scale-x: X scale of the scene item.
|
||||
- --scale-y: Y scale of the scene item.
|
||||
- args: SceneName ItemName
|
||||
|
||||
```console
|
||||
gobs-cli sceneitem transform \
|
||||
--rotation=5 \
|
||||
--position-x=250.8 \
|
||||
Scene "Colour Source 3"
|
||||
```
|
||||
|
||||
### GroupCmd
|
||||
|
||||
- list: List all groups.
|
||||
@@ -273,7 +308,7 @@ gobs-cli scenecollection list
|
||||
gobs-cli scenecollection current
|
||||
```
|
||||
|
||||
- switch: "Switch scene collection.
|
||||
- switch: Switch scene collection.
|
||||
- args: Name
|
||||
|
||||
```console
|
||||
@@ -305,21 +340,21 @@ gobs-cli profile current
|
||||
- args: Name
|
||||
|
||||
```console
|
||||
gobs-cli profile switch test-collection
|
||||
gobs-cli profile switch test-profile
|
||||
```
|
||||
|
||||
- create: Create profile.
|
||||
- args: Name
|
||||
|
||||
```console
|
||||
gobs-cli profile create test-collection
|
||||
gobs-cli profile create test-profile
|
||||
```
|
||||
|
||||
- remove: Remove profile.
|
||||
- args: Name
|
||||
|
||||
```console
|
||||
gobs-cli profile create test-collection
|
||||
gobs-cli profile remove test-profile
|
||||
```
|
||||
|
||||
### ReplayBufferCmd
|
||||
|
||||
@@ -50,3 +50,8 @@ tasks:
|
||||
desc: Clean the build artifacts
|
||||
cmds:
|
||||
- '{{.SHELL}} rm -r {{.BIN_DIR}}'
|
||||
|
||||
manfile:
|
||||
desc: Generate manfile
|
||||
cmds:
|
||||
- go run . --man > manfile
|
||||
|
||||
3
go.mod
3
go.mod
@@ -4,6 +4,7 @@ go 1.24.0
|
||||
|
||||
require (
|
||||
github.com/alecthomas/kong v1.10.0
|
||||
github.com/alecthomas/mango-kong v0.1.0
|
||||
github.com/andreykaipov/goobs v1.5.6
|
||||
)
|
||||
|
||||
@@ -13,5 +14,7 @@ require (
|
||||
github.com/hashicorp/logutils v1.0.0 // indirect
|
||||
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||
github.com/mmcloughlin/profile v0.1.1 // indirect
|
||||
github.com/muesli/mango v0.1.1-0.20220205060214-77e2058169ab // indirect
|
||||
github.com/muesli/roff v0.1.0 // indirect
|
||||
github.com/nu7hatch/gouuid v0.0.0-20131221200532-179d4d0c4d8d // indirect
|
||||
)
|
||||
|
||||
6
go.sum
6
go.sum
@@ -2,6 +2,8 @@ github.com/alecthomas/assert/v2 v2.11.0 h1:2Q9r3ki8+JYXvGsDyBXwH3LcJ+WK5D0gc5E8v
|
||||
github.com/alecthomas/assert/v2 v2.11.0/go.mod h1:Bze95FyfUr7x34QZrjL+XP+0qgp/zg8yS+TtBj1WA3k=
|
||||
github.com/alecthomas/kong v1.10.0 h1:8K4rGDpT7Iu+jEXCIJUeKqvpwZHbsFRoebLbnzlmrpw=
|
||||
github.com/alecthomas/kong v1.10.0/go.mod h1:p2vqieVMeTAnaC83txKtXe8FLke2X07aruPWXyMPQrU=
|
||||
github.com/alecthomas/mango-kong v0.1.0 h1:iFVfP1k1K4qpml3JUQmD5I8MCQYfIvsD9mRdrw7jJC4=
|
||||
github.com/alecthomas/mango-kong v0.1.0/go.mod h1:t+TYVdsONUolf/BwVcm+15eqcdAj15h4Qe9MMFAwwT4=
|
||||
github.com/alecthomas/repr v0.4.0 h1:GhI2A8MACjfegCPVq9f1FLvIBS+DrQ2KQBFZP1iFzXc=
|
||||
github.com/alecthomas/repr v0.4.0/go.mod h1:Fr0507jx4eOXV7AlPV6AVZLYrLIuIeSOWtW57eE/O/4=
|
||||
github.com/andreykaipov/goobs v1.5.6 h1:eIkEqYN99+2VJvmlY/56Ah60nkRKS6efMQvpM3oUgPQ=
|
||||
@@ -20,6 +22,10 @@ github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyua
|
||||
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
|
||||
github.com/mmcloughlin/profile v0.1.1 h1:jhDmAqPyebOsVDOCICJoINoLb/AnLBaUw58nFzxWS2w=
|
||||
github.com/mmcloughlin/profile v0.1.1/go.mod h1:IhHD7q1ooxgwTgjxQYkACGA77oFTDdFVejUS1/tS/qU=
|
||||
github.com/muesli/mango v0.1.1-0.20220205060214-77e2058169ab h1:m7QFONkzLK0fVXCjwX5tANcnj1yXxTnYQtnfJiY3tcA=
|
||||
github.com/muesli/mango v0.1.1-0.20220205060214-77e2058169ab/go.mod h1:5XFpbC8jY5UUv89YQciiXNlbi+iJgt29VDC5xbzrLL4=
|
||||
github.com/muesli/roff v0.1.0 h1:YD0lalCotmYuF5HhZliKWlIx7IEhiXeSfq7hNjFqGF8=
|
||||
github.com/muesli/roff v0.1.0/go.mod h1:pjAHQM9hdUUwm/krAfrLGgJkXJ+YuhtsfZ42kieB2Ig=
|
||||
github.com/nu7hatch/gouuid v0.0.0-20131221200532-179d4d0c4d8d h1:VhgPp6v9qf9Agr/56bj7Y/xa04UccTW04VP0Qed4vnQ=
|
||||
github.com/nu7hatch/gouuid v0.0.0-20131221200532-179d4d0c4d8d/go.mod h1:YUTz3bUH2ZwIWBy3CJBeOBEugqcmXREj14T+iG/4k4U=
|
||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||
|
||||
8
main.go
8
main.go
@@ -10,6 +10,7 @@ import (
|
||||
"time"
|
||||
|
||||
"github.com/alecthomas/kong"
|
||||
mangokong "github.com/alecthomas/mango-kong"
|
||||
"github.com/andreykaipov/goobs"
|
||||
)
|
||||
|
||||
@@ -26,6 +27,8 @@ type ObsConfig struct {
|
||||
type cli struct {
|
||||
ObsConfig `embed:"" help:"OBS WebSocket configuration."`
|
||||
|
||||
Man mangokong.ManFlag `help:"Print man page."`
|
||||
|
||||
Version VersionCmd `help:"Show version." cmd:"" aliases:"v"`
|
||||
Scene SceneCmd `help:"Manage scenes." cmd:"" aliases:"sc"`
|
||||
Sceneitem SceneItemCmd `help:"Manage scene items." cmd:"" aliases:"si"`
|
||||
@@ -46,7 +49,6 @@ type context struct {
|
||||
}
|
||||
|
||||
func main() {
|
||||
var client *goobs.Client
|
||||
cli := cli{}
|
||||
ctx := kong.Parse(
|
||||
&cli,
|
||||
@@ -55,9 +57,7 @@ func main() {
|
||||
)
|
||||
|
||||
client, err := connectObs(cli.ObsConfig)
|
||||
if err != nil {
|
||||
ctx.FatalIfErrorf(err)
|
||||
}
|
||||
ctx.FatalIfErrorf(err)
|
||||
|
||||
ctx.Bind(&context{
|
||||
Client: client,
|
||||
|
||||
@@ -12,7 +12,7 @@ type ProfileCmd struct {
|
||||
List ListProfileCmd `help:"List profiles." cmd:"" aliases:"ls"`
|
||||
Current CurrentProfileCmd `help:"Get current profile." cmd:"" aliases:"c"`
|
||||
Switch SwitchProfileCmd `help:"Switch profile." cmd:"" aliases:"sw"`
|
||||
Create CreateProfileCmd `help:"Create profile." cmd:"" aliases:"cr"`
|
||||
Create CreateProfileCmd `help:"Create profile." cmd:"" aliases:"new"`
|
||||
Remove RemoveProfileCmd `help:"Remove profile." cmd:"" aliases:"rm"`
|
||||
}
|
||||
|
||||
|
||||
@@ -11,7 +11,7 @@ type SceneCollectionCmd struct {
|
||||
List ListSceneCollectionCmd `help:"List scene collections." cmd:"" aliases:"ls"`
|
||||
Current CurrentSceneCollectionCmd `help:"Get current scene collection." cmd:"" aliases:"c"`
|
||||
Switch SwitchSceneCollectionCmd `help:"Switch scene collection." cmd:"" aliases:"sw"`
|
||||
Create CreateSceneCollectionCmd `help:"Create scene collection." cmd:"" aliases:"cr"`
|
||||
Create CreateSceneCollectionCmd `help:"Create scene collection." cmd:"" aliases:"new"`
|
||||
}
|
||||
|
||||
// ListSceneCollectionCmd provides a command to list all scene collections.
|
||||
|
||||
139
sceneitem.go
139
sceneitem.go
@@ -9,11 +9,12 @@ import (
|
||||
|
||||
// SceneItemCmd provides commands to manage scene items in OBS Studio.
|
||||
type SceneItemCmd struct {
|
||||
List SceneItemListCmd `cmd:"" help:"List all scene items." aliases:"ls"`
|
||||
Show SceneItemShowCmd `cmd:"" help:"Show scene item." aliases:"sh"`
|
||||
Hide SceneItemHideCmd `cmd:"" help:"Hide scene item." aliases:"h"`
|
||||
Toggle SceneItemToggleCmd `cmd:"" help:"Toggle scene item." aliases:"tg"`
|
||||
Visible SceneItemVisibleCmd `cmd:"" help:"Get scene item visibility." aliases:"v"`
|
||||
List SceneItemListCmd `cmd:"" help:"List all scene items." aliases:"ls"`
|
||||
Show SceneItemShowCmd `cmd:"" help:"Show scene item." aliases:"sh"`
|
||||
Hide SceneItemHideCmd `cmd:"" help:"Hide scene item." aliases:"h"`
|
||||
Toggle SceneItemToggleCmd `cmd:"" help:"Toggle scene item." aliases:"tg"`
|
||||
Visible SceneItemVisibleCmd `cmd:"" help:"Get scene item visibility." aliases:"v"`
|
||||
Transform SceneItemTransformCmd `cmd:"" help:"Transform scene item." aliases:"t"`
|
||||
}
|
||||
|
||||
// SceneItemListCmd provides a command to list all scene items in a scene.
|
||||
@@ -85,6 +86,13 @@ func (cmd *SceneItemShowCmd) Run(ctx *context) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if cmd.Parent != "" {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in group '%s' is now visible.\n", cmd.ItemName, cmd.Parent)
|
||||
} else {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in scene '%s' is now visible.\n", cmd.ItemName, cmd.SceneName)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -110,6 +118,13 @@ func (cmd *SceneItemHideCmd) Run(ctx *context) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if cmd.Parent != "" {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in group '%s' is now hidden.\n", cmd.ItemName, cmd.Parent)
|
||||
} else {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in scene '%s' is now hidden.\n", cmd.ItemName, cmd.SceneName)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -151,6 +166,13 @@ func (cmd *SceneItemToggleCmd) Run(ctx *context) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if itemEnabled {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in scene '%s' is now hidden.\n", cmd.ItemName, cmd.SceneName)
|
||||
} else {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in scene '%s' is now visible.\n", cmd.ItemName, cmd.SceneName)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -181,3 +203,110 @@ func (cmd *SceneItemVisibleCmd) Run(ctx *context) error {
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// SceneItemTransformCmd provides a command to transform a scene item.
|
||||
type SceneItemTransformCmd struct {
|
||||
SceneName string `arg:"" help:"Scene name."`
|
||||
ItemName string `arg:"" help:"Item name."`
|
||||
|
||||
Parent string `flag:"" help:"Parent group name."`
|
||||
|
||||
Alignment float64 `flag:"" help:"Alignment of the scene item."`
|
||||
BoundsAlignment float64 `flag:"" help:"Bounds alignment of the scene item."`
|
||||
BoundsHeight float64 `flag:"" help:"Bounds height of the scene item." default:"1.0"`
|
||||
BoundsType string `flag:"" help:"Bounds type of the scene item." default:"OBS_BOUNDS_NONE"`
|
||||
BoundsWidth float64 `flag:"" help:"Bounds width of the scene item." default:"1.0"`
|
||||
CropToBounds bool `flag:"" help:"Whether to crop the scene item to bounds."`
|
||||
CropBottom float64 `flag:"" help:"Crop bottom value of the scene item."`
|
||||
CropLeft float64 `flag:"" help:"Crop left value of the scene item."`
|
||||
CropRight float64 `flag:"" help:"Crop right value of the scene item."`
|
||||
CropTop float64 `flag:"" help:"Crop top value of the scene item."`
|
||||
PositionX float64 `flag:"" help:"X position of the scene item."`
|
||||
PositionY float64 `flag:"" help:"Y position of the scene item."`
|
||||
Rotation float64 `flag:"" help:"Rotation of the scene item."`
|
||||
ScaleX float64 `flag:"" help:"X scale of the scene item."`
|
||||
ScaleY float64 `flag:"" help:"Y scale of the scene item."`
|
||||
}
|
||||
|
||||
// Run executes the command to transform a scene item.
|
||||
func (cmd *SceneItemTransformCmd) Run(ctx *context) error {
|
||||
sceneName, sceneItemID, err := getSceneNameAndItemID(ctx.Client, cmd.SceneName, cmd.ItemName, cmd.Parent)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Get the current transform of the scene item
|
||||
resp, err := ctx.Client.SceneItems.GetSceneItemTransform(sceneitems.NewGetSceneItemTransformParams().
|
||||
WithSceneName(sceneName).
|
||||
WithSceneItemId(sceneItemID))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Update the transform with the provided values
|
||||
transform := resp.SceneItemTransform
|
||||
|
||||
if cmd.Alignment != 0 {
|
||||
transform.Alignment = cmd.Alignment
|
||||
}
|
||||
if cmd.BoundsAlignment != 0 {
|
||||
transform.BoundsAlignment = cmd.BoundsAlignment
|
||||
}
|
||||
|
||||
if cmd.BoundsHeight != 0 {
|
||||
transform.BoundsHeight = cmd.BoundsHeight
|
||||
}
|
||||
if cmd.BoundsType != "" {
|
||||
transform.BoundsType = cmd.BoundsType
|
||||
}
|
||||
if cmd.BoundsWidth != 0 {
|
||||
transform.BoundsWidth = cmd.BoundsWidth
|
||||
}
|
||||
|
||||
if cmd.CropToBounds {
|
||||
transform.CropToBounds = cmd.CropToBounds
|
||||
}
|
||||
if cmd.CropBottom != 0 {
|
||||
transform.CropBottom = cmd.CropBottom
|
||||
}
|
||||
if cmd.CropLeft != 0 {
|
||||
transform.CropLeft = cmd.CropLeft
|
||||
}
|
||||
if cmd.CropRight != 0 {
|
||||
transform.CropRight = cmd.CropRight
|
||||
}
|
||||
if cmd.CropTop != 0 {
|
||||
transform.CropTop = cmd.CropTop
|
||||
}
|
||||
if cmd.PositionX != 0 {
|
||||
transform.PositionX = cmd.PositionX
|
||||
}
|
||||
if cmd.PositionY != 0 {
|
||||
transform.PositionY = cmd.PositionY
|
||||
}
|
||||
if cmd.Rotation != 0 {
|
||||
transform.Rotation = cmd.Rotation
|
||||
}
|
||||
if cmd.ScaleX != 0 {
|
||||
transform.ScaleX = cmd.ScaleX
|
||||
}
|
||||
if cmd.ScaleY != 0 {
|
||||
transform.ScaleY = cmd.ScaleY
|
||||
}
|
||||
|
||||
_, err = ctx.Client.SceneItems.SetSceneItemTransform(sceneitems.NewSetSceneItemTransformParams().
|
||||
WithSceneName(sceneName).
|
||||
WithSceneItemId(sceneItemID).
|
||||
WithSceneItemTransform(transform))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if cmd.Parent != "" {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in group '%s' transformed.\n", cmd.ItemName, cmd.Parent)
|
||||
} else {
|
||||
fmt.Fprintf(ctx.Out, "Scene item '%s' in scene '%s' transformed.\n", cmd.ItemName, cmd.SceneName)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user