Files
letro-ios/ElementX/Sources/Screens/AppLock/AppLockScreen/View/AppLockScreen.swift
Mauro 56eec826df Fix A11y tests (#5104)
* replace NavigationStack with ElementNavigationStack to allow the content to be rendered without a NavigationStack in a11y tests

* fix a11y tests

* update xcodeproject

* swiftformat fix

* use iOS 26.1 for CI

* use a wrapper to solve the issue for a11y tests

* ElementNavigationStack only uses the trick in DEBUG mode, and added a swiftlint rule to prevent the usage of NavigationStack
2026-02-13 16:45:58 +01:00

124 lines
4.6 KiB
Swift

//
// Copyright 2025 Element Creations Ltd.
// Copyright 2022-2025 New Vector Ltd.
//
// SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial.
// Please see LICENSE files in the repository root for full details.
//
import Compound
import SwiftUI
struct AppLockScreen: View {
@ObservedObject var context: AppLockScreenViewModel.Context
/// Used to animate the PIN input field on failure.
@State private var pinInputFieldOffset = 0.0
/// A focus state to highlight a failed PIN entry in VoiceOver.
@AccessibilityFocusState private var accessibilitySubtitleFocus: Bool
var subtitleColor: Color {
context.viewState.isSubtitleWarning ? .compound.textCriticalPrimary : .compound.textPrimary
}
var body: some View {
FullscreenDialog {
VStack(spacing: 32) {
header
pinInputField
.padding(.bottom, 16)
.offset(x: pinInputFieldOffset)
.onChange(of: context.viewState.numberOfPINAttempts) { _, newValue in
guard newValue > 0 else { return } // Reset without animation in Previews.
accessibilitySubtitleFocus = true
Task { await animatePINFailure() }
}
.accessibilityLabel(L10n.a11yPinField)
.accessibilityValue(L10n.a11yDigitsEntered(context.viewState.numberOfDigitsEntered))
AppLockScreenPINKeypad(pinCode: $context.pinCode)
}
} bottomContent: {
Button(L10n.screenAppLockForgotPin) {
context.send(viewAction: .forgotPIN)
}
.font(.compound.bodyMDSemibold)
}
.background()
.backgroundStyle(.compound.bgCanvasDefault)
.disabled(context.viewState.forcedLogoutIndicator != nil)
.overlay {
context.viewState.forcedLogoutIndicator.map(UserIndicatorModalView.init)
.animation(.elementDefault, value: context.viewState.forcedLogoutIndicator)
}
.alert(item: $context.alertInfo)
}
var header: some View {
VStack(spacing: 8) {
CompoundIcon(\.lockSolid, size: .medium, relativeTo: .compound.headingMDBold)
.padding(.bottom, 8)
.accessibilityHidden(true)
Text(L10n.commonEnterYourPin)
.font(.compound.headingMDBold)
.foregroundColor(.compound.textPrimary)
.multilineTextAlignment(.center)
Text(context.viewState.subtitle)
.font(.compound.bodyMD)
.foregroundColor(subtitleColor)
.multilineTextAlignment(.center)
.accessibilityFocused($accessibilitySubtitleFocus)
}
}
/// The row of dots showing how many digits have been entered.
var pinInputField: some View {
HStack(spacing: 24) {
// The size of each dot within the PIN input field.
let pinDotSize: CGFloat = 14
Circle()
.fill(context.viewState.numberOfDigitsEntered > 0 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.scaledFrame(size: pinDotSize)
Circle()
.fill(context.viewState.numberOfDigitsEntered > 1 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.scaledFrame(size: pinDotSize)
Circle()
.fill(context.viewState.numberOfDigitsEntered > 2 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.scaledFrame(size: pinDotSize)
Circle()
.fill(context.viewState.numberOfDigitsEntered > 3 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.scaledFrame(size: pinDotSize)
}
}
func animatePINFailure() async {
withAnimation(.spring(response: 0, dampingFraction: 0.7, blendDuration: 0.0)) {
pinInputFieldOffset = 15
}
try? await Task.sleep(for: .milliseconds(50))
withAnimation(.spring(response: 0.1, dampingFraction: 0.3, blendDuration: 0.1)) {
pinInputFieldOffset = 0
}
try? await Task.sleep(for: .milliseconds(100))
context.send(viewAction: .clearPINCode)
}
}
// MARK: - Previews
struct AppLockScreen_Previews: PreviewProvider, TestablePreview {
static let viewModel = AppLockScreenViewModel(appLockService: AppLockServiceMock.mock())
static var previews: some View {
ElementNavigationStack {
AppLockScreen(context: viewModel.context)
}
}
}