NCUtility.swift 45 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020
  1. //
  2. // NCUtility.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 25/06/18.
  6. // Copyright © 2018 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 UIKit
  24. import SVGKit
  25. import KTVHTTPCache
  26. import NextcloudKit
  27. import PDFKit
  28. import Accelerate
  29. import CoreMedia
  30. import Queuer
  31. import Photos
  32. import JGProgressHUD
  33. class NCUtility: NSObject {
  34. @objc static let shared: NCUtility = {
  35. let instance = NCUtility()
  36. return instance
  37. }()
  38. func convertSVGtoPNGWriteToUserData(svgUrlString: String, fileName: String?, width: CGFloat?, rewrite: Bool, account: String, closure: @escaping (String?) -> Void) {
  39. var fileNamePNG = ""
  40. guard let svgUrlString = svgUrlString.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed),
  41. let iconURL = URL(string: svgUrlString) else {
  42. return closure(nil)
  43. }
  44. if let fileName = fileName {
  45. fileNamePNG = fileName
  46. } else {
  47. fileNamePNG = iconURL.deletingPathExtension().lastPathComponent + ".png"
  48. }
  49. let imageNamePath = CCUtility.getDirectoryUserData() + "/" + fileNamePNG
  50. if !FileManager.default.fileExists(atPath: imageNamePath) || rewrite == true {
  51. NextcloudKit.shared.downloadContent(serverUrl: iconURL.absoluteString) { _, data, error in
  52. if error == .success && data != nil {
  53. if let image = UIImage(data: data!) {
  54. var newImage: UIImage = image
  55. if width != nil {
  56. let ratio = image.size.height / image.size.width
  57. let newSize = CGSize(width: width!, height: width! * ratio)
  58. let renderFormat = UIGraphicsImageRendererFormat.default()
  59. renderFormat.opaque = false
  60. let renderer = UIGraphicsImageRenderer(size: CGSize(width: newSize.width, height: newSize.height), format: renderFormat)
  61. newImage = renderer.image {
  62. _ in
  63. image.draw(in: CGRect(x: 0, y: 0, width: newSize.width, height: newSize.height))
  64. }
  65. }
  66. guard let pngImageData = newImage.pngData() else {
  67. return closure(nil)
  68. }
  69. try? pngImageData.write(to: URL(fileURLWithPath: imageNamePath))
  70. return closure(imageNamePath)
  71. } else {
  72. guard let svgImage: SVGKImage = SVGKImage(data: data) else {
  73. return closure(nil)
  74. }
  75. if width != nil {
  76. let scale = svgImage.size.height / svgImage.size.width
  77. svgImage.size = CGSize(width: width!, height: width! * scale)
  78. }
  79. guard let image: UIImage = svgImage.uiImage else {
  80. return closure(nil)
  81. }
  82. guard let pngImageData = image.pngData() else {
  83. return closure(nil)
  84. }
  85. try? pngImageData.write(to: URL(fileURLWithPath: imageNamePath))
  86. return closure(imageNamePath)
  87. }
  88. } else {
  89. return closure(nil)
  90. }
  91. }
  92. } else {
  93. return closure(imageNamePath)
  94. }
  95. }
  96. @objc func isSimulatorOrTestFlight() -> Bool {
  97. guard let path = Bundle.main.appStoreReceiptURL?.path else {
  98. return false
  99. }
  100. return path.contains("CoreSimulator") || path.contains("sandboxReceipt")
  101. }
  102. @objc func isSimulator() -> Bool {
  103. guard let path = Bundle.main.appStoreReceiptURL?.path else {
  104. return false
  105. }
  106. return path.contains("CoreSimulator")
  107. }
  108. @objc func isRichDocument(_ metadata: tableMetadata) -> Bool {
  109. guard let mimeType = CCUtility.getMimeType(metadata.fileNameView) else {
  110. return false
  111. }
  112. guard let richdocumentsMimetypes = NCManageDatabase.shared.getCapabilitiesServerArray(account: metadata.account, elements: NCElementsJSON.shared.capabilitiesRichdocumentsMimetypes) else {
  113. return false
  114. }
  115. // contentype
  116. for richdocumentMimetype: String in richdocumentsMimetypes {
  117. if richdocumentMimetype.contains(metadata.contentType) || metadata.contentType == "text/plain" {
  118. return true
  119. }
  120. }
  121. // mimetype
  122. if richdocumentsMimetypes.count > 0 && mimeType.components(separatedBy: ".").count > 2 {
  123. let mimeTypeArray = mimeType.components(separatedBy: ".")
  124. let mimeType = mimeTypeArray[mimeTypeArray.count - 2] + "." + mimeTypeArray[mimeTypeArray.count - 1]
  125. for richdocumentMimetype: String in richdocumentsMimetypes {
  126. if richdocumentMimetype.contains(mimeType) {
  127. return true
  128. }
  129. }
  130. }
  131. return false
  132. }
  133. @objc func isDirectEditing(account: String, contentType: String) -> [String] {
  134. var editor: [String] = []
  135. guard let results = NCManageDatabase.shared.getDirectEditingEditors(account: account) else {
  136. return editor
  137. }
  138. for result: tableDirectEditingEditors in results {
  139. for mimetype in result.mimetypes {
  140. if mimetype == contentType {
  141. editor.append(result.editor)
  142. }
  143. // HARDCODE
  144. // https://github.com/nextcloud/text/issues/913
  145. if mimetype == "text/markdown" && contentType == "text/x-markdown" {
  146. editor.append(result.editor)
  147. }
  148. if contentType == "text/html" {
  149. editor.append(result.editor)
  150. }
  151. }
  152. for mimetype in result.optionalMimetypes {
  153. if mimetype == contentType {
  154. editor.append(result.editor)
  155. }
  156. }
  157. }
  158. // HARDCODE
  159. // if editor.count == 0 {
  160. // editor.append(NCGlobal.shared.editorText)
  161. // }
  162. return Array(Set(editor))
  163. }
  164. @objc func removeAllSettings() {
  165. URLCache.shared.memoryCapacity = 0
  166. URLCache.shared.diskCapacity = 0
  167. KTVHTTPCache.cacheDeleteAllCaches()
  168. NCManageDatabase.shared.clearDatabase(account: nil, removeAccount: true)
  169. CCUtility.removeGroupDirectoryProviderStorage()
  170. CCUtility.removeGroupLibraryDirectory()
  171. CCUtility.removeDocumentsDirectory()
  172. CCUtility.removeTemporaryDirectory()
  173. CCUtility.createDirectoryStandard()
  174. CCUtility.deleteAllChainStore()
  175. }
  176. @objc func permissionsContainsString(_ metadataPermissions: String, permissions: String) -> Bool {
  177. for char in permissions {
  178. if metadataPermissions.contains(char) == false {
  179. return false
  180. }
  181. }
  182. return true
  183. }
  184. @objc func getCustomUserAgentNCText() -> String {
  185. let userAgent: String = CCUtility.getUserAgent()
  186. if UIDevice.current.userInterfaceIdiom == .phone {
  187. // NOTE: Hardcoded (May 2022)
  188. // Tested for iPhone SE (1st), iOS 12; iPhone Pro Max, iOS 15.4
  189. // 605.1.15 = WebKit build version
  190. // 15E148 = frozen iOS build number according to: https://chromestatus.com/feature/4558585463832576
  191. return userAgent + " " + "AppleWebKit/605.1.15 Mobile/15E148"
  192. } else {
  193. return userAgent
  194. }
  195. }
  196. @objc func getCustomUserAgentOnlyOffice() -> String {
  197. let appVersion = Bundle.main.object(forInfoDictionaryKey: "CFBundleShortVersionString")!
  198. if UIDevice.current.userInterfaceIdiom == .pad {
  199. return "Mozilla/5.0 (iPad) Nextcloud-iOS/\(appVersion)"
  200. } else {
  201. return "Mozilla/5.0 (iPhone) Mobile Nextcloud-iOS/\(appVersion)"
  202. }
  203. }
  204. @objc func pdfThumbnail(url: URL, width: CGFloat = 240) -> UIImage? {
  205. guard let data = try? Data(contentsOf: url), let page = PDFDocument(data: data)?.page(at: 0) else {
  206. return nil
  207. }
  208. let pageSize = page.bounds(for: .mediaBox)
  209. let pdfScale = width / pageSize.width
  210. // Apply if you're displaying the thumbnail on screen
  211. let scale = UIScreen.main.scale * pdfScale
  212. let screenSize = CGSize(width: pageSize.width * scale, height: pageSize.height * scale)
  213. return page.thumbnail(of: screenSize, for: .mediaBox)
  214. }
  215. @objc func isQuickLookDisplayable(metadata: tableMetadata) -> Bool {
  216. return true
  217. }
  218. @objc func ocIdToFileId(ocId: String?) -> String? {
  219. guard let ocId = ocId else { return nil }
  220. let items = ocId.components(separatedBy: "oc")
  221. if items.count < 2 { return nil }
  222. guard let intFileId = Int(items[0]) else { return nil }
  223. return String(intFileId)
  224. }
  225. func getUserStatus(userIcon: String?, userStatus: String?, userMessage: String?) -> (onlineStatus: UIImage?, statusMessage: String, descriptionMessage: String) {
  226. var onlineStatus: UIImage?
  227. var statusMessage: String = ""
  228. var descriptionMessage: String = ""
  229. var messageUserDefined: String = ""
  230. if userStatus?.lowercased() == "online" {
  231. onlineStatus = UIImage(named: "circle_fill")!.image(color: UIColor(red: 103.0/255.0, green: 176.0/255.0, blue: 134.0/255.0, alpha: 1.0), size: 50)
  232. messageUserDefined = NSLocalizedString("_online_", comment: "")
  233. }
  234. if userStatus?.lowercased() == "away" {
  235. onlineStatus = UIImage(named: "userStatusAway")!.image(color: UIColor(red: 233.0/255.0, green: 166.0/255.0, blue: 75.0/255.0, alpha: 1.0), size: 50)
  236. messageUserDefined = NSLocalizedString("_away_", comment: "")
  237. }
  238. if userStatus?.lowercased() == "dnd" {
  239. onlineStatus = UIImage(named: "userStatusDnd")?.resizeImage(size: CGSize(width: 100, height: 100), isAspectRation: false)
  240. messageUserDefined = NSLocalizedString("_dnd_", comment: "")
  241. descriptionMessage = NSLocalizedString("_dnd_description_", comment: "")
  242. }
  243. if userStatus?.lowercased() == "offline" || userStatus?.lowercased() == "invisible" {
  244. onlineStatus = UIImage(named: "userStatusOffline")!.image(color: .black, size: 50)
  245. messageUserDefined = NSLocalizedString("_invisible_", comment: "")
  246. descriptionMessage = NSLocalizedString("_invisible_description_", comment: "")
  247. }
  248. if let userIcon = userIcon {
  249. statusMessage = userIcon + " "
  250. }
  251. if let userMessage = userMessage {
  252. statusMessage += userMessage
  253. }
  254. statusMessage = statusMessage.trimmingCharacters(in: .whitespaces)
  255. if statusMessage == "" {
  256. statusMessage = messageUserDefined
  257. }
  258. return(onlineStatus, statusMessage, descriptionMessage)
  259. }
  260. // MARK: -
  261. func extractFiles(from metadata: tableMetadata, viewController: UIViewController?, hud: JGProgressHUD, completition: @escaping (_ metadatas: [tableMetadata]) -> Void) {
  262. let chunckSize = CCUtility.getChunkSize() * 1000000
  263. var metadatas: [tableMetadata] = []
  264. let metadataSource = tableMetadata.init(value: metadata)
  265. guard !metadata.isExtractFile else { return completition([metadataSource]) }
  266. guard !metadataSource.assetLocalIdentifier.isEmpty else {
  267. let filePath = CCUtility.getDirectoryProviderStorageOcId(metadataSource.ocId, fileNameView: metadataSource.fileName)!
  268. metadataSource.size = NCUtilityFileSystem.shared.getFileSize(filePath: filePath)
  269. let results = NKCommon.shared.getInternalType(fileName: metadataSource.fileNameView, mimeType: metadataSource.contentType, directory: false)
  270. metadataSource.contentType = results.mimeType
  271. metadataSource.iconName = results.iconName
  272. metadataSource.classFile = results.classFile
  273. if let date = NCUtilityFileSystem.shared.getFileCreationDate(filePath: filePath) {
  274. metadataSource.creationDate = date
  275. }
  276. if let date = NCUtilityFileSystem.shared.getFileModificationDate(filePath: filePath) {
  277. metadataSource.date = date
  278. }
  279. metadataSource.chunk = chunckSize != 0 && metadata.size > chunckSize
  280. metadataSource.e2eEncrypted = NCUtility.shared.isDirectoryE2EE(metadata: metadata)
  281. metadataSource.isExtractFile = true
  282. if let metadata = NCManageDatabase.shared.addMetadata(metadataSource) {
  283. metadatas.append(metadata)
  284. }
  285. return completition(metadatas)
  286. }
  287. extractImageVideoFromAssetLocalIdentifier(metadata: metadataSource, modifyMetadataForUpload: true, viewController: viewController, hud: hud) { metadata, fileNamePath, returnError in
  288. if let metadata = metadata, let fileNamePath = fileNamePath, !returnError {
  289. metadatas.append(metadata)
  290. let toPath = CCUtility.getDirectoryProviderStorageOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
  291. NCUtilityFileSystem.shared.moveFile(atPath: fileNamePath, toPath: toPath)
  292. } else {
  293. return completition(metadatas)
  294. }
  295. let fetchAssets = PHAsset.fetchAssets(withLocalIdentifiers: [metadataSource.assetLocalIdentifier], options: nil)
  296. if metadataSource.livePhoto, fetchAssets.count > 0 {
  297. NCUtility.shared.createMetadataLivePhotoFromMetadata(metadataSource, asset: fetchAssets.firstObject) { metadata in
  298. if let metadata = metadata, let metadata = NCManageDatabase.shared.addMetadata(metadata) {
  299. metadatas.append(metadata)
  300. }
  301. completition(metadatas)
  302. }
  303. } else {
  304. completition(metadatas)
  305. }
  306. }
  307. }
  308. func extractImageVideoFromAssetLocalIdentifier(metadata: tableMetadata, modifyMetadataForUpload: Bool, viewController: UIViewController?, hud: JGProgressHUD, completion: @escaping (_ metadata: tableMetadata?, _ fileNamePath: String?, _ error: Bool) -> ()) {
  309. var fileNamePath: String?
  310. let metadata = tableMetadata.init(value: metadata)
  311. let chunckSize = CCUtility.getChunkSize() * 1000000
  312. var compatibilityFormat: Bool = false
  313. let isDirectoryE2EE = NCUtility.shared.isDirectoryE2EE(metadata: metadata)
  314. func callCompletionWithError(_ error: Bool = true) {
  315. if error {
  316. completion(nil, nil, true)
  317. } else {
  318. var metadataReturn = metadata
  319. if modifyMetadataForUpload {
  320. metadata.chunk = chunckSize != 0 && metadata.size > chunckSize
  321. //metadata.e2eEncrypted = NCUtility.shared.isFolderEncrypted(serverUrl: metadata.serverUrl, e2eEncrypted: metadata.e2eEncrypted, account: metadata.account, urlBase: metadata.urlBase, userId: metadata.userId)
  322. metadata.isExtractFile = true
  323. if let metadata = NCManageDatabase.shared.addMetadata(metadata) {
  324. metadataReturn = metadata
  325. }
  326. }
  327. completion(metadataReturn, fileNamePath, error)
  328. }
  329. }
  330. let fetchAssets = PHAsset.fetchAssets(withLocalIdentifiers: [metadata.assetLocalIdentifier], options: nil)
  331. guard fetchAssets.count > 0, let asset = fetchAssets.firstObject, let extensionAsset = (asset.value(forKey: "filename") as? NSString)?.pathExtension.uppercased() else { return callCompletionWithError() }
  332. let creationDate = asset.creationDate ?? Date()
  333. let modificationDate = asset.modificationDate ?? Date()
  334. if asset.mediaType == PHAssetMediaType.image && (extensionAsset == "HEIC" || extensionAsset == "DNG") && CCUtility.getFormatCompatibility() {
  335. let fileName = (metadata.fileNameView as NSString).deletingPathExtension + ".jpg"
  336. metadata.contentType = "image/jpeg"
  337. fileNamePath = NSTemporaryDirectory() + fileName
  338. metadata.fileNameView = fileName
  339. if !isDirectoryE2EE {
  340. metadata.fileName = fileName
  341. }
  342. compatibilityFormat = true
  343. } else {
  344. fileNamePath = NSTemporaryDirectory() + metadata.fileNameView
  345. }
  346. guard let fileNamePath = fileNamePath else { return callCompletionWithError() }
  347. if asset.mediaType == PHAssetMediaType.image {
  348. let options = PHImageRequestOptions()
  349. options.isNetworkAccessAllowed = true
  350. if compatibilityFormat {
  351. options.deliveryMode = .opportunistic
  352. } else {
  353. options.deliveryMode = .highQualityFormat
  354. }
  355. options.isSynchronous = true
  356. if extensionAsset == "DNG" {
  357. options.version = PHImageRequestOptionsVersion.original
  358. }
  359. options.progressHandler = { (progress, error, stop, info) in
  360. print(progress)
  361. if error != nil { return callCompletionWithError() }
  362. }
  363. PHImageManager.default().requestImageDataAndOrientation(for: asset, options: options) { data, dataUI, orientation, info in
  364. guard var data = data else { return callCompletionWithError() }
  365. if compatibilityFormat {
  366. guard let ciImage = CIImage.init(data: data), let colorSpace = ciImage.colorSpace, let dataJPEG = CIContext().jpegRepresentation(of: ciImage, colorSpace: colorSpace) else { return callCompletionWithError() }
  367. data = dataJPEG
  368. }
  369. NCUtilityFileSystem.shared.deleteFile(filePath: fileNamePath)
  370. do {
  371. try data.write(to: URL(fileURLWithPath: fileNamePath), options: .atomic)
  372. } catch { return callCompletionWithError() }
  373. metadata.creationDate = creationDate as NSDate
  374. metadata.date = modificationDate as NSDate
  375. metadata.size = NCUtilityFileSystem.shared.getFileSize(filePath: fileNamePath)
  376. return callCompletionWithError(false)
  377. }
  378. } else if asset.mediaType == PHAssetMediaType.video {
  379. let options = PHVideoRequestOptions()
  380. options.isNetworkAccessAllowed = true
  381. options.version = PHVideoRequestOptionsVersion.current
  382. options.progressHandler = { (progress, error, stop, info) in
  383. print(progress)
  384. if error != nil { return callCompletionWithError() }
  385. }
  386. PHImageManager.default().requestAVAsset(forVideo: asset, options: options) { asset, audioMix, info in
  387. if let asset = asset as? AVURLAsset {
  388. NCUtilityFileSystem.shared.deleteFile(filePath: fileNamePath)
  389. do {
  390. try FileManager.default.copyItem(at: asset.url, to: URL(fileURLWithPath: fileNamePath))
  391. metadata.creationDate = creationDate as NSDate
  392. metadata.date = modificationDate as NSDate
  393. metadata.size = NCUtilityFileSystem.shared.getFileSize(filePath: fileNamePath)
  394. return callCompletionWithError(false)
  395. } catch { return callCompletionWithError() }
  396. } else if let asset = asset as? AVComposition, asset.tracks.count > 1, let exporter = AVAssetExportSession(asset: asset, presetName: AVAssetExportPresetHighestQuality), let viewController = viewController {
  397. DispatchQueue.main.async {
  398. hud.indicatorView = JGProgressHUDRingIndicatorView()
  399. if let indicatorView = hud.indicatorView as? JGProgressHUDRingIndicatorView {
  400. indicatorView.ringWidth = 1.5
  401. }
  402. hud.textLabel.text = NSLocalizedString("_exporting_video_", comment: "")
  403. hud.show(in: viewController.view)
  404. hud.tapOnHUDViewBlock = { hud in
  405. exporter.cancelExport()
  406. }
  407. }
  408. exporter.outputURL = URL(fileURLWithPath: fileNamePath)
  409. exporter.outputFileType = AVFileType.mp4
  410. exporter.shouldOptimizeForNetworkUse = true
  411. exporter.exportAsynchronously {
  412. DispatchQueue.main.async { hud.dismiss() }
  413. if exporter.status == .completed {
  414. metadata.creationDate = creationDate as NSDate
  415. metadata.date = modificationDate as NSDate
  416. metadata.size = NCUtilityFileSystem.shared.getFileSize(filePath: fileNamePath)
  417. return callCompletionWithError(false)
  418. } else { return callCompletionWithError() }
  419. }
  420. while exporter.status == AVAssetExportSession.Status.exporting || exporter.status == AVAssetExportSession.Status.waiting {
  421. hud.progress = exporter.progress
  422. }
  423. } else {
  424. return callCompletionWithError()
  425. }
  426. }
  427. } else {
  428. return callCompletionWithError()
  429. }
  430. }
  431. func createMetadataLivePhotoFromMetadata(_ metadata: tableMetadata, asset: PHAsset?, completion: @escaping (_ metadata: tableMetadata?) -> ()) {
  432. guard let asset = asset else { return completion(nil) }
  433. let options = PHLivePhotoRequestOptions()
  434. options.deliveryMode = PHImageRequestOptionsDeliveryMode.fastFormat
  435. options.isNetworkAccessAllowed = true
  436. let chunckSize = CCUtility.getChunkSize() * 1000000
  437. let e2eEncrypted = NCUtility.shared.isDirectoryE2EE(metadata: metadata)
  438. let ocId = NSUUID().uuidString
  439. let fileName = (metadata.fileName as NSString).deletingPathExtension + ".mov"
  440. let fileNamePath = CCUtility.getDirectoryProviderStorageOcId(ocId, fileNameView: fileName)!
  441. PHImageManager.default().requestLivePhoto(for: asset, targetSize: UIScreen.main.bounds.size, contentMode: PHImageContentMode.default, options: options) { livePhoto, info in
  442. guard let livePhoto = livePhoto else { return completion(nil) }
  443. var videoResource: PHAssetResource?
  444. for resource in PHAssetResource.assetResources(for: livePhoto) {
  445. if resource.type == PHAssetResourceType.pairedVideo {
  446. videoResource = resource
  447. break
  448. }
  449. }
  450. guard let videoResource = videoResource else { return completion(nil) }
  451. NCUtilityFileSystem.shared.deleteFile(filePath: fileNamePath)
  452. PHAssetResourceManager.default().writeData(for: videoResource, toFile: URL(fileURLWithPath: fileNamePath), options: nil) { error in
  453. if error != nil { return completion(nil) }
  454. let metadataLivePhoto = NCManageDatabase.shared.createMetadata(account: metadata.account, user: metadata.user, userId: metadata.userId, fileName: fileName, fileNameView: fileName, ocId: ocId, serverUrl: metadata.serverUrl, urlBase: metadata.urlBase, url: "", contentType: "", isLivePhoto: true)
  455. metadataLivePhoto.classFile = NKCommon.typeClassFile.video.rawValue
  456. metadataLivePhoto.e2eEncrypted = e2eEncrypted
  457. metadataLivePhoto.isExtractFile = true
  458. metadataLivePhoto.session = metadata.session
  459. metadataLivePhoto.sessionSelector = metadata.sessionSelector
  460. metadataLivePhoto.size = NCUtilityFileSystem.shared.getFileSize(filePath: fileNamePath)
  461. metadataLivePhoto.status = metadata.status
  462. metadataLivePhoto.chunk = chunckSize != 0 && metadata.size > chunckSize
  463. return completion(NCManageDatabase.shared.addMetadata(metadataLivePhoto))
  464. }
  465. }
  466. }
  467. func imageFromVideo(url: URL, at time: TimeInterval) -> UIImage? {
  468. let asset = AVURLAsset(url: url)
  469. let assetIG = AVAssetImageGenerator(asset: asset)
  470. assetIG.appliesPreferredTrackTransform = true
  471. assetIG.apertureMode = AVAssetImageGenerator.ApertureMode.encodedPixels
  472. let cmTime = CMTime(seconds: time, preferredTimescale: 60)
  473. let thumbnailImageRef: CGImage
  474. do {
  475. thumbnailImageRef = try assetIG.copyCGImage(at: cmTime, actualTime: nil)
  476. } catch let error {
  477. print("Error: \(error)")
  478. return nil
  479. }
  480. return UIImage(cgImage: thumbnailImageRef)
  481. }
  482. func imageFromVideo(url: URL, at time: TimeInterval, completion: @escaping (UIImage?) -> Void) {
  483. DispatchQueue.global().async {
  484. let asset = AVURLAsset(url: url)
  485. let assetIG = AVAssetImageGenerator(asset: asset)
  486. assetIG.appliesPreferredTrackTransform = true
  487. assetIG.apertureMode = AVAssetImageGenerator.ApertureMode.encodedPixels
  488. let cmTime = CMTime(seconds: time, preferredTimescale: 60)
  489. let thumbnailImageRef: CGImage
  490. do {
  491. thumbnailImageRef = try assetIG.copyCGImage(at: cmTime, actualTime: nil)
  492. } catch let error {
  493. print("Error: \(error)")
  494. return completion(nil)
  495. }
  496. DispatchQueue.main.async {
  497. completion(UIImage(cgImage: thumbnailImageRef))
  498. }
  499. }
  500. }
  501. func createImageFrom(fileNameView: String, ocId: String, etag: String, classFile: String) {
  502. var originalImage, scaleImagePreview, scaleImageIcon: UIImage?
  503. let fileNamePath = CCUtility.getDirectoryProviderStorageOcId(ocId, fileNameView: fileNameView)!
  504. let fileNamePathPreview = CCUtility.getDirectoryProviderStoragePreviewOcId(ocId, etag: etag)!
  505. let fileNamePathIcon = CCUtility.getDirectoryProviderStorageIconOcId(ocId, etag: etag)!
  506. if CCUtility.fileProviderStorageSize(ocId, fileNameView: fileNameView) > 0 && FileManager().fileExists(atPath: fileNamePathPreview) && FileManager().fileExists(atPath: fileNamePathIcon) { return }
  507. if classFile != NKCommon.typeClassFile.image.rawValue && classFile != NKCommon.typeClassFile.video.rawValue { return }
  508. if classFile == NKCommon.typeClassFile.image.rawValue {
  509. originalImage = UIImage(contentsOfFile: fileNamePath)
  510. scaleImagePreview = originalImage?.resizeImage(size: CGSize(width: NCGlobal.shared.sizePreview, height: NCGlobal.shared.sizePreview))
  511. scaleImageIcon = originalImage?.resizeImage(size: CGSize(width: NCGlobal.shared.sizeIcon, height: NCGlobal.shared.sizeIcon))
  512. try? scaleImagePreview?.jpegData(compressionQuality: 0.7)?.write(to: URL(fileURLWithPath: fileNamePathPreview))
  513. try? scaleImageIcon?.jpegData(compressionQuality: 0.7)?.write(to: URL(fileURLWithPath: fileNamePathIcon))
  514. } else if classFile == NKCommon.typeClassFile.video.rawValue {
  515. let videoPath = NSTemporaryDirectory()+"tempvideo.mp4"
  516. NCUtilityFileSystem.shared.linkItem(atPath: fileNamePath, toPath: videoPath)
  517. originalImage = imageFromVideo(url: URL(fileURLWithPath: videoPath), at: 0)
  518. try? originalImage?.jpegData(compressionQuality: 0.7)?.write(to: URL(fileURLWithPath: fileNamePathPreview))
  519. try? originalImage?.jpegData(compressionQuality: 0.7)?.write(to: URL(fileURLWithPath: fileNamePathIcon))
  520. }
  521. }
  522. @objc func getVersionApp(withBuild: Bool = true) -> String {
  523. if let dictionary = Bundle.main.infoDictionary {
  524. if let version = dictionary["CFBundleShortVersionString"], let build = dictionary["CFBundleVersion"] {
  525. if withBuild {
  526. return "\(version).\(build)"
  527. } else {
  528. return "\(version)"
  529. }
  530. }
  531. }
  532. return ""
  533. }
  534. func loadImage(named imageName: String, color: UIColor = NCBrandColor.shared.gray, size: CGFloat = 50, symbolConfiguration: Any? = nil) -> UIImage {
  535. var image: UIImage?
  536. // see https://stackoverflow.com/questions/71764255
  537. let sfSymbolName = imageName.replacingOccurrences(of: "_", with: ".")
  538. if let symbolConfiguration = symbolConfiguration {
  539. image = UIImage(systemName: sfSymbolName, withConfiguration: symbolConfiguration as? UIImage.Configuration)?.withTintColor(color, renderingMode: .alwaysOriginal)
  540. } else {
  541. image = UIImage(systemName: sfSymbolName)?.withTintColor(color, renderingMode: .alwaysOriginal)
  542. }
  543. if image == nil {
  544. image = UIImage(named: imageName)?.image(color: color, size: size)
  545. }
  546. if let image = image {
  547. return image
  548. }
  549. return UIImage(named: "file")!.image(color: color, size: size)
  550. }
  551. @objc func loadUserImage(for user: String, displayName: String?, userBaseUrl: NCUserBaseUrl) -> UIImage {
  552. let fileName = userBaseUrl.userBaseUrl + "-" + user + ".png"
  553. let localFilePath = String(CCUtility.getDirectoryUserData()) + "/" + fileName
  554. if let localImage = UIImage(contentsOfFile: localFilePath) {
  555. return createAvatar(image: localImage, size: 30)
  556. } else if let loadedAvatar = NCManageDatabase.shared.getImageAvatarLoaded(fileName: fileName) {
  557. return loadedAvatar
  558. } else if let displayName = displayName, !displayName.isEmpty, let avatarImg = createAvatar(displayName: displayName, size: 30) {
  559. return avatarImg
  560. } else { return getDefaultUserIcon() }
  561. }
  562. func getDefaultUserIcon() -> UIImage {
  563. let config = UIImage.SymbolConfiguration(pointSize: 30)
  564. return NCUtility.shared.loadImage(named: "person.crop.circle", symbolConfiguration: config)
  565. }
  566. @objc func createAvatar(image: UIImage, size: CGFloat) -> UIImage {
  567. var avatarImage = image
  568. let rect = CGRect(x: 0, y: 0, width: size, height: size)
  569. UIGraphicsBeginImageContextWithOptions(rect.size, false, 3.0)
  570. UIBezierPath(roundedRect: rect, cornerRadius: rect.size.height).addClip()
  571. avatarImage.draw(in: rect)
  572. avatarImage = UIGraphicsGetImageFromCurrentImageContext() ?? image
  573. UIGraphicsEndImageContext()
  574. return avatarImage
  575. }
  576. func createAvatar(displayName: String, size: CGFloat) -> UIImage? {
  577. guard let initials = displayName.uppercaseInitials else {
  578. return nil
  579. }
  580. let userColor = NCGlobal.shared.usernameToColor(displayName)
  581. let rect = CGRect(x: 0, y: 0, width: size, height: size)
  582. var avatarImage: UIImage?
  583. UIGraphicsBeginImageContextWithOptions(rect.size, false, 3.0)
  584. let context = UIGraphicsGetCurrentContext()
  585. UIBezierPath(roundedRect: rect, cornerRadius: rect.size.height).addClip()
  586. context?.setFillColor(userColor)
  587. context?.fill(rect)
  588. let textStyle = NSMutableParagraphStyle()
  589. textStyle.alignment = NSTextAlignment.center
  590. let lineHeight = UIFont.systemFont(ofSize: UIFont.systemFontSize).pointSize
  591. NSString(string: initials)
  592. .draw(
  593. in: CGRect(x: 0, y: (size - lineHeight) / 2, width: size, height: lineHeight),
  594. withAttributes: [NSAttributedString.Key.paragraphStyle: textStyle])
  595. avatarImage = UIGraphicsGetImageFromCurrentImageContext()
  596. UIGraphicsEndImageContext()
  597. return avatarImage
  598. }
  599. /*
  600. Facebook's comparison algorithm:
  601. */
  602. func compare(tolerance: Float, expected: Data, observed: Data) throws -> Bool {
  603. enum customError: Error {
  604. case unableToGetUIImageFromData
  605. case unableToGetCGImageFromData
  606. case unableToGetColorSpaceFromCGImage
  607. case imagesHasDifferentSizes
  608. case unableToInitializeContext
  609. }
  610. guard let expectedUIImage = UIImage(data: expected), let observedUIImage = UIImage(data: observed) else {
  611. throw customError.unableToGetUIImageFromData
  612. }
  613. guard let expectedCGImage = expectedUIImage.cgImage, let observedCGImage = observedUIImage.cgImage else {
  614. throw customError.unableToGetCGImageFromData
  615. }
  616. guard let expectedColorSpace = expectedCGImage.colorSpace, let observedColorSpace = observedCGImage.colorSpace else {
  617. throw customError.unableToGetColorSpaceFromCGImage
  618. }
  619. if expectedCGImage.width != observedCGImage.width || expectedCGImage.height != observedCGImage.height {
  620. throw customError.imagesHasDifferentSizes
  621. }
  622. let imageSize = CGSize(width: expectedCGImage.width, height: expectedCGImage.height)
  623. let numberOfPixels = Int(imageSize.width * imageSize.height)
  624. // Checking that our `UInt32` buffer has same number of bytes as image has.
  625. let bytesPerRow = min(expectedCGImage.bytesPerRow, observedCGImage.bytesPerRow)
  626. assert(MemoryLayout<UInt32>.stride == bytesPerRow / Int(imageSize.width))
  627. let expectedPixels = UnsafeMutablePointer<UInt32>.allocate(capacity: numberOfPixels)
  628. let observedPixels = UnsafeMutablePointer<UInt32>.allocate(capacity: numberOfPixels)
  629. let expectedPixelsRaw = UnsafeMutableRawPointer(expectedPixels)
  630. let observedPixelsRaw = UnsafeMutableRawPointer(observedPixels)
  631. let bitmapInfo = CGBitmapInfo(rawValue: CGImageAlphaInfo.premultipliedLast.rawValue)
  632. guard let expectedContext = CGContext(data: expectedPixelsRaw, width: Int(imageSize.width), height: Int(imageSize.height),
  633. bitsPerComponent: expectedCGImage.bitsPerComponent, bytesPerRow: bytesPerRow,
  634. space: expectedColorSpace, bitmapInfo: bitmapInfo.rawValue) else {
  635. expectedPixels.deallocate()
  636. observedPixels.deallocate()
  637. throw customError.unableToInitializeContext
  638. }
  639. guard let observedContext = CGContext(data: observedPixelsRaw, width: Int(imageSize.width), height: Int(imageSize.height),
  640. bitsPerComponent: observedCGImage.bitsPerComponent, bytesPerRow: bytesPerRow,
  641. space: observedColorSpace, bitmapInfo: bitmapInfo.rawValue) else {
  642. expectedPixels.deallocate()
  643. observedPixels.deallocate()
  644. throw customError.unableToInitializeContext
  645. }
  646. expectedContext.draw(expectedCGImage, in: CGRect(origin: .zero, size: imageSize))
  647. observedContext.draw(observedCGImage, in: CGRect(origin: .zero, size: imageSize))
  648. let expectedBuffer = UnsafeBufferPointer(start: expectedPixels, count: numberOfPixels)
  649. let observedBuffer = UnsafeBufferPointer(start: observedPixels, count: numberOfPixels)
  650. var isEqual = true
  651. if tolerance == 0 {
  652. isEqual = expectedBuffer.elementsEqual(observedBuffer)
  653. } else {
  654. // Go through each pixel in turn and see if it is different
  655. var numDiffPixels = 0
  656. for pixel in 0 ..< numberOfPixels where expectedBuffer[pixel] != observedBuffer[pixel] {
  657. // If this pixel is different, increment the pixel diff count and see if we have hit our limit.
  658. numDiffPixels += 1
  659. let percentage = 100 * Float(numDiffPixels) / Float(numberOfPixels)
  660. if percentage > tolerance {
  661. isEqual = false
  662. break
  663. }
  664. }
  665. }
  666. expectedPixels.deallocate()
  667. observedPixels.deallocate()
  668. return isEqual
  669. }
  670. func stringFromTime(_ time: CMTime) -> String {
  671. let interval = Int(CMTimeGetSeconds(time))
  672. let seconds = interval % 60
  673. let minutes = (interval / 60) % 60
  674. let hours = (interval / 3600)
  675. if hours > 0 {
  676. return String(format: "%02d:%02d:%02d", hours, minutes, seconds)
  677. } else {
  678. return String(format: "%02d:%02d", minutes, seconds)
  679. }
  680. }
  681. func colorNavigationController(_ navigationController: UINavigationController?, backgroundColor: UIColor, titleColor: UIColor, tintColor: UIColor?, withoutShadow: Bool) {
  682. let appearance = UINavigationBarAppearance()
  683. appearance.titleTextAttributes = [.foregroundColor: titleColor]
  684. appearance.largeTitleTextAttributes = [.foregroundColor: titleColor]
  685. if withoutShadow {
  686. appearance.shadowColor = .clear
  687. appearance.shadowImage = UIImage()
  688. }
  689. if let tintColor = tintColor {
  690. navigationController?.navigationBar.tintColor = tintColor
  691. }
  692. navigationController?.view.backgroundColor = backgroundColor
  693. navigationController?.navigationBar.barTintColor = titleColor
  694. navigationController?.navigationBar.standardAppearance = appearance
  695. navigationController?.navigationBar.compactAppearance = appearance
  696. navigationController?.navigationBar.scrollEdgeAppearance = appearance
  697. }
  698. func getEncondingDataType(data: Data) -> String.Encoding? {
  699. if let _ = String(data: data, encoding: .utf8) {
  700. return .utf8
  701. }
  702. if let _ = String(data: data, encoding: .ascii) {
  703. return .ascii
  704. }
  705. if let _ = String(data: data, encoding: .isoLatin1) {
  706. return .isoLatin1
  707. }
  708. if let _ = String(data: data, encoding: .isoLatin2) {
  709. return .isoLatin2
  710. }
  711. if let _ = String(data: data, encoding: .windowsCP1250) {
  712. return .windowsCP1250
  713. }
  714. if let _ = String(data: data, encoding: .windowsCP1251) {
  715. return .windowsCP1251
  716. }
  717. if let _ = String(data: data, encoding: .windowsCP1252) {
  718. return .windowsCP1252
  719. }
  720. if let _ = String(data: data, encoding: .windowsCP1253) {
  721. return .windowsCP1253
  722. }
  723. if let _ = String(data: data, encoding: .windowsCP1254) {
  724. return .windowsCP1254
  725. }
  726. if let _ = String(data: data, encoding: .macOSRoman) {
  727. return .macOSRoman
  728. }
  729. if let _ = String(data: data, encoding: .japaneseEUC) {
  730. return .japaneseEUC
  731. }
  732. if let _ = String(data: data, encoding: .nextstep) {
  733. return .nextstep
  734. }
  735. if let _ = String(data: data, encoding: .nonLossyASCII) {
  736. return .nonLossyASCII
  737. }
  738. if let _ = String(data: data, encoding: .shiftJIS) {
  739. return .shiftJIS
  740. }
  741. if let _ = String(data: data, encoding: .symbol) {
  742. return .symbol
  743. }
  744. if let _ = String(data: data, encoding: .unicode) {
  745. return .unicode
  746. }
  747. if let _ = String(data: data, encoding: .utf16) {
  748. return .utf16
  749. }
  750. if let _ = String(data: data, encoding: .utf16BigEndian) {
  751. return .utf16BigEndian
  752. }
  753. if let _ = String(data: data, encoding: .utf16LittleEndian) {
  754. return .utf16LittleEndian
  755. }
  756. if let _ = String(data: data, encoding: .utf32) {
  757. return .utf32
  758. }
  759. if let _ = String(data: data, encoding: .utf32BigEndian) {
  760. return .utf32BigEndian
  761. }
  762. if let _ = String(data: data, encoding: .utf32LittleEndian) {
  763. return .utf32LittleEndian
  764. }
  765. return nil
  766. }
  767. func SYSTEM_VERSION_LESS_THAN(version: String) -> Bool {
  768. return UIDevice.current.systemVersion.compare(version,
  769. options: NSString.CompareOptions.numeric) == ComparisonResult.orderedAscending
  770. }
  771. func getAvatarFromIconUrl(metadata: tableMetadata) -> String? {
  772. var ownerId: String?
  773. if metadata.iconUrl.contains("http") && metadata.iconUrl.contains("avatar") {
  774. let splitIconUrl = metadata.iconUrl.components(separatedBy: "/")
  775. var found:Bool = false
  776. for item in splitIconUrl {
  777. if found {
  778. ownerId = item
  779. break
  780. }
  781. if item == "avatar" { found = true}
  782. }
  783. }
  784. return ownerId
  785. }
  786. // https://stackoverflow.com/questions/25471114/how-to-validate-an-e-mail-address-in-swift
  787. func isValidEmail(_ email: String) -> Bool {
  788. let emailRegEx = "[A-Z0-9a-z._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{2,64}"
  789. let emailPred = NSPredicate(format:"SELF MATCHES %@", emailRegEx)
  790. return emailPred.evaluate(with: email)
  791. }
  792. func createFilePreviewImage(ocId: String, etag: String, fileNameView: String, classFile: String, status: Int, createPreviewMedia: Bool) -> UIImage? {
  793. var imagePreview: UIImage?
  794. let filePath = CCUtility.getDirectoryProviderStorageOcId(ocId, fileNameView: fileNameView)!
  795. let iconImagePath = CCUtility.getDirectoryProviderStorageIconOcId(ocId, etag: etag)!
  796. if FileManager().fileExists(atPath: iconImagePath) {
  797. imagePreview = UIImage(contentsOfFile: iconImagePath)
  798. } else if !createPreviewMedia {
  799. return nil
  800. } else if createPreviewMedia && status >= NCGlobal.shared.metadataStatusNormal && classFile == NKCommon.typeClassFile.image.rawValue && FileManager().fileExists(atPath: filePath) {
  801. if let image = UIImage(contentsOfFile: filePath), let image = image.resizeImage(size: CGSize(width: NCGlobal.shared.sizeIcon, height: NCGlobal.shared.sizeIcon)), let data = image.jpegData(compressionQuality: 0.5) {
  802. do {
  803. try data.write(to: URL.init(fileURLWithPath: iconImagePath), options: .atomic)
  804. imagePreview = image
  805. } catch { }
  806. }
  807. } else if createPreviewMedia && status >= NCGlobal.shared.metadataStatusNormal && classFile == NKCommon.typeClassFile.video.rawValue && FileManager().fileExists(atPath: filePath) {
  808. if let image = NCUtility.shared.imageFromVideo(url: URL(fileURLWithPath: filePath), at: 0), let image = image.resizeImage(size: CGSize(width: NCGlobal.shared.sizeIcon, height: NCGlobal.shared.sizeIcon)), let data = image.jpegData(compressionQuality: 0.5) {
  809. do {
  810. try data.write(to: URL.init(fileURLWithPath: iconImagePath), options: .atomic)
  811. imagePreview = image
  812. } catch { }
  813. }
  814. }
  815. return imagePreview
  816. }
  817. @discardableResult
  818. func convertDataToImage(data: Data?, size:CGSize, fileNameToWrite: String?) -> UIImage? {
  819. guard let data = data else { return nil }
  820. var returnImage: UIImage?
  821. if let image = UIImage(data: data), let image = image.resizeImage(size: size) {
  822. returnImage = image
  823. } else if let image = SVGKImage(data: data) {
  824. image.size = size
  825. returnImage = image.uiImage
  826. } else {
  827. print("error")
  828. }
  829. if let fileName = fileNameToWrite, let image = returnImage {
  830. do {
  831. let fileNamePath: String = CCUtility.getDirectoryUserData() + "/" + fileName + ".png"
  832. try image.pngData()?.write(to: URL(fileURLWithPath: fileNamePath), options: .atomic)
  833. } catch { }
  834. }
  835. return returnImage
  836. }
  837. func isDirectoryE2EE(serverUrl: String, userBase: NCUserBaseUrl) -> Bool {
  838. return isDirectoryE2EE(serverUrl: serverUrl, account: userBase.account, urlBase: userBase.urlBase, userId: userBase.userId)
  839. }
  840. func isDirectoryE2EE(file: NKFile) -> Bool {
  841. return isDirectoryE2EE(serverUrl: file.serverUrl, account: file.account, urlBase: file.urlBase, userId: file.userId)
  842. }
  843. @objc func isDirectoryE2EE(metadata: tableMetadata) -> Bool {
  844. return isDirectoryE2EE(serverUrl: metadata.serverUrl, account: metadata.account, urlBase: metadata.urlBase, userId: metadata.userId)
  845. }
  846. @objc func isDirectoryE2EE(serverUrl: String, account: String, urlBase: String, userId: String) -> Bool {
  847. if serverUrl == NCUtilityFileSystem.shared.getHomeServer(urlBase: urlBase, userId: userId) || serverUrl == ".." { return false }
  848. if let directory = NCManageDatabase.shared.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", account, serverUrl)) {
  849. return directory.e2eEncrypted
  850. }
  851. return false
  852. }
  853. }