summaryrefslogtreecommitdiff
path: root/app/src/main/java/net/taler/merchantpos/ProcessPayment.kt
blob: 26ef1c7dd5bedd2cb9dfd5037391da8007c10659 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
package net.taler.merchantpos

import android.content.Context
import android.graphics.Bitmap
import android.graphics.Color
import android.net.Uri
import android.os.Bundle
import androidx.fragment.app.Fragment
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import com.google.zxing.BarcodeFormat
import com.google.zxing.common.BitMatrix
import com.google.zxing.qrcode.QRCodeWriter
import android.opengl.ETC1.getWidth
import android.opengl.ETC1.getHeight
import android.os.Handler
import android.util.Log
import android.widget.Button
import android.widget.ImageView
import android.widget.TextView
import androidx.activity.OnBackPressedCallback
import androidx.activity.addCallback
import androidx.lifecycle.ViewModelProviders
import androidx.navigation.findNavController
import androidx.navigation.fragment.findNavController
import com.android.volley.Request
import com.android.volley.RequestQueue
import com.android.volley.Response
import com.android.volley.VolleyError
import com.android.volley.toolbox.Volley
import com.google.android.material.snackbar.Snackbar
import org.json.JSONObject
import java.net.URLEncoder


// TODO: Rename parameter arguments, choose names that match
// the fragment initialization parameters, e.g. ARG_ITEM_NUMBER
private const val ARG_PARAM1 = "param1"
private const val ARG_PARAM2 = "param2"

/**
 * A simple [Fragment] subclass.
 *
 */
class ProcessPayment : Fragment() {

    private var paused: Boolean = true
    private lateinit var queue: RequestQueue
    private lateinit var model: PosTerminalViewModel

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        model = activity?.run {
            ViewModelProviders.of(this)[PosTerminalViewModel::class.java]
        } ?: throw Exception("Invalid Activity")

        queue = Volley.newRequestQueue(context)

    }

    private fun onCheckPayment(checkPaymentResponse: JSONObject) {
        if (paused) {
            return
        }
        //Log.v("taler-merchant", "got check payment result ${checkPaymentResponse}")
        if (checkPaymentResponse.getBoolean("paid")) {
            queue.cancelAll { true }
            findNavController().navigate(R.id.action_processPayment_to_paymentSuccess)
            return
        }
    }

    private fun onNetworkError(volleyError: VolleyError?) {
        val mySnackbar = Snackbar.make(view!!, "Network Error", Snackbar.LENGTH_SHORT)
        mySnackbar.show()
    }

    private fun checkPaid() {
        if (paused) {
            return
        }
        //Log.v("taler-merchant", "checkig if payment happened")
        val params = mapOf("order_id" to model.activeOrderId!!, "instance" to model.merchantConfig!!.instance)
        var req = MerchantInternalRequest(Request.Method.GET, model.merchantConfig!!, "check-payment", params, null,
            Response.Listener { onCheckPayment(it) }, Response.ErrorListener { onNetworkError(it) })
        queue.add(req)
        val handler = Handler()
        handler.postDelayed({
            checkPaid()
        }, 500)

    }

    override fun onResume() {
        this.paused = false
        checkPaid()
        super.onResume()
    }

    override fun onPause() {
        this.paused = true
        super.onPause()
        queue.cancelAll { true }
    }

    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        // Inflate the layout for this fragment
        val view = inflater.inflate(R.layout.fragment_process_payment, container, false)
        val img = view.findViewById<ImageView>(R.id.qrcode)
        val talerPayUrl = "talerpay:" + URLEncoder.encode(model.activeContractUri!!, "utf-8")
        val myBitmap = makeQrCode(talerPayUrl)
        img.setImageBitmap(myBitmap)
        val cancelPaymentButton = view.findViewById<Button>(R.id.button_cancel_payment)
        cancelPaymentButton.setOnClickListener {
            onPaymentCancel()
        }
        val textViewAmount = view.findViewById<TextView>(R.id.text_view_amount)
        textViewAmount.text = model.activeAmountPretty()
        val textViewOrderId = view.findViewById<TextView>(R.id.text_view_order_reference)
        textViewOrderId.text = "Order Reference: " + model.activeOrderId
        return view
    }

    private fun onPaymentCancel() {
        val navController = findNavController()
        navController.popBackStack()

        val mySnackbar = Snackbar.make(view!!, "Payment Canceled", Snackbar.LENGTH_SHORT)
        mySnackbar.show()
    }

    fun makeQrCode(text: String): Bitmap {
        val qrCodeWriter: QRCodeWriter = QRCodeWriter()
        val bitMatrix: BitMatrix = qrCodeWriter.encode(text, BarcodeFormat.QR_CODE, 256, 256)
        val height = bitMatrix.height
        val width = bitMatrix.width
        val bmp = Bitmap.createBitmap(width, height, Bitmap.Config.RGB_565)
        for (x in 0 until width) {
            for (y in 0 until height) {
                bmp.setPixel(x, y, if (bitMatrix.get(x, y)) Color.BLACK else Color.WHITE)
            }
        }
        return bmp
    }
}