Files
GitHawk/Classes/Issues/IssueViewModels.swift
2017-07-27 13:21:54 -06:00

126 lines
4.1 KiB
Swift

//
// IssueViewModels.swift
// Freetime
//
// Created by Ryan Nystrom on 5/19/17.
// Copyright © 2017 Ryan Nystrom. All rights reserved.
//
import UIKit
import IGListKit
func createViewModels(
issue: IssueType,
width: CGFloat
) -> (viewModels: [ListDiffable], timeline: [ListDiffable]) {
var viewModels = [ListDiffable]()
let status: IssueStatus = issue.merged ? .merged : issue.closableFields.closed ? .closed : .open
viewModels.append(IssueStatusModel(status: status, pullRequest: issue.pullRequest, locked: issue.locked))
viewModels.append(titleStringSizing(title: issue.title, width: width))
viewModels.append(IssueLabelsModel(labels: issue.labelableFields.issueLabelModels))
viewModels.append(createAssigneeModel(assigneeFields: issue.assigneeFields))
if let reviewers = issue.reviewRequestModel {
viewModels.append(reviewers)
}
if let root = createCommentModel(
id: issue.id,
commentFields: issue.commentFields,
reactionFields: issue.reactionFields,
width: width,
threadState: .single
) {
viewModels.append(root)
}
return (viewModels, issue.timelineViewModels(width: width))
}
func titleStringSizing(title: String, width: CGFloat) -> NSAttributedStringSizing {
let attributedString = NSAttributedString(
string: title,
attributes: [
NSFontAttributeName: Styles.Fonts.headline,
NSForegroundColorAttributeName: Styles.Colors.Gray.dark.color
])
return NSAttributedStringSizing(
containerWidth: width,
attributedText: attributedString,
inset: IssueTitleCell.inset
)
}
func createIssueReactions(reactions: ReactionFields) -> IssueCommentReactionViewModel {
var models = [ReactionViewModel]()
for group in reactions.reactionGroups ?? [] {
// do not display reactions for 0 count
let count = group.users.totalCount
guard count > 0 else { continue }
let nodes: [String] = {
guard let filtered = group.users.nodes?.filter({ $0?.login != nil }) as? [ReactionFields.ReactionGroup.User.Node] else {
return []
}
return filtered.map({ $0.login })
}()
models.append(ReactionViewModel(content: group.content, count: count, viewerDidReact: group.viewerHasReacted, users: nodes))
}
return IssueCommentReactionViewModel(models: models)
}
func createCommentModel(
id: String,
commentFields: CommentFields,
reactionFields: ReactionFields,
width: CGFloat,
threadState: IssueCommentModel.ThreadState
) -> IssueCommentModel? {
guard let author = commentFields.author,
let date = GithubAPIDateFormatter().date(from: commentFields.createdAt),
let avatarURL = URL(string: author.avatarUrl)
else { return nil }
let editedAt: Date? = {
guard let editedDate = commentFields.lastEditedAt else { return nil }
return GithubAPIDateFormatter().date(from: editedDate)
}()
let details = IssueCommentDetailsViewModel(
date: date,
login: author.login,
avatarURL: avatarURL,
didAuthor: commentFields.viewerDidAuthor,
editedBy: commentFields.editor?.login,
editedAt: editedAt
)
let bodies = CreateCommentModels(markdown: commentFields.body, width: width)
let reactions = createIssueReactions(reactions: reactionFields)
let collapse = IssueCollapsedBodies(bodies: bodies, width: width)
return IssueCommentModel(
id: id,
details: details,
bodyModels: bodies,
reactions: reactions,
collapse: collapse,
threadState: threadState
)
}
func createAssigneeModel(assigneeFields: AssigneeFields) -> IssueAssigneesModel {
var models = [IssueAssigneeViewModel]()
for node in assigneeFields.assignees.nodes ?? [] {
guard let node = node,
let url = URL(string: node.avatarUrl)
else { continue }
models.append(IssueAssigneeViewModel(login: node.login, avatarURL: url))
}
return IssueAssigneesModel(users: models, type: .assigned)
}