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:

FieldTypeDescription
enabledbooleanIndicates if the tip mode should be offered to the end user.
amountSettingsArray[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:

FieldExample valueDescription
serviceIdSL-1234-1234The sales location ID you want to start the payment for.
secret6ba7ef49e89c4178a1eabff5a7ab82d9The 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.

Field

Type

Description

orderId

MV-CODE

ID of the transaction in the TGU

statusAction

PAID CANCELLED ERROR NFC_READ OFFLINE EXPIRED

Status of the transaction

reference

Alphanummeric

Your reference of the transaction

statusCode

100 / -63 / -64

Nummeric detailed reasoncode for statuus

ticket

Plain text - Base 64

Printable ticket for the cardholder

activationCode

-####-####

Code that is valid for 24 hours, to activate the terminal via my.pay.nl or the

Terminal:Create

API

amountInfo

Plain text - Base 64

Decodable JSON string with the different objects that sum up to the total paid amount

nfcCardID

String (64)

If an external Card ID is scanned, we return the ID of that card.


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.