NCDetailViewController.swift 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546
  1. //
  2. // NCDetailViewController.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 07/02/2020.
  6. // Copyright © 2020 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. import Foundation
  24. import WebKit
  25. import NCCommunication
  26. class NCDetailViewController: UIViewController {
  27. @IBOutlet weak var backgroundView: UIImageView!
  28. @objc var metadata: tableMetadata?
  29. @objc var selector: String?
  30. @objc var favoriteFilterImage: Bool = false
  31. @objc var mediaFilterImage: Bool = false
  32. @objc var offlineFilterImage: Bool = false
  33. private let appDelegate = UIApplication.shared.delegate as! AppDelegate
  34. private var mediaBrowser: MediaBrowserViewController?
  35. private var metadatas = [tableMetadata]()
  36. private var mediaBrowserIndexStart = 0
  37. //MARK: -
  38. required init?(coder: NSCoder) {
  39. super.init(coder: coder)
  40. appDelegate.activeDetail = self
  41. }
  42. override func viewDidLoad() {
  43. super.viewDidLoad()
  44. NotificationCenter.default.addObserver(self, selector: #selector(self.changeTheming), name: NSNotification.Name(rawValue: k_notificationCenter_changeTheming), object: nil)
  45. NotificationCenter.default.addObserver(self, selector: #selector(self.changeDisplayMode), name: NSNotification.Name(rawValue: k_notificationCenter_splitViewChangeDisplayMode), object: nil)
  46. NotificationCenter.default.addObserver(self, selector: #selector(self.deleteFile(_:)), name: NSNotification.Name(rawValue: k_notificationCenter_deleteFile), object: nil)
  47. NotificationCenter.default.addObserver(self, selector: #selector(self.uploadFile(_:)), name: NSNotification.Name(rawValue: k_notificationCenter_uploadFile), object: nil)
  48. NotificationCenter.default.addObserver(self, selector: #selector(self.renameFile(_:)), name: NSNotification.Name(rawValue: k_notificationCenter_renameFile), object: nil)
  49. NotificationCenter.default.addObserver(self, selector: #selector(self.moveFile(_:)), name: NSNotification.Name(rawValue: k_notificationCenter_moveFile), object: nil)
  50. changeTheming()
  51. if metadata != nil {
  52. viewFile(metadata: metadata!, selector: selector)
  53. }
  54. }
  55. override func viewDidDisappear(_ animated: Bool) {
  56. super.viewDidDisappear(animated)
  57. if appDelegate.player != nil && appDelegate.player.rate != 0 {
  58. appDelegate.player.pause()
  59. }
  60. if appDelegate.isMediaObserver {
  61. appDelegate.isMediaObserver = false
  62. NCViewerMedia.sharedInstance.removeObserver()
  63. }
  64. }
  65. override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
  66. super.viewWillTransition(to: size, with: coordinator)
  67. coordinator.animate(alongsideTransition: nil) { _ in
  68. }
  69. }
  70. //MARK: - Utility
  71. func subViewActive() -> UIView? {
  72. return backgroundView.subviews.first
  73. }
  74. func viewUnload() {
  75. metadata = nil
  76. selector = nil
  77. if let splitViewController = self.splitViewController as? NCSplitViewController {
  78. if splitViewController.isCollapsed {
  79. if let navigationController = splitViewController.viewControllers.last as? UINavigationController {
  80. navigationController.popToRootViewController(animated: true)
  81. }
  82. } else {
  83. for view in backgroundView.subviews {
  84. view.removeFromSuperview()
  85. }
  86. self.navigationController?.navigationBar.topItem?.title = ""
  87. }
  88. }
  89. self.splitViewController?.preferredDisplayMode = .allVisible
  90. self.navigationController?.isNavigationBarHidden = false
  91. view.backgroundColor = NCBrandColor.sharedInstance.backgroundView
  92. backgroundView.image = CCGraphics.changeThemingColorImage(UIImage.init(named: "logo"), multiplier: 2, color: NCBrandColor.sharedInstance.brand.withAlphaComponent(0.4))
  93. }
  94. //MARK: - NotificationCenter
  95. @objc func changeTheming() {
  96. if backgroundView.image != nil {
  97. backgroundView.image = CCGraphics.changeThemingColorImage(UIImage.init(named: "logo"), multiplier: 2, color: NCBrandColor.sharedInstance.brand.withAlphaComponent(0.4))
  98. }
  99. if navigationController?.isNavigationBarHidden == false {
  100. view.backgroundColor = NCBrandColor.sharedInstance.backgroundView
  101. }
  102. }
  103. @objc func changeDisplayMode() {
  104. DispatchQueue.main.async {
  105. self.mediaBrowser?.changeInViewSize(to: self.backgroundView.frame.size)
  106. }
  107. }
  108. @objc func moveFile(_ notification: NSNotification) {
  109. if let userInfo = notification.userInfo as NSDictionary? {
  110. if let metadata = userInfo["metadata"] as? tableMetadata, let errorCode = userInfo["errorCode"] as? Int {
  111. if errorCode != 0 || metadata.serverUrl != self.metadata?.serverUrl { return }
  112. self.deleteFile(notification)
  113. }
  114. }
  115. }
  116. @objc func deleteFile(_ notification: NSNotification) {
  117. if let userInfo = notification.userInfo as NSDictionary? {
  118. if let metadata = userInfo["metadata"] as? tableMetadata, let errorCode = userInfo["errorCode"] as? Int {
  119. if errorCode != 0 { return }
  120. // IMAGE
  121. if mediaBrowser != nil && metadata.account == self.metadata?.account && metadata.serverUrl == self.metadata?.serverUrl && metadata.typeFile == k_metadataTypeFile_image {
  122. if let metadatas = getMetadatasMediaBrowser() {
  123. var index = mediaBrowser!.index - 1
  124. if index < 0 { index = 0}
  125. self.metadata = metadatas[index]
  126. viewImage()
  127. } else {
  128. viewUnload()
  129. }
  130. // OTHER SINGLE FILE TYPE
  131. } else if metadata.ocId == self.metadata?.ocId {
  132. viewUnload()
  133. }
  134. }
  135. }
  136. }
  137. @objc func uploadFile(_ notification: NSNotification) {
  138. if let userInfo = notification.userInfo as NSDictionary? {
  139. if let metadata = userInfo["metadata"] as? tableMetadata, let errorCode = userInfo["errorCode"] as? Int {
  140. if errorCode != 0 { return }
  141. // IMAGE
  142. if mediaBrowser != nil && metadata.account == self.metadata?.account && metadata.serverUrl == self.metadata?.serverUrl && metadata.typeFile == k_metadataTypeFile_image {
  143. if getMetadatasMediaBrowser() != nil {
  144. viewImage()
  145. } else {
  146. viewUnload()
  147. }
  148. }
  149. }
  150. }
  151. }
  152. @objc func renameFile(_ notification: NSNotification) {
  153. if let userInfo = notification.userInfo as NSDictionary? {
  154. if let metadata = userInfo["metadata"] as? tableMetadata, let errorCode = userInfo["errorCode"] as? Int {
  155. if errorCode != 0 { return }
  156. // IMAGE
  157. if mediaBrowser != nil && metadata.account == self.metadata?.account && metadata.serverUrl == self.metadata?.serverUrl && metadata.typeFile == k_metadataTypeFile_image {
  158. if getMetadatasMediaBrowser() != nil {
  159. viewImage()
  160. } else {
  161. viewUnload()
  162. }
  163. // OTHER SINGLE FILE TYPE
  164. } else if metadata.ocId == self.metadata?.ocId {
  165. self.navigationController?.navigationBar.topItem?.title = metadata.fileNameView
  166. }
  167. }
  168. }
  169. }
  170. //MARK: -
  171. @objc func viewFile(metadata: tableMetadata, selector: String?) {
  172. self.metadata = metadata
  173. self.selector = selector
  174. self.backgroundView.image = nil
  175. for view in backgroundView.subviews { view.removeFromSuperview() }
  176. self.navigationController?.navigationBar.topItem?.title = metadata.fileNameView
  177. if FileManager().fileExists(atPath: CCUtility.getDirectoryProviderStorageIconOcId(metadata.ocId, fileNameView: metadata.fileNameView)) == false {
  178. CCGraphics.createNewImage(from: metadata.fileNameView, ocId: metadata.ocId, extension: (metadata.fileNameView as NSString).pathExtension, filterGrayScale: false, typeFile: metadata.typeFile, writeImage: true)
  179. }
  180. if appDelegate.isMediaObserver {
  181. appDelegate.isMediaObserver = false
  182. NCViewerMedia.sharedInstance.removeObserver()
  183. }
  184. // IMAGE
  185. if metadata.typeFile == k_metadataTypeFile_image {
  186. viewImage()
  187. return
  188. }
  189. // AUDIO VIDEO
  190. if metadata.typeFile == k_metadataTypeFile_audio || metadata.typeFile == k_metadataTypeFile_video {
  191. NCViewerMedia.sharedInstance.viewMedia(metadata, view: backgroundView)
  192. return
  193. }
  194. // DOCUMENT - INTERNAL VIEWER
  195. if metadata.typeFile == k_metadataTypeFile_document && selector != nil && selector == selectorLoadFileInternalView {
  196. NCViewerDocumentWeb.sharedInstance.viewDocumentWebAt(metadata, view: backgroundView)
  197. return
  198. }
  199. // DOCUMENT
  200. if metadata.typeFile == k_metadataTypeFile_document {
  201. // PDF
  202. if metadata.contentType == "application/pdf" {
  203. if #available(iOS 11.0, *) {
  204. let viewerPDF = NCViewerPDF.init(frame: backgroundView.frame)
  205. let filePath = CCUtility.getDirectoryProviderStorageOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
  206. if CCUtility.fileProviderStorageExists(metadata.ocId, fileNameView: metadata.fileNameView) == false {
  207. return
  208. }
  209. viewerPDF.setupPdfView(filePath: URL(fileURLWithPath: filePath), detailViewController: self)
  210. }
  211. return
  212. }
  213. // DirectEditinf: Nextcloud Text - OnlyOffice
  214. if NCUtility.sharedInstance.isDirectEditing(metadata) != nil && appDelegate.reachability.isReachable() {
  215. let editor = NCUtility.sharedInstance.isDirectEditing(metadata)!
  216. if editor == k_editor_text || editor == k_editor_onlyoffice {
  217. NCUtility.sharedInstance.startActivityIndicator(view: backgroundView, bottom: 0)
  218. if metadata.url == "" {
  219. var customUserAgent: String?
  220. let fileNamePath = CCUtility.returnFileNamePath(fromFileName: metadata.fileName, serverUrl: metadata.serverUrl, activeUrl: appDelegate.activeUrl)!
  221. if editor == k_editor_onlyoffice {
  222. customUserAgent = NCUtility.sharedInstance.getCustomUserAgentOnlyOffice()
  223. }
  224. NCCommunication.sharedInstance.NCTextOpenFile(urlString: appDelegate.activeUrl, fileNamePath: fileNamePath, editor: editor, customUserAgent: customUserAgent, account: appDelegate.activeAccount) { (account, url, errorCode, errorMessage) in
  225. if errorCode == 0 && account == self.appDelegate.activeAccount && url != nil {
  226. let nextcloudText = NCViewerNextcloudText.init(frame: self.backgroundView.frame, configuration: WKWebViewConfiguration())
  227. nextcloudText.viewerAt(url!, metadata: metadata, editor: editor, view: self.backgroundView, viewController: self)
  228. if editor == k_editor_text && self.splitViewController!.isCollapsed {
  229. self.navigationController?.navigationItem.hidesBackButton = true
  230. }
  231. } else if errorCode != 0 {
  232. NCContentPresenter.shared.messageNotification("_error_", description: errorMessage, delay: TimeInterval(k_dismissAfterSecond), type: NCContentPresenter.messageType.error, errorCode: errorCode)
  233. self.navigationController?.popViewController(animated: true)
  234. } else {
  235. self.navigationController?.popViewController(animated: true)
  236. }
  237. }
  238. } else {
  239. let nextcloudText = NCViewerNextcloudText.init(frame: backgroundView.frame, configuration: WKWebViewConfiguration())
  240. nextcloudText.viewerAt(metadata.url, metadata: metadata, editor: editor, view: backgroundView, viewController: self)
  241. if editor == k_editor_text && self.splitViewController!.isCollapsed {
  242. self.navigationController?.navigationItem.hidesBackButton = true
  243. }
  244. }
  245. }
  246. return
  247. }
  248. // RichDocument: Collabora
  249. if NCUtility.sharedInstance.isRichDocument(metadata) && appDelegate.reachability.isReachable() {
  250. NCUtility.sharedInstance.startActivityIndicator(view: backgroundView, bottom: 0)
  251. if metadata.url == "" {
  252. OCNetworking.sharedManager()?.createLinkRichdocuments(withAccount: appDelegate.activeAccount, fileId: metadata.fileId, completion: { (account, url, errorMessage, errorCode) in
  253. if errorCode == 0 && account == self.appDelegate.activeAccount && url != nil {
  254. let richDocument = NCViewerRichdocument.init(frame: self.backgroundView.frame, configuration: WKWebViewConfiguration())
  255. richDocument.viewRichDocumentAt(url!, metadata: metadata, view: self.backgroundView, viewController: self)
  256. if self.splitViewController != nil && self.splitViewController!.isCollapsed {
  257. self.navigationController?.navigationItem.hidesBackButton = true
  258. }
  259. } else if errorCode != 0 {
  260. NCContentPresenter.shared.messageNotification("_error_", description: errorMessage, delay: TimeInterval(k_dismissAfterSecond), type: NCContentPresenter.messageType.error, errorCode: errorCode)
  261. self.navigationController?.popViewController(animated: true)
  262. } else {
  263. self.navigationController?.popViewController(animated: true)
  264. }
  265. })
  266. } else {
  267. let richDocument = NCViewerRichdocument.init(frame: backgroundView.frame, configuration: WKWebViewConfiguration())
  268. richDocument.viewRichDocumentAt(metadata.url, metadata: metadata, view: backgroundView, viewController: self)
  269. if self.splitViewController != nil && self.splitViewController!.isCollapsed {
  270. self.navigationController?.navigationItem.hidesBackButton = true
  271. }
  272. }
  273. }
  274. }
  275. // OTHER
  276. NCViewerDocumentWeb.sharedInstance.viewDocumentWebAt(metadata, view: backgroundView)
  277. }
  278. }
  279. //MARK: - MediaBrowser - Delegate/DataSource
  280. extension NCDetailViewController: MediaBrowserViewControllerDelegate, MediaBrowserViewControllerDataSource {
  281. func viewImage() {
  282. for view in backgroundView.subviews { view.removeFromSuperview() }
  283. if let metadatas = getMetadatasMediaBrowser() {
  284. var counter = 0
  285. for metadata in metadatas {
  286. if metadata.ocId == self.metadata!.ocId { mediaBrowserIndexStart = counter }
  287. counter += 1
  288. }
  289. self.metadatas = metadatas
  290. mediaBrowser = MediaBrowserViewController(index: mediaBrowserIndexStart, dataSource: self, delegate: self)
  291. if mediaBrowser != nil {
  292. self.backgroundView.image = nil
  293. mediaBrowser!.view.isHidden = true
  294. mediaBrowser!.enableInteractiveDismissal = true
  295. addChild(mediaBrowser!)
  296. backgroundView.addSubview(mediaBrowser!.view)
  297. mediaBrowser!.view.frame = CGRect(x: 0, y: 0, width: backgroundView.frame.width, height: backgroundView.frame.height)
  298. mediaBrowser!.view.autoresizingMask = [.flexibleWidth, .flexibleHeight]
  299. mediaBrowser!.didMove(toParent: self)
  300. DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
  301. self.mediaBrowser!.changeInViewSize(to: self.backgroundView.frame.size)
  302. self.mediaBrowser!.view.isHidden = false
  303. }
  304. }
  305. }
  306. }
  307. func numberOfItems(in mediaBrowser: MediaBrowserViewController) -> Int {
  308. return metadatas.count
  309. }
  310. func mediaBrowser(_ mediaBrowser: MediaBrowserViewController, imageAt index: Int, completion: @escaping MediaBrowserViewControllerDataSource.CompletionBlock) {
  311. if index >= metadatas.count { return }
  312. let metadata = metadatas[index]
  313. // Refresh self metadata && title
  314. if mediaBrowser.index < metadatas.count {
  315. self.metadata = metadatas[mediaBrowser.index]
  316. self.navigationController?.navigationBar.topItem?.title = self.metadata!.fileNameView
  317. }
  318. // Original only for the first
  319. if CCUtility.fileProviderStorageSize(metadata.ocId, fileNameView: metadata.fileNameView) > 0 && index == mediaBrowserIndexStart {
  320. var image: UIImage?
  321. let imagePath = CCUtility.getDirectoryProviderStorageOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
  322. let ext = CCUtility.getExtension(metadata.fileNameView)
  323. if ext == "GIF" { image = UIImage.animatedImage(withAnimatedGIFURL: URL(fileURLWithPath: imagePath)) }
  324. else { image = UIImage.init(contentsOfFile: imagePath) }
  325. if let image = image {
  326. completion(index, image, ZoomScale.default, nil)
  327. } else {
  328. completion(index, self.getImageOffOutline(), ZoomScale.default, nil)
  329. }
  330. // Preview
  331. } else if CCUtility.fileProviderStorageIconExists(metadata.ocId, fileNameView: metadata.fileNameView) {
  332. let imagePath = CCUtility.getDirectoryProviderStorageIconOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
  333. if let image = UIImage.init(contentsOfFile: imagePath) {
  334. completion(index, image, ZoomScale.default, nil)
  335. } else {
  336. completion(index, self.getImageOffOutline(), ZoomScale.default, nil)
  337. }
  338. } else {
  339. let fileNamePath = CCUtility.returnFileNamePath(fromFileName: metadata.fileName, serverUrl: metadata.serverUrl, activeUrl: appDelegate.activeUrl)!
  340. let fileNameLocalPath = CCUtility.getDirectoryProviderStorageIconOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
  341. NCCommunication.sharedInstance.downloadPreview(serverUrl: appDelegate.activeUrl, fileNamePath: fileNamePath, fileNameLocalPath: fileNameLocalPath, width: NCUtility.sharedInstance.getScreenWidthForPreview(), height: NCUtility.sharedInstance.getScreenHeightForPreview(), account: metadata.account) { (account, data, errorCode, errorMessage) in
  342. if errorCode == 0 && data != nil {
  343. do {
  344. let url = URL.init(fileURLWithPath: fileNameLocalPath)
  345. try data!.write(to: url, options: .atomic)
  346. completion(index, UIImage.init(data: data!), ZoomScale.default, nil)
  347. } catch {
  348. completion(index, self.getImageOffOutline(), ZoomScale.default, nil)
  349. }
  350. } else {
  351. completion(index, self.getImageOffOutline(), ZoomScale.default, nil)
  352. }
  353. }
  354. }
  355. }
  356. func mediaBrowser(_ mediaBrowser: MediaBrowserViewController, didChangeFocusTo index: Int, view: MediaContentView) {
  357. if index >= metadatas.count { return }
  358. let metadata = metadatas[index]
  359. DispatchQueue.global().async {
  360. if CCUtility.fileProviderStorageSize(metadata.ocId, fileNameView: metadata.fileNameView) > 0 {
  361. var image: UIImage?
  362. let imagePath = CCUtility.getDirectoryProviderStorageOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
  363. let ext = CCUtility.getExtension(metadata.fileNameView)
  364. if ext == "GIF" { image = UIImage.animatedImage(withAnimatedGIFURL: URL(fileURLWithPath: imagePath)) }
  365. else { image = UIImage.init(contentsOfFile: imagePath) }
  366. if let image = image {
  367. DispatchQueue.main.asyncAfter(deadline: .now() + .milliseconds(140)) {
  368. view.image = image
  369. }
  370. }
  371. }
  372. }
  373. }
  374. func mediaBrowserTap(_ mediaBrowser: MediaBrowserViewController) {
  375. guard let navigationController = self.navigationController else { return }
  376. if navigationController.isNavigationBarHidden {
  377. navigationController.isNavigationBarHidden = false
  378. view.backgroundColor = NCBrandColor.sharedInstance.backgroundView
  379. } else {
  380. navigationController.isNavigationBarHidden = true
  381. view.backgroundColor = .black
  382. }
  383. mediaBrowser.changeInViewSize(to: self.backgroundView.frame.size)
  384. }
  385. func mediaBrowserDismiss() {
  386. viewUnload()
  387. }
  388. func getMetadatasMediaBrowser() -> [tableMetadata]? {
  389. guard let metadata = self.metadata else { return nil }
  390. if favoriteFilterImage {
  391. return NCManageDatabase.sharedInstance.getMetadatas(predicate: NSPredicate(format: "account == %@ AND favorite == 1 AND typeFile == %@", metadata.account, k_metadataTypeFile_image), sorted: CCUtility.getOrderSettings(), ascending: CCUtility.getAscendingSettings())
  392. } else if mediaFilterImage {
  393. return NCManageDatabase.sharedInstance.getMedias(account: metadata.account, predicate: NSPredicate(format: "account == %@ AND typeFile == %@", metadata.account, k_metadataTypeFile_image))
  394. } else if offlineFilterImage {
  395. var datasourceSorted = ""
  396. var datasourceAscending = true
  397. (_, datasourceSorted, datasourceAscending, _, _) = NCUtility.sharedInstance.getLayoutForView(key: k_layout_view_offline)
  398. if let files = NCManageDatabase.sharedInstance.getTableLocalFiles(predicate: NSPredicate(format: "account == %@ AND offline == true", appDelegate.activeAccount), sorted: datasourceSorted, ascending: datasourceAscending) {
  399. var ocIds = [String]()
  400. for file: tableLocalFile in files {
  401. ocIds.append(file.ocId)
  402. }
  403. return NCManageDatabase.sharedInstance.getMetadatas(predicate: NSPredicate(format: "account == %@ AND ocId IN %@", appDelegate.activeAccount, ocIds), sorted: datasourceSorted, ascending: datasourceAscending)
  404. }
  405. } else {
  406. return NCManageDatabase.sharedInstance.getMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@ AND typeFile == %@", metadata.account, metadata.serverUrl, k_metadataTypeFile_image), sorted: CCUtility.getOrderSettings(), ascending: CCUtility.getAscendingSettings())
  407. }
  408. return nil
  409. }
  410. func getImageOffOutline() -> UIImage {
  411. let image = CCGraphics.changeThemingColorImage(UIImage.init(named: "imageOffOutline"), width: self.view.frame.width, height: self.view.frame.width, color: NCBrandColor.sharedInstance.brand)
  412. return image!
  413. }
  414. }