From c11b579d594907a87cf79ae5378b9c4c24ac8391 Mon Sep 17 00:00:00 2001 From: kean Date: Wed, 22 Mar 2023 11:54:12 -0400 Subject: [PATCH 1/9] Remove unused code from BlogDashboardCardFrameView The chevronImage and the iconImageView were removed during the recent dashboard card redesign --- .../Posts/BlogDashboardCardFrameView.swift | 53 ++----------------- .../Posts/DashboardPostsListCardCell.swift | 1 - .../Prompts/DashboardPromptsCardCell.swift | 1 - .../DashboardQuickStartCardCell.swift | 1 - .../Cards/Stats/DashboardStatsCardCell.swift | 1 - .../View/JetpackRemoteInstallCardView.swift | 1 - 6 files changed, 3 insertions(+), 55 deletions(-) diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift index a65ed8154bb5..3eaa9e67a069 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift @@ -25,16 +25,6 @@ class BlogDashboardCardFrameView: UIView { return topStackView }() - /// Card's icon image view - private lazy var iconImageView: UIImageView = { - let iconImageView = UIImageView(image: UIImage.gridicon(.posts, size: Constants.iconSize).withRenderingMode(.alwaysTemplate)) - iconImageView.tintColor = .label - iconImageView.frame = CGRect(x: 0, y: 0, width: Constants.iconSize.width, height: Constants.iconSize.height) - iconImageView.setContentHuggingPriority(.defaultHigh, for: .horizontal) - iconImageView.isAccessibilityElement = false - return iconImageView - }() - /// Card's title private lazy var titleLabel: UILabel = { let titleLabel = UILabel() @@ -45,19 +35,6 @@ class BlogDashboardCardFrameView: UIView { return titleLabel }() - /// Chevron displayed in case there's any action associated - private lazy var chevronImageView: UIImageView = { - let chevronImage = UIImage.gridicon(.chevronRight, size: Constants.iconSize).withRenderingMode(.alwaysTemplate) - let chevronImageView = UIImageView(image: chevronImage.imageFlippedForRightToLeftLayoutDirection()) - chevronImageView.frame = CGRect(x: 0, y: 0, width: Constants.iconSize.width, height: Constants.iconSize.height) - chevronImageView.tintColor = .listIcon - chevronImageView.setContentHuggingPriority(.defaultHigh, for: .horizontal) - chevronImageView.isAccessibilityElement = false - chevronImageView.isHidden = true - chevronImageView.setContentHuggingPriority(.defaultHigh, for: .horizontal) - return chevronImageView - }() - /// Ellipsis Button displayed on the top right corner of the view. /// Displayed only when an associated action is set private(set) lazy var ellipsisButton: UIButton = { @@ -102,19 +79,10 @@ class BlogDashboardCardFrameView: UIView { } } - /// The icon to be displayed at the header - var icon: UIImage? { - didSet { - iconImageView.image = icon?.withRenderingMode(.alwaysTemplate) - iconImageView.isHidden = icon == nil - } - } - /// Closure to be called when anywhere in the view is tapped. /// If set, the chevron image is displayed. var onViewTap: (() -> Void)? { didSet { - updateChevronImageState() addViewTapGestureIfNeeded() } } @@ -124,10 +92,8 @@ class BlogDashboardCardFrameView: UIView { /// If set, the chevron image is displayed. var onHeaderTap: (() -> Void)? { didSet { - updateChevronImageState() addHeaderTapGestureIfNeeded() } - } /// Closure to be called when the ellipsis button is tapped.. @@ -177,7 +143,7 @@ class BlogDashboardCardFrameView: UIView { headerStackView.isHidden = true buttonContainerStackView.isHidden = false - if !ellipsisButton.isHidden || !chevronImageView.isHidden { + if !ellipsisButton.isHidden { mainStackViewTrailingConstraint?.constant = -Constants.mainStackViewTrailingPadding } } @@ -228,30 +194,17 @@ class BlogDashboardCardFrameView: UIView { ellipsisButton.setImage(UIImage.gridicon(.ellipsis).imageWithTintColor(.listIcon), for: .normal) } - private func updateChevronImageState() { - chevronImageView.isHidden = onViewTap == nil && onHeaderTap == nil - assertOnTapRecognitionCorrectUsage() - } - private func updateEllipsisButtonState() { ellipsisButton.isHidden = onEllipsisButtonTap == nil let headerPadding = ellipsisButton.isHidden ? Constants.headerPaddingWithEllipsisButtonHidden : Constants.headerPaddingWithEllipsisButtonShown headerStackView.layoutMargins = headerPadding - assertOnTapRecognitionCorrectUsage() - } - - /// Only one of two types of action should be associated with the card. - /// Either ellipsis button tap, or view/header tap - private func assertOnTapRecognitionCorrectUsage() { - let bothTypesUsed = (onViewTap != nil || onHeaderTap != nil) && onEllipsisButtonTap != nil - assert(!bothTypesUsed, "Using onViewTap or onHeaderTap alongside onEllipsisButtonTap is not supported and will result in unexpected behavior.") } private func addHeaderTapGestureIfNeeded() { // Reset any previously added gesture recognizers - headerStackView.gestureRecognizers?.forEach {headerStackView.removeGestureRecognizer($0)} + headerStackView.gestureRecognizers?.forEach { headerStackView.removeGestureRecognizer($0) } // Add gesture recognizer if needed if onHeaderTap != nil { @@ -262,7 +215,7 @@ class BlogDashboardCardFrameView: UIView { private func addViewTapGestureIfNeeded() { // Reset any previously added gesture recognizers - self.gestureRecognizers?.forEach {self.removeGestureRecognizer($0)} + self.gestureRecognizers?.forEach { self.removeGestureRecognizer($0) } // Add gesture recognizer if needed if onViewTap != nil { diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift index 2dc355cd48df..630b486887e7 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift @@ -69,7 +69,6 @@ class DashboardPostsListCardCell: UICollectionViewCell, Reusable { private func addSubviews() { let frameView = BlogDashboardCardFrameView() - frameView.icon = UIImage.gridicon(.posts, size: Constants.iconSize) frameView.translatesAutoresizingMaskIntoConstraints = false frameView.add(subview: tableView) diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift index 4cae4f76fb01..1c67645a8858 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift @@ -11,7 +11,6 @@ class DashboardPromptsCardCell: UICollectionViewCell, Reusable { let frameView = BlogDashboardCardFrameView() frameView.translatesAutoresizingMaskIntoConstraints = false frameView.title = Strings.cardFrameTitle - frameView.icon = Style.frameIconImage // NOTE: Remove the logic when support for iOS 14 is dropped if #available (iOS 15.0, *) { diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Quick Start/DashboardQuickStartCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Quick Start/DashboardQuickStartCardCell.swift index 38a689f9bb7b..fdc9eafeed2f 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Quick Start/DashboardQuickStartCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Quick Start/DashboardQuickStartCardCell.swift @@ -51,7 +51,6 @@ final class DashboardQuickStartCardCell: UICollectionViewCell, Reusable, BlogDas fallthrough case .newSite: - cardFrameView.icon = UIImage.gridicon(.listOrdered, size: Metrics.iconSize) configureOnEllipsisButtonTap(sourceRect: cardFrameView.ellipsisButton.frame) cardFrameView.showHeader() diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift index 9d4e54f731f3..8039eb4d5904 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift @@ -42,7 +42,6 @@ class DashboardStatsCardCell: UICollectionViewCell, Reusable { let frameView = BlogDashboardCardFrameView() frameView.title = Strings.statsTitle frameView.titleHint = Strings.statsTitleHint - frameView.icon = UIImage.gridicon(.statsAlt, size: Constants.iconSize) self.frameView = frameView let statsStackview = DashboardStatsStackView() diff --git a/WordPress/Classes/ViewRelated/Jetpack/Install/View/JetpackRemoteInstallCardView.swift b/WordPress/Classes/ViewRelated/Jetpack/Install/View/JetpackRemoteInstallCardView.swift index 7c177bc2c21e..6260de7578f3 100644 --- a/WordPress/Classes/ViewRelated/Jetpack/Install/View/JetpackRemoteInstallCardView.swift +++ b/WordPress/Classes/ViewRelated/Jetpack/Install/View/JetpackRemoteInstallCardView.swift @@ -72,7 +72,6 @@ class JetpackRemoteInstallCardView: UIView { private lazy var cardFrameView: BlogDashboardCardFrameView = { let frameView = BlogDashboardCardFrameView() frameView.translatesAutoresizingMaskIntoConstraints = false - frameView.icon = .none frameView.onEllipsisButtonTap = {} frameView.ellipsisButton.showsMenuAsPrimaryAction = true frameView.ellipsisButton.menu = contextMenu From 1d2c7fb47b248b6b3fb8d7504d7d0967f371018c Mon Sep 17 00:00:00 2001 From: kean Date: Wed, 22 Mar 2023 11:55:28 -0400 Subject: [PATCH 2/9] Fix localizable strings comments in Blaze card --- .../Blog/Blog Dashboard/Cards/Blaze/BlazeCardView.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/BlazeCardView.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/BlazeCardView.swift index 24c48cbae22e..84a6dd156149 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/BlazeCardView.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/BlazeCardView.swift @@ -126,7 +126,7 @@ extension BlazeCardView { comment: "Description for the Blaze dashboard card.") static let hideThis = NSLocalizedString("blaze.dashboard.card.menu.hide", value: "Hide this", - comment: "Title for a menu action in the context menu on the Jetpack install card.") + comment: "Title for a menu action in the context menu on the Blaze card.") } } From 60b88d889a025d5e7e69f9231b73db34e06e39d6 Mon Sep 17 00:00:00 2001 From: kean Date: Wed, 22 Mar 2023 11:57:08 -0400 Subject: [PATCH 3/9] Fix size of the context menu buttons on Dashboard (was too small) --- .../Cards/Posts/BlogDashboardCardFrameView.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift index 3eaa9e67a069..7e7809457398 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift @@ -177,7 +177,7 @@ class BlogDashboardCardFrameView: UIView { addSubview(buttonContainerStackView) NSLayoutConstraint.activate([ - buttonContainerStackView.topAnchor.constraint(equalTo: topAnchor, constant: Constants.buttonContainerStackViewPadding), + buttonContainerStackView.topAnchor.constraint(equalTo: topAnchor), buttonContainerStackView.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -Constants.buttonContainerStackViewPadding) ]) @@ -251,11 +251,11 @@ class BlogDashboardCardFrameView: UIView { private enum Constants { static let bottomPadding: CGFloat = 8 static let headerPaddingWithEllipsisButtonHidden = UIEdgeInsets(top: 12, left: 16, bottom: 8, right: 16) - static let headerPaddingWithEllipsisButtonShown = UIEdgeInsets(top: 12, left: 16, bottom: 8, right: 8) + static let headerPaddingWithEllipsisButtonShown = UIEdgeInsets(top: 4, left: 16, bottom: 0, right: 8) static let headerHorizontalSpacing: CGFloat = 5 static let iconSize = CGSize(width: 18, height: 18) static let cornerRadius: CGFloat = 10 - static let ellipsisButtonPadding = UIEdgeInsets(top: 0, left: 8, bottom: 0, right: 8) + static let ellipsisButtonPadding = UIEdgeInsets(top: 8, left: 8, bottom: 8, right: 8) static let buttonContainerStackViewPadding: CGFloat = 8 static let mainStackViewTrailingPadding: CGFloat = 32 } From c72dc43e90f2ae6ba3d6a82d55f91283c0a2d0e2 Mon Sep 17 00:00:00 2001 From: kean Date: Wed, 22 Mar 2023 15:07:54 -0400 Subject: [PATCH 4/9] Add context menu to Drafts and Scheduled posts --- .../Utility/Analytics/WPAnalyticsEvent.swift | 6 ++++ .../Cards/Blaze/DashboardBlazeCardCell.swift | 2 ++ .../Posts/DashboardPostsListCardCell.swift | 28 ++++++++++++------- .../Prompts/DashboardPromptsCardCell.swift | 6 +++- .../Helpers/BlogDashboardAnalytics.swift | 8 ++++++ .../Helpers/BlogDashboardHelpers.swift | 19 +++++++++++++ WordPress/WordPress.xcodeproj/project.pbxproj | 6 ++++ 7 files changed, 64 insertions(+), 11 deletions(-) create mode 100644 WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardHelpers.swift diff --git a/WordPress/Classes/Utility/Analytics/WPAnalyticsEvent.swift b/WordPress/Classes/Utility/Analytics/WPAnalyticsEvent.swift index 4a012e9dfdb2..5a094b521e41 100644 --- a/WordPress/Classes/Utility/Analytics/WPAnalyticsEvent.swift +++ b/WordPress/Classes/Utility/Analytics/WPAnalyticsEvent.swift @@ -329,6 +329,8 @@ import Foundation // My Site Dashboard case dashboardCardShown case dashboardCardItemTapped + case dashboardCardContextualMenuAccessed + case dashboardCardHideTapped case mySiteTabTapped case mySiteSiteMenuShown case mySiteDashboardShown @@ -1033,6 +1035,10 @@ import Foundation return "my_site_dashboard_card_shown" case .dashboardCardItemTapped: return "my_site_dashboard_card_item_tapped" + case .dashboardCardContextualMenuAccessed: + return "my_site_dashboard_contextual_menu_accessed" + case .dashboardCardHideTapped: + return "my_site_dashboard_card_hide_tapped" case .mySiteTabTapped: return "my_site_tab_tapped" case .mySiteSiteMenuShown: diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/DashboardBlazeCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/DashboardBlazeCardCell.swift index fac42f823c48..9f286cb096f6 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/DashboardBlazeCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Blaze/DashboardBlazeCardCell.swift @@ -19,10 +19,12 @@ class DashboardBlazeCardCell: DashboardCollectionViewCell { } let onEllipsisTap: () -> Void = { [weak self] in + BlogDashboardAnalytics.trackContextualMenuAccessed(for: .blaze) BlazeEventsTracker.trackContextualMenuAccessed(for: .dashboardCard) } let onHideThisTap: UIActionHandler = { [weak self] _ in + BlogDashboardAnalytics.trackHideTapped(for: .blaze) BlazeEventsTracker.trackHideThisTapped(for: .dashboardCard) BlazeHelper.hideBlazeCard(for: self?.blog) } diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift index 630b486887e7..182a9cabe06f 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift @@ -16,7 +16,7 @@ class DashboardPostsListCardCell: UICollectionViewCell, Reusable { // MARK: Views - private var frameView: BlogDashboardCardFrameView? + private let frameView = BlogDashboardCardFrameView() lazy var tableView: UITableView = { let tableView = PostCardTableView() @@ -68,13 +68,9 @@ class DashboardPostsListCardCell: UICollectionViewCell, Reusable { } private func addSubviews() { - let frameView = BlogDashboardCardFrameView() frameView.translatesAutoresizingMaskIntoConstraints = false - frameView.add(subview: tableView) - self.frameView = frameView - contentView.addSubview(frameView) contentView.pinSubviewToAllEdges(frameView, priority: Constants.constraintPriority) } @@ -103,23 +99,35 @@ extension DashboardPostsListCardCell { assertionFailure("Cell used with wrong card type") return } + addContextMenu(card: cardType, blog: blog) + viewModel = PostsCardViewModel(blog: blog, status: status, view: self) viewModel?.viewDidLoad() tableView.dataSource = viewModel?.diffableDataSource viewModel?.refresh() } + private func addContextMenu(card: DashboardCard, blog: Blog) { + frameView.onEllipsisButtonTap = { + BlogDashboardAnalytics.trackContextualMenuAccessed(for: card) + } + frameView.ellipsisButton.showsMenuAsPrimaryAction = true + frameView.ellipsisButton.menu = UIMenu(title: "", options: .displayInline, children: [ + BlogDashboardHelpers.makeHideCardAction(for: card, siteID: blog.dotComID?.intValue ?? 0) + ]) + } + private func configureDraftsList(blog: Blog) { - frameView?.title = Strings.draftsTitle - frameView?.titleHint = Strings.draftsTitleHint - frameView?.onHeaderTap = { [weak self] in + frameView.title = Strings.draftsTitle + frameView.titleHint = Strings.draftsTitleHint + frameView.onHeaderTap = { [weak self] in self?.presentPostList(with: .draft) } } private func configureScheduledList(blog: Blog) { - frameView?.title = Strings.scheduledTitle - frameView?.onHeaderTap = { [weak self] in + frameView.title = Strings.scheduledTitle + frameView.onHeaderTap = { [weak self] in self?.presentPostList(with: .scheduled) } } diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift index 1c67645a8858..c574841428c7 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Prompts/DashboardPromptsCardCell.swift @@ -15,7 +15,9 @@ class DashboardPromptsCardCell: UICollectionViewCell, Reusable { // NOTE: Remove the logic when support for iOS 14 is dropped if #available (iOS 15.0, *) { // assign an empty closure so the button appears. - frameView.onEllipsisButtonTap = {} + frameView.onEllipsisButtonTap = { + BlogDashboardAnalytics.trackContextualMenuAccessed(for: .prompts) + } frameView.ellipsisButton.showsMenuAsPrimaryAction = true frameView.ellipsisButton.menu = contextMenu } else { @@ -23,6 +25,7 @@ class DashboardPromptsCardCell: UICollectionViewCell, Reusable { // iOS 13 doesn't support showing UIMenu programmatically. // iOS 14 doesn't support `UIDeferredMenuElement.uncached`. frameView.onEllipsisButtonTap = { [weak self] in + BlogDashboardAnalytics.trackContextualMenuAccessed(for: .prompts) self?.showMenuSheet() } } @@ -500,6 +503,7 @@ private extension DashboardPromptsCardCell { return } WPAnalytics.track(.promptsDashboardCardMenuRemove) + BlogDashboardAnalytics.trackHideTapped(for: .prompts) let service = BlogDashboardPersonalizationService(siteID: siteID) service.setEnabled(false, for: .prompts) let notice = Notice(title: Strings.promptRemovedTitle, message: Strings.promptRemovedSubtitle, feedbackType: .success, actionTitle: Strings.undoSkipTitle) { _ in diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardAnalytics.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardAnalytics.swift index 53078b8fa6da..b3e90f6d86bc 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardAnalytics.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardAnalytics.swift @@ -30,4 +30,12 @@ class BlogDashboardAnalytics { } } } + + static func trackContextualMenuAccessed(for card: DashboardCard) { + WPAnalytics.track(.dashboardCardContextualMenuAccessed, properties: ["card": card.rawValue]) + } + + static func trackHideTapped(for card: DashboardCard) { + WPAnalytics.track(.dashboardCardHideTapped, properties: ["card": card.rawValue]) + } } diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardHelpers.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardHelpers.swift new file mode 100644 index 000000000000..beb295506968 --- /dev/null +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Helpers/BlogDashboardHelpers.swift @@ -0,0 +1,19 @@ +import Foundation + +struct BlogDashboardHelpers { + static func makeHideCardAction(for card: DashboardCard, siteID: Int) -> UIAction { + UIAction( + title: Strings.hideThis, + image: UIImage(systemName: "minus.circle"), + attributes: [.destructive], + handler: { _ in + BlogDashboardAnalytics.trackHideTapped(for: card) + BlogDashboardPersonalizationService(siteID: siteID) + .setEnabled(false, for: card) + }) + } + + private enum Strings { + static let hideThis = NSLocalizedString("blogDashboard.contextMenu.hideThis", value: "Hide this", comment: "Title for the context menu action that hides the dashboard card.") + } +} diff --git a/WordPress/WordPress.xcodeproj/project.pbxproj b/WordPress/WordPress.xcodeproj/project.pbxproj index a3f6cc684c75..26e07e8311d5 100644 --- a/WordPress/WordPress.xcodeproj/project.pbxproj +++ b/WordPress/WordPress.xcodeproj/project.pbxproj @@ -304,6 +304,8 @@ 0A9610FA28B2E56300076EBA /* UserSuggestion+Comparable.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0A9610F828B2E56300076EBA /* UserSuggestion+Comparable.swift */; }; 0A9687BC28B40771009DCD2F /* FullScreenCommentReplyViewModelMock.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0A9687BB28B40771009DCD2F /* FullScreenCommentReplyViewModelMock.swift */; }; 0C35FFF429CBA6DA00D224EB /* BlogDashboardPersonalizationViewModelTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0C35FFF329CBA6DA00D224EB /* BlogDashboardPersonalizationViewModelTests.swift */; }; + 0C35FFF129CB81F700D224EB /* BlogDashboardHelpers.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0C35FFF029CB81F700D224EB /* BlogDashboardHelpers.swift */; }; + 0C35FFF229CB81F700D224EB /* BlogDashboardHelpers.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0C35FFF029CB81F700D224EB /* BlogDashboardHelpers.swift */; }; 0CB4056B29C78F06008EED0A /* BlogDashboardPersonalizationService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0CB4056A29C78F06008EED0A /* BlogDashboardPersonalizationService.swift */; }; 0CB4056C29C78F06008EED0A /* BlogDashboardPersonalizationService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0CB4056A29C78F06008EED0A /* BlogDashboardPersonalizationService.swift */; }; 0CB4056E29C7BA63008EED0A /* BlogDashboardPersonalizationServiceTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0CB4056D29C7BA63008EED0A /* BlogDashboardPersonalizationServiceTests.swift */; }; @@ -5919,6 +5921,7 @@ 0A9610F828B2E56300076EBA /* UserSuggestion+Comparable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UserSuggestion+Comparable.swift"; sourceTree = ""; }; 0A9687BB28B40771009DCD2F /* FullScreenCommentReplyViewModelMock.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FullScreenCommentReplyViewModelMock.swift; sourceTree = ""; }; 0C35FFF329CBA6DA00D224EB /* BlogDashboardPersonalizationViewModelTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlogDashboardPersonalizationViewModelTests.swift; sourceTree = ""; }; + 0C35FFF029CB81F700D224EB /* BlogDashboardHelpers.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlogDashboardHelpers.swift; sourceTree = ""; }; 0CB4056A29C78F06008EED0A /* BlogDashboardPersonalizationService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlogDashboardPersonalizationService.swift; sourceTree = ""; }; 0CB4056D29C7BA63008EED0A /* BlogDashboardPersonalizationServiceTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlogDashboardPersonalizationServiceTests.swift; sourceTree = ""; }; 0CB4057029C8DCF4008EED0A /* BlogDashboardPersonalizationViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BlogDashboardPersonalizationViewModel.swift; sourceTree = ""; }; @@ -13539,6 +13542,7 @@ isa = PBXGroup; children = ( 8B15D27328009EBF0076628A /* BlogDashboardAnalytics.swift */, + 0C35FFF029CB81F700D224EB /* BlogDashboardHelpers.swift */, 8BC81D6427CFC0DA0057F790 /* BlogDashboardState.swift */, 80EF9285280D272E0064A971 /* DashboardPostsSyncManager.swift */, ); @@ -21306,6 +21310,7 @@ B538F3891EF46EC8001003D5 /* UnknownEditorViewController.swift in Sources */, 937D9A1119F838C2007B9D5F /* AccountToAccount22to23.swift in Sources */, 400A2C772217A8A0000A8A59 /* VisitsSummaryStatsRecordValue+CoreDataClass.swift in Sources */, + 0C35FFF129CB81F700D224EB /* BlogDashboardHelpers.swift in Sources */, D8212CB320AA6861008E8AE8 /* ReaderFollowAction.swift in Sources */, F5D399302541F25B0058D0AB /* SheetActions.swift in Sources */, 93F7214F271831820021A09F /* SiteStatsPinnedItemStore.swift in Sources */, @@ -24935,6 +24940,7 @@ FABB256B2602FC2C00C8785C /* InteractivePostView.swift in Sources */, FABB256C2602FC2C00C8785C /* LinearGradientView.swift in Sources */, FABB256D2602FC2C00C8785C /* WizardStep.swift in Sources */, + 0C35FFF229CB81F700D224EB /* BlogDashboardHelpers.swift in Sources */, C395FB242821FE4B00AE7C11 /* SiteDesignSection.swift in Sources */, 837B49D8283C2AE80061A657 /* BloggingPromptSettings+CoreDataClass.swift in Sources */, FABB256E2602FC2C00C8785C /* PostPostViewController.swift in Sources */, From 6cd31fd2cc5d087ce42ea00a1825f179acc1aff1 Mon Sep 17 00:00:00 2001 From: kean Date: Wed, 22 Mar 2023 15:20:00 -0400 Subject: [PATCH 5/9] Add Hide This action to Todays Stats card on Dashboard --- .../Posts/DashboardPostsListCardCell.swift | 2 ++ .../Cards/Stats/DashboardStatsCardCell.swift | 17 ++++++++++++----- 2 files changed, 14 insertions(+), 5 deletions(-) diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift index 182a9cabe06f..0f8127af752d 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/DashboardPostsListCardCell.swift @@ -108,6 +108,8 @@ extension DashboardPostsListCardCell { } private func addContextMenu(card: DashboardCard, blog: Blog) { + guard FeatureFlag.personalizeHomeTab.enabled else { return } + frameView.onEllipsisButtonTap = { BlogDashboardAnalytics.trackContextualMenuAccessed(for: card) } diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift index 8039eb4d5904..0b371d238a24 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Stats/DashboardStatsCardCell.swift @@ -6,7 +6,7 @@ class DashboardStatsCardCell: UICollectionViewCell, Reusable { // MARK: Private Variables private var viewModel: DashboardStatsViewModel? - private var frameView: BlogDashboardCardFrameView? + private let frameView = BlogDashboardCardFrameView() private var nudgeView: DashboardStatsNudgeView? private var statsStackView: DashboardStatsStackView? @@ -39,10 +39,8 @@ class DashboardStatsCardCell: UICollectionViewCell, Reusable { } private func addSubviews() { - let frameView = BlogDashboardCardFrameView() frameView.title = Strings.statsTitle frameView.titleHint = Strings.statsTitleHint - self.frameView = frameView let statsStackview = DashboardStatsStackView() frameView.add(subview: statsStackview) @@ -73,11 +71,20 @@ extension DashboardStatsCardCell: BlogDashboardCardConfigurable { } private func configureCard(for blog: Blog, in viewController: UIViewController) { - - frameView?.onViewTap = { [weak self] in + frameView.onViewTap = { [weak self] in self?.showStats(for: blog, from: viewController) } + if FeatureFlag.personalizeHomeTab.enabled { + frameView.onEllipsisButtonTap = { + BlogDashboardAnalytics.trackContextualMenuAccessed(for: .todaysStats) + } + frameView.ellipsisButton.showsMenuAsPrimaryAction = true + frameView.ellipsisButton.menu = UIMenu(title: "", options: .displayInline, children: [ + BlogDashboardHelpers.makeHideCardAction(for: .todaysStats, siteID: blog.dotComID?.intValue ?? 0) + ]) + } + statsStackView?.views = viewModel?.todaysViews statsStackView?.visitors = viewModel?.todaysVisitors statsStackView?.likes = viewModel?.todaysLikes From 9d2cd357e32f18eaa52b693a054d4aded460d4a5 Mon Sep 17 00:00:00 2001 From: kean Date: Wed, 22 Mar 2023 17:58:38 -0400 Subject: [PATCH 6/9] Update release notes --- RELEASE-NOTES.txt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/RELEASE-NOTES.txt b/RELEASE-NOTES.txt index 06d6073cb3ab..32922bb5e918 100644 --- a/RELEASE-NOTES.txt +++ b/RELEASE-NOTES.txt @@ -2,8 +2,9 @@ ----- * [**] [internal] Refactor updating account related Core Data operations, which ususally happens during log in and out of the app. [#20394] * [***] [internal] Refactor uploading photos (from the device photo, the Free Photo library, and other sources) to the WordPress Media Library. Affected areas are where you can choose a photo and upload, including the "Media" screen, adding images to a post, updating site icon, etc. [#20322] -* [**] [WordPress-only] Warns user about sites with only individual plugins not supporting core app features and offers the option to switch to the Jetpack app. [#20408] * [**] Add a "Personalize Home Tab" button to the bottom of the Home tab that allows changing cards visibility. [#20369] +* [**] Add context menus to "Drafts", "Scheduled Posts", and "Today's Stats" cards with an option to hide these cards. [#20384] +* [**] [WordPress-only] Warns user about sites with only individual plugins not supporting core app features and offers the option to switch to the Jetpack app. [#20408] 22.0 ----- From 07bbcfd785f761eea007ecdf8191fb03d02c6484 Mon Sep 17 00:00:00 2001 From: Alex Grebenyuk Date: Fri, 31 Mar 2023 08:14:36 -0400 Subject: [PATCH 7/9] Revert "Fix size of the context menu buttons on Dashboard (was too small)" This reverts commit 60b88d889a025d5e7e69f9231b73db34e06e39d6. --- .../Cards/Posts/BlogDashboardCardFrameView.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift index 7e7809457398..3eaa9e67a069 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift @@ -177,7 +177,7 @@ class BlogDashboardCardFrameView: UIView { addSubview(buttonContainerStackView) NSLayoutConstraint.activate([ - buttonContainerStackView.topAnchor.constraint(equalTo: topAnchor), + buttonContainerStackView.topAnchor.constraint(equalTo: topAnchor, constant: Constants.buttonContainerStackViewPadding), buttonContainerStackView.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -Constants.buttonContainerStackViewPadding) ]) @@ -251,11 +251,11 @@ class BlogDashboardCardFrameView: UIView { private enum Constants { static let bottomPadding: CGFloat = 8 static let headerPaddingWithEllipsisButtonHidden = UIEdgeInsets(top: 12, left: 16, bottom: 8, right: 16) - static let headerPaddingWithEllipsisButtonShown = UIEdgeInsets(top: 4, left: 16, bottom: 0, right: 8) + static let headerPaddingWithEllipsisButtonShown = UIEdgeInsets(top: 12, left: 16, bottom: 8, right: 8) static let headerHorizontalSpacing: CGFloat = 5 static let iconSize = CGSize(width: 18, height: 18) static let cornerRadius: CGFloat = 10 - static let ellipsisButtonPadding = UIEdgeInsets(top: 8, left: 8, bottom: 8, right: 8) + static let ellipsisButtonPadding = UIEdgeInsets(top: 0, left: 8, bottom: 0, right: 8) static let buttonContainerStackViewPadding: CGFloat = 8 static let mainStackViewTrailingPadding: CGFloat = 32 } From 0389d8c4887094e9439dd95b7180528a9b019b86 Mon Sep 17 00:00:00 2001 From: Alex Grebenyuk Date: Fri, 31 Mar 2023 10:14:46 -0400 Subject: [PATCH 8/9] Remove personalization from release notes for now --- RELEASE-NOTES.txt | 2 -- 1 file changed, 2 deletions(-) diff --git a/RELEASE-NOTES.txt b/RELEASE-NOTES.txt index 32922bb5e918..9eef518266f9 100644 --- a/RELEASE-NOTES.txt +++ b/RELEASE-NOTES.txt @@ -2,8 +2,6 @@ ----- * [**] [internal] Refactor updating account related Core Data operations, which ususally happens during log in and out of the app. [#20394] * [***] [internal] Refactor uploading photos (from the device photo, the Free Photo library, and other sources) to the WordPress Media Library. Affected areas are where you can choose a photo and upload, including the "Media" screen, adding images to a post, updating site icon, etc. [#20322] -* [**] Add a "Personalize Home Tab" button to the bottom of the Home tab that allows changing cards visibility. [#20369] -* [**] Add context menus to "Drafts", "Scheduled Posts", and "Today's Stats" cards with an option to hide these cards. [#20384] * [**] [WordPress-only] Warns user about sites with only individual plugins not supporting core app features and offers the option to switch to the Jetpack app. [#20408] 22.0 From 8c96c77f869ffa55305c65e9268a2a3667a11f67 Mon Sep 17 00:00:00 2001 From: Alex Grebenyuk Date: Fri, 31 Mar 2023 12:43:58 -0400 Subject: [PATCH 9/9] Update context menu icon on dashboard cards --- .../Cards/Posts/BlogDashboardCardFrameView.swift | 12 ++---------- .../more-horizontal-mobile.imageset/Contents.json | 15 +++++++++++++++ .../more-horizontal-mobile.svg | 3 +++ 3 files changed, 20 insertions(+), 10 deletions(-) create mode 100644 WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/Contents.json create mode 100644 WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/more-horizontal-mobile.svg diff --git a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift index 1ffa1d05f362..542c39799b8c 100644 --- a/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift +++ b/WordPress/Classes/ViewRelated/Blog/Blog Dashboard/Cards/Posts/BlogDashboardCardFrameView.swift @@ -40,7 +40,8 @@ class BlogDashboardCardFrameView: UIView { /// Displayed only when an associated action is set private(set) lazy var ellipsisButton: UIButton = { let button = UIButton(type: .custom) - button.setImage(UIImage.gridicon(.ellipsis).imageWithTintColor(.listIcon), for: .normal) + button.setImage(UIImage(named: "more-horizontal-mobile"), for: .normal) + button.tintColor = UIColor.listIcon button.contentEdgeInsets = Constants.ellipsisButtonPadding button.isAccessibilityElement = true button.accessibilityLabel = Strings.ellipsisButtonAccessibilityLabel @@ -129,11 +130,6 @@ class BlogDashboardCardFrameView: UIView { } } - override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) { - super.traitCollectionDidChange(previousTraitCollection) - updateColors() - } - /// Add a subview inside the card frame func add(subview: UIView) { mainStackView.addArrangedSubview(subview) @@ -194,10 +190,6 @@ class BlogDashboardCardFrameView: UIView { buttonContainerStackView.removeFromSuperview() } - private func updateColors() { - ellipsisButton.setImage(UIImage.gridicon(.ellipsis).imageWithTintColor(.listIcon), for: .normal) - } - private func updateEllipsisButtonState() { ellipsisButton.isHidden = onEllipsisButtonTap == nil let headerPadding = ellipsisButton.isHidden ? diff --git a/WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/Contents.json b/WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/Contents.json new file mode 100644 index 000000000000..08a009949956 --- /dev/null +++ b/WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/Contents.json @@ -0,0 +1,15 @@ +{ + "images" : [ + { + "filename" : "more-horizontal-mobile.svg", + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + }, + "properties" : { + "template-rendering-intent" : "template" + } +} diff --git a/WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/more-horizontal-mobile.svg b/WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/more-horizontal-mobile.svg new file mode 100644 index 000000000000..f8af22821782 --- /dev/null +++ b/WordPress/Resources/AppImages.xcassets/more-horizontal-mobile.imageset/more-horizontal-mobile.svg @@ -0,0 +1,3 @@ + + +