// // NCNotification.swift // Nextcloud // // Created by Marino Faggiana on 27/01/17. // Copyright (c) 2017 Marino Faggiana. All rights reserved. // // Author Marino Faggiana // Author Henrik Storch // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // import UIKit import NextcloudKit import SwiftyJSON import JGProgressHUD class NCNotification: UITableViewController, NCNotificationCellDelegate, NCEmptyDataSetDelegate { let appDelegate = UIApplication.shared.delegate as! AppDelegate var notifications: [NKNotifications] = [] var emptyDataSet: NCEmptyDataSet? var isReloadDataSourceNetworkInProgress: Bool = false // MARK: - View Life Cycle override func viewDidLoad() { super.viewDidLoad() title = NSLocalizedString("_notifications_", comment: "") view.backgroundColor = .systemBackground tableView.tableFooterView = UIView() tableView.rowHeight = UITableView.automaticDimension tableView.estimatedRowHeight = 50.0 tableView.backgroundColor = .systemBackground refreshControl?.addTarget(self, action: #selector(getNetwokingNotification), for: .valueChanged) // Navigation controller is being presented modally if navigationController?.presentingViewController != nil { navigationItem.leftBarButtonItem = UIBarButtonItem(title: NSLocalizedString("_cancel_", comment: ""), style: .plain, action: { [weak self] in self?.dismiss(animated: true) }) } // Empty let offset = (self.navigationController?.navigationBar.bounds.height ?? 0) - 20 emptyDataSet = NCEmptyDataSet(view: tableView, offset: -offset, delegate: self) } override func viewWillAppear(_ animated: Bool) { super.viewWillAppear(animated) appDelegate.activeViewController = self navigationController?.setFileAppreance() getNetwokingNotification() } @objc func viewClose() { self.dismiss(animated: true, completion: nil) } // MARK: - Empty func emptyDataSetView(_ view: NCEmptyView) { if isReloadDataSourceNetworkInProgress { view.emptyImage.image = UIImage(named: "networkInProgress")?.image(color: .gray, size: UIScreen.main.bounds.width) view.emptyTitle.text = NSLocalizedString("_request_in_progress_", comment: "") view.emptyDescription.text = "" } else { view.emptyImage.image = NCUtility.shared.loadImage(named: "bell", color: .gray, size: UIScreen.main.bounds.width) view.emptyTitle.text = NSLocalizedString("_no_notification_", comment: "") view.emptyDescription.text = "" } } // MARK: - Table override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { emptyDataSet?.numberOfItemsInSection(notifications.count, section: section) return notifications.count } override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { guard let notification = NCApplicationHandle().didSelectNotification(notifications[indexPath.row], viewController: self) else { return } do { if let subjectRichParameters = notification.subjectRichParameters, let json = try JSONSerialization.jsonObject(with: subjectRichParameters, options: .mutableContainers) as? [String: Any], let file = json["file"] as? [String: Any], file["type"] as? String == "file" { if let id = file["id"] { NCActionCenter.shared.viewerFile(account: appDelegate.account, fileId: ("\(id)"), viewController: self) } } } catch { print("Something went wrong") } } override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let cell = self.tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath) as! NCNotificationCell cell.delegate = self cell.selectionStyle = .none cell.indexPath = indexPath let notification = notifications[indexPath.row] let urlIcon = URL(string: notification.icon) var image: UIImage? if let urlIcon = urlIcon { let pathFileName = String(CCUtility.getDirectoryUserData()) + "/" + urlIcon.deletingPathExtension().lastPathComponent + ".png" image = UIImage(contentsOfFile: pathFileName) } if let image = image { cell.icon.image = image.withTintColor(NCBrandColor.shared.brandElement, renderingMode: .alwaysOriginal) } else { cell.icon.image = NCUtility.shared.loadImage(named: "bell", color: NCBrandColor.shared.brandElement) } // Avatar cell.avatar.isHidden = true cell.avatarLeadingMargin.constant = 10 if let subjectRichParameters = notification.subjectRichParameters, let json = JSON(subjectRichParameters).dictionary, let user = json["user"]?["id"].stringValue { cell.avatar.isHidden = false cell.avatarLeadingMargin.constant = 50 let fileName = appDelegate.userBaseUrl + "-" + user + ".png" let fileNameLocalPath = String(CCUtility.getDirectoryUserData()) + "/" + fileName if let image = UIImage(contentsOfFile: fileNameLocalPath) { cell.avatar.image = image } else if !FileManager.default.fileExists(atPath: fileNameLocalPath) { cell.fileUser = user NCOperationQueue.shared.downloadAvatar(user: user, dispalyName: json["user"]?["name"].string, fileName: fileName, cell: cell, view: tableView, cellImageView: cell.fileAvatarImageView) } } cell.date.text = DateFormatter.localizedString(from: notification.date as Date, dateStyle: .medium, timeStyle: .medium) cell.notification = notification cell.date.text = CCUtility.dateDiff(notification.date as Date) cell.date.textColor = .gray cell.subject.text = notification.subject cell.subject.textColor = .label cell.message.text = notification.message.replacingOccurrences(of: "
", with: "\n") cell.message.textColor = .gray cell.remove.setImage(UIImage(named: "xmark")!.image(color: .gray, size: 20), for: .normal) cell.primary.isEnabled = false cell.primary.isHidden = true cell.primary.titleLabel?.font = .systemFont(ofSize: 15) cell.primary.layer.cornerRadius = 15 cell.primary.layer.masksToBounds = true cell.primary.layer.backgroundColor = NCBrandColor.shared.brandElement.cgColor cell.primary.setTitleColor(NCBrandColor.shared.brandText, for: .normal) cell.more.isEnabled = false cell.more.isHidden = true cell.more.titleLabel?.font = .systemFont(ofSize: 15) cell.more.layer.cornerRadius = 15 cell.more.layer.masksToBounds = true cell.more.layer.backgroundColor = NCBrandColor.shared.brandElement.cgColor cell.more.setTitleColor(NCBrandColor.shared.brandText, for: .normal) cell.secondary.isEnabled = false cell.secondary.isHidden = true cell.secondary.titleLabel?.font = .systemFont(ofSize: 15) cell.secondary.layer.cornerRadius = 15 cell.secondary.layer.masksToBounds = true cell.secondary.layer.borderWidth = 1 cell.secondary.layer.borderColor = UIColor.systemGray.cgColor cell.secondary.layer.backgroundColor = UIColor.secondarySystemBackground.cgColor cell.secondary.setTitleColor(.gray, for: .normal) // Action if let actions = notification.actions, let jsonActions = JSON(actions).array { if jsonActions.count == 1 { let action = jsonActions[0] cell.primary.isEnabled = true cell.primary.isHidden = false cell.primary.setTitle(action["label"].stringValue, for: .normal) } else if jsonActions.count == 2 { cell.primary.isEnabled = true cell.primary.isHidden = false cell.secondary.isEnabled = true cell.secondary.isHidden = false for action in jsonActions { let label = action["label"].stringValue let primary = action["primary"].boolValue if primary { cell.primary.setTitle(label, for: .normal) } else { cell.secondary.setTitle(label, for: .normal) } } } else if jsonActions.count >= 3 { cell.more.isEnabled = true cell.more.isHidden = false cell.more.setTitle("…", for: .normal) } var buttonWidth = max(cell.primary.intrinsicContentSize.width, cell.secondary.intrinsicContentSize.width) buttonWidth += 30 cell.primaryWidth.constant = buttonWidth cell.secondaryWidth.constant = buttonWidth } return cell } // MARK: - tap Action func tapRemove(with notification: NKNotifications) { NextcloudKit.shared.setNotification(serverUrl: nil, idNotification: notification.idNotification, method: "DELETE") { account, error in if error == .success && account == self.appDelegate.account { if let index = self.notifications .firstIndex(where: { $0.idNotification == notification.idNotification }) { self.notifications.remove(at: index) } self.tableView.reloadData() } else if error != .success { NCContentPresenter.shared.showError(error: error) } else { print("[Error] The user has been changed during networking process.") } } } func tapAction(with notification: NKNotifications, label: String) { if notification.app == NCGlobal.shared.spreedName, let roomToken = notification.objectId.split(separator: "/").first, let talkUrl = URL(string: "nextcloudtalk://open-conversation?server=\(appDelegate.urlBase)&user=\(appDelegate.userId)&withRoomToken=\(roomToken)"), UIApplication.shared.canOpenURL(talkUrl) { UIApplication.shared.open(talkUrl) } else if let actions = notification.actions, let jsonActions = JSON(actions).array, let action = jsonActions.first(where: { $0["label"].string == label }) { let serverUrl = action["link"].stringValue let method = action["type"].stringValue if method == "WEB", let url = action["link"].url { UIApplication.shared.open(url, options: [:], completionHandler: nil) return } NextcloudKit.shared.setNotification(serverUrl: serverUrl, idNotification: 0, method: method) { account, error in if error == .success && account == self.appDelegate.account { if let index = self.notifications.firstIndex(where: { $0.idNotification == notification.idNotification }) { self.notifications.remove(at: index) } self.tableView.reloadData() if self.navigationController?.presentingViewController != nil, notification.app == NCGlobal.shared.twoFactorNotificatioName { self.dismiss(animated: true) } } else if error != .success { NCContentPresenter.shared.showError(error: error) } else { print("[Error] The user has been changed during networking process.") } } } // else: Action not found } func tapMore(with notification: NKNotifications) { toggleMenu(notification: notification) } // MARK: - Load notification networking @objc func getNetwokingNotification() { isReloadDataSourceNetworkInProgress = true self.tableView.reloadData() NextcloudKit.shared.getNotifications { account, notifications, _, error in if error == .success && account == self.appDelegate.account { self.notifications.removeAll() let sortedListOfNotifications = (notifications! as NSArray).sortedArray(using: [NSSortDescriptor(key: "date", ascending: false)]) for notification in sortedListOfNotifications { if let icon = (notification as! NKNotifications).icon { NCUtility.shared.convertSVGtoPNGWriteToUserData(svgUrlString: icon, fileName: nil, width: 25, rewrite: false, account: self.appDelegate.account, completion: { _, _ in }) } self.notifications.append(notification as! NKNotifications) } self.refreshControl?.endRefreshing() self.isReloadDataSourceNetworkInProgress = false self.tableView.reloadData() } } } } // MARK: - class NCNotificationCell: UITableViewCell, NCCellProtocol { @IBOutlet weak var icon: UIImageView! @IBOutlet weak var avatar: UIImageView! @IBOutlet weak var date: UILabel! @IBOutlet weak var subject: UILabel! @IBOutlet weak var message: UILabel! @IBOutlet weak var remove: UIButton! @IBOutlet weak var primary: UIButton! @IBOutlet weak var secondary: UIButton! @IBOutlet weak var more: UIButton! @IBOutlet weak var avatarLeadingMargin: NSLayoutConstraint! @IBOutlet weak var primaryWidth: NSLayoutConstraint! @IBOutlet weak var secondaryWidth: NSLayoutConstraint! private var user = "" private var index = IndexPath() weak var delegate: NCNotificationCellDelegate? var notification: NKNotifications? var indexPath: IndexPath { get { return index } set { index = newValue } } var fileAvatarImageView: UIImageView? { get { return avatar } } var fileUser: String? { get { return user } set { user = newValue ?? "" } } override func awakeFromNib() { super.awakeFromNib() } @IBAction func touchUpInsideRemove(_ sender: Any) { guard let notification = notification else { return } delegate?.tapRemove(with: notification) } @IBAction func touchUpInsidePrimary(_ sender: Any) { guard let notification = notification, let button = sender as? UIButton, let label = button.titleLabel?.text else { return } delegate?.tapAction(with: notification, label: label) } @IBAction func touchUpInsideSecondary(_ sender: Any) { guard let notification = notification, let button = sender as? UIButton, let label = button.titleLabel?.text else { return } delegate?.tapAction(with: notification, label: label) } @IBAction func touchUpInsideMore(_ sender: Any) { guard let notification = notification else { return } delegate?.tapMore(with: notification) } } protocol NCNotificationCellDelegate: AnyObject { func tapRemove(with notification: NKNotifications) func tapAction(with notification: NKNotifications, label: String) func tapMore(with notification: NKNotifications) }