diff --git a/README.md b/README.md index a6118d6..1c33d1d 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ # Vietnam QR Pay -Thư viện hỗ trợ encode/decode mã QR của VietQR & VNPayQR +Thư viện hỗ trợ encode/decode mã QR của VietQR (QR Ngân hàng, QR Đa năng Momo/ZaloPay) & VNPayQR [![npm](https://img.shields.io/npm/v/vietnam-qr-pay.svg)](https://www.npmjs.com/package/vietnam-qr-pay) [![download](https://img.shields.io/npm/dt/vietnam-qr-pay.svg)](https://www.npmjs.com/package/vietnam-qr-pay) @@ -49,6 +49,67 @@ console.log(content) ``` +### QR Đa năng của MoMo và ZaloPay + +Hiện tại, QR Đa năng của MoMo và ZaloPay đang thông qua Ngân hàng Bản Việt (BVBank) để nhận tiền. + +Mỗi tài khoản MoMo/ZaloPay sẽ được gán một STK tương ứng tại BVBank. Tiền chuyển đến STK này sẽ được chuyển tiếp đến ví MoMo/ZaloPay. + +Bạn có thể lấy STK này tại trang chi tiết của QR Nhận tiền trong ứng dụng MoMo/ZaloPay. + +#### MoMo + +```javascript +import { QRPay, BanksObject } from 'vietnam-qr-pay'; + +// Số tài khoản trong ví MoMo +const accountNumber = '99MM24011M34875080' + +const momoQR = QRPay.initVietQR({ + bankBin: BanksObject.banviet.bin, + bankNumber: accountNumber, + // amount: '10000', // Số tiền (không bắt buộc) + // purpose: 'Chuyen tien', // Nội dung (không bắt buộc) +}) + +// Trong mã QR của MoMo có chứa thêm 1 mã tham chiếu tương ứng với STK +momoQR.additionalData.reference = 'MOMOW2W' + accountNumber.slice(10) + +// Mã QR của MoMo có thêm 1 trường ID 80 với giá trị là 3 số cuối của SỐ ĐIỆN THOẠI của tài khoản nhận tiền +momoQR.setUnreservedField('80', '046') + +const content = momoQR.build() + +// 00020101021138620010A00000072701320006970454011899MM24011M348750800208QRIBFTTA53037045802VN62190515MOMOW2W3487508080030466304EBC8 + +``` + + + +#### ZaloPay + +> Trong mã QR của ZaloPay có chứa một số thông tin bổ sung ở trường ID 26. Tuy nhiên chưa rõ chức năng của các thông tin này (có thể là dùng để định danh từng mã QR đc tạo trên hệ thống của ZaloPay). Trong ví dụ dưới sẽ bỏ qua các thông tin này. + +```javascript +import { QRPay, BanksObject } from 'vietnam-qr-pay'; + +// Số tài khoản trong ví ZaloPay +const accountNumber = '99ZP24009M07248267' + +const zaloPayQR = QRPay.initVietQR({ + bankBin: BanksObject.banviet.bin, + bankNumber: accountNumber, + // amount: '10000', // Số tiền (không bắt buộc) + // purpose: 'Chuyen tien', // Nội dung (không bắt buộc) +}) + +const content = zaloPayQR.build() +// 00020101021138620010A00000072701320006970454011899ZP24009M072482670208QRIBFTTA53037045802VN6304073C +``` + + + + ### VNPay ```javascript diff --git a/package.json b/package.json index 8e7c541..0c48d04 100644 --- a/package.json +++ b/package.json @@ -36,6 +36,7 @@ "@typescript-eslint/parser": "^6.7.2", "eslint": "^8.50.0", "jest": "^28.1.1", + "qrto": "^0.0.3", "ts-jest": "^28.0.5", "ts-node": "^10.8.1", "typescript": "^4.7.4" diff --git a/test/momo.svg b/test/momo.svg new file mode 100644 index 0000000..e3d61a4 --- /dev/null +++ b/test/momo.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/test/test-momo-zalopay.spec.ts b/test/test-momo-zalopay.spec.ts new file mode 100644 index 0000000..0c20331 --- /dev/null +++ b/test/test-momo-zalopay.spec.ts @@ -0,0 +1,62 @@ +import fs from 'node:fs' +import { toSVG } from 'qrto' + +import { BanksObject } from '../src' +import { QRPay } from '../src/qr-pay' + + +test('MoMo', () => { + const accountNumber = '99MM24011M34875080' + const momoQR = QRPay.initVietQR({ + bankBin: BanksObject.banviet.bin, + bankNumber: accountNumber, + }) + momoQR.additionalData.reference = 'MOMOW2W' + accountNumber.slice(10) + momoQR.setUnreservedField('80', '046') + // QR Content generated from MoMo app + const content = momoQR.build() + expect(content).toBe('00020101021138620010A00000072701320006970454011899MM24011M348750800208QRIBFTTA53037045802VN62190515MOMOW2W3487508080030466304EBC8') + + const svg = toSVG(content, { + ecLevel: 'L', + style: 'round', + foregroundColor: { + type: 'linear', + colors: ['#d82f8b', '#a50064'] + }, + logo: { + url: '' + } + }) + fs.writeFileSync(__dirname + '/momo.svg', svg) +}) + + +test('ZaloPay', () => { + const accountNumber = '99ZP24009M07248267' + const zaloPayQR = QRPay.initVietQR({ + bankBin: BanksObject.banviet.bin, + bankNumber: accountNumber, + }) + + const content = zaloPayQR.build() + expect(content).toBe('00020101021138620010A00000072701320006970454011899ZP24009M072482670208QRIBFTTA53037045802VN6304073C') + + const svg = toSVG(content, { + ecLevel: 'L', + marker: { + style: { + outer: 'round', + inner: 'circle', + } + }, + foregroundColor: { + type: 'linear', + colors: ['#0068ff', '#003e99'], + }, + logo: { + url: '' + } + }) + fs.writeFileSync(__dirname + '/zalopay.svg', svg) +}) diff --git a/test/zalopay.svg b/test/zalopay.svg new file mode 100644 index 0000000..3076fa4 --- /dev/null +++ b/test/zalopay.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/yarn.lock b/yarn.lock index 75faec9..6428164 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2445,6 +2445,11 @@ punycode@^2.1.0: resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5" integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg== +qrto@^0.0.3: + version "0.0.3" + resolved "https://registry.yarnpkg.com/qrto/-/qrto-0.0.3.tgz#5609878601de9baff7b7f84e1fa2564ddb99d5be" + integrity sha512-2NWE4JPrxilv/8Kbtjh5O08Tv7vw1K6mxtwyMnAyj4sgtEE71NlOJMD1DUESo1mB8NX8vKCVxMxkPABquZ0ZAQ== + queue-microtask@^1.2.2: version "1.2.3" resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243"