mortogo321/laravel-thai-promptpay

Thai PromptPay QR Code Payment Generator for Laravel

Installs: 0

Dependents: 0

Suggesters: 0

Security: 0

Stars: 0

Watchers: 0

Forks: 0

Open Issues: 0

pkg:composer/mortogo321/laravel-thai-promptpay

v1.0.0 2025-10-11 06:00 UTC

This package is auto-updated.

Last update: 2025-11-11 06:19:28 UTC


README

Latest Version on Packagist Total Downloads GitHub License PHP Version

A Laravel package for generating Thai PromptPay QR codes for payment processing. This package supports generating QR codes for both Thai phone numbers and National ID numbers, with optional payment amounts.

Features

  • Generate PromptPay QR codes for Thai phone numbers
  • Generate PromptPay QR codes for Thai National ID
  • Support for fixed amount payments
  • Support for open amount (user enters amount)
  • Returns QR code as data URI or binary PNG
  • Built-in AJAX/API endpoints for dynamic generation
  • Works with Axios, Fetch, Vue.js, React, and any frontend framework
  • Beautiful example page included
  • Laravel auto-discovery support
  • Fully compliant with Thai PromptPay EMV QR Code specification

Requirements

  • PHP 8.1 or higher
  • Laravel 10.x or higher

Installation

Install the package via Composer:

composer require mortogo321/laravel-thai-promptpay

The package will automatically register its service provider.

Publish Configuration (Optional)

php artisan vendor:publish --tag=promptpay-config

Usage

Using Facade

use Mortogo321\LaravelThaiPromptPay\Facades\PromptPay;

// Generate QR code with phone number (data URI)
$qrCode = PromptPay::generateQRCode('0812345678');

// Generate QR code with fixed amount
$qrCode = PromptPay::generateQRCode('0812345678', 100.50);

// Generate QR code with National ID
$qrCode = PromptPay::generateQRCode('1234567890123', 250.00);

// Generate QR code with custom size (default: 300px)
$qrCode = PromptPay::generateQRCode('0812345678', 100.00, 400);

// Generate payload string only
$payload = PromptPay::generatePayload('0812345678', 100.00);

// Generate QR code as binary PNG
$binary = PromptPay::generateQRCodeBinary('0812345678', 100.00);

Using Dependency Injection

use Mortogo321\LaravelThaiPromptPay\PromptPayQR;

class PaymentController extends Controller
{
    public function generateQR(PromptPayQR $promptpay)
    {
        $qrCode = $promptpay->generateQRCode('0812345678', 150.00);

        return view('payment', compact('qrCode'));
    }
}

Display QR Code in Blade

<div class="payment-qr">
    <h3>Scan to Pay</h3>
    <img src="{{ $qrCode }}" alt="PromptPay QR Code">
    <p>Amount: ฿{{ number_format($amount, 2) }}</p>
</div>

Download QR Code as Image

use Mortogo321\LaravelThaiPromptPay\Facades\PromptPay;

Route::get('/qr-download', function () {
    $binary = PromptPay::generateQRCodeBinary('0812345678', 100.00);

    return response($binary)
        ->header('Content-Type', 'image/png')
        ->header('Content-Disposition', 'attachment; filename="promptpay-qr.png"');
});

AJAX/API Usage

The package provides built-in API endpoints for generating QR codes dynamically via AJAX/Axios.

Available API Endpoints

1. Generate QR Code

POST /promptpay/generate

axios.post('/promptpay/generate', {
    identifier: '0812345678',  // Phone or National ID
    amount: 100.50,            // Optional
    size: 300                  // Optional (default: 300)
})
.then(response => {
    // response.data.qr_code contains the data URI
    document.getElementById('qr-image').src = response.data.qr_code;
})
.catch(error => {
    console.error(error.response.data.message);
});

Response:

{
    "success": true,
    "qr_code": "...",
    "identifier": "0812345678",
    "amount": 100.50
}

2. Get Payload Only

POST /promptpay/payload

axios.post('/promptpay/payload', {
    identifier: '0812345678',
    amount: 100.50
})
.then(response => {
    console.log(response.data.payload);
});

3. Download QR Code

POST /promptpay/download

axios.post('/promptpay/download', {
    identifier: '0812345678',
    amount: 100.50,
    size: 500
}, {
    responseType: 'blob'
})
.then(response => {
    const url = window.URL.createObjectURL(new Blob([response.data]));
    const link = document.createElement('a');
    link.href = url;
    link.download = 'promptpay-qr.png';
    link.click();
});

Complete AJAX Example

<!DOCTYPE html>
<html>
<head>
    <meta name="csrf-token" content="{{ csrf_token() }}">
    <script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
</head>
<body>
    <input type="text" id="phone" placeholder="0812345678">
    <input type="number" id="amount" placeholder="100.00">
    <button onclick="generateQR()">Generate QR</button>

    <div id="result">
        <img id="qr-image" style="display:none;">
    </div>

    <script>
        // Setup CSRF token
        axios.defaults.headers.common['X-CSRF-TOKEN'] =
            document.querySelector('meta[name="csrf-token"]').content;

        async function generateQR() {
            try {
                const response = await axios.post('/promptpay/generate', {
                    identifier: document.getElementById('phone').value,
                    amount: document.getElementById('amount').value || null
                });

                const img = document.getElementById('qr-image');
                img.src = response.data.qr_code;
                img.style.display = 'block';
            } catch (error) {
                alert(error.response.data.message);
            }
        }
    </script>
</body>
</html>

Vue.js Example

<template>
  <div>
    <input v-model="identifier" placeholder="Phone or ID">
    <input v-model="amount" type="number" placeholder="Amount">
    <button @click="generateQR">Generate</button>

    <img v-if="qrCode" :src="qrCode" alt="PromptPay QR">
  </div>
</template>

<script>
export default {
  data() {
    return {
      identifier: '',
      amount: null,
      qrCode: null
    }
  },
  methods: {
    async generateQR() {
      try {
        const response = await axios.post('/promptpay/generate', {
          identifier: this.identifier,
          amount: this.amount
        });
        this.qrCode = response.data.qr_code;
      } catch (error) {
        alert(error.response.data.message);
      }
    }
  }
}
</script>

React Example

import { useState } from 'react';
import axios from 'axios';

function PromptPayGenerator() {
  const [identifier, setIdentifier] = useState('');
  const [amount, setAmount] = useState('');
  const [qrCode, setQrCode] = useState(null);

  const generateQR = async () => {
    try {
      const response = await axios.post('/promptpay/generate', {
        identifier,
        amount: amount || null
      });
      setQrCode(response.data.qr_code);
    } catch (error) {
      alert(error.response.data.message);
    }
  };

  return (
    <div>
      <input
        value={identifier}
        onChange={(e) => setIdentifier(e.target.value)}
        placeholder="Phone or ID"
      />
      <input
        value={amount}
        onChange={(e) => setAmount(e.target.value)}
        type="number"
        placeholder="Amount"
      />
      <button onClick={generateQR}>Generate</button>

      {qrCode && <img src={qrCode} alt="PromptPay QR" />}
    </div>
  );
}

Publishing Example View

To use the included example page with beautiful UI:

php artisan vendor:publish --tag=promptpay-views

Then add a route in your routes/web.php:

Route::get('/promptpay-demo', function () {
    return view('vendor.promptpay.example');
});

Visit /promptpay-demo to see the interactive demo page.

Supported Identifier Formats

Phone Numbers

  • 0812345678 - Thai phone format (10 digits starting with 0)
  • 812345678 - Without leading 0 (9 digits)
  • 66812345678 - International format (automatically converted)

National ID

  • 1234567890123 - 13-digit Thai National ID

API Reference

generatePayload(string $identifier, ?float $amount = null): string

Generates the PromptPay EMV QR code payload string.

Parameters:

  • $identifier - Thai phone number or National ID
  • $amount - Payment amount (optional, null for open amount)

Returns: PromptPay payload string

generateQRCode(string $identifier, ?float $amount = null, int $size = 300): string

Generates QR code image as data URI.

Parameters:

  • $identifier - Thai phone number or National ID
  • $amount - Payment amount (optional, null for open amount)
  • $size - QR code size in pixels (default: 300)

Returns: Data URI string (can be used directly in <img src="">)

generateQRCodeBinary(string $identifier, ?float $amount = null, int $size = 300): string

Generates QR code as binary PNG data.

Parameters:

  • $identifier - Thai phone number or National ID
  • $amount - Payment amount (optional, null for open amount)
  • $size - QR code size in pixels (default: 300)

Returns: Binary PNG data

Configuration

The config file (config/promptpay.php) allows you to set:

return [
    // Default QR code size in pixels
    'qr_size' => env('PROMPTPAY_QR_SIZE', 300),

    // Default merchant identifier (optional)
    'default_identifier' => env('PROMPTPAY_DEFAULT_IDENTIFIER', null),
];

Add to your .env file:

PROMPTPAY_QR_SIZE=300
PROMPTPAY_DEFAULT_IDENTIFIER=0812345678

Testing

composer test

Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

License

MIT License

Credits

Support

If you discover any issues, please email morgoto321@gmail.com or create an issue on GitHub.