Skip to content

Commit 736da58

Browse files
authored
🔀 Merge pull request #20 from davep/save-pep-source
Add support for saving a PEP's source to a file
2 parents 665f937 + a13e1e1 commit 736da58

File tree

6 files changed

+140
-1
lines changed

6 files changed

+140
-1
lines changed

ChangeLog.md

+2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@
1111
([#18](https://github.com/davep/peplum/pull/18))
1212
- Dropped Python 3.8 as a supported Python version.
1313
([#19](https://github.com/davep/peplum/pull/19))
14+
- Added support for saving the source of a PEP to a file.
15+
(#20[](https://github.com/davep/peplum/pull/20))
1416

1517
## v0.2.0
1618

pyproject.toml

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ dependencies = [
1212
"typing-extensions>=4.12.2",
1313
"packaging>=24.2",
1414
"humanize>=4.11.0",
15+
"textual-fspicker>=0.1.1",
1516
]
1617
readme = "README.md"
1718
requires-python = ">= 3.9"

requirements-dev.lock

+3
Original file line numberDiff line numberDiff line change
@@ -104,8 +104,11 @@ sniffio==1.3.1
104104
textual==1.0.0
105105
# via peplum
106106
# via textual-dev
107+
# via textual-fspicker
107108
# via textual-serve
108109
textual-dev==1.7.0
110+
textual-fspicker==0.1.1
111+
# via peplum
109112
textual-serve==1.1.1
110113
# via textual-dev
111114
typing-extensions==4.12.2

requirements.lock

+3
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,9 @@ sniffio==1.3.1
4848
# via anyio
4949
textual==1.0.0
5050
# via peplum
51+
# via textual-fspicker
52+
textual-fspicker==0.1.1
53+
# via peplum
5154
typing-extensions==4.12.2
5255
# via peplum
5356
# via textual

src/peplum/app/screens/confirm.py

+101
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,101 @@
1+
"""Provides a confirmation dialog."""
2+
3+
##############################################################################
4+
# Textual imports.
5+
from textual import on
6+
from textual.app import ComposeResult
7+
from textual.containers import Horizontal, Vertical
8+
from textual.screen import ModalScreen
9+
from textual.widgets import Button, Label
10+
11+
12+
##############################################################################
13+
class Confirm(ModalScreen[bool]):
14+
"""A modal dialog for confirming things."""
15+
16+
CSS = """
17+
Confirm {
18+
align: center middle;
19+
20+
&> Vertical {
21+
padding: 1 2;
22+
height: auto;
23+
width: auto;
24+
max-width: 80vw;
25+
background: $surface;
26+
border: panel $error;
27+
border-title-color: $text;
28+
29+
&> Horizontal {
30+
height: auto;
31+
width: 100%;
32+
align-horizontal: center;
33+
}
34+
}
35+
36+
Label {
37+
width: auto;
38+
max-width: 70vw;
39+
padding-left: 1;
40+
padding-right: 1;
41+
margin-bottom: 1;
42+
}
43+
44+
Button {
45+
margin-right: 1;
46+
}
47+
}
48+
"""
49+
50+
BINDINGS = [
51+
("escape", "no"),
52+
("f2", "yes"),
53+
("left", "focus_previous"),
54+
("right", "focus_next"),
55+
]
56+
57+
def __init__(
58+
self, title: str, question: str, yes_text: str = "Yes", no_text: str = "No"
59+
) -> None:
60+
"""Initialise the dialog.
61+
62+
Args:
63+
title: The title for the dialog.
64+
question: The question to ask the user.
65+
yes_text: The text for the yes button.
66+
no_text: The text for the no button.
67+
"""
68+
super().__init__()
69+
self._title = title
70+
"""The title for the dialog."""
71+
self._question = question
72+
"""The question to ask the user."""
73+
self._yes = yes_text
74+
"""The text of the yes button."""
75+
self._no = no_text
76+
"""The text of the no button."""
77+
78+
def compose(self) -> ComposeResult:
79+
"""Compose the layout of the dialog."""
80+
key_colour = (
81+
"dim" if self.app.current_theme is None else self.app.current_theme.accent
82+
)
83+
with Vertical() as dialog:
84+
dialog.border_title = self._title
85+
yield Label(self._question)
86+
with Horizontal():
87+
yield Button(f"{self._no} [{key_colour}]\\[Esc][/]", id="no")
88+
yield Button(f"{self._yes} [{key_colour}]\\[F2][/]", id="yes")
89+
90+
@on(Button.Pressed, "#yes")
91+
def action_yes(self) -> None:
92+
"""Send back the positive response."""
93+
self.dismiss(True)
94+
95+
@on(Button.Pressed, "#no")
96+
def action_no(self) -> None:
97+
"""Send back the negative response."""
98+
self.dismiss(False)
99+
100+
101+
### confirm.py ends here

src/peplum/app/screens/pep_viewer.py

+30-1
Original file line numberDiff line numberDiff line change
@@ -12,11 +12,16 @@
1212
from textual.screen import ModalScreen
1313
from textual.widgets import Button, TextArea
1414

15+
##############################################################################
16+
# Textual fspicker imports.
17+
from textual_fspicker import FileSave
18+
1519
##############################################################################
1620
# Local imports.
1721
from ...peps import API
1822
from ..data import PEP, cache_dir
1923
from ..widgets import TextViewer
24+
from .confirm import Confirm
2025

2126

2227
##############################################################################
@@ -58,7 +63,12 @@ class PEPViewer(ModalScreen[None]):
5863
}
5964
"""
6065

61-
BINDINGS = [("escape", "close"), ("ctrl+r", "refresh"), ("ctrl+c", "copy")]
66+
BINDINGS = [
67+
("ctrl+c", "copy"),
68+
("ctrl+r", "refresh"),
69+
("ctrl+s", "save"),
70+
("escape", "close"),
71+
]
6272

6373
def __init__(self, pep: PEP) -> None:
6474
"""Initialise the dialog.
@@ -80,6 +90,7 @@ def compose(self) -> ComposeResult:
8090
yield TextViewer()
8191
with Horizontal(id="buttons"):
8292
yield Button(f"Copy [{key_colour}]\\[^c][/]", id="copy")
93+
yield Button(f"Save [{key_colour}]\\[^s][/]", id="save")
8394
yield Button(f"Refresh [{key_colour}]\\[^r][/]", id="refresh")
8495
yield Button(f"Close [{key_colour}]\\[Esc][/]", id="close")
8596

@@ -147,5 +158,23 @@ async def action_copy(self) -> None:
147158
"""Copy PEP text to the clipboard."""
148159
await self.query_one(TextArea).run_action("copy")
149160

161+
@on(Button.Pressed, "#save")
162+
@work
163+
async def action_save(self) -> None:
164+
"""Save the source of the PEP to a file."""
165+
if target := await self.app.push_screen_wait(FileSave()):
166+
if target.exists() and not await self.app.push_screen_wait(
167+
Confirm(
168+
"Overwrite?", f"{target}\n\nAre you sure you want to overwrite?"
169+
)
170+
):
171+
return
172+
try:
173+
target.write_text(self.query_one(TextArea).text, encoding="utf-8")
174+
except IOError as error:
175+
self.notify(str(error), title="Save Failed", severity="error")
176+
return
177+
self.notify(str(target), title="Saved")
178+
150179

151180
### pep_viewer.py ends here

0 commit comments

Comments
 (0)