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:

ProviderPackage nameDownload
Play Store (Android)PAY.POS<https://play.google.com/store/apps/details?id=com.paynl.softpos&pli=1>
Sunmi P-DevicesPAY.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 iPhonePAY.POS<https://apps.apple.com/nl/app/pay-pos/id6468886450>

App Versions

Version Updates Release date .APK download for Android

v1.1.3

  • Dark mode improvements
  • Boot stability
  • Cloud Connect speed improvements and stability fixes
  • General settings menu performance update
  • Send logs via Inject
    **Android only **
  • Payment automatically retries after failed to read NFC
  • Dual screen support
  • Mifare support is here! (Via inject only)
    **Sunmi only **
  • Add support for Sunmi non-p non-cloud printers.

31-01-2025

v1.0.57

  • Service inject per payment docs
  • Fix where users would sometimes log out in the background on iOS
  • Sunmi P2_SMARTPAD numpad fixes
  • Sizing fixes on several devices
  • Dark mode
  • POS mode updates
  • Stability fixes

19-11-2024

download

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

download

v1.0.20

QR Code and Barcode scanning for quick activation.
Offline sync delay moved from 10 to 2 seconds.

11-06-2024

download

v1.0.19

Pin prompt cancelation status.
Minor fixes to prevent app crashes during usage.

30-05-2024

download

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
Improved logging

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

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

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 location

If tipModeis enabled, we will append a new object in the returnUrl called amountInfo 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>>

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:

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

FieldTypeDescription
orderIdMV-CODEID of the transaction in the processing unit
statusActionCHANGE / PAID / CANCEL / ERRORStatus of the transaction
referenceAlphanummericYour reference of the transaction
statusCode100 / -63 / -64Nummeric detailed reasoncode for statuus
ticketPlain text - Base 64Printable ticket for the cardholder
activationCode#### -####-####Code that is valid for 24 hours, to activate the terminal via the portal of API
amountInfoPlain text - Base 64Decodable 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 confirmation

For 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)

VersionDownload link
1.0.33download .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.


  1. Open a terminal in the folder where the APK was downloaded.
  2. 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.
  3. For the app to function properly and prevent tampering we unfortunately then need to disable all development options.
  4. Go back to the 'Developer Options'-settings page and uncheck USB Debugging.
  5. 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'.
  6. Complete instructions within the app. When completed your device is able to accept contactless payments.