mirror of
https://github.com/onyx-and-iris/simple-recorder.git
synced 2026-04-21 00:33:36 +00:00
Compare commits
15 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 5aae021aed | |||
| a2e1c5280b | |||
| d5b7991584 | |||
| d7edaab2d1 | |||
| 09270683d9 | |||
| b7c25525a1 | |||
| c39b909d11 | |||
| 856ab6cbec | |||
| c7baa9ffd6 | |||
| d0401d0457 | |||
| 8813c25819 | |||
| 6035e09e43 | |||
| 0fe22debb6 | |||
| d8d7fce5cc | |||
| ca7604c279 |
67
README.md
67
README.md
@@ -28,7 +28,18 @@ pipx install simple-recorder
|
|||||||
|
|
||||||
*with pyz*
|
*with pyz*
|
||||||
|
|
||||||
An executable pyz has been included in [Releases](https://github.com/onyx-and-iris/simple-recorder/releases) which you can run in Windows. Follow the steps in this [Setting up Windows for Zipapps](https://jhermann.github.io/blog/python/deployment/2020/02/29/python_zippapps_on_windows.html#Setting-Up-Windows-10-for-Zipapps) guide.
|
- Download the pyz file in [Releases](https://github.com/onyx-and-iris/simple-recorder/releases)
|
||||||
|
- Optional step: for automatic discovery of the pyz file follow this guide on [Setting Up Windows for Zippapps](https://jhermann.github.io/blog/python/deployment/2020/02/29/python_zippapps_on_windows.html#Setting-Up-Windows-10-for-Zipapps)
|
||||||
|
|
||||||
|
Finally run the pyz with python (CLI)/pythonw (GUI):
|
||||||
|
|
||||||
|
```console
|
||||||
|
python simple-recorder.pyz <subcommand>
|
||||||
|
|
||||||
|
pythonw simple-recorder.pyz
|
||||||
|
```
|
||||||
|
|
||||||
|
note, the pyz extension won't be required if you followed the optional step and made it discoverable.
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
@@ -49,28 +60,18 @@ OBS_THEME=Reds
|
|||||||
|
|
||||||
## Use
|
## Use
|
||||||
|
|
||||||
### GUI
|
|
||||||
|
|
||||||
To launch the GUI run the root command without any subcommands:
|
|
||||||
|
|
||||||
```console
|
|
||||||
simple-recorder
|
|
||||||
```
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
Just enter the filename and click *Start*.
|
|
||||||
|
|
||||||
#### Themes
|
|
||||||
|
|
||||||
You can change the colour theme with the --theme option:
|
|
||||||
|
|
||||||
```console
|
|
||||||
simple-recorder --theme="Light Purple"
|
|
||||||
```
|
|
||||||
|
|
||||||
### CLI
|
### CLI
|
||||||
|
|
||||||
|
To launch the CLI:
|
||||||
|
|
||||||
|
```console
|
||||||
|
simple-recorder start "File Name"
|
||||||
|
|
||||||
|
simple-recorder stop
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Commands:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
Usage: simple-recorder [OPTIONS] COMMAND
|
Usage: simple-recorder [OPTIONS] COMMAND
|
||||||
|
|
||||||
@@ -79,6 +80,9 @@ Usage: simple-recorder [OPTIONS] COMMAND
|
|||||||
┃ stop Stop recording ┃
|
┃ stop Stop recording ┃
|
||||||
┃ pause Pause recording ┃
|
┃ pause Pause recording ┃
|
||||||
┃ resume Resume recording ┃
|
┃ resume Resume recording ┃
|
||||||
|
┃ split Split the current recording into a new file ┃
|
||||||
|
┃ chapter Create a chapter in the current recording ┃
|
||||||
|
┃ directory Get or set the recording directory ┃
|
||||||
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
|
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
|
||||||
|
|
||||||
┏━ Options ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
|
┏━ Options ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
|
||||||
@@ -90,15 +94,24 @@ Usage: simple-recorder [OPTIONS] COMMAND
|
|||||||
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
|
┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
|
||||||
```
|
```
|
||||||
|
|
||||||
To launch the CLI pass any subcommand (start/stop etc...), for example:
|
### GUI
|
||||||
|
|
||||||
|
To launch the GUI:
|
||||||
|
|
||||||
```console
|
```console
|
||||||
simple-recorder start "File Name"
|
simple-recorder-gui
|
||||||
|
|
||||||
simple-recorder stop
|
|
||||||
```
|
```
|
||||||
|
|
||||||
- If no filename is passed to start then you will be prompted for one.
|

|
||||||
- A default_name will be used if none is supplied to the prompt.
|
|
||||||
|
Just enter the filename and click *Start*.
|
||||||
|
|
||||||
|
#### Themes
|
||||||
|
|
||||||
|
You can change the colour theme with the --theme option:
|
||||||
|
|
||||||
|
```console
|
||||||
|
simple-recorder-gui --theme="Light Purple"
|
||||||
|
```
|
||||||
|
|
||||||
[obs-studio]: https://obsproject.com/
|
[obs-studio]: https://obsproject.com/
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 84 KiB After Width: | Height: | Size: 78 KiB |
8
pdm.lock
generated
8
pdm.lock
generated
@@ -5,7 +5,7 @@
|
|||||||
groups = ["default", "build"]
|
groups = ["default", "build"]
|
||||||
strategy = ["inherit_metadata"]
|
strategy = ["inherit_metadata"]
|
||||||
lock_version = "4.5.0"
|
lock_version = "4.5.0"
|
||||||
content_hash = "sha256:292c5ea319597e3539895c1ac50004c884c5d46edd5f7b195ede79156558feab"
|
content_hash = "sha256:e744505d4be91a30830dd1d57d956a6c8775ea75385351b3f4c0d29faddd9ce3"
|
||||||
|
|
||||||
[[metadata.targets]]
|
[[metadata.targets]]
|
||||||
requires_python = ">=3.11"
|
requires_python = ">=3.11"
|
||||||
@@ -63,7 +63,7 @@ files = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "obsws-python"
|
name = "obsws-python"
|
||||||
version = "1.7.2"
|
version = "1.8.0"
|
||||||
requires_python = ">=3.9"
|
requires_python = ">=3.9"
|
||||||
summary = "A Python SDK for OBS Studio WebSocket v5.0"
|
summary = "A Python SDK for OBS Studio WebSocket v5.0"
|
||||||
groups = ["default"]
|
groups = ["default"]
|
||||||
@@ -72,8 +72,8 @@ dependencies = [
|
|||||||
"websocket-client",
|
"websocket-client",
|
||||||
]
|
]
|
||||||
files = [
|
files = [
|
||||||
{file = "obsws_python-1.7.2-py3-none-any.whl", hash = "sha256:acda31852ad9d7165de915b0603c13f6df527d3f61619970bf5fb562e300bc85"},
|
{file = "obsws_python-1.8.0-py3-none-any.whl", hash = "sha256:537bde416e149b6f59e0b2f31761d4a40329feafec171bde6fc1346ab8516e28"},
|
||||||
{file = "obsws_python-1.7.2.tar.gz", hash = "sha256:b5cdaad30fbe1f6d4787b6530048b9882f070c3ee7830abb6dad4a47f84d7fa0"},
|
{file = "obsws_python-1.8.0.tar.gz", hash = "sha256:e082894f80deb0836861fdc3c222e497308c8f66328da6075baba5b456a20971"},
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
[project]
|
[project]
|
||||||
name = "simple-recorder"
|
name = "simple-recorder"
|
||||||
version = "0.3.0"
|
version = "0.5.0"
|
||||||
description = "A simple OBS recorder"
|
description = "A simple OBS recorder"
|
||||||
authors = [{ name = "onyx-and-iris", email = "code@onyxandiris.online" }]
|
authors = [{ name = "onyx-and-iris", email = "code@onyxandiris.online" }]
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"clypi>=1.8.1",
|
"clypi>=1.8.1",
|
||||||
"FreeSimpleGUI>=5.2.0.post1",
|
"FreeSimpleGUI>=5.2.0.post1",
|
||||||
"obsws-python>=1.7.2",
|
"obsws-python>=1.8.0",
|
||||||
]
|
]
|
||||||
requires-python = ">=3.11"
|
requires-python = ">=3.11"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
@@ -15,6 +15,9 @@ license = { text = "MIT" }
|
|||||||
[project.scripts]
|
[project.scripts]
|
||||||
simple-recorder = "simple_recorder:run"
|
simple-recorder = "simple_recorder:run"
|
||||||
|
|
||||||
|
[project.gui-scripts]
|
||||||
|
simple-recorder-gui = "simple_recorder:run"
|
||||||
|
|
||||||
[build-system]
|
[build-system]
|
||||||
requires = ["pdm-backend"]
|
requires = ["pdm-backend"]
|
||||||
build-backend = "pdm.backend"
|
build-backend = "pdm.backend"
|
||||||
@@ -28,6 +31,4 @@ compile = "shiv -c simple-recorder -o bin/simple-recorder.pyz ."
|
|||||||
|
|
||||||
|
|
||||||
[dependency-groups]
|
[dependency-groups]
|
||||||
build = [
|
build = ["shiv>=1.0.8"]
|
||||||
"shiv>=1.0.8",
|
|
||||||
]
|
|
||||||
|
|||||||
48
src/simple_recorder/chapter.py
Normal file
48
src/simple_recorder/chapter.py
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
import obsws_python as obsws
|
||||||
|
from clypi import Command, Positional, arg
|
||||||
|
from typing_extensions import override
|
||||||
|
|
||||||
|
from .errors import SimpleRecorderError
|
||||||
|
from .styler import highlight
|
||||||
|
|
||||||
|
|
||||||
|
class Chapter(Command):
|
||||||
|
"""Create a chapter in the current recording."""
|
||||||
|
|
||||||
|
chapter_name: Positional[str] = arg(
|
||||||
|
help="Name of the chapter to create.",
|
||||||
|
prompt="Enter the name for the chapter.",
|
||||||
|
default="unnamed",
|
||||||
|
)
|
||||||
|
host: str = arg(inherited=True)
|
||||||
|
port: int = arg(inherited=True)
|
||||||
|
password: str = arg(inherited=True)
|
||||||
|
|
||||||
|
@override
|
||||||
|
async def run(self):
|
||||||
|
"""Run the chapter command."""
|
||||||
|
try:
|
||||||
|
with obsws.ReqClient(
|
||||||
|
host=self.host, port=self.port, password=self.password, timeout=3
|
||||||
|
) as client:
|
||||||
|
resp = client.get_record_status()
|
||||||
|
if not resp.output_active:
|
||||||
|
raise SimpleRecorderError(
|
||||||
|
"No active recording to create a chapter."
|
||||||
|
)
|
||||||
|
|
||||||
|
# Allow OBS to set unnamed chapters (it will increment the name)
|
||||||
|
if self.chapter_name == "unnamed":
|
||||||
|
client.create_record_chapter()
|
||||||
|
else:
|
||||||
|
client.create_record_chapter(self.chapter_name)
|
||||||
|
print(f"Chapter {highlight(self.chapter_name)} created successfully.")
|
||||||
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
except obsws.error.OBSSDKRequestError as e:
|
||||||
|
if e.code == 702:
|
||||||
|
raise SimpleRecorderError(
|
||||||
|
"Unable to create chapter, please check your OBS settings."
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
raise SimpleRecorderError(f"Error: {e}")
|
||||||
@@ -3,16 +3,16 @@ import logging
|
|||||||
from clypi import ClypiConfig, ClypiException, Command, arg, configure
|
from clypi import ClypiConfig, ClypiException, Command, arg, configure
|
||||||
from typing_extensions import override
|
from typing_extensions import override
|
||||||
|
|
||||||
|
from .chapter import Chapter
|
||||||
from .directory import Directory
|
from .directory import Directory
|
||||||
from .errors import SimpleRecorderError
|
from .errors import SimpleRecorderError
|
||||||
from .gui import SimpleRecorderWindow
|
from .gui import SimpleRecorderWindow
|
||||||
from .pause import Pause
|
from .pause import Pause
|
||||||
from .resume import Resume
|
from .resume import Resume
|
||||||
|
from .split import Split
|
||||||
from .start import Start
|
from .start import Start
|
||||||
from .stop import Stop
|
from .stop import Stop
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
config = ClypiConfig(
|
config = ClypiConfig(
|
||||||
nice_errors=(SimpleRecorderError,),
|
nice_errors=(SimpleRecorderError,),
|
||||||
)
|
)
|
||||||
@@ -37,8 +37,11 @@ def theme_parser(value: str) -> str:
|
|||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
|
SUBCOMMANDS = Start | Stop | Pause | Resume | Split | Chapter | Directory
|
||||||
|
|
||||||
|
|
||||||
class SimpleRecorder(Command):
|
class SimpleRecorder(Command):
|
||||||
subcommand: Start | Stop | Pause | Resume | Directory | None = None
|
subcommand: SUBCOMMANDS | None = None
|
||||||
host: str = arg(default="localhost", env="OBS_HOST", help="OBS WebSocket host")
|
host: str = arg(default="localhost", env="OBS_HOST", help="OBS WebSocket host")
|
||||||
port: int = arg(default=4455, env="OBS_PORT", help="OBS WebSocket port")
|
port: int = arg(default=4455, env="OBS_PORT", help="OBS WebSocket port")
|
||||||
password: str | None = arg(
|
password: str | None = arg(
|
||||||
@@ -52,17 +55,20 @@ class SimpleRecorder(Command):
|
|||||||
)
|
)
|
||||||
debug: bool = arg(
|
debug: bool = arg(
|
||||||
default=False,
|
default=False,
|
||||||
env="DEBUG",
|
|
||||||
help="Enable debug logging",
|
help="Enable debug logging",
|
||||||
hidden=True,
|
hidden=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
@override
|
@override
|
||||||
async def run(self):
|
async def pre_run_hook(self):
|
||||||
"""Run the Simple Recorder GUI."""
|
|
||||||
if self.debug:
|
if self.debug:
|
||||||
logging.basicConfig(level=logging.DEBUG)
|
logging.basicConfig(level=logging.DEBUG)
|
||||||
|
else:
|
||||||
|
logging.basicConfig(level=logging.disable())
|
||||||
|
|
||||||
|
@override
|
||||||
|
async def run(self):
|
||||||
|
"""Run the Simple Recorder GUI."""
|
||||||
window = SimpleRecorderWindow(self.host, self.port, self.password, self.theme)
|
window = SimpleRecorderWindow(self.host, self.port, self.password, self.theme)
|
||||||
await window.run()
|
await window.run()
|
||||||
|
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ class Directory(Command):
|
|||||||
async def run(self):
|
async def run(self):
|
||||||
try:
|
try:
|
||||||
with obsws.ReqClient(
|
with obsws.ReqClient(
|
||||||
host=self.host, port=self.port, password=self.password
|
host=self.host, port=self.port, password=self.password, timeout=3
|
||||||
) as client:
|
) as client:
|
||||||
if self.directory:
|
if self.directory:
|
||||||
client.set_record_directory(self.directory)
|
client.set_record_directory(self.directory)
|
||||||
@@ -32,5 +32,5 @@ class Directory(Command):
|
|||||||
f"Current recording directory: {highlight(resp.record_directory)}"
|
f"Current recording directory: {highlight(resp.record_directory)}"
|
||||||
)
|
)
|
||||||
return resp.record_directory
|
return resp.record_directory
|
||||||
except Exception as e:
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
raise SimpleRecorderError(f"Failed to get recording directory: {e}")
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
|||||||
@@ -3,10 +3,12 @@ import logging
|
|||||||
import FreeSimpleGUI as fsg
|
import FreeSimpleGUI as fsg
|
||||||
import obsws_python as obsws
|
import obsws_python as obsws
|
||||||
|
|
||||||
|
from .chapter import Chapter
|
||||||
from .directory import Directory
|
from .directory import Directory
|
||||||
from .errors import SimpleRecorderError
|
from .errors import SimpleRecorderError
|
||||||
from .pause import Pause
|
from .pause import Pause
|
||||||
from .resume import Resume
|
from .resume import Resume
|
||||||
|
from .split import Split
|
||||||
from .start import Start
|
from .start import Start
|
||||||
from .stop import Stop
|
from .stop import Stop
|
||||||
|
|
||||||
@@ -31,6 +33,7 @@ class SimpleRecorderWindow(fsg.Window):
|
|||||||
current_directory = resp.record_directory
|
current_directory = resp.record_directory
|
||||||
except (ConnectionRefusedError, TimeoutError):
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
status_message = "Failed to connect to OBS. Is it running?"
|
status_message = "Failed to connect to OBS. Is it running?"
|
||||||
|
current_directory = ""
|
||||||
|
|
||||||
recorder_layout = [
|
recorder_layout = [
|
||||||
[fsg.Text("Enter recording filename:", key="-PROMPT-")],
|
[fsg.Text("Enter recording filename:", key="-PROMPT-")],
|
||||||
@@ -93,8 +96,10 @@ class SimpleRecorderWindow(fsg.Window):
|
|||||||
self["Stop Recording"].bind("<Return>", " || RETURN")
|
self["Stop Recording"].bind("<Return>", " || RETURN")
|
||||||
self["Pause Recording"].bind("<Return>", " || RETURN")
|
self["Pause Recording"].bind("<Return>", " || RETURN")
|
||||||
self["Resume Recording"].bind("<Return>", " || RETURN")
|
self["Resume Recording"].bind("<Return>", " || RETURN")
|
||||||
|
self["Split Recording"].bind("<Return>", " || RETURN")
|
||||||
|
self["Add Chapter"].bind("<Return>", " || RETURN")
|
||||||
|
self["Add Chapter"].bind("<Shift-Return>", " || SHIFT-RETURN")
|
||||||
|
|
||||||
self["-FILENAME-"].bind("<KeyPress>", " || KEYPRESS")
|
|
||||||
self["-FILENAME-"].update(select=True)
|
self["-FILENAME-"].update(select=True)
|
||||||
self["Add Chapter"].bind("<FocusIn>", " || FOCUS")
|
self["Add Chapter"].bind("<FocusIn>", " || FOCUS")
|
||||||
self["Add Chapter"].bind("<Enter>", " || FOCUS")
|
self["Add Chapter"].bind("<Enter>", " || FOCUS")
|
||||||
@@ -102,7 +107,12 @@ class SimpleRecorderWindow(fsg.Window):
|
|||||||
self["Add Chapter"].bind("<Leave>", " || LEAVE")
|
self["Add Chapter"].bind("<Leave>", " || LEAVE")
|
||||||
self["Add Chapter"].bind("<Button-3>", " || RIGHT_CLICK")
|
self["Add Chapter"].bind("<Button-3>", " || RIGHT_CLICK")
|
||||||
|
|
||||||
|
self["-GET-CURRENT-"].bind("<Return>", " || RETURN")
|
||||||
|
self["-UPDATE-"].bind("<Return>", " || RETURN")
|
||||||
|
|
||||||
async def run(self):
|
async def run(self):
|
||||||
|
chapter_name = "unnamed"
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
event, values = self.read()
|
event, values = self.read()
|
||||||
self.logger.debug(f"Event: {event}, Values: {values}")
|
self.logger.debug(f"Event: {event}, Values: {values}")
|
||||||
@@ -173,19 +183,47 @@ class SimpleRecorderWindow(fsg.Window):
|
|||||||
else:
|
else:
|
||||||
self["-OUTPUT-RECORDER-"].update("", text_color="white")
|
self["-OUTPUT-RECORDER-"].update("", text_color="white")
|
||||||
|
|
||||||
case ["Add Chapter", "RIGHT_CLICK"]:
|
case ["Split Recording"] | ["Split Recording", "RETURN"]:
|
||||||
_ = fsg.popup_get_text(
|
try:
|
||||||
|
await Split(
|
||||||
|
host=self.host, port=self.port, password=self.password
|
||||||
|
).run()
|
||||||
|
self["-OUTPUT-RECORDER-"].update(
|
||||||
|
"Recording split successfully", text_color="green"
|
||||||
|
)
|
||||||
|
except SimpleRecorderError as e:
|
||||||
|
self["-OUTPUT-RECORDER-"].update(
|
||||||
|
f"Error: {e.raw_message}", text_color="red"
|
||||||
|
)
|
||||||
|
|
||||||
|
case ["Add Chapter", "RIGHT_CLICK" | "SHIFT-RETURN"]:
|
||||||
|
chapter_name = fsg.popup_get_text(
|
||||||
"Enter chapter name:",
|
"Enter chapter name:",
|
||||||
"Add Chapter",
|
"Add Chapter",
|
||||||
default_text="unnamed",
|
default_text="unnamed",
|
||||||
)
|
)
|
||||||
|
|
||||||
case ["Split Recording" | "Add Chapter"]:
|
case ["Add Chapter"] | ["Add Chapter", "RETURN"]:
|
||||||
|
try:
|
||||||
|
await Chapter(
|
||||||
|
chapter_name=chapter_name,
|
||||||
|
host=self.host,
|
||||||
|
port=self.port,
|
||||||
|
password=self.password,
|
||||||
|
).run()
|
||||||
self["-OUTPUT-RECORDER-"].update(
|
self["-OUTPUT-RECORDER-"].update(
|
||||||
"This feature is not implemented yet", text_color="orange"
|
f"Chapter {chapter_name if chapter_name else 'unnamed'} added successfully",
|
||||||
|
text_color="green",
|
||||||
|
)
|
||||||
|
except SimpleRecorderError:
|
||||||
|
fsg.popup_error(
|
||||||
|
"Unable to create chapter, please check your OBS settings.\n"
|
||||||
|
"Note, currently only Hybrid MP4 is supported for chapters.",
|
||||||
|
title="Chapter Error",
|
||||||
|
keep_on_top=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
case ["-GET-CURRENT-"]:
|
case ["-GET-CURRENT-"] | ["-GET-CURRENT-", "RETURN"]:
|
||||||
try:
|
try:
|
||||||
current_directory = await Directory(
|
current_directory = await Directory(
|
||||||
host=self.host, port=self.port, password=self.password
|
host=self.host, port=self.port, password=self.password
|
||||||
@@ -196,7 +234,7 @@ class SimpleRecorderWindow(fsg.Window):
|
|||||||
f"Error: {e.raw_message}", text_color="red"
|
f"Error: {e.raw_message}", text_color="red"
|
||||||
)
|
)
|
||||||
|
|
||||||
case ["-UPDATE-"]:
|
case ["-UPDATE-"] | ["-UPDATE-", "RETURN"]:
|
||||||
filepath = values["-FILEPATH-"]
|
filepath = values["-FILEPATH-"]
|
||||||
if not filepath:
|
if not filepath:
|
||||||
self["-OUTPUT-SETTINGS-"].update(
|
self["-OUTPUT-SETTINGS-"].update(
|
||||||
|
|||||||
@@ -26,5 +26,5 @@ class Pause(Command):
|
|||||||
|
|
||||||
client.pause_record()
|
client.pause_record()
|
||||||
print("Recording paused successfully.")
|
print("Recording paused successfully.")
|
||||||
except TimeoutError:
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
|||||||
@@ -26,5 +26,5 @@ class Resume(Command):
|
|||||||
|
|
||||||
client.resume_record()
|
client.resume_record()
|
||||||
print("Recording resumed successfully.")
|
print("Recording resumed successfully.")
|
||||||
except TimeoutError:
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
|||||||
36
src/simple_recorder/split.py
Normal file
36
src/simple_recorder/split.py
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import obsws_python as obsws
|
||||||
|
from clypi import Command, arg
|
||||||
|
from typing_extensions import override
|
||||||
|
|
||||||
|
from .errors import SimpleRecorderError
|
||||||
|
|
||||||
|
|
||||||
|
class Split(Command):
|
||||||
|
"""Split the current recording into a new file."""
|
||||||
|
|
||||||
|
host: str = arg(inherited=True)
|
||||||
|
port: int = arg(inherited=True)
|
||||||
|
password: str = arg(inherited=True)
|
||||||
|
|
||||||
|
@override
|
||||||
|
async def run(self):
|
||||||
|
"""Run the split command."""
|
||||||
|
try:
|
||||||
|
with obsws.ReqClient(
|
||||||
|
host=self.host, port=self.port, password=self.password, timeout=3
|
||||||
|
) as client:
|
||||||
|
resp = client.get_record_status()
|
||||||
|
if not resp.output_active:
|
||||||
|
raise SimpleRecorderError("No active recording to split.")
|
||||||
|
|
||||||
|
client.split_record_file()
|
||||||
|
print("Recording split successfully.")
|
||||||
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
except obsws.error.OBSSDKRequestError as e:
|
||||||
|
if e.code == 702:
|
||||||
|
raise SimpleRecorderError(
|
||||||
|
"Unable to split file, please check your OBS settings."
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
raise SimpleRecorderError(f"Error: {e.code} - {e.message}")
|
||||||
@@ -45,5 +45,5 @@ class Start(Command):
|
|||||||
)
|
)
|
||||||
client.start_record()
|
client.start_record()
|
||||||
print(f"Recording started with filename: {highlight(filename)}")
|
print(f"Recording started with filename: {highlight(filename)}")
|
||||||
except TimeoutError:
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ from clypi import Command, arg
|
|||||||
from typing_extensions import override
|
from typing_extensions import override
|
||||||
|
|
||||||
from .errors import SimpleRecorderError
|
from .errors import SimpleRecorderError
|
||||||
from .styler import highlight
|
|
||||||
|
|
||||||
|
|
||||||
class Stop(Command):
|
class Stop(Command):
|
||||||
@@ -24,6 +23,6 @@ class Stop(Command):
|
|||||||
raise SimpleRecorderError("Recording is not active.")
|
raise SimpleRecorderError("Recording is not active.")
|
||||||
|
|
||||||
client.stop_record()
|
client.stop_record()
|
||||||
print(highlight("Recording stopped successfully."))
|
print("Recording stopped successfully.")
|
||||||
except TimeoutError:
|
except (ConnectionRefusedError, TimeoutError):
|
||||||
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
raise SimpleRecorderError("Failed to connect to OBS. Is it running?")
|
||||||
|
|||||||
Reference in New Issue
Block a user