mirror of
https://github.com/iluvcapra/pycmx.git
synced 2025-12-31 17:00:53 +00:00
Compare commits
101 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ce1a0d32bb | ||
|
|
daa5d58a66 | ||
|
|
38ce1445a1 | ||
|
|
552d007360 | ||
|
|
d79fdcc6a8 | ||
|
|
3187a50a6b | ||
|
|
8e8d4f5753 | ||
|
|
cc371cd486 | ||
|
|
795a666a74 | ||
|
|
cd35f4f80d | ||
|
|
6525840151 | ||
|
|
b78ae05d8c | ||
|
|
51ed92f5df | ||
|
|
a6f042c76f | ||
|
|
179808fbf2 | ||
|
|
2b38d8aaf9 | ||
|
|
e19bb3c5c5 | ||
|
|
d0e2e5bbe5 | ||
|
|
e90897e503 | ||
|
|
4daa008e91 | ||
|
|
cf1d71800e | ||
|
|
fcb25ae183 | ||
|
|
97ada84bfe | ||
|
|
d157512c32 | ||
|
|
1e2d31716e | ||
|
|
6be7c54de5 | ||
|
|
a3109cdb7a | ||
|
|
3a9d81417e | ||
|
|
9838e6b357 | ||
|
|
41fdeeaf56 | ||
|
|
b3b6e57f6c | ||
|
|
62e7f10cf4 | ||
|
|
f855d3581d | ||
|
|
0b5555d333 | ||
|
|
7123a6f1bb | ||
|
|
173493a610 | ||
|
|
d20d9d1bdd | ||
|
|
d328e12283 | ||
|
|
4d83f81fc8 | ||
|
|
69b6c7236d | ||
|
|
71ffe8cd0d | ||
|
|
3fff6c8d2a | ||
|
|
e350565430 | ||
|
|
9ec30ede02 | ||
|
|
e5130b8011 | ||
|
|
00eaccabac | ||
|
|
521d86e444 | ||
|
|
0fdba2408b | ||
|
|
199bba2466 | ||
|
|
d8f0b5694e | ||
|
|
742b0f96c3 | ||
|
|
d44948bdc1 | ||
|
|
93cff15446 | ||
|
|
fc914409ce | ||
|
|
770e7f45a4 | ||
|
|
d3cf5fa5f2 | ||
|
|
67d2bd7093 | ||
|
|
26a7eae437 | ||
|
|
407aa1c1fd | ||
|
|
13d0a80a10 | ||
|
|
dcd2a22a43 | ||
|
|
c586740269 | ||
|
|
e28dbbbe5e | ||
|
|
41df450452 | ||
|
|
c37464036d | ||
|
|
9d89834eb3 | ||
|
|
8b53d2249c | ||
|
|
0cdbc4e9be | ||
|
|
e229e807b1 | ||
|
|
a7ee1f6737 | ||
|
|
9097de8efa | ||
|
|
bc6d7f34c0 | ||
|
|
b642f859f3 | ||
|
|
29a9a5fba7 | ||
|
|
20ff7d7ee8 | ||
|
|
183f121cfc | ||
|
|
e2dffcb745 | ||
|
|
8c2ba3cc09 | ||
|
|
c7569045c1 | ||
|
|
50bcac23bb | ||
|
|
67b1631ba9 | ||
|
|
85cbafba8f | ||
|
|
595cf35e57 | ||
|
|
7fa22d4b85 | ||
|
|
42f2de54b5 | ||
|
|
f7d1432014 | ||
|
|
db4eadb73e | ||
|
|
3305bc7920 | ||
|
|
6ba77b3568 | ||
|
|
c68f8bca80 | ||
|
|
284267c9c0 | ||
|
|
bd196f2dbf | ||
|
|
b14a9a6319 | ||
|
|
0cbd01f418 | ||
|
|
50d48708e9 | ||
|
|
f67c4ac2c5 | ||
|
|
1b8a3c3288 | ||
|
|
b37b57d7c9 | ||
|
|
a9937683e5 | ||
|
|
4fae65fa8d | ||
|
|
566e6257f4 |
41
.github/workflows/python-package.yml
vendored
Normal file
41
.github/workflows/python-package.yml
vendored
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
# This workflow will install Python dependencies, run tests and lint with a variety of Python versions
|
||||||
|
# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-python
|
||||||
|
|
||||||
|
name: Lint and Test
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [ "master" ]
|
||||||
|
pull_request:
|
||||||
|
branches: [ "master" ]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
python-version: ["3.7", "3.8", "3.9", "3.10", "3.11", "3.12"]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- name: Set up Python ${{ matrix.python-version }}
|
||||||
|
uses: actions/setup-python@v4.3.0
|
||||||
|
with:
|
||||||
|
python-version: ${{ matrix.python-version }}
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
python -m pip install flake8 pytest
|
||||||
|
python3 -m pip install -e .
|
||||||
|
# if [ -f requirements.txt ]; then pip install -r requirements.txt; fi
|
||||||
|
- name: Lint with flake8
|
||||||
|
run: |
|
||||||
|
# stop the build if there are Python syntax errors or undefined names
|
||||||
|
flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics
|
||||||
|
# exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide
|
||||||
|
flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics
|
||||||
|
- name: Test with pytest
|
||||||
|
run: |
|
||||||
|
pytest
|
||||||
40
.github/workflows/pythonpublish.yml
vendored
Normal file
40
.github/workflows/pythonpublish.yml
vendored
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
name: Upload Python Package
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
id-token: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
environment:
|
||||||
|
name: release
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3.5.2
|
||||||
|
- name: Set up Python
|
||||||
|
uses: actions/setup-python@v4.6.0
|
||||||
|
with:
|
||||||
|
python-version: '3.x'
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
pip install build
|
||||||
|
- name: Build package
|
||||||
|
run: python -m build
|
||||||
|
- name: pypi-publish
|
||||||
|
uses: pypa/gh-action-pypi-publish@v1.8.6
|
||||||
|
- name: Report to Mastodon
|
||||||
|
uses: cbrgm/mastodon-github-action@v1.0.1
|
||||||
|
with:
|
||||||
|
message: |
|
||||||
|
I just released a new version of pycmx, my library for reading CMX EDLs!
|
||||||
|
#sounddesign #filmmaking #python
|
||||||
|
${{ github.server_url }}/${{ github.repository }}
|
||||||
|
env:
|
||||||
|
MASTODON_URL: ${{ secrets.MASTODON_URL }}
|
||||||
|
MASTODON_ACCESS_TOKEN: ${{ secrets.MASTODON_ACCESS_TOKEN }}
|
||||||
10
.gitignore
vendored
10
.gitignore
vendored
@@ -10,3 +10,13 @@
|
|||||||
|
|
||||||
# Vim Swapfiles
|
# Vim Swapfiles
|
||||||
*.swp
|
*.swp
|
||||||
|
.DS_Store
|
||||||
|
|
||||||
|
venv/
|
||||||
|
|
||||||
|
.coverage
|
||||||
|
lcov.info
|
||||||
|
|
||||||
|
|
||||||
|
# venv
|
||||||
|
venv/
|
||||||
|
|||||||
32
.readthedocs.yaml
Normal file
32
.readthedocs.yaml
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
# .readthedocs.yaml
|
||||||
|
# Read the Docs configuration file
|
||||||
|
# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details
|
||||||
|
|
||||||
|
# Required
|
||||||
|
version: 2
|
||||||
|
|
||||||
|
# Set the version of Python and other tools you might need
|
||||||
|
build:
|
||||||
|
os: ubuntu-20.04
|
||||||
|
tools:
|
||||||
|
python: "3.10"
|
||||||
|
# You can also specify other tool versions:
|
||||||
|
# nodejs: "16"
|
||||||
|
# rust: "1.55"
|
||||||
|
# golang: "1.17"
|
||||||
|
|
||||||
|
# Build documentation in the docs/ directory with Sphinx
|
||||||
|
sphinx:
|
||||||
|
configuration: docs/source/conf.py
|
||||||
|
|
||||||
|
#If using Sphinx, optionally build your docs in additional formats such as PDF
|
||||||
|
formats:
|
||||||
|
- pdf
|
||||||
|
|
||||||
|
#Optionally declare the Python requirements required to build your docs
|
||||||
|
python:
|
||||||
|
install:
|
||||||
|
- method: pip
|
||||||
|
path: .
|
||||||
|
extra_requirements:
|
||||||
|
- doc
|
||||||
17
.travis.yml
17
.travis.yml
@@ -1,17 +0,0 @@
|
|||||||
language: python
|
|
||||||
python:
|
|
||||||
- "3.7"
|
|
||||||
- "3.6"
|
|
||||||
- "3.5"
|
|
||||||
- "2.7"
|
|
||||||
script:
|
|
||||||
- "python setup.py test"
|
|
||||||
- "py.test tests/ -v --cov pycmx --cov-report term-missing"
|
|
||||||
install:
|
|
||||||
- "pip install setuptools"
|
|
||||||
before_install:
|
|
||||||
- "pip install coverage"
|
|
||||||
- "pip install codecov"
|
|
||||||
- "pip install pytest-cov==2.5.0"
|
|
||||||
after_success:
|
|
||||||
- "codecov"
|
|
||||||
13
CONTRIBUTING.md
Normal file
13
CONTRIBUTING.md
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Contributing
|
||||||
|
|
||||||
|
Contributions to this project are welcome!
|
||||||
|
|
||||||
|
The best way to contribute code to this project is to find this project on [Github][github] and submit a pull request.
|
||||||
|
|
||||||
|
## Call for EDLs
|
||||||
|
|
||||||
|
If you have EDLs you are having trouble working with becuase of unusual formatting, please send me a copy! Contact us
|
||||||
|
through [Github].
|
||||||
|
|
||||||
|
|
||||||
|
[github]: https://github.com/iluvcapra/pycmx
|
||||||
2
LICENSE
2
LICENSE
@@ -1,4 +1,4 @@
|
|||||||
Copyright (c) 2018 Jamie Hardt.
|
Copyright (c) 2022 Jamie Hardt.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[](https://travis-ci.com/iluvcapra/pycmx)
|
|
||||||
[](https://codecov.io/gh/iluvcapra/pycmx)
|
|
||||||
[](https://pycmx.readthedocs.io/en/latest/?badge=latest)   [](https://pypi.org/project/pycmx/) 
|
[](https://pycmx.readthedocs.io/en/latest/?badge=latest)   [](https://pypi.org/project/pycmx/) 
|
||||||
|

|
||||||
|
[](https://github.com/iluvcapra/pycmx/actions/workflows/python-package.yml)
|
||||||
|
|
||||||
|
|
||||||
# pycmx
|
# pycmx
|
||||||
@@ -83,3 +83,5 @@ Audio channel 7 is present
|
|||||||
>>> events[2].edits[0].channels.video
|
>>> events[2].edits[0].channels.video
|
||||||
False
|
False
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
BIN
docs/CMX3600.pdf
BIN
docs/CMX3600.pdf
Binary file not shown.
@@ -1,35 +0,0 @@
|
|||||||
@ECHO OFF
|
|
||||||
|
|
||||||
pushd %~dp0
|
|
||||||
|
|
||||||
REM Command file for Sphinx documentation
|
|
||||||
|
|
||||||
if "%SPHINXBUILD%" == "" (
|
|
||||||
set SPHINXBUILD=sphinx-build
|
|
||||||
)
|
|
||||||
set SOURCEDIR=source
|
|
||||||
set BUILDDIR=build
|
|
||||||
|
|
||||||
if "%1" == "" goto help
|
|
||||||
|
|
||||||
%SPHINXBUILD% >NUL 2>NUL
|
|
||||||
if errorlevel 9009 (
|
|
||||||
echo.
|
|
||||||
echo.The 'sphinx-build' command was not found. Make sure you have Sphinx
|
|
||||||
echo.installed, then set the SPHINXBUILD environment variable to point
|
|
||||||
echo.to the full path of the 'sphinx-build' executable. Alternatively you
|
|
||||||
echo.may add the Sphinx directory to PATH.
|
|
||||||
echo.
|
|
||||||
echo.If you don't have Sphinx installed, grab it from
|
|
||||||
echo.http://sphinx-doc.org/
|
|
||||||
exit /b 1
|
|
||||||
)
|
|
||||||
|
|
||||||
%SPHINXBUILD% -M %1 %SOURCEDIR% %BUILDDIR% %SPHINXOPTS%
|
|
||||||
goto end
|
|
||||||
|
|
||||||
:help
|
|
||||||
%SPHINXBUILD% -M help %SOURCEDIR% %BUILDDIR% %SPHINXOPTS%
|
|
||||||
|
|
||||||
:end
|
|
||||||
popd
|
|
||||||
24
docs/source/classes.rst
Normal file
24
docs/source/classes.rst
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
pycmx Classes
|
||||||
|
=============
|
||||||
|
|
||||||
|
.. autoclass:: pycmx.edit_list.EditList
|
||||||
|
:members:
|
||||||
|
|
||||||
|
|
||||||
|
.. autoclass:: pycmx.event.Event
|
||||||
|
:members:
|
||||||
|
|
||||||
|
|
||||||
|
.. autoclass:: pycmx.edit.Edit
|
||||||
|
:members:
|
||||||
|
|
||||||
|
|
||||||
|
.. autoclass:: pycmx.transition.Transition
|
||||||
|
:members:
|
||||||
|
|
||||||
|
|
||||||
|
.. autoclass:: pycmx.channel_map.ChannelMap
|
||||||
|
:members:
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -14,19 +14,20 @@
|
|||||||
#
|
#
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
sys.path.insert(0, os.path.abspath('../../pycmx'))
|
sys.path.insert(0, os.path.abspath('../..'))
|
||||||
|
|
||||||
|
import pycmx
|
||||||
|
|
||||||
# -- Project information -----------------------------------------------------
|
# -- Project information -----------------------------------------------------
|
||||||
|
|
||||||
project = u'pycmx'
|
project = u'pycmx'
|
||||||
copyright = u'2018, Jamie Hardt'
|
copyright = u'(c) 2023, Jamie Hardt'
|
||||||
author = u'Jamie Hardt'
|
author = u'Jamie Hardt'
|
||||||
|
|
||||||
# The short X.Y version
|
# The short X.Y version
|
||||||
version = u''
|
version = pycmx.__version__
|
||||||
# The full version, including alpha/beta/rc tags
|
# The full version, including alpha/beta/rc tags
|
||||||
release = u''
|
release = pycmx.__version__
|
||||||
|
|
||||||
|
|
||||||
# -- General configuration ---------------------------------------------------
|
# -- General configuration ---------------------------------------------------
|
||||||
@@ -63,7 +64,7 @@ master_doc = 'index'
|
|||||||
#
|
#
|
||||||
# This is also used if you do content translation via gettext catalogs.
|
# This is also used if you do content translation via gettext catalogs.
|
||||||
# Usually you set "language" from the command line for these cases.
|
# Usually you set "language" from the command line for these cases.
|
||||||
language = None
|
language = 'em'
|
||||||
|
|
||||||
# List of patterns, relative to source directory, that match files and
|
# List of patterns, relative to source directory, that match files and
|
||||||
# directories to ignore when looking for source files.
|
# directories to ignore when looking for source files.
|
||||||
@@ -144,7 +145,7 @@ latex_documents = [
|
|||||||
# (source start file, name, description, authors, manual section).
|
# (source start file, name, description, authors, manual section).
|
||||||
man_pages = [
|
man_pages = [
|
||||||
(master_doc, 'pycmx', u'pycmx Documentation',
|
(master_doc, 'pycmx', u'pycmx Documentation',
|
||||||
[author], 1)
|
[author], "3p")
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
7
docs/source/function.rst
Normal file
7
docs/source/function.rst
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
Parse Function
|
||||||
|
==============
|
||||||
|
|
||||||
|
|
||||||
|
.. autofunction:: pycmx.parse_cmx_events.parse_cmx3600
|
||||||
|
|
||||||
|
|
||||||
@@ -8,13 +8,13 @@ Welcome to pycmx's documentation!
|
|||||||
|
|
||||||
.. toctree::
|
.. toctree::
|
||||||
:maxdepth: 5
|
:maxdepth: 5
|
||||||
:caption: API Reference:
|
:caption: API Reference
|
||||||
|
|
||||||
pycmx
|
function
|
||||||
|
classes
|
||||||
|
|
||||||
Indices and tables
|
Indices and tables
|
||||||
==================
|
==================
|
||||||
|
|
||||||
* :ref:`genindex`
|
* :ref:`genindex`
|
||||||
* :ref:`modindex`
|
|
||||||
* :ref:`search`
|
* :ref:`search`
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
pycmx
|
|
||||||
=====
|
|
||||||
|
|
||||||
.. toctree::
|
|
||||||
:maxdepth: 4
|
|
||||||
|
|
||||||
pycmx
|
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
pycmx package
|
|
||||||
=============
|
|
||||||
|
|
||||||
Submodules
|
|
||||||
----------
|
|
||||||
|
|
||||||
pycmx.channel\_map module
|
|
||||||
-------------------------
|
|
||||||
|
|
||||||
.. automodule:: pycmx.channel_map
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
:show-inheritance:
|
|
||||||
|
|
||||||
pycmx.parse\_cmx\_events module
|
|
||||||
-------------------------------
|
|
||||||
|
|
||||||
.. automodule:: pycmx.parse_cmx_events
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
:show-inheritance:
|
|
||||||
|
|
||||||
pycmx.parse\_cmx\_statements module
|
|
||||||
-----------------------------------
|
|
||||||
|
|
||||||
.. automodule:: pycmx.parse_cmx_statements
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
:show-inheritance:
|
|
||||||
|
|
||||||
pycmx.util module
|
|
||||||
-----------------
|
|
||||||
|
|
||||||
.. automodule:: pycmx.util
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
:show-inheritance:
|
|
||||||
|
|
||||||
|
|
||||||
Module contents
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: pycmx
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
:show-inheritance:
|
|
||||||
@@ -1,15 +1,13 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
"""
|
"""
|
||||||
pycmx is a module for parsing CMX 3600-style EDLs. For more information and
|
pycmx is a parser for CMX 3600-style EDLs.
|
||||||
examples see README.md
|
|
||||||
|
|
||||||
This module (c) 2018 Jamie Hardt. For more information on your rights to
|
This module (c) 2023 Jamie Hardt. For more information on your rights to
|
||||||
copy and reuse this software, refer to the LICENSE file included with the
|
copy and reuse this software, refer to the LICENSE file included with the
|
||||||
distribution.
|
distribution.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
__version__ = '1.0'
|
__version__ = '1.2.2'
|
||||||
__author__ = 'Jamie Hardt'
|
|
||||||
|
|
||||||
from .parse_cmx_events import parse_cmx3600
|
from .parse_cmx_events import parse_cmx3600
|
||||||
from .transition import Transition
|
from .transition import Transition
|
||||||
|
|||||||
@@ -2,13 +2,14 @@
|
|||||||
# (c) 2018 Jamie Hardt
|
# (c) 2018 Jamie Hardt
|
||||||
|
|
||||||
from re import (compile, match)
|
from re import (compile, match)
|
||||||
|
from typing import Dict, Tuple, Generator
|
||||||
|
|
||||||
class ChannelMap:
|
class ChannelMap:
|
||||||
"""
|
"""
|
||||||
Represents a set of all the channels to which an event applies.
|
Represents a set of all the channels to which an event applies.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
_chan_map = {
|
_chan_map : Dict[str, Tuple] = {
|
||||||
"V" : (True, False, False),
|
"V" : (True, False, False),
|
||||||
"A" : (False, True, False),
|
"A" : (False, True, False),
|
||||||
"A2" : (False, False, True),
|
"A2" : (False, False, True),
|
||||||
@@ -23,65 +24,70 @@ class ChannelMap:
|
|||||||
self.v = v
|
self.v = v
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def video(self):
|
def video(self) -> bool:
|
||||||
'True if video is included'
|
'True if video is included'
|
||||||
return self.v
|
return self.v
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def channels(self):
|
def audio(self) -> bool:
|
||||||
|
'True if an audio channel is included'
|
||||||
|
return len(self._audio_channel_set) > 0
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channels(self) -> Generator[int, None, None]:
|
||||||
'A generator for each audio channel'
|
'A generator for each audio channel'
|
||||||
for c in self._audio_channel_set:
|
for c in self._audio_channel_set:
|
||||||
yield c
|
yield c
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def a1(self):
|
def a1(self) -> bool:
|
||||||
"""True if A1 is included."""
|
"""True if A1 is included"""
|
||||||
return self.get_audio_channel(1)
|
return self.get_audio_channel(1)
|
||||||
|
|
||||||
@a1.setter
|
@a1.setter
|
||||||
def a1(self,val):
|
def a1(self, val: bool):
|
||||||
self.set_audio_channel(1,val)
|
self.set_audio_channel(1,val)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def a2(self):
|
def a2(self) -> bool:
|
||||||
"""True if A2 is included."""
|
"""True if A2 is included"""
|
||||||
return self.get_audio_channel(2)
|
return self.get_audio_channel(2)
|
||||||
|
|
||||||
@a2.setter
|
@a2.setter
|
||||||
def a2(self,val):
|
def a2(self, val: bool):
|
||||||
self.set_audio_channel(2,val)
|
self.set_audio_channel(2,val)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def a3(self):
|
def a3(self) -> bool:
|
||||||
"""True if A3 is included."""
|
"""True if A3 is included"""
|
||||||
return self.get_audio_channel(3)
|
return self.get_audio_channel(3)
|
||||||
|
|
||||||
@a3.setter
|
@a3.setter
|
||||||
def a3(self,val):
|
def a3(self, val: bool):
|
||||||
self.set_audio_channel(3,val)
|
self.set_audio_channel(3,val)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def a4(self):
|
def a4(self) -> bool:
|
||||||
"""True if A4 is included."""
|
"""True if A4 is included"""
|
||||||
return self.get_audio_channel(4)
|
return self.get_audio_channel(4)
|
||||||
|
|
||||||
@a4.setter
|
@a4.setter
|
||||||
def a4(self,val):
|
def a4(self,val: bool):
|
||||||
self.set_audio_channel(4,val)
|
self.set_audio_channel(4,val)
|
||||||
|
|
||||||
def get_audio_channel(self,chan_num):
|
def get_audio_channel(self, chan_num) -> bool:
|
||||||
"""True if chan_num is included."""
|
"""True if chan_num is included"""
|
||||||
return (chan_num in self._audio_channel_set)
|
return (chan_num in self._audio_channel_set)
|
||||||
|
|
||||||
def set_audio_channel(self,chan_num,enabled):
|
def set_audio_channel(self,chan_num, enabled: bool):
|
||||||
"""If enabled is true, chan_num will be included."""
|
"""If enabled is true, chan_num will be included"""
|
||||||
if enabled:
|
if enabled:
|
||||||
self._audio_channel_set.add(chan_num)
|
self._audio_channel_set.add(chan_num)
|
||||||
elif self.get_audio_channel(chan_num):
|
elif self.get_audio_channel(chan_num):
|
||||||
self._audio_channel_set.remove(chan_num)
|
self._audio_channel_set.remove(chan_num)
|
||||||
|
|
||||||
def _append_event(self, event_str):
|
def _append_event(self, event_str):
|
||||||
alt_channel_re = compile('^A(\d+)')
|
alt_channel_re = compile(r'^A(\d+)')
|
||||||
if event_str in self._chan_map:
|
if event_str in self._chan_map:
|
||||||
channels = self._chan_map[event_str]
|
channels = self._chan_map[event_str]
|
||||||
self.v = channels[0]
|
self.v = channels[0]
|
||||||
@@ -93,14 +99,15 @@ class ChannelMap:
|
|||||||
self.set_audio_channel(int( matchresult.group(1)), True )
|
self.set_audio_channel(int( matchresult.group(1)), True )
|
||||||
|
|
||||||
def _append_ext(self, audio_ext):
|
def _append_ext(self, audio_ext):
|
||||||
self.a3 = ext.audio3
|
self.a3 = audio_ext.audio3
|
||||||
self.a4 = ext.audio4
|
self.a4 = audio_ext.audio4
|
||||||
|
|
||||||
def __or__(self, other):
|
def __or__(self, other):
|
||||||
"""
|
"""
|
||||||
Return the logical union of this channel map with another
|
the logical union of this channel map with another
|
||||||
"""
|
"""
|
||||||
out_v = self.video | other.video
|
out_v = self.video | other.video
|
||||||
out_a = self._audio_channel_set | other._audio_channel_set
|
out_a = self._audio_channel_set | other._audio_channel_set
|
||||||
|
|
||||||
return ChannelMap(v=out_v,audio_channels = out_a)
|
return ChannelMap(v=out_v,audio_channels = out_a)
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,9 @@
|
|||||||
|
|
||||||
from .transition import Transition
|
from .transition import Transition
|
||||||
from .channel_map import ChannelMap
|
from .channel_map import ChannelMap
|
||||||
from .parse_cmx_statements import StmtEffectsName
|
# from .parse_cmx_statements import StmtEffectsName
|
||||||
|
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
class Edit:
|
class Edit:
|
||||||
"""
|
"""
|
||||||
@@ -18,7 +20,7 @@ class Edit:
|
|||||||
self.trans_name_statement = trans_name_statement
|
self.trans_name_statement = trans_name_statement
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def line_number(self):
|
def line_number(self) -> int:
|
||||||
"""
|
"""
|
||||||
Get the line number for the "standard form" statement associated with
|
Get the line number for the "standard form" statement associated with
|
||||||
this edit. Line numbers a zero-indexed, such that the
|
this edit. Line numbers a zero-indexed, such that the
|
||||||
@@ -27,7 +29,7 @@ class Edit:
|
|||||||
return self.edit_statement.line_number
|
return self.edit_statement.line_number
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def channels(self):
|
def channels(self) -> ChannelMap:
|
||||||
"""
|
"""
|
||||||
Get the :obj:`ChannelMap` object associated with this Edit.
|
Get the :obj:`ChannelMap` object associated with this Edit.
|
||||||
"""
|
"""
|
||||||
@@ -38,7 +40,7 @@ class Edit:
|
|||||||
return cm
|
return cm
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def transition(self):
|
def transition(self) -> Transition:
|
||||||
"""
|
"""
|
||||||
Get the :obj:`Transition` object associated with this edit.
|
Get the :obj:`Transition` object associated with this edit.
|
||||||
"""
|
"""
|
||||||
@@ -48,14 +50,14 @@ class Edit:
|
|||||||
return Transition(self.edit_statement.trans, self.edit_statement.trans_op, None)
|
return Transition(self.edit_statement.trans, self.edit_statement.trans_op, None)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def source_in(self):
|
def source_in(self) -> str:
|
||||||
"""
|
"""
|
||||||
Get the source in timecode.
|
Get the source in timecode.
|
||||||
"""
|
"""
|
||||||
return self.edit_statement.source_in
|
return self.edit_statement.source_in
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def source_out(self):
|
def source_out(self) -> str:
|
||||||
"""
|
"""
|
||||||
Get the source out timecode.
|
Get the source out timecode.
|
||||||
"""
|
"""
|
||||||
@@ -63,7 +65,7 @@ class Edit:
|
|||||||
return self.edit_statement.source_out
|
return self.edit_statement.source_out
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def record_in(self):
|
def record_in(self) -> str:
|
||||||
"""
|
"""
|
||||||
Get the record in timecode.
|
Get the record in timecode.
|
||||||
"""
|
"""
|
||||||
@@ -71,7 +73,7 @@ class Edit:
|
|||||||
return self.edit_statement.record_in
|
return self.edit_statement.record_in
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def record_out(self):
|
def record_out(self) -> str:
|
||||||
"""
|
"""
|
||||||
Get the record out timecode.
|
Get the record out timecode.
|
||||||
"""
|
"""
|
||||||
@@ -79,7 +81,7 @@ class Edit:
|
|||||||
return self.edit_statement.record_out
|
return self.edit_statement.record_out
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def source(self):
|
def source(self) -> str:
|
||||||
"""
|
"""
|
||||||
Get the source column. This is the 8, 32 or 128-character string on the
|
Get the source column. This is the 8, 32 or 128-character string on the
|
||||||
event record line, this usually references the tape name of the source.
|
event record line, this usually references the tape name of the source.
|
||||||
@@ -87,21 +89,21 @@ class Edit:
|
|||||||
return self.edit_statement.source
|
return self.edit_statement.source
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def black(self):
|
def black(self) -> bool:
|
||||||
"""
|
"""
|
||||||
Black video or silence should be used as the source for this event.
|
Black video or silence should be used as the source for this event.
|
||||||
"""
|
"""
|
||||||
return self.source == "BL"
|
return self.source == "BL"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def aux_source(self):
|
def aux_source(self) -> bool:
|
||||||
"""
|
"""
|
||||||
An auxiliary source is the source of this event.
|
An auxiliary source is the source of this event.
|
||||||
"""
|
"""
|
||||||
return self.source == "AX"
|
return self.source == "AX"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def source_file(self):
|
def source_file(self) -> Optional[str]:
|
||||||
"""
|
"""
|
||||||
Get the source file, as attested by a "* SOURCE FILE" remark on the
|
Get the source file, as attested by a "* SOURCE FILE" remark on the
|
||||||
EDL. This will return None if the information is not present.
|
EDL. This will return None if the information is not present.
|
||||||
@@ -112,7 +114,7 @@ class Edit:
|
|||||||
return self.source_file_statement.filename
|
return self.source_file_statement.filename
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def clip_name(self):
|
def clip_name(self) -> Optional[str]:
|
||||||
"""
|
"""
|
||||||
Get the clip name, as attested by a "* FROM CLIP NAME" or "* TO CLIP
|
Get the clip name, as attested by a "* FROM CLIP NAME" or "* TO CLIP
|
||||||
NAME" remark on the EDL. This will return None if the information is
|
NAME" remark on the EDL. This will return None if the information is
|
||||||
|
|||||||
@@ -5,21 +5,21 @@ from .parse_cmx_statements import (StmtUnrecognized, StmtFCM, StmtEvent, StmtSou
|
|||||||
from .event import Event
|
from .event import Event
|
||||||
from .channel_map import ChannelMap
|
from .channel_map import ChannelMap
|
||||||
|
|
||||||
|
from typing import Generator
|
||||||
|
|
||||||
class EditList:
|
class EditList:
|
||||||
"""
|
"""
|
||||||
Represents an entire edit decision list as returned by `parse_cmx3600()`.
|
Represents an entire edit decision list as returned by :func:`~pycmx.parse_cmx3600()`.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
def __init__(self, statements):
|
def __init__(self, statements):
|
||||||
self.title_statement = statements[0]
|
self.title_statement = statements[0]
|
||||||
self.event_statements = statements[1:]
|
self.event_statements = statements[1:]
|
||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def format(self):
|
def format(self) -> str:
|
||||||
"""
|
"""
|
||||||
The detected format of the EDL. Possible values are: `3600`,`File32`,
|
The detected format of the EDL. Possible values are: "3600", "File32",
|
||||||
`File128`, and `unknown`
|
"File128", and "unknown".
|
||||||
"""
|
"""
|
||||||
first_event = next( (s for s in self.event_statements if type(s) is StmtEvent), None)
|
first_event = next( (s for s in self.event_statements if type(s) is StmtEvent), None)
|
||||||
|
|
||||||
@@ -37,7 +37,7 @@ class EditList:
|
|||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def channels(self):
|
def channels(self) -> ChannelMap:
|
||||||
"""
|
"""
|
||||||
Return the union of every channel channel.
|
Return the union of every channel channel.
|
||||||
"""
|
"""
|
||||||
@@ -51,17 +51,15 @@ class EditList:
|
|||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def title(self):
|
def title(self) -> str:
|
||||||
"""
|
"""
|
||||||
The title of this edit list, as attensted by the 'TITLE:' statement on
|
The title of this edit list.
|
||||||
the first line.
|
|
||||||
"""
|
"""
|
||||||
'The title of the edit list'
|
|
||||||
return self.title_statement.title
|
return self.title_statement.title
|
||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def unrecognized_statements(self):
|
def unrecognized_statements(self) -> Generator[StmtUnrecognized, None, None]:
|
||||||
"""
|
"""
|
||||||
A generator for all the unrecognized statements in the list.
|
A generator for all the unrecognized statements in the list.
|
||||||
"""
|
"""
|
||||||
@@ -71,7 +69,7 @@ class EditList:
|
|||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def events(self):
|
def events(self) -> Generator[Event, None, None]:
|
||||||
'A generator for all the events in the edit list'
|
'A generator for all the events in the edit list'
|
||||||
is_drop = None
|
is_drop = None
|
||||||
current_event_num = None
|
current_event_num = None
|
||||||
@@ -99,7 +97,7 @@ class EditList:
|
|||||||
yield Event(statements=event_statements)
|
yield Event(statements=event_statements)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def sources(self):
|
def sources(self) -> Generator[StmtSourceUMID, None, None]:
|
||||||
"""
|
"""
|
||||||
A generator for all of the sources in the list
|
A generator for all of the sources in the list
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -1,26 +1,28 @@
|
|||||||
# pycmx
|
# pycmx
|
||||||
# (c) 2018 Jamie Hardt
|
# (c) 2023 Jamie Hardt
|
||||||
|
|
||||||
from .parse_cmx_statements import (StmtEvent, StmtClipName, StmtSourceFile, StmtAudioExt, StmtUnrecognized, StmtEffectsName)
|
from .parse_cmx_statements import (StmtEvent, StmtClipName, StmtSourceFile, StmtAudioExt, StmtUnrecognized, StmtEffectsName)
|
||||||
from .edit import Edit
|
from .edit import Edit
|
||||||
|
|
||||||
|
from typing import List, Generator, Optional, Tuple, Any
|
||||||
|
|
||||||
class Event:
|
class Event:
|
||||||
"""
|
"""
|
||||||
Represents a collection of :class:`Edit`s, all with the same event number.
|
Represents a collection of :class:`~pycmx.edit.Edit` s, all with the same event number.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, statements):
|
def __init__(self, statements):
|
||||||
self.statements = statements
|
self.statements = statements
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def number(self):
|
def number(self) -> int:
|
||||||
"""
|
"""
|
||||||
Return the event number.
|
Return the event number.
|
||||||
"""
|
"""
|
||||||
return int(self._edit_statements()[0].event)
|
return int(self._edit_statements()[0].event)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def edits(self):
|
def edits(self) -> List[Edit]:
|
||||||
"""
|
"""
|
||||||
Returns the edits. Most events will have a single edit, a single event
|
Returns the edits. Most events will have a single edit, a single event
|
||||||
will have multiple edits when a dissolve, wipe or key transition needs
|
will have multiple edits when a dissolve, wipe or key transition needs
|
||||||
@@ -30,17 +32,19 @@ class Event:
|
|||||||
clip_names = self._clip_name_statements()
|
clip_names = self._clip_name_statements()
|
||||||
source_files= self._source_file_statements()
|
source_files= self._source_file_statements()
|
||||||
|
|
||||||
the_zip = [edits_audio]
|
the_zip: List[List[Any]] = [edits_audio]
|
||||||
|
|
||||||
if len(edits_audio) == 2:
|
if len(edits_audio) == 2:
|
||||||
cn = [None, None]
|
start_name: Optional[StmtClipName] = None
|
||||||
|
end_name: Optional[StmtClipName] = None
|
||||||
|
|
||||||
for clip_name in clip_names:
|
for clip_name in clip_names:
|
||||||
if clip_name.affect == 'from':
|
if clip_name.affect == 'from':
|
||||||
cn[0] = clip_name
|
start_name = clip_name
|
||||||
elif clip_name.affect == 'to':
|
elif clip_name.affect == 'to':
|
||||||
cn[1] = clip_name
|
end_name = clip_name
|
||||||
|
|
||||||
the_zip.append(cn)
|
the_zip.append([start_name, end_name])
|
||||||
else:
|
else:
|
||||||
if len(edits_audio) == len(clip_names):
|
if len(edits_audio) == len(clip_names):
|
||||||
the_zip.append(clip_names)
|
the_zip.append(clip_names)
|
||||||
@@ -57,17 +61,20 @@ class Event:
|
|||||||
# attach trans name to last event
|
# attach trans name to last event
|
||||||
try:
|
try:
|
||||||
trans_statement = self._trans_name_statements()[0]
|
trans_statement = self._trans_name_statements()[0]
|
||||||
trans_names = [None] * (len(edits_audio) - 1)
|
trans_names: List[Optional[Any]] = [None] * (len(edits_audio) - 1)
|
||||||
trans_names.append(trans_statement)
|
trans_names.append(trans_statement)
|
||||||
the_zip.append(trans_names)
|
the_zip.append(trans_names)
|
||||||
except IndexError:
|
except IndexError:
|
||||||
the_zip.append([None] * len(edits_audio) )
|
the_zip.append([None] * len(edits_audio) )
|
||||||
|
|
||||||
|
return [ Edit(edit_statement=e1[0],
|
||||||
return [ Edit(e1[0],e1[1],n1,s1,u1) for (e1,n1,s1,u1) in zip(*the_zip) ]
|
audio_ext_statement=e1[1],
|
||||||
|
clip_name_statement=n1,
|
||||||
|
source_file_statement=s1,
|
||||||
|
trans_name_statement=u1) for (e1,n1,s1,u1) in zip(*the_zip) ]
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def unrecognized_statements(self):
|
def unrecognized_statements(self) -> Generator[StmtUnrecognized, None, None]:
|
||||||
"""
|
"""
|
||||||
A generator for all the unrecognized statements in the event.
|
A generator for all the unrecognized statements in the event.
|
||||||
"""
|
"""
|
||||||
@@ -75,19 +82,19 @@ class Event:
|
|||||||
if type(s) is StmtUnrecognized:
|
if type(s) is StmtUnrecognized:
|
||||||
yield s
|
yield s
|
||||||
|
|
||||||
def _trans_name_statements(self):
|
def _trans_name_statements(self) -> List[StmtEffectsName]:
|
||||||
return [s for s in self.statements if type(s) is StmtEffectsName]
|
return [s for s in self.statements if type(s) is StmtEffectsName]
|
||||||
|
|
||||||
def _edit_statements(self):
|
def _edit_statements(self) -> List[StmtEvent]:
|
||||||
return [s for s in self.statements if type(s) is StmtEvent]
|
return [s for s in self.statements if type(s) is StmtEvent]
|
||||||
|
|
||||||
def _clip_name_statements(self):
|
def _clip_name_statements(self) -> List[StmtClipName]:
|
||||||
return [s for s in self.statements if type(s) is StmtClipName]
|
return [s for s in self.statements if type(s) is StmtClipName]
|
||||||
|
|
||||||
def _source_file_statements(self):
|
def _source_file_statements(self) -> List[StmtSourceFile]:
|
||||||
return [s for s in self.statements if type(s) is StmtSourceFile]
|
return [s for s in self.statements if type(s) is StmtSourceFile]
|
||||||
|
|
||||||
def _statements_with_audio_ext(self):
|
def _statements_with_audio_ext(self) -> Generator[Tuple[StmtEvent, Optional[StmtAudioExt]], None, None]:
|
||||||
for (s1, s2) in zip(self.statements, self.statements[1:]):
|
for (s1, s2) in zip(self.statements, self.statements[1:]):
|
||||||
if type(s1) is StmtEvent and type(s2) is StmtAudioExt:
|
if type(s1) is StmtEvent and type(s2) is StmtAudioExt:
|
||||||
yield (s1,s2)
|
yield (s1,s2)
|
||||||
|
|||||||
@@ -1,20 +1,19 @@
|
|||||||
# pycmx
|
# pycmx
|
||||||
# (c) 2018 Jamie Hardt
|
# (c) 2018 Jamie Hardt
|
||||||
|
|
||||||
from collections import namedtuple
|
# from collections import namedtuple
|
||||||
|
|
||||||
from .parse_cmx_statements import (parse_cmx3600_statements, StmtEvent,StmtFCM )
|
from .parse_cmx_statements import (parse_cmx3600_statements)
|
||||||
from .edit_list import EditList
|
from .edit_list import EditList
|
||||||
|
|
||||||
def parse_cmx3600(f):
|
from typing import TextIO
|
||||||
|
|
||||||
|
def parse_cmx3600(f: TextIO):
|
||||||
"""
|
"""
|
||||||
Parse a CMX 3600 EDL.
|
Parse a CMX 3600 EDL.
|
||||||
|
|
||||||
Args:
|
:param TextIO f: a file-like object, anything that's readlines-able.
|
||||||
f : a file-like object, anything that's readlines-able.
|
:returns: An :class:`pycmx.edit_list.EditList`.
|
||||||
|
|
||||||
Returns:
|
|
||||||
An :class:`EditList`.
|
|
||||||
"""
|
"""
|
||||||
statements = parse_cmx3600_statements(f)
|
statements = parse_cmx3600_statements(f)
|
||||||
return EditList(statements)
|
return EditList(statements)
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ import re
|
|||||||
import sys
|
import sys
|
||||||
from collections import namedtuple
|
from collections import namedtuple
|
||||||
from itertools import count
|
from itertools import count
|
||||||
|
from typing import TextIO, List
|
||||||
|
|
||||||
|
|
||||||
from .util import collimate
|
from .util import collimate
|
||||||
|
|
||||||
@@ -18,12 +20,12 @@ StmtSourceFile = namedtuple("SourceFile",["filename","line_number"])
|
|||||||
StmtRemark = namedtuple("Remark",["text","line_number"])
|
StmtRemark = namedtuple("Remark",["text","line_number"])
|
||||||
StmtEffectsName = namedtuple("EffectsName",["name","line_number"])
|
StmtEffectsName = namedtuple("EffectsName",["name","line_number"])
|
||||||
StmtSourceUMID = namedtuple("Source",["name","umid","line_number"])
|
StmtSourceUMID = namedtuple("Source",["name","umid","line_number"])
|
||||||
StmtSplitEdit = namedtuple("SplitEdit",["video","magnitue", "line_number"])
|
StmtSplitEdit = namedtuple("SplitEdit",["video","magnitude", "line_number"])
|
||||||
StmtMotionMemory = namedtuple("MotionMemory",["source","fps"]) # FIXME needs more fields
|
StmtMotionMemory = namedtuple("MotionMemory",["source","fps"]) # FIXME needs more fields
|
||||||
StmtUnrecognized = namedtuple("Unrecognized",["content","line_number"])
|
StmtUnrecognized = namedtuple("Unrecognized",["content","line_number"])
|
||||||
|
|
||||||
|
|
||||||
def parse_cmx3600_statements(file):
|
def parse_cmx3600_statements(file: TextIO) -> List[object]:
|
||||||
"""
|
"""
|
||||||
Return a list of every statement in the file argument.
|
Return a list of every statement in the file argument.
|
||||||
"""
|
"""
|
||||||
@@ -109,7 +111,7 @@ def _parse_extended_audio_channels(line, line_number):
|
|||||||
else:
|
else:
|
||||||
return StmtUnrecognized(content=line, line_number=line_number)
|
return StmtUnrecognized(content=line, line_number=line_number)
|
||||||
|
|
||||||
def _parse_remark(line, line_number):
|
def _parse_remark(line, line_number) -> object:
|
||||||
if line.startswith("FROM CLIP NAME:"):
|
if line.startswith("FROM CLIP NAME:"):
|
||||||
return StmtClipName(name=line[15:].strip() , affect="from", line_number=line_number)
|
return StmtClipName(name=line[15:].strip() , affect="from", line_number=line_number)
|
||||||
elif line.startswith("TO CLIP NAME:"):
|
elif line.startswith("TO CLIP NAME:"):
|
||||||
@@ -119,7 +121,7 @@ def _parse_remark(line, line_number):
|
|||||||
else:
|
else:
|
||||||
return StmtRemark(text=line, line_number=line_number)
|
return StmtRemark(text=line, line_number=line_number)
|
||||||
|
|
||||||
def _parse_effects_name(line, line_number):
|
def _parse_effects_name(line, line_number) -> StmtEffectsName:
|
||||||
name = line[16:].strip()
|
name = line[16:].strip()
|
||||||
return StmtEffectsName(name=name, line_number=line_number)
|
return StmtEffectsName(name=name, line_number=line_number)
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
# pycmx
|
# pycmx
|
||||||
# (c) 2018 Jamie Hardt
|
# (c) 2023 Jamie Hardt
|
||||||
|
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
class Transition:
|
class Transition:
|
||||||
"""
|
"""
|
||||||
@@ -19,7 +21,7 @@ class Transition:
|
|||||||
self.name = name
|
self.name = name
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def kind(self):
|
def kind(self) -> Optional[str]:
|
||||||
"""
|
"""
|
||||||
Return the kind of transition: Cut, Wipe, etc
|
Return the kind of transition: Cut, Wipe, etc
|
||||||
"""
|
"""
|
||||||
@@ -37,22 +39,22 @@ class Transition:
|
|||||||
return Transition.KeyOut
|
return Transition.KeyOut
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def cut(self):
|
def cut(self) -> bool:
|
||||||
"`True` if this transition is a cut."
|
"`True` if this transition is a cut."
|
||||||
return self.transition == 'C'
|
return self.transition == 'C'
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def dissolve(self):
|
def dissolve(self) -> bool:
|
||||||
"`True` if this traansition is a dissolve."
|
"`True` if this traansition is a dissolve."
|
||||||
return self.transition == 'D'
|
return self.transition == 'D'
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def wipe(self):
|
def wipe(self) -> bool:
|
||||||
"`True` if this transition is a wipe."
|
"`True` if this transition is a wipe."
|
||||||
return self.transition.startswith('W')
|
return self.transition.startswith('W')
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def effect_duration(self):
|
def effect_duration(self) -> int:
|
||||||
"""The duration of this transition, in frames of the record target.
|
"""The duration of this transition, in frames of the record target.
|
||||||
|
|
||||||
In the event of a key event, this is the duration of the fade in.
|
In the event of a key event, this is the duration of the fade in.
|
||||||
@@ -60,7 +62,7 @@ class Transition:
|
|||||||
return int(self.operand)
|
return int(self.operand)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def wipe_number(self):
|
def wipe_number(self) -> Optional[int]:
|
||||||
"Wipes are identified by a particular number."
|
"Wipes are identified by a particular number."
|
||||||
if self.wipe:
|
if self.wipe:
|
||||||
return int(self.transition[1:])
|
return int(self.transition[1:])
|
||||||
@@ -68,19 +70,21 @@ class Transition:
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def key_background(self):
|
def key_background(self) -> bool:
|
||||||
"`True` if this edit is a key background."
|
"`True` if this edit is a key background."
|
||||||
return self.transition == KeyBackground
|
return self.transition == Transition.KeyBackground
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def key_foreground(self):
|
def key_foreground(self) -> bool:
|
||||||
"`True` if this edit is a key foreground."
|
"`True` if this edit is a key foreground."
|
||||||
return self.transition == Key
|
return self.transition == Transition.Key
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def key_out(self):
|
def key_out(self) -> bool:
|
||||||
"""
|
"""
|
||||||
`True` if this edit is a key out. This material will removed from
|
`True` if this edit is a key out. This material will removed from
|
||||||
the key foreground and replaced with the key background.
|
the key foreground and replaced with the key background.
|
||||||
"""
|
"""
|
||||||
return self.transition == KeyOut
|
return self.transition == Transition.KeyOut
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,2 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
python3 -m twine upload --repository-url https://upload.pypi.org/legacy/ dist/*
|
|
||||||
61
pyproject.toml
Normal file
61
pyproject.toml
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
[build-system]
|
||||||
|
requires = ["flit_core >=3.2,<4"]
|
||||||
|
build-backend = "flit_core.buildapi"
|
||||||
|
|
||||||
|
[project]
|
||||||
|
name = "pycmx"
|
||||||
|
authors = [{name = "Jamie Hardt", email = "jamiehardt@me.com"}]
|
||||||
|
readme = "README.md"
|
||||||
|
dynamic = ["version", "description"]
|
||||||
|
requires-python = "~=3.7"
|
||||||
|
classifiers = [
|
||||||
|
'Development Status :: 5 - Production/Stable',
|
||||||
|
'License :: OSI Approved :: MIT License',
|
||||||
|
'Topic :: Multimedia',
|
||||||
|
'Topic :: Multimedia :: Video',
|
||||||
|
'Topic :: Text Processing',
|
||||||
|
'Programming Language :: Python :: 3.7',
|
||||||
|
'Programming Language :: Python :: 3.8',
|
||||||
|
'Programming Language :: Python :: 3.9',
|
||||||
|
'Programming Language :: Python :: 3.10',
|
||||||
|
'Programming Language :: Python :: 3.11',
|
||||||
|
'Programming Language :: Python :: 3.12'
|
||||||
|
]
|
||||||
|
dependencies = [
|
||||||
|
|
||||||
|
]
|
||||||
|
keywords = [
|
||||||
|
'parser',
|
||||||
|
'film',
|
||||||
|
'broadcast'
|
||||||
|
]
|
||||||
|
|
||||||
|
[tool.flit.module]
|
||||||
|
name = "pycmx"
|
||||||
|
|
||||||
|
[project.optional-dependencies]
|
||||||
|
doc = [
|
||||||
|
'sphinx >= 5.3.0',
|
||||||
|
'sphinx_rtd_theme >= 1.1.1',
|
||||||
|
]
|
||||||
|
|
||||||
|
[project.urls]
|
||||||
|
Home = "https://github.com/iluvcapra/pycmx"
|
||||||
|
Documentation = "https://pycmx.readthedocs.io/"
|
||||||
|
Source = "https://github.com/iluvcapra/pycmx.git"
|
||||||
|
Issues = "https://github.com/iluvcapra/pycmx/issues"
|
||||||
|
|
||||||
|
[tool.pyright]
|
||||||
|
typeCheckingMode = "basic"
|
||||||
|
|
||||||
|
[tool.pylint]
|
||||||
|
max-line-length = 88
|
||||||
|
disable = [
|
||||||
|
"C0103", # (invalid-name)
|
||||||
|
"C0114", # (missing-module-docstring)
|
||||||
|
"C0115", # (missing-class-docstring)
|
||||||
|
"C0116", # (missing-function-docstring)
|
||||||
|
"R0903", # (too-few-public-methods)
|
||||||
|
"R0913", # (too-many-arguments)
|
||||||
|
"W0105", # (pointless-string-statement)
|
||||||
|
]
|
||||||
24
setup.py
24
setup.py
@@ -1,24 +0,0 @@
|
|||||||
from setuptools import setup
|
|
||||||
|
|
||||||
with open("README.md", "r") as fh:
|
|
||||||
long_description = fh.read()
|
|
||||||
|
|
||||||
setup(name='pycmx',
|
|
||||||
version='1.0.1',
|
|
||||||
author='Jamie Hardt',
|
|
||||||
author_email='jamiehardt@me.com',
|
|
||||||
description='CMX 3600 Edit Decision List Parser',
|
|
||||||
long_description_content_type="text/markdown",
|
|
||||||
long_description=long_description,
|
|
||||||
url='https://github.com/iluvcapra/pycmx',
|
|
||||||
classifiers=['Development Status :: 5 - Production/Stable',
|
|
||||||
'License :: OSI Approved :: MIT License',
|
|
||||||
'Topic :: Multimedia',
|
|
||||||
'Topic :: Multimedia :: Video',
|
|
||||||
'Topic :: Text Processing',
|
|
||||||
'Programming Language :: Python :: 2.7',
|
|
||||||
'Programming Language :: Python :: 3.5',
|
|
||||||
'Programming Language :: Python :: 3.6',
|
|
||||||
'Programming Language :: Python :: 3.7',
|
|
||||||
],
|
|
||||||
packages=['pycmx'])
|
|
||||||
@@ -66,6 +66,7 @@ class TestParse(TestCase):
|
|||||||
self.assertFalse( events[0].edits[0].channels.a1)
|
self.assertFalse( events[0].edits[0].channels.a1)
|
||||||
self.assertTrue( events[0].edits[0].channels.a2)
|
self.assertTrue( events[0].edits[0].channels.a2)
|
||||||
self.assertTrue( events[2].edits[0].channels.get_audio_channel(7) )
|
self.assertTrue( events[2].edits[0].channels.get_audio_channel(7) )
|
||||||
|
self.assertTrue( events[2].edits[0].channels.audio)
|
||||||
|
|
||||||
|
|
||||||
def test_multi_edit_events(self):
|
def test_multi_edit_events(self):
|
||||||
|
|||||||
Reference in New Issue
Block a user