@objc(LKGroupMembersVC) final class GroupMembersVC : BaseVC, UITableViewDataSource { private let thread: TSGroupThread private lazy var members: [String] = { func getDisplayName(for hexEncodedPublicKey: String) -> String { return UserDisplayNameUtilities.getPrivateChatDisplayName(for: hexEncodedPublicKey) ?? hexEncodedPublicKey } return GroupUtilities.getClosedGroupMembers(thread).sorted { getDisplayName(for: $0) < getDisplayName(for: $1) } }() // MARK: Components @objc private lazy var tableView: UITableView = { let result = UITableView() result.dataSource = self result.register(Cell.self, forCellReuseIdentifier: "Cell") result.separatorStyle = .none result.backgroundColor = .clear result.showsVerticalScrollIndicator = false result.alwaysBounceVertical = false return result }() // MARK: Lifecycle @objc init(thread: TSGroupThread) { self.thread = thread super.init(nibName: nil, bundle: nil) } required init?(coder: NSCoder) { fatalError("Using GroupMembersVC.init(nibName:bundle:) isn't allowed. Use GroupMembersVC.init(thread:) instead.") } override init(nibName: String?, bundle: Bundle?) { fatalError("Using GroupMembersVC.init(nibName:bundle:) isn't allowed. Use GroupMembersVC.init(thread:) instead.") } override func viewDidLoad() { super.viewDidLoad() setUpGradientBackground() setUpNavBarStyle() setNavBarTitle("Group Members") // Set up explanation label let explanationLabel = UILabel() explanationLabel.textColor = Colors.text.withAlphaComponent(Values.mediumOpacity) explanationLabel.font = .systemFont(ofSize: Values.smallFontSize) explanationLabel.text = "The ability to add members to a closed group is coming soon." explanationLabel.numberOfLines = 0 explanationLabel.textAlignment = .center explanationLabel.lineBreakMode = .byWordWrapping // Set up view hierarchy view.addSubview(explanationLabel) explanationLabel.pin(.leading, to: .leading, of: view, withInset: Values.largeSpacing) explanationLabel.pin(.top, to: .top, of: view, withInset: Values.mediumSpacing) explanationLabel.pin(.trailing, to: .trailing, of: view, withInset: -Values.largeSpacing) let separator = UIView() separator.backgroundColor = Colors.separator separator.set(.height, to: Values.separatorThickness) view.addSubview(separator) separator.pin(.leading, to: .leading, of: view) separator.pin(.top, to: .bottom, of: explanationLabel, withInset: Values.mediumSpacing) separator.pin(.trailing, to: .trailing, of: view) view.addSubview(tableView) tableView.pin(.leading, to: .leading, of: view) tableView.pin(.top, to: .bottom, of: separator) tableView.pin(.trailing, to: .trailing, of: view) tableView.pin(.bottom, to: .bottom, of: view) } // MARK: Data func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return members.count } func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let cell = tableView.dequeueReusableCell(withIdentifier: "Cell") as! Cell let contact = members[indexPath.row] cell.hexEncodedPublicKey = contact return cell } @objc private func close() { dismiss(animated: true, completion: nil) } } // MARK: - Cell private extension GroupMembersVC { final class Cell : UITableViewCell { var hexEncodedPublicKey = "" { didSet { update() } } // MARK: Components private lazy var profilePictureView = ProfilePictureView() private lazy var displayNameLabel: UILabel = { let result = UILabel() result.textColor = Colors.text result.font = .boldSystemFont(ofSize: Values.mediumFontSize) result.lineBreakMode = .byTruncatingTail return result }() private lazy var separator: UIView = { let result = UIView() result.backgroundColor = Colors.separator result.set(.height, to: Values.separatorThickness) return result }() // MARK: Initialization override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) { super.init(style: style, reuseIdentifier: reuseIdentifier) setUpViewHierarchy() } required init?(coder: NSCoder) { super.init(coder: coder) setUpViewHierarchy() } private func setUpViewHierarchy() { // Set the cell background color backgroundColor = Colors.cellBackground // Set up the highlight color let selectedBackgroundView = UIView() selectedBackgroundView.backgroundColor = .clear // Disabled for now self.selectedBackgroundView = selectedBackgroundView // Set up the profile picture image view let profilePictureViewSize = Values.smallProfilePictureSize profilePictureView.set(.width, to: profilePictureViewSize) profilePictureView.set(.height, to: profilePictureViewSize) profilePictureView.size = profilePictureViewSize // Set up the main stack view let stackView = UIStackView(arrangedSubviews: [ profilePictureView, displayNameLabel ]) stackView.axis = .horizontal stackView.alignment = .center stackView.spacing = Values.mediumSpacing stackView.set(.height, to: profilePictureViewSize) contentView.addSubview(stackView) stackView.pin(.leading, to: .leading, of: contentView, withInset: Values.mediumSpacing) stackView.pin(.top, to: .top, of: contentView, withInset: Values.mediumSpacing) contentView.pin(.bottom, to: .bottom, of: stackView, withInset: Values.mediumSpacing) stackView.set(.width, to: UIScreen.main.bounds.width - 2 * Values.mediumSpacing) // Set up the separator addSubview(separator) separator.pin(.leading, to: .leading, of: self) separator.pin(.bottom, to: .bottom, of: self) separator.set(.width, to: UIScreen.main.bounds.width) } // MARK: Updating private func update() { profilePictureView.publicKey = hexEncodedPublicKey profilePictureView.update() displayNameLabel.text = UserDisplayNameUtilities.getPrivateChatDisplayName(for: hexEncodedPublicKey) ?? hexEncodedPublicKey } } }