Avi Drissman | 8ba1bad | 2022-09-13 19:22:36 | [diff] [blame] | 1 | // Copyright 2014 The Chromium Authors |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
Clark DuVall | 484c256 | 2020-01-23 22:05:09 | [diff] [blame] | 5 | #ifndef COMPONENTS_PERMISSIONS_PERMISSION_REQUEST_H_ |
| 6 | #define COMPONENTS_PERMISSIONS_PERMISSION_REQUEST_H_ |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 7 | |
Jan Wilken Dörrie | ad587c3 | 2021-03-11 14:09:27 | [diff] [blame] | 8 | #include <string> |
| 9 | |
Avi Drissman | 12be031 | 2023-01-11 09:16:09 | [diff] [blame] | 10 | #include "base/functional/callback.h" |
Clark DuVall | 484c256 | 2020-01-23 22:05:09 | [diff] [blame] | 11 | #include "build/build_config.h" |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 12 | #include "components/content_settings/core/common/content_settings.h" |
lshang | ada00c1 | 2016-10-17 04:51:10 | [diff] [blame] | 13 | #include "components/content_settings/core/common/content_settings_types.h" |
Andy Paicu | 4a88f42 | 2020-11-12 18:21:39 | [diff] [blame] | 14 | #include "components/permissions/permission_request_enums.h" |
Illia Klimov | fabd8b5 | 2021-10-21 07:15:40 | [diff] [blame] | 15 | #include "components/permissions/request_type.h" |
Illia Klimov | e406ecc1 | 2022-11-22 15:53:29 | [diff] [blame] | 16 | #include "content/public/browser/global_routing_id.h" |
Anton Bikineev | 1156b5f | 2021-05-15 22:35:36 | [diff] [blame] | 17 | #include "third_party/abseil-cpp/absl/types/optional.h" |
[email protected] | d23cdeee | 2014-03-10 06:39:53 | [diff] [blame] | 18 | #include "url/gurl.h" |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 19 | |
Clark DuVall | 484c256 | 2020-01-23 22:05:09 | [diff] [blame] | 20 | namespace permissions { |
Bret Sepulveda | 362cce4 | 2021-01-13 18:47:54 | [diff] [blame] | 21 | enum class RequestType; |
Clark DuVall | 484c256 | 2020-01-23 22:05:09 | [diff] [blame] | 22 | |
tsergeant | 58defcfb | 2016-07-19 23:47:28 | [diff] [blame] | 23 | // Describes the interface a feature making permission requests should |
| 24 | // implement. A class of this type is registered with the permission request |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 25 | // manager to receive updates about the result of the permissions request |
tsergeant | 58defcfb | 2016-07-19 23:47:28 | [diff] [blame] | 26 | // from the bubble or infobar. It should live until it is unregistered or until |
[email protected] | 634e598 | 2014-04-18 19:20:48 | [diff] [blame] | 27 | // RequestFinished is called. |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 28 | // Note that no particular guarantees are made about what exact UI surface |
| 29 | // is presented to the user. The delegate may be coalesced with other bubble |
| 30 | // requests, or depending on the situation, not shown at all. |
tsergeant | 58defcfb | 2016-07-19 23:47:28 | [diff] [blame] | 31 | class PermissionRequest { |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 32 | public: |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 33 | // If `result` is CONTENT_SETTING_ALLOW, the permission was granted by the |
| 34 | // user. If it's CONTENT_SETTING_BLOCK, the permission was blocked by the |
| 35 | // user. If it's CONTENT_SETTING_DEFAULT, the permission was ignored or |
| 36 | // dismissed without an explicit decision. No other ContentSetting value will |
| 37 | // be passed into this callback. |
| 38 | // If `is_one_time` is true, the decision will last until all tabs of |
| 39 | // `requesting_origin_` are closed or navigated away from. |
| 40 | using PermissionDecidedCallback = |
Illia Klimov | e406ecc1 | 2022-11-22 15:53:29 | [diff] [blame] | 41 | base::RepeatingCallback<void(ContentSetting /*result*/, |
| 42 | bool /*is_one_time*/, |
| 43 | bool /*is_final_decision*/)>; |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 44 | |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 45 | // `permission_decided_callback` is called when the permission request is |
| 46 | // resolved by the user (see comment on PermissionDecidedCallback above). |
| 47 | // `delete_callback` is called when the permission request is no longer needed |
| 48 | // by the permission system. Therefore, it is safe to delete `this` inside |
| 49 | // `delete_callback`. It will always be called eventually by the permission |
| 50 | // system. |
| 51 | // `delete_callback` may be called before `permission_decided_callback`, for |
| 52 | // example if the tab is closed without user interaction. In this case, the |
| 53 | // javascript promise from the requesting origin will not be resolved. |
| 54 | PermissionRequest(const GURL& requesting_origin, |
| 55 | RequestType request_type, |
| 56 | bool has_gesture, |
| 57 | PermissionDecidedCallback permission_decided_callback, |
| 58 | base::OnceClosure delete_callback); |
| 59 | |
| 60 | PermissionRequest(const PermissionRequest&) = delete; |
| 61 | PermissionRequest& operator=(const PermissionRequest&) = delete; |
| 62 | |
Florian Jacky | eeb6206 | 2022-10-05 18:04:07 | [diff] [blame] | 63 | enum ChipTextType { |
| 64 | LOUD_REQUEST, |
| 65 | QUIET_REQUEST, |
| 66 | ALLOW_CONFIRMATION, |
| 67 | BLOCKED_CONFIRMATION, |
| 68 | ACCESSIBILITY_ALLOWED_CONFIRMATION, |
| 69 | ACCESSIBILITY_BLOCKED_CONFIRMATION |
| 70 | }; |
| 71 | |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 72 | virtual ~PermissionRequest(); |
| 73 | |
| 74 | GURL requesting_origin() const { return requesting_origin_; } |
| 75 | RequestType request_type() const { return request_type_; } |
[email protected] | d23cdeee | 2014-03-10 06:39:53 | [diff] [blame] | 76 | |
Bret Sepulveda | d7e4d44 | 2021-04-20 13:46:41 | [diff] [blame] | 77 | // Whether |this| and |other_request| are duplicates and therefore don't both |
| 78 | // need to be shown in the UI. |
| 79 | virtual bool IsDuplicateOf(PermissionRequest* other_request) const; |
| 80 | |
Xiaohan Wang | e81358230 | 2022-01-14 14:50:46 | [diff] [blame] | 81 | #if BUILDFLAG(IS_ANDROID) |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 82 | // Returns prompt text appropriate for displaying in an Android dialog. |
| 83 | virtual std::u16string GetDialogMessageText() const; |
timloh | aa3ce26 | 2017-06-01 05:29:40 | [diff] [blame] | 84 | #endif |
| 85 | |
Xiaohan Wang | e81358230 | 2022-01-14 14:50:46 | [diff] [blame] | 86 | #if !BUILDFLAG(IS_ANDROID) |
Florian Jacky | 4748cf3 | 2022-10-06 09:04:18 | [diff] [blame] | 87 | // Returns whether displaying a confirmation chip for the request is |
| 88 | // supported. |
| 89 | bool IsConfirmationChipSupported(); |
| 90 | |
Illia Klimov | fabd8b5 | 2021-10-21 07:15:40 | [diff] [blame] | 91 | // Returns prompt icon appropriate for displaying on the chip button in the |
| 92 | // location bar. |
| 93 | IconId GetIconForChip(); |
| 94 | |
| 95 | // Returns prompt icon appropriate for displaying on the quiet chip button in |
| 96 | // the location bar. |
| 97 | IconId GetBlockedIconForChip(); |
| 98 | |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 99 | // Returns prompt text appropriate for displaying on the chip button in the |
| 100 | // location bar. |
Florian Jacky | eeb6206 | 2022-10-05 18:04:07 | [diff] [blame] | 101 | absl::optional<std::u16string> GetRequestChipText(ChipTextType type) const; |
Olesia Marukhno | f8a4bed8 | 2020-06-17 13:35:31 | [diff] [blame] | 102 | |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 103 | // Returns prompt text appropriate for displaying under the dialog title |
| 104 | // "[domain] wants to:". |
| 105 | virtual std::u16string GetMessageTextFragment() const; |
Bret Sepulveda | d7e4d44 | 2021-04-20 13:46:41 | [diff] [blame] | 106 | #endif |
[email protected] | dd1ba69 | 2014-01-24 23:17:37 | [diff] [blame] | 107 | |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 108 | // Called when the user has granted the requested permission. |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 109 | // If |is_one_time| is true the permission will last until all tabs of |
| 110 | // |origin| are closed or navigated away from, and then the permission will |
Ravjit Singh Uppal | c73b5a6 | 2020-11-13 01:38:52 | [diff] [blame] | 111 | // automatically expire after 1 day. |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 112 | void PermissionGranted(bool is_one_time); |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 113 | |
| 114 | // Called when the user has denied the requested permission. |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 115 | void PermissionDenied(); |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 116 | |
| 117 | // Called when the user has cancelled the permission request. This |
| 118 | // corresponds to a denial, but is segregated in case the context needs to |
| 119 | // be able to distinguish between an active refusal or an implicit refusal. |
Illia Klimov | e406ecc1 | 2022-11-22 15:53:29 | [diff] [blame] | 120 | void Cancelled(bool is_final_decision = true); |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 121 | |
tsergeant | 58defcfb | 2016-07-19 23:47:28 | [diff] [blame] | 122 | // The UI this request was associated with was answered by the user. |
[email protected] | e2ff17e | 2014-02-06 02:32:33 | [diff] [blame] | 123 | // It is safe for the request to be deleted at this point -- it will receive |
tsergeant | 58defcfb | 2016-07-19 23:47:28 | [diff] [blame] | 124 | // no further message from the permission request system. This method will |
[email protected] | e2ff17e | 2014-02-06 02:32:33 | [diff] [blame] | 125 | // eventually be called on every request which is not unregistered. |
Anatoliy Potapchuk | 1c46f7e | 2020-01-23 13:31:03 | [diff] [blame] | 126 | // It is ok to call this method without actually resolving the request via |
| 127 | // PermissionGranted(), PermissionDenied() or Canceled(). However, it will not |
| 128 | // resolve the javascript promise from the requesting origin. |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 129 | void RequestFinished(); |
benwells | 46b02fa | 2016-04-20 02:37:02 | [diff] [blame] | 130 | |
benwells | 471d1f1 | 2016-07-25 23:58:04 | [diff] [blame] | 131 | // Used to record UMA for whether requests are associated with a user gesture. |
| 132 | // To keep things simple this metric is only recorded for the most popular |
| 133 | // request types. |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 134 | PermissionRequestGestureType GetGestureType() const; |
dominickn | d4e446a | 2016-09-13 07:44:13 | [diff] [blame] | 135 | |
lshang | ada00c1 | 2016-10-17 04:51:10 | [diff] [blame] | 136 | // Used on Android to determine what Android OS permissions are needed for |
| 137 | // this permission request. |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 138 | ContentSettingsType GetContentSettingsType() const; |
lshang | ada00c1 | 2016-10-17 04:51:10 | [diff] [blame] | 139 | |
Illia Klimov | e406ecc1 | 2022-11-22 15:53:29 | [diff] [blame] | 140 | void set_requesting_frame_id(content::GlobalRenderFrameHostId id) { |
| 141 | request_frame_id_ = id; |
| 142 | } |
| 143 | |
| 144 | content::GlobalRenderFrameHostId& get_requesting_frame_id() { |
| 145 | return request_frame_id_; |
| 146 | } |
| 147 | |
dominickn | d4e446a | 2016-09-13 07:44:13 | [diff] [blame] | 148 | private: |
Illia Klimov | e406ecc1 | 2022-11-22 15:53:29 | [diff] [blame] | 149 | content::GlobalRenderFrameHostId request_frame_id_; |
| 150 | |
Bret Sepulveda | 5327d8b5 | 2021-07-21 17:44:23 | [diff] [blame] | 151 | // The origin on whose behalf this permission request is being made. |
| 152 | GURL requesting_origin_; |
| 153 | |
| 154 | // The type of this request. |
| 155 | RequestType request_type_; |
| 156 | |
| 157 | // Whether the request was associated with a user gesture. |
| 158 | bool has_gesture_; |
| 159 | |
| 160 | // Called once a decision is made about the permission. |
| 161 | PermissionDecidedCallback permission_decided_callback_; |
| 162 | |
| 163 | // Called when the request is no longer in use so it can be deleted by the |
| 164 | // caller. |
| 165 | base::OnceClosure delete_callback_; |
[email protected] | efad90f | 2014-01-17 00:45:54 | [diff] [blame] | 166 | }; |
| 167 | |
Clark DuVall | 484c256 | 2020-01-23 22:05:09 | [diff] [blame] | 168 | } // namespace permissions |
| 169 | |
| 170 | #endif // COMPONENTS_PERMISSIONS_PERMISSION_REQUEST_H_ |