1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
|
//
// OutOfTimeContentView.swift
// MullvadVPN
//
// Created by Andreas Lif on 2022-07-26.
// Copyright © 2025 Mullvad VPN AB. All rights reserved.
//
import Foundation
import UIKit
class OutOfTimeContentView: UIView {
let statusActivityView: StatusActivityView = {
let statusActivityView = StatusActivityView(state: .failure)
statusActivityView.translatesAutoresizingMaskIntoConstraints = false
return statusActivityView
}()
private lazy var titleLabel: UILabel = {
let label = UILabel()
label.text = NSLocalizedString("Out of time", comment: "")
label.font = .mullvadLarge
label.adjustsFontForContentSizeCategory = true
label.textColor = .white
label.numberOfLines = 0
return label
}()
private lazy var bodyLabel: UILabel = {
let label = UILabel()
label.textColor = .white
label.numberOfLines = 0
label.adjustsFontForContentSizeCategory = true
return label
}()
lazy var disconnectButton: AppButton = {
let button = AppButton(style: .danger)
button.translatesAutoresizingMaskIntoConstraints = false
button.isHidden = true
let localizedString = NSLocalizedString("Disconnect", comment: "")
button.setTitle(localizedString, for: .normal)
return button
}()
lazy var purchaseButton: InAppPurchaseButton = {
let button = InAppPurchaseButton()
button.translatesAutoresizingMaskIntoConstraints = false
let localizedString = NSLocalizedString("Add time", comment: "")
button.setTitle(localizedString, for: .normal)
return button
}()
lazy var restoreButton: AppButton = {
let button = AppButton(style: .default)
button.translatesAutoresizingMaskIntoConstraints = false
button.setTitle(NSLocalizedString("Restore purchases", comment: ""), for: .normal)
return button
}()
private let scrollView = UIScrollView()
private lazy var topStackView: UIStackView = {
let stackView = UIStackView(arrangedSubviews: [statusActivityView, titleLabel, bodyLabel])
stackView.translatesAutoresizingMaskIntoConstraints = false
stackView.axis = .vertical
stackView.spacing = UIMetrics.TableView.sectionSpacing
return stackView
}()
private lazy var bottomStackView: UIStackView = {
let stackView = UIStackView(
arrangedSubviews: [disconnectButton, purchaseButton, restoreButton]
)
stackView.translatesAutoresizingMaskIntoConstraints = false
stackView.axis = .vertical
stackView.spacing = UIMetrics.interButtonSpacing
stackView.backgroundColor = .secondaryColor
return stackView
}()
override init(frame: CGRect) {
super.init(frame: frame)
setAccessibilityIdentifier(.outOfTimeView)
translatesAutoresizingMaskIntoConstraints = false
backgroundColor = .secondaryColor
setUpSubviews()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
func enableDisconnectButton(_ enabled: Bool, animated: Bool) {
disconnectButton.isEnabled = enabled
UIView.animate(withDuration: animated ? 0.25 : 0) {
self.disconnectButton.isHidden = !enabled
}
}
func enablePurchaseButton(_ enabled: Bool) {
purchaseButton.isEnabled = enabled
}
// MARK: - Private Functions
func setUpSubviews() {
scrollView.addConstrainedSubviews([topStackView]) {
topStackView.pinEdgesToSuperviewMargins(
PinnableEdges([
.leading(0),
.trailing(0),
]))
topStackView.pinEdgesToSuperview(
PinnableEdges([
.top(0),
.bottom(0),
]))
}
addConstrainedSubviews([scrollView, bottomStackView]) {
scrollView.pinEdgesToSuperviewMargins(
PinnableEdges([
.top(UIMetrics.contentLayoutMargins.top),
.leading(0),
.trailing(0),
]))
bottomStackView.pinEdgesToSuperviewMargins(
PinnableEdges([
.leading(UIMetrics.padding8),
.trailing(UIMetrics.padding8),
.bottom(UIMetrics.contentLayoutMargins.bottom),
]))
bottomStackView.topAnchor.constraint(
equalTo: scrollView.bottomAnchor,
constant: UIMetrics.contentLayoutMargins.top
)
}
}
func setBodyLabelText(_ text: String) {
bodyLabel.attributedText = NSAttributedString(
markdownString: text,
options: MarkdownStylingOptions(font: .mullvadSmall)
)
}
}
|