/inventree-template-pro

A plugin for InvenTree to enhance the clarity and professionalism of your reports and labels.

Primary LanguagePythonMIT LicenseMIT

Have you carefully crafted your InvenTree deployment only to get labels and reports that appear meaningless:

C25076 | 0402WGF1000TCE

Or have wordy and inconsistent descriptions:

0402WGF1000TCE Passives / Resistors 62.5mW 100Ω 50V Thick Film Resistors ±200ppm/℃ ±1% 0402 Chip Resistor - Surface Mount ROHS

Wouldn't you rather have concise, detailed information from your part's parameters?

100ohm 62.5mw 1% SMD 0402

InvenTree Part Templates

InvenTree-Part-Templates is a plugin for InvenTree that enhances InvenTree's label and reporting templates. You can create new custom context properties, adjust template values for those properties based on the parts' category, and utilize the part_templates context variable in your report and label templates for easy, consistent reporting. Additionally, this plugin allows you to clean up inconsistencies in parameter values, significantly improving the quality of your labels and reports.

Introducing Context Properties

Context Properties are a feature of InvenTree Part Templates that provide a straightforward way to include contextual information about a part in a report. A single Context Property can have different values depending on the context of the part, such as the category. This is achieved by defining the context property template on catalog items (or directly on a part). For example, a part that is a resistor might include information about resistance, wattage, and tolerance, whereas a connector might display pin count, pitch, and current ratings. You can define custom templates for on any catalog item, and even directly on a part, for any of the defined context properties.

In short, it's a way to define a name to use in your reporting templates, which then uses the part to find the best template, which in turn defines the information to include in the report. It may be a bit confusing at first, with templates referring to other templates, but it's surprisingly easy to use and highly capable.

Pulling this all together in a simple example, let's say you have already created a context property called part_details. You can reference this property in your labels and reports just like any other template property (as a property of part_parameters):

<div class='category'>
  {{ part_templates.part_details }}
</div>

When you created the part_details context property, you defined a default template value (optional, but highly recommended). This is used when no contextual template can be found for the part, ensuring the report still has usable content. A default template might be:

{{ part.name }} ({{ part.ipn }}): {{ category.name }}

When you run your report, {{ part_templates.part_details }} will resolve to {{ part.name }} ({{ part.ipn }}): {{ category.name }} which in turn will process that template and result in something like this:

MFK3293C99-3Q (C19239): Resistor

Where this gets much more interesting is when you define templates for these context properties on different catalog items in your InvenTree database. You can add template values on catalog entries, and even directly on parts, to supply detailed contextual templates for those parts. You have full access to all the properties of part, category, parameter and stock (if applicable to a stock report/label; otherwise, it is empty) in the context property template. When a context property is used, the "closest" template is used, starting with the part and moving up the catalog hierarchy, and if none is found, the default template is used.

For example, let's say your catalog has an entry called "Integrated Circuits" and a subcategory called "Buffers." You can add a part_details template on "Integrated Circuits" to include the "Number of Gates" property ({{ parameter|item:"Number of Gates" }}), but have none on the "Integrated Circuits / Buffers." When the report is run, it checks first on the part (which has no part_details template), then it checks "Buffers" (also none), and then it finds it on "Integrated Circuits" and returns the "Number of Gates" property on the part.

Similarly, any value that isn't found (such as a property not on a part) will be ignored (resulting in a blank output). This is beneficial for defining templates higher up in the catalog hierarchy that might reference properties that do not exist on all children.

Installation

InvenTree Part Templates is installed like most plugins. First, verify that plugins are enabled by visiting the Settings / Plugin Settings / Plugins page, and ensure the Enable URL integration, Enable app integration, and Check plugins on startup settings (if using Docker containers) are all enabled.

Then, install the plugin using your preferred method. The easiest methods are:

  • The best approach, especially when using Docker, is to edit your inventree_data/plugins.txt file to add the package name (inventree-part-templates). Restart InvenTree for the package to be downloaded and installed.
  • Visit the Settings / Plugin Settings / Plugins page in the management console and install it from there.

Once installed, verify the installation by checking the Settings / Plugin Settings / Plugins page. There should be no errors from plugins at the bottom of the page, and the PartTemplatesPlugin should be listed.

Plugin settings

The first step is to determine which context properties you want to start with. You can have up to five context properties (if you need more, please open a GitHub issue). Commonly, a part description and a category property are used, both of which are set as defaults. You can remove or edit these as needed.

The reason a category context property is commonly used is that sometimes the category name needs to show the parent category to make sense, such as "Capacitor / Aluminum." However, other times it is unnecessary, such as "Passive / Resistor." A default context property template of {{ category.parent.name }} / {{ category.name }} works well for "Capacitor / Aluminum", with {{ category.name }} (or just Resistor) used as the template on the "Registor" category.

Visit the Settings / Plugin Settings / Plugins page to set up the context properties as well as the settings for rights to manage templates:

InvenTree Part Templates Settings page

The first two fields control access rights for viewing and editing the templates on Category and Part pages. If both rights are disabled (edit and view), then the Part Templates panel will not be included on the page.

The remaining ten fields are for defining the name and default template for up to five context properties.

Use in Labels and Reports

With the plugin installed and some context properties defined, you can now update your reports and labels to incorporate them. Initially, they will display values from the default template, but as you add category- and part-specific templates, the content will automatically update to reflect these changes. By integrating them into your labels and reports now, you can fine-tune and adjust your context property templates to meet your usability and size requirements.

Adjusting a template to use context properties is straightforward. Simply open your template in any text editor and incorporate the syntax {{ part_templates.<your_context_property_name> }}. Afterward, upload it to your label or report system.

You can then print your report or label, for instance, using the InvenTree PDF Label Printer, to view how the default template for your context property functions on the screen.

Context Templates

The next step is to adjust the template value so that it is contextually appropriate. This adjustment is typically made on catalog items but can also be applied to individual parts. For instance, if you have the catalog structure Electronics / Passives / Capacitors / Aluminum, you might want a context template on Capacitors that is specific to all capacitor types (Aluminum, Tantalum, MLCC, etc.), allowing shared values across these different types.

On the Part and Category pages, provided you have the rights to view and edit the Part Templates (refer to Plugin Settings), you will see a new "Part Templates" panel. For Part templates, the panel will display as follows:

InvenTree Part Templates Panel

This panel shows the context properties, the values each property will resolve to when the template is applied to a specific part, and the template currently in use (including any inherited templates).

When you view a Catalog item, the template values column is not included:

InvenTree Catalog Templates Panel

Use the edit buttons on the right (edit, delete) to modify the templates for this specific catalog or part template.

Template Context Variables and Filters

The context templates are standard InvenTree/Django templates and support several context variables along with a few new filters. The context variables include:

  • part: The current report/label part
  • category: The current part category
  • parameters: A shorthand for part.parameters
  • stock: The current stock item, if being reported on. If there is no specific stock item, such as in a part-only report, this variable will be empty. Use {% if stock %} to check for stock availability.

These variables facilitate dynamic content generation in templates, enhancing the adaptability and relevance of your reports and labels.

Context Property Template Filters

The following filters are available on context property templates and can be accessed by loading with {% load part_templates %} in your labels/reports:

  • item:"<name>": Retrieves a property name from a dictionary and scrubs it if filters exist. Example: {{ parameters|item:"Rated Voltage" }}
  • value:"<name>": Retrieves a property name from a dictionary without any scrubbing.
  • scrub:"<name>": Scrubs the associated string using a filter. Example: {{ part.name|scrub:"MPN" }}

The following are some examples of using context parameter templates with filters:

  • Resistor description: {{ parameters|item:"Resistance" }} {{ parameters|item:"Rated Power" }} {{ parameters|item:"Tolerance" }} {{ parameters|item:"Mounting Type" }} {{ parameters|item:"Package Type" }}
  • Resistor short category: RES
  • Capacitor description: {{ parameters|item:"Capacitance" }} {{ parameters|item:"Tolerance" }} {{ parameters|item:"Rated Voltage" }} {{ parameters|item:"Mounting Type" }} {{ parameters|item:"Package Type" }}
  • Capacitor short category: CAP
  • Capacitor/Tantalum short category: CAP TANT
  • Conditional example: {{ parameters|item:"Supply Voltage" }} {% if not parameters|item:"Supply Voltage" %}{{ parameters|item:"Input Voltage" }} {% endif %} {{ parameters|item:"Type" }}

Parameter Scrubbing

Parameters on parts are convienent, but often not very consistent. This is especially true when automatically importing parts and parameters from supplies such as DigiKey, Mouser and LCSC. For example, a parameter of "Mounting Type" might be "SMT", "SMD", "Surface Mount" or even "Brick Nogging"!

Parameter scrubbing is a way to cleanse properties, using Regular Expressions. The cleansing does not change the property value in the database, so there is no risk of accidential loss of information by scrubbing. It is used only to resolve templates for labels/reports, and only when you use the item or scrub filters.

The file part_templates.yaml defines a collection of filters organized by names (which are usually the same as a part's property name), followed by a list of regular expression filters and replacement strings. The default part_templates.yaml file provided with the plugin is based on property names used by InvenTree Part Import using a configuration similar to this one. You can specify your own template file using the environment variable PART_TEMPLATES_CONFIG_FILE, such as in your .env file:

PART_TEMPLATES_CONFIG_FILE=./inventree-data/part_templates.yaml

Here is a simple example of filtering Brick Nogging and Surface Mount to be SMD:

filters:
  "Mounting Type":
    - pattern: "^Brick .*"
      replacement: "SMD"
    - pattern: "^Surface Mount.*"
      replacement: "SMD"

While the item:"<Property Name>" filter automatically applies the name of the property to find the filter, you can also use scrub:"<Any Name>" to scrub a string. For example, you might define a filter called MPN like this:

filters:
  "MPN":
    # remove everything after a comma
    - pattern: "^\\s*([^,\\s]*).*"
      replacement: "\\1"

And then in your template use {{ part.name|scrub:"MPN" }} to apply the filter to the string resulting from part.name. In this case, if the MPN has a comma in it, everything following the comma is removed.

There is also a _GLOBAL filter category, which is applied automatically on any scrubbed value. For example:

  "_GLOBAL":
    # currently, InvenTree does not handle reporting with chinese characters (they
    # become spaces).  Until that is fixed, this rule removes them
    - pattern: "[\u4e00-\u9fff\u3400-\u4dbf\uff00-\uffef]"
      replacement: ""

Example Labels

See the example labels folder for various labels designed for the Brother QL-810W label printer. These labels utilize 29mm endless tape and are available in three sizes:

  • inventree-label-part-29mm: This is a full-width label that includes long descriptions and detailed category names. Set the label width to 52 and height to 27 in the part label settings.
  • inventree-label-part-16mm: This narrow and long label is ideal for placing on the edge of a container. It features a horizontal cut line for manual cutting (easily adaptable for 12mm endless tape). Set the label width to 60 and height to 28.
  • inventree-label-part-smdbox: A very small label, perfectly sized for small, spring-hinged, modular "WENTAI" boxes. This label is an excellent example of the need for short category names. Set the label width to 15 and height to 26.

The label width determines the cutting point of the label, and the height influences the scaling. A smaller height results in a larger label appearance, and these dimensions have been selected to ensure a clean fit.

These labels utilize the following context properties:

  • description: A long contextual part description.
  • short-desc: A shorter version of the part description, suitable for the smaller labels.
  • category: A descriptive category name.
  • short-cat: A very short category, limited to a maximum of 6 characters per word and up to two words.

Also, note the use of another property, part_templates.error. This is set whenever the templates fail to render, facilitating easier debugging of template issues. It is a best practice to check for both part_templates and part_templates.error as shown below:

{% if not part_templates %}
    <div class="error">Plugin inventree-part-templates required</div>
{% elif part_templates.error %}
    <div class="error">{{ part_templates.error }}</div>
{% else %}
    ... your label here ...
{% endif %}