Skip to content

ysfchn/dymo-bluetooth

Repository files navigation

dymo-bluetooth

Use DYMO LetraTag LT-200B thermal label printer over Bluetooth in Python, without depending on its app.

I was not expecting to own a label printer, but after caming across to this online, I thought why I shouldn't get it for whatever reason since it was affordable. After a while, I decided to tinker with this printer so I can print anything whatever I would like instead of being limited with its mobile app, so here it is.

The image on the printed label shown in the picture can be found here.

More about this printer model

This printer doesn't use inks, it instead writes on the labels with thermal. And obviously, it does require a label cartridge (or called "casette"). From what I've seen, all LetraTag branded label casettes have the same same shape and dimensions, so it is easy to get another if the casette that came with the printer runs out. There are different kind of labels (paper, plastic, metalic etc.) available, and all of them are self-adhesive. The casettes don't have an electronical part, which makes the printer to be able to use casettes manufactured by someone else.

The label dimensions are 12 millimeter in height, however the printable area is just 32 pixels, so labels will have an very obvious white space even if all printable space is filled with black. The print resolution is 200 DPI and, as stated on the original website, it has a print speed of up to 7 millimeters per second. So unfortunately the print resolution is not perfect for everything, but still, it does its job, and in the end, I enjoyed playing around with this printer, even if it was too simple.

Installation

python -m pip install --upgrade https://github.com/ysfchn/dymo-bluetooth/archive/refs/heads/main.zip

Python 3.10 or up is targeted, but 3.9 should work too. It depends on bleak for Bluetooth communication, and pillow for importing images from files. If python-barcode is installed, it can be used to print barcodes.

Usage

There is a small CLI to print image files in a nearby printer.

python -m dymo_bluetooth --help

Or in a Python program:

from dymo_bluetooth import discover_printers, Canvas
import asyncio

async def main():
    # Canvas class allows constructing 1-bit images in ease.
    canvas = Canvas()

    # Pixels can be set and retrieved with methods.
    canvas.set_pixel(0, 0, True)

    # Images needs to be stretched by at least 2 times like
    # how its mobile app does, otherwise printer will render
    # the labels too thin.
    canvas = canvas.stretch_image(2)

    # Get a list of printers.
    printers = await discover_printers()

    # Get the first found printer and print the 
    # constructed Canvas. Returns the print status.
    await printers[0].print(canvas)

asyncio.run(main())

Disclaimer

This program is licensed under MIT License.

It is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See MIT license for details.

This program is not supported, sponsored, affiliated, approved, or endorsed in any way with DYMO. LetraTag is an trademark of DYMO. All other mentioned trademarks are the property of respective owners.

Protocol

This printer uses Bluetooth LE (GATT) to send labels & retrieve its status. It does uses this service UUIDs:

UUID Used for
be3dd650-2b3d-42f1-99c1-f0f749dd0678 Service
be3dd651-2b3d-42f1-99c1-f0f749dd0678 Print request (write -> printer)
be3dd652-2b3d-42f1-99c1-f0f749dd0678 Print reply (notify <- printer)
be3dd653-2b3d-42f1-99c1-f0f749dd0678 Unknown

2b3d part in the UUID may not be future-proof, but so far all produced printers of the same model seems to share the same UUID.

To discover the nearby printers, filter for the service UUID or/and check for the MAC address range. This printer has a MAC address in range of 58:CF:79:00:00:00 and 58:CF:79:FF:FF:FF, which this MAC range block seems to be owned by Espressif Inc. according to Bluetooth vendor list.

All data structures are explained below so it can be used as a future reference.

Directive

Directives are simply command types that found in the payload. A directive directly follows another directive, and so on. A payload may contain more than one directive.

Each directive consists of 2 bytes, first byte is always an ASCII escape character 1B (in hexadecimal, or 27 in decimal), and second byte is the ASCII code point of the character that specifies the command type.

COMMAND_TYPE[1] = 73 | 4D | 43 | 44 | 45 | 41 | 51
DIRECTIVE[2] = 1B + COMMAND_TYPE[1]
Command Character = Code point (hex) = Code point (decimal) Notes
START s 0x73 115 Start command. Each payload starts with this directive.
MEDIA_TYPE M 0x4d 77 ?
PRINT_DENSITY C 0x43 67 Seems unused.
PRINT_DATA D 0x44 68 Defines the image that will be printed, see below for byte format.
FORM_FEED E 0x45 69 Follows after PRINT_DATA.
STATUS A 0x41 65 Requests the status (?)
END Q 0x51 81 End command. Each payload ends with this directive.

Below is the structure of each command that follows after its directive:

START command

Each payload starts with directive. This command follows with 4 constant bytes after its directive.

START[6] = DIRECTIVE[2] + 9A 02 00 00

MEDIA_TYPE command

This command follows with 1 byte containing a number of some type (?) to be set for the printer. It doesn't really seem to be used by this printer, but this command was found in its mobile app anyway.

When this number has set to anything, the printer should advertise this number when searching for Bluetooth devices nearby. (not tested) So I guess it was planned to be used for some sort of casette checking & changing casette type, so printer can hold this value even when its turned off.

MEDIA_TYPE[3] = DIRECTIVE[2] + ??

PRINT_DENSITY command

Seems unused by the printer. Not sure what it does.

PRINT_DENSITY[?] = DIRECTIVE[2] + ...

FORM_FEED command

Used after PRINT_DATA command. Doesn't take additional bytes other than its directive.

FORM_FEED[2] = DIRECTIVE[2]

STATUS command

Used after FORM_FEED command. Doesn't take additional bytes other than its directive.

STATUS[2] = DIRECTIVE[2]

END command

Each payload ends with directive. Doesn't take additional bytes other than its directive.

END[2] = DIRECTIVE[2]

PRINT_DATA command

This command defines the data of the image that will be printed to the label. It is followed by:

  • Bits per pixel value; this is always equals to 1, so in hex that would be 0x01.

  • Alignment; this is always equals to 2, so in hex that would be 0x02. This is always seems the case regardless of the content of the image, so not sure how printer will react for values other than 2.

  • Width of the image. This value consists of 4 bytes, in little-endian. Since the label is printed horizontally, it might be changed based on your input image, so continue reading about image format.

  • Height of the image. This value consists of 4 bytes, in little-endian. Since labels can only contain 32 pixels from top to the bottom, simply use 32 as a height, so in hexadecimal, that would be: 20 00 00 00

  • Image data, see "Image format".

# Printer have fixed 32 pixel size
HEIGHT[4] = 20 00 00 00

PRINT_DATA[variable] = DIRECTIVE[2] + 01 02 + WIDTH[4] + HEIGHT[4] + IMAGE_DATA[4 * WIDTH]

Image format

The printer accepts images in portrait direction, formatted as 1-bit monochrome and stored in little endian. To put it simply, it uses bits (1 and 0) to set each pixel, and since a byte contains 8 bits, the printer will require 4 bytes (4 * 8 = 32 pixels, which is the height of the printable area) to print a full line from top to the bottom.

    8 bits                                                                   4 bytes = 32 bits (8 * 4) = 1 printing line
┌─────────────┐                                                            ┌─────────┐
│             │                                                            │         │
0 0 0 0 0 0 0 0   0 0 0 0 0 0 0 0   0 0 0 0 0 0 0 0   0 0 0 0 0 0 0 0  ->  00 00 00 00  -> 1 full empty line
    binary            binary            binary            binary               hex

1 1 1 1 1 1 1 1   1 1 1 1 1 1 1 1   1 1 1 1 1 1 1 1   1 1 1 1 1 1 1 1  ->  FF FF FF FF  -> 1 full black line
    binary            binary            binary            binary               hex

Since the format is in little endian, the first left-top pixel (X = 0, Y = 0) is inserted in the last byte of 4 bytes, and the biggest bit in that byte set.

0 0 0 0 0 0 0 0   0 0 0 0 0 0 0 0   0 0 0 0 0 0 0 0   1 0 0 0 0 0 0 0  ->  00 00 00 80  -> 1 pixel in [x = 0, y = 0]
    binary            binary            binary        │   binary               hex
                                                      │
      0                 0                 0           └─── 128
   decimal           decimal           decimal           decimal

As explained above, line always take up 4 bytes, so the image width can be found by dividing the byte count to 4. And since the image height doesn't vary (labels are printed horizontally in this printer, from left to right) the image height is always 32 pixels.

Each line is directly followed by another line (if any), so there is no additional separator between lines, and it is ordered sequantally, so the first 4 bytes in the image data will be the first line that will be printed out on the label.

Payloads

There are 2 payloads that can be sent to the printer. Each payload starts with an HEADER, and it does follow with the payload data.

Payload Requires chunking? Usage
Print request Yes Sends an image to be printed out on the label.
Set media type No ?

Magic value

Magic value consists of 2 bytes, which is 12 34 (in hexadecimal) or [18, 52] in decimal.

MAGIC[2] = 12 34

Header

Header consists of 9 bytes in total. First two bytes are constant, which is FF F0. Third and forth byte is MAGIC, which is explained above. And it is followed by the length of the payload stored in 4 bytes (little-endian) that is going to follow after this header.

The ninth byte of the header is the checksum of the header, and it is calculated by simply getting sum of the all previous 8 bytes and getting the first byte of the sum.

PARTIAL_HEADER[8] = FF F0 + MAGIC[2] + PAYLOAD_LENGTH[4]

CHECKSUM = SUM(PARTIAL_HEADER) & 0xFF

HEADER[9] = PARTIAL_HEADER[8] + CHECKSUM

Print request

A print request payload consists of these directives; START, PRINT_DATA, FORM_FEED, STATUS and END respectively. Directives follow after HEADER, so it would be the first.

PRINT_REQUEST[variable] = HEADER[9] + START[6] + PRINT_DATA[variable] + FORM_FEED[2] + STATUS[2] + END[2]

Due to both the way the printer works and the way the Bluetooth itself work, this payload needs to be broken down into chunks in each 500 bytes. Each chunk must be send separately over Bluetooth, the printer will start printing until it receives the last chunk.

Chunking needs to be start right after HEADER, so only the header is sent first (which is the 9 bytes), then the each chunk is separately sent via Bluetooth. So number of the total Bluetooth writes would be 1 + chunk count.

Each chunk must contain its index before chunk data itself followed by. The chunk data can be smaller than 500, but it can only contain maximum 500 bytes, inclusive. In the end, you will have the full chunk to be sent over Bluetooth which contains the index and the data.

CHUNK[0...501] = CHUNK_INDEX[1] + CHUNK_DATA[0...500]

The last chunk must contain the MAGIC value appended end of the chunk.

CHUNK[0...503] = CHUNK_INDEX[1] + CHUNK_DATA[0...500] + MAGIC[2]

Set media type

A media type set payload consists of these directives; START, MEDIA_TYPE and END respectively. Directives follow after HEADER, so it would be the first.

SET_MEDIA_TYPE[20] = HEADER[9] + START[6] + MEDIA_TYPE[3] + END[2]

This payload doesn't require chunking and can be sent with single Bluetooth transaction.

Result

Printer will notify on the print reply UUID, giving the status of the printing operation. First byte is always an ASCII escape character 1B (in hexadecimal, or 27 in decimal), and second byte is the ASCII code point of the R character (0x52 in hexadecimal, or 82 in decimal) that stands for the result.

Third byte represents the status code. The printer may be inconsistent when it comes to the status codes, see Result enum in the code for more details about available status codes and their (supposedly) meanings.

RESULT[3] = 1B 52 + STATUS[1]

Extras

The printer will automatically print its MAC address if power button was pressed long.

About

Print labels on DYMO LetraTag LT-200B in Python.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages