blob: 78b35a0c38734c23f18a0e2baee7ee774d7f1aea [file] [log] [blame]
krb7f6421b2016-11-18 17:46:211// Copyright 2016 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
rouslan908248c2017-02-27 21:30:245#ifndef COMPONENTS_PAYMENTS_CONTENT_PAYMENT_REQUEST_H_
6#define COMPONENTS_PAYMENTS_CONTENT_PAYMENT_REQUEST_H_
mathpf709499d2017-01-09 20:48:367
8#include <memory>
tmartino68c0a272017-01-19 17:44:089#include <vector>
krb7f6421b2016-11-18 17:46:2110
rouslan908248c2017-02-27 21:30:2411#include "base/macros.h"
Anthony Vallee-Duboisdc1dbf1a2017-07-17 15:01:1312#include "base/memory/weak_ptr.h"
Rouslan Solomakhin27064702018-12-14 21:15:3313#include "components/payments/content/developer_console_logger.h"
Anthony Vallee-Duboisc7ae7332017-12-19 20:44:0714#include "components/payments/content/payment_request_display_manager.h"
mathpf1a7a3752017-03-15 11:23:3715#include "components/payments/content/payment_request_spec.h"
16#include "components/payments/content/payment_request_state.h"
sebsg20b49d7b2017-05-04 20:23:1717#include "components/payments/core/journey_logger.h"
krb7f6421b2016-11-18 17:46:2118#include "mojo/public/cpp/bindings/binding.h"
rouslan8fdbfb242017-03-15 01:12:4919#include "mojo/public/cpp/bindings/interface_request.h"
Han Leonabbe7c902018-08-30 02:08:1020#include "third_party/blink/public/mojom/payments/payment_request.mojom.h"
Rouslan Solomakhin6e979ab2017-08-30 17:30:3921#include "url/gurl.h"
krb7f6421b2016-11-18 17:46:2122
23namespace content {
rouslan690997682017-05-09 18:07:3924class RenderFrameHost;
krb7f6421b2016-11-18 17:46:2125class WebContents;
Rouslan Solomakhin115f7232017-08-01 15:24:3826} // namespace content
krb7f6421b2016-11-18 17:46:2127
28namespace payments {
29
Rouslan Solomakhin4eea9bc22017-10-10 15:18:5130class ContentPaymentRequestDelegate;
mathpf709499d2017-01-09 20:48:3631class PaymentRequestWebContentsManager;
32
mathpf1a7a3752017-03-15 11:23:3733// This class manages the interaction between the renderer (through the
34// PaymentRequestClient and Mojo stub implementation) and the UI (through the
35// PaymentRequestDelegate). The API user (merchant) specification (supported
36// payment methods, required information, order details) is stored in
37// PaymentRequestSpec, and the current user selection state (and related data)
38// is stored in PaymentRequestSpec.
mathp4b85b582017-03-08 21:07:1639class PaymentRequest : public mojom::PaymentRequest,
mathpf1a7a3752017-03-15 11:23:3740 public PaymentRequestSpec::Observer,
41 public PaymentRequestState::Delegate {
krb7f6421b2016-11-18 17:46:2142 public:
mathp300fa542017-03-27 19:29:3743 class ObserverForTest {
44 public:
45 virtual void OnCanMakePaymentCalled() = 0;
gogerald8189d522017-09-15 17:52:1846 virtual void OnCanMakePaymentReturned() = 0;
Danyao Wangce175bf2018-12-21 22:35:5847 virtual void OnHasEnrolledInstrumentCalled() = 0;
48 virtual void OnHasEnrolledInstrumentReturned() = 0;
rouslan6e3cf7c62017-04-17 21:23:2849 virtual void OnNotSupportedError() = 0;
rouslanb28f4532017-05-08 15:41:4750 virtual void OnConnectionTerminated() = 0;
Anthony Vallee-Dubois6813c1442017-05-17 19:32:5651 virtual void OnAbortCalled() = 0;
mathp300fa542017-03-27 19:29:3752
53 protected:
54 virtual ~ObserverForTest() {}
55 };
56
rouslan690997682017-05-09 18:07:3957 PaymentRequest(content::RenderFrameHost* render_frame_host,
58 content::WebContents* web_contents,
Rouslan Solomakhin4eea9bc22017-10-10 15:18:5159 std::unique_ptr<ContentPaymentRequestDelegate> delegate,
rouslan8fdbfb242017-03-15 01:12:4960 PaymentRequestWebContentsManager* manager,
Anthony Vallee-Duboisc7ae7332017-12-19 20:44:0761 PaymentRequestDisplayManager* display_manager,
mathp300fa542017-03-27 19:29:3762 mojo::InterfaceRequest<mojom::PaymentRequest> request,
63 ObserverForTest* observer_for_testing);
mathpf709499d2017-01-09 20:48:3664 ~PaymentRequest() override;
krb7f6421b2016-11-18 17:46:2165
rouslan8fdbfb242017-03-15 01:12:4966 // mojom::PaymentRequest
67 void Init(mojom::PaymentRequestClientPtr client,
68 std::vector<mojom::PaymentMethodDataPtr> method_data,
69 mojom::PaymentDetailsPtr details,
70 mojom::PaymentOptionsPtr options) override;
Rouslan Solomakhin833f8512018-04-03 23:19:2571 void Show(bool is_user_gesture) override;
Jinho Bangbe463a22018-08-02 10:26:5072 void Retry(mojom::PaymentValidationErrorsPtr errors) override;
mathp151bd31e2017-04-03 21:07:2473 void UpdateWith(mojom::PaymentDetailsPtr details) override;
Rouslan Solomakhina9ff9282017-10-31 21:58:0574 void NoUpdatedPaymentDetails() override;
mathpf4bc50e2017-01-24 05:17:5075 void Abort() override;
rouslan8fdbfb242017-03-15 01:12:4976 void Complete(mojom::PaymentComplete result) override;
mathp4b85b582017-03-08 21:07:1677 void CanMakePayment() override;
Rouslan Solomakhin5683eb282019-01-29 18:06:0378 void HasEnrolledInstrument(bool per_method_quota) override;
mathp4b85b582017-03-08 21:07:1679
mathpf1a7a3752017-03-15 11:23:3780 // PaymentRequestSpec::Observer:
mathp151bd31e2017-04-03 21:07:2481 void OnSpecUpdated() override {}
mathpf1a7a3752017-03-15 11:23:3782
83 // PaymentRequestState::Delegate:
mathpf1a7a3752017-03-15 11:23:3784 void OnPaymentResponseAvailable(mojom::PaymentResponsePtr response) override;
mathp151bd31e2017-04-03 21:07:2485 void OnShippingOptionIdSelected(std::string shipping_option_id) override;
86 void OnShippingAddressSelected(mojom::PaymentAddressPtr address) override;
Jinho Bangbb178152018-09-13 09:44:4387 void OnPayerInfoSelected(mojom::PayerDetailPtr payer_info) override;
krb7f6421b2016-11-18 17:46:2188
Rouslan Solomakhin16ee55202017-10-03 19:04:4289 // Called when the user explicitly cancelled the flow. Will send a message
mathpf4bc50e2017-01-24 05:17:5090 // to the renderer which will indirectly destroy this object (through
91 // OnConnectionTerminated).
92 void UserCancelled();
93
sebsgd56b3e422017-10-20 18:08:0894 // Called when the main frame attached to this PaymentRequest is navigating to
95 // another document, but before the PaymentRequest is destroyed.
96 void DidStartMainFrameNavigationToDifferentDocument(bool is_user_initiated);
sebsg2c8558a2017-05-17 18:54:1097
mathpf4bc50e2017-01-24 05:17:5098 // As a result of a browser-side error or renderer-initiated mojo channel
99 // closure (e.g. there was an error on the renderer side, or payment was
100 // successful), this method is called. It is responsible for cleaning up,
101 // such as possibly closing the dialog.
102 void OnConnectionTerminated();
mathp6758be032017-01-13 04:49:50103
mathpd4be8de82017-03-01 00:51:48104 // Called when the user clicks on the "Pay" button.
105 void Pay();
106
Anthony Vallee-Duboisc7ae7332017-12-19 20:44:07107 // Hide this Payment Request if it's already showing.
108 void HideIfNecessary();
109
Rouslan Solomakhind2cae95a2018-08-09 00:16:10110 // Record the "dialog shown" event in the journey logger.
111 void RecordDialogShownEventInJourneyLogger();
112
Anthony Vallee-Dubois10d131a2018-02-22 15:41:04113 bool IsIncognito() const;
114
anthonyvd3d7f9722016-12-07 18:43:54115 content::WebContents* web_contents() { return web_contents_; }
116
gogerald2ddfc64c2018-08-23 16:20:21117 bool skipped_payment_request_ui() { return skipped_payment_request_ui_; }
118
mathpf1a7a3752017-03-15 11:23:37119 PaymentRequestSpec* spec() { return spec_.get(); }
120 PaymentRequestState* state() { return state_.get(); }
mathpd4be8de82017-03-01 00:51:48121
Anthony Vallee-Dubois968ae4d2018-03-15 16:56:36122 PaymentRequestSpec* spec() const { return spec_.get(); }
123 PaymentRequestState* state() const { return state_.get(); }
124
krb7f6421b2016-11-18 17:46:21125 private:
Rouslan Solomakhin27064702018-12-14 21:15:33126 // Returns true after init() has been called and the mojo connection has been
127 // established. If the mojo connection gets later disconnected, this will
128 // returns false.
129 bool IsInitialized() const;
130
131 // Returns true after show() has been called and the payment sheet is showing.
132 // If the payment sheet is later hidden, this will return false.
133 bool IsThisPaymentRequestShowing() const;
134
gogerald2ddfc64c2018-08-23 16:20:21135 // Returns true if this payment request supports skipping the Payment Sheet.
136 // Typically, this means only one payment method is supported, it's a URL
137 // based method, and no other info is requested from the user.
138 bool SatisfiesSkipUIConstraints() const;
139
sebsgfcdd13c2017-06-08 15:49:33140 // Only records the abort reason if it's the first completion for this Payment
141 // Request. This is necessary since the aborts cascade into one another with
142 // the first one being the most precise.
143 void RecordFirstAbortReason(JourneyLogger::AbortReason completion_status);
sebsg2c8558a2017-05-17 18:54:10144
gogerald0a7ee6c2017-11-13 18:23:19145 // The callback for PaymentRequestState::CanMakePayment. Checks for query
146 // quota and may send QUERY_QUOTA_EXCEEDED.
gogerald8189d522017-09-15 17:52:18147 void CanMakePaymentCallback(bool can_make_payment);
148
Danyao Wangce175bf2018-12-21 22:35:58149 // The callback for PaymentRequestState::HasEnrolledInstrument. Checks for
150 // query quota and may send QUERY_QUOTA_EXCEEDED.
Rouslan Solomakhin5683eb282019-01-29 18:06:03151 void HasEnrolledInstrumentCallback(bool per_method_quota,
152 bool has_enrolled_instrument);
Danyao Wangce175bf2018-12-21 22:35:58153
gogerald0a7ee6c2017-11-13 18:23:19154 // The callback for PaymentRequestState::AreRequestedMethodsSupported.
155 void AreRequestedMethodsSupportedCallback(bool methods_supported);
156
Danyao Wangce175bf2018-12-21 22:35:58157 // Sends either HAS_ENROLLED_INSTRUMENT or HAS_NO_ENROLLED_INSTRUMENT to the
Danyao Wang4bc0606a2018-12-27 16:54:53158 // renderer, depending on |has_enrolled_instrument| value. Does not check
159 // query quota so never sends QUERY_QUOTA_EXCEEDED. If
160 // |warn_localhost_or_file| is true, then sends
Danyao Wangce175bf2018-12-21 22:35:58161 // WARNING_HAS_ENROLLED_INSTRUMENT or WARNING_HAS_NO_ENROLLED_INSTRUMENT
162 // version of the values instead.
163 void RespondToHasEnrolledInstrumentQuery(bool has_enrolled_instrument,
164 bool warn_localhost_or_file);
165
krb7f6421b2016-11-18 17:46:21166 content::WebContents* web_contents_;
Rouslan Solomakhin27064702018-12-14 21:15:33167 DeveloperConsoleLogger log_;
Rouslan Solomakhin4eea9bc22017-10-10 15:18:51168 std::unique_ptr<ContentPaymentRequestDelegate> delegate_;
mathpf709499d2017-01-09 20:48:36169 // |manager_| owns this PaymentRequest.
170 PaymentRequestWebContentsManager* manager_;
Anthony Vallee-Duboisc7ae7332017-12-19 20:44:07171 PaymentRequestDisplayManager* display_manager_;
172 std::unique_ptr<PaymentRequestDisplayManager::DisplayHandle> display_handle_;
rouslan8fdbfb242017-03-15 01:12:49173 mojo::Binding<mojom::PaymentRequest> binding_;
174 mojom::PaymentRequestClientPtr client_;
mathpd4be8de82017-03-01 00:51:48175
mathpf1a7a3752017-03-15 11:23:37176 std::unique_ptr<PaymentRequestSpec> spec_;
177 std::unique_ptr<PaymentRequestState> state_;
anthonyvd3d7f9722016-12-07 18:43:54178
Rouslan Solomakhin115f7232017-08-01 15:24:38179 // The RFC 6454 origin of the top level frame that has invoked PaymentRequest
180 // API. This is what the user sees in the address bar.
Rouslan Solomakhin6e979ab2017-08-30 17:30:39181 const GURL top_level_origin_;
Rouslan Solomakhin115f7232017-08-01 15:24:38182
rouslan690997682017-05-09 18:07:39183 // The RFC 6454 origin of the frame that has invoked PaymentRequest API. This
184 // can be either the main frame or an iframe.
Rouslan Solomakhin6e979ab2017-08-30 17:30:39185 const GURL frame_origin_;
rouslan690997682017-05-09 18:07:39186
mathp300fa542017-03-27 19:29:37187 // May be null, must outlive this object.
188 ObserverForTest* observer_for_testing_;
189
sebsg20b49d7b2017-05-04 20:23:17190 JourneyLogger journey_logger_;
191
sebsgfcdd13c2017-06-08 15:49:33192 // Whether a completion was already recorded for this Payment Request.
193 bool has_recorded_completion_ = false;
sebsg2c8558a2017-05-17 18:54:10194
Rouslan Solomakhin833f8512018-04-03 23:19:25195 // Whether PaymentRequest.show() was invoked with a user gesture.
196 bool is_show_user_gesture_ = false;
197
gogerald2ddfc64c2018-08-23 16:20:21198 // Whether PaymentRequest.show() was invoked by skipping payment request UI.
199 bool skipped_payment_request_ui_ = false;
200
Rouslan Solomakhin27064702018-12-14 21:15:33201 // Whether PaymentRequest mojo connection has been initialized from the
202 // renderer.
203 bool is_initialized_ = false;
204
205 // Whether PaymentRequest.show() has been called.
206 bool is_show_called_ = false;
207
Anthony Vallee-Duboisdc1dbf1a2017-07-17 15:01:13208 base::WeakPtrFactory<PaymentRequest> weak_ptr_factory_;
209
mathpf709499d2017-01-09 20:48:36210 DISALLOW_COPY_AND_ASSIGN(PaymentRequest);
krb7f6421b2016-11-18 17:46:21211};
212
213} // namespace payments
214
rouslan908248c2017-02-27 21:30:24215#endif // COMPONENTS_PAYMENTS_CONTENT_PAYMENT_REQUEST_H_