SoftPoS Android & Tap to pay on iPhone (MPOS)
MPOS means, Mobile Payments Of Shelf. Accept Card Present Transactions from an iPhone or an Android device.
Before you start:
1: You need a valid <<glossary:Merchant>>
account, don't have it?
Signup via <https://signup.pay.nl>
2: Download the SoftPos APP from the store:
Provider | Package name | Download |
---|---|---|
Play Store (Android) | PAY.POS | <https://play.google.com/store/apps/details?id=com.paynl.softpos&pli=1> |
Sunmi P-Devices | PAY.POS (h) | Use the Sunmi store and search for PAY.POS to find the hardware version of our PAY.POS app. |
Tap to Pay on iPhone | PAY.POS | <https://apps.apple.com/nl/app/pay-pos/id6468886450> |
App Versions
Version | Updates | Release date | .APK download for Android |
---|---|---|---|
v1.1.3 |
|
31-01-2025 |
|
v1.0.57 |
|
19-11-2024 |
|
v1.0.22 |
Tip mode deeplinks now supported, more info here |
10-07-2024 |
N/A |
v1.0.21 |
Fixed issue on co-branded cards (Bancontact with Visa Debit in relation to some issuers). |
13-06-2024 |
|
v1.0.20 |
QR Code and Barcode scanning for quick activation. |
11-06-2024 |
|
v1.0.19 |
Pin prompt cancelation status. |
30-05-2024 |
|
v1.0.18 |
Mitigated an issue that caused the application to exit when device is low on memory. |
25-05-2024 |
N/A |
V1.0.17 |
Fix memory usage issues, affecting older devices. |
24-05-2024 |
N/A |
V1.0.16 |
Enabled root Logging to check a memory leak |
21-05-2024 |
N/A |
V1.0.15 |
Resolve critical issue regarding retry payment |
18-05-2024 |
N/A |
V1.0.14 |
Added tip module and update the Offline PIN |
12-05-2024 |
N/A |
Send Logs
If you click 3 times on the LOGO of PAY (or whitelabel logo) on top in the Payment Application and the Logs will be send.
Sending logs through deeplink
You can send logs through a deeplink from your own app like so:
paycpoc://?transaction={"returnUrl":"https://google.com"}&command=sendLogs
The Flow

We suggest that you start within the application the merchant is using. This generates a simple transaction that can be send to your backend and to the SoftPos APP. You can also first start a payment in your backend and send the reference via the merchant application to the APP.
This application has two tasks. If it's opened without an inject, merchants can start payments in different flows, request the TIP module or activate options like offline processing or boot the DONO mode (simple option to get easy payments).
If the application is opened by an inject and the application is activated it reads the card data, encrypts the data and if a PINcode is needed, it will ask for the pin and encrypt that as well. It sends the data to the <<glossary: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 <<glossary:Exchange URL>>
. And transfer the Payment to the <<glossary:Global Management System>>
where the reporting engine collects the necessary data to report, clear and settle the funds to your account.
There is more in between!Between the TGU and the issuing bank is more technology, like HSM's, different processing hosts and
<<glossary:Card schemes>>
. For this manual those steps are less informative. We get the card data, we route to touch the bank in the cheapest and most effective way.
Place the Button
Let's send an order (Inject)
The fastest way to start a payment is via the direct inject. You can start a transaction by sending it to the Pay. SoftPoS application.
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"
}
}
Branding the SoftPoS app
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>
Ask the customer for a tip through 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 to the transaction object. 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 |
Custom tip percentage options
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:
The tipMode will change the return locationIf
tipMode
is enabled, we will append a new object in the returnUrl calledamountInfo
the appended value will look something like this:amountInfo=eyJwYXltZW50Ijp7ImFtb3VudCI6MjQwMCwiY3VycmVuY3kiOiJFVVIifSwidGlwIjp7ImFtb3VudCI6NDg1LCJjdXJyZW5jeSI6IkVVUiJ9fQ==
The object is base64 if you decode this you'll get this JSON format:
{ "payment":{ "amount":2400, "currency":"EUR" }, "tip":{ "amount":485, "currency":"EUR" } }
Injecting payments from another <<glossary:serviceId>>
<<glossary:serviceId>>
You can provide Softpos with an alternate sales location than the one that is currently activated.
paycpoc://?transaction=<PARTIAL_TRANSACTION_OBJECT>&authorization=<AUTH_CODE>&service={"serviceId": "SL-XXXX-XXXX", "secret": "<SECRET>"}
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 |
Copy these values and fill them in the deeplink, so this service would be:
paycpoc://?transaction=<PARTIAL_TRANSACTION_OBJECT>&authorization=<AUTH_CODE>&service={"serviceId": "SL-1233-1233", "secret": "abc23230sdfsabasd03..."}
After the Payment
We will load the returnUrl. This can be a website or another application installed on the device.
returnUrl: <<https://www.yourdomain.com>
Field | Type | Description |
---|---|---|
orderId | MV-CODE | ID of the transaction in the processing unit |
statusAction | CHANGE / PAID / CANCEL / ERROR | 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 the portal of API |
amountInfo | Plain text - Base 64 | Decodable JSON string with the different objects that sum up to the total paid amount |
Example CHANGE button:
<https://www.yourdomain.com?orderId=1985773695Xe163a&statusAction=CHANGE&statusCode=0&reference={order.reference}&ticket=BASE64>
<<glossary:Payer>>
clicked the 'Change' button
APP-DEEPLINK://?statusAction=CHANGE&reference={order.reference}
Terminal is not activated
APP-DEEPLINK://?statusAction=ACTIVATE&reference={order.reference}
&activationCode=####-####-####
Terminal is not ready due to techical problems
APP-DEEPLINK://?statusAction=ERROR&reference={order.reference}
Declined transaction by the Payment Method or bank
APP-DEEPLINK://?orderId=1985773695Xe163a&statusAction=CANCEL&statusCode=-64&reference={order.reference}
&ticket=BASE64
Completed transaction by the Payment Method or bank
APP-DEEPLINK://?orderId=1985773695Xe163a&statusAction=PAID&statusCode=100&reference={order.reference}
&ticket=BASE64
Completed transaction in offline mode
APP-DEEPLINK://?statusAction=OFFLINE&reference={order.reference}
If you receive this deeplink this means the payment was not able to be processed because of no internet connection on the terminal. The transaction will be processed later when internet connection resumes.
Note: in an OFFLINE scenario we don't have an orderId and ticket
Completed transaction by the Payment Method or bank with TipMode = true
APP-DEEPLINK://?orderId=1985773695Xe163a&statusAction=PAID&statusCode=100&reference={order.reference}
&ticket=BASE64&amountInfo=BASE64
The Exchange and payment confirmationFor Softpos transactions, you should take the payment state of the return URL. The exchange is send asynchronous to you backend. This can take longer than the transaction is completed.
Don't bother about PIN codes
Payments that are authorised by the customer on a device (like a smartwatch), no pincode is needed, not even for higher amounts. For amounts up untill some tressholds we will always first try to receive the funds without the PIN code, based on the brand of the card that is used. Sometimes the issuing bank will request to secure the transaction with a PIN-code. We will open up a pincode challenge window within the softpos application. If the pincode is providedby the customer, we will then retry the transaction. If the payment is completed (even if we needed a two-step verification via PIN), you will get back control of the flow after that second option.
Why is the pinpad is scrambled?Because softpos can be installed on a regular consumer devices instead of secure and certified devices like payment terminals the schemes rules require us to scramble the pinpad. We know it's not a great experience, but we also know that only 4% of all our transactions are PIN-required. That's a small effort for extra security and people are more and more getting used to it. Data shows 98% of entered pincodes is right the first time. Not a bad result!
Offline PIN
If you use <<glossary:MAT>>
to process payments when there is temporary no internet, you need to test the flow from End2End with the OFFLINE deeplink
Completed transaction in offline mode
APP-DEEPLINK://?statusAction=OFFLINE&reference={order.reference}
Requirements
- The Pay. SoftPoS app only runs on Android devices.
- The app should have the latest Android Security Patch installed within the last 3 months. Older security patches might not work.
- The device should have a NFC reader and NFC capabilities exposed to the Pay SoftPoS app.
- The user must accept all permission requests made by the app.
Store installation (Google Play Store)
We recommend downloading the Pay SoftPoS app through Google's Play Store. This way you will receive automated updates with bug-fixes and new features.
<https://play.google.com/store/apps/details?id=com.paynl.softpos&hl=nl>
Manual installation (APK)
Version | Download link |
---|---|
1.0.33 | download .apk |
For a manual installation of the Pay SoftPoS app follow these instructions. Please note that you will not get automated updates.
Currently there is no publicly available APK. Use the manual installation (APK) method when requested to do so by Pay.
Assuming you've acquired a copy of the SoftPoS app you need to go to the download folder locate your .apk.
- Open a terminal in the folder where the APK was downloaded.
- Execute the following command:
adb install -i com.android.vending .\[version].apk
Note: com.android.vending is required otherwise the app will throw sideloading errors and will not function on your android device. - For the app to function properly and prevent tampering we unfortunately then need to disable all development options.
- Go back to the 'Developer Options'-settings page and uncheck USB Debugging.
- Then disable developer options entirely. This is usually the button on top.
13 Start the app from your app launcher by search for 'Pay' or 'CPOC'. - Complete instructions within the app. When completed your device is able to accept contactless payments.
Updated 3 months ago