Skip to content

Commit

Permalink
Update cell layout and behavior for obfuscation methods
Browse files Browse the repository at this point in the history
  • Loading branch information
rablador committed Oct 21, 2024
1 parent 36a6113 commit 2591de0
Show file tree
Hide file tree
Showing 12 changed files with 212 additions and 54 deletions.
8 changes: 8 additions & 0 deletions ios/MullvadVPN.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -466,6 +466,8 @@
7A27E3C92CAE85710088BCFF /* SettingsInfoButtonItem.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A27E3C82CAE85660088BCFF /* SettingsInfoButtonItem.swift */; };
7A27E3CB2CAE861D0088BCFF /* SettingsViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A27E3CA2CAE86170088BCFF /* SettingsViewModel.swift */; };
7A27E3CD2CB814EF0088BCFF /* DAITAInfoView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A27E3CC2CB814EA0088BCFF /* DAITAInfoView.swift */; };
7A27E3CF2CBD4A8C0088BCFF /* SelectableSettingsDetailsCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A27E3CE2CBD4A830088BCFF /* SelectableSettingsDetailsCell.swift */; };
7A27E3D12CC299F90088BCFF /* VPNSettingsDetailsButtonItem.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A27E3D02CC299E60088BCFF /* VPNSettingsDetailsButtonItem.swift */; };
7A28826A2BA8336600FD9F20 /* VPNSettingsCoordinator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A2882692BA8336600FD9F20 /* VPNSettingsCoordinator.swift */; };
7A2960F62A963F7500389B82 /* AlertCoordinator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A2960F52A963F7500389B82 /* AlertCoordinator.swift */; };
7A2960FD2A964BB700389B82 /* AlertPresentation.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A2960FC2A964BB700389B82 /* AlertPresentation.swift */; };
Expand Down Expand Up @@ -1803,6 +1805,8 @@
7A27E3C82CAE85660088BCFF /* SettingsInfoButtonItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SettingsInfoButtonItem.swift; sourceTree = "<group>"; };
7A27E3CA2CAE86170088BCFF /* SettingsViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SettingsViewModel.swift; sourceTree = "<group>"; };
7A27E3CC2CB814EA0088BCFF /* DAITAInfoView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DAITAInfoView.swift; sourceTree = "<group>"; };
7A27E3CE2CBD4A830088BCFF /* SelectableSettingsDetailsCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SelectableSettingsDetailsCell.swift; sourceTree = "<group>"; };
7A27E3D02CC299E60088BCFF /* VPNSettingsDetailsButtonItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VPNSettingsDetailsButtonItem.swift; sourceTree = "<group>"; };
7A2882692BA8336600FD9F20 /* VPNSettingsCoordinator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VPNSettingsCoordinator.swift; sourceTree = "<group>"; };
7A2960F52A963F7500389B82 /* AlertCoordinator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AlertCoordinator.swift; sourceTree = "<group>"; };
7A2960FC2A964BB700389B82 /* AlertPresentation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AlertPresentation.swift; sourceTree = "<group>"; };
Expand Down Expand Up @@ -2803,6 +2807,7 @@
children = (
7A9FA1432A2E3FE5000B728D /* CheckableSettingsCell.swift */,
7A1A264A2A29D65E00B978AA /* SelectableSettingsCell.swift */,
7A27E3CE2CBD4A830088BCFF /* SelectableSettingsDetailsCell.swift */,
5819C2162729595500D6EC38 /* SettingsAddDNSEntryCell.swift */,
582BB1AE229566420055B6EF /* SettingsCell.swift */,
5864AF0029C7879B005B0CD9 /* SettingsCellFactory.swift */,
Expand Down Expand Up @@ -2845,6 +2850,7 @@
5864AF0229C7879B005B0CD9 /* VPNSettingsCellFactory.swift */,
584D26C3270C855A004EA533 /* VPNSettingsDataSource.swift */,
587EB6732714520600123C75 /* VPNSettingsDataSourceDelegate.swift */,
7A27E3D02CC299E60088BCFF /* VPNSettingsDetailsButtonItem.swift */,
7A6F2FAE2AFE36E7006D0856 /* VPNSettingsInfoButtonItem.swift */,
5871167E2910035700D41AAC /* VPNSettingsInteractor.swift */,
58ACF6482655365700ACE4B7 /* VPNSettingsViewController.swift */,
Expand Down Expand Up @@ -5658,6 +5664,7 @@
A91614D62B10B26B00F416EB /* TunnelControlViewModel.swift in Sources */,
7A5869972B32EA4500640D27 /* AppButton.swift in Sources */,
586C0D8F2B03D88100E7CDD7 /* ProxyProtocolConfigurationItemIdentifier.swift in Sources */,
7A27E3CF2CBD4A8C0088BCFF /* SelectableSettingsDetailsCell.swift in Sources */,
7A27E3CB2CAE861D0088BCFF /* SettingsViewModel.swift in Sources */,
588527B2276B3F0700BAA373 /* LoadTunnelConfigurationOperation.swift in Sources */,
7A9F29392CABFAFC005F2089 /* InfoHeaderView.swift in Sources */,
Expand Down Expand Up @@ -5854,6 +5861,7 @@
F0DA87492A9CBA9F006044F1 /* AccountDeviceRow.swift in Sources */,
58FF9FE42B075BDD00E4C97D /* EditAccessMethodItemIdentifier.swift in Sources */,
5878A27329091D6D0096FC88 /* TunnelBlockObserver.swift in Sources */,
7A27E3D12CC299F90088BCFF /* VPNSettingsDetailsButtonItem.swift in Sources */,
A9E034642ABB302000E59A5A /* UIEdgeInsets+Extensions.swift in Sources */,
58CEB2E92AFBBA4A00E6E088 /* AddAccessMethodCoordinator.swift in Sources */,
58DFF7D02B02560400F864E0 /* NSAttributedString+Extensions.swift in Sources */,
Expand Down
3 changes: 2 additions & 1 deletion ios/MullvadVPN/Classes/AccessbilityIdentifier.swift
Original file line number Diff line number Diff line change
Expand Up @@ -174,7 +174,8 @@ public enum AccessibilityIdentifier: String {
case wireGuardObfuscationAutomatic
case wireGuardObfuscationPort
case wireGuardObfuscationOff
case wireGuardObfuscationOn
case wireGuardObfuscationUDPTCP
case wireGuardObfuscationShadowsocks
case wireGuardPort

// Custom DNS
Expand Down
3 changes: 3 additions & 0 deletions ios/MullvadVPN/UI appearance/UIMetrics.swift
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ enum UIMetrics {
static let layoutMargins = NSDirectionalEdgeInsets(top: 16, leading: 24, bottom: 16, trailing: 12)
static let inputCellTextFieldLayoutMargins = UIEdgeInsets(top: 0, left: 8, bottom: 0, right: 8)
static let selectableSettingsCellLeftViewSpacing: CGFloat = 12
static let settingsCellRightViewSpacing: CGFloat = 12
static let checkableSettingsCellLeftViewSpacing: CGFloat = 20

/// Cell layout margins used in table views that use inset style.
Expand All @@ -93,6 +94,8 @@ enum UIMetrics {
static let customListsCellHeight: CGFloat = 44
static let apiAccessSwitchCellTrailingMargin: CGFloat = apiAccessInsetLayoutMargins.trailing - 4
static let apiAccessPickerListContentInsetTop: CGFloat = 16
static let buttonSeparatorHeight: CGFloat = 22
static let detailsButtonSize: CGFloat = 60
}

enum InAppBannerNotification {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ class CheckableSettingsCell: SettingsCell {
override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)

setLeftView(checkboxView, spacing: UIMetrics.SettingsCell.checkableSettingsCellLeftViewSpacing)
setLeadingView(checkboxView, spacing: UIMetrics.SettingsCell.checkableSettingsCellLeftViewSpacing)
selectedBackgroundView?.backgroundColor = .clear
}

Expand All @@ -25,7 +25,7 @@ class CheckableSettingsCell: SettingsCell {
override func prepareForReuse() {
super.prepareForReuse()

setLeftView(checkboxView, spacing: UIMetrics.SettingsCell.checkableSettingsCellLeftViewSpacing)
setLeadingView(checkboxView, spacing: UIMetrics.SettingsCell.checkableSettingsCellLeftViewSpacing)
}

override func setSelected(_ selected: Bool, animated: Bool) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import UIKit
class SelectableSettingsCell: SettingsCell {
let tickImageView: UIImageView = {
let imageView = UIImageView(image: UIImage(named: "IconTick"))
imageView.contentMode = .center
imageView.tintColor = .white
imageView.alpha = 0
return imageView
Expand All @@ -19,7 +20,7 @@ class SelectableSettingsCell: SettingsCell {
override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)

setLeftView(tickImageView, spacing: UIMetrics.SettingsCell.selectableSettingsCellLeftViewSpacing)
setLeadingView(tickImageView, spacing: UIMetrics.SettingsCell.selectableSettingsCellLeftViewSpacing)
selectedBackgroundView?.backgroundColor = UIColor.Cell.Background.selected
}

Expand All @@ -30,7 +31,7 @@ class SelectableSettingsCell: SettingsCell {
override func prepareForReuse() {
super.prepareForReuse()

setLeftView(tickImageView, spacing: UIMetrics.SettingsCell.selectableSettingsCellLeftViewSpacing)
setLeadingView(tickImageView, spacing: UIMetrics.SettingsCell.selectableSettingsCellLeftViewSpacing)
}

override func setSelected(_ selected: Bool, animated: Bool) {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
//
// SelectableSettingsDetailsCell.swift
// MullvadVPN
//
// Created by Jon Petersson on 2024-10-14.
// Copyright © 2024 Mullvad VPN AB. All rights reserved.
//

import UIKit

class SelectableSettingsDetailsCell: SelectableSettingsCell {
var action: (() -> Void)?

override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
super.init(style: .subtitle, reuseIdentifier: reuseIdentifier)

let actionButton = IncreasedHitButton(type: .system)
actionButton.setImage(UIImage(systemName: "ellipsis"), for: .normal)
actionButton.tintColor = .white

let separatorView = UIView()
separatorView.backgroundColor = .white

let viewContainer = UIView()
viewContainer.addConstrainedSubviews([separatorView, actionButton]) {
separatorView.leadingAnchor.constraint(equalTo: viewContainer.leadingAnchor, constant: 16)
separatorView.centerYAnchor.constraint(equalTo: viewContainer.centerYAnchor)
separatorView.heightAnchor.constraint(equalToConstant: UIMetrics.SettingsCell.buttonSeparatorHeight)
separatorView.widthAnchor.constraint(equalToConstant: 1)

actionButton.pinEdgesToSuperview(.all().excluding(.leading))
actionButton.leadingAnchor.constraint(equalTo: separatorView.trailingAnchor)
actionButton.widthAnchor.constraint(equalToConstant: UIMetrics.SettingsCell.detailsButtonSize)
}

setTrailingView { superview in
superview.addConstrainedSubviews([viewContainer]) {
viewContainer.pinEdgesToSuperview()
}
}

actionButton.addTarget(
self,
action: #selector(didPressActionButton),
for: .valueChanged
)
}

required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

// MARK: - Actions

@objc private func didPressActionButton() {
action?()
}
}
94 changes: 57 additions & 37 deletions ios/MullvadVPN/View controllers/Settings/SettingsCell.swift
Original file line number Diff line number Diff line change
Expand Up @@ -32,10 +32,11 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
typealias InfoButtonHandler = () -> Void

let contentContainerSubviewMaxCount = 2
let titleLabel = UILabel()
let detailTitleLabel = UILabel()
let disclosureImageView = UIImageView(image: nil)
let contentContainer = UIStackView()
let mainContent = UIView()
let mainContentContainer = UIStackView()
let leftContentContainer = UIStackView()
let rightContentContainer = UIView()
var infoButtonHandler: InfoButtonHandler? { didSet {
infoButton.isHidden = infoButtonHandler == nil
}}
Expand All @@ -59,8 +60,27 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
}
}

let titleLabel: UILabel = {
let label = UILabel()
label.font = UIFont.systemFont(ofSize: 17)
label.textColor = UIColor.Cell.titleTextColor
label.setContentHuggingPriority(.defaultHigh, for: .horizontal)
label.setContentCompressionResistancePriority(.defaultHigh, for: .horizontal)
return label
}()

let detailTitleLabel: UILabel = {
let label = UILabel()
label.font = UIFont.systemFont(ofSize: 13)
label.textColor = UIColor.Cell.detailTextColor
label.setContentHuggingPriority(.defaultLow, for: .horizontal)
label.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
return label
}()

private var subCellLeadingIndentation: CGFloat = 0
private let buttonWidth: CGFloat = 24

private let infoButton: UIButton = {
let button = UIButton(type: .custom)
button.accessibilityIdentifier = .infoButton
Expand All @@ -83,38 +103,29 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
backgroundColor = .clear
contentView.backgroundColor = .clear

infoButton.isHidden = true
infoButton.addTarget(self, action: #selector(handleInfoButton(_:)), for: .touchUpInside)

subCellLeadingIndentation = contentView.layoutMargins.left + UIMetrics.TableView.cellIndentationWidth

titleLabel.translatesAutoresizingMaskIntoConstraints = false
titleLabel.font = UIFont.systemFont(ofSize: 17)
titleLabel.textColor = UIColor.Cell.titleTextColor

detailTitleLabel.translatesAutoresizingMaskIntoConstraints = false
detailTitleLabel.font = UIFont.systemFont(ofSize: 13)
detailTitleLabel.textColor = UIColor.Cell.detailTextColor

titleLabel.setContentHuggingPriority(.defaultHigh, for: .horizontal)
detailTitleLabel.setContentHuggingPriority(.defaultLow, for: .horizontal)

titleLabel.setContentCompressionResistancePriority(.defaultHigh, for: .horizontal)
detailTitleLabel.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
rightContentContainer.setContentHuggingPriority(.required, for: .horizontal)

setLayoutMargins()

let buttonAreaWidth = UIMetrics.contentLayoutMargins.leading + UIMetrics
.contentLayoutMargins.trailing + buttonWidth

let content = UIView()
content.addConstrainedSubviews([titleLabel, infoButton, detailTitleLabel]) {
let infoButtonConstraint = infoButton.trailingAnchor.constraint(
greaterThanOrEqualTo: mainContent.trailingAnchor
)
infoButtonConstraint.priority = .defaultLow

mainContent.addConstrainedSubviews([titleLabel, infoButton, detailTitleLabel, rightContentContainer]) {
switch style {
case .subtitle:
titleLabel.pinEdgesToSuperview(.init([.top(0), .leading(0)]))
detailTitleLabel.pinEdgesToSuperview(.all().excluding(.top))
detailTitleLabel.topAnchor.constraint(equalToSystemSpacingBelow: titleLabel.bottomAnchor, multiplier: 1)
infoButton.trailingAnchor.constraint(greaterThanOrEqualTo: content.trailingAnchor)
detailTitleLabel.pinEdgesToSuperview(.all().excluding([.top, .trailing]))
detailTitleLabel.topAnchor.constraint(equalTo: titleLabel.bottomAnchor)
infoButtonConstraint

default:
titleLabel.pinEdgesToSuperview(.all().excluding(.trailing))
Expand All @@ -123,7 +134,7 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
}

infoButton.pinEdgesToSuperview(.init([.top(0)]))
infoButton.bottomAnchor.constraint(lessThanOrEqualTo: content.bottomAnchor)
infoButton.bottomAnchor.constraint(lessThanOrEqualTo: mainContent.bottomAnchor)
infoButton.leadingAnchor.constraint(
equalTo: titleLabel.trailingAnchor,
constant: -UIMetrics.interButtonSpacing
Expand All @@ -132,10 +143,14 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
infoButton.widthAnchor.constraint(equalToConstant: buttonAreaWidth)
}

contentContainer.addArrangedSubview(content)
mainContentContainer.addArrangedSubview(leftContentContainer)
mainContentContainer.addArrangedSubview(mainContent)

contentView.addConstrainedSubviews([contentContainer]) {
contentContainer.pinEdgesToSuperviewMargins()
contentView.addConstrainedSubviews([mainContentContainer, rightContentContainer]) {
mainContentContainer.pinEdgesToSuperviewMargins(.all().excluding(.trailing))

rightContentContainer.pinEdgesToSuperview(.all().excluding(.leading))
rightContentContainer.leadingAnchor.constraint(equalTo: mainContentContainer.trailingAnchor)
}
}

Expand All @@ -147,7 +162,7 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
super.prepareForReuse()

infoButton.isHidden = true
removeLeftView()
removeLeadingView()
setLayoutMargins()
}

Expand All @@ -156,20 +171,25 @@ class SettingsCell: UITableViewCell, CustomCellDisclosureHandling {
backgroundView?.backgroundColor = UIColor.Cell.Background.indentationLevelOne
}

func setLeftView(_ view: UIView, spacing: CGFloat) {
removeLeftView()
func setLeadingView(_ view: UIView, spacing: CGFloat) {
removeLeadingView()

if contentContainer.arrangedSubviews.count <= 1 {
contentContainer.insertArrangedSubview(view, at: 0)
}
leftContentContainer.addArrangedSubview(view)
mainContentContainer.setCustomSpacing(spacing, after: leftContentContainer)
}

contentContainer.spacing = spacing
func removeLeadingView() {
leftContentContainer.arrangedSubviews.forEach { $0.removeFromSuperview() }
mainContentContainer.setCustomSpacing(0, after: leftContentContainer)
}

func removeLeftView() {
if contentContainer.arrangedSubviews.count >= contentContainerSubviewMaxCount {
contentContainer.arrangedSubviews.first?.removeFromSuperview()
}
func setTrailingView(superviewProvider: (UIView) -> Void) {
removeTrailingView()
superviewProvider(rightContentContainer)
}

func removeTrailingView() {
rightContentContainer.subviews.forEach { $0.removeFromSuperview() }
}

@objc private func handleInfoButton(_ sender: UIControl) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import UIKit

protocol VPNSettingsCellEventHandler {
func showInfo(for button: VPNSettingsInfoButtonItem)
func showDetails(for button: VPNSettingsDetailsButtonItem)
func addCustomPort(_ port: UInt16)
func selectCustomPortEntry(_ port: UInt16) -> Bool
func selectObfuscationState(_ state: WireGuardObfuscationState)
Expand Down Expand Up @@ -135,17 +136,42 @@ final class VPNSettingsCellFactory: CellFactoryProtocol {
cell.accessibilityIdentifier = item.accessibilityIdentifier
cell.applySubCellStyling()

case .wireGuardObfuscationOn:
guard let cell = cell as? SelectableSettingsCell else { return }
case .wireGuardObfuscationUDPTCP:
guard let cell = cell as? SelectableSettingsDetailsCell else { return }

cell.titleLabel.text = NSLocalizedString(
"WIRE_GUARD_OBFUSCATION_UDP_TCP_LABEL",
tableName: "VPNSettings",
value: "UDP-over-TCP",
comment: ""
)
// TODO: When ready, add implementation for selected obfuscation options.
cell.detailTitleLabel.text = "Todo"
cell.accessibilityIdentifier = item.accessibilityIdentifier
cell.applySubCellStyling()

cell.action = { [weak self] in
self?.delegate?.showDetails(for: .udpTcp)
}

case .wireGuardObfuscationShadowsocks:
guard let cell = cell as? SelectableSettingsDetailsCell else { return }

cell.titleLabel.text = NSLocalizedString(
"WIRE_GUARD_OBFUSCATION_ON_LABEL",
"WIRE_GUARD_OBFUSCATION_SHADOWSOCKS_LABEL",
tableName: "VPNSettings",
value: "On (UDP-over-TCP)",
value: "Shadowsocks",
comment: ""
)
// TODO: When ready, add implementation for selected obfuscation options.
cell.detailTitleLabel.text = "Todo"
cell.accessibilityIdentifier = item.accessibilityIdentifier
cell.applySubCellStyling()

cell.action = { [weak self] in
self?.delegate?.showDetails(for: .shadowsocks)
}

case .wireGuardObfuscationOff:
guard let cell = cell as? SelectableSettingsCell else { return }

Expand Down
Loading

0 comments on commit 2591de0

Please sign in to comment.