Same Device (Inject)
Integrate our PAY.POS application on your own device.
We suggest that you start within the app the merchant is using. This generates a simple transaction that can be send to your backend and to PAY.POS. You can also first start a payment in your backend and send the reference via the merchant application to the PAY.POS.

If PAY.POS is opened by an inject and PAY.POS is activated it reads the card data and encrypts the data. If a PINcode is needed, it will ask for the PIN and encrypt that as well. It sends the data to the Transaction Gateway Unit where the transaction is routed via the different schemes to the Issuing bank where the payment is completed. If the Payment fails, we return to the original application or URL (via the returnUrl) where the next action can be performed.
After the payment we asynchronously send a data request to the Exchange URL. And transfer the Payment to the Global Management System where the reporting engine collects the necessary data to report, clear and settle the funds to your account.
Inject an order
The fastest way to start a payment is via the direct inject. You can start a transaction by sending it to Pay.POS.
Keep it simple at the start
transaction = {
"description": "Field 6.2 for EUR 30,02",
"reference": "FIELD6.2",
"returnUrl": "https:\/\/cpoc.buddy.nl", // Or YOURAPP://UUID
"amount": {
"value": 123,
"currency": "EUR"
}
}
But you are allowed to use the full order object
transaction = {
"serviceId": "SL-1234-1234",
"description": "Field 6.2 for EUR 30,02",
"reference": "FIELD6.2",
"returnUrl": "https:\/\/cpoc.buddy.nl", // Or YOURAPP://UUID
"exchangeUrl": "https:\/\/demo.pay.nl\/exchange.php",
"amount": {
"value": 4321,
"currency": "EUR"
},
"integration": {
"testMode": true
},
"customer": {
"firstName": "John",
"lastName": "Doe",
"phone": "0031612345678",
"birthDate": "1999-02-15",
"gender": "m",
"email": "[email protected]",
"ipAddress": "213.126.82.230",
"trust": 5,
"reference": "NL87654321",
"company": {
"countryCode": "nl",
"coc": "12345678",
"vat": "NL0123456789",
"name": "CompanyName"
}
},
"order": {
"countryCode": "NL",
"deliveryDate": "2022-12-30",
"invoiceDate": "2022-12-30",
"deliveryAddress": {
"firstName": "John",
"lastName": "Doe",
"streetName": "Deliverylane",
"streetNumber": "70",
"streetNumberExtension": "A",
"zipCode": "5678CD",
"city": "Amsterdam",
"countryCode": "NL"
},
"invoiceAddress": {
"firstName": "Samanta",
"lastName": "Doe - Jones",
"streetName": "Invoicestreet",
"streetNumber": "2",
"streetNumberExtension": "B",
"zipCode": "SW36LQ",
"city": "London",
"countryCode": "GB"
},
"products": [{
"id": "TEST_01",
"description": "Caramels sweet roll",
"type": "article",
"price": {
"value": 1,
"currency": "EUR"
},
"quantity": 2,
"vatCode": "H"
}, {
"id": "TEST_02",
"description": "Cookie tart sugar",
"type": "article",
"price": {
"value": 3,
"currency": "EUR"
},
"quantity": 1,
"vatCode": "H"
}, {
"id": "TEST_03",
"description": "Lollipop chocolate bar",
"type": "article",
"price": {
"value": 1,
"currency": "EUR"
},
"quantity": 5,
"vatCode": "H"
}]
},
"notification": {
"type": "email",
"recipient": "[email protected]"
},
"stats": {
"object": "POS System",
"info": "Campagne 99",
"tool": "Google",
"extra1": "Customer 6985615",
"extra2": "Invoice 21.3695",
"extra3": "Shop Amsterdam"
}
}
Brand the PAY.POS
To change the icon
and the colour you can send an image (JPG or PNG format) in base64. To change the colours of the loading icons and the buttons youcan also send in buttonTextColor
<a href='paycpoc://?data=ab9c9a73-d986-ce76-1937-508381ac626e&layout={
"icon": "",
"buttonTextColor": "F75513"
}'>Link to the APP
</a>
TIP through an inject
It is also possible to provide the user with an option to leave a tip. To do this you have to slightly alter your inject request:
{
"description": "Field 6.2 for EUR 30,02",
"reference": "FIELD6.2",
"returnUrl": "https:\/\/cpoc.buddy.nl", // Or YOURAPP://UUID
"amount": {
"value": 123,
"currency": "EUR"
},
"integration": {
"tipMode":{
"enabled":true,
"amountSettings": [2.33, 15.6, 20.2]
},
}
}
We added the integration
-> tipMode
section. If you set the enabled property to true the user will be asked if they want to leave a tip.
tipMode
fields:
Field | Type | Description |
---|---|---|
enabled | boolean | Indicates if the tip mode should be offered to the end user. |
amountSettings | Array[number] | An array containing slots for tip options. (Limited at 3 slots for now). The value is a percentage and limited to a value between 0 and 100 |
For now we offer three slots (as a maximum) for you to provide the user with custom tip options. See integration
-> tipMode
-> amountSettings
each of these values is a percentage with a maximum value of 100 and minimum of 0.
Example
If tipMode
is enabled, we will append a new object in the returnURL called amountInfo
. The value will look something like:
amountInfo=eyJwYXltZW50Ijp7ImFtb3VudCI6MjQwMCwiY3VycmVuY3kiOiJFVVIifSwidGlwIjp7ImFtb3VudCI6NDg1LCJjdXJyZW5jeSI6IkVVUiJ9fQ==
This value is coded in base64, if you decode this, you will get this JSON format
{
"payment":{
"amount":2400,
"currency":"EUR"
},
"tip":{
"amount":485,
"currency":"EUR"
}
}
Using another Service/sales location
You can provide the PAY.POS with an alternative service/sales location than the one that the PAY.POS is currently activated on.
To set this up you require two elements:
Field | Example value | Description |
---|---|---|
serviceId | SL-1234-1234 | The sales location ID you want to start the payment for. |
secret | 6ba7ef49e89c4178a1eabff5a7ab82d9 | The secret you retrieve from the sales location |
Use these values in the deeplink, e.g.
paycpoc://?transaction=<PARTIAL_TRANSACTION_OBJECT>
&service={"serviceId": "SL-1234-1234", "secret": "6ba7ef49e89c4178a1eabff5a7ab82d9"}
Sending logs through deeplink
You can send logs through a deeplink from your own app.
paycpoc://?transaction={"returnUrl":"https://google.com"}&command=sendLogs
After the payment
We will load the returnURL. This can be a website or another app installed on the device. We will append the returnURL with additional parameters.
Status actions
Status | Meaning |
---|---|
PAID | Payment was successful. |
CANCELLED | Cancel button has been hit or payment has been DECLINED. if DECLINED, you will get the statusCode & orderId |
ERROR | An error occurred during payment. |
NFC_READ | NFC_READ is triggered when a Mifare card is scanned. It allows merchants to offer their own (prepaid) cards using the nfcCardID parameter. While the merchant still initiates a transaction, the standard payment flow is skipped, and we immediately redirect back to the merchant’s app. The merchant is responsible for handling acceptance or rejection of the transaction. Note: This status is not triggered when a regular bank card is used. |
OFFLINE | Card has been read ofline, transaction will be performed at a later date. (Expires after 28 days). |
EXPIRED | Transaction reached it's expiration limit. (45 sec by default). |
Ticket example
Below you will find the source base64 of the ticket and the ASCII variant that showcases the result when the base64 is decoded.
ICAgICAgICAgICAgIENsYXNzaWMgRmxvd2VyIFN0b3JlIEIuVi4gICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgIEVuc2NoZWRlICAgICAgICAgICAgICAgICAgICAKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCkF1dG9tYWF0OiAgICAyMjE5MDA5MSAgICAgVmVya29wZXI6ICAgICAxMDczNTMxMwpCZXRhbGluZzogICAgICAgICAgICAgICAgIE1WLTExMTEtMTExMS0xMTExLTExMTEKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgIE1BRVNUUk8gICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgIChBMDAwMDAwMDAwMDAwMSkgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAwMDAxNjY2KioqKioqKioqMDAxMjMgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKS2FhcnQgdm9sZ25yOiAwMCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCkdlbGRpZyB0b3Q6IDI3MDYgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKRGF0dW06IDIyLTEyLTIwMjQgMDk6MzAgICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIApBdXRoLiBjb2RlOiAxMTBYMjQgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKVG90YWFsOiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDAsMDEgRVVSCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAozRjAwMDIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA=
Classic Flower Store B.V.
Enschede
------------------------------------------------
Automaat: 22190091 Verkoper: 10735313
Betaling: MV-1111-1111-1111-1111
MAESTRO
(A0000000000001)
0001666*********00123
Kaart volgnr: 00
Geldig tot: 2706
Datum: 22-12-2024 09:30
Auth. code: 110X24
Totaal: 0,01 EUR
3F0002
------------------------------------------------
Activation by Inject
If you open the APP, that is not activated yet via an inject, we will provide the activationCode
APP-DEEPLINK://?statusAction=ACTIVATE&reference={order.reference}&activationCode=####-####-####
you can use the Terminal:Create API to activate the terminal. After Activation you restart the Payment.
Refunds
In order to do refunds you need to check the following:
-
Refunds are enabled on your TH-code: Contact PayNL support to do this
-
Refunds are enabled on your Sales Location: If you do not see this option, contact PayNL support
Once this is done, you can inject a Refund using the following:
paycpoc://?transaction={"type":"REFUND","amount":{"value":1,"currency":"EUR"}}
Frequently asked questions
Can I integrate SoftPOS using a browser-to-app (Browser2App) approach? While it is technically possible to integrate SoftPOS via a browser-to-app (Browser2App) implementation, you may face some potential limitations and user experience issues.
For optimal performance and a seamless user journey, we recommend using app-to-app (App2App) integration If Browser2App is currently your only viable option, you may encounter issues such as multiple browser tabs opening after each transaction. To help mitigate this, we suggest exploring Progressive Web Apps (PWAs) using resources like web.dev's PWA. If you want to convert the website to an app, consider using tools like Capacitor to convert your web application into a native mobile app. In summary, for the best SoftPOS experience, App2App integration would be the preferred and most effective method.
Updated 8 days ago