Fix PyPI installation issues

This commit is contained in:
A S Lewis 2020-03-28 07:35:24 +00:00
parent 35e70c91a6
commit 24efd5450b
216 changed files with 3191 additions and 58509 deletions

View File

@ -5,10 +5,5 @@ Authors ordered by first contribution:
(none yet)
Image credits:
Vectorgraphit <https://www.iconfinder.com/vectorgraphit>
FatCow Web Hosting https://www.fatcow.com/>
Mr. Hopnguyen <https://www.iconfinder.com/Mr.hopnguyen>
Other credits:
Tartube is partially based on youtube-dl-gui
Carlo Rodriguez <https://www.iconfinder.com/icons/512534/exercise_fitness_gym_gymnasium_icon>

964
CHANGES
View File

@ -1,965 +1,9 @@
v2.0.0 (29 Feb 2020)
v1.002 (28 Mar 2020)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- Tartube can now be installed from PyPI, or by using the new DEB/RPM packages
(Linux/BSD only; installation from PyPI does not work on MS Windows)
- DEB/RPM packages marked 'STRICT' are also available for uploads to
repositories with lots of rules, such as the official Debian repository.
In 'STRICT' packages, updating youtube-dl from within Tartube is disabled.
The 'STRICT' packages are compiled using new environment variables,
TARTUBE_PKG and TARTUBE_PKG_STRICT (replacing the old TARTUBE_DEBIAN)
environment variable. See the comments in setup.py for more details
- During a download operation, in the Progress Tab, you can now right-click a
video and select 'Stop after these videos'. This allows all of the current
video downloads to finish, before halting the download operation
- The download options window (in the Formats tab) did not allow users to
select an audio format before selecting a video format. The reason for this
restriction was that youtube-dl did not download the right formats, if an
audio format was selected first. Unfortuantely, it prevented users from
downloading a separate audio file, when this was available (e.g. an .m4a
file from YouTube). The restriction has now been removed; instead, Tartube
will automatically reorder the specified video/audio formats, so that video
formats are passed to youtube-dl first
- Fix for PyPI installation problems
MAJOR FIXES
- If an upload operation is automatically performed before a download
operation, and if the user tried to download a single video/channel/
playlist/folder, everything was downloaded instead of the single video/
channel/playlist/folder. Fixed
- Fixed an error in the 'Show system command' dialogue window, that prevented
it from opening at all
- Fixed parsing of download options inside double quotes "..."
MINOR NEW FEATURES
- Added a 'Cancel' button to some dialogue windows that didn't already have one
- Added a copy of the XDG module to the Tartube code, so it is no longer
necessary to install it before running/installing Tartube (Linux/BSD only)
MINOR FIXES
- Fixed a system error during a forced youtube-dl update (MS Windows only)
- Fixed wrong location for config file backups (MS Windows only)
- Fixed wrong location for Tartube temporary/test folders (all systems)
- Fixed missing (or duplicate) dialogue windows after failing to load the
config file and/or database file, in some rare situations
- The config file could not be created if its parent directory did not exist;
fixed
- Fixed loading of the wrong database file, in some rare situations
- Removed the old 'hello world' code intended for testing on MS Windows; it's
no longer required
- If Tartube can't find its icon files, a simple error message is now generated
rather than a long traceback
v1.5.0 (22 Feb 2020)
v1.0 (27 Mar 2020)
-------------------------------------------------------------------------------
This is the first release candidate for v2.0.0.
MAJOR NEW FEATURES
- You can now run multiple instances of Tartube on your system at the same
time. Multiple instances cannot load the same Tartube database; they must
each load their own database. Tartube will now remember the databases it
has loaded. If there are three databases (perhaps one on your main hard
disk and two on an external drive), you can start Tartube three times, and
they will each load a different database. This behaviour can be configured,
if necessary. Click 'Edit > System preferences... > Filesystem > Database'
- HookTube acts as a redirection service for YouTube. Because of lawyers and
their evil machinations, HookTube's functionality is not as extensive as it
once was. Added the Invidious website (https://invidio.us/) as an
alternative
- Added custom downloads. To start a custom download, click 'Operations >
Custom download all', or right-click a video/channel/playlist/folder. A
custom download is just like a normal download, until you customise it. To
do that, click 'Edit > System preferences > Operations > Custom'. Custom
downloads can be used to divert YouTube requests to HookTube or Invidious,
and to insert a delay between video downloads when the website is
complaining about robots
- Added a new toolbar at the bottom of the main window, below the list of
videos. The toolbar is hidden, by default. To reveal it, click the 'Show
filter options' button in the bottom right-hand corner. Buttons in the new
toolbar can be used to sort the videos alphabetically, rather than by date,
and to search for videos whose name matches a string (or regex). The button
to search for videos by date has been moved into this toolbar
- If you find a video that can't be downloaded, and you're not sure why, you
can now perform a test download. First, click 'Operations > Test
youtube-dl...' (or right-click a video in the main window's list). Copy the
video's URL into the dialogue window, and then click the OK button. Click
the Output Tab to see the results. If the test successfully downloads the
video, then the problem was with Tartube. If the test fails to download the
video, then the problem is with the underlying youtube-dl software (or with
the video website)
- During a test, it's possible to omit the video URL, while specifying some
youtube-dl download options. For example, you could fetch the youtube-dl
version number with the option --version
- Added a new operation for tidying up files in Tartube's data directory
(folder). To start the operation, click 'Operations > Tidy up files...'.
You could also right-click a channel and select 'Channel actions > Tidy up
channel', and so on. A dialogue window appears, in which you can specify
which files should be tidied up. Choose carefully, because any files
deleted as a result of this operation cannot be recovered
- The main window's switch button (in the toolbar near the top of the window)
now has six settings, rather than four. Click the button repeatedly to
cycle through them
- Interesting and important videos can now be bookmarked (e.g., by right-
clicking a video and selecting 'Mark video > Video is bookmarked').
Bookmarked videos are visible in the new 'Bookmarks' folder. Bookmarking is
an alternative to favourites; bookmarks usually apply to a single video,
whereas favourites usually apply to a whole channel, playlist or folder
- Also added a new 'Waiting Videos' folder. This acts as your own private
playlist - a list of videos that are waiting to be watched. To make a video
visible in this folder, right-click it and select 'Mark video > Video is in
waiting list'. When you watch the video, it will automatically disappear
from the 'Waiting Videos' folder (this doesn't happen to bookmarked videos)
- The previous version was unable to delete a channel, playlist or folder (see
below). That error caused a partially-deleted channel/playlist/folder to
appear in the Videos Tab, on the left-hand side. In case similar errors
occur in the future, a feature has been added to look for errors and
inconsistencies in the Tartube database and automatically fix them. Click
'Edit > System preferences... > Filesystem > DB Errors > Check' to use it
- Tartube can now fetch a list of available video formats for a video. Right-
click the video and select 'Fetch > Available formats'. Click the Output
Tab to see the results
- Tartube can also fetch a list of available subtitles for a video. Right-click
the video and select 'Fecth > Available subtitles'. Click the Output Tab to
see the results
- Tartube can now remember the size of its main window, and use the same size
when it restarts. This feature is disabled by default. To enable it, click
'Edit > System preferences... > Windows > Main window > Remember the size
of the main window when shutting down'
MAJOR FIXES
- In the previous version, Tartube was unable to delete a channel, playlist or
folder. Fixed
- Some procedures took an extremely long time. For example, after right-
clicking a channel and selecting 'Channel contents > Mark videos as new',
the procedure could take several minutes if the channel had hundreds of
videos, or several hours if it contained thousands of videos. The faulty
code has been fixed, and the procedure now takes just a few seconds, even
for many thousands of videos
- Videos in a Tartube folder (for example the 'Unsorted Videos' folder) were
added to the download list in a 'Check all' operation, even when they had
been checked before. This no longer happens, by default. To restore the
original behaviour, click 'Edit > System preferences... > Operations >
Downloads > For simulate downloads, don't check a video in a folder more
than once' to deselect it
- Fixed the button for finding videos by date, which was not working at all in
the previous version
- Fixed an occasional 'signal is not defined' error when the user stops an
operation (for example, a download operation)
- Various inconsistencies in the way alternative download destinations are
handled have all been fixed. Download operations sometimes freezed
indefinitely, because Tartube doesn't download two channels/playlists/
folders with the same download destination at the same time. The code has
been updated to prevent the freeze from ever happening again
- Fixed some more crashes caused by Gtk during a download operation
MINOR NEW FEATURES
- Videos downloaded into a temporary folder are deleted when Tartube restarts.
After shutting down Tartube, users often like to copy these videos
somewhere else on their hard drive. You can now ask Tartube to open the
temporary directories (folders), before shutting down, which will remind
you to do something with the videos. To enable this behaviour, click
'Edit > System preferences... > Filesystem > Temporary folders'
- In the main window's list of videos, the date is now displayed as 'today' and
'yesterday' when possible. This behaviour can be disabled in 'Edit > System
preferences... > Windows > Main window'
- During refresh operations, a progress bar is now visible in the bottom-left
corner of the window (just like the one visible during a download
operation)
- When setting an alternative download destination for a channel/playlist/
folder (for example, by right-clicking a channel and selecting 'Channel
actions > Set download destination...'), the dialogue window has been
updated to show the previously selected alternative at the top of the list.
This should save a lot of time when setting the alternative download
destination for many channels/playlists/folders
- The alternative download destination, if any, is now visible in the tooltips
for the channel/playlist/folder
- Improved the appearance of the dialogue windows seen when Tartube runs for
the first time
- Tweaked the appearance of the list of channels/playlists/folders in the
main window, so that for items with long names, more text is visible
MINOR FIXES
- Fixed a 'No such file or directory' error seen during a download operation,
if an external hard drive suddenly become disconnected (for example, if
the cable falls out)
- Fixed rare problems in loading Tartube's config file
- After a download operation, the list in the top half of the progress tab
often had one or two items in it, even when 'Hide active rows after they
are finished' was selected. Fixed
- The length of lines of text, and spacing between lines, in various dialogue
windows has been made uniform
- Improved the appearance of the main window by adding frames around everything
- Renamed some misnamed icon files. The old icon files were being used in the
MS Windows installer, so fixed that too
- If the user performed two successive refresh operations, the second one
halted after a couple of seconds. Fixed
- When videos are deleted from Tartube's database, any post-processing
artefacts are now deleted with them
- Fixed a few incorrect regex-matching actions
- The user can specify that the main 'Download all' button should be
desensitised, but the setting was not applied correctly after Tartube
restarted. Fixed
- Removed a duplicate menu option in the Video Index popup menu
- Tartube channels, playlists and folders keep counts of the number of videos
inside them, including the number of favourite videos, downloaded videos,
and so on. The code was not working correctly, so the counts were not
always accurate. This version updates the code and recalculates all of the
counts
- Fixed folder icons with an incorrect colour in various edit windows
- Fixed markup errors for videos whose URL contained an ampersand character
- Fixed the Gtk warning when closing the 'Add new video(s)' dialogue window
- Updated the installer scripts for MS Windows, so they don't try to update the
Windows registry (the code has never worked)
- You can no longer set videos as favourite, or new (etc), in an empty channel,
playlist or folder
- In the video list, labels can be right-clicked to copy a video's location
(for example, so it can be copy-pasted somewhere else). This did not work
the same way for every clickable label, and in some cases did not work at
all. Fixed
- Tooltips for videos contained &amp; rather than a simple ampersand character.
Fixed
- Tartube debug messages for the mainapp.py file (which can only be enabled
by editing the file) now have a second debug flag, so the timer functions
can be filtered out
- Checked all keyboard shortcuts to remove duplicates
v1.4.0 (2 Feb 2020)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- The structures of files and directories (folders) in Tartube's data
directory (into which all videos are downloaded) has been changed in
response to Git #28. Tartube will be able to recognise both structures
forever, so there is no need to move anything around on your computer. (If
you actually want to move things around, see the README file)
- Creating a channel/playlist/folder starting with a full stop (period) is no
longer allowed; some channels/playlists/folders might be automatically
renamed when you open Tartube
- The edit and preference windows have been reorganised, adding a second
layer of tabs in many windows. This should hopefully make things a little
easier to find
- In the download options window, you can now specify multiple languages for
your subtitles, instead of just one (Git #47)
- Added some more filename formats (in Edit > General download options...
> Files > File names). When downloading a partial playlist (for example,
starting at the 5th video), youtube-dl cannot create files with the correct
number (naming the first file downloaded #1, instead of #5). Tartube can
now handle this correctly. In the drop-down box, use one of the formats
containing 'Autonumber' (Git #47)
- You can now limit the length of a download operation. This is particularly
useful on small devices, or when leaving Tartube to run overnight. Click
Edit > System preferences... > Scheduling > Stop, and choose one or more of
the new options (Git #47)
- When adding new videos, channels or playlists, you can now turn on clipboard
monitoring. Simply select a URL (for example, in your web browser), press
CTRL+C to copy it to your system's clipboard, and then Tartube will
automatically paste it into the dialogue window (Git #52)
- The MS Windows installer now includes a copy of AtomicParsley, so there is no
need to install it yourself. This does not affect Linux/BSD users, who can
continue installing AtomicParsley by the usual methods
- The list in the top half of the Progress Tab is often full, and it's
sometimes difficult to see what is being downloaded right at this moment.
You can now hide finished rows, if you want to, so that active rows appear
at the top of the list
MAJOR FIXES
- The Gtk graphics libraries have historically been full of bugs, which made
applications using Gtk unstable. Most of these bugs are fixed, but the
fixes can take years before they propogate into operating systems. If Gtk
v3.22 (or lower) is installed on your system, Tartube automatically
disables some minor cosmetic features to prevent crashes. If you are using
Gtk v3.24 or later, and are still experiencing unexplainable crashes, you
can now disable the cosmetic features regardless of Gtk version. Click
Edit > System preferences... > General > Modules, and select 'Assume that
Gtk is broken...'
- On Linux/BSD, attempts to update youtube-dl from the Tartube menu so,etimes
produced a 'permission denied' error. There are now new settings available
in 'Edit > System prefences... > youtube-dl > Shell command for update
operations'. If you installed youtube-dl using pip/pip3, the 'recommended'
options should now work, if they didn't work before. Some pip3 warning
messages, which caused Tartube to think the update had failed, are now
filtered out
- A user complained that his Tartube database file had been corrupted. We are
still not sure what the cause was, but the code has been changed to make
that kind of corruption impossible
- Fixed some occasional crashes when, during a download operation, Tartube
tried to sort the videos in the selected channel/playlist/folder
- When switching databases, if Tartube couldn't load the new database, it tried
again after being restarted, rather than trying to load the previous
(readable) database. This has now been fixed
- Some youtube-dl download options could be applied to playlists, but not
channels, even though youtube-dl allows them to be applied to be both.
Fixed, and updated some labels to make it clearer what the options are for
(Git #47)
- In all edit windows, the 'Apply' button at the bottom of the window did not
work. Fixed
MINOR NEW FEATURES
- Tartube icons have been updated, in some cases making them easier to identify
- In the Progress Tab, added tooltips to assist with identifying undownloaded
videos (Git #51)
- More types of YouTube error message can now be filtered out
- We have also added a customisable list of strings (or regular expressions);
if set, any matching error/warning messages (on any website) are filtered
out
- In the Video Index popup menus, 'rename default location' has been changed
to a much more comprehensible 'rename channel', etc
- You can now open a video in its system directory (folder) by right-clicking
it, and selecting 'Show location'
- You can now switch databases from the main menu. Click File > Change database
(which opens the preference window at the correct page; hopefully this is
quicker than trying to find the right page yourself)
- There was no way to save Tartube's config file (except by shutting down
Tartube). To do that, you can now click File > Save all
- If Tartube is unable to read the config file and/or database file, the text
in the resulting dialogue windows has been improved. In some circumstances,
multiple dialogue windows were produced; this has now been fixed
- In the download options window, the option to 'embed subtitles with video'
now appears in two different places, to make it easier to find (Git #47)
- If the 'Add new video(s)', 'Add a new channel' or 'Add a new playlist'
dialogue windows are open, you can now drag-and-drop into them (Linux/BSD
only). Modifications to the code mean that it's no longer possible to
drop one URL into the middle of an existing one, rendering both of them
useless
- Tartube checked URLs for validity before adding them, but this did not work
as well as intended. The code has been improved, so less garbage should
appear in the 'Add new video(s)' dialogue window, and so on
- You could already download a temporary copy of video(s) by right-clicking
them and selecting 'Temporary > Download', but that can be inconvenient
for multiple videos, as you had to wait for each download to finish. You
can now select 'Temporary > Mark for download' instead, which creates a
copy of the video in the 'Temporary Videos' folder. When you're ready to
download them all, just download that folder
- Minor improvements to aesthetics for some textviews and treeviews
MINOR FIXES
- Fixed incorrect operation of the checkbuttons in the Errors/Warnings Tab.
Added new checkbuttons to separate Tartube errors/warnings from youtube-dl
errors/warnings (Git #50)
- 'Child process exited with non-zero code' errors still appeared in the
Errors/Warnins tab, even if the user has disabled them. Fixed
- Tooltips for videos could not be enabled/disabled if no channel/playlist/
folder was selected. Fixed
- On MS Windows, edit/preference windows will no longer increase in size, if
there isn't enough room for each window's tabs
- Fixed rare 'Permission denied' errors when trying to create a directory
(folder) on the filesystem
- In the download options edit window, the combobox for audio formats had
multiple and ever-increasing empty spaces. Fixed
- In the download options window, File > File names, the default value for the
custom format was garbled. Fixed, and it should now be working as intended
- During a simulated download, videos which are not in a channel or playlist
(for example, videos in the 'Unsorted Videos' folder) did not appear in
the Results List in the Progress Tab. Fixed
- Fixed an unprintable character in the licence declaration, visible in
Tartube's 'About' window
- When deleting a video, Tartube will now delete more related files (such as
those produced when post-processing a video)
- Removed a few duplicate ISO 639-1 language codes
v1.3.077 (26 Jan 2020)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- Drag and drop (for example, from a web browser into Tartube's main window)
is now fully working on Linux/BSD. On MS Windows, drag and drop does not
work at all for any Gtk application. It is unlikely that the Tartube
authors can do anything about this (Git #35)
- The 'Add new video(s)' dialogue window can now handle URLs representing
channels and playlists, as well as URLs representing individual videos.
During a download operation, if Tartube is expecting an individual video
but receives a channel/playlist, it will automatically create a new
channel, and download videos into that channel. You can change this default
behaviour, if you want (Edit > System preferences... > URL flexibility
preferences)
- To change the name of the new channel/playlist, right-click it and select
'Filesystem > Rename default location...'
- If Tartube creates a channel, which should really be a playlist, then you
can now convert one to the other. Right-click a channel and select
'Channel actions > Convert to playlist'. Right-click a playlist and select
'Playlist actions > Convert to channel'
- In the download options windows, it's now very easy to tell Tartube to
convert videos to sound files. Open the window by clicking 'Edit >
General download options...', click the 'Hide advanced download options'
button if necessary, click the 'Sound only' tab, select your preferences,
and apply them by clicking the OK button at the bottom of the window
- You can now see the download options applied to a video, channel, playlist
or folder without having to download anything. Right-click a video/channel/
playlist/folder and select 'Downloads > Show system command'
- During a download operation, the system commands used are now visible (by
default) in the Output Tab. The system command can also be displayed in the
terminal, if required; this is disabled by default
MINOR NEW FEATURES
- In the Output Tab, the summary page is now hidden by default. To make it
visible, click 'Edit > System Preferences... > Output >
Show a summary of active threads' and then restart Tartube
- In the Errors/Warnings Tab, added checkbuttons to filter out errors and/or
warning messages, if required (Git #50)
- In the Progress tab, in the top half of the window, you can now right-click
an unnamed video to open it in your web browser. This will be useful in
identifying videos that did not download, and whose name is unknown to
Tartube (Git #51)
- Columns in the Progress tab have been rearranged a little, so that the
user can more easily see how quickly the download is progressing, when
Tartube's main window is small
MAJOR FIXES
- Fixed multiple issues with Tartube, when running under Python 3.8
- Replaced all remaining references to the Python os.rename() function, which
can cause crashes on some filesystems (Git #34)
- Fixed crashes caused by the new YouTube error messages (January 2020), which
some versions of youtube-dl cannot handle correctly
- Fixed issues with the default location for videos, again. Fixed an issue
with adding folders inside the currently selected folder (Git #36, #46)
MINOR FIXES
- Fixed various Gtk warning messages, visible only on some systems
- Videos whose name contains an ampersand (&) character could not be opened by
clicking the 'Media player' label in the Video Catalogue. Fixed
- The properties windows for videos, channels and playlists showed a folder
icon, instead of a video/channel/playlist icon. Fixed
- The popup menu in the Progress tab, in the top half of the tab, did not work
as intended during a download operation, and again after a download
operation. Fixed both sets of issues
- Coloured text was not displayed in the Output Tab correctly. Fixed
v1.3.048 (23 Jan 2020)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- Tartube now creates an icon in the user's system tray. Closing the main
window now closes to the tray, by default. To disable this behaviour,
click Edit > System preferences > Windows > Deselect 'Close to the tray...'
- When that functionality is enabled, Tartube can be shutdown by clicking
File > Quit. Scheduled download operations will still take place if Tartube
has been closed to the tray. Implements Github issue #37
- Tartube can now show a desktop notification at the end of a download
operation, rather that a dialogue window. This does not work on MS Windows.
On other operating systems, enable desktop notifications by clicking
Edit > System preferences... > Operations > Show a desktop notification...
- When you click the 'Add new video(s)' button, the folder displayed in the
dialogue window is now the same folder that's selected in the main window
(if any). The same applies for adding channels, playlists and folders.
Fixes Github issue #36
- If you normally use the 'Check all' button rather than the 'Download all'
button, and if you want to download a temporary copy of one of the videos,
there's now an easier way to do it. In the Videos tab, right-click the
video, and select 'Temporary > Download' or 'Temporary > Download and
Watch'. A copy of the video is downloaded into the 'Temporary Videos'
folder, without affecting any other folders
MINOR NEW FEATURES
- The icons for channels and playlists have been replaced, to make it easier to
tell them apart. Some other icons have been replaced too
- Videos can now be dragged and dropped from a web browser (or similar
application) into Tartube's main window, which automatically adds the video
to the currently selected folder (or 'Unsorted Videos', if no folder is
selected). Unfortunately, the code is not yet working reliably. We are
looking for a solution (Github issue #35)
- The layout of the Format tab in the download options window has been improved
to alleviate confusion experienced by users trying to download a video to
a sound format such as .mp3 (only). See the new section in the README file
- A number of new video/audio formats have been added, for example several new
60fps formats, implementing Github issue #40
- When you apply download options to a video/channel/playlist/folder, the
options are now cloned from the default set of options (those visible in
Edit > General download options...). To disable this behaviour, click
Edit > System preferences... > Operations > When applying download options,
automatically clone general download options. Implements Github issue #39
- The options already applied to a video/channel/playlist/folder can now be
reset to match the general options, any time you want. Use the new button
at the bottom of the download options window, in the General tab
MAJOR FIXES
- Fixed a rare crash when the video's JSON filename was too long for the
operating system
- In the download options window, Formats tab, the user can add up to three
video formats. The third format, if added, was always ignored. Fixed
MINOR FIXES
- If you perform a refresh operation on a folder, the operation now applies to
all videos, channels, playlists and folders inside it
- After adding a video to the folder that's currently selected, the video does
not appear immediately in the video catalogue. Fixed
- In the Progress and Errors/Warnings tabs, the column headers scrolled away
along with the rest of the list. Fixed; they are now always visible
- Video nicknames were not set correctly after an update operation. Fixed
- During a refresh operation, a video's name was compared against the full
filepath (filename and extension), which produced none of the intended
matches. Fixed
- The edit/preference windows had a tendency to increase in size without
limits. Fixed
- A video's annotations.xml file was not deleted correctly, when required.
Fixed
- In the download options window, the option 'hls-prefer-ffmpeg' is now working
correctly
- In the download options window, the 'prefer avconv over ffmpeg' options have
been desensitised on MS Windows, as there is no known method of using
Tartube with avconv on MS Windows
- youtube-dl creates a file, ytdl-archive.txt, recording all the videos that
it has downloaded. This can interfere if the user tries to re-download the
video(s) for any reason. Create of the ytdl-archive.txt file can now be
disabled (Edit > System preferences... > youtube-dl > Deselect 'Allow
youtube-dl to create its own archive...')
- If creation of the archive file is nonetheless enabled, Tartube can now
re-download video(s) without problems
- In rare circumstances, Tartube was unable to redraw the video catalogue
(the right-hand side of the Videos tab). Fxied
v1.3.007 (20 Dec 2019)
-------------------------------------------------------------------------------
MAJOR FIXES
- v1.3.007 was completely broken when replacing an earlier installation. Fixed
- When Tartube's data directory was copied from one place to another (for
example, from one external drive to another), Tartube did not adapt to the
change very well. The way file paths are stored in Tartube's database has
been changed to eliminate this problem
MINOR FIXES
- Fixed an invalid time value which (sometimes) prevented a refresh operation
from completing correctly
v1.3.0 (20 Dec 2019)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- Tartube on MS Windows did not recognise FFmpeg or AVConv. You can now tell
Tartube to download and install a compatible version of FFmpeg from the
main menu (Operations > Install FFmpeg). Tartube still cannot recognise
the ordinary version of FFmpeg, and it still does not recognise AVConv at
all. It is unlikely that this situation can be remedied
- A new Output Tab has been added, in which you can see what is happening
internally when you check or download videos, update youtube-dl, install
FFmpeg, or refresh the Tartube database. The amount of information shown
can be customised in the System preferences window. The information can
still be written to STDOUT/STDERR, if required
- For users on other operating systems, the system preferences window
displayed the wrong location of the FFmpeg/AVConv executable. This has now
been fixed
- There are now two simple ways to specify the video resolution you want to
download (for example, 1080p). You can use the download options window
(Edit > General download options... > Formats, and then choose a video
format like 'any format [1080p]'). You can also use the new spinbutton at
the bottom of the Progress Tab. Both of these methods have the same effect,
so it's not necessary to use both of them. Tartube will download videos in
that resolution if possible, or in the next highest available resolution
otherwise
- The download options window has been simplified, with only the most useful
options visible. If you want to see the full range of options, open any
download options window, and in the General tab, click the new 'Show
advanced download options' button
MINOR NEW FEATURES
- By default, temporary folders are no longer emptied when Tartube shuts down,
but only when Tartube starts up. This means you can continue watching
temporary videos you've downloaded even after shutting down Tartube. If you
want temporary folders to be emptied on shutdown, as before, select Edit >
System preferences... > Videos > Empty temporary folders when Tartube shuts
down
MAJOR FIXES
- Tartube experienced a whole range of problems when downloading videos to a
hard drive that was running out of space. Tartube now checks the available
disk space before starting to download anything, and continues checking it
throughout the download process. You can specify how much disk space should
be available in the System Preferences window. If the hard drive, despite
your best efforts, actually does run out of space, Tartube is now much more
resilient (and can usually halt the download process, rather than
crashing). The amount of disk space available is now visible in the System
Preferences window
- Adding a channel/playlist/folder whose name included a slash, for example
'Adam/Eve's Channel', had unfortunate consequences, with Tartube creating
a directory (folder) at the wrong location. Slashes are now automatically
converted to hyphens, which solves the problem
- In Tartube's window, dragging a channel/playlist/folder to a new location in
the tree changes the hard drive, moving a directory (folder) to a new
location in the filesystem. If a directory (folder) with the same name
already existed at that location, an invisible error occurs. Tartube now
displays a visible error so the user can delete the duplicate directory
(folder) manually
- When refreshing the Tartube database (e.g. Operations > Refresh database),
the moviepy module freezes if it encounters a corrupted video file. We
can't fix the moviepy module, but the Tartube code has been made much more
resilient
- When refreshing the Tartube database, Tartube made bad decisions if it was
looking for a video called 'ymca.mp4', but found a video called
'ymca.webm'. This has been fixed
- Tartube is now able to detect if its data directory (into which videos are
downloaded) doesn't exist. Usually this is because an external hard drive
has not been mounted; the user is now warned about this, so they can mount
it
- On MS Windows, if the user has updated youtube-dl or installed FFmpeg,
Tartube no longer freezes on shutdown
- On MS Windows, Tartube was unable to open a video file in the system's
default media player, if the name contained an ampersand. Fixed
MINOR FIXES
- When deleting large channels/playlists/folders, sometimes not everything was
deleted, and the user had to delete the item a second time. This was due to
inconsistencies in the Tartube database, which have now been fixed
- Channels/playlists/folders beginning with a number, e.g. '5 Pewdiepie', were
supposed to be displayed in numerical order, rather than in strict
alphabetical order. This did not work as intended (e.g. '11 Pewdiepie' was
listed before '1 T-Series'). Fixed again
- In the 'Delete channel' dialogue window (and so on), the name of the channel
to be deleted is now displayed prominently
- The size of the MS Windows installer has been reduced by about 40%
v1.2.008 (30 Sep 2019)
-------------------------------------------------------------------------------
MINOR NEW FEATURES
- Tartube now ignores the YouTube 'WARNING: video doesn't have subtitles'
by default. You can change this setting, if you want to
MAJOR FIXES
- When moving a channel/playlist/folder to a different place on your
filesystem, or when renaming a channel/playlist/folder, in certain rare
situations data in the Tartube database isn't updated correctly. This may
lead to a freeze or a crash. I'm not sure yet what the cause is, but I have
added temporary code to prevent the problem affecting any user
- Fixed error messages generated when checking/downloading individual
channels/playlists/folders
- Fixed faulty code for importing videos/channels/playlists/folders into the
database
v1.2.0 (31 Aug 2019)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- Multiple channels, playlists and/or folders can now download their videos to
a single location. The README.rst file explains how it works, and why you
might want to do it
- You can also tell Tartube to download all videos into the 'Unsorted Videos'
or 'Temporary Videos' folders, instead of downloading them into separate
directories/folders for each channel and playlist
- Added automatic deletion of videos, disabled by default. Before enabling it,
you should do a 'Check all' or 'Download all' operation, which will create
the necessary youtube-dl archive files
- Added archiving. A video, channel, playlist or folder that is marked
archived won't be auto-deleted (but can still be deleted manually by the
user)
- You can now disable both checking and downloading a channel, playlist or
folder, if you want to. (It was already possible to just disabled
downloading them)
- Download operations can now be scheduled to take place at regular intervals
- You can now 'Download and watch' a video. The video is opened in your
system's default media player as soon as it has been downloaded
- Tartube can now download a video's annotations file automatically. Warnings
generated by YouTube about the lack of annotations are ignored by default
- For channels/playlists/folders containing many videos, you can now skip to
the first video uploaded after a certain date, using the new button in the
toolbar at the bottom of the Videos Tab
- The lists in the Results Tab can now be right-clicked, so you can change the
order in which videos/channels/playlists/folders are checked/downloaded,
abandon a download, play a video directly from the Results List, delete a
video directly from the Results List, and so on
- The edit window for youtube-dl options has been improved, adding many new
options to the GUI interface
- Plain text exports of Tartube's database can now be re-imported. Some
inconsistencies with the import process (from JSON and plain text files)
have been fixed
MINOR NEW FEATURES
- Added tooltips in several places. If you don't want to see tooltips above
videos/channels/playlists/folders, you can turn them off
- You can now select multiple videos in the Video Catalogue, and apply an
action to all of them (by right-clicking them)
- You can now switch to smaller icons in the Video Index (on the left side of
the Videos Tab), if you want to
- You can now force the Video Index to expand its tree whenever you click on a
folder, revealing any channels/playlists it contains. This is disabled by
default
- If you only want to check videos, and never download them, you can disable
the 'Download all' buttons. Individual videos/channels/playlists/folders
can still be downloaded by right-clicking them
- Tartube's file structure has changed. If you run it from the command line,
you might need to use a (slightly) different command. See the README.rst
for details of what command to use
- The path to the FFmpeg/AVConv executable can now be specified by the user.
This will be especially helpful for MS Windows users
- Columns in the Progress Tab can now be manually resized
- The Tartube website can now be opened from the main window menu
- XDG has been added as an optional dependency, for the benefit of Debian
packagers
MAJOR FIXES
- The MS Windows installer should now work for everyone
- Refresh operations are now stable (should not crash) on systems with Gtk 3.22
or earlier
- When downloads were disabled for a folder, downloads for channels/playlists/
folders inside that folder were still enabled. This is counter-intuitive,
so disabling downloads for a folder disables downloads for everything it
contains
- Marking/unmarking a video as favourite caused certain problems, which should
now be fixed
MINOR FIXES
- Fixed some unicode errors in reading JSON and plain text files
- Fixed the wrong page size displayed in the toolbar at the bototm of the
Videos Tab
- Empty lines in a video's description are now preserved when they're displayed
in Tartube's main window
v1.1.0 (18 Aug 2019)
-------------------------------------------------------------------------------
MAJOR NEW FEATURES
- You can now create an export of Tartube's database. This export contains
details of videos, channels, playlists and/or folders, but not the videos
themselves (or any of the thumbnail/description/metadata files). The
export can take two forms: JSON data, or plain text. If JSON data, the
exported file can later be imported into another Tartube database (imports
from plain text are not implemented yet). You can export either the
entire database, or just one channel/playlist/folder (and everythng it
contains)
- You can now change the name of a channel, playlist of folder. This doesn't
have any effect on your filesystem; it only changes the name displayed in
the Video Index (the left-hand side of the Videos Tab). This might be
useful for channels and playlists that have weird or very short names. For
example, right-click a folder and select 'Folder actions > Set nickname...'
- You can also rename the channel, playlist or folder, and this action DOES
affect the filesystem, changing the directory/folder on your hard drive
where the channel/playlist/folder videos are stored. For example, right-
click a folder and select 'Filesystem > Rename location...'
- If you change the format of a downloaded video file from the default 'Title'
to, for example, 'Title + ID', the Video catalogue (the right-hand side of
the Videos Tab) will now simply display the video's title (which should be
easier to read). To see the actual filename, you can right-click the video
and select 'Show properties'. This only works if the video's metadata was
downloaded when the video itself was downloaded; this is now turned on by
default for all new users
MINOR NEW FEATURES
- When adding videos, channels and playlists, the contents of the system's
clipboard was automatically copied into the window. This can now be turned
off, if you wish (Edit > System preferences... > Windows > When adding
videos/channels/playlists, copy URLs from the system clipboard)
- When adding channels/playlists, you can set the dialogue window to stay open,
which makes adding multiple channels/playlists quicker (Edit >
System preferences... > Windows > When adding channels/playlists, keep the
dialogue window open)
- When creating channels/playlists/folders inside an existing parent folder, a
dialogue window which stays open can be told to continuously re-use that
parent folder (Edit > System preferences... > Windows > When adding
channels/playlists, re-use the optional parent folder)
- When checking/downloading videos, the Results List (the bottom half of the
Progress Tab) can now display videos in reverse order, so you don't have to
scroll down to see the video that was just checked/downloaded (Edit >
System Preferences... > Windows > Show results in reverse order)
- The number of ystem error and warning messages displayed in their own tab
is visible in the tab's label. The label is usually reset when the tab
is made visible. You can now disable this behaviour, preserving the numbers
until the 'Clear the list' button is explicitly clicked (Edit >
System preferences... > Windows > Don't remove number of system messages
from tab label until 'Clear' button is clicked)
- Items in the Video Index (on the left-hand side of the Videos Tab) are sorted
alphabetically. The sorting algorithm has been improved to take account of
numbered items, such that '1 Music' will now appear before '11 Comedy'
- For the benefit of package maintainers (such as a Debian package), Tartube
now uses an environment variable which will prevent Tartube from updating
the youtube-dl binary, if specified. See the comments in setup.py
MAJOR FIXES
- Users whose system Gtk is earlier than v3.24 (this includes many current
Linux distros, but the MS Windows installer) will have experienced graphics
issues, and endless error messages in the terminal window, if open. If your
system Gtk is earlier than v3.24, Tartube will no longer update the Video
Index during a download operation; this should fix the issue at the cost of
disabling real-time updates of the number of videos in each channel,
playlist and folder. Your system's Gtk version is now visible in Tartube's
System Preferences window
- After loading the config file, the download limits were set, but not
displayed in the Progress Tab. Fixed
- Rarely, Tartube crashes (or freezes) when loading a video's JSON metadata
file from your filesystem (but now when downloading it). This should no
longer happen
MINOR FIXES
- The 'Channel properties', 'Folder properties' (etc) windows used the wrong
icon (displaying a folder in the wrong colour). Fixed
- Fixed a 'list modified during sort' error during a download operation
v1.0.0 (31 Jul 2019)
-------------------------------------------------------------------------------
- First beta release
- Fixed some issues with the MS Windows installer
- Some parts of the Tartube window displayed the wrong icons. Fixed
v0.7.0 (7 Jul 2019)
-------------------------------------------------------------------------------
- This is the first release candidate for v1.0.0
- The MS Windows installer has been redesigned again (thanks to slartie for his
generous assistance in getting it working). Some MS Windows 10 users were
still complaining that Tartube would not run; this should be fixed now
- MS Windows should find that the annoying terminal window is no longer visible
- Deleting individual videos, then adding them to the Tartube database again,
could cause problems with the statistics displayed in the Video Index (e.g.
'All Videos (0, -1)'. Fixed
- Deleting and individual video by right-clicking it removed the video from the
database, but didn't delete the video file itself. Fixed
- If an empty channel/playlist was selected, new videos did not automatically
appear in the Video Catalogue during a download operation. Fixed
- Fixed some issues with the Temporary Videos folder
- Fixed more issues with videos being dislayed in the wrong order in the Video
Catalogue
- Fixed more issues with the scrollbars not resetting themselves when switching
between channels/playlists/folders
- After right-clicking a folder, selecting Mark Videos > New didn't work. Fixed
- When marking videos as new/favourite, you can now do this to all videos
in a folder, including all child channels/playlists/folders, or you can do
it just to the videos actually inside that folder. There are several new
popup menu options for emptying a folder, or for removing all its videos
- Tartube will no longer issue a system error if you drag a folder onto itself
in the Video Index
- Fixed the remaining issues caused by the Gtk graphics libraries
- Confirmed that various Gtk issues present in Gtk3.22 are not present in
Gtk 3.24. Users with Gtk 3.22 on their system will be warned to update it.
These warnings can be disabled, if required
v0.6.0 (4 Jul 2019)
-------------------------------------------------------------------------------
- Some MS Windows users, especially on Windows 10, report that they can't run
Tartube at all. In an effort to get around this, the installer has been
redesigned. The way Tartube communicates with youtube-dl on MS Windows has
been changed. youtube-dl update operations should now work flawlessly.
Please report any further problems at our GitHub page; this might be a fix
to issue #10
- Users on Linux/*BSD can now run Tartube directly from the command line, after
installing it (see the README)
- Occasionally, videos were downloaded (or checked) successfully, but Tartube
failed to notice them. This issue should now be fixed
- When checking videos/channels/playlists/folders, only new videos will now
appear in the Results List (in the 'Progress' Tab)
- The toolbar has been redesigned. MS Windows users won't see labels at all
(so everything should fit). Users on all system can turn labels on or off.
Tooltips have been added to the buttons, in case the labels are turned off
- Fixed yet another problem with button to switch the location of Tartube's
data folder (#6)
- Tartube no longer requires the python 'validators' module
- Tartube can now ignore YouTube copyright messages, and also 'Child process
exited with non-zero code' messages, meaning that they won't appear in
Tartube's Errors/Warnings Tab. (They are not ignored by default)
- Tartube now applies a 60-second timeout when youtube-dl tries to download
a video's metadata (since youtube-dl uses a 10-minute timeout); this can be
turned off, if required (#9)
- Fixed some more issues with the way videos are sorted in the Video Catalogue
(it's still not 100%)
- Users can no longer type in comboboxes
- Tartube now spots when the user adds a channel or playlist URL as a video;
only the first video in the channel/playlist is now downloaded
- You can no longer remove download options from a media data object when the
download options edit window is still open
- When you update youtube-dl, Tartube will now tell you which youtube-dl
version is installed. Tartube will no longer claim the update operation
fails if you're using pip (rather than pip3)
v0.5.0 (1 Jul 2019)
-------------------------------------------------------------------------------
- On MS Windows, the fix from v0.4.0 to prevent a crash whenever the user tries
to change the location of Tartube's data directory, did not work. Fixed it
again (#6), and added some dialogue window to make it clearer to the user
what is going on
- Fixed problems with dragging-and-dropping (or otherwise moving) a channel,
playlist or folder to a new location (such as another folder). The videos
were not updated with their new location. The new code will fix any
problems in the Tartube database
- Fixed numerous problems with the code that sorts videos into the right order,
and which displays videos in the Video Catalogue in the right order
- The 'Switch' button no longer resets the page back to the first one
- When switching between channels/playlists/folders, scrollbars are
automatically moved back to the top
- In the toolbar beneath the Video Catalogue, there are two new buttons for
scrolling to the top or bottom of the visible page. Users can now select
a different page just by typing the page number and pressing RETURN. The
same applies to the page size - there is no button to click any more, just
type the new size and press RETURN
- Fixed several problems which were still preventing selection of different
video formats (#3)
- Added bare-bones aac, m4a, mp3, ogg and wav as recognised video formats
- A set of download options can now be completely reset to their default values
- The 'Add Videos' dialogue window, and some others, don't behave well when the
user resizes them. Fixed (#4)
- Added a small folder icon to the the 'Add Videos' dialogue window, and
others, so the user is less likely to forget to set a custom location
- In the 'Add Videos' dialogue window, and others, URLs are now tidied up a
bit because being copied from the clipboard, eliminating leading/trailing
whitespace and empty lines
- The Gtk test window, available for MS Windows after using the installer,
now contains some text (to make it clear that the window is working as
intended)
v0.4.0 (29 Jun 2019)
------------------------------------------------------------------------------
- Drastic improvements to overall performance. Download operations are now much
smoother. You should notice a much lighter burden on your machine's CPU
- The graphics libraries struggled to draw lists containing hundreds (or
thousands!) of videos, so the Video Catalogue has been split into pages. It
typically takes less then a second to show the 'All Videos' folder, if it
contains hundreds of videos, rather than several minutes
- If you just want to find new videos, you can now tell Tartube to stop
checking/downloading channels/playlists as soon as notifications of videos
you've already checked/downloaded start arriving. This works well on
YouTube, which sends the newest videos first, but might not work well on
all websites. The new functionality is turned off by default. Click
'Edit > System preferences > Performance > Time-saving preferences' to turn
it on
- The installer for 32-bit MS Windows failed under all circumstances. Applied a
fix
- On MS Windows, the uninstaller was invisible. It can now be executed from the
Start Menu
- On MS Windows, fixed a crash whenever the user tried to changed the location
of Tartube's data directory (#6)
- Fixed some problems with the 'Add videos' dialogue window, which tried to add
all sorts of invalid URLs (such as lines containing only empty space). The
other 'Add' dialogue windows were also fixed
- A limit to the length of a channel/playlist/folder name now applies
- Leading/trailing whitespace is now removed from URLs and channel/playlist/
folder names
- The 'Add videos' dialogue window now checks for duplicate URLs, and when
found, doesn't add them to the database
- Fixed various problems caused by deleting a video/channel/playlist/folder
(such as the numbers visible in the Video Index). If you've been using an
earlier version of Tartube, any such problems with the database will be
automatically repaired for you (#8)
- Videos which haven't actually been downloaded can now be deleted from the
database by right-clicking them
- Added new options for making backups of the Tartube database file, in case it
becomes corrupted. See the settings in
'Edit > System preferences... > Backups'
- The 'Switch' button now switches between four 'skins' (instead of the
previous two). Two of them show the name of a video's parent channel/
playlist/folder; the other two show the video's description
- Fixed incorrect formatting in the DASH and 3D file formats (#3)
v0.3.0 (25 Jun 2019)
-------------------------------------------------------------------------------
- Tartube will now run on MS Windows
- Fixed some more crashes
- Fixed some issues with video descriptions containing quotes and ampersand
characters
v0.2.0 (23 Jun 2019)
-------------------------------------------------------------------------------
- Corrected old Python2 code to work on Python3
- Greatly expanded the README file
- Fixed the constant crashes
- Fixed some Gtk problems (but others remain unfixed)
- Fixed downloads for users who haven't installed Ffmpeg, and for sites that
don't support Ffmpeg
- Several other minor tweaks/fixes
v0.1.0 (27 May 2019)
-------------------------------------------------------------------------------
- This is the first public release of Tartube
- This is the first public release of GymBob

View File

@ -3,5 +3,4 @@ recursive-include docs *
recursive-include icons *
recursive-include pack *
recursive-include screenshots *
recursive-include share *

File diff suppressed because it is too large Load Diff

View File

@ -1 +1 @@
name = "tartube"
name = "gymbob"

View File

@ -1 +1 @@
#Tartube
#GymBob

758
gymbob/editwin.py Normal file
View File

@ -0,0 +1,758 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (C) 2020 A S Lewis
#
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# this program. If not, see <http://www.gnu.org/licenses/>.
"""Edit window classes."""
# Import Gtk modules
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GObject, GdkPixbuf
# Import other modules
import re
# Import our modules
# ...
# Classes
class ProgEditWin(Gtk.Window):
"""Python class for an 'edit window' to modify values in a gymprog.GymProg
object.
Args:
app_obj (mainapp.TartubeApp): The main application object
edit_obj (gymprog.GymProg): The object whose attributes will be edited
in this window
"""
def __init__(self, app_obj, edit_obj):
Gtk.Window.__init__(self, title='Edit \'' + edit_obj.name + '\'')
# IV list - class objects
# -----------------------
# The mainapp.GymBobApp object
self.app_obj = app_obj
# The gymprog.GymProg object being edited
self.edit_obj = edit_obj
# IV list - Gtk widgets
# ---------------------
self.main_grid = None # Gtk.Grid
self.reset_button = None # Gtk.Button
self.apply_button = None # Gtk.Button
self.ok_button = None # Gtk.Button
self.cancel_button = None # Gtk.Button
self.treeview = None # Gtk.TreeView
self.liststore = None # Gtk.ListStore
# IV list - other
# ---------------
# Size (in pixels) of gaps between edit window widgets
self.spacing_size = self.app_obj.default_spacing_size
# When the user changes a value, it is not applied to self.edit_obj
# immediately; instead, it is stored temporarily in this dictionary
# If the user clicks the 'OK' or 'Apply' buttons at the bottom of the
# window, the changes are applied to self.edit_obj
# If the user clicks the 'Reset' or 'Cancel' buttons, the dictionary
# is emptied and the changes are lost
# The key-value pairs in the dictionary correspond directly to the
# names of attributes, and their values in self.edit_obj
# Key-value pairs are added to this dictionary whenever the user makes
# a change (so if no changes are made when the window is closed, the
# dictionary will still be empty)
self.edit_dict = {}
# Code
# ----
# Set up the edit window
self.setup()
# Public class methods
def setup(self):
"""Called by self.__init__().
Sets up the edit window when it opens.
"""
# Set the default window size
self.set_default_size(
self.app_obj.edit_win_width,
self.app_obj.edit_win_height,
)
# Set the window's Gtk icon list
self.set_icon_list(self.app_obj.main_win_obj.icon_pixbuf_list)
# Set up the window's containing box
self.main_grid = Gtk.Grid()
self.add(self.main_grid)
# Set up main widgets
self.setup_tab()
self.setup_button_strip()
self.setup_gap()
# Procedure complete
self.show_all()
# Inform the main window of this window's birth (so that GymBox doesn't
# the clock to start until all configuration windows have closed)
self.app_obj.main_win_obj.add_child_window(self)
# Add a callback so we can inform the main window of this window's
# destruction
self.connect('destroy', self.close)
def setup_tab(self):
"""Called by self.setup().
Sets up all widgets (except the button strip at the bottom of the
window).
"""
mini_grid = Gtk.Grid()
self.main_grid.attach(mini_grid, 0, 0, 1, 1)
mini_grid.set_border_width(self.spacing_size)
mini_grid.set_column_spacing(self.spacing_size)
mini_grid.set_row_spacing(self.spacing_size)
# Add a treeview
frame = Gtk.Frame()
mini_grid.attach(frame, 0, 0, 5, 1)
frame.set_hexpand(True)
frame.set_vexpand(True)
scrolled = Gtk.ScrolledWindow()
frame.add(scrolled)
scrolled.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
self.treeview = Gtk.TreeView()
scrolled.add(self.treeview)
self.treeview.set_headers_visible(True)
for i, column_title in enumerate( ['#', 'Time', 'Message', 'Sound'] ):
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn(
column_title,
renderer_text,
text=i,
)
self.treeview.append_column(column_text)
self.treeview_reset()
self.treeview_refill()
# Add editing widgets beneath the treeview
label = Gtk.Label('Time (in seconds)')
mini_grid.attach(label, 0, 1, 1, 1)
entry = Gtk.Entry()
mini_grid.attach(entry, 1, 1, 2, 1)
entry.set_hexpand(True)
entry.set_max_width_chars(4)
label2 = Gtk.Label('Message')
mini_grid.attach(label2, 0, 2, 1, 1)
entry2 = Gtk.Entry()
mini_grid.attach(entry2, 1, 2, 4, 1)
entry2.set_hexpand(True)
label3 = Gtk.Label('Sound (optional)')
mini_grid.attach(label3, 0, 3, 1, 1)
combostore = Gtk.ListStore(str)
combostore.append('')
for sound_file in self.app_obj.sound_list:
combostore.append( [sound_file] )
combo = Gtk.ComboBox.new_with_model(combostore)
mini_grid.attach(combo, 1, 3, 2, 1)
renderer_text = Gtk.CellRendererText()
combo.pack_start(renderer_text, True)
combo.add_attribute(renderer_text, 'text', 0)
combo.set_entry_text_column(0)
combo.set_active(0)
button = Gtk.Button('Add message')
mini_grid.attach(button, 0, 4, 1, 1)
button.connect(
'clicked',
self.on_button_add_clicked,
entry,
entry2,
combo,
)
button2 = Gtk.Button('Update message')
mini_grid.attach(button2, 1, 4, 1, 1)
button2.connect(
'clicked',
self.on_button_update_clicked,
entry,
entry2,
combo,
)
button3 = Gtk.Button('Delete message')
mini_grid.attach(button3, 2, 4, 1, 1)
button3.connect('clicked', self.on_button_delete_clicked)
button4 = Gtk.Button('Move up')
mini_grid.attach(button4, 3, 4, 1, 1)
button4.connect('clicked', self.on_button_move_up_clicked)
button5 = Gtk.Button('Move down')
mini_grid.attach(button5, 4, 4, 1, 1)
button5.connect('clicked', self.on_button_move_down_clicked)
def setup_button_strip(self):
"""Called by self.setup().
Creates a strip of buttons at the bottom of the window. Any changes the
user has made are applied by clicking the 'OK' or 'Apply' buttons, and
cancelled by using the 'Reset' or 'Cancel' buttons.
The window is closed by using the 'OK' and 'Cancel' buttons.
"""
hbox = Gtk.HBox()
self.main_grid.attach(hbox, 0, 1, 1, 1)
# 'Reset' button
self.reset_button = Gtk.Button('Reset')
hbox.pack_start(self.reset_button, False, False, self.spacing_size)
self.reset_button.get_child().set_width_chars(10)
self.reset_button.set_tooltip_text(
'Reset changes without closing the window',
);
self.reset_button.connect('clicked', self.on_button_reset_clicked)
# 'Apply' button
self.apply_button = Gtk.Button('Apply')
hbox.pack_start(self.apply_button, False, False, self.spacing_size)
self.apply_button.get_child().set_width_chars(10)
self.apply_button.set_tooltip_text(
'Apply changes without closing the window',
);
self.apply_button.connect('clicked', self.on_button_apply_clicked)
# 'OK' button
self.ok_button = Gtk.Button('OK')
hbox.pack_end(self.ok_button, False, False, self.spacing_size)
self.ok_button.get_child().set_width_chars(10)
self.ok_button.set_tooltip_text('Apply changes');
self.ok_button.connect('clicked', self.on_button_ok_clicked)
# 'Cancel' button
self.cancel_button = Gtk.Button('Cancel')
hbox.pack_end(self.cancel_button, False, False, self.spacing_size)
self.cancel_button.get_child().set_width_chars(10)
self.cancel_button.set_tooltip_text('Cancel changes');
self.cancel_button.connect(
'clicked',
self.on_button_cancel_clicked,
)
def setup_gap(self):
"""Called by self.setup().
Adds an empty box beneath the button strip for aesthetic purposes.
"""
hbox = Gtk.HBox()
self.main_grid.attach(hbox, 0, 2, 1, 1)
hbox.set_border_width(self.spacing_size)
def retrieve_val(self, name):
"""Can be called by anything.
Any changes the user has made are temporarily stored in self.edit_dict.
Each key corresponds to an attribute in the object being edited,
self.edit_obj.
If 'name' exists as a key in that dictionary, retrieve the
corresponding value and return it. Otherwise, the user hasn't yet
modified the value, so retrieve directly from the attribute in the
object being edited.
Args:
name (str): The name of the attribute in the object being edited
Returns:
The original or modified value of that attribute.
"""
if name in self.edit_dict:
return self.edit_dict[name]
else:
attrib = getattr(self.edit_obj, name)
return attrib.copy()
def treeview_reset(self):
"""Called by self.setup_tab() and several callback functions.
Creates a model for the Gtk.TreeView to use, replacing any previous
model in use.
"""
self.liststore = Gtk.ListStore(int, int, str, str)
self.treeview.set_model(self.liststore)
def treeview_refill(self):
"""Called by self.setup_tab() and several callback functions, usually
after a call to self.treeview_reset().
Fills the Gtk.Treeview with data.
"""
msg_group_list = self.retrieve_val('msg_group_list')
count = 0
for mini_list in msg_group_list:
count += 1
mod_list = mini_list.copy()
mod_list.insert(0, count)
self.liststore.append(mod_list)
def check_entries(self, entry, entry2):
"""Called by self.on_button_add_clicked() and
.on_button_update_clicked().
Extracts the value of the two Gtk.Entrys. If either value is invalid,
displays an error message.
Args:
entry, entry2 (Gtk.Entry): The entry boxes to check
Return values:
time (int), msg (str): The contents of the boxes, or (None, None)
if either value is invalid
"""
time = entry.get_text()
msg = entry2.get_text()
if time == '' or not re.search('^\d+$', time):
msg_dialogue_win = Gtk.MessageDialog(
self,
0,
Gtk.MessageType.ERROR,
Gtk.ButtonsType.OK,
'Invalid time value (must\nbe an integer in seconds)',
)
msg_dialogue_win.run()
msg_dialogue_win.destroy()
return None, None
# elif msg == '':
#
# msg_dialogue_win = Gtk.MessageDialog(
# self,
# 0,
# Gtk.MessageType.ERROR,
# Gtk.ButtonsType.OK,
# 'Invalid message (must\ncontain some characters)',
# )
# msg_dialogue_win.run()
# msg_dialogue_win.destroy()
# return None, None
else:
return time, msg
def apply_changes(self):
"""Called by self.on_button_ok_clicked() and
self.on_button_apply_clicked().
Any changes the user has made are temporarily stored in self.edit_dict.
Apply to those changes to the object being edited.
"""
# Apply any changes the user has made
for key in self.edit_dict.keys():
setattr(self.edit_obj, key, self.edit_dict[key])
# The changes can now be cleared
self.edit_dict = {}
# Save the programme file
self.app_obj.save_prog(self.edit_obj)
def close(self, also_self):
"""Called from callback in self.setup().
Inform the main window that this window is closing.
Args:
also_self (editwin.ProgEditWin): Another copy of self
"""
self.app_obj.main_win_obj.del_child_window(self)
# (Callbacks)
def on_button_add_clicked(self, button, entry, entry2, combo):
"""Called from a callback in self.setup_tab().
Adds a message to the treeview.
Args:
button (Gtk.Button): The widget clicked
entry, entry2 (Gtk.Entry): The contents of these entry boxes are
aded to the treeview (and the gymprog.GymProg programme)
combo (Gtk.ComboBox): The contents of this combobox is added to the
treeview
"""
time, msg = self.check_entries(entry, entry2)
tree_iter = combo.get_active_iter()
model = combo.get_model()
filename = model[tree_iter][0]
if time is not None:
msg_group_list = self.retrieve_val('msg_group_list')
# mini_list is in the form
# (time_in_seconds, message, optional_sound_file)
mini_list = [int(time), str(msg), str(filename)]
msg_group_list.append(mini_list)
self.edit_dict['msg_group_list'] = msg_group_list
mini_list2 = [
len(msg_group_list), int(time), str(msg), str(filename),
]
self.liststore.append(mini_list2)
def on_button_apply_clicked(self, button):
"""Called from a callback in self.setup_button_strip().
Applies any changes made by the user, but doesn't close the window.
Args:
button (Gtk.Button): The widget clicked
"""
# Apply any changes the user has made
self.apply_changes()
def on_button_cancel_clicked(self, button):
"""Called from a callback in self.setup_button_strip().
Destroys any changes made by the user and closes the window.
Args:
button (Gtk.Button): The widget clicked
"""
# Destroy the window
self.destroy()
def on_button_delete_clicked(self, button):
"""Called from a callback in self.setup_tab().
Deletes a message from the treeview.
Args:
button (Gtk.Button): The widget clicked
"""
selection = self.treeview.get_selection()
(model, iter) = selection.get_selected()
if iter is None:
# Nothing selected
return
row_num = model[iter][0]
msg_group_list = self.retrieve_val('msg_group_list')
count = 0
mod_list = []
# mini_list is in the form
# (time_in_seconds, message, optional_sound_file)
for mini_list in msg_group_list:
count += 1
if count != row_num:
mod_list.append(mini_list)
self.edit_dict['msg_group_list'] = mod_list
self.treeview_reset()
self.treeview_refill()
def on_button_ok_clicked(self, button):
"""Called from a callback in self.setup_button_strip().
Destroys any changes made by the user and then closes the window.
Args:
button (Gtk.Button): The widget clicked
"""
# Apply any changes the user has made
self.apply_changes()
# Destroy the window
self.destroy()
def on_button_move_down_clicked(self, button):
"""Called from a callback in self.setup_tab().
Moves a message one place down in the treeview.
Args:
button (Gtk.Button): The widget clicked
"""
selection = self.treeview.get_selection()
(model, iter) = selection.get_selected()
if iter is None:
# Nothing selected
return
row_num = model[iter][0]
msg_group_list = self.retrieve_val('msg_group_list')
if row_num < len(msg_group_list):
count = 0
mod_list = []
# mini_list is in the form
# (time_in_seconds, message, optional_sound_file)
for mini_list in msg_group_list:
count += 1
if count != row_num:
mod_list.append(mini_list)
else:
insert_list = mini_list.copy()
mod_list.insert(row_num, insert_list)
self.edit_dict['msg_group_list'] = mod_list
self.treeview_reset()
self.treeview_refill()
selection = self.treeview.get_selection()
selection.select_path(row_num)
def on_button_move_up_clicked(self, button):
"""Called from a callback in self.setup_tab().
Moves a message one place up in the treeview.
Args:
button (Gtk.Button): The widget clicked
"""
selection = self.treeview.get_selection()
(model, iter) = selection.get_selected()
if iter is None:
# Nothing selected
return
row_num = model[iter][0]
msg_group_list = self.retrieve_val('msg_group_list')
if row_num > 1:
count = 0
mod_list = []
# mini_list is in the form
# (time_in_seconds, message, optional_sound_file)
for mini_list in msg_group_list:
count += 1
if count != row_num:
mod_list.append(mini_list)
else:
insert_list = mini_list.copy()
mod_list.insert(row_num - 2, insert_list)
self.edit_dict['msg_group_list'] = mod_list
self.treeview_reset()
self.treeview_refill()
selection = self.treeview.get_selection()
selection.select_path(row_num - 2)
def on_button_reset_clicked(self, button):
"""Called from a callback in self.setup_button_strip().
Destroys any changes made by the user and updates the window, showing
self.edit_obj's original values
Args:
button (Gtk.Button): The widget clicked
"""
# Empty self.edit_dict, destroying any changes the user has made
self.edit_dict = {}
# Reset the treeview
self.treeview_reset()
self.treeview_refill()
# Render the changes
self.show_all()
def on_button_update_clicked(self, button, entry, entry2, combo):
"""Called from a callback in self.setup_tab().
Updates the selected message in the treeview.
Args:
button (Gtk.Button): The widget clicked
entry, entry2 (Gtk.Entry): The contents of these entry boxes are
aded to the treeview, updating the selected line (and the
gymprog.GymProg programme)
combo (Gtk.ComboBox): The contents of this combobox is added to the
treeview
"""
selection = self.treeview.get_selection()
(model, iter) = selection.get_selected()
if iter is None:
# Nothing selected
return
row_num = model[iter][0]
time, msg = self.check_entries(entry, entry2)
tree_iter = combo.get_active_iter()
model = combo.get_model()
filename = model[tree_iter][0]
if time is not None:
msg_group_list = self.retrieve_val('msg_group_list')
# mini_list is in the form
# (time_in_seconds, message, optional_sound_file)
mini_list = [int(time), str(msg), str(filename)]
msg_group_list[row_num - 1] = mini_list
self.edit_dict['msg_group_list'] = msg_group_list
self.treeview_reset()
self.treeview_refill()

37
tartube/tartube → gymbob/gymbob Executable file → Normal file
View File

@ -1,7 +1,7 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (C) 2019-2020 A S Lewis
# Copyright (C) 2020 A S Lewis
#
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
@ -17,7 +17,7 @@
# this program. If not, see <http://www.gnu.org/licenses/>.
"""Tartube main file."""
"""Gymbob main file."""
# Import Gtk modules
@ -31,23 +31,23 @@ import importlib.util
# Add module directory to path to prevent import issues
spec = importlib.util.find_spec('tartube')
spec = importlib.util.find_spec('gymbob')
if spec is not None:
sys.path.append(os.path.abspath(os.path.dirname(spec.origin)))
# Import our modules
import mainapp
# 'Global' variables
__packagename__ = 'tartube'
__prettyname__ = 'Tartube'
__version__ = '2.0.006'
__date__ = '3 Mar 2020'
__copyright__ = 'Copyright \xa9 2019-2020 A S Lewis'
__packagename__ = 'gymbob'
__prettyname__ = 'GymBob'
__version__ = '1.002'
__date__ = '28 Mar 2020'
__copyright__ = 'Copyright \xa9 2020 A S Lewis'
__license__ = """
Copyright \xa9 2019-2020 A S Lewis.
Copyright \xa9 2020 A S Lewis.
This program is free software: you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free Software
@ -64,18 +64,11 @@ this program. If not, see <http://www.gnu.org/licenses/>.
__author_list__ = [
'A S Lewis',
]
__description__ = 'A front-end GUI for youtube-dl,\n' \
+ 'partly based on youtube-dl-gui\n' \
+ 'and written in Python 3 / Gtk 3'
__website__ = 'http://tartube.sourceforge.io'
__app_id__ = 'io.sourceforge.tartube'
# There are three executables; this default one, and two others used in Debian/
# RPM packaging. The others are identical, except for the values of these
# variables
__pkg_install_flag__ = False
__pkg_strict_install_flag__ = False
__description__ = 'Simple script to prompt the user during a workout'
__website__ = 'http://gymbob.sourceforge.io'
__app_id__ = 'io.sourceforge.gymbob'
# Start Tartube
app = mainapp.TartubeApp()
# Start Gymbob
app = mainapp.GymBobApp()
app.run(sys.argv)

75
gymbob/gymprog.py Normal file
View File

@ -0,0 +1,75 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (C) 2020 A S Lewis
#
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# this program. If not, see <http://www.gnu.org/licenses/>.
"""Workout programme classes."""
# Import Gtk modules
# ...
# Import other modules
# ...
# Import our modules
# ...
# Classes
class GymProg(object):
"""Can be called by anything.
Python class that handles a workout programme.
Args:
name (str): A unique programme name
msg_group_list (list): List of messages to display in the main window,
in the form described below
"""
# Standard class methods
def __init__(self, name, msg_group_list=[]):
# IV list - other
# ---------------
# A unique programme name
self.name = name
# The programme is made up of a sequence of messages, displayed after
# fixed intervals
# A message group is a list in the form [int, txt, txt]:
# int: Time in seconds between this message and the previous one
# txt: The message itself (or an empty string to overwrite the
# previous message, leaving no message visible)
# txt: Name of the sound file (.wav or .ogg) in the ../sounds
# folder to play when this message is displayed (or an empty
# string to play no sound)
# The message groups are stored in this sequential list
self.msg_group_list = msg_group_list

901
gymbob/mainapp.py Normal file
View File

@ -0,0 +1,901 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (C) 2020 A S Lewis
#
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# this program. If not, see <http://www.gnu.org/licenses/>.
"""Main application class."""
# Import Gtk modules
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GObject, GdkPixbuf
# Import other modules
from gi.repository import Gio
import datetime
import json
import os
import playsound
import re
import subprocess
import sys
import threading
import time
# Import our modules
import editwin
import gymprog
import __main__
import mainwin
# Classes
class GymBobApp(Gtk.Application):
"""Main python class for the GymBob application."""
def __init__(self, *args, **kwargs):
super(GymBobApp, self).__init__(
*args,
application_id=None,
flags=Gio.ApplicationFlags.FLAGS_NONE,
**kwargs)
# Instance variable (IV) list - class objects
# -------------------------------------------
# The main window object, set as soon as it's created
self.main_win_obj = None
# Instance variable (IV) list - other
# -----------------------------------
# Default window sizes (in pixels)
self.main_win_width = 550
self.main_win_height = 400
# Default size of edit windows (in pixels)
self.edit_win_width = 600
self.edit_win_height = 400
# Default size (in pixels) of space between various widgets
self.default_spacing_size = 5
# For quick lookup, the directory in which the 'gymbob' executable
# file is found, and its parent directory
self.script_dir = sys.path[0]
self.script_parent_dir = os.path.abspath(
os.path.join(self.script_dir, os.pardir),
)
# The directory in which workout programmes can be stored (as .json
# files)
self.data_dir = os.path.abspath(
os.path.join(
os.path.expanduser('~'),
__main__.__packagename__ + '-data',
),
)
# List of sound files found in the ../sounds directory
self.sound_list = []
# So that a sound can be played from within its own thread, the name of
# the sound file to be played is stored here (temporarily) before
# self.play_sound() can be called
self.sound_file = None
# At all times (after initial setup), a GObject timer runs
# The timer's ID
self.script_timer_id = None
# The timer interval time (in milliseconds)
self.script_timer_time = 100
# Flag set to True if the clock is running
self.clock_running_flag = False
# The time (matches time.time()) at which the user clicked the START
# button
self.clock_start_time = 0
# The time at which the STOP button was clicked. If the START button is
# subsequently clicked, the value of self.clock_start_time is
# adjusted
self.clock_stop_time = 0
# Dictionary of gymprog.GymProg objects, each one handling a single
# workout programme. Dictionary in the form
# prog_dict[unique_name] = GymProg object
# ...where 'unique_name' is a string with a maximum length of 16
self.prog_dict = {}
# The current workout programme object
self.current_prog_obj = None
# Flag set to True when the clock is started (from 0), which starts the
# workout programme
self.current_prog_started_flag = False
# When the clock is started (from 0), the contents of the workout
# programme is copied into this IV
self.current_prog_msg_group_list = []
# The time (matches the clock time, not time.time()) at which the first
# message in self.current_msg_group_list should be displayed. As soon
# as the message is displayed, it is removed from the list
self.current_prog_next_update_time = None
# Flag set to True if sound should has been muted
self.mute_sound_flag = False
def do_startup(self):
"""Gio.Application standard function."""
Gtk.Application.do_startup(self)
# Menu actions
# ------------
# 'GymBob' column
quit_menu_action = Gio.SimpleAction.new('quit_menu', None)
quit_menu_action.connect('activate', self.on_menu_quit)
self.add_action(quit_menu_action)
# 'Programmes' column
new_prog_menu_action = Gio.SimpleAction.new('new_prog_menu', None)
new_prog_menu_action.connect('activate', self.on_menu_new_prog)
self.add_action(new_prog_menu_action)
switch_prog_menu_action = Gio.SimpleAction.new(
'switch_prog_menu',
None,
)
switch_prog_menu_action.connect('activate', self.on_menu_switch_prog)
self.add_action(switch_prog_menu_action)
edit_prog_menu_action = Gio.SimpleAction.new('edit_prog_menu', None)
edit_prog_menu_action.connect('activate', self.on_menu_edit_prog)
self.add_action(edit_prog_menu_action)
delete_prog_menu_action = Gio.SimpleAction.new(
'delete_prog_menu',
None,
)
delete_prog_menu_action.connect('activate', self.on_menu_delete_prog)
self.add_action(delete_prog_menu_action)
# 'Help' column
about_menu_action = Gio.SimpleAction.new('about_menu', None)
about_menu_action.connect('activate', self.on_menu_about)
self.add_action(about_menu_action)
go_website_menu_action = Gio.SimpleAction.new('go_website_menu', None)
go_website_menu_action.connect('activate', self.on_menu_go_website)
self.add_action(go_website_menu_action)
# Button actions
# --------------
start_button_action = Gio.SimpleAction.new('start_button', None)
start_button_action.connect('activate', self.on_button_start)
self.add_action(start_button_action)
stop_button_action = Gio.SimpleAction.new('stop_button', None)
stop_button_action.connect('activate', self.on_button_stop)
self.add_action(stop_button_action)
reset_button_action = Gio.SimpleAction.new('reset_button', None)
reset_button_action.connect('activate', self.on_button_reset)
self.add_action(reset_button_action)
def do_activate(self):
"""Gio.Application standard function."""
self.start()
def do_shutdown(self):
"""Gio.Application standard function."""
# Stop immediately
Gtk.Application.do_shutdown(self)
# Public class methods
def start(self):
"""Called by self.do_activate().
Performs general initialisation.
"""
# Create the main window and make it visible
self.main_win_obj = mainwin.MainWin(self)
self.main_win_obj.show_all()
# Start the GObject timer, which runs continuously, even when the
# visible clock/stopwatch is not running
self.script_timer_id = GObject.timeout_add(
self.script_timer_time,
self.script_timer_callback,
)
# Get a list of available sound files, and sort alphabetically
sound_dir = os.path.abspath(
os.path.join(self.script_parent_dir, 'sounds'),
)
for (dirpath, dir_list, file_list) in os.walk(sound_dir):
for filename in file_list:
if filename != 'COPYING':
self.sound_list.append(filename)
self.sound_list.sort()
# Create the data directory (in which workout programmes are stored as
# .json files), if it doesn't already exist
if not os.path.isdir(self.data_dir):
os.makedirs(self.data_dir)
# Load any workout programmes found in the data directory
for (dirpath, dir_list, file_list) in os.walk(self.data_dir):
# The first file (alphabetically) is the new current workout
# programme
file_list.sort()
for filename in file_list:
if re.search('\.json$', filename):
self.load_prog(filename)
def stop(self):
"""Called by self.on_menu_quit().
Performs a clean shutdown.
"""
# I'm outta here!
self.quit()
def save_prog(self, prog_obj):
"""Called by self.on_menu_new_prog() and
editwin.ProgEditWin.apply_changes().
Saves the specified workout programme as .json file.
Args:
prog_obj (gymprog.GymProg): The workout programme to save
"""
# Prepare the file's data
utc = datetime.datetime.utcfromtimestamp(time.time())
file_path = os.path.abspath(
os.path.join(self.data_dir, prog_obj.name + '.json'),
)
json_dict = {
# Metadata
'script_name': __main__.__packagename__,
'script_version': __main__.__version__,
'save_date': str(utc.strftime('%d %b %Y')),
'save_time': str(utc.strftime('%H:%M:%S')),
# Data
'prog_name': prog_obj.name,
'prog_msg_group_list': prog_obj.msg_group_list,
}
# Try to save the file
try:
with open(file_path, 'w') as outfile:
json.dump(json_dict, outfile, indent=4)
except:
# Save failed
msg_dialogue_win = Gtk.MessageDialog(
self.main_win_obj,
0,
Gtk.MessageType.ERROR,
Gtk.ButtonsType.OK,
'Failed to save the \'' + prog_obj.name + '\' programme',
)
msg_dialogue_win.run()
msg_dialogue_win.destroy()
def load_prog(self, filename):
"""Called by self.start().
Loads the named workout programme (a .json file).
Args:
filename (str): The name of the workout programme to load, matching
a key in self.prog_dict, and the name of a file in the data
directory
"""
# Get the full filepath for the specified file
filepath = os.path.abspath(
os.path.join(self.data_dir, filename),
)
# Try to load the file, ignoring any failures
try:
with open(filepath) as infile:
json_dict = json.load(infile)
except:
return
# Do some basic checks on the loaded data
if not json_dict \
or not 'script_name' in json_dict \
or not 'script_version' in json_dict \
or not 'save_date' in json_dict \
or not 'save_time' in json_dict \
or not 'prog_name' in json_dict \
or not 'prog_msg_group_list' in json_dict \
or json_dict['script_name'] != __main__.__packagename__:
return
# Ignore duplicate names (the file name and programme name should be
# the same, but perhaps the user has been editing it by hand...)
if json_dict['prog_name'] in self.prog_dict:
return
# File loaded; create a gymprog.GymProg object for it
prog_obj = gymprog.GymProg(
json_dict['prog_name'],
json_dict['prog_msg_group_list'],
)
self.prog_dict[json_dict['prog_name']] = prog_obj
# The first file loaded is the new current programme. (Files are loaded
# in alphabetical order)
if not self.current_prog_obj:
self.current_prog_obj = prog_obj
self.main_win_obj.update_win_title(prog_obj.name)
self.main_win_obj.update_buttons_on_current_prog()
self.main_win_obj.update_menu_items_on_prog()
def delete_prog(self, prog_obj):
"""Called by self.on_menu_delete_prog().
Deletes the .json file corresponding to the specified workout
programme.
Args:
prog_obj (gymprog.GymProg): The workout programme to delete
"""
file_path = os.path.abspath(
os.path.join(self.data_dir, prog_obj.name + '.json'),
)
# Delete the file
try:
os.remove(file_path)
except:
# Deletion failed
msg_dialogue_win = Gtk.MessageDialog(
self.main_win_obj,
0,
Gtk.MessageType.ERROR,
Gtk.ButtonsType.OK,
'Failed to delete the \'' + prog_obj.name + '\' programme',
)
msg_dialogue_win.run()
msg_dialogue_win.destroy()
def play_sound(self):
"""Called periodically by self.script_timer_callback().
If a sound file has been set, play it (unless sound is muted).
"""
sound_file = self.sound_file
self.sound_file = None
# (The value might be None or an empty string)
if not self.mute_sound_flag and sound_file:
path = os.path.abspath(
os.path.join(self.script_parent_dir, 'sounds', sound_file),
)
if os.path.isfile(path):
playsound.playsound(path)
# Callback class methods
# (Timers)
def script_timer_callback(self):
"""Called by GObject timer created by self.start(), ten times a second.
Returns:
1 to keep the timer going, or None to halt it
"""
# (Use the same time value to update all main window widgets)
current_time = time.time()
# If the programme is running...
if self.clock_running_flag:
# Update the contents of textviews in the main window
# (De)sensitise menu items and/or buttons
clock_time = current_time - self.clock_start_time
self.main_win_obj.update_clock_textview(clock_time)
if self.current_prog_next_update_time is not None \
and self.current_prog_next_update_time <= clock_time \
and self.current_prog_msg_group_list:
mini_list = self.current_prog_msg_group_list.pop(0)
self.main_win_obj.update_this_info_textview(str(mini_list[1]))
if mini_list[2]:
self.sound_file = mini_list[2]
if self.current_prog_msg_group_list:
# Programme not finished; show the next message as well (in
# a different colour)
next_mini_list = self.current_prog_msg_group_list[0]
self.current_prog_next_update_time += next_mini_list[0]
self.main_win_obj.update_next_info_textview(
next_mini_list[1],
)
else:
# Programme finished
self.current_prog_next_update_time = None
self.main_win_obj.update_countdown_textview('')
self.main_win_obj.update_next_info_textview('')
if self.current_prog_next_update_time is not None:
next_time = self.current_prog_next_update_time - clock_time
self.main_win_obj.update_countdown_textview(next_time)
# Play a sound file, if required. In case of long sounds, perform this
# action inside its own thread
if self.sound_file is not None:
thread = threading.Thread(target=self.play_sound)
thread.start()
# Return 1 to keep the timer going
return 1
# (Widgets)
def on_button_start(self, action, par):
"""Called from a callback in self.do_startup().
Starts the clock.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Update IVs
self.clock_running_flag = True
if not self.clock_start_time:
self.clock_start_time = time.time()
else:
self.clock_start_time += (time.time() - self.clock_stop_time)
# (De)sensitise buttons
self.main_win_obj.update_buttons_on_start()
if not self.current_prog_started_flag:
# Clock is at 0; start the workout programme
self.current_prog_started_flag = True
self.current_prog_msg_group_list \
= self.current_prog_obj.msg_group_list.copy()
# (The workout programme might be empty, so we have to check for
# that)
if self.current_prog_msg_group_list:
# mini_list is in the form
# (time_in_seconds, message, optional_sound_file)
mini_list = self.current_prog_msg_group_list[0]
self.current_prog_next_update_time = mini_list[0]
if len(self.current_prog_msg_group_list) > 1:
# Programme not finished; show the next message as well (in
# a different colour)
next_mini_list = self.current_prog_msg_group_list[0]
self.main_win_obj.update_next_info_textview(
next_mini_list[1],
)
# (De)sensitise menu items
self.main_win_obj.update_menu_items_on_prog()
def on_button_stop(self, action, par):
"""Called from a callback in self.do_startup().
Stops the clock.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Update IVs
self.clock_running_flag = False
self.clock_stop_time = time.time()
# (De)sensitise buttons
self.main_win_obj.update_buttons_on_stop()
# (De)sensitise menu items
self.main_win_obj.update_menu_items_on_prog()
def on_button_reset(self, action, par):
"""Called from a callback in self.do_startup().
Resets the clock.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Update IVs
self.clock_running_flag = False
self.clock_start_time = 0
self.clock_stop_time = 0
self.current_prog_started_flag = False
self.current_prog_msg_group_list = []
self.current_prog_next_update_time = None
# Reset textviews
self.main_win_obj.update_clock_textview('')
self.main_win_obj.update_countdown_textview('')
self.main_win_obj.update_this_info_textview('')
self.main_win_obj.update_next_info_textview('')
# (De)sensitise buttons
self.main_win_obj.update_buttons_on_reset()
# (De)sensitise menu items
self.main_win_obj.update_menu_items_on_prog()
def on_menu_about(self, action, par):
"""Called from a callback in self.do_startup().
Show a standard 'about' dialogue window.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
dialogue_win = Gtk.AboutDialog()
dialogue_win.set_transient_for(self.main_win_obj)
dialogue_win.set_destroy_with_parent(True)
dialogue_win.set_program_name(__main__.__packagename__.title())
dialogue_win.set_version('v' + __main__.__version__)
dialogue_win.set_copyright(__main__.__copyright__)
dialogue_win.set_license(__main__.__license__)
dialogue_win.set_website(__main__.__website__)
dialogue_win.set_website_label(
__main__.__prettyname__ + ' website'
)
dialogue_win.set_comments(__main__.__description__)
dialogue_win.set_logo(
self.main_win_obj.icon_pixbuf_dict['icon_64'],
)
dialogue_win.set_authors(__main__.__author_list__)
dialogue_win.set_title('')
dialogue_win.connect('response', self.on_menu_about_close)
dialogue_win.show()
def on_menu_about_close(self, action, par):
"""Called from a callback in self.on_menu_about().
Close the 'about' dialogue window.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
action.destroy()
def on_menu_delete_prog(self, action, par):
"""Called from a callback in self.do_startup().
Prompts the user to delete a workout programme.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Open a dialogue window
dialogue_win = mainwin.DeleteProgDialogue(self.main_win_obj)
response = dialogue_win.run()
# Retrieve user choices from the dialogue window...
prog_name = dialogue_win.prog_name
# ...before destroying the dialogue window
dialogue_win.destroy()
if response == Gtk.ResponseType.OK:
# Remove the workout programme from the registry
del self.prog_dict[prog_name]
# If it's the current programme, update some things
if self.current_prog_obj is not None \
and self.current_prog_obj.name == prog_name:
# Reset the window title
self.main_win_obj.update_win_title()
# Delete the corresponding file
self.delete_prog(self.current_prog_obj)
self.current_prog_obj = None
# Artificially click the RESET button to reset the clock and
# empty the main window of text
self.main_win_obj.reset_button.clicked()
# (De)sensitise buttons
self.main_win_obj.update_buttons_on_current_prog()
# Desensitise menu items, if there are no workout programmes left
self.main_win_obj.update_menu_items_on_prog()
def on_menu_edit_prog(self, action, par):
"""Called from a callback in self.do_startup().
Prompts the user to edit the current workout programme.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Open the edit window immediately
editwin.ProgEditWin(self, self.current_prog_obj)
def on_menu_go_website(self, action, par):
"""Called from a callback in self.do_startup().
Opens the GymBob website.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
uri = __main__.__website__
# Open the GymBob home page in the user's default browser
if sys.platform == "win32":
os.startfile(uri)
else:
opener ="open" if sys.platform == "darwin" else "xdg-open"
subprocess.call([opener, uri])
def on_menu_new_prog(self, action, par):
"""Called from a callback in self.do_startup().
Prompts the user for the name of a new workout programme.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Open a dialogue window
dialogue_win = mainwin.NewProgDialogue(self.main_win_obj)
response = dialogue_win.run()
# Retrieve user choices from the dialogue window...
prog_name = dialogue_win.entry.get_text()
# ...before destroying the dialogue window
dialogue_win.destroy()
if response != Gtk.ResponseType.OK:
return
if prog_name in self.prog_dict:
# Duplicate programme names are not allowed
msg_dialogue_win = Gtk.MessageDialog(
self.main_win_obj,
0,
Gtk.MessageType.ERROR,
Gtk.ButtonsType.OK,
'A programme called \'' + prog_name + '\' already exists!',
)
msg_dialogue_win.run()
msg_dialogue_win.destroy()
return
# Create the new workout programme
prog_obj = gymprog.GymProg(prog_name)
self.prog_dict[prog_name] = prog_obj
# Set it as the current programme
self.current_prog_obj = prog_obj
# Save the empty programme, creating a .json file
self.save_prog(prog_obj)
# Update the main window title to show the current programme
self.main_win_obj.update_win_title(prog_obj.name)
# (De)sensitise menu items and buttons
self.main_win_obj.update_buttons_on_current_prog()
self.main_win_obj.update_menu_items_on_prog()
# Open an edit window for the new programme immediately
editwin.ProgEditWin(self, prog_obj)
def on_menu_switch_prog(self, action, par):
"""Called from a callback in self.do_startup().
Prompts the user with a list of workout programmes from which to
select.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Open a dialogue window
dialogue_win = mainwin.SwitchProgDialogue(self.main_win_obj)
response = dialogue_win.run()
# Retrieve user choices from the dialogue window...
prog_name = dialogue_win.prog_name
# ...before destroying the dialogue window
dialogue_win.destroy()
if response == Gtk.ResponseType.OK:
# Set the current programme
self.current_prog_obj = self.prog_dict[prog_name]
# (De)sensitise menu items and buttons
self.main_win_obj.update_buttons_on_current_prog()
self.main_win_obj.update_menu_items_on_prog()
# Artificially click the RESET button to reset the clock and empty
# the main window of text
self.main_win_obj.reset_button.clicked()
# Update the main window title to show the current programme
self.main_win_obj.update_win_title(self.current_prog_obj.name)
def on_menu_quit(self, action, par):
"""Called from a callback in self.do_startup().
Terminates the GymBob app.
Args:
action (Gio.SimpleAction): Object generated by Gio
par (None): Ignored
"""
# Terminate the script
self.stop()
# Set accessors
def set_mute_sound_flag(self, flag):
if not flag:
self.mute_sound_flag = False
else:
self.mute_sound_flag = True

998
gymbob/mainwin.py Normal file
View File

@ -0,0 +1,998 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (C) 2020 A S Lewis
#
# This program is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License as published by the Free Software
# Foundation, either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# this program. If not, see <http://www.gnu.org/licenses/>.
"""Main window class."""
# Import Gtk modules
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GObject, Gdk, GdkPixbuf
# Import other modules
from gi.repository import Gio
import math
import os
# Import our modules
import __main__
import mainapp
# Classes
class MainWin(Gtk.ApplicationWindow):
"""Called by mainapp.GymBobApp.start().
Python class that handles the main window.
Args:
app_obj (mainapp.TartubeApp): The main application object
"""
def __init__(self, app_obj):
super(MainWin, self).__init__(
title=__main__.__prettyname__,
application=app_obj
)
# IV list - class objects
# -----------------------
# The main application
self.app_obj = app_obj
# IV list - Gtk widgets
# ---------------------
# (from self.setup_grid)
self.grid = None # Gtk.Grid
# (from self.setup_menubar)
self.menubar = None # Gtk.MenuBar
self.new_prog_menu_item = None # Gtk.MenuItem
self.switch_prog_menu_item = None # Gtk.MenuItem
self.edit_prog_menu_item = None # Gtk.MenuItem
self.delete_prog_menu_item = None # Gtk.MenuItem
# (from self.setup_win)
self.clock_textview = None # Gtk.TextView
self.countdown_textview = None # Gtk.TextView
self.this_info_textview = None # Gtk.TextView
self.next_info_textview = None # Gtk.TextView
self.start_button = None # Gtk.Button
self.pause_button = None # Gtk.Button
self.stop_button = None # Gtk.Button
self.reset_button = None # Gtk.Button
# IV list - other
# ---------------
# Colours to use in the upper/lower textviews, and the font size (in
# points, default value is 10)
self.clock_bg_colour = '#000000'
self.clock_text_colour = '#FFFFFF'
self.clock_font_size = 40
self.countdown_bg_colour = '#000000'
self.countdown_text_colour = '#FF0000'
self.countdown_font_size = 40
self.this_info_bg_colour = '#000000'
self.this_info_text_colour = '#FFFFFF'
self.this_info_font_size = 30
self.next_info_bg_colour = '#000000'
self.next_info_text_colour = '#FF0000'
self.next_info_font_size = 30
# Gymbob icon files are loaded into pixbufs, ready for use. Dictionary
# in the form:
# key - a string like 'icon_512'
# value - a Gdk.Pixbuf
self.icon_pixbuf_dict = {}
# The same list of pixbufs in sequential order
self.icon_pixbuf_list = []
# List of edit windows (editwin.ProgEditWin objects) that are currently
# open. The clock can't be started if any edit windows are open
self.edit_win_list = []
# Code
# ----
# Set up icon pixbufs
self.setup_icons()
# Set up the main window
self.setup_win()
# Public class methods
def setup_icons(self):
"""Called by self.__init__().
Sets up pixbufs for GymBob icons.
"""
# The default location for icons is ../icons
# When installed via PyPI, the icons are moved to ../gymbob/icons
# When installed via a Debian/RPM package, the icons are moved to
# /usr/share/gymbob/icons
icon_dir_list = []
icon_dir_list.append(
os.path.abspath(
os.path.join(self.app_obj.script_parent_dir, 'icons'),
),
)
icon_dir_list.append(
os.path.abspath(
os.path.join(
os.path.dirname(os.path.realpath(__file__)),
'icons',
),
),
)
icon_dir_list.append(
os.path.join(
'/', 'usr', 'share', __main__.__packagename__, 'icons',
)
)
self.icon_pixbuf_list = []
for icon_dir_path in icon_dir_list:
if os.path.isdir(icon_dir_path):
for size in [16, 24, 32, 48, 64, 128, 256, 512]:
path = os.path.abspath(
os.path.join(
icon_dir_path,
'win',
__main__.__packagename__ + '_icon_' + str(size) \
+ '.png',
),
)
pixbuf = GdkPixbuf.Pixbuf.new_from_file(path)
self.icon_pixbuf_list.append(pixbuf)
self.icon_pixbuf_dict['icon_' + str(size)] = pixbuf
# Pass the list of pixbufs to Gtk
self.set_icon_list(self.icon_pixbuf_list)
def setup_win(self):
"""Called by self.__init__().
Sets up the main window, calling various function to create its
widgets.
"""
spacing = self.app_obj.default_spacing_size
# Set the default window size
self.set_default_size(
self.app_obj.main_win_width,
self.app_obj.main_win_height,
)
# Create main window widgets
self.grid = Gtk.Grid()
self.add(self.grid)
self.setup_menubar()
self.clock_textview = self.setup_textview(
1,
self.clock_bg_colour,
self.clock_text_colour,
self.clock_font_size,
0, 1, 1, 1,
)
self.countdown_textview = self.setup_textview(
2,
self.countdown_bg_colour,
self.countdown_text_colour,
self.countdown_font_size,
1, 1, 1, 1,
)
self.this_info_textview = self.setup_textview(
3,
self.this_info_bg_colour,
self.this_info_text_colour,
self.this_info_font_size,
0, 2, 2, 1,
)
self.next_info_textview = self.setup_textview(
4,
self.next_info_bg_colour,
self.next_info_text_colour,
self.next_info_font_size,
0, 3, 2, 1,
)
self.setup_dummy_textview()
# Separator
self.grid.attach(Gtk.Separator(), 0, 4, 2, 1)
hbox = Gtk.HBox()
self.grid.attach(hbox, 0, 5, 2, 1)
hbox.set_border_width(spacing * 2)
hbox.set_vexpand(True)
self.start_button = Gtk.Button('START')
hbox.pack_start(self.start_button, True, True, spacing)
self.start_button.set_action_name('app.start_button')
# (These buttons are desensitised until a programme is loaded/created)
self.start_button.set_sensitive(False)
self.stop_button = Gtk.Button('STOP')
hbox.pack_start(self.stop_button, True, True, spacing)
self.stop_button.set_sensitive(False)
self.stop_button.set_action_name('app.stop_button')
self.stop_button.set_sensitive(False)
self.reset_button = Gtk.Button('RESET')
hbox.pack_start(self.reset_button, True, True, spacing)
self.reset_button.set_action_name('app.reset_button')
self.reset_button.set_sensitive(False)
def setup_menubar(self):
"""Called by self.setup_win().
Sets up a Gtk.Menu at the top of the main window.
"""
self.menubar = Gtk.MenuBar()
self.grid.attach(self.menubar, 0, 0, 2, 1)
# GymBob column
file_menu_column = Gtk.MenuItem.new_with_mnemonic(
'_' + __main__.__prettyname__,
)
self.menubar.add(file_menu_column)
file_sub_menu = Gtk.Menu()
file_menu_column.set_submenu(file_sub_menu)
mute_sound_menu_item = Gtk.CheckMenuItem.new_with_mnemonic(
'_Mute sound',
)
mute_sound_menu_item.set_active(self.app_obj.mute_sound_flag)
mute_sound_menu_item.connect(
'activate',
self.on_menu_mute_sound,
)
file_sub_menu.append(mute_sound_menu_item)
# Separator
file_sub_menu.append(Gtk.SeparatorMenuItem())
quit_menu_item = Gtk.MenuItem.new_with_mnemonic('_Quit')
file_sub_menu.append(quit_menu_item)
quit_menu_item.set_action_name('app.quit_menu')
# Programmes column
edit_menu_column = Gtk.MenuItem.new_with_mnemonic('_Programmes')
self.menubar.add(edit_menu_column)
edit_sub_menu = Gtk.Menu()
edit_menu_column.set_submenu(edit_sub_menu)
self.new_prog_menu_item = Gtk.MenuItem.new_with_mnemonic(
'_New programme...',
)
edit_sub_menu.append(self.new_prog_menu_item)
self.new_prog_menu_item.set_action_name('app.new_prog_menu')
self.switch_prog_menu_item = Gtk.MenuItem.new_with_mnemonic(
'_Switch programme...',
)
edit_sub_menu.append(self.switch_prog_menu_item)
self.switch_prog_menu_item.set_action_name('app.switch_prog_menu')
self.edit_prog_menu_item = Gtk.MenuItem.new_with_mnemonic(
'_Edit current programme...',
)
edit_sub_menu.append(self.edit_prog_menu_item)
self.edit_prog_menu_item.set_action_name('app.edit_prog_menu')
# Separator
edit_sub_menu.append(Gtk.SeparatorMenuItem())
self.delete_prog_menu_item = Gtk.MenuItem.new_with_mnemonic(
'_Delete programme...',
)
edit_sub_menu.append(self.delete_prog_menu_item)
self.delete_prog_menu_item.set_action_name('app.delete_prog_menu')
# Help column
help_menu_column = Gtk.MenuItem.new_with_mnemonic('_Help')
self.menubar.add(help_menu_column)
help_sub_menu = Gtk.Menu()
help_menu_column.set_submenu(help_sub_menu)
about_menu_item = Gtk.MenuItem.new_with_mnemonic('_About...')
help_sub_menu.append(about_menu_item)
about_menu_item.set_action_name('app.about_menu')
go_website_menu_item = Gtk.MenuItem.new_with_mnemonic('Go to _website')
help_sub_menu.append(go_website_menu_item)
go_website_menu_item.set_action_name('app.go_website_menu')
# (Some menu items are desensitised until a programme is loaded/
# created)
self.edit_prog_menu_item.set_sensitive(False)
self.switch_prog_menu_item.set_sensitive(False)
self.delete_prog_menu_item.set_sensitive(False)
def setup_textview(self, widget_id, bg_colour, text_colour, font_size, \
x_pos, y_pos, width, height):
"""Called by self.setup_win().
Creates one of the main window textviews (there are four in all).
Args:
widget_id (int): Unique number for this textview (1-4)
bg_colour, text_colour (str): The colours to use in this textview
(e.g. '#FFFFFF')
font_size (int): The font size (in points, e.g. 10)
x_pos, y_pos, width, height (int): Coordinates on the Gtk3.Grid
Return values:
The Gtk.TextView created
"""
# Add a textview to the grid, using a css style sheet to provide (for
# example) monospaced white text on a black background
scrolled = Gtk.ScrolledWindow()
self.grid.attach(scrolled, x_pos, y_pos, width, height)
scrolled.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
frame = Gtk.Frame()
scrolled.add_with_viewport(frame)
style_provider = self.set_textview_css(
'#css_text_id_' + str(widget_id) \
+ ', textview text {\n' \
+ ' background-color: ' + bg_colour + ';\n' \
+ ' color: ' + text_colour + ';\n' \
+ '}\n' \
+ '#css_label_id_' + str(widget_id) \
+ ', textview {\n' \
+ ' font-family: monospace, monospace;\n' \
+ ' font-size: ' + str(font_size) + 'pt;\n' \
+ '}'
)
textview = Gtk.TextView()
frame.add(textview)
textview.set_wrap_mode(Gtk.WrapMode.WORD)
textview.set_editable(False)
textview.set_cursor_visible(False)
textview.set_hexpand(True)
textview.set_vexpand(True)
context = textview.get_style_context()
context.add_provider(style_provider, 600)
return textview
def setup_dummy_textview(self):
"""Called by self.setup_win(), immediately after calls to
self.setup_textview().
Resets css properties for the next Gtk.TextView created (presumably by
another application), so it uses the default style, not the css style
specified in the calls to self.setup_textview().
"""
# Create a dummy textview that's not visible in the main window
textview = Gtk.TextView()
style_provider = self.set_textview_css(
'#css_text_id_default, textview text {\n' \
+ ' background-color: unset;\n' \
+ ' color: unset;\n' \
+ '}\n' \
+ '#css_label_id_default, textview {\n' \
+ ' font-family: unset;\n' \
+ ' font-size: unset;\n' \
+ '}'
)
context = textview.get_style_context()
context.add_provider(style_provider, 600)
def set_textview_css(self, css_string):
"""Called by self.setup_upper_textview() and .setup_lower_textview().
Applies a CSS style to the current screen, which is used for the
Gtk.TextView that has just been created.
Called a third time to create a dummy textview with default properties.
Args:
css_string (str): The CSS style to apply
Returns:
The Gtk.CssProvider created
"""
style_provider = Gtk.CssProvider()
style_provider.load_from_data(bytes(css_string.encode()))
Gtk.StyleContext.add_provider_for_screen(
Gdk.Screen.get_default(),
style_provider,
Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION
)
return style_provider
# (Update widgets)
def update_win_title(self, prog_name=None):
"""Called by various functions.
Changes the title of the main window. If there is a current programme,
display its name alongside the name of the script.
Args:
prog_name (str): The name of a workout programme, matching a key
in self.app_obj.prog_dict
"""
if prog_name is not None:
self.set_title(__main__.__prettyname__ + ' [' + prog_name + ']')
else:
self.set_title(__main__.__prettyname__)
def update_clock_textview(self, time):
"""Called by various functions.
Updates the main window textview showing the current time.
Args:
time (int): The current time (since the programme begun) in seconds
"""
if not time:
self.clock_textview.get_buffer().set_text('')
else:
self.clock_textview.get_buffer().set_text(
self.convert_time_to_string(time),
)
def update_countdown_textview(self, time):
"""Called by various functions.
Updates the main window textview showing the time until the next
message to be displayed.
Args:
time (int): The time in seconds
"""
if not time:
self.countdown_textview.get_buffer().set_text('')
else:
self.countdown_textview.get_buffer().set_text(
self.convert_time_to_string(math.ceil(time)),
)
def update_this_info_textview(self, msg):
"""Called by various functions.
Updates the main window textview with the current message (if any).
Args:
msg (str): The message to display (use an empty string to clear the
textview)
"""
self.this_info_textview.get_buffer().set_text(str(msg))
def update_next_info_textview(self, msg):
"""Called by various functions.
Updates the main window textview with the next message (if any).
Args:
msg (str): The message to display (use an empty string to clear the
textview)
"""
self.next_info_textview.get_buffer().set_text(str(msg))
def update_menu_items_on_prog(self):
"""Called by various functions.
(De)sensitises menu items depending on whether any workout programmes
exist, or not.
"""
if self.app_obj.current_prog_started_flag:
self.new_prog_menu_item.set_sensitive(False)
else:
self.new_prog_menu_item.set_sensitive(True)
if not self.app_obj.prog_dict:
self.edit_prog_menu_item.set_sensitive(False)
else:
self.edit_prog_menu_item.set_sensitive(True)
if not self.app_obj.prog_dict \
or self.app_obj.current_prog_started_flag:
self.switch_prog_menu_item.set_sensitive(False)
self.delete_prog_menu_item.set_sensitive(False)
else:
self.switch_prog_menu_item.set_sensitive(True)
self.delete_prog_menu_item.set_sensitive(True)
def update_buttons_on_start(self):
"""Called by mainapp.GymBobApp.on_button_start().
(De)sensitises buttons after the user clicks the START button.
"""
self.start_button.set_sensitive(False)
self.stop_button.set_sensitive(True)
self.reset_button.set_sensitive(False)
def update_buttons_on_stop(self):
"""Called by mainapp.GymBobApp.on_button_stop().
(De)sensitises buttons after the user clicks the STOP button.
"""
self.start_button.set_sensitive(True)
self.stop_button.set_sensitive(False)
self.reset_button.set_sensitive(True)
def update_buttons_on_reset(self):
"""Called by mainapp.GymBobApp.on_button_reset().
(De)sensitises buttons after the user clicks the RESET button.
"""
self.start_button.set_sensitive(True)
self.stop_button.set_sensitive(False)
self.reset_button.set_sensitive(True)
def update_buttons_on_current_prog(self):
"""Called by various functions.
(De)sensitises the START, STOP and RESET buttons, depending on whether
there is a current workout programme, or not.
"""
if not self.app_obj.current_prog_obj:
self.start_button.set_sensitive(False)
self.stop_button.set_sensitive(False)
self.reset_button.set_sensitive(False)
else:
self.start_button.set_sensitive(True)
self.stop_button.set_sensitive(True)
self.reset_button.set_sensitive(True)
# (Support functions)
def convert_time_to_string(self, time):
"""Called by various functions.
Converts a time value (an integer in seconds) into a formatted string,
e.g. '1:27:02'.
Args:
time (int): A value in seconds (0 or above)
Return values:
The converted string
"""
minutes = int(time / 60)
seconds = int(time % 60)
hours = int(minutes / 60)
minutes = int(minutes % 60)
time_str = str(minutes).zfill(2) + ':' + str(seconds).zfill(2)
if hours > 0:
time_str = str(hours) + ':' + time_str
return time_str
def add_child_window(self, edit_win_obj):
"""Called by editwin.ProgEditWin.setup().
When an edit window opens, add it to our list of such windows. (A
workout programme will not start while the window(s) are open.)
Args:
edit_win_obj (edit.ProgEditWin): The window to add
"""
# Check that the window isn't already in the list (unlikely, but check
# anyway)
if not edit_win_obj in self.edit_win_list:
# Update the IV
self.edit_win_list.append(edit_win_obj)
def del_child_window(self, edit_win_obj):
"""Called by editwin.ProgEditWin.close().
When an edit window closes, remove it to our list of such windows.
Args:
edit_win_obj (edit.ProgEditWin): The window to remove
"""
# Update the IV
if edit_win_obj in self.edit_win_list:
self.edit_win_list.remove(edit_win_obj)
# Callbacks
def on_menu_mute_sound(self, checkbutton):
"""Called from a callback in self.setup_menubar().
Mutes (or unmutes) sound effects.
Args:
checkbutton (Gtk.CheckButton): The widget clicked.
"""
self.app_obj.set_mute_sound_flag(checkbutton.get_active())
class DeleteProgDialogue(Gtk.Dialog):
"""Called by mainapp.GymBobApp.on_menu_switch_prog().
Python class handling a dialogue window that prompts the user to delete a
workout programme.
Args:
main_win_obj (mainwin.MainWin): The parent main window
"""
def __init__(self, main_win_obj):
# IV list - Gtk widgets
# ---------------------
self.combo = None # Gtk.ComboBox
# IV list - other
# ---------------
self.prog_name = None
self.prog_list = []
# Code
# ----
Gtk.Dialog.__init__(
self,
'Delete programme',
main_win_obj,
Gtk.DialogFlags.DESTROY_WITH_PARENT,
(
Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL,
Gtk.STOCK_OK, Gtk.ResponseType.OK,
)
)
self.set_modal(True)
# Set up the dialogue window
box = self.get_content_area()
grid = Gtk.Grid()
box.add(grid)
grid.set_border_width(main_win_obj.app_obj.default_spacing_size)
grid.set_row_spacing(main_win_obj.app_obj.default_spacing_size)
label = Gtk.Label('Select the programme to delete')
grid.attach(label, 0, 0, 1, 1)
# Import and display a sorted list of workout programmes
self.prog_list = list(main_win_obj.app_obj.prog_dict.keys())
self.prog_list.sort()
self.prog_name = self.prog_list[0]
listmodel = Gtk.ListStore(str)
for item in self.prog_list:
listmodel.append([item])
self.combo = Gtk.ComboBox.new_with_model(listmodel)
grid.attach(self.combo, 0, 1, 1, 1)
self.combo.set_hexpand(True)
cell = Gtk.CellRendererText()
self.combo.pack_start(cell, False)
self.combo.add_attribute(cell, 'text', 0)
self.combo.set_active(0)
self.combo.connect('changed', self.on_combo_changed)
# Display the dialogue window
self.show_all()
def on_combo_changed(self, combo):
"""Called from callback in self.__init__().
Store the combobox's selected item, so the calling function can
retrieve it.
Args:
combo (Gtk.ComboBox): The clicked widget
"""
self.prog_name = self.prog_list[combo.get_active()]
class NewProgDialogue(Gtk.Dialog):
"""Called by mainapp.GymBobApp.on_menu_new_prog().
Python class handling a dialogue window that prompts the user for the name
of a new workout programme.
Args:
main_win_obj (mainwin.MainWin): The parent main window
"""
def __init__(self, main_win_obj):
# IV list - Gtk widgets
# ---------------------
self.entry = None # Gtk.Entry
# Code
# ----
Gtk.Dialog.__init__(
self,
'New programme',
main_win_obj,
Gtk.DialogFlags.DESTROY_WITH_PARENT,
(
Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL,
Gtk.STOCK_OK, Gtk.ResponseType.OK,
)
)
self.set_modal(True)
# Set up the dialogue window
box = self.get_content_area()
grid = Gtk.Grid()
box.add(grid)
grid.set_border_width(main_win_obj.app_obj.default_spacing_size)
grid.set_row_spacing(main_win_obj.app_obj.default_spacing_size)
label = Gtk.Label('Enter the name of a new workout programme')
grid.attach(label, 0, 0, 1, 1)
self.entry = Gtk.Entry()
grid.attach(self.entry, 0, 1, 1, 1)
self.entry.set_hexpand(True)
self.entry.set_max_length(16)
# Display the dialogue window
self.show_all()
class SwitchProgDialogue(Gtk.Dialog):
"""Called by mainapp.GymBobApp.on_menu_switch_prog().
Python class handling a dialogue window that prompts the user to switch to
a new workout programme.
Args:
main_win_obj (mainwin.MainWin): The parent main window
"""
def __init__(self, main_win_obj):
# IV list - Gtk widgets
# ---------------------
self.combo = None # Gtk.ComboBox
# IV list - other
# ---------------
self.prog_name = None
self.prog_list = []
# Code
# ----
Gtk.Dialog.__init__(
self,
'Switch programme',
main_win_obj,
Gtk.DialogFlags.DESTROY_WITH_PARENT,
(
Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL,
Gtk.STOCK_OK, Gtk.ResponseType.OK,
)
)
self.set_modal(True)
# Set up the dialogue window
box = self.get_content_area()
grid = Gtk.Grid()
box.add(grid)
grid.set_border_width(main_win_obj.app_obj.default_spacing_size)
grid.set_row_spacing(main_win_obj.app_obj.default_spacing_size)
label = Gtk.Label('Set the new workout programme')
grid.attach(label, 0, 0, 1, 1)
# Import a sorted list of programmes. The current programme should be
# the first item in the list
import_list = list(main_win_obj.app_obj.prog_dict.keys())
sorted_list = []
self.prog_name = main_win_obj.app_obj.current_prog_obj.name
for item in import_list:
if item != self.prog_name:
self.prog_list.append(item)
self.prog_list.sort()
self.prog_list.insert(0, self.prog_name)
listmodel = Gtk.ListStore(str)
for item in self.prog_list:
listmodel.append([item])
self.combo = Gtk.ComboBox.new_with_model(listmodel)
grid.attach(self.combo, 0, 1, 1, 1)
self.combo.set_hexpand(True)
cell = Gtk.CellRendererText()
self.combo.pack_start(cell, False)
self.combo.add_attribute(cell, 'text', 0)
self.combo.set_active(0)
self.combo.connect('changed', self.on_combo_changed)
# Display the dialogue window
self.show_all()
def on_combo_changed(self, combo):
"""Called from callback in self.__init__().
Store the combobox's selected item, so the calling function can
retrieve it.
Args:
combo (Gtk.ComboBox): The clicked widget
"""
self.prog_name = self.prog_list[combo.get_active()]

View File

@ -1,39 +1,9 @@
COPYING
All files in the ../dialogue sub-directory
All files in the ../status sub-directory
All files in the ../win sub-directory
All files in ../win sub-directory
Author: Vectorgraphit
Source: https://www.iconfinder.com/icons/199499/
Author: bekeen studio
Source: https://www.iconfinder.com/bekeenstudio
Author: Carlo Rodriguez
Source: https://www.iconfinder.com/icons/512534/exercise_fitness_gym_gymnasium_icon
This work is licensed under the Creative Commons Attribution 3.0 Unported
License. To view a copy of this license, visit
http://creativecommons.org/licenses/by/3.0/ or send a letter to Creative
Commons, 444 Castro Street, Suite 900, Mountain View, California, 94041, USA.
All files in the ../large sub-directory
All files in the ../msg sub-directory
All files in the ../small sub-directory
All files in the ../toolbar sub-directory
Author: FatCow Web Hosting
Source: https://www.fatcow.com/free-icons
This work is licensed under the Creative Commons Attribution 3.0 Generic
License. To view a copy of this license, visit
http://creativecommons.org/licenses/by/3.0/ or send a letter to Creative
Commons, 444 Castro Street, Suite 900, Mountain View, California, 94041, USA.
All files in the ../locale directory
Author: Mr. Hopnguyen
Source: https://www.iconfinder.com/icons/2634450/
This work is licensed under the Creative Commons Attribution 3.0 Unported
License. To view a copy of this license, visit
http://creativecommons.org/licenses/by/3.0/ or send a letter to Creative
Commons, 444 Castro Street, Suite 900, Mountain View, California, 94041, USA.
"Free for commercial use"

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 958 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 381 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 391 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 780 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 574 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 543 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 632 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 626 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 541 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 765 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 621 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 7.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 7.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 780 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 574 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 632 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 626 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 727 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.0 KiB

Some files were not shown because too many files have changed in this diff Show More