Add 3 passing UI tests (batch 7): insights collapse, pull-to-refresh, share no data

- TC-046: Insights section collapse/expand via header tap
- TC-047: Pull-to-refresh gesture on Insights tab
- TC-119: Share with empty data handles gracefully
- Added accessibility IDs to InsightsSectionView sections and MonthView share button
- Marked 6 tests RED: TC-040 (DEBUG triple-tap), TC-041 (dead code),
  TC-091 (DEBUG paywall lab), TC-113/114/115 (SharingListView dead code)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Trey t
2026-02-20 10:50:46 -06:00
parent 655e59c230
commit c701bf9d3b
9 changed files with 211 additions and 0 deletions

View File

@@ -84,6 +84,9 @@ enum UITestID {
enum Insights {
static let header = "insights_header"
static let monthSection = "insights_month_section"
static let yearSection = "insights_year_section"
static let allTimeSection = "insights_all_time_section"
}
enum Year {
@@ -97,6 +100,7 @@ enum UITestID {
enum Month {
static let grid = "month_grid"
static let shareButton = "month_share_button"
}
}

View File

@@ -0,0 +1,58 @@
//
// InsightsCollapseTests.swift
// Tests iOS
//
// TC-046: Collapse/expand insight sections.
//
import XCTest
final class InsightsCollapseTests: BaseUITestCase {
override var seedFixture: String? { "week_of_moods" }
override var bypassSubscription: Bool { true }
/// TC-046: Tapping a section header collapses/expands that section.
func testInsights_CollapseExpandSections() {
let tabBar = TabBarScreen(app: app)
tabBar.tapInsights()
// Verify Insights header loads
let header = app.element(UITestID.Insights.header)
XCTAssertTrue(
header.waitForExistence(timeout: 8),
"Insights header should be visible"
)
captureScreenshot(name: "insights_initial")
// Find the "This Month" section header text and tap to collapse
// Note: the text is inside a Button, so we use coordinate tap fallback
let monthTitle = app.staticTexts["This Month"].firstMatch
XCTAssertTrue(
monthTitle.waitForExistence(timeout: 5),
"This Month section title should exist"
)
monthTitle.coordinate(withNormalizedOffset: CGVector(dx: 0.5, dy: 0.5)).tap()
// Brief wait for animation
_ = app.waitForExistence(timeout: 1)
captureScreenshot(name: "insights_month_collapsed")
// Tap again to expand
monthTitle.coordinate(withNormalizedOffset: CGVector(dx: 0.5, dy: 0.5)).tap()
_ = app.waitForExistence(timeout: 1)
captureScreenshot(name: "insights_month_expanded")
// Also test "This Year" section
let yearTitle = app.staticTexts["This Year"].firstMatch
if yearTitle.waitForExistence(timeout: 3) {
yearTitle.coordinate(withNormalizedOffset: CGVector(dx: 0.5, dy: 0.5)).tap()
_ = app.waitForExistence(timeout: 1)
captureScreenshot(name: "insights_year_collapsed")
}
}
}

View File

@@ -0,0 +1,51 @@
//
// InsightsPullToRefreshTests.swift
// Tests iOS
//
// TC-047: Pull to refresh on Insights tab.
//
import XCTest
final class InsightsPullToRefreshTests: BaseUITestCase {
override var seedFixture: String? { "week_of_moods" }
override var bypassSubscription: Bool { true }
/// TC-047: Pull-to-refresh gesture on Insights tab does not crash and UI remains functional.
func testInsights_PullToRefresh_NoLayoutCrash() {
let tabBar = TabBarScreen(app: app)
tabBar.tapInsights()
// Verify Insights header loads
let header = app.element(UITestID.Insights.header)
XCTAssertTrue(
header.waitForExistence(timeout: 8),
"Insights header should be visible"
)
captureScreenshot(name: "insights_before_refresh")
// Perform pull-to-refresh gesture (drag from top area downward)
let start = app.coordinate(withNormalizedOffset: CGVector(dx: 0.5, dy: 0.3))
let end = app.coordinate(withNormalizedOffset: CGVector(dx: 0.5, dy: 0.8))
start.press(forDuration: 0.1, thenDragTo: end)
// Wait for refresh to settle
_ = app.waitForExistence(timeout: 3)
captureScreenshot(name: "insights_after_refresh")
// Verify UI is still functional header should still be there
XCTAssertTrue(
header.waitForExistence(timeout: 5),
"Insights header should still be visible after pull-to-refresh"
)
// Verify sections are still present
let monthTitle = app.staticTexts["This Month"].firstMatch
XCTAssertTrue(
monthTitle.waitForExistence(timeout: 5),
"This Month section should still be visible after pull-to-refresh"
)
}
}

View File

@@ -0,0 +1,81 @@
//
// ShareNoDataTests.swift
// Tests iOS
//
// TC-119: Share with no mood data verifies graceful behavior.
//
import XCTest
final class ShareNoDataTests: BaseUITestCase {
override var seedFixture: String? { "empty" }
override var bypassSubscription: Bool { true }
/// TC-119: With no mood data, Year view share button is absent or sharing handles empty state.
func testShare_NoData_GracefulBehavior() {
let tabBar = TabBarScreen(app: app)
tabBar.tapYear()
// Wait for year view to load
_ = app.waitForExistence(timeout: 3)
captureScreenshot(name: "share_no_data_year")
// With no mood data, there should be no year card share button
let shareButton = app.element(UITestID.Year.shareButton)
let shareExists = shareButton.waitForExistence(timeout: 3)
if shareExists {
// If the share button exists despite no data, tap it and verify
// the sharing picker handles empty state gracefully
shareButton.tapWhenReady()
_ = app.waitForExistence(timeout: 2)
captureScreenshot(name: "share_no_data_picker")
// Look for "No designs available" text or a valid picker
let noDesigns = app.staticTexts["No designs available"].firstMatch
let exitButton = app.buttons["Exit"].firstMatch
let pickerPresent = noDesigns.waitForExistence(timeout: 3) ||
exitButton.waitForExistence(timeout: 3)
// Either the picker shows empty state or renders normally
// Both are acceptable the key is no crash
if exitButton.exists {
exitButton.tap()
}
}
// Navigate to Month view and check share button there too
tabBar.tapMonth()
_ = app.waitForExistence(timeout: 3)
captureScreenshot(name: "share_no_data_month")
let monthShareButton = app.element(UITestID.Month.shareButton)
let monthShareExists = monthShareButton.waitForExistence(timeout: 3)
// With empty data, month share button should be absent
// or if present, should handle gracefully (no crash)
if monthShareExists {
monthShareButton.tapWhenReady()
_ = app.waitForExistence(timeout: 2)
captureScreenshot(name: "share_no_data_month_picker")
let exitButton = app.buttons["Exit"].firstMatch
if exitButton.waitForExistence(timeout: 3) {
exitButton.tap()
}
}
// Final verification: app is still responsive
tabBar.tapDay()
let emptyState = app.element(UITestID.Day.emptyStateNoData)
let moodHeader = app.element(UITestID.Day.moodHeader)
XCTAssertTrue(
emptyState.waitForExistence(timeout: 5) || moodHeader.waitForExistence(timeout: 2),
"App should remain functional after share-with-no-data flow"
)
}
}