@barcode(FORMAT,WIDTH,HEIGHT,CONTENT)@
**NOTE:** Don't put any spaces before or after the surrounding parentheses or the commas separating the fields!
The following values are supported in place of the ''FORMAT'' parameter:
  * CODEBAR
  * CODE_39
  * CODE_93
  * CODE_128
  * DATA_MATRIX
  * EAN_8
  * EAN_13
  * QR_CODE
  * UPC_A
  * UPC_E
The values given for the ''WIDTH'' and ''HEIGHT'' parameters are merely guidelines for the barcode generator.  If the generated barcode doesn't fit in the given width/height, they will be exceeded.
For the ''CONTENT'' parameter, you can use any other placeholder, free text, or a combination thereof.
Here's a full example:
@barcode(EAN_13,100,50,@itemBarcodeGTIN13@)@
This placeholder will be replaced with an EAN-13 barcode, containing the primary barcode of the item according to the barcodes table in the item master data in SAP Business One.
Since we've used the placeholder ''itemBarcodeGTIN13'', the barcode of the item may actually be a GTIN-12, or it may be saved as a GTIN-14 in SAP Business One with a leading zero.  In both cases, the EAN-13 barcode will be generated just fine, since the ''itemBarcodeGTIN13'' adds or removes zeroes as necessary.  If the conversion to the correct number of digits is not possible (e.g. barcode is a full-length GTIN-14, so there's no leading zero to remove), then the value ''0000000000000'' (13 zeroes) will be used instead; this way it's ensured that an EAN-13 barcode is still generated on the label without messing up the whole print layout.  The printed barcode will of course be non-functional.
=== GS1 Support ===
The formats ''CODE_128'', ''DATA_MATRIX'', and ''QR_CODE'' support generating [[GS1 barcodes]].
To make use of this, the contents of the barcode must begin with the vertical bar symbol: **''|''**
GS1 fields with variable length that need to be terminated are likewise terminated with a vertical bar symbol.
Example:
@barcode(CODE_128,100,75,|01@itemBarcodeGTIN14@10@batchNumber@|...)@
In this example, the first GS1 barcode field is ''01'' which symbolizes that a GTIN-14 follows.  By using the placeholder ''itemBarcodeGTIN14'' we ensure that exactly 14 digits are inserted here.  This field does not need to be terminated with a vertical bar symbol, since it has a fixed length of 14 digits.
The next field is ''10'' i.e. the batch number.  This field is allowed to contain 1 to 20 digits or arbitrary characters.  If we use batch numbers of 10 characters, this means we must end this GS1 field explicitly with a vertical bar symbol.
Here's a list of commonly used GS1 field identifiers:
https://www.activebarcode.de/codes/ean128_ucc128_ai.html
Here's a full list from GS1.org currently containing 480 entries:
https://www.gs1.org/standards/barcodes/application-identifiers
=== Better control of generated HTML ===
The ''barcode()'' placeholder generates a whole ''
@) 
The ''barcodeBase64'' placeholder generates only the Base64 string.  Example:
@) 
The parameters of the ''barcodeSrc()'' and ''barcodeBase64()'' placeholders are exactly the same as that of the regular ''barcode()'' placeholder.  That means you must still provide the ''WIDTH'' and ''HEIGHT'' parameters; these will be passed to the barcode generation system that produces the Base64 PNG.  (This has no effect on the width and height of the ''img'' element in HTML/CSS.)
==== User-input values ====
Label templates may contain a special type of placeholder that tells the app to show the user a popup for values to insert manually every time the label is to be printed.  The simplest syntax for this type of placeholder looks as follows:
@input(NAME)@
The ''NAME'' parameter uniquely identifies the input and may only consist of normal English letters and digits.  (No spaces, special characters, umlauts, etc.)
The following logic is applied to label templates with input placeholders:
  * The template is scanned for all input placeholders.
  * Whenever such a placeholder is found and the ''NAME'' has not been encountered before, it's registered into a list of needed inputs.
  * The user is then presented with a popup that contains one field per needed input.  The value provided by the user is saved.
  * Finally, all input placeholders are replaced with the value that the user had entered for them.  Placeholders with the same ''NAME'' get the same value.
Since the name of an input may not contain arbitrary text, an optional "display name" may be specified in square brackets, which will be used in the popup where the user has to enter the value for that input.  Example:
@input(quantity[Number of crates])@
In this example, the input is identified by the simple name ''qty'' but the COBI.wms user will see it as "Number of crates" in the input values popup.  If no display name is specified in square brackets, then the regular name of the input is also used as the display name.
Also, an input may be limited to a predefined set of choices by listing them in the following format:
@input(Currency;EUR,GBP,USD)@
**NOTE:** Don't put any spaces after the semicolon, or before or after the commas.
This means that in the input popup shown to the user, there will be a drop-down list of selections instead of a free-form text field.
The display name and the list of choices can be combined:
@input(shipType[Freight type];Less Than Truckload,Full Truckload,Air Freight,Ocean Freight)@
The configuration for an input (display name and list of values) only has to appear for the first time the input with that ''NAME'' is encountered.  A good strategy is to list all the inputs with their configuration at the top of the HTML file within an HTML comment, and only refer to them as ''@input(NAME)@'' in the main body of the HTML, to make the template easier to read.  Refer to the second example template for an example of this strategy.
==== Date placeholders ====
You can insert the date and/or time of the moment of the printing by using the following special placeholder:
@date(FORMAT)@
Valid values for the ''FORMAT'' parameters correspond to the parameters of the Android class ''SimpleDateFormat'' which is documented in the following page:
https://developer.android.com/reference/java/text/SimpleDateFormat
However, for your convenience, here are the values that are likely to be used most commonly:
| yyyy | Year, 4 digits |
| yy   | Year, 2 digits |
| MM   | Month number, 2 digits |
| dd   | Day of month, 2 digits |
| HH   | Hour of day, 2 digits |
| mm   | Minute of hour, 2 digits |
| ss   | Second of minute, 2 digits |
Examples follow.  Let's say the date is February 23rd, 2021, and the clock reads 16:11.
''@date(yyyy-MM-dd HH:mm)@'' becomes ''2021-02-23 16:11''
''@date(dd.MM.yyyy)@'' becomes ''23.02.2021''
==== Date reformatting ====
The special date-reformat placeholder can be used to interpret a date in one format and then have it printed in another format.  The usage looks as follows:
@dateReformat(DATE|FROM_FORMAT|TO_FORMAT)@
**NOTE:** Don't put any spaces before or after the parentheses or the separating vertical bar symbols.  (The spaces would be considered part of the corresponding parameter.)
This placeholder uses the vertical bar symbol instead of commas to separate its parameters, because the parameters themselves may contain commas.
The mechanism is as follows: the text found within the ''DATE'' parameter is interpreted in accordance to the format specification ''FROM_FORMAT'' so the software knows what date/time it represents, and it's then turned into ''TO_FORMAT''.
The ''FROM_FORMAT'' and ''TO_FORMAT'' parameters are akin to the ''FORMAT'' parameter of the regular ''date()'' placeholder described in the previous section.
Following is an example for printing the expiry date of a batch number in a custom format, exploiting the fact that the placeholder ''@expiryDateGS1@'' will always produce a date of the format ''yyMMdd''.
@dateReformat(@batchExpiryGS1@,yyMMdd,dd.MM.yyyy)@
===== JavaScript bindings =====
The HTML document resulting from the interpretation of a COBI.wms label template is rendered by a full-fledged browser engine.  This means that you can use not only CSS but even JavaScript in the template.  However, this has to be enabled explicitly by placing the following pseudo-placeholder anywhere in the file, such as in an HTML comment somewhere at the top:
@useJavaScript@
**NOTE:** When JS is enabled, the actual printing of the rendered HTML document will not be triggered automatically anymore.  Instead, you have to trigger the printing manually from your JS code by calling ''cobiwms.print()'' at some point.
During execution of your JS code, you have access to the special object ''cobiwms'' which contains a number of functions.  Their explanation follows.
cobiwms.get(name)
The parameter ''name'' must be a string.  It will be interpreted as a placeholder, and its value returned as a string.  For example, calling ''cobiwms.get("itemName")'' will give you the item name as a string.
(You could also just use placeholders like ''@itemName@'' in your code, but then the value would be injected as-is into your code.  That means: you would have to wrap it in quotation marks like ''"@itemName@"'', and any quotation marks appearing within the actual value would break the code.  Therefore it's much better to use ''cobiwms.get()'' instead.)
cobiwms.prompt(title, callback)
Calling this function makes the app present a popup with an input field to the user.  The parameter ''title'', which must be a string, will be the title of the popup.  The parameter ''callback'' must also be a string, and must represent a function.  Said function will be called with one argument (a string representing the value entered by the user) after the user has confirmed their input.
Example usage:
window.handleCurrency = function (currency) {
    ...
}
cobiwms.prompt("Enter currency", "window.handleCurrency")
===== Example template files =====
The following example files serve to demonstrate the syntax and some of the features.
==== File 1 ====
This is a very simple template demonstrating the use of placeholders to print the item code and name, price, and a timestamp.
    
    
    
        Code: @itemCode@
    
    
        Price: @itemPriceCurrencySymbol(1)@ @itemPriceFormattedValue(1)@
    
    
        @itemName@
    
    
        @barcode(CODE_128,150,75,|90@itemCode@|10@batchNumber@)@
    
    
        (90)@itemCode@(10)@batchNumber@
    
    
        @date(yyyy-MM-dd HH:mm)@
    
    
        COBI.wms Sample Label
    
==== File 2 ====
This file demonstrates the use of user-input values and JavaScript integration.
    
    
    
    
    
        JS-Insert: 
    
    
        Selection: @input(selection)@
    
    
        @itemName@
    
    
        @barcode(@input(codeFormat)@,150,75,|90@itemCode@|10@batchNumber@)@
    
    
        (90)@itemCode@(10)@batchNumber@
    
    
        Input: @input(text)@
    
    
        COBI.wms Sample Label