Skip to main content

Python module for playing and recording videos with sound inside tkinter Label widget using Pillow, imageio, and PyAudio, including media playback control, slider, and fps aware buffering.

Project description

tkVideoUtils - Python module for playing videos and camera feeds in a Tkinter label, with recording functionality

Contributors Forks Stargazers Issues MIT License

About The Project

tkVideoUtils is a Python module for playing and recording videos in GUIs created with tkinter. Using imageio-ffmpeg, webcams can be indexed and streamed in a tkinter Label. They can also be recorded (in their original resolution) by calling the start_recording() function! This project was heavily inspired by huskeee's tkvideo library, check their project out!

Built With

Installation

End-users:

  • Clone the repo and run setup.py
git clone https://github.com/wsarce/tkVideoUtils.git
python ./tkvideo/setup.py

or

  • Install the package from PyPI
pip install tkVideoUtils

Developers and contributors

  • Clone the repo and install the module in developer mode
git clone https://github.com/wsarce/tkVideoUtils.git
python ./tkvideo/setup.py develop

or

  • Install the package from PyPI in editable mode
pip install -e tkvideoplayer

This will create a shim between your code and the module binaries that gets updated every time you change your code.

Usage

Playing a Video in a Label

from tkinter import *

from ttkwidgets import TickScale

from tkvideoutils import VideoPlayer
from tkinter import filedialog, messagebox


def on_closing():
    player.loading = False
    root.quit()
    root.destroy()


if __name__ == '__main__':
    # create instance of window
    root = Tk()
    # set window title
    root.title('Video Player')
    # load images
    pause_image = PhotoImage(file='pause.png')
    play_image = PhotoImage(file='play.png')
    skip_backward = PhotoImage(file='skip_backward.png')
    skip_forward = PhotoImage(file='skip_forward.png')
    # create user interface
    button = Button(root, image=play_image)
    forward_button = Button(root, image=skip_forward)
    backward_button = Button(root, image=skip_backward)
    video_label = Label(root)
    video_path = 'recorded_video.mp4'
    audio_path = 'recorded_video.wav'
    slider_var = IntVar(root)
    slider = TickScale(root, orient="horizontal", variable=slider_var)
    # place elements
    video_label.pack()
    button.pack()
    forward_button.pack()
    backward_button.pack()
    slider.pack()

    if video_path:
        # read video to display on label
        player = VideoPlayer(root, video_path, audio_path, video_label, size=(700, 500),
                             play_button=button, play_image=play_image, pause_image=pause_image,
                             slider=slider, slider_var=slider_var, keep_ratio=True, cleanup_audio=True)
    else:
        messagebox.showwarning("Select Video File", "Please retry and select a video file.")
        sys.exit(1)
    player.set_clip(50, 70)
    forward_button.config(command=player.skip_video_forward)
    backward_button.config(command=player.skip_video_backward)
    root.protocol("WM_DELETE_WINDOW", on_closing)
    root.mainloop()

Playing and Recording a Webcam in a Label

from tkinter import *
from tkvideoutils import VideoRecorder
from tkinter import messagebox


def merge_sources():
    if player.merge_sources(merged_path, ffmpeg_exe):
        print("Sources merged!")
    else:
        print("Something went wrong")


def stop_recording():
    player.stop_recording()
    player.stop_playback()
    button['command'] = merge_sources


def start_recording():
    player.start_recording()
    button['command'] = stop_recording


if __name__ == '__main__':
    # create instance of window
    root = Tk()
    # set window title
    root.title('Video Player')
    # load images
    pause_image = PhotoImage(file='pause.png')
    play_image = PhotoImage(file='play.png')
    # create user interface
    button = Button(root, image=play_image)
    video_label = Label(root)
    video_path = 'raw_video.mp4'
    audio_path = 'recorded_video.wav'
    merged_path = 'recorded_video.mp4'
    # place elements
    video_label.pack()
    button.pack()
    # Get existing video sources
    video_sources = VideoRecorder.get_video_sources()
    audio_sources = VideoRecorder.get_audio_sources()
    print(video_sources, audio_sources)
    # TODO: Fill out FFMPEG path
    ffmpeg_exe = r''

    if video_sources:
        if video_path:
            # read video to display on label
            player = VideoRecorder(video_source=video_sources[0], audio_source=audio_sources[1],
                                   video_path=video_path, audio_path=audio_path, fps=8, label=video_label,
                                   size=(700, 500))
            player.start_playback()
        else:
            messagebox.showwarning("Select Video File", "Please retry and select a video file.")
            sys.exit(1)
        button.config(command=start_recording)
        root.mainloop()
    else:
        print("No video sources found!")

Issues / Suggestions

Have a problem that needs to be solved or a suggestion to make? See the issues page.

Notes on Freezing

If you want to freeze your Python scripts that reference tkVideoUtils, there are some general notes to keep in mind:

  • Due to the use of ImageIO, the lazy import error causes frozen Python scripts to fail to start or fail to execute the imageio codepaths. This can make it look like imageio-ffmpeg is failing with a backend issue. To circumvent this, follow this answer. This answer says to overwrite your imageio folder in your exe output directory with the imageio folder from your virtual environment (venv\Lib\site-packages\imageio). I can verify that this works.
  • I prefer copying the ffmpeg exe version that is packaged with imageio-ffmpeg into your exe directory in a known folder and updating os.environ['IMAGEIO_FFMPEG_EXE'] with that directory. This ensures a known path.

License

Distributed under the MIT License. See LICENSE for more information.

Contact

Walker Arce - wsarcera@gmail.com Project Link: https://github.com/wsarce/tkVideoUtils

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

tkVideoUtils-1.0.2.tar.gz (11.7 kB view details)

Uploaded Source

Built Distributions

tkVideoUtils-1.0.2-py3.8.egg (22.7 kB view details)

Uploaded Source

tkVideoUtils-1.0.2-py3-none-any.whl (12.1 kB view details)

Uploaded Python 3

File details

Details for the file tkVideoUtils-1.0.2.tar.gz.

File metadata

  • Download URL: tkVideoUtils-1.0.2.tar.gz
  • Upload date:
  • Size: 11.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.8.0 pkginfo/1.8.2 readme-renderer/32.0 requests/2.27.1 requests-toolbelt/0.9.1 urllib3/1.26.8 tqdm/4.62.3 importlib-metadata/4.11.1 keyring/23.5.0 rfc3986/2.0.0 colorama/0.4.4 CPython/3.8.10

File hashes

Hashes for tkVideoUtils-1.0.2.tar.gz
Algorithm Hash digest
SHA256 412c985e28b851a4e2b337604d759fe41b1802e9be2b60ca9edb8c23360c3e84
MD5 e072dbc3c0a26fa8a5d98dbdee0ba1ef
BLAKE2b-256 8b37cb2526eec993f9d902ed19ca9f7c4988cd8c36f5255bcf626c29210c78aa

See more details on using hashes here.

File details

Details for the file tkVideoUtils-1.0.2-py3.8.egg.

File metadata

  • Download URL: tkVideoUtils-1.0.2-py3.8.egg
  • Upload date:
  • Size: 22.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.8.0 pkginfo/1.8.2 readme-renderer/32.0 requests/2.27.1 requests-toolbelt/0.9.1 urllib3/1.26.8 tqdm/4.62.3 importlib-metadata/4.11.1 keyring/23.5.0 rfc3986/2.0.0 colorama/0.4.4 CPython/3.8.10

File hashes

Hashes for tkVideoUtils-1.0.2-py3.8.egg
Algorithm Hash digest
SHA256 964cb623ef111abf2ce4747f0adea7ba2fb5f2f48e8907a19b4b5cd377951cb0
MD5 9583b8dc4350a099a38e1fd44dc08598
BLAKE2b-256 117e9c19ae7e48c86846047bee1f31a2d3f684b98ce81840dbae6cdd179b07dc

See more details on using hashes here.

File details

Details for the file tkVideoUtils-1.0.2-py3-none-any.whl.

File metadata

  • Download URL: tkVideoUtils-1.0.2-py3-none-any.whl
  • Upload date:
  • Size: 12.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.8.0 pkginfo/1.8.2 readme-renderer/32.0 requests/2.27.1 requests-toolbelt/0.9.1 urllib3/1.26.8 tqdm/4.62.3 importlib-metadata/4.11.1 keyring/23.5.0 rfc3986/2.0.0 colorama/0.4.4 CPython/3.8.10

File hashes

Hashes for tkVideoUtils-1.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 f0dd01dc901d1cd7f1f0fa69d72f599341d60113aa6c4bece4e84e7c18a395c9
MD5 82d761f515c88c6b2d254b4b41aeac18
BLAKE2b-256 37643557d4ddbb36116f32ddd331e5cc6c7a94be92afc444c180683bdca91966

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page