inventree-zebra-plugin


Nameinventree-zebra-plugin JSON
Version 0.5.2 PyPI version JSON
download
home_pagehttps://github.com/SergeoLacruz/inventree-zebra-plugin
SummaryZebra label printer plugin for InvenTree
upload_time2024-03-15 14:44:02
maintainer
docs_urlNone
authorMichael Buchmann
requires_python>=3.6
licenseMIT
keywords inventree label printer printing inventory
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)


# inventree-zebra-plugin

This is a label printing plugin for [InvenTree](https://inventree.org), which provides support for Zebra Label printers .
It was only tested with  GK420T but should work for other ZPL printers too. It uses the ZPL library to
convert the png data provided by InvenTree to Zebra's bitmap format. 

It can output the print data either to a local printer connected to the computer via USB or to a network printer
with an IP address. The output can be configured in the InvenTree plugin user interface.  

Error handling is very basic. 

## Installation

The plugin is on pypi. Install this plugin using pip with the following command:

```
pip install inventree-zebra-plugin
```
 
## Configuration Options
### Printer Interface
Here you can chose between Local printer or network printer. Default value is a local printer.

### IP address
In case you use an IP printer set the IPv4 address here.

### Port 
In case you use an IP printer set the port number here. The default port number is 9100.

### Local Device
In case of a local printer set the device here. The plugin actually puts the data directly to the
device /dev/usb/lp0. No printer spooler is involved so far. 

### Threshold 
The image from pillow comes in greyscale. The plugin converts it ti pure BW because this gives a much 
better print result. The threshold between black and white can be adjusted here.

### Darkness 
This is a value that influences the darkness of the print. Allowed values are 0 (white) to 30 (black).
It is directly converted to a SD command in ZPL. If your black areas tend to blur out reduce the 
darkness.

### Dots per mm 
This sets the resolution of the printer. You can choose between 8, 12 and 24
dpmm depending on your printer model.

### Printer init 
This string added to the printer output. It can be used to set special commands
e.g. label rotation, mirror or white on black. Please refer to the ZPL manual
for more information.

Zebra printers store settings after printing. So if a rotated label has been
printed all following label will be rotated unless you change it. The default
sets the printer to settings that have been useful for me. You might want to
change it according to your requirements. Please keep in mind that this string
is passed directly to the printer without any checks. So be careful when editing
here.

## Label Template
The label needs a template described in html and css. The template should start with a page definition
that defines the label size as shown below: 

```
    @page {
        {% localize off %}
        height: {{ height }}mm;
        width: {{ width }}mm;
        {% endlocalize %}
        padding: 0mm;
        margin: 0px 0px 0px 0px;
        background-color: white;
    }
```

The height and width parameters are defined in the InvenTree admin panel in the label section. These values
have to fit the label size that is in the printer. See the example templates for details on template definition.

## How it works
First import all the stuff you need. Here we use the translation mechanism from Django for multi language support.
The import the InvenTree libs and everything you need for plugin. Here we have ZPL for the Zebra bitmaps and socket
for the IP connection to the printer. 

The next part is this:

```python
class ZebraLabelPlugin(LabelPrintingMixin, SettingsMixin, IntegrationPluginBase):

    AUTHOR = "Michael Buchmann"
    DESCRIPTION = "Label printing plugin for Zebra printers"
    VERSION = ZEBRA_PLUGIN_VERSION
    NAME = "Zebra"
    SLUG = "zebra"
    TITLE = "Zebra Label Printer"
```

The name of the class can be freely chosen. You reference to it in the entry_points section of the setup.py file.
The parameters need to be like in the example. Then there is the description block. The keywords are fixed and 
need to be like that. The values are found in the UI as shown in the pictures below.

![Admin](https://github.com/SergeoLacruz/inventree-zebra-plugin/blob/master/pictures/plugin_admin.png)
![Config](https://github.com/SergeoLacruz/inventree-zebra-plugin/blob/master/pictures/plugin.png)


Then we add the configuration parameters.
```python
SETTINGS = {
        'CONNECTION': {
            'name': _('Printer Interface'),
            'description': _('Select local or network printer'),
            'choices': [('local','Local printer e.g. USB'),('network','Network printer with IP address')],
            'default': 'local',
        },
        'PORT': {
            'name': _('Port'),
            'description': _('Network port in case of network printer'),
            'default': '9100',
        },
    }

```

We need to define a dict with the name SETTINGS. Please be aware the keys need to be in all CAPITAL letters like CONNECTION.
Simple parameters are just text strings like the port. We can set a default. The name and description shows up in the UI. 
Instead of a simple text we can also use choices. The first string like "local" it the key you use in the code. The second
one is the description in the UI. 
After that we need to define a function:

```python
def print_label(self, **kwargs){
```

The kwargs is a dict with the following keys:

- pdf_data
- user
- filename
- label_instance
- item_instance
- width
- height
- png_file

The item_instance is the part to be printed. This allows direct access to all part data. The arguments width and height 
come from the settings of the label in the admin interface. NOT from the html template. 
For the Zebra printer we use the png_file. This is a PIL (python Pillow) object with the graphic of the label in PNG format. 
The PIL object is a greyscale image. Because the printer can just print pure BW we convert this to a BW picture. 

```python
fn = lambda x : 255 if x > Threshold else 0
label_image = label_image.convert('L').point(fn, mode='1')
```

The threshold can by modified by a plugin parameter. 200 is a good starting value.  This trick gives much better prints. 
We can put the result of this directly into the ZPL library. 

```python
l = zpl.Label(Height, Width, dpmm)
l.origin(0, 0)
...
l.write_graphic(label_image, Width)
l.endorigin()
```

Width and Height define is the size of the label in millimeters as described above.
The third parameter is the resolution of the printer in dots per mm. 
write_graphic converts the pillow data to ZPL. 

The plugin was tested with a labels of various sizes defined using css and html. The DPI scaling
can be chosen in the InvenTree settings. 800 is a good value because it gives high quality.

The rest of the code is just output to the printer on different interfaces.

## Quality matters 
The InvenTree printer system uses a graphical representation of the label. The label is described
in HTML, converted to a pixel graphic and printed. The advantage is independency from  printer
models and systems. Disadvantage is larger data and quality problems with darkness and scaling.
Let's have a look at the following printout:

![QRcodes](https://github.com/SergeoLacruz/inventree-zebra-plugin/blob/master/pictures/qr.png)

Both codes have been printed with the same printer on the same reel. The left one is 
hardly readable using my mobile. The right one reads easily even as it is smaller. 

### Secret 1, Scale
The printer resolution is 8 dots per mm resulting in a dot size of 0.125mm. The QR code pixel 
and the printer pixel size should be integrally divisible. The code in the picture has 21
pixels plus one in the frame, so 23 pixel. The frame is set in the HTML description. 

```
{% qrcode qr_data border=1 %}
```

I selected two dots per pixel. So 23 * 2 * 0.125 = 6.125mm. If the size is something different
scaling takes place and the result might be worse. If you like a larger printout select more 
dots per pixel. From a certain size upwards the value does not matter any more because the code
gets large enough to be readable in any quality. 

### Secret 2: Darkness
Zebra printers allow to set the darkness of the print in values between 0 (white) and 30 (max)
The left code was printed with a value 0r 30. The black dots tend to blur out a bit resulting
in smaller white areas. The right code was printed with a value of 25 resulting in larger white
pixels.  The darkness values are just examples. Your values will differ based on printer model,
media type and printer age. The printer head tends to wear out and the darkness value might
need an adjustment from time to time. 

### Alternative
You can also bypass the InvenTree template and printing system and directly create ZPL from 
the parts data. The printer knows best how to render the label and the print quality is best.
If you are interested in this way have a look at the [inventree-zpl-plugin](https://github.com/yellowcrescent/inventree-zpl-plugin) 
that does exactly that. 

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/SergeoLacruz/inventree-zebra-plugin",
    "name": "inventree-zebra-plugin",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.6",
    "maintainer_email": "",
    "keywords": "inventree label printer printing inventory",
    "author": "Michael Buchmann",
    "author_email": "michael@buchmann.ruhr",
    "download_url": "https://files.pythonhosted.org/packages/ac/de/5b02843ada01f8e524ad57673f50afbbd7aabeb736e8f432b5efdbca3e3d/inventree-zebra-plugin-0.5.2.tar.gz",
    "platform": null,
    "description": "[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)\n\n\n# inventree-zebra-plugin\n\nThis is a label printing plugin for [InvenTree](https://inventree.org), which provides support for Zebra Label printers .\nIt was only tested with  GK420T but should work for other ZPL printers too. It uses the ZPL library to\nconvert the png data provided by InvenTree to Zebra's bitmap format. \n\nIt can output the print data either to a local printer connected to the computer via USB or to a network printer\nwith an IP address. The output can be configured in the InvenTree plugin user interface.  \n\nError handling is very basic. \n\n## Installation\n\nThe plugin is on pypi. Install this plugin using pip with the following command:\n\n```\npip install inventree-zebra-plugin\n```\n \n## Configuration Options\n### Printer Interface\nHere you can chose between Local printer or network printer. Default value is a local printer.\n\n### IP address\nIn case you use an IP printer set the IPv4 address here.\n\n### Port \nIn case you use an IP printer set the port number here. The default port number is 9100.\n\n### Local Device\nIn case of a local printer set the device here. The plugin actually puts the data directly to the\ndevice /dev/usb/lp0. No printer spooler is involved so far. \n\n### Threshold \nThe image from pillow comes in greyscale. The plugin converts it ti pure BW because this gives a much \nbetter print result. The threshold between black and white can be adjusted here.\n\n### Darkness \nThis is a value that influences the darkness of the print. Allowed values are 0 (white) to 30 (black).\nIt is directly converted to a SD command in ZPL. If your black areas tend to blur out reduce the \ndarkness.\n\n### Dots per mm \nThis sets the resolution of the printer. You can choose between 8, 12 and 24\ndpmm depending on your printer model.\n\n### Printer init \nThis string added to the printer output. It can be used to set special commands\ne.g. label rotation, mirror or white on black. Please refer to the ZPL manual\nfor more information.\n\nZebra printers store settings after printing. So if a rotated label has been\nprinted all following label will be rotated unless you change it. The default\nsets the printer to settings that have been useful for me. You might want to\nchange it according to your requirements. Please keep in mind that this string\nis passed directly to the printer without any checks. So be careful when editing\nhere.\n\n## Label Template\nThe label needs a template described in html and css. The template should start with a page definition\nthat defines the label size as shown below: \n\n```\n    @page {\n        {% localize off %}\n        height: {{ height }}mm;\n        width: {{ width }}mm;\n        {% endlocalize %}\n        padding: 0mm;\n        margin: 0px 0px 0px 0px;\n        background-color: white;\n    }\n```\n\nThe height and width parameters are defined in the InvenTree admin panel in the label section. These values\nhave to fit the label size that is in the printer. See the example templates for details on template definition.\n\n## How it works\nFirst import all the stuff you need. Here we use the translation mechanism from Django for multi language support.\nThe import the InvenTree libs and everything you need for plugin. Here we have ZPL for the Zebra bitmaps and socket\nfor the IP connection to the printer. \n\nThe next part is this:\n\n```python\nclass ZebraLabelPlugin(LabelPrintingMixin, SettingsMixin, IntegrationPluginBase):\n\n    AUTHOR = \"Michael Buchmann\"\n    DESCRIPTION = \"Label printing plugin for Zebra printers\"\n    VERSION = ZEBRA_PLUGIN_VERSION\n    NAME = \"Zebra\"\n    SLUG = \"zebra\"\n    TITLE = \"Zebra Label Printer\"\n```\n\nThe name of the class can be freely chosen. You reference to it in the entry_points section of the setup.py file.\nThe parameters need to be like in the example. Then there is the description block. The keywords are fixed and \nneed to be like that. The values are found in the UI as shown in the pictures below.\n\n![Admin](https://github.com/SergeoLacruz/inventree-zebra-plugin/blob/master/pictures/plugin_admin.png)\n![Config](https://github.com/SergeoLacruz/inventree-zebra-plugin/blob/master/pictures/plugin.png)\n\n\nThen we add the configuration parameters.\n```python\nSETTINGS = {\n        'CONNECTION': {\n            'name': _('Printer Interface'),\n            'description': _('Select local or network printer'),\n            'choices': [('local','Local printer e.g. USB'),('network','Network printer with IP address')],\n            'default': 'local',\n        },\n        'PORT': {\n            'name': _('Port'),\n            'description': _('Network port in case of network printer'),\n            'default': '9100',\n        },\n    }\n\n```\n\nWe need to define a dict with the name SETTINGS. Please be aware the keys need to be in all CAPITAL letters like CONNECTION.\nSimple parameters are just text strings like the port. We can set a default. The name and description shows up in the UI. \nInstead of a simple text we can also use choices. The first string like \"local\" it the key you use in the code. The second\none is the description in the UI. \nAfter that we need to define a function:\n\n```python\ndef print_label(self, **kwargs){\n```\n\nThe kwargs is a dict with the following keys:\n\n- pdf_data\n- user\n- filename\n- label_instance\n- item_instance\n- width\n- height\n- png_file\n\nThe item_instance is the part to be printed. This allows direct access to all part data. The arguments width and height \ncome from the settings of the label in the admin interface. NOT from the html template. \nFor the Zebra printer we use the png_file. This is a PIL (python Pillow) object with the graphic of the label in PNG format. \nThe PIL object is a greyscale image. Because the printer can just print pure BW we convert this to a BW picture. \n\n```python\nfn = lambda x : 255 if x > Threshold else 0\nlabel_image = label_image.convert('L').point(fn, mode='1')\n```\n\nThe threshold can by modified by a plugin parameter. 200 is a good starting value.  This trick gives much better prints. \nWe can put the result of this directly into the ZPL library. \n\n```python\nl = zpl.Label(Height, Width, dpmm)\nl.origin(0, 0)\n...\nl.write_graphic(label_image, Width)\nl.endorigin()\n```\n\nWidth and Height define is the size of the label in millimeters as described above.\nThe third parameter is the resolution of the printer in dots per mm. \nwrite_graphic converts the pillow data to ZPL. \n\nThe plugin was tested with a labels of various sizes defined using css and html. The DPI scaling\ncan be chosen in the InvenTree settings. 800 is a good value because it gives high quality.\n\nThe rest of the code is just output to the printer on different interfaces.\n\n## Quality matters \nThe InvenTree printer system uses a graphical representation of the label. The label is described\nin HTML, converted to a pixel graphic and printed. The advantage is independency from  printer\nmodels and systems. Disadvantage is larger data and quality problems with darkness and scaling.\nLet's have a look at the following printout:\n\n![QRcodes](https://github.com/SergeoLacruz/inventree-zebra-plugin/blob/master/pictures/qr.png)\n\nBoth codes have been printed with the same printer on the same reel. The left one is \nhardly readable using my mobile. The right one reads easily even as it is smaller. \n\n### Secret 1, Scale\nThe printer resolution is 8 dots per mm resulting in a dot size of 0.125mm. The QR code pixel \nand the printer pixel size should be integrally divisible. The code in the picture has 21\npixels plus one in the frame, so 23 pixel. The frame is set in the HTML description. \n\n```\n{% qrcode qr_data border=1 %}\n```\n\nI selected two dots per pixel. So 23 * 2 * 0.125 = 6.125mm. If the size is something different\nscaling takes place and the result might be worse. If you like a larger printout select more \ndots per pixel. From a certain size upwards the value does not matter any more because the code\ngets large enough to be readable in any quality. \n\n### Secret 2: Darkness\nZebra printers allow to set the darkness of the print in values between 0 (white) and 30 (max)\nThe left code was printed with a value 0r 30. The black dots tend to blur out a bit resulting\nin smaller white areas. The right code was printed with a value of 25 resulting in larger white\npixels.  The darkness values are just examples. Your values will differ based on printer model,\nmedia type and printer age. The printer head tends to wear out and the darkness value might\nneed an adjustment from time to time. \n\n### Alternative\nYou can also bypass the InvenTree template and printing system and directly create ZPL from \nthe parts data. The printer knows best how to render the label and the print quality is best.\nIf you are interested in this way have a look at the [inventree-zpl-plugin](https://github.com/yellowcrescent/inventree-zpl-plugin) \nthat does exactly that. \n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Zebra label printer plugin for InvenTree",
    "version": "0.5.2",
    "project_urls": {
        "Homepage": "https://github.com/SergeoLacruz/inventree-zebra-plugin"
    },
    "split_keywords": [
        "inventree",
        "label",
        "printer",
        "printing",
        "inventory"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "e6b8b9df84798c16ae71c8cef04106cab7eab3377d2a4990d037030ec341e3c0",
                "md5": "3e1de57ec645a704fbe632bd6a28f9a2",
                "sha256": "faea8daba731accff78db7f237793b55d7ca7cc5833c040a9d39d7901560da42"
            },
            "downloads": -1,
            "filename": "inventree_zebra_plugin-0.5.2-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "3e1de57ec645a704fbe632bd6a28f9a2",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.6",
            "size": 8632,
            "upload_time": "2024-03-15T14:44:01",
            "upload_time_iso_8601": "2024-03-15T14:44:01.403686Z",
            "url": "https://files.pythonhosted.org/packages/e6/b8/b9df84798c16ae71c8cef04106cab7eab3377d2a4990d037030ec341e3c0/inventree_zebra_plugin-0.5.2-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "acde5b02843ada01f8e524ad57673f50afbbd7aabeb736e8f432b5efdbca3e3d",
                "md5": "2453b2576fa7ec937b3f75326bb0d89b",
                "sha256": "84c6b7d9c1d8d56f1ca39d7c360beb04a487a819e00157249d9635672d3b317a"
            },
            "downloads": -1,
            "filename": "inventree-zebra-plugin-0.5.2.tar.gz",
            "has_sig": false,
            "md5_digest": "2453b2576fa7ec937b3f75326bb0d89b",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.6",
            "size": 7512,
            "upload_time": "2024-03-15T14:44:02",
            "upload_time_iso_8601": "2024-03-15T14:44:02.671649Z",
            "url": "https://files.pythonhosted.org/packages/ac/de/5b02843ada01f8e524ad57673f50afbbd7aabeb736e8f432b5efdbca3e3d/inventree-zebra-plugin-0.5.2.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-03-15 14:44:02",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "SergeoLacruz",
    "github_project": "inventree-zebra-plugin",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "inventree-zebra-plugin"
}
        
Elapsed time: 0.21760s