/chrome/browser/commerce/merchant_viewer/android/java/src/org/chromium/chrome/browser/merchant_viewer/MerchantTrustMetrics.java

https://github.com/chromium/chromium · Java · 273 lines · 191 code · 32 blank · 50 comment · 21 complexity · d6a6f13729ebe4b97f49668765000978 MD5 · raw file

  1. // Copyright 2021 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. package org.chromium.chrome.browser.merchant_viewer;
  5. import androidx.annotation.IntDef;
  6. import org.chromium.base.TimeUtils;
  7. import org.chromium.base.metrics.RecordHistogram;
  8. import org.chromium.base.metrics.RecordUserAction;
  9. import org.chromium.components.browser_ui.bottomsheet.BottomSheetController.StateChangeReason;
  10. import org.chromium.components.messages.DismissReason;
  11. import java.lang.annotation.Retention;
  12. import java.lang.annotation.RetentionPolicy;
  13. /**
  14. * Metrics util class for merchant trust.
  15. */
  16. public class MerchantTrustMetrics {
  17. /**
  18. * The reason why we clear the prepared message.
  19. *
  20. * Needs to stay in sync with MerchantTrustMessageClearReason in enums.xml. These values are
  21. * persisted to logs. Entries should not be renumbered and numeric values should never be
  22. * reused.
  23. */
  24. @IntDef({MessageClearReason.UNKNOWN, MessageClearReason.NAVIGATE_TO_SAME_DOMAIN,
  25. MessageClearReason.NAVIGATE_TO_DIFFERENT_DOMAIN,
  26. MessageClearReason.MESSAGE_CONTEXT_NO_LONGER_VALID,
  27. MessageClearReason.SWITCH_TO_DIFFERENT_WEBCONTENTS})
  28. @Retention(RetentionPolicy.SOURCE)
  29. public @interface MessageClearReason {
  30. int UNKNOWN = 0;
  31. int NAVIGATE_TO_SAME_DOMAIN = 1;
  32. int NAVIGATE_TO_DIFFERENT_DOMAIN = 2;
  33. int MESSAGE_CONTEXT_NO_LONGER_VALID = 3;
  34. int SWITCH_TO_DIFFERENT_WEBCONTENTS = 4;
  35. // Always update MAX_VALUE to match the last reason in the list.
  36. int MAX_VALUE = 4;
  37. }
  38. /**
  39. * Which ui the bottom sheet is opened from.
  40. *
  41. * Needs to stay in sync with MerchantTrustBottomSheetOpenedSource in enums.xml. These values
  42. * are persisted to logs. Entries should not be renumbered and numeric values should never be
  43. * reused.
  44. */
  45. @IntDef({BottomSheetOpenedSource.UNKNOWN, BottomSheetOpenedSource.FROM_MESSAGE,
  46. BottomSheetOpenedSource.FROM_PAGE_INFO})
  47. @Retention(RetentionPolicy.SOURCE)
  48. public @interface BottomSheetOpenedSource {
  49. int UNKNOWN = 0;
  50. int FROM_MESSAGE = 1;
  51. int FROM_PAGE_INFO = 2;
  52. // Always update MAX_VALUE to match the last item in the list.
  53. int MAX_VALUE = 2;
  54. }
  55. // Metrics for merchant trust signal message.
  56. private boolean mDidRecordMessagePrepared;
  57. private boolean mDidRecordMessageShown;
  58. private long mMessagePreparedNanoseconds;
  59. private long mMessageVisibleNanoseconds;
  60. // Metrics for merchant trust detailed bottom sheet.
  61. private boolean mDidRecordBottomSheetFirstPeek;
  62. private boolean mDidRecordBottomSheetFirstHalfOpen;
  63. private boolean mDidRecordBottomSheetFirstFullyOpen;
  64. private boolean mIsBottomSheetHalfViewed;
  65. private boolean mIsBottomSheetFullyViewed;
  66. private long mBottomSheetPeekedNanoseconds;
  67. private long mBottomSheetHalfOpenedNanoseconds;
  68. private long mBottomSheetFullyOpenedNanoseconds;
  69. /** Records metrics when the message is prepared. */
  70. public void recordMetricsForMessagePrepared() {
  71. startMessagePreparedTimer();
  72. }
  73. /** Records metrics when user leaves a rate-eligible page without seeing the message. */
  74. public void recordMetricsForMessageCleared(@MessageClearReason int clearReason) {
  75. finishMessagePreparedTimer();
  76. RecordHistogram.recordEnumeratedHistogram(
  77. "MerchantTrust.Message.ClearReason", clearReason, MessageClearReason.MAX_VALUE + 1);
  78. resetMessageMetrics();
  79. }
  80. /** Records metrics when the message is shown. */
  81. public void recordMetricsForMessageShown() {
  82. startMessageShownTimer();
  83. finishMessagePreparedTimer();
  84. }
  85. /** Records metrics when the message is dismissed. */
  86. public void recordMetricsForMessageDismissed(@DismissReason int dismissReason) {
  87. finishMessageShownTimer();
  88. RecordHistogram.recordEnumeratedHistogram(
  89. "MerchantTrust.Message.DismissReason", dismissReason, DismissReason.COUNT);
  90. resetMessageMetrics();
  91. }
  92. /** Records metrics when the message is tapped. */
  93. public void recordMetricsForMessageTapped() {
  94. finishMessageShownTimer();
  95. RecordUserAction.record("MerchantTrust.Message.Tapped");
  96. resetMessageMetrics();
  97. }
  98. /** Starts timing when the message is prepared. */
  99. private void startMessagePreparedTimer() {
  100. mMessagePreparedNanoseconds = System.nanoTime();
  101. }
  102. /** Finishes timing when the message is cleared or shown. */
  103. private void finishMessagePreparedTimer() {
  104. if (!mDidRecordMessagePrepared && mMessagePreparedNanoseconds != 0) {
  105. mDidRecordMessagePrepared = true;
  106. long durationPrepared = (System.nanoTime() - mMessagePreparedNanoseconds)
  107. / TimeUtils.NANOSECONDS_PER_MILLISECOND;
  108. RecordHistogram.recordMediumTimesHistogram(
  109. "MerchantTrust.Message.DurationPrepared", durationPrepared);
  110. }
  111. }
  112. /** Starts timing when the message is shown. */
  113. private void startMessageShownTimer() {
  114. mMessageVisibleNanoseconds = System.nanoTime();
  115. }
  116. /** Finishes timing when the message is dismissed or tapped. */
  117. private void finishMessageShownTimer() {
  118. if (!mDidRecordMessageShown && mMessageVisibleNanoseconds != 0) {
  119. mDidRecordMessageShown = true;
  120. long durationShow = (System.nanoTime() - mMessageVisibleNanoseconds)
  121. / TimeUtils.NANOSECONDS_PER_MILLISECOND;
  122. RecordHistogram.recordMediumTimesHistogram(
  123. "MerchantTrust.Message.DurationShown", durationShow);
  124. }
  125. }
  126. /** Resets all message-related metrics. */
  127. private void resetMessageMetrics() {
  128. mDidRecordMessagePrepared = false;
  129. mDidRecordMessageShown = false;
  130. mMessagePreparedNanoseconds = 0;
  131. mMessageVisibleNanoseconds = 0;
  132. }
  133. /** Records metrics for the peeked panel state. */
  134. public void recordMetricsForBottomSheetPeeked() {
  135. startBottomSheetPeekTimer();
  136. finishBottomSheetHalfOpenTimer();
  137. finishBottomSheetFullyOpenTimer();
  138. }
  139. /** Records metrics when the panel has been half opened. */
  140. public void recordMetricsForBottomSheetHalfOpened() {
  141. mIsBottomSheetHalfViewed = true;
  142. startBottomSheetHalfOpenTimer();
  143. finishBottomSheetPeekTimer();
  144. finishBottomSheetFullyOpenTimer();
  145. }
  146. /** Records metrics when the panel has been fully opened. */
  147. public void recordMetricsForBottomSheetFullyOpened() {
  148. mIsBottomSheetFullyViewed = true;
  149. startBottomSheetFullyOpenTimer();
  150. finishBottomSheetPeekTimer();
  151. finishBottomSheetHalfOpenTimer();
  152. }
  153. /** Records metrics when the panel has been closed. */
  154. public void recordMetricsForBottomSheetClosed(@StateChangeReason int stateChangeReason) {
  155. finishBottomSheetPeekTimer();
  156. finishBottomSheetHalfOpenTimer();
  157. finishBottomSheetFullyOpenTimer();
  158. RecordHistogram.recordBooleanHistogram(
  159. "MerchantTrust.BottomSheet.IsHalfViewed", mIsBottomSheetHalfViewed);
  160. RecordHistogram.recordBooleanHistogram(
  161. "MerchantTrust.BottomSheet.IsFullyViewed", mIsBottomSheetFullyViewed);
  162. RecordHistogram.recordEnumeratedHistogram("MerchantTrust.BottomSheet.CloseReason",
  163. stateChangeReason, StateChangeReason.MAX_VALUE + 1);
  164. resetBottomSheetMetrics();
  165. }
  166. /** Records a user action that navigates to a new link on the bottom sheet. */
  167. public void recordNavigateLinkOnBottomSheet() {
  168. RecordUserAction.record("MerchantTrust.BottomSheet.NavigateLink");
  169. }
  170. /** Resets all bottom sheet-related metrics. */
  171. private void resetBottomSheetMetrics() {
  172. mDidRecordBottomSheetFirstPeek = false;
  173. mDidRecordBottomSheetFirstHalfOpen = false;
  174. mDidRecordBottomSheetFirstFullyOpen = false;
  175. mIsBottomSheetHalfViewed = false;
  176. mIsBottomSheetFullyViewed = false;
  177. mBottomSheetPeekedNanoseconds = 0;
  178. mBottomSheetHalfOpenedNanoseconds = 0;
  179. mBottomSheetFullyOpenedNanoseconds = 0;
  180. }
  181. /** Starts timing the peek state if it's not already been started. */
  182. private void startBottomSheetPeekTimer() {
  183. if (mBottomSheetPeekedNanoseconds == 0) {
  184. mBottomSheetPeekedNanoseconds = System.nanoTime();
  185. }
  186. }
  187. /** Finishes timing metrics for the first peek state, unless that has already been done. */
  188. private void finishBottomSheetPeekTimer() {
  189. if (!mDidRecordBottomSheetFirstPeek && mBottomSheetPeekedNanoseconds != 0) {
  190. mDidRecordBottomSheetFirstPeek = true;
  191. long durationPeeking = (System.nanoTime() - mBottomSheetPeekedNanoseconds)
  192. / TimeUtils.NANOSECONDS_PER_MILLISECOND;
  193. RecordHistogram.recordMediumTimesHistogram(
  194. "MerchantTrust.BottomSheet.DurationPeeked", durationPeeking);
  195. }
  196. }
  197. /** Starts timing the half open state if it's not already been started. */
  198. private void startBottomSheetHalfOpenTimer() {
  199. if (mBottomSheetHalfOpenedNanoseconds == 0) {
  200. mBottomSheetHalfOpenedNanoseconds = System.nanoTime();
  201. }
  202. }
  203. /** Finishes timing metrics for the first half open state, unless that has already been done. */
  204. private void finishBottomSheetHalfOpenTimer() {
  205. if (!mDidRecordBottomSheetFirstHalfOpen && mBottomSheetHalfOpenedNanoseconds != 0) {
  206. mDidRecordBottomSheetFirstHalfOpen = true;
  207. long durationOpened = (System.nanoTime() - mBottomSheetHalfOpenedNanoseconds)
  208. / TimeUtils.NANOSECONDS_PER_MILLISECOND;
  209. RecordHistogram.recordMediumTimesHistogram(
  210. "MerchantTrust.BottomSheet.DurationHalfOpened", durationOpened);
  211. }
  212. }
  213. /** Starts timing the fully open state if it's not already been started. */
  214. private void startBottomSheetFullyOpenTimer() {
  215. if (mBottomSheetFullyOpenedNanoseconds == 0) {
  216. mBottomSheetFullyOpenedNanoseconds = System.nanoTime();
  217. }
  218. }
  219. /**
  220. * Finishes timing metrics for the first fully open state, unless that has already been done.
  221. */
  222. private void finishBottomSheetFullyOpenTimer() {
  223. if (!mDidRecordBottomSheetFirstFullyOpen && mBottomSheetFullyOpenedNanoseconds != 0) {
  224. mDidRecordBottomSheetFirstFullyOpen = true;
  225. long durationOpened = (System.nanoTime() - mBottomSheetFullyOpenedNanoseconds)
  226. / TimeUtils.NANOSECONDS_PER_MILLISECOND;
  227. RecordHistogram.recordMediumTimesHistogram(
  228. "MerchantTrust.BottomSheet.DurationFullyOpened", durationOpened);
  229. }
  230. }
  231. /** Records metrics when the page info is opened. */
  232. public void recordMetricsForStoreInfoRowVisible(boolean visible) {
  233. RecordHistogram.recordBooleanHistogram(
  234. "MerchantTrust.PageInfo.IsStoreInfoVisible", visible);
  235. }
  236. /** Records metrics for the bottom sheet opened source. */
  237. public void recordMetricsForBottomSheetOpenedSource(@BottomSheetOpenedSource int source) {
  238. RecordHistogram.recordEnumeratedHistogram("MerchantTrust.BottomSheet.OpenSource", source,
  239. BottomSheetOpenedSource.MAX_VALUE + 1);
  240. }
  241. }