- Development Setup
- Creating a Release
- Coding Conventions
- Writing Commit Messages
- Writing Documentation
- Element = module, class, function or variable
- PDB = GIMP procedural database
This section explains how to set up development environment for Batcher.
If you are using a Linux-based environment, the easiest to get set up is to download and run the bash script that automatically installs any required dependencies and sets up the environment. Users of Windows 10 and above may use e.g. the Windows Subsystem for Linux to run the script.
If you cannot run the script, perform manual setup as per the instructions below.
Locate the directory for local GIMP plug-ins (e.g. [home directory]/.config/GIMP/[version]
on Linux-based systems) and create a directory named batcher
.
To make GIMP recognize the new directory as a directory containing the development version of Batcher, open GIMP, go to Edit → Preferences → Folders → Plug-ins
and add the new directory to the list.
GIMP needs to be restarted for changes to take effect.
Clone the gh-pages
branch (acting as the GitHub page for Batcher) to [Batcher directory]/docs/gh-pages
.
Several scripts depend on this directory location.
Some scripts require that the GitHub page be run locally. To set up GitHub page locally:
-
Install Ruby language.
-
Install
bundler
gem:gem install bundler
-
Switch to the
gh-pages
directory:cd docs/gh-pages
-
Run
bundle
to install required dependencies:bundle install
Set up git hooks located in dev/git_hooks
by creating symbolic links:
ln -s dev/git_hooks/commig_msg.py .git/hooks/commit-msg
ln -s dev/git_hooks/pre_commit.py .git/hooks/pre-commit
The commit-msg
hook enforces several conventions for commit messages.
The pre-commit
hook automatically propagates changes in files in docs
to files comprising the end user documentation, such as the Readme and GitHub pages (located in docs/gh-pages
).
See User Documentation for more information.
The repository contains readily available configuration for the PyCharm IDE.
Among other things, the configuration suppresses several formatting warnings and missing references to global variables (such as _
used as a function to return a translated string).
Run dev/make_installers.sh
to create installer(s).
Use the -i
option to specify platform(s) for which to create installers.
The installers will be created in the installers/output
directory.
For Python modules, follow PEP 8 conventions unless specified otherwise below.
Plug-in-related Python modules must be placed under the batcher/src
directory.
This avoids a problem where Python attempts to import modules from the top-level module (batcher.py
) rather than the package with the same name (batcher
).
Development-related modules (such as documentation generation) must be placed under the dev
directory.
The maximum line length is:
- 100 characters for code,
- 80 characters for docstrings and comments.
Use two spaces for indentation.
Use hanging indents, with indented lines having two extra spaces.
For multi-line lists of variables (or arguments in function calls or definitions):
- if the variables fit in one line, use one line for all variables,
- if the variables do not fit in one line, use one line for each variable.
For multi-line conditions, align expressions vertically.
For multi-line function or class definitions, loops and with
statements, use hanging indents and add two spaces after the beginning of the name of the function/class, first loop variable or the expression after the with
keyword.
Example:
def __init__(
self,
default_value):
with open(
'/a/very/very/long/path/to/file.txt', 'w') as f:
pass
Use single quotes except for cases when double quotes must be used.
Use double quotes:
- for docstrings,
- for emphasizing text in docstrings or comments,
- for string literals containing single quotes (to avoid inserting backslashes), e.g.
"the plug-in's interactive dialog"
In comments and docstrings, wrap element names in backquotes.
Format function and method names as `function()
`, i.e. with the trailing ()
.
Use the following conventions for terms and variables:
Term | Variable name | Meaning |
---|---|---|
File name | filename |
File basename |
Directory name | dirname |
Directory basename |
File path | filepath |
Absolute or relative file path |
Directory path | dirpath |
Absolute or relative directory path |
Import modules at the beginning of a module.
Avoid importing individual objects, classes or functions. Notable exceptions:
- types from the
typing
andcollections.abc
modules pdb
object inpygimplib
Do not use wildcard imports. Exceptions:
- internal modules whose public elements should be used directly through a package.
These modules must define
__all__
to list all public elements to be imported.
Append _
to imported modules with common names (e.g. settings
becomes settings_
) to avoid clashes with variable names.
Prepend _
to unused parameter names or return values (e.g. settings
becomes _settings
).
Do not call code directly on the module or class level. Exceptions to this rule include:
- initializing module- or class-level variables or constants,
- initializing application configuration,
- initializing a package or a library,
- standalone scripts such as test runners or git hooks.
Avoid calling functions from the GIMP API and GTK API on the module or class level.
These APIs are not fully initialized during GIMP startup and may result in plug-in procedures failing to be registered by GIMP.
Some functions such as Gimp.directory()
as safe to call on the module or class level, but functions manipulating images or creating GUI widgets will yield errors.
For standalone scripts (such as utility scripts for creating a new release), the code inside if __name__ == '__main__':
must be enclosed in a single function in order to avoid introducing global variables.
The name of the enclosing function should be main()
.
Yes:
def main():
# code
if __name__ == '__main__':
main()
No:
if __name__ == '__main__':
# code
Mixins can access and modify only those attributes defined in it.
No other classes may directly modify attributes defined in mixins.
Use @classmethod
for methods using class variables only.
Use @staticmethod
for methods not using instance or class variables and logically belonging to the class.
Do not use keys()
when iterating over dictionary keys, except where it improves clarity (e.g. passing keys as a parameter to a function).
For public classes, methods and functions, provide type hints for parameters and return values. For brevity, you can omit Any
which indicates an arbitrary type.
Always use UTF-8 encoding when loading and saving data provided by this plug-in.
Always use GObject
types (for Gtk.TreeView
columns, __gsignals__
, etc.) instead of Python types if such GObject
types exist.
For example, use GObject.TYPE_STRING
instead of str
for Gtk.TreeView
columns of string type.
If it is necessary to get the dimensions or the relative position of a widget not yet realized, connect to the 'size-allocate'
signal and continue processing in the connected event handler.
Do not use Gtk.main_iteration()
(which forces the GUI to update) for this purpose as it introduces flickering in the GUI.
Make sure the code is compatible with Python 3.7 and above.
This section explains how to write good commit messages. The conventions are based on the following guidelines:
Some conventions are automatically enforced by the git hook commit-msg
.
These conventions are marked by a trailing "*".
Each commit should change one and one thing only. For example, a bug fix and introducing a new feature should be treated as separate commits.
Use the following format for commit messages*:
<scope>: <header>
<body>
- Limit the first line to 80 characters*. Strive for brevity and clarity.
- Do not end the first line with a period*.
- Begin the header with a verb in the imperative.
- Begin the header with a capital letter*.
- Be concise. Limit the message to the first line unless further explanation is required.
- Wrap the message body in 72 characters*.
- Wrap element names with backquotes.
- Format function and method names as `
function()
`.
Scope in the first line is optional, but highly recommended.
Use one of the following types of scope:
- package name
- module name
- directory name (e.g.
docs
for the directory containing documentation) - filename without extension (for files that are not Python modules)
To indicate a more detailed scope, use .
as a path separator, e.g. gui.settings: ...
.
Documentation for modules, classes and functions are written as docstrings directly in the source code.
Each module except test modules should contain a short module-level docstring.
This docstring should be placed at the top of the module (after #!/usr/bin/env python3
if present), followed by one empty blank line.
Provide docstrings for public classes, methods and functions.
Use the Sphinx syntax and either the Google-style docstrings (for more complex functions and methods) or a "plain" style as seen in the documentation for Python standard modules (e.g. as seen in functions in the Python os
module).
For functions and methods, the first line provides a summary and begins with a verb in the third person (as seen in the Google Python style guide).
Similarly, class and module docstrings should begin with a one-line summary.
Avoid beginning the module docstring with This module...
and directly describe its purpose (e.g. Loading and saving settings.
instead of This module defines classes to load and save settings.
).
To update documentation for end users, modify the "raw" documentation - files located in docs
(except files in gh-pages
).
Do not modify other documentation files outside docs
as they are automatically updated by git hooks when committing changes to the files in docs
.
Any changes in user documentation propagated to files in docs/gh-pages
should be reviewed first before pushing to the gh-pages
branch.
In Markdown files, break lines on sentences. For long sentences, rely on soft wrap or split them into multiple shorter sentences.