How to print raw BOCA FGL commands from Blazor
Product JSPrintManager for Blazor Published 07/07/2021 Updated 11/28/2024 Author Neodynamic
Overview
FRIENDLY GHOST LANGUAGE (FGL) is Boca's industry standard language that allows developers to print data, graphics, boxes and bar codes anywhere on a ticket or label in virtually any orientation.
BOCA FGL commands are very simple and text plain! The main advantage of using raw BOCA FGL commands for printing instead of using the built-in browser javascript printing (window.print();) is that the printing performance will be way faster; a factor that is key in the aforementioned scenarios. You'll be able to use raw printing feature with the help of our JSPrintManager for Blazor solution that was specially designed for this kind of printing needs.
In this walkthrough, you'll learn how to print raw FGL commands from Blazor directly to the client printer without displaying a print dialog at all. You'll be able to print BOCA FGL commands to the Default client printer as well as to any other installed printer at the client machine. This solution works with any popular browser like Chrome, Firefox, IE/Edge & Safari on Windows, Linux, Raspberry Pi and Mac systems!
The BOCA FGL commands that we'll use in this article will print out a ticket passport that will look like this:
A Sample Ticket Passport printed from Blazor and created by using BOCA FGL commands
Follow up these steps
- Be sure you install in your dev machine JSPrintManager (JSPM) (Available for Windows, Linux, Raspberry Pi & Mac)
This small app must be installed on each client that will print from your website! - In your Blazor project...
- Add a NuGet reference to the JSPrintManager Razor Component
- Add the JSPrintManager service...
- Add the following statement at the top of your
Startup
file
using Neodynamic.Blazor;
- For Blazor Server
Add the following line in theStartup's ConfigureServices
method
services.AddJSPrintManager();
For .NET 8+ you must use Interactive Server components and render mode, so add these settings:
builder.Services.AddRazorComponents() .AddInteractiveServerComponents();
app.MapRazorComponents<App>() .AddInteractiveServerRenderMode()
- For Blazor WebAssembly
Add the following line in theProgram's Main
method
builder.Services.AddJSPrintManager();
- For Blazor Server
- Add the following statement at the top of your
- Add the following statement in the
_Imports.razor
file
@using Neodynamic.Blazor
- Add a new Razor Page and copy/paste the following code. Please read the source code comments to understand the printing logic!
For .NET 8+ Blazor Server you must use Interactive Server render mode, so add these settings:
@page "/" @rendermode InteractiveServer @inject JSPrintManager JSPrintManager
@page "/" @inject JSPrintManager JSPrintManager <div> <strong>JSPM </strong><span>WebSocket Status </span> @if (JSPrintManager.Status == JSPMWSStatus.Open) { <span class="badge badge-success"> <i class="fa fa-check" /> Open </span> } else if (JSPrintManager.Status == JSPMWSStatus.Closed) { <span class="badge badge-danger"> <i class="fa fa-exclamation-circle" /> Closed! </span> <div> <strong>JSPrintManager (JSPM) App</strong> is not installed or not running! <a href="https://neodynamic.com/downloads/jspm" target="_blank">Download JSPM Client App...</a> </div> } else if (JSPrintManager.Status == JSPMWSStatus.Blocked) { <span class="badge badge-warning"> <i class="fa fa-times-circle" /> This Website is Blocked! </span> } else if (JSPrintManager.Status == JSPMWSStatus.WaitingForUserResponse) { <span class="badge badge-warning"> <i class="fa fa-user-circle" /> Waiting for user response... </span> } </div> @if (JSPrintManager.Status == JSPMWSStatus.Open) { @if (JSPrintManager.Printers == null) { <hr /> <div class="spinner-border text-info" role="status"> <span class="sr-only">Please wait...</span> </div> <strong><em>Getting local printers...</em></strong> } else { <div> <h1>Print BOCA FGL commands from Blazor</h1> <hr /> <EditForm Model="@MyPrinter"> <div class="form-check"> <InputCheckbox class="form-check-input" @bind-Value="UseDefaultPrinter" />Print to <strong>Default Printer?</strong><br /> </div> <p>or...</p> <p>Select an <strong>Installed Printer</strong>:</p> <InputSelect @bind-Value="MyPrinter.PrinterName" class="form-control form-control-sm"> @foreach (var p in JSPrintManager.Printers) { <option value="@p">@p</option> } </InputSelect> </EditForm> <br /><br /> <input type="button" value="Print Now..." @onclick="DoPrint" /> </div> } } @code { // An Installed Printer instance private InstalledPrinter MyPrinter { get; set; } = new(); // Use default printer? private bool UseDefaultPrinter = false; // Printing... private void DoPrint() { // Status = Open means that JSPM Client App is up and running! if (JSPrintManager.Status == JSPMWSStatus.Open) { // Create a ClientPrintJob var cpj = new ClientPrintJob(); // Set target Printer cpj.ClientPrinter = UseDefaultPrinter ? new DefaultPrinter() : MyPrinter; // Create BOCA FGL commands for sample label var cmds = ""; cmds += "<RL><RC360,10><F3><HW1,1>GHOSTWRITER WORLD"; cmds += "<RC380,76><F6><HW1,1><BS26,44>ALL<F2>"; cmds += "<F6><BS26,44>THREE<F2> <F6><BS26,44>PARKS"; cmds += "<RC348,130><F6><HW1,1><BS42,44>PASSPORT"; cmds += "<RC324,240><RL><F6><HW2,2>6"; cmds += "<RC210,240><HW1,1>DAY"; cmds += "<RC230,290><F3><HW1,1>ADMIT ONE"; cmds += "<RC230,320><F6><HW1,1>GUEST"; cmds += "<F1><RC230,370><F1><HW1,1>VERY SMALL PRINT"; cmds += "<RC24,530><LT2><BX340,50>"; cmds += "<RC25,528><LT2><VX338>"; cmds += "<RC216,550><HW1,1><F2>DAY 1"; cmds += "<RC24,580><LT2><BX340,50>"; cmds += "<RC216,600>DAY 2"; cmds += "<RC24,630><LT2><BX340,50>"; cmds += "<RC216,650>DAY 3"; cmds += "<RC24,680><LT2><BX340,50>"; cmds += "<RC216,700>DAY 4"; cmds += "<RC24,730><LT2><BX340,50>"; cmds += "<RC25,780><LT2><VX338>"; cmds += "<RC216,760>DAY 5"; cmds += "<RC340,400><RL><F6><BS36,44><HW1,1>DAY GUEST"; cmds += "<RC260,450><F3><HW1,1>$112.00"; cmds += "<RC240,482><F3>PLUS TAX"; cmds += "<RC280,1010><F3><HW1,1>12345678"; cmds += "<RC60,990><NL10><X2>*01000407*"; cmds += "<RC360,820><F9><HW1,1>VALID ONLY ON DATE STAMPED"; cmds += "NONTRANSFERABLE NONREFUNDABLE"; cmds += "<RC280,870><F3><HW1,1>01000407"; cmds += "<RC20,1079><RR><F3><HW1,1>GHOSTWRITER WORLD"; cmds += "<p>"; // Set the RAW commands to send to the printer... cpj.PrinterCommands = cmds; // PRINT IT!!! JSPrintManager.SendClientPrintJob(cpj); } } protected override void OnAfterRender(bool firstRender) { if (firstRender) { // Handle OnGetPrinters event... JSPrintManager.OnGetPrinters += () => { if (JSPrintManager.Printers != null && JSPrintManager.Printers.Length > 0) { // Display installed printers... StateHasChanged(); } else { Console.WriteLine("No printers found..."); } }; // Handle OnStatusChanged event to detect any WSS status change JSPrintManager.OnStatusChanged += () => { StateHasChanged(); // Status = Open means that JSPM Client App is up and running! if (JSPrintManager.Status == JSPMWSStatus.Open) { //Try getting local printers... JSPrintManager.TryGetPrinters(); } }; // Start WebSocket comm JSPrintManager.Start(); } base.OnAfterRender(firstRender); } }
- That's it! Run your website and test it. Click on Print Now... to print the BOCA FGL Commands without print dialog. You can print it to the Default client printer or you can get a list of the installed printers available at the client machine.
NOTE You can also print directly to any LPT Parallel Port, RS232 Serial Port or IP/Ethernet Printer although these scenarios have not been considered in this article for simplicity. For further details on those scenarios, please contact our tech support.