PHPackages                             devxisas/qr-studio - PHPackages - PHPackages  [Skip to content](#main-content)[PHPackages](/)[Directory](/)[Categories](/categories)[Trending](/trending)[Leaderboard](/leaderboard)[Changelog](/changelog)[Analyze](/analyze)[Collections](/collections)[Log in](/login)[Sign up](/register)

1. [Directory](/)
2. /
3. devxisas/qr-studio

ActiveLibrary

devxisas/qr-studio
==================

A modern QR code studio for Laravel — SVG, PNG, EPS, styles, gradients, data types, Blade directive, response macro, and model traits.

v1.0.0(1mo ago)00[1 PRs](https://github.com/devxisas/qr-studio/pulls)MITPHPPHP ^8.2CI passing

Since Mar 17Pushed 1mo agoCompare

[ Source](https://github.com/devxisas/qr-studio)[ Packagist](https://packagist.org/packages/devxisas/qr-studio)[ Docs](https://github.com/devxisas/qr-studio)[ RSS](/packages/devxisas-qr-studio/feed)WikiDiscussions main Synced 1mo ago

READMEChangelog (1)Dependencies (12)Versions (5)Used By (0)

devxisas/qr-studio
==================

[](#devxisasqr-studio)

[![Latest Version on Packagist](https://camo.githubusercontent.com/06a7f9ee1fa3707833c0e3436e2dc625eff4e241030d28b9548c49fd3b2d09b1/68747470733a2f2f696d672e736869656c64732e696f2f7061636b61676973742f762f64657678697361732f71722d73747564696f2e7376673f7374796c653d666c61742d737175617265)](https://packagist.org/packages/devxisas/qr-studio)[![GitHub Tests Action Status](https://camo.githubusercontent.com/094dab0e5cd310c90af3976dcc8359ccbcf60e96ec89b6636136b0ad9657b1d8/68747470733a2f2f696d672e736869656c64732e696f2f6769746875622f616374696f6e732f776f726b666c6f772f7374617475732f64657678697361732f71722d73747564696f2f72756e2d74657374732e796d6c3f6272616e63683d6d61696e266c6162656c3d7465737473267374796c653d666c61742d737175617265)](https://github.com/devxisas/qr-studio/actions?query=workflow%3Arun-tests+branch%3Amain)[![Total Downloads](https://camo.githubusercontent.com/9532cd7d8d20f26ed4e508a5379dcba5c89542682af953ebd678581a28d3704b/68747470733a2f2f696d672e736869656c64732e696f2f7061636b61676973742f64742f64657678697361732f71722d73747564696f2e7376673f7374796c653d666c61742d737175617265)](https://packagist.org/packages/devxisas/qr-studio)[![License](https://camo.githubusercontent.com/b4724deef4231c52cdea62102fd956790f95ec4d1178a70c4f511ea020bcfa1c/68747470733a2f2f696d672e736869656c64732e696f2f7061636b61676973742f6c2f64657678697361732f71722d73747564696f2e7376673f7374796c653d666c61742d737175617265)](LICENSE)

A modern QR code studio for Laravel. Inspired by and built upon the foundation of [simplesoftwareio/simple-qrcode](https://github.com/SimpleSoftwareIO/simple-qrcode), this package brings full PHP 8.2+ type safety, enum-based APIs, new data types (vCard, MeCard, Calendar Events), a Blade directive, a response macro, a `toDataUri()` helper, an Artisan command, and the `HasQrCode` Eloquent trait — while keeping the same familiar fluent interface.

Note

**Migrating from `simplesoftwareio/simple-qrcode`?** See the [migration guide](#migrating-from-simplesoftwareiosimple-qrcode) at the bottom of this page.

---

[![Basic QR code — plain SVG output](docs/images/basic.png)](docs/images/basic.png)

---

Requirements
------------

[](#requirements)

Package versionLaravelPHP1.x11, 12, 138.2+Installation
------------

[](#installation)

```
composer require devxisas/qr-studio
```

The service provider is registered automatically via Laravel's package auto-discovery.

PNG generation uses `ext-imagick` when available, and falls back to `ext-gd` (installed by default on most servers) when Imagick is not installed. Note: the GD fallback does not support gradients.

To install Imagick for best PNG quality:

```
composer require ext-imagick
```

---

Basic Usage
-----------

[](#basic-usage)

Use the `QrCode` facade anywhere in your application:

```
use Devxisas\QrStudio\Facades\QrCode;

// Generate an SVG (default) — safe to render directly in Blade with {!! !!}
$svg = QrCode::generate('https://devxi.com');

// Save to a file
QrCode::generate('https://devxi.com', '/path/to/qrcode.svg');
```

In a Blade template:

```
{!! QrCode::size(200)->generate('https://devxi.com') !!}
```

[![Basic QR code — plain SVG output](docs/images/basic.png)](docs/images/basic.png)

---

Configuration
-------------

[](#configuration)

Publish the config file to set package-wide defaults:

```
php artisan vendor:publish --tag="qr-studio-config"
```

This creates `config/qr-studio.php`:

```
use Devxisas\QrStudio\Enums\ErrorCorrection;
use Devxisas\QrStudio\Enums\Format;
use Devxisas\QrStudio\Enums\Theme;

return [
    // Output
    'format'           => Format::Svg,            // Format enum or 'svg' | 'eps' | 'png'
    'size'             => 100,                     // pixels
    'margin'           => 0,                       // quiet zone around the code
    'error_correction' => ErrorCorrection::Medium, // enum or 'L' | 'M' | 'Q' | 'H'
    'encoding'         => 'UTF-8',                 // character encoding

    // Visual theme — applies a full visual preset to every QR code globally
    'theme'            => null,                    // null | 'ocean'|'sunset'|'forest'|'midnight'|'coral' | Theme enum

    // Storage — used by saveToDisk() when no disk/path is specified per call
    'disk'             => 'local',                 // any disk from config/filesystems.php
    'path'             => 'qrcodes',               // default directory on that disk
];
```

These defaults are applied to every QR code generated through the `QrCode` facade, the `@qrcode` Blade directive, and the `response()->qrcode()` macro. Per-call options always take precedence over config defaults.

KeyScopeDefault`format`Global`Format::Svg``size`Global`100``margin`Global`0``error_correction`Global`ErrorCorrection::Medium``encoding`Global`'UTF-8'``theme`Global`null` (disabled)`disk`saveToDisk`'local'``path`saveToDisk`'qrcodes'`> **Note:** `style`, `eye`, `color`, `gradient`, and `eyeColor` are always per-call — they are not configurable globally as they are visual choices that vary per use case.

---

Formats
-------

[](#formats)

Three output formats are supported. You can pass a string or the `Format` enum.

```
use Devxisas\QrStudio\Enums\Format;

QrCode::generate('...');                          // SVG (default)
QrCode::format('png')->generate('...');           // PNG
QrCode::format(Format::Png)->generate('...');     // PNG via enum
QrCode::format('eps')->generate('...');           // EPS (vector, no browser preview)
```

> **PNG and the GD fallback:** PNG generation uses `ext-imagick` when available. When Imagick is not installed the package falls back to `ext-gd`. The GD fallback works for most use cases, but gradients are not supported under GD — use Imagick if you need gradient PNGs.

[![Output formats — SVG, PNG, EPS](docs/images/formats.png)](docs/images/formats.png)

---

Size &amp; Margin
-----------------

[](#size--margin)

```
QrCode::size(300)->generate('...');
QrCode::size(300)->margin(4)->generate('...');
```

[![Size and margin variations](docs/images/size-margin.png)](docs/images/size-margin.png)

---

Error Correction
----------------

[](#error-correction)

Higher levels allow the QR code to be read even when partially obscured (e.g. with a logo overlay).

```
use Devxisas\QrStudio\Enums\ErrorCorrection;

QrCode::errorCorrection('H')->generate('...');
QrCode::errorCorrection(ErrorCorrection::High)->generate('...');
```

LevelEnumData recovery`L``ErrorCorrection::Low`7%`M``ErrorCorrection::Medium`15% (default)`Q``ErrorCorrection::Quartile`25%`H``ErrorCorrection::High`30%[![Error correction levels L / M / Q / H](docs/images/error-correction.png)](docs/images/error-correction.png)

---

Module Styles
-------------

[](#module-styles)

```
use Devxisas\QrStudio\Enums\Style;

QrCode::style('square')->generate('...');          // default
QrCode::style('dot', 0.5)->generate('...');
QrCode::style('round', 0.7)->generate('...');

// Enum API
QrCode::style(Style::Dot, 0.5)->generate('...');
```

[![Module styles — Square, Dot, Round](docs/images/styles.png)](docs/images/styles.png)

---

Eye Styles
----------

[](#eye-styles)

```
use Devxisas\QrStudio\Enums\EyeStyle;

QrCode::eye('square')->generate('...');   // default
QrCode::eye('circle')->generate('...');
QrCode::eye('pointy')->generate('...');   // new in BaconQrCode 3.x — curved outer + circle inner

// Enum API
QrCode::eye(EyeStyle::Circle)->generate('...');
QrCode::eye(EyeStyle::Pointy)->generate('...');
```

ValueEnumDescription`square``EyeStyle::Square`Default square finder eye`circle``EyeStyle::Circle`Circular finder eye`pointy``EyeStyle::Pointy`Curved outer corner + circle inner (BaconQrCode 3.x)[![Eye styles — Square, Circle, Pointy](docs/images/eyes.png)](docs/images/eyes.png)

---

Colors
------

[](#colors)

### Foreground &amp; background

[](#foreground--background)

```
QrCode::color(59, 130, 246)->generate('...');
QrCode::color(255, 255, 255)->backgroundColor(15, 23, 42)->generate('...');

// With alpha (0–127, where 127 = fully transparent)
QrCode::color(59, 130, 246, 50)->generate('...');
```

### Per-eye colors

[](#per-eye-colors)

Each of the three finder eyes (0, 1, 2) can have independent inner and outer colors.

```
QrCode::eyeColor(0, 239, 68, 68)           // eye 0 — red (inner = outer)
      ->eyeColor(1, 34, 197, 94)           // eye 1 — green
      ->eyeColor(2, 59, 130, 246)          // eye 2 — blue
      ->generate('...');

// With separate inner and outer colors
// eyeColor(eye, innerR, innerG, innerB, outerR, outerG, outerB)
QrCode::eyeColor(0, 255, 255, 255, 59, 130, 246)->generate('...');
```

[![Foreground and background colors](docs/images/colors.png)](docs/images/colors.png)

[![Per-eye color customization](docs/images/eye-colors.png)](docs/images/eye-colors.png)

---

Gradients
---------

[](#gradients)

```
use Devxisas\QrStudio\Enums\GradientType;

// gradient(startR, startG, startB, endR, endG, endB, type)
QrCode::gradient(59, 130, 246, 168, 85, 247, 'radial')->generate('...');

// Enum API
QrCode::gradient(59, 130, 246, 168, 85, 247, GradientType::Radial)->generate('...');
```

TypeEnum`horizontal``GradientType::Horizontal``vertical``GradientType::Vertical``diagonal``GradientType::Diagonal``inverse_diagonal``GradientType::InverseDiagonal``radial``GradientType::Radial`> **Note:** Gradients require `ext-imagick`. They are not supported when falling back to `ext-gd`.

[![Gradient types — horizontal, vertical, diagonal, inverse diagonal, radial](docs/images/gradients.png)](docs/images/gradients.png)

---

Image Merging (Logo overlay)
----------------------------

[](#image-merging-logo-overlay)

Requires PNG format and `ErrorCorrection::High` (`H`) for reliable scanning.

```
// From a file path
QrCode::format('png')
      ->errorCorrection('H')
      ->merge('/path/to/logo.png', 0.3)
      ->generate('https://devxi.com');

// From a string (e.g. fetched via HTTP)
QrCode::format('png')
      ->errorCorrection('H')
      ->mergeString(file_get_contents('/path/to/logo.png'), 0.3)
      ->generate('https://devxi.com');
```

The second argument is the percentage of the QR code the image should occupy (default `0.2`).

[![Logo overlay on PNG QR code](docs/images/merge.png)](docs/images/merge.png)

---

Data URI (`toDataUri`)
----------------------

[](#data-uri-todatauri)

Generates a base64-encoded data URI — ideal for embedding QR codes in emails, PDFs, or anywhere external URLs are unavailable.

```
$uri = QrCode::size(200)->toDataUri('https://devxi.com');
// → "data:image/svg+xml;base64,..."

// PNG
$uri = QrCode::size(200)->format('png')->toDataUri('https://devxi.com');
// → "data:image/png;base64,..."
```

In Blade:

```

```

[![toDataUri — SVG and PNG data URI examples](docs/images/data-uri.png)](docs/images/data-uri.png)

---

Blade Directive
---------------

[](#blade-directive)

A `@qrcode` directive is registered automatically.

```
{{-- Uses config defaults for format and size --}}
@qrcode('https://devxi.com')

{{-- Override format and size per call --}}
@qrcode('https://devxi.com', 'svg', 200)

{{-- Using enums --}}
@php use Devxisas\QrStudio\Enums\Format; @endphp
@qrcode('https://devxi.com', Format::Png, 300)
```

When no `format` argument is passed, the directive reads `qr-studio.format` from config (default `svg`). When no `size` argument is passed, the `` width/height uses `qr-studio.size` from config.

SVG output is echoed directly as HTML. PNG and EPS output is wrapped in an `` tag automatically so no raw binary reaches the browser.

---

Response Macro
--------------

[](#response-macro)

Stream a QR code directly as an HTTP response with the correct `Content-Type` header.

```
// Uses config defaults for format and size
return response()->qrcode('https://devxi.com');

// Override format and size per call
return response()->qrcode('https://devxi.com', 'png', 300);

// Using enum
use Devxisas\QrStudio\Enums\Format;
return response()->qrcode('https://devxi.com', Format::Png, 300);
```

When `size` is not passed, the macro uses `qr-studio.size` from config. The `X-Content-Type-Options: nosniff` header is always set automatically.

```
// Route example
Route::get('/qr/{url}', fn (string $url) => response()->qrcode($url, 'png'));
```

---

Artisan Command
---------------

[](#artisan-command)

Generate QR codes from the command line.

```
# Print SVG to stdout
php artisan qrcode:generate "https://devxi.com"

# Save PNG to a file
php artisan qrcode:generate "https://devxi.com" --format=png --output=public/qr.png

# All options
php artisan qrcode:generate "https://devxi.com" \
    --format=svg \
    --size=300 \
    --margin=2 \
    --error-correction=H \
    --output=/path/to/output.svg
```

OptionDefaultValues`--format``svg``svg`, `eps`, `png``--size``200`1–4096`--margin``0`integer`--error-correction``M``L`, `M`, `Q`, `H``--output`stdoutfile path---

Data Types
----------

[](#data-types)

All data types return an `HtmlString` just like `generate()` and follow the same fluent interface.

### URL / Plain text

[](#url--plain-text)

```
QrCode::generate('https://devxi.com');
QrCode::generate('Plain text content');
```

### Email

[](#email)

```
QrCode::email('hello@example.com', 'Subject', 'Body text');
```

### Phone number

[](#phone-number)

```
QrCode::phoneNumber('+50312345678');
```

### SMS

[](#sms)

```
QrCode::sms('+50312345678', 'Message body');
```

### Geo location

[](#geo-location)

```
QrCode::geo(13.6929, -89.2182);  // latitude, longitude
```

### WiFi

[](#wifi)

```
QrCode::wifi([
    'encryption' => 'WPA',       // WPA | WEP | nopass
    'ssid'       => 'NetworkName',
    'password'   => 'secret123',
    'hidden'     => false,        // optional
]);
```

### Bitcoin

[](#bitcoin)

```
QrCode::btc('1A1zP1eP5QGefi2DMPTfTL5SLmv7Divf', '0.001', [
    'label'         => 'Donation',   // optional
    'message'       => 'Thank you',  // optional
    'returnAddress' => 'bc1q...',    // optional
]);
```

### vCard 3.0

[](#vcard-30)

```
QrCode::errorCorrection('H')->vCard([
    'name'    => 'Sorto;Elmer',   // LastName;FirstName
    'email'   => 'elmer@devxi.com',
    'phone'   => '+50312345678',
    'org'     => 'Devxisas',
    'title'   => 'Developer',
    'url'     => 'https://devxi.com',
    'address' => 'San Salvador, El Salvador',  // optional
    'note'    => 'Some note',                   // optional
]);
```

### MeCard (iOS / Android)

[](#mecard-ios--android)

```
QrCode::meCard([
    'name'    => 'Sorto,Elmer',
    'phone'   => '+50312345678',
    'email'   => 'elmer@devxi.com',
    'url'     => 'https://devxi.com',
    'address' => 'San Salvador',   // optional
    'note'    => 'Some note',      // optional
]);
```

### Calendar Event (iCal)

[](#calendar-event-ical)

Accepts ISO 8601 strings, Unix timestamps, or any `DateTimeInterface`.

```
QrCode::calendarEvent([
    'summary'     => 'Laravel Meetup SV',
    'start'       => '2025-06-15 18:00:00',
    'end'         => '2025-06-15 20:00:00',
    'location'    => 'San Salvador, El Salvador',  // optional
    'description' => 'Monthly Laravel meetup',      // optional
    'url'         => 'https://devxi.com',            // optional
]);

// With Carbon / DateTimeInterface
QrCode::calendarEvent([
    'summary' => 'Meeting',
    'start'   => now()->addDay(),
    'end'     => now()->addDay()->addHours(2),
]);
```

### WhatsApp

[](#whatsapp)

Generates a `wa.me` deep-link QR code. Scanning it opens WhatsApp with the number pre-filled and, when a message is provided, the text pre-typed in the message box.

```
QrCode::whatsApp('+50312345678');

// With pre-filled message
QrCode::whatsApp('+50312345678', '¡Hola! Vi tu QR y quiero más información.');
```

The phone number is normalised to digits only — international format is required (`+503` → `503`).

[![Data types — Email, Phone, SMS, Geo, WiFi, BTC, vCard, MeCard, Calendar, WhatsApp](docs/images/data-types.png)](docs/images/data-types.png)

---

Themes
------

[](#themes)

Apply a complete visual preset — colors, gradient, module style, and eye style — with a single call. Any option chained after `theme()` overrides the preset.

```
use Devxisas\QrStudio\Enums\Theme;

QrCode::theme('ocean')->generate('...');
QrCode::theme(Theme::Sunset)->generate('...');

// Per-call options override the theme
QrCode::theme('ocean')->size(300)->eye('square')->generate('...');

// Set a global default in config/qr-studio.php
'theme' => 'ocean',
'theme' => Theme::Midnight,
```

ThemeDescription`ocean`Deep blue → cyan radial gradient · dot modules · circle eyes`sunset`Orange → crimson diagonal gradient · square eyes`forest`Dark green → mint vertical gradient · round modules · square eyes`midnight`Light blue-white text on deep navy background`coral`Coral → hot-pink horizontal gradient · dot modules · circle eyes> **Note:** Themes that use `dot` or `round` modules automatically set `errorCorrection('H')` to ensure scannability.

[![Built-in themes — ocean, sunset, forest, midnight, coral](docs/images/themes.png)](docs/images/themes.png)

---

saveToDisk()
------------

[](#savetodisk)

Save any QR code directly to a Laravel filesystem disk (local, S3, public, etc.).

```
// Save to default disk ('local') at default path ('qrcodes/')
$path = QrCode::format('png')->size(300)->saveToDisk(
    'https://devxi.com',
    'my-qr.png'
);
// → saves to storage/app/qrcodes/my-qr.png
// → returns 'qrcodes/my-qr.png'

// Full path — directory separator present, no prefix added
QrCode::saveToDisk('https://devxi.com', 'invoices/2025/001F.svg');

// Override disk per call (any disk from config/filesystems.php)
QrCode::format('png')->saveToDisk('https://devxi.com', 'qr.png', 's3');

// Get a public URL after saving
$url = Storage::disk('s3')->url($path);

// Combine with themes
QrCode::theme('ocean')->format('png')->size(400)->saveToDisk(
    'https://devxi.com',
    'branded.png',
    's3'
);
```

**Path resolution:** when `$filename` contains no `/`, the `path` from config is prepended automatically (`'qr.png'` → `'qrcodes/qr.png'`). Pass a full relative path to override this.

Configure the defaults in `config/qr-studio.php`:

```
'disk' => 's3',         // default disk for all saveToDisk() calls
'path' => 'qrcodes',    // default directory when filename has no '/'
```

**`HasQrCode` models** also get `saveQrCodeToDisk()`:

```
// Save the model's QR code to disk
$document->saveQrCodeToDisk('facturas/001F.png');
$document->saveQrCodeToDisk('001F.png', 's3');            // override disk
$user->saveQrCodeToDisk('contacts/elmer.png', null, Format::Svg);  // SVG
```

---

HasQrCode Trait
---------------

[](#hasqrcode-trait)

Add QR code generation directly to any Eloquent model. The trait is flexible: return a plain string for URL / text QR codes, or return an array and override `qrCodeType()` for structured data types (MeCard, vCard, WiFi, etc.).

### URL / plain-text model

[](#url--plain-text-model)

```
use Devxisas\QrStudio\Traits\HasQrCode;

class Product extends Model
{
    use HasQrCode;

    public function qrCodeData(): string
    {
        return route('products.show', $this);
    }
}

// In controllers or Blade:
$product->qrCodeSvg();          // inline SVG (uses size from config)
$product->qrCodeSvg(300);       // 300 px SVG
$product->qrCodeDataUri();      // PNG data URI for
$product->qrCodeDataUri(300);   // PNG at 300 px
```

In Blade:

```
{!! $product->qrCodeSvg() !!}

```

### Structured data types

[](#structured-data-types)

Return an array from `qrCodeData()` and override `qrCodeType()` to match the generator's data type method:

```
use Devxisas\QrStudio\Traits\HasQrCode;

// Contact card — encodes a MeCard QR code
class Contact extends Model
{
    use HasQrCode;

    public function qrCodeType(): string { return 'meCard'; }

    public function qrCodeData(): array
    {
        return [
            'name'  => $this->last_name . ',' . $this->first_name,
            'email' => $this->email,
            'phone' => $this->phone ?? '',
            'url'   => route('contacts.show', $this),
        ];
    }
}

// Office / venue — encodes a WiFi QR code
class Office extends Model
{
    use HasQrCode;

    public function qrCodeType(): string { return 'wifi'; }

    public function qrCodeData(): array
    {
        return [
            'encryption' => 'WPA',
            'ssid'       => $this->wifi_ssid,
            'password'   => $this->wifi_password,
        ];
    }
}

// User profile — encodes a vCard QR code
class User extends Model
{
    use HasQrCode;

    public function qrCodeType(): string { return 'vCard'; }

    public function qrCodeData(): array
    {
        return [
            'name'  => $this->last_name . ';' . $this->first_name,
            'email' => $this->email,
            'phone' => $this->phone ?? '',
            'url'   => route('users.show', $this),
        ];
    }
}
```

Supported `qrCodeType()` return values (must match a generator magic method):

ValueData type`meCard`MeCard (iOS/Android)`vCard`vCard 3.0`wifi`WiFi network`email`Email with subject/body`phoneNumber`Phone number`sMS`SMS with body`geo`GPS coordinates`bTC`Bitcoin payment`calendarEvent`iCal event### Trait API

[](#trait-api)

MethodSignatureDescription`qrCodeData()``qrCodeData(): string|array`**Must be implemented.** Return a string for URL/text, or an array for structured types. Throws `BadMethodCallException` if not overridden.`qrCodeType()``qrCodeType(): string`Override to set the data type when returning an array from `qrCodeData()`. Default: `'text'`.`qrCodeSvg()``qrCodeSvg(int $size = 0): string`Returns an inline SVG string. `$size = 0` uses the package config default.`qrCodeDataUri()``qrCodeDataUri(int $size = 0, Format $format = Format::Png): string`Returns a base64 data URI. Defaults to PNG.---

Combining Options
-----------------

[](#combining-options)

Options are fully composable via fluent chaining:

```
QrCode::size(250)
      ->format('png')
      ->style('dot', 0.5)
      ->eye('circle')
      ->gradient(59, 130, 246, 99, 102, 241, 'radial')
      ->errorCorrection('H')
      ->generate('https://devxi.com');
```

```
QrCode::size(250)
      ->style('round', 0.7)
      ->eye('square')
      ->color(16, 185, 129)
      ->backgroundColor(15, 23, 42)
      ->margin(2)
      ->generate('https://devxi.com');
```

[![Combination examples — dot+gradient, round+color, full dark theme](docs/images/combinations.png)](docs/images/combinations.png)

---

API Reference
-------------

[](#api-reference)

Complete reference for all public methods on `QrCodeGenerator` (accessed via the `QrCode` facade).

### generate()

[](#generate)

**Signature:** `generate(string $text, ?string $filename = null): HtmlString|string|null`

**Description:** Generates the QR code for the given text. When `$filename` is provided the output is written to that path and `null` is returned. Without a filename, an `HtmlString` is returned (or a plain `string` outside of Laravel). Calls `reset()` automatically after execution, so all per-call options are cleared.

**Parameters:**

NameTypeRequiredDescription`$text``string`YesThe content to encode in the QR code`$filename``?string`NoAbsolute path to write the output file**Exceptions:** `WriterException` if the underlying BaconQrCode writer fails.

**Example:**

```
// Returns HtmlString — embed directly in Blade
$svg = QrCode::size(200)->generate('https://devxi.com');

// Write to file — returns null
QrCode::format('png')->size(300)->generate('https://devxi.com', '/var/www/qr.png');
```

---

### toDataUri()

[](#todatauri)

**Signature:** `toDataUri(string $text): string`

**Description:** Generates the QR code and returns a base64-encoded data URI. Ideal for embedding in emails, PDFs, or any context where external URLs are blocked. Calls `reset()` automatically after execution.

- SVG format returns `data:image/svg+xml;base64,...`
- PNG format returns `data:image/png;base64,...`

**Parameters:**

NameTypeRequiredDescription`$text``string`YesThe content to encode**Exceptions:** `WriterException` if the underlying writer fails.

**Example:**

```
$uri = QrCode::size(200)->toDataUri('https://devxi.com');
// → "data:image/svg+xml;base64,..."

// In Blade

```

---

### format()

[](#format)

**Signature:** `format(Format|string $format): static`

**Description:** Sets the output format. Accepts a `Format` enum (recommended) or the strings `'svg'`, `'eps'`, or `'png'`.

**Parameters:**

NameTypeRequiredDescription`$format``Format|string`YesOutput format: `svg`, `eps`, `png`**Exceptions:** `InvalidArgumentException` if a string other than `svg`, `eps`, or `png` is passed.

**Example:**

```
QrCode::format('png')->generate('...');
QrCode::format(Format::Png)->generate('...');
```

---

### size()

[](#size)

**Signature:** `size(int $pixels): static`

**Description:** Sets the width and height of the generated QR code in pixels. Minimum value is 1.

**Parameters:**

NameTypeRequiredDescription`$pixels``int`YesSize in pixels (minimum: 1)**Exceptions:** `InvalidArgumentException` if `$pixels` is less than 1.

**Example:**

```
QrCode::size(300)->generate('...');
```

---

### margin()

[](#margin)

**Signature:** `margin(int $margin): static`

**Description:** Sets the quiet zone (whitespace border) around the QR code. Minimum value is 0.

**Parameters:**

NameTypeRequiredDescription`$margin``int`YesQuiet zone width in pixels (min 0)**Exceptions:** `InvalidArgumentException` if `$margin` is negative.

**Example:**

```
QrCode::size(300)->margin(4)->generate('...');
```

---

### encoding()

[](#encoding)

**Signature:** `encoding(string $encoding): static`

**Description:** Sets the character encoding used when encoding the text. The value is converted to uppercase internally. Defaults to `UTF-8`.

**Parameters:**

NameTypeRequiredDescription`$encoding``string`YesCharacter encoding (e.g. `UTF-8`)**Example:**

```
QrCode::encoding('ISO-8859-1')->generate('...');
```

---

### errorCorrection()

[](#errorcorrection)

**Signature:** `errorCorrection(ErrorCorrection|string $errorCorrection): static`

**Description:** Sets the error correction level. Higher levels allow the code to be read even when partially obscured (e.g. with a logo). Accepts an `ErrorCorrection` enum or the strings `'L'`, `'M'`, `'Q'`, `'H'`.

**Parameters:**

NameTypeRequiredDescription`$errorCorrection``ErrorCorrection|string`YesError correction level: L, M, Q or H**Exceptions:** `InvalidArgumentException` if a string other than `L`, `M`, `Q`, or `H` is passed.

**Example:**

```
QrCode::errorCorrection('H')->generate('...');
QrCode::errorCorrection(ErrorCorrection::High)->generate('...');
```

---

### style()

[](#style)

**Signature:** `style(Style|string $style, float $size = 0.5): static`

**Description:** Sets the module (dot) rendering style. Accepts a `Style` enum or the strings `'square'`, `'dot'`, or `'round'`. The `$size` parameter controls the fill ratio of each module and must be strictly between 0 and 1.

> **Important:** `dot` and `round` styles render all pixels as dots, including the three finder-pattern eyes. Without an explicit `eye()` call the finder squares disappear and scanners fail to orient the code. Always combine `style('dot', ...)` or `style('round', ...)` with `->eye('square')` or `->eye('circle')`.

**Parameters:**

NameTypeRequiredDescription`$style``Style|string`YesModule style: `square`, `dot`, or `round``$size``float`NoFill ratio, must be &gt; 0 and &lt; 1. Default: `0.5`**Exceptions:** `InvalidArgumentException` if an invalid style string is passed, or if `$size` is `>= 1` or `eye('square')->errorCorrection('H')->generate('...');

// Wrong — finder eyes will be invisible, code will not scan
QrCode::style('dot', 0.5)->generate('...');
```

---

### eye()

[](#eye)

**Signature:** `eye(EyeStyle|string $style): static`

**Description:** Sets the finder-eye (corner square) rendering style. Accepts an `EyeStyle` enum or the strings `'square'`, `'circle'`, or `'pointy'`.

> **Required when using `style('dot')` or `style('round')`** to ensure the finder pattern remains recognizable to scanners.

**Parameters:**

NameTypeRequiredDescription`$style``EyeStyle|string`YesEye style: `square`, `circle`, or `pointy`**Exceptions:** `InvalidArgumentException` if a string other than `square`, `circle`, or `pointy` is passed.

**Example:**

```
QrCode::eye('circle')->generate('...');
QrCode::eye(EyeStyle::Pointy)->generate('...');

// Required with dot/round modules
QrCode::style('dot', 0.5)->eye('square')->errorCorrection('H')->generate('...');
```

---

### color()

[](#color)

**Signature:** `color(int $red, int $green, int $blue, ?int $alpha = null): static`

**Description:** Sets the foreground color of the QR code modules. Alpha channel is optional: `0` = fully opaque, `127` = fully transparent.

**Parameters:**

NameTypeRequiredDescription`$red``int`YesRed channel (0–255)`$green``int`YesGreen channel (0–255)`$blue``int`YesBlue channel (0–255)`$alpha``?int`NoAlpha (0 = opaque, 127 = transparent)**Example:**

```
QrCode::color(59, 130, 246)->generate('...');
QrCode::color(59, 130, 246, 50)->generate('...');  // semi-transparent
```

---

### backgroundColor()

[](#backgroundcolor)

**Signature:** `backgroundColor(int $red, int $green, int $blue, ?int $alpha = null): static`

**Description:** Sets the background color of the QR code. Alpha channel follows the same convention as `color()`: `0` = fully opaque, `127` = fully transparent.

**Parameters:**

NameTypeRequiredDescription`$red``int`YesRed channel (0–255)`$green``int`YesGreen channel (0–255)`$blue``int`YesBlue channel (0–255)`$alpha``?int`NoAlpha (0 = opaque, 127 = transparent)**Example:**

```
QrCode::color(255, 255, 255)->backgroundColor(15, 23, 42)->generate('...');
```

---

### eyeColor()

[](#eyecolor)

**Signature:** `eyeColor(int $eyeNumber, int $innerRed, int $innerGreen, int $innerBlue, int $outerRed = 0, int $outerGreen = 0, int $outerBlue = 0): static`

**Description:** Sets the color of one of the three finder eyes independently. `$eyeNumber` is `0`, `1`, or `2`. Each eye has an inner color (the center dot) and an outer color (the surrounding frame). When only inner colors are provided, outer defaults to black (`0, 0, 0`).

**Parameters:**

NameTypeRequiredDescription`$eyeNumber``int`YesEye index: 0, 1, or 2`$innerRed``int`YesInner dot red channel (0–255)`$innerGreen``int`YesInner dot green channel (0–255)`$innerBlue``int`YesInner dot blue channel (0–255)`$outerRed``int`NoOuter frame red channel. Default: `0``$outerGreen``int`NoOuter frame green channel. Default: `0``$outerBlue``int`NoOuter frame blue channel. Default: `0`**Exceptions:** `InvalidArgumentException` if `$eyeNumber` is not 0, 1, or 2.

**Example:**

```
QrCode::eyeColor(0, 239, 68, 68)           // eye 0 — red inner, black outer
      ->eyeColor(1, 34, 197, 94)           // eye 1 — green inner, black outer
      ->eyeColor(2, 59, 130, 246)          // eye 2 — blue inner, black outer
      ->generate('...');

// Separate inner and outer colors
QrCode::eyeColor(0, 255, 255, 255, 59, 130, 246)->generate('...');
```

---

### gradient()

[](#gradient)

**Signature:** `gradient(int $startR, int $startG, int $startB, int $endR, int $endG, int $endB, GradientType|string $type): static`

**Description:** Applies a gradient fill to the QR code foreground modules. The gradient runs from the start color to the end color according to the specified type. Only works with SVG format or PNG with `ext-imagick` installed. Passing a `GradientType` enum is recommended; strings are accepted for compatibility.

**Parameters:**

NameTypeRequiredDescription`$startR``int`YesStart color red channel (0–255)`$startG``int`YesStart color green channel (0–255)`$startB``int`YesStart color blue channel (0–255)`$endR``int`YesEnd color red channel (0–255)`$endG``int`YesEnd color green channel (0–255)`$endB``int`YesEnd color blue channel (0–255)`$type``GradientType|string`YesGradient direction/shape (see Enum Reference below)**Exceptions:** `InvalidArgumentException` if an invalid gradient type string is passed. `RuntimeException` at render time if PNG format is used without `ext-imagick`.

**Example:**

```
QrCode::gradient(59, 130, 246, 168, 85, 247, 'radial')->generate('...');
QrCode::gradient(59, 130, 246, 168, 85, 247, GradientType::Radial)->generate('...');
```

---

### theme()

[](#theme)

**Signature:** `theme(Theme|string $theme): static`

**Description:** Applies a built-in visual preset to the QR code. The preset configures colors, gradient, module style, and eye style in a single call. Any method chained *after* `theme()` overrides the corresponding preset value.

**Available themes:** `ocean`, `sunset`, `forest`, `midnight`, `coral` (use the `Theme` enum or a plain string).

**Parameters:**

NameTypeRequiredDescription`$theme``Theme|string`YesTheme name or `Theme` enum case**Exceptions:** `InvalidArgumentException` if an unrecognised theme name is passed.

**Example:**

```
use Devxisas\QrStudio\Enums\Theme;

// Apply a preset
QrCode::theme('ocean')->size(300)->generate('https://devxi.com');

// Override one value from the preset
QrCode::theme(Theme::Sunset)->size(300)->color(0, 0, 0)->generate('...');
```

---

### saveToDisk()

[](#savetodisk-1)

**Signature:** `saveToDisk(string $text, string $filename, ?string $disk = null): string`

**Description:** Generates the QR code and persists it to a Laravel filesystem disk. Returns the storage path where the file was written. Calls `reset()` automatically after execution.

- When `$filename` contains no `/`, the configured `path` from `config('qr-studio.path')` is prepended automatically.
- When `$disk` is `null`, the configured `disk` from `config('qr-studio.disk')` is used.
- The file extension in `$filename` determines nothing — use `format()` to control the output format.

**Parameters:**

NameTypeRequiredDescription`$text``string`YesThe content to encode in the QR code`$filename``string`YesStorage path (e.g. `'user-42.png'` or `'invoices/qr.svg'`)`$disk``?string`NoFilesystem disk name. `null` = use config default**Returns:** `string` — the storage path where the file was saved.

**Exceptions:** `WriterException` if the QR writer fails. `RuntimeException` if the disk cannot write.

**Example:**

```
// Save to default disk and path (config defaults)
$path = QrCode::format('png')->size(300)->saveToDisk('https://devxi.com', 'user-42.png');
// → "qrcodes/user-42.png"

// Explicit directory in filename — skips config path prefix
$path = QrCode::saveToDisk('https://devxi.com', 'invoices/2025/001F.svg');
// → "invoices/2025/001F.svg"

// Save to S3
$path = QrCode::format('png')->saveToDisk('https://devxi.com', 'qr.png', 's3');

// Combine theme + saveToDisk
QrCode::theme('ocean')->format('png')->size(400)->saveToDisk('https://devxi.com', 'ocean.png', 's3');
```

---

### merge()

[](#merge)

**Signature:** `merge(string $filepath, float $percentage = 0.2, bool $absolute = false): static`

**Description:** Overlays an image (e.g. a logo) centered over the QR code. Only applies when using PNG format — silently has no effect for SVG or EPS. `$percentage` is the fraction of the QR code's width that the image should occupy (0.0–1.0). When `$absolute` is `false` and `$filepath` does not start with `/`, the path is prepended with `base_path()`. Use `errorCorrection('H')` to ensure the code remains scannable after the overlay.

**Parameters:**

NameTypeRequiredDescription`$filepath``string`YesPath to the image file`$percentage``float`NoLogo size as fraction of QR width. Default: `0.2``$absolute``bool`NoIf `true`, skip `base_path()` prepending. Default: `false`**Exceptions:** `InvalidArgumentException` if the file cannot be read.

**Example:**

```
QrCode::format('png')
      ->errorCorrection('H')
      ->merge('/path/to/logo.png', 0.25)
      ->generate('https://devxi.com');
```

---

### mergeString()

[](#mergestring)

**Signature:** `mergeString(string $content, float $percentage = 0.2): static`

**Description:** Same as `merge()` but accepts the raw binary content of the image directly instead of a file path. Useful when the image has already been loaded into memory (e.g. fetched via HTTP). Only applies when using PNG format.

**Parameters:**

NameTypeRequiredDescription`$content``string`YesBinary image content`$percentage``float`NoLogo size as fraction of QR width. Default: `0.2`**Example:**

```
QrCode::format('png')
      ->errorCorrection('H')
      ->mergeString(file_get_contents('/path/to/logo.png'), 0.25)
      ->generate('https://devxi.com');
```

---

Limitations &amp; Known Issues
------------------------------

[](#limitations--known-issues)

### 1. Dot/Round style without eye() — code will not scan

[](#1-dotround-style-without-eye--code-will-not-scan)

`DotsModule` and `RoundnessModule` in BaconQrCode render every pixel as a dot, including the three finder-pattern eyes. Without those recognizable squares, scanners cannot determine the orientation of the code and will fail to read it, even though the QR data is valid.

**Rule:** always call `->eye('square')` or `->eye('circle')` whenever you use `style('dot', ...)` or `style('round', ...)`.

```
// Wrong — finder eyes are invisible, code will not scan
QrCode::style('dot', 0.5)->generate('...');

// Correct
QrCode::style('dot', 0.5)->eye('square')->errorCorrection('H')->generate('...');
```

---

### 2. PNG gradient without ext-imagick throws RuntimeException

[](#2-png-gradient-without-ext-imagick-throws-runtimeexception)

Gradient rendering for PNG requires Imagick. When only GD is available (the fallback path), calling `->gradient()` before generating a PNG throws a `RuntimeException` at render time — not at configuration time.

**Solutions:**

- Install `ext-imagick` for full PNG gradient support.
- Use SVG format instead — gradients always work in SVG with no extra extension required.

```
// Throws RuntimeException if Imagick is not installed:
QrCode::format('png')->gradient(59, 130, 246, 168, 85, 247, 'radial')->generate('...');

// Always works — use SVG when Imagick is unavailable:
QrCode::gradient(59, 130, 246, 168, 85, 247, 'radial')->generate('...');
```

---

### 3. merge() / mergeString() is PNG-only

[](#3-merge--mergestring-is-png-only)

Logo overlay is performed using bitmap compositing and silently has no effect when the output format is SVG or EPS. No error is thrown and no warning is emitted — the image just does not appear.

Always combine `merge()` with `format('png')` and `errorCorrection('H')` (H provides 30% redundancy, which compensates for the area covered by the logo).

```
// Correct
QrCode::format('png')->errorCorrection('H')->merge('/logo.png', 0.25)->generate('...');

// Silent no-op — SVG output, logo ignored
QrCode::merge('/logo.png', 0.25)->generate('...');
```

---

### 4. Gradient without eyeColor() may produce low-contrast finder eyes

[](#4-gradient-without-eyecolor-may-produce-low-contrast-finder-eyes)

When `eyeColor()` is not set, finder eyes inherit the gradient fill. If the gradient produces a light color at the eye positions and the background is also light, the eye borders become invisible and scanners fail.

**Solution:** Use `eyeColor()` on all three eyes with high-contrast colors whenever using a gradient.

```
QrCode::gradient(200, 220, 255, 240, 245, 255, 'radial')
      ->eyeColor(0, 10, 10, 80, 10, 10, 80)
      ->eyeColor(1, 10, 10, 80, 10, 10, 80)
      ->eyeColor(2, 10, 10, 80, 10, 10, 80)
      ->generate('...');
```

---

### 5. style() size must be strictly less than 1.0

[](#5-style-size-must-be-strictly-less-than-10)

The `$size` parameter in `style('dot', $size)` must be in the range `(0, 1)`. Passing `1.0` or any value `>= 1` throws an `InvalidArgumentException` immediately. Values close to `1.0` also produce nearly solid modules with no visible gap between them.

Recommended range: `0.5` to `0.7`.

```
QrCode::style('dot', 1.0);  // throws InvalidArgumentException
QrCode::style('dot', 0.6);  // recommended
```

---

### 6. Low error correction with dot/round modules reduces scan reliability

[](#6-low-error-correction-with-dotround-modules-reduces-scan-reliability)

Error correction levels L (7%) or M (15%) combined with dot or round modules often produce codes that real-world scanners refuse. The reduced per-module fill area leaves very little redundancy to compensate for noise, printing imperfections, or screen glare.

Always use `errorCorrection('H')` when using dot or round module styles.

```
// Unreliable — default EC is M
QrCode::style('dot', 0.5)->eye('square')->generate('...');

// Reliable
QrCode::style('dot', 0.5)->eye('square')->errorCorrection('H')->generate('...');
```

---

### 7. EPS format has no browser preview

[](#7-eps-format-has-no-browser-preview)

EPS (Encapsulated PostScript) is a valid vector format understood by print software (Illustrator, InDesign, macOS Preview), but browsers cannot render it inline. Using an EPS data URI in an `` tag will display nothing.

EPS is intended for print pipelines. Use `response()->qrcode('...', Format::Eps)` to stream it from a route, or write it to a file with `generate($text, $path)`. For web display, use SVG.

---

### 8. State resets automatically after generate() and toDataUri()

[](#8-state-resets-automatically-after-generate-and-todatauri)

Every call to `generate()` or `toDataUri()` triggers an internal `reset()`, clearing all per-call options (format, size, style, colors, etc.) and returning the instance to config defaults. Do not split configuration across multiple statements expecting state to persist:

```
// Wrong — state is reset after first generate(), second call uses defaults only
QrCode::size(300)->format('png');
QrCode::generate('...');

// Correct — configure everything in a single fluent chain
QrCode::size(300)->format('png')->generate('...');
```

---

Enum Reference
--------------

[](#enum-reference)

All enums are backed enums so they work alongside their string equivalents.

```
use Devxisas\QrStudio\Enums\Format;
use Devxisas\QrStudio\Enums\Style;
use Devxisas\QrStudio\Enums\EyeStyle;
use Devxisas\QrStudio\Enums\ErrorCorrection;
use Devxisas\QrStudio\Enums\GradientType;
use Devxisas\QrStudio\Enums\Theme;

Format::Svg          // 'svg'
Format::Eps          // 'eps'
Format::Png          // 'png'

Style::Square        // 'square'
Style::Dot           // 'dot'
Style::Round         // 'round'

EyeStyle::Square     // 'square'
EyeStyle::Circle     // 'circle'
EyeStyle::Pointy     // 'pointy'  — curved outer corner + circle inner (BaconQrCode 3.x)

ErrorCorrection::Low       // 'L'
ErrorCorrection::Medium    // 'M'
ErrorCorrection::Quartile  // 'Q'
ErrorCorrection::High      // 'H'

GradientType::Horizontal       // 'horizontal'
GradientType::Vertical         // 'vertical'
GradientType::Diagonal         // 'diagonal'
GradientType::InverseDiagonal  // 'inverse_diagonal'
GradientType::Radial           // 'radial'

Theme::Ocean     // 'ocean'
Theme::Sunset    // 'sunset'
Theme::Forest    // 'forest'
Theme::Midnight  // 'midnight'
Theme::Coral     // 'coral'
```

---

Testing
-------

[](#testing)

```
composer test           # all tests
composer test:unit      # unit tests only
composer analyse        # PHPStan static analysis
composer format         # Pint code style fix
composer format:check   # Pint check only
```

---

Migrating from `simplesoftwareio/simple-qrcode`
-----------------------------------------------

[](#migrating-from-simplesoftwareiosimple-qrcode)

This package was built as a modernized continuation of `simple-qrcode`. The core fluent API is identical, so most projects require only minor changes.

### Installation

[](#installation-1)

```
composer remove simplesoftwareio/simple-qrcode
composer require devxisas/qr-studio
```

### Facade

[](#facade)

```
// Before
use SimpleSoftwareIO\QrCode\Facades\QrCode;

// After
use Devxisas\QrStudio\Facades\QrCode;
```

That's usually the only change needed. All existing method calls (`generate`, `size`, `format`, `color`, `style`, `eye`, `gradient`, `merge`, etc.) work identically.

### What's new

[](#whats-new)

Featuresimple-qrcodeqr-studioPHP 8.2+ strict types—✓Backed enums for all options—✓`toDataUri()`—✓vCard 3.0 data type—✓MeCard data type—✓Calendar Event data type—✓`@qrcode` Blade directive—✓`response()->qrcode()` macro—✓Artisan `qrcode:generate` command—✓Publishable config file—✓`HasQrCode` Eloquent trait—✓Automatic state reset after `generate()`—✓`EyeStyle::Pointy` (BaconQrCode 3.x)—✓PNG via `ext-gd` fallback (no Imagick needed)—✓WhatsApp data type—✓Built-in visual themes (ocean, sunset, forest…)—✓`saveToDisk()` — persist to any filesystem disk—✓Laravel 11 / 12 / 13 support✓✓SVG / EPS / PNG formats✓✓Image merging (logo overlay)✓✓Colors, gradients, eye colors✓✓Email, Phone, SMS, Geo, WiFi, BTC✓✓---

Changelog
---------

[](#changelog)

See [CHANGELOG.md](CHANGELOG.md).

Contributing
------------

[](#contributing)

See [CONTRIBUTING.md](CONTRIBUTING.md).

Security
--------

[](#security)

See [SECURITY.md](SECURITY.md).

License
-------

[](#license)

MIT — see [LICENSE](LICENSE).

###  Health Score

40

—

FairBetter than 87% of packages

Maintenance96

Actively maintained with recent releases

Popularity0

Limited adoption so far

Community6

Small or concentrated contributor base

Maturity49

Maturing project, gaining track record

 Bus Factor1

Top contributor holds 100% of commits — single point of failure

How is this calculated?**Maintenance (25%)** — Last commit recency, latest release date, and issue-to-star ratio. Uses a 2-year decay window.

**Popularity (30%)** — Total and monthly downloads, GitHub stars, and forks. Logarithmic scaling prevents top-heavy scores.

**Community (15%)** — Contributors, dependents, forks, watchers, and maintainers. Measures real ecosystem engagement.

**Maturity (30%)** — Project age, version count, PHP version support, and release stability.

###  Release Activity

Cadence

Unknown

Total

1

Last Release

53d ago

### Community

Maintainers

![](https://www.gravatar.com/avatar/e99fd40f951486893fc76817ea831c33a68b362e5df5d59bfade2d2313469bfb?d=identicon)[elmersh](/maintainers/elmersh)

---

Top Contributors

[![elmersh](https://avatars.githubusercontent.com/u/26351000?v=4)](https://github.com/elmersh "elmersh (42 commits)")

---

Tags

bacon-qr-codelaravellaravel-packagephppngqr-codeqrcodesvgqr codeqrcodeqrlaravelgeneratorsvgbarcodepngdevxisasbacon-qr-codeqr-studio

###  Code Quality

TestsPest

Static AnalysisPHPStan

Code StyleLaravel Pint

### Embed Badge

![Health badge](/badges/devxisas-qr-studio/health.svg)

```
[![Health](https://phpackages.com/badges/devxisas-qr-studio/health.svg)](https://phpackages.com/packages/devxisas-qr-studio)
```

###  Alternatives

[milon/barcode

Barcode generator like Qr Code, PDF417, C39, C39+, C39E, C39E+, C93, S25, S25+, I25, I25+, C128, C128A, C128B, C128C, 2-Digits UPC-Based Extention, 5-Digits UPC-Based Extention, EAN 8, EAN 13, UPC-A, UPC-E, MSI (Variation of Plessey code)

1.5k13.3M39](/packages/milon-barcode)[linkxtr/laravel-qrcode

A clean, modern, and easy-to-use QR code generator for Laravel

295.1k](/packages/linkxtr-laravel-qrcode)[vormkracht10/laravel-mails

Laravel Mails can collect everything you might want to track about the mails that has been sent by your Laravel app.

24149.7k](/packages/vormkracht10-laravel-mails)[tuncaybahadir/quar

A simple QR Code generation tool for your projects with Laravel 10, 11, 12 versions, php 8.2, 8.3, 8.4 and 8.5

6966.5k4](/packages/tuncaybahadir-quar)[yellowskies/qr-code-bundle

Symfony Barcode &amp; QR Code Generator Bundle with Twig extension

36682.9k](/packages/yellowskies-qr-code-bundle)[akira/laravel-qrcode

A clean, modern, and easy-to-use QR code generator for Laravel

431.4k](/packages/akira-laravel-qrcode)

PHPackages © 2026

[Directory](/)[Categories](/categories)[Trending](/trending)[Changelog](/changelog)[Analyze](/analyze)
