Swiftgram/TelegramUI/CreateGroupController.swift
2017-04-18 19:53:47 +03:00

240 lines
9.3 KiB
Swift

import Foundation
import Display
import SwiftSignalKit
import Postbox
import TelegramCore
private struct CreateGroupArguments {
let account: Account
let updateEditingName: (ItemListAvatarAndNameInfoItemName) -> Void
let done: () -> Void
}
private enum CreateGroupSection: Int32 {
case info
case members
}
private enum CreateGroupEntry: ItemListNodeEntry {
case groupInfo(Peer?, ItemListAvatarAndNameInfoItemState)
case setProfilePhoto
case member(Int32, Peer, PeerPresence?)
var section: ItemListSectionId {
switch self {
case .groupInfo, .setProfilePhoto:
return CreateGroupSection.info.rawValue
case .member:
return CreateGroupSection.members.rawValue
}
}
var stableId: Int32 {
switch self {
case .groupInfo:
return 0
case .setProfilePhoto:
return 1
case let .member(index, _, _):
return 2 + index
}
}
static func ==(lhs: CreateGroupEntry, rhs: CreateGroupEntry) -> Bool {
switch lhs {
case let .groupInfo(lhsPeer, lhsEditingState):
if case let .groupInfo(rhsPeer, rhsEditingState) = rhs {
if let lhsPeer = lhsPeer, let rhsPeer = rhsPeer {
if !lhsPeer.isEqual(rhsPeer) {
return false
}
} else if (lhsPeer != nil) != (rhsPeer != nil) {
return false
}
if lhsEditingState != rhsEditingState {
return false
}
return true
} else {
return false
}
case .setProfilePhoto:
if case .setProfilePhoto = rhs {
return true
} else {
return false
}
case let .member(lhsIndex, lhsPeer, lhsPresence):
if case let .member(rhsIndex, rhsPeer, rhsPresence) = rhs {
if lhsIndex != rhsIndex {
return false
}
if !lhsPeer.isEqual(rhsPeer) {
return false
}
if let lhsPresence = lhsPresence, let rhsPresence = rhsPresence {
if !lhsPresence.isEqual(to: rhsPresence) {
return false
}
} else if (lhsPresence != nil) != (rhsPresence != nil) {
return false
}
return true
} else {
return false
}
}
}
static func <(lhs: CreateGroupEntry, rhs: CreateGroupEntry) -> Bool {
return lhs.stableId < rhs.stableId
}
func item(_ arguments: CreateGroupArguments) -> ListViewItem {
switch self {
case let .groupInfo(peer, state):
return ItemListAvatarAndNameInfoItem(account: arguments.account, peer: peer, presence: nil, cachedData: nil, state: state, sectionId: ItemListSectionId(self.section), style: .blocks, editingNameUpdated: { editingName in
arguments.updateEditingName(editingName)
}, avatarTapped: {
})
case .setProfilePhoto:
return ItemListActionItem(title: "Set Profile Photo", kind: .generic, alignment: .natural, sectionId: ItemListSectionId(self.section), style: .blocks, action: {
})
case let .member(_, peer, presence):
return ItemListPeerItem(account: arguments.account, peer: peer, presence: presence, text: .presence, label: .none, editing: ItemListPeerItemEditing(editable: false, editing: false, revealed: false), switchValue: nil, enabled: true, sectionId: self.section, action: nil, setPeerIdWithRevealedOptions: { _ in }, removePeer: { _ in })
}
}
}
private struct CreateGroupState: Equatable {
let creating: Bool
let editingName: ItemListAvatarAndNameInfoItemName
static func ==(lhs: CreateGroupState, rhs: CreateGroupState) -> Bool {
if lhs.creating != rhs.creating {
return false
}
if lhs.editingName != rhs.editingName {
return false
}
return true
}
}
private func createGroupEntries(state: CreateGroupState, peerIds: [PeerId], view: MultiplePeersView) -> [CreateGroupEntry] {
var entries: [CreateGroupEntry] = []
let groupInfoState = ItemListAvatarAndNameInfoItemState(editingName: state.editingName, updatingName: nil)
let peer = TelegramGroup(id: PeerId(namespace: 100, id: 0), title: state.editingName.composedTitle, photo: [], participantCount: 0, role: .creator, membership: .Member, flags: [], migrationReference: nil, creationDate: 0, version: 0)
entries.append(.groupInfo(peer, groupInfoState))
entries.append(.setProfilePhoto)
var peers: [Peer] = []
for peerId in peerIds {
if let peer = view.peers[peerId] {
peers.append(peer)
}
}
peers.sort(by: { lhs, rhs in
let lhsPresence = view.presences[lhs.id] as? TelegramUserPresence
let rhsPresence = view.presences[rhs.id] as? TelegramUserPresence
if let lhsPresence = lhsPresence, let rhsPresence = rhsPresence {
if lhsPresence.status < rhsPresence.status {
return false
} else if lhsPresence.status > rhsPresence.status {
return true
} else {
return lhs.id < rhs.id
}
} else if let _ = lhsPresence {
return true
} else if let _ = rhsPresence {
return false
} else {
return lhs.id < rhs.id
}
})
for i in 0 ..< peers.count {
entries.append(.member(Int32(i), peers[i], view.presences[peers[i].id]))
}
return entries
}
public func createGroupController(account: Account, peerIds: [PeerId]) -> ViewController {
let initialState = CreateGroupState(creating: false, editingName: .title(title: ""))
let statePromise = ValuePromise(initialState, ignoreRepeated: true)
let stateValue = Atomic(value: initialState)
let updateState: ((CreateGroupState) -> CreateGroupState) -> Void = { f in
statePromise.set(stateValue.modify { f($0) })
}
var replaceControllerImpl: ((ViewController) -> Void)?
let actionsDisposable = DisposableSet()
let arguments = CreateGroupArguments(account: account, updateEditingName: { editingName in
updateState { current in
return CreateGroupState(creating: current.creating, editingName: editingName)
}
}, done: {
let (creating, title) = stateValue.with { state -> (Bool, String) in
return (state.creating, state.editingName.composedTitle)
}
if !creating && !title.isEmpty {
updateState { current in
return CreateGroupState(creating: true, editingName: current.editingName)
}
actionsDisposable.add((createGroup(account: account, title: title, peerIds: peerIds) |> deliverOnMainQueue |> afterDisposed {
Queue.mainQueue().async {
updateState { current in
return CreateGroupState(creating: false, editingName: current.editingName)
}
}
}).start(next: { peerId in
if let peerId = peerId {
let controller = ChatController(account: account, peerId: peerId)
replaceControllerImpl?(controller)
}
}))
}
})
let signal = combineLatest(statePromise.get(), account.postbox.multiplePeersView(peerIds))
|> map { state, view -> (ItemListControllerState, (ItemListNodeState<CreateGroupEntry>, CreateGroupEntry.ItemGenerationArguments)) in
let rightNavigationButton: ItemListNavigationButton
if state.creating {
rightNavigationButton = ItemListNavigationButton(title: "", style: .activity, enabled: true, action: {})
} else {
rightNavigationButton = ItemListNavigationButton(title: "Create", style: .bold, enabled: !state.editingName.composedTitle.isEmpty, action: {
arguments.done()
})
}
let controllerState = ItemListControllerState(title: .text("Create Group"), leftNavigationButton: nil, rightNavigationButton: rightNavigationButton)
let listState = ItemListNodeState(entries: createGroupEntries(state: state, peerIds: peerIds, view: view), style: .blocks)
return (controllerState, (listState, arguments))
} |> afterDisposed {
actionsDisposable.dispose()
}
let controller = ItemListController(signal)
controller.navigationItem.backBarButtonItem = UIBarButtonItem(title: "Back", style: .plain, target: nil, action: nil)
replaceControllerImpl = { [weak controller] value in
(controller?.navigationController as? NavigationController)?.replaceAllButRootController(value, animated: true)
}
return controller
}