Mengimplementasikan Pembayaran Link Stripe di Aplikasi Capacitor Menurut Pedoman Baru Apple
Sejak tanggal 1 Mei 2025, Apple telah menerapkan perubahan signifikan pada Pedoman Ulasan Aplikasi App Store setelah putusan pengadilan dalam kasus Epic v. Apple. Perubahan-perubahan ini secara khusus memungkinkan pengembang aplikasi di Amerika Serikat untuk menghubungkan ke metode pembayaran eksternal untuk barang-barang digital dan layanan, membuka alternatif untuk sistem pembelian dalam aplikasi Apple.
Pertarungan Epik yang Mengubah Pembayaran Seluler Selamanya
Jalan menuju momen ini telah panjang dan kontroversial. Semua dimulai pada Agustus 2020 ketika Epic Games, pembuat permainan populer Fortnite, secara sengaja melanggar pedoman App Store Apple dengan mengimplementasikan opsi pembayaran langsung yang menghindari komisi 30% Apple. Apple segera menghapus Fortnite dari App Store, dan Epic bereaksi dengan mengajukan gugatan yang menantang kontrol Apple atas distribusi aplikasi iOS dan pembayaran dalam aplikasi.
Setelah bertahun-tahun pertarungan hukum, banding, dan banding balik, pengadilan akhirnya memutuskan bahwa Apple harus memungkinkan pengembang untuk mengarahkan pengguna ke metode pembayaran alternatif di luar aplikasi mereka. Keputusan ini secara fundamental mengubah ekonomi ekosistem App Store, yang telah beroperasi di bawah model keuangan dasar yang sama sejak awalnya pada tahun 2008.
Keputusan Akhir - Tidak Ada Banding Lagi
Apa yang membuat keputusan ini sangat signifikan adalah bahwa itu adalah keputusan akhir dan tidak dapat dibandingkan lagi. Mahkamah Agung menolak mendengarkan banding Apple pada awal 2025, memastikan keputusan pengadilan tingkat rendah sebagai hukum tanah. Ini berarti pengembang dapat mengimplementasikan metode pembayaran eksternal dengan percaya diri bahwa Apple tidak dapat membalikkan keputusan ini melalui tantangan hukum yang lebih lanjut.
Perlakuan Sama yang Dijamin oleh Hukum
Yang paling penting, putusan ini secara eksplisit menyatakan bahwa Apple tidak dapat diskriminasi terhadap aplikasi yang menggunakan metode pembayaran eksternal. Pengadilan secara spesifik melarang Apple dari:
- Mengenakan biaya tambahan atau menetapkan persyaratan tambahan pada aplikasi yang menggunakan metode pembayaran eksternal
- Memberikan perlakuan istimewa dalam hasil pencarian atau fitur untuk aplikasi yang secara eksklusif menggunakan sistem IAP Apple
- Menggunakan tindakan teknis untuk membuat pengalaman pembayaran eksternal menjadi kurang baik daripada sistem Apple sendiri
- Menetapkan persyaratan diskusi yang berat di luar informasi konsumen dasar
Ketentuan-ketentuan ini secara eksplisit berarti bahwa pengembang dapat menerapkan Stripe atau penyedia pembayaran eksternal lainnya tanpa takut akan balasan atau diskriminasi dari Apple. Lapangan permainan telah menjadi rata, dan Apple harus menganggap semua aplikasi sama tanpa memandang pilihan metode pembayaran mereka.
Putusan ini merupakan salah satu tantangan paling signifikan terhadap pendekatan taman Apple dan menandai perubahan besar dalam bagaimana aplikasi mobile dapat di monetisasi. Bagi pengembang yang telah lama mengeluh tentang komisi 30% Apple (diperkecil menjadi 15% untuk bisnis kecil), putusan ini menawarkan jalan untuk meningkatkan margin laba dan lebih banyak kontrol atas pengalaman pelanggan.
Manfaat Keuangan Menggunakan Stripe Atas Pembelian Dalam Aplikasi Apple
Implikasi keuangan dari perubahan ini sangat besar bagi pengembang:
-
Biaya Pengolahan Pembayaran yang Dikurangi: Apple biasanya mengenakan komisi 30% pada pembelian aplikasi (15% untuk bisnis kecil), sementara biaya Stripe hanya sekitar 2,9% + $0,30 per transaksi. Perbedaan ini dapat meningkatkan signifikan margin laba Anda.
-
Fleksibilitas Pembayaran: Dengan Apple, Anda biasanya menunggu 45-90 hari untuk menerima dana Anda. Stripe, di sisi lain, mentransfer pembayaran ke rekening bank Anda dalam 2-3 hari kerja.
-
Sistem Pengembalian Uang yang Sederhana: Proses pengembalian uang dapat diatur langsung melalui dashboard Stripe daripada harus melalui sistem pengembalian uang Apple yang lebih kompleks.
Keuntungan ini dapat berubah permainan, terutama bagi pengembang dan bisnis kecil.
Di artikel ini, kami akan menjelajahi cara mengimplementasikan Tautan Pembayaran Stripe di aplikasi Capacitor Anda untuk memanfaatkan aturan baru ini, sambil memastikan konsistensi dengan pedoman Apple yang diupdate.
Implementasi ini didasarkan pada documentasi resmi Stripe untuk Tautan Pembayaran, yang telah disesuaikan secara spesifik untuk aplikasi Capacitor.
Pahami Pedoman Baru
The App Store Review Guidelines yang diperbarui sekarang memungkinkan pengembang untuk mengarahkan pengguna ke situs web eksternal untuk pengolahan pembayaran, khususnya untuk barang digital dan langganan. Perubahan ini hanya berlaku untuk aplikasi yang didistribusikan di App Store Amerika Serikat.
Poin-poin penting untuk dipahami:
- Kamu sekarang dapat menghubungkan opsi pembayaran eksternal untuk barang digital di dalam aplikasi kamu
- Perubahan ini hanya berlaku untuk aplikasi di App Store Amerika Serikat
- Kamu masih harus mematuhi persyaratan pengungkapan Apple
- Kamu tetap bertanggung jawab atas semua dukungan pelanggan dan pengelolaan pengembalian uang
Mengatur Tautan Pembayaran Stripe di Aplikasi Capacitor Kamu
Ayo kita masuk ke implementasi teknis:
Langkah 1: Buat Tautan Pembayaran di Dashboard Stripe
Pertama, buatlah tautan pembayaran di Dashboard Stripe kamu:
- Navigasikan ke bagian Tautan Pembayaran di Dashboard Stripe kamu
- Klik ”+ Baru” untuk membuat tautan pembayaran baru
- Tentukan detail produk atau langganan Anda
- Di pengaturan “Setelah pembayaran”, pilih “Jangan tampilkan halaman konfirmasi”
- Tentukan tautan universal sebagai URL kesuksesan Anda (kami akan mengkonfigurasi ini kemudian)
- Klik “Buat Tautan” untuk menghasilkan tautan pembayaran Anda
Langkah 2: Atur Tautan Universal di Aplikasi Anda Capacitor
Untuk mengarahkan pengguna kembali ke aplikasi Anda setelah pembayaran selesai, konfigurasi tautan universal:
- Buat sebuah
apple-app-site-associationfile di domain Anda:
{
"applinks": {
"apps": [],
"details": [
{
"appIDs": ["YOURTEAMID.com.yourdomain.yourapp"],
"components": [
{
"/": "/checkout_redirect*",
"comment": "Matches any URL whose path starts with /checkout_redirect"
}
]
}
]
}
}
-
Tampilkan file ini di
https://yourdomain.com/.well-known/apple-app-site-association -
Pastikan file tersebut disajikan dengan MIME type yang benar
application/json -
Konfigurasi aplikasi Capacitor Anda untuk menghandle tautan universal dengan menambahkan hak akses yang tepat. Pertama, di project Xcode Anda:
capacitor.config.ts:
import { CapacitorConfig } from '@capacitor/cli';
const config: CapacitorConfig = {
// Your existing app configuration (appId, appName, etc.)
plugins: {
Geolocation: {
// Request precise location access on iOS
iosLocationAccuracy: 'reduced'
}
}
};
export default config;
- Tambahkan hak akses Domain Asosiasi ke project Xcode Anda:
- Buka proyek Xcode Anda
- Pilih target aplikasi Anda
- Pergi ke “Tanda Tangan & Kemampuan”
- Klik ”+ Kemampuan” dan pilih “Domain yang Dikaitkan”
- Tambah
applinks:yourdomain.com
Langkah 3: Buat Halaman Pemadaman
Buat halaman pemadaman di URL pengalihan untuk mengatasi kasus di mana aplikasi tidak terinstal:
<!DOCTYPE html>
<html>
<head>
<title>Redirecting...</title>
<meta http-equiv="refresh" content="0;url=https://yourdomain.com/app-download">
</head>
<body>
<p>Redirecting to download page...</p>
</body>
</html>
Langkah 4: Implementasi Tombol Pembayaran di Aplikasi Anda Capacitor
Sekarang, tambahkan tombol pembayaran ke aplikasi Anda:
import { Capacitor } from '@capacitor/core';
export async function openPaymentLink(userEmail, userId) {
// Use your actual Stripe payment link
const baseUrl = 'https://buy.stripe.com/your_payment_link';
// Add URL parameters to customize the experience
const params = new URLSearchParams({
prefilled_email: encodeURIComponent(userEmail),
client_reference_id: userId
});
const fullUrl = `${baseUrl}?${params.toString()}`;
// Simple window.open works in both web and Capacitor
// Using _blank opens in Safari on iOS which is important for users with saved Stripe Link credentials
window.open(fullUrl, '_blank');
}
Mengapa Safari Penting: Membuka tautan pembayaran di Safari (via
window.open) daripada browser aplikasi internal lebih bermanfaat karena pengguna yang telah menyimpan informasi pembayaran mereka dengan Stripe Link sebelumnya akan memiliki kredit mereka tersedia secara otomatis. Ini menciptakan pengalaman checkout yang lebih halus di mana pengguna tidak perlu memasukkan kembali informasi kartu kredit mereka, secara signifikan mengurangi gesekan dan tingkat kecualian.
Langkah 5: Tangani Tautan Universal di Aplikasi Anda
Konfigurasi aplikasi Anda untuk menangani tautan universal ketika pengguna diarahkan kembali:
- Pertama, instal plugin Aplikasi:
npm install @capacitor/app
- Mendaftarkan plugin Aplikasi di aplikasi Anda:
import { App } from '@capacitor/app';
// In your initialization code
App.addListener('appUrlOpen', (event) => {
// Example URL: https://yourdomain.com/checkout_redirect?session_id=cs_test_...
const url = new URL(event.url);
if (url.pathname.startsWith('/checkout_redirect')) {
// Extract any parameters you need
const params = new URLSearchParams(url.search);
const sessionId = params.get('session_id');
// Handle successful payment
if (sessionId) {
// Verify the payment on your server if needed
verifyPayment(sessionId);
// Update UI to reflect successful purchase
updatePurchaseStatus(true);
}
}
});
async function verifyPayment(sessionId) {
// Call your backend to verify the payment
// This is optional if you're relying on webhooks
}
function updatePurchaseStatus(success) {
// Update your app UI to reflect purchase status
}
Langkah 6: Atur Webhook untuk Penukaran Pesanan
Akhirnya, konfigurasi webhook pada server Anda untuk menangani pembayaran sukses:
// Using Express.js as an example
const express = require('express');
const stripe = require('stripe')('sk_test_your_stripe_secret_key');
const app = express();
// Use raw body parser for webhook signature verification
app.post('/webhook', express.raw({type: 'application/json'}), async (req, res) => {
const sig = req.headers['stripe-signature'];
const webhookSecret = 'whsec_your_webhook_secret';
let event;
try {
event = stripe.webhooks.constructEvent(req.body, sig, webhookSecret);
} catch (err) {
console.log(`Webhook Error: ${err.message}`);
return res.status(400).send(`Webhook Error: ${err.message}`);
}
// Handle the checkout.session.completed event
if (event.type === 'checkout.session.completed') {
const session = event.data.object;
// Retrieve client_reference_id (your user ID)
const userId = session.client_reference_id;
// Grant access to the purchased content
await grantAccess(userId, session.id);
}
res.status(200).send();
});
async function grantAccess(userId, sessionId) {
// Your logic to grant access to the purchased content
// This could be updating a database, sending a notification, etc.
}
app.listen(3000, () => console.log('Webhook server running on port 3000'));
Kemampuan Kompabilitas Android
Marilah kita jelas: putusan Epic v. Apple telah mengubah secara fundamental peta pembayaran mobile. Tidak hanya mempengaruhi langsung aplikasi iOS, tetapi juga memperkuat posisi pengembang Android yang telah menggunakan metode pembayaran eksternal.
Pengembang Android dapat mengeksekusi solusi pembayaran eksternal dengan keyakinan yang lengkap. Preceden yang ditetapkan oleh putusan Apple secara efektif melindungi pengembang di berbagai platform dari potensi pembatasan masa depan. Keputusan pengadilan ini telah memvalidasi apa yang banyak pengembang Android lakukan selama bertahun-tahun - menawarkan opsi pembayaran alternatif dengan biaya yang lebih rendah.
Google Play Store telah selalu kurang restriktif tentang metode pembayaran eksternal daripada Apple, dan sekarang dengan precedent hukum yang ditetapkan, ada hampir tidak ada risiko dalam mengeksekusi Stripe atau penyedia pembayaran eksternal lainnya di aplikasi Android Anda. Anda dapat melanjutkan dengan implementasi ini dengan yakin bahwa Anda berada di atas tanah hukum yang solid.
Implementasi yang telah kita bahas untuk iOS hampir sama untuk perangkat Android. Karena Google Play Store tidak memiliki batasan yang sama pada metode pembayaran eksternal, Anda dapat menggunakan metode Stripe Payment Links yang sama tanpa perlu dialog pengungkapan khusus.
Untuk menghandle deep linking (setara dengan universal links pada iOS), Anda perlu:
- Mengatur App Links di aplikasi Anda
AndroidManifest.xmluntuk menghandle URL redirect - Membuat sebuah
.well-known/assetlinks.jsonfile di domain Anda dengan detail aplikasi Anda - Menggunakan logika pemanggilan yang sama untuk memproses pembayaran sukses
appUrlOpenKeindahan dari __CAPGO_KEEP_0__ adalah bahwa setelah Anda telah mengimplementasikan konfigurasi platform khusus, aliran pembayaran __CAPGO_KEEP_1__ sebenarnya tetap sama di kedua platform.
The beauty of Capacitor is that once you’ve implemented the platform-specific configurations, the actual payment flow code remains the same across both platforms.
Berikut adalah contoh komponen tombol pembayaran di Vue yang dapat Anda tambahkan ke aplikasi __CAPGO_KEEP_0__ Anda:
Here’s an example of a payment button component in Vue that you can add to your Capacitor app:
<template>
<div class="payment-container">
<div class="pricing-card">
<h2 class="mb-4 text-xl font-bold">{{ product.name }}</h2>
<p class="mb-6 text-gray-600">{{ product.description }}</p>
<div class="mb-6 price-tag">
<span class="text-2xl font-bold">${{ product.price }}</span>
<span v-if="product.isSubscription" class="text-sm text-gray-500">/month</span>
</div>
<button
@click="handlePayment"
class="py-3 w-full font-medium text-white bg-indigo-600 rounded-lg transition-colors hover:bg-indigo-700"
>
Purchase Now
</button>
</div>
</div>
</template>
<script setup>
import { ref } from 'vue';
import { Dialog } from '@capacitor/dialog';
const props = defineProps({
product: {
type: Object,
required: true
},
userEmail: {
type: String,
default: ''
},
userId: {
type: String,
required: true
}
});
const isLoading = ref(false);
async function showExternalPaymentDisclosure() {
const { value } = await Dialog.confirm({
title: 'Leaving App for Payment',
message: 'You are about to leave this app to make a payment. Apple is not responsible for the privacy or security of payments that are not made through the App Store. All payment-related issues, including refunds, must be handled by our support team.',
okButtonTitle: 'Continue',
cancelButtonTitle: 'Cancel'
});
return value;
}
async function openPaymentLink() {
// Use your actual Stripe payment link
const baseUrl = 'https://buy.stripe.com/your_payment_link';
// Add URL parameters to customize the experience
const params = new URLSearchParams({
prefilled_email: encodeURIComponent(props.userEmail),
client_reference_id: props.userId
});
const fullUrl = `${baseUrl}?${params.toString()}`;
// Simple window.open works in both web and Capacitor
// Using _blank opens in Safari on iOS which is important for users with saved Stripe Link credentials
window.open(fullUrl, '_blank');
}
async function handlePayment() {
isLoading.value = true;
try {
// Only show the disclosure on iOS
if (window.Capacitor?.getPlatform() === 'ios') {
const userConfirmed = await showExternalPaymentDisclosure();
if (!userConfirmed) return;
}
await openPaymentLink();
} catch (error) {
console.error('Payment error:', error);
await Dialog.alert({
title: 'Payment Error',
message: 'There was an error initiating the payment. Please try again.'
});
} finally {
isLoading.value = false;
}
}
</script>
Menghandle Wilayah yang Berbeda
Karena pedoman baru Apple hanya berlaku untuk aplikasi di App Store Amerika Serikat, Anda perlu strategi untuk mendeteksi wilayah pengguna dan menerapkan metode pembayaran yang tepat. Berikut adalah pendekatan yang lebih dapat diandalkan menggunakan lokasi geografis IP:
import { Capacitor } from '@capacitor/core';
async function determinePaymentMethod() {
// Always use Stripe for Android
if (Capacitor.getPlatform() !== 'ios') {
return 'external';
}
try {
// Use a geolocation service to determine user's country
const response = await fetch('https://ipapi.co/json/');
const locationData = await response.json();
// Check if the user is in the United States
if (locationData.country_code === 'US') {
return 'external'; // Can use Stripe Payment Links
} else {
return 'iap'; // Must use In-App Purchases
}
} catch (error) {
console.error('Error detecting region:', error);
return 'iap'; // Default to IAP to be safe
}
}
export async function processPayment(product, userEmail, userId) {
const paymentMethod = await determinePaymentMethod();
if (paymentMethod === 'external') {
// Use Stripe Payment Links
await initiateExternalPayment(userEmail, userId);
} else {
// Use Apple's In-App Purchase
await initiateInAppPurchase(product.appleProductId);
}
}
Pendekatan ini menggunakan layanan gratis ipapi.co untuk menentukan negara pengguna berdasarkan alamat IP mereka. Anda juga dapat menggunakan layanan lokasi geografis lain seperti MaxMind, atau menerapkan periksa ini di sisi server untuk keamanan tambahan.
Peringatan: Meskipun pendekatan ini berfungsi, penting untuk diingat bahwa lokasi geografis IP tidak selalu 100% akurat. Untuk aplikasi mission-critical, pertimbangkan menggunakan metode deteksi yang lebih akurat atau memungkinkan pengguna untuk memilih wilayah secara manual.
Pengenalan Lokasi yang Lebih Akurat dengan Capacitor Plugins
Untuk pengenalan lokasi yang lebih akurat, Anda dapat menggunakan plugin Capacitor Geolocation bersama dengan @capgo/nativegeocoder untuk menentukan negara pengguna dengan presisi yang lebih tinggi:
- Pertama, instal plugin yang diperlukan:
npm install @capacitor/geolocation @capgo/nativegeocoder
- Konfigurasi plugin di project Capacitor Anda. Tambahkan kode berikut ke
capacitor.config.ts:
import { CapacitorConfig } from '@capacitor/cli';
const config: CapacitorConfig = {
// Your existing app configuration (appId, appName, etc.)
plugins: {
Geolocation: {
// Request precise location access on iOS
iosLocationAccuracy: 'reduced'
}
}
};
export default config;
- Mengimplementasikan deteksi wilayah berdasarkan lokasi:
import { Capacitor } from '@capacitor/core';
import { Geolocation } from '@capacitor/geolocation';
import { NativeGeocoder } from '@capgo/nativegeocoder';
async function isUserInUSA() {
try {
// Request permission first
const permissionStatus = await Geolocation.requestPermissions();
if (permissionStatus.location === 'granted') {
// Get current position
const position = await Geolocation.getCurrentPosition({
timeout: 10000,
enableHighAccuracy: false
});
// Use NativeGeocoder to reverse geocode the coordinates
const results = await NativeGeocoder.reverseGeocode({
latitude: position.coords.latitude,
longitude: position.coords.longitude,
useLocale: true,
maxResults: 1
});
if (results.addresses.length > 0) {
// Check if the user is in the USA
return results.addresses[0].countryCode === 'US';
}
}
// If we couldn't determine location or permission denied, fall back to IP detection
return await isUserInUSAByIP();
} catch (error) {
console.error('Error detecting location:', error);
// Fall back to IP detection on error
return await isUserInUSAByIP();
}
}
async function isUserInUSAByIP() {
try {
const response = await fetch('https://ipapi.co/json/');
const data = await response.json();
return data.country_code === 'US';
} catch (error) {
console.error('Error detecting IP location:', error);
return false; // Default to false to be safe
}
}
export async function determinePaymentMethod() {
// Always use Stripe for Android
if (Capacitor.getPlatform() !== 'ios') {
return 'external';
}
// Check if user is in the USA
const isUSA = await isUserInUSA();
return isUSA ? 'external' : 'iap';
}
export async function processPayment(product, userEmail, userId) {
const paymentMethod = await determinePaymentMethod();
if (paymentMethod === 'external') {
// Use Stripe Payment Links
await initiateExternalPayment(userEmail, userId);
} else {
// Use Apple's In-App Purchase
await initiateInAppPurchase(product.appleProductId);
}
}
Pengimplementasian ini menyediakan cara yang lebih akurat untuk menentukan apakah pengguna berada di Amerika Serikat secara fisik. Pertama, ia mencoba menggunakan GPS perangkat dan geokoder native untuk menentukan negara. Jika itu gagal (disebabkan oleh masalah izin atau kesalahan lainnya), ia kembali ke deteksi berdasarkan IP.
Jangan lupa menambahkan izin yang diperlukan ke info.plist (iOS) dan AndroidManifest.xml file-file (
Untuk iOS (ios/App/App/Info.plist):
<key>NSLocationWhenInUseUsageDescription</key>
<string>We need your location to determine which payment method to use based on regional availability.</string>
Untuk Android (android/app/src/main/AndroidManifest.xml):
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
Menggunakan pendekatan ini memberikan cara yang paling akurat untuk menentukan apakah pengguna berhak untuk menggunakan opsi pembayaran eksternal sesuai dengan pedoman baru Apple.
Mengelola Langganan
Salah satu kelebihan utama menggunakan Stripe untuk pembayaran adalah kemampuan untuk menawarkan dan mengelola langganan. Berikut cara mengelola manajemen langganan di aplikasi Capacitor Anda:
1. Membuat Halaman Manajemen Langganan
Tambahkan halaman manajemen langganan di aplikasi Anda untuk menampilkan langganan aktif pengguna:
<template>
<div class="subscription-manager">
<div v-if="isLoading" class="loading-indicator">
Loading subscription data...
</div>
<div v-else-if="subscription" class="subscription-info">
<h2 class="mb-4 text-xl font-bold">Your Subscription</h2>
<div class="mb-6 plan-details">
<p><span class="font-medium">Plan:</span> {{ subscription.planName }}</p>
<p><span class="font-medium">Status:</span> {{ subscription.status }}</p>
<p><span class="font-medium">Renews:</span> {{ formatDate(subscription.currentPeriodEnd) }}</p>
</div>
<button
@click="manageSubscription"
class="py-3 w-full font-medium text-white bg-indigo-600 rounded-lg transition-colors hover:bg-indigo-700"
>
Manage Subscription
</button>
</div>
<div v-else class="no-subscription">
<p class="mb-4">You don't have an active subscription.</p>
<button
@click="goToPricingPage"
class="py-3 w-full font-medium text-white bg-indigo-600 rounded-lg transition-colors hover:bg-indigo-700"
>
View Plans
</button>
</div>
</div>
</template>
<script setup>
import { ref, onMounted } from 'vue';
import { getUserSubscription } from '../services/subscription';
const subscription = ref(null);
const isLoading = ref(true);
onMounted(async () => {
try {
const userData = await getUserSubscription();
subscription.value = userData.subscription;
} catch (error) {
console.error('Failed to load subscription:', error);
} finally {
isLoading.value = false;
}
});
function formatDate(timestamp) {
return new Date(timestamp * 1000).toLocaleDateString();
}
function manageSubscription() {
// Open Stripe Customer Portal
window.open(subscription.value.portalUrl, '_blank');
}
function goToPricingPage() {
// Navigate to pricing page
// router.push('/pricing');
}
</script>
2. Portal Pelanggan untuk Manajemen Langganan
Stripe menawarkan Portal Pelanggan yang memungkinkan pengguna mengelola langganan mereka. Anda dapat membuat tautan ke portal ini dari server Anda:
// Server-side code (Node.js)
const stripe = require('stripe')('sk_your_stripe_secret_key');
async function createPortalSession(customerId) {
const session = await stripe.billingPortal.sessions.create({
customer: customerId,
return_url: 'https://yourdomain.com/account',
});
return session.url;
}
Menggunakan Pedoman Aplikasi App Store
Untuk memastikan implementasi Anda sesuai dengan pedoman Apple:
- Termasuk diskusi yang tepat tentang pembelian eksternal
- Implementasikan jendela modal yang memberitahu pengguna bahwa mereka meninggalkan aplikasi (seperti yang diperlukan oleh Apple)
- Jangan mencoba untuk menghindari komisi Apple pada pembelian yang dilakukan di dalam aplikasi
- Komunikasikan dengan jelas kepada pengguna bahwa Apple tidak bertanggung jawab atas transaksi
Berikut adalah contoh implementasi jendela modal diskusi yang diperlukan:
import { Dialog } from '@capacitor/dialog';
async function showExternalPaymentDisclosure() {
const { value } = await Dialog.confirm({
title: 'Leaving App for Payment',
message: 'You are about to leave this app to make a payment. Apple is not responsible for the privacy or security of payments that are not made through the App Store. All payment-related issues, including refunds, must be handled by our support team.',
okButtonTitle: 'Continue',
cancelButtonTitle: 'Cancel'
});
return value;
}
export async function initiateExternalPayment(userEmail, userId) {
const userConfirmed = await showExternalPaymentDisclosure();
if (userConfirmed) {
await openPaymentLink(userEmail, userId);
}
}
Menguji Implementasi Anda
Untuk menguji implementasi Anda:
- Klik tombol pembayaran Anda di aplikasi, yang harus menampilkan diskusi dan kemudian membuka halaman pembayaran Stripe
- Lakukan pembayaran uji menggunakan kartu uji Stripe
4242 4242 4242 4242 - Setelah pembayaran, Anda harus diarahkan kembali ke aplikasi Anda melalui tautan universal
- Periksa bahwa webhook Anda menerima
checkout.session.completedevent
Kesimpulan
Fungsi untuk menggunakan opsi pembayaran eksternal untuk barang digital di aplikasi iOS adalah perubahan yang signifikan yang memberikan lebih banyak fleksibilitas bagi pengembang. Meskipun perubahan ini saat ini hanya berlaku untuk aplikasi di App Store Amerika Serikat, namun memberikan alternatif penting dari sistem pembelian dalam aplikasi Apple.
Dengan menggunakan Stripe Payment Links dengan Capacitor, Anda dapat melaksanakan pengalaman pembayaran yang terstruktur dengan cepat sambil mempertahankan konsistensi dengan pedoman Apple. Pendekatan ini juga memberikan keuntungan dari infrastruktur pembayaran yang kuat Stripe, biaya pengolahan yang lebih rendah (3% vs 30%), dan pembayaran yang lebih cepat (hari-hari bukan bulan-bulan) dibandingkan dengan sistem pembelian dalam aplikasi Apple.
Ingatlah bahwa Anda perlu menangani semua masalah dukungan pelanggan dan pengembalian uang secara langsung, karena transaksi ini terjadi di luar ekosistem Apple.
Apakah Anda telah menerapkan Stripe Payment Links di aplikasi Capacitor Anda? Bagikan pengalaman Anda di komentar di bawah!
FAQs
Q: Apakah pendekatan ini konsisten dengan pedoman Apple?
A: Ya, sejak tanggal 1 Mei 2025, Apple memungkinkan penghubungan ke metode pembayaran eksternal untuk barang digital dan layanan di aplikasi yang didistribusikan di App Store Amerika Serikat, selama Anda mencakup diskusi yang diperlukan.
Q: Apakah saya perlu membayar komisi Apple ketika menggunakan metode pembayaran eksternal?
A: Tidak, salah satu keuntungan besar dari aturan baru adalah bahwa pembayaran yang diproses di luar sistem Apple tidak berlaku untuk komisi mereka.
Q: Apakah perusahaan saya perlu berbasis di Amerika Serikat untuk menikmati aturan baru ini?
A: Tidak, perusahaan dari mana saja di dunia dapat menerapkan metode pembayaran eksternal selama aplikasi Anda tersedia di App Store Amerika Serikat dan pengguna yang melakukan pembelian berada di Amerika Serikat. Keputusan ini berlaku untuk pasar (App Store Amerika Serikat) dan lokasi pengguna, bukan lokasi perusahaan Anda. Ini berarti pengembang dari Eropa, Asia, Amerika Selatan, atau tempat lain dapat menerapkan Stripe Payment Links untuk pelanggan Amerika Serikat mereka.
Q: Apa yang terjadi jika pengguna di luar AS mencoba menggunakan opsi pembayaran eksternal?
A: Anda harus menerapkan deteksi wilayah (seperti yang ditunjukkan dalam artikel) untuk hanya menawarkan opsi pembayaran eksternal kepada pengguna di AS. Untuk wilayah lain, Anda harus terus menggunakan sistem pembelian dalam aplikasi Apple.
Q: Apakah saya dapat menggunakan ini untuk barang fisik atau jasa yang dikonsumsi di luar aplikasi?
A: Ya, Apple telah selalu memungkinkan metode pembayaran eksternal untuk barang fisik dan jasa yang dikonsumsi di luar aplikasi (seperti berbagi perjalanan atau pengiriman makanan).