Пользователи Эвотор могут установить на смарт-терминал приложения для работы с различными платёжными системами. После установки приложения платёжной системы, на экране оплаты/возврата товара, помимо кнопок Наличные и Банковская карта, добавляется кнопка платёжной системы.
Чтобы добавить поддержку сторонней платёжной системы:
В манифесте приложения, добавьте разрешение:
<uses-permission android:name="ru.evotor.permission.PAYMENT_SYSTEM" />
Создайте службу и назовите её, например .PaymentService
.
Служба будет обрабатывать событие evo.v2.receipt.paymentSystem
.
Событие содержит поле OperationType
, которое указывает тип операции:
OperationType.SELL
– продажа;OperationType.SELL_CANCEL
– отмена продажи;OperationType.PAYBACK
– возврат;OperationType.PAYBACK_CANCEL
– отмена возврата.Каждому типу операции соответствует событие и метод, который его обрабатывает. Методы представлены в классе PaymentSystemProcessor.kt
:
abstract class PaymentSystemProcessor : ActionProcessor() {
override fun process(action: String, bundle: Bundle?, callback: ActionProcessor.Callback) {
val event = PaymentSystemEvent.create(bundle) ?: return
when (event.operationType) {
PaymentSystemEvent.OperationType.SELL -> sell(action, event as PaymentSystemSellEvent, callback)
PaymentSystemEvent.OperationType.SELL_CANCEL -> sellCancel(action, event as PaymentSystemSellCancelEvent, callback)
PaymentSystemEvent.OperationType.PAYBACK -> payback(action, event as PaymentSystemPaybackEvent, callback)
PaymentSystemEvent.OperationType.PAYBACK_CANCEL -> paybackCancel(action, event as PaymentSystemPaybackCancelEvent, callback)
else -> null
}
}
Ваша служба должна реализовывать все методы, описанные в классе.
Измените секцию службы .PaymentService
в манифесте приложения:
<service
android:name=".PaymentService"
android:exported="true"
android:icon="@android:drawable/ic_dialog_map"
android:label="Текст на кнопке">
<meta-data
android:name="ru.evotor.sales_screen.BACKGROUND_COLOR"
android:value="@android:color/holo_green_light" />
<meta-data
android:name="ru.evotor.paymentSystem.PAYMENT_SYSTEM_ID"
android:value="ru.test.testapplication.paymentSystem.packageName" />
<intent-filter>
<action android:name="evo.v2.receipt.paymentSystem" />
</intent-filter>
</service>
Где:
ru.evotor.sales_screen.BACKGROUND_COLOR
задаёт цвет кнопки платёжной системы.ru.evotor.paymentSystem.PAYMENT_SYSTEM_ID
задаёт уникальный идентификатор приложения. Уникальность идентификатора необходима, чтобы смарт-терминал всегда мог определить с какой платёжной системой он взаимодействует. Идентификатор остаётся неизменным даже, если название службы изменилось при обновлении приложения.intent-filter
подпишите службу на получение события evo.v2.receipt.paymentSystem
.После обработки события evo.v2.receipt.paymentSystem
, служба .PaymentService
должна возвращать соответствующий результат.
В случае удачной обработки события служба возвращает результат PaymentSystemPaymentOkResult
.
В случае неудачной обработки события служба возвращает результат PaymentSystemPaymentErrorResult
.
В зависимости от операции на смарт-терминале в службу могут поступать четыре типа событий.
При продаже товара приходит событие PaymentSystemSellEvent.kt
:
class PaymentSystemSellEvent(
val receiptUuid: String,
val accountId: String?,
val sum: BigDecimal,
val description: String?
)
Где:
receiptUuid
– идентификатор чека.accountId
– счёт (учётная запись) в платёжной системе. Поле указывать необязательно.sum
– сумма платежа.description
– текстовое описание. Поле может быть указано любым приложением, например, при разделении чека. В приложении требуется предусмотреть возможность отображения этого поля.При отмене продажи приходит событие PaymentSystemSellCancelEvent.kt
:
class PaymentSystemSellCancelEvent(
val receiptUuid: String,
val accountId: String?,
val sum: BigDecimal,
val rrn: String?,
val description: String?
)
Где:
receiptUuid
– идентификатор чека.accoundId
– счёт (учётная запись) в платёжной системе.sum
– сумма платежа.rrn
– уникальный идентификатор платежа, который будет отменён.description
– текстовое описание. Поле может быть указано любым приложением, например, при разделении чека. В приложении требуется предусмотреть возможность отображения этого поля.При возврате товара приходит событие PaymentSystemPaybackEvent.kt
:
class PaymentSystemPaybackEvent(
val receiptUuid: String,
val accountId: String?,
val sum: BigDecimal,
val rrn: String?,
val description: String?
)
Где:
receiptUuid
– идентификатор чека.accoundId
– счёт (учётная запись) в платёжной системе.sum
– сумма платежа.rrn
– уникальный идентификатор платежа, по которому будет проведён возврат.description
– текстовое описание. Поле может быть указано любым приложением, например, при разделении чека. В приложении требуется предусмотреть возможность отображения этого поля.При отмене возврата приходит событие PaymentSystemPaybackCancelEvent.kt
:
class PaymentSystemPaybackCancelEvent(
val receiptUuid: String,
val accountId: String?,
val sum: BigDecimal,
val rrn: String?,
val description: String?
)
Где:
receiptUuid
– идентификатор чека.accoundId
– счёт (учётная запись) в платёжной системе.sum
– сумма платежа.rrn
– уникальный идентификатор платежа, который будет отменён.description
– текстовое описание. Поле может быть указано любым приложением, например, при разделении чека. В приложении требуется предусмотреть возможность отображения этого поля.В случае успешной обработки события служба должна возвращать результат PaymentSystemPaymentOkResult
:
class PaymentSystemPaymentOkResult(
val rrn: String,
val slip: List<String>,
val paymentInfo: String?,
val paymentType: PaymentType = PaymentType.ELECTRON,
val cashlessInfo: CashlessInfo? = null
)
Где:
rrn
– уникальный идентификатор платежа, который понадобится при отмене транзакции.slip
– текст, который будет напечатан на чеке в двух экземплярах.paymentInfo
– поле для хранения статистической информации. Приложение не заполняет это поле.paymentType
– тип оплаты. По умолчанию ELECTRON
.cashlessInfo
– расширенная информация о платеже. Подробное описание в разделе “Передача расширенной информации о платеже”.На рынке появляются новые способы безналичной оплаты, помимо оплаты картой, такие как оплата по «СБП», оплата «Улыбкой» (по биометрии) и так далее. В некоторых случаях все эти способы оплаты реализованы в одном драйвере платёжных систем, например, драйвер Сбербанка. Так же комиссия при выполнении операции может отличаться в зависимости способа платежа. Мы дали возможность сторонним разработчикам платежных приложений передавать в товароучётные системы через наше облако подробную детализацию по способам этих платежей.
Чтобы передать расширенную информацию о способе оплаты, заполните поле cashlessInfo
объекта PaymentSystemPaymentOkResult
:
data class CashlessInfo(
val uuid: String,
val description: String,
val method: Method
)
Где:
method (Enum)
– Способ безналичной оплаты. description (String)
– Расширенное описание способа безналичной оплаты. Например: SberPay, MtsPay, AliPay.uuid (String)
– Уникальный uuid способа безналичной оплаты. Формат обязательно должен быть uuid4
, при создании объекта будет добавлена валидация на проверку формата uuid
.Примеры чеков с заполняемой в CashlessInfo
информацией приведены в статье: https://developer.evotor.ru/docs/doc_java_extended_api_data.html.
В случае ошибки служба должна возвращать результат PaymentSystemPaymentErrorResult.kt
:
class PaymentSystemPaymentErrorResult(
val errorDescription: String?
)
Где:
errorDescription
– описание ошибки.PaymentService
Пример работы с PaymentSystem API в демонстрационном приложении.
Пример службы PaymentService
, которая поддерживает все методы класса PaymentSystemProcessor.kt
:
public class PaymentService extends IntegrationService {
public static final String TAG = "PaymentService";
@Nullable
@Override
protected Map<String, ActionProcessor> createProcessors() {
Map<String, ActionProcessor> processorMap = new HashMap<>();
processorMap.put(
PaymentSystemEvent.NAME_ACTION,
new PaymentSystemProcessor() {
@Override
public void sell(String s, PaymentSystemSellEvent paymentSystemSellEvent, Callback callback) {
Log.e(TAG, "sell " + paymentSystemSellEvent);
Intent intent = new Intent(PaymentService.this, PaymentActivity.class);
intent.putExtra(EXTRA_NAME_OPERATION, "sell");
try {
callback.startActivity(intent);
} catch (RemoteException exc) {
exc.printStackTrace();
}
}
@Override
public void sellCancel(String s, PaymentSystemSellCancelEvent paymentSystemSellCancelEvent, Callback callback) {
Log.e(TAG, "sellCancel " + paymentSystemSellCancelEvent);
Intent intent = new Intent(PaymentService.this, PaymentActivity.class);
intent.putExtra(EXTRA_NAME_OPERATION, "sellCancel");
try {
callback.startActivity(intent);
} catch (RemoteException exc) {
exc.printStackTrace();
}
}
@Override
public void payback(String s, PaymentSystemPaybackEvent paymentSystemPaybackEvent, Callback callback) {
Log.e(TAG, "payback " + paymentSystemPaybackEvent);
Intent intent = new Intent(PaymentService.this, PaymentActivity.class);
intent.putExtra(EXTRA_NAME_OPERATION, "payback");
try {
callback.startActivity(intent);
} catch (RemoteException exc) {
exc.printStackTrace();
}
}
@Override
public void paybackCancel(String s, PaymentSystemPaybackCancelEvent paymentSystemPaybackCancelEvent, Callback callback) {
Log.e(TAG, "paybackCancel " + paymentSystemPaybackCancelEvent);
Intent intent = new Intent(PaymentService.this, PaymentActivity.class);
intent.putExtra(EXTRA_NAME_OPERATION, "paybackCancel");
try {
callback.startActivity(intent);
} catch (RemoteException exc) {
exc.printStackTrace();
}
}
}
}
);
}
package ru.apptest.test.testapplication;
import android.os.Bundle;
import android.view.View;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import ru.evotor.framework.core.IntegrationActivity;
import ru.evotor.framework.core.action.event.receipt.payment.system.result.PaymentSystemPaymentErrorResult;
import ru.evotor.framework.core.action.event.receipt.payment.system.result.PaymentSystemPaymentOkResult;
import ru.evotor.framework.payment.PaymentType;
public class PaymentActivity extends IntegrationActivity {
public static final String EXTRA_NAME_OPERATION = "EXTRA_NAME_OPERATION";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_payment);
findViewById(R.id.button_ok).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String rrn = "";
Random random = new Random();
for (int i = 0; i < 10; i++) {
rrn += random.nextInt(10);
}
List<String> slip = new ArrayList<String>();
slip.add("SLIP START");
slip.add("RRN:");
slip.add(rrn);
slip.add("SLIP EMD");
setIntegrationResult(new PaymentSystemPaymentOkResult(rrn, slip, "123qwe", PaymentType.ELECTRON));
finish();
}
});
findViewById(R.id.button_error).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
setIntegrationResult(new PaymentSystemPaymentErrorResult("beda was happened"));
finish();
}
});
if (getIntent().hasExtra(EXTRA_NAME_OPERATION)) {
((TextView) findViewById(R.id.textView_operation)).setText(getIntent().getStringExtra(EXTRA_NAME_OPERATION));
}
}
@Override
public void onBackPressed() {
setIntegrationResult(new PaymentSystemPaymentErrorResult("onBackPressed was happened"));
finish();
}
}