// // ZLThumbnailViewController.swift // ZLPhotoBrowser // // Created by long on 2020/8/19. // // Copyright (c) 2020 Long Zhang <495181165@qq.com> // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. import UIKit import Photos extension ZLThumbnailViewController { enum SlideSelectType { case none case select case cancel } } class ZLThumbnailViewController: UIViewController { var albumList: ZLAlbumListModel var externalNavView: ZLExternalAlbumListNavView? var embedNavView: ZLEmbedAlbumListNavView? var embedAlbumListView: ZLEmbedAlbumListView? var collectionView: UICollectionView! var bottomView: UIView! var bottomBlurView: UIVisualEffectView? var limitAuthTipsView: ZLLimitedAuthorityTipsView? var previewBtn: UIButton! var originalBtn: UIButton! var doneBtn: UIButton! var arrDataSources: [ZLPhotoModel] = [] var showCameraCell: Bool { if ZLPhotoConfiguration.default().allowTakePhotoInLibrary && self.albumList.isCameraRoll { return true } return false } /// 所有滑动经过的indexPath lazy var arrSlideIndexPaths: [IndexPath] = [] /// 所有滑动经过的indexPath的初始选择状态 lazy var dicOriSelectStatus: [IndexPath: Bool] = [:] var isLayoutOK = false /// 设备旋转前第一个可视indexPath var firstVisibleIndexPathBeforeRotation: IndexPath? /// 是否触发了横竖屏切换 var isSwitchOrientation = false /// 是否开始出发滑动选择 var beginPanSelect = false /// 滑动选择 或 取消 /// 当初始滑动的cell处于未选择状态,则开始选择,反之,则开始取消选择 var panSelectType: ZLThumbnailViewController.SlideSelectType = .none /// 开始滑动的indexPath var beginSlideIndexPath: IndexPath? /// 最后滑动经过的index,开始的indexPath不计入 /// 优化拖动手势计算,避免单个cell中冗余计算多次 var lastSlideIndex: Int? /// 预览所选择图片,手势返回时候不调用scrollToIndex var isPreviewPush = false /// 拍照后置为true,需要刷新相册列表 var hasTakeANewAsset = false var slideCalculateQueue = DispatchQueue(label: "com.ZLhotoBrowser.slide") var autoScrollTimer: CADisplayLink? var lastPanUpdateTime = CACurrentMediaTime() let showLimitAuthTipsView: Bool = { if #available(iOS 14.0, *), PHPhotoLibrary.authorizationStatus(for: .readWrite) == .limited, ZLPhotoConfiguration.default().showEnterSettingTips { return true } else { return false } }() private enum AutoScrollDirection { case none case top case bottom } private var autoScrollInfo: (direction: AutoScrollDirection, speed: CGFloat) = (.none, 0) @available(iOS 14, *) var showAddPhotoCell: Bool { PHPhotoLibrary.authorizationStatus(for: .readWrite) == .limited && ZLPhotoConfiguration.default().showAddPhotoButton && self.albumList.isCameraRoll } /// 照相按钮+添加图片按钮的数量 /// the count of addPhotoButton & cameraButton private var offset: Int { if #available(iOS 14, *) { return Int(self.showAddPhotoCell) + Int(self.showCameraCell) } else { return Int(self.showCameraCell) } } override var prefersStatusBarHidden: Bool { return false } override var preferredStatusBarStyle: UIStatusBarStyle { return ZLPhotoConfiguration.default().statusBarStyle } var panGes: UIPanGestureRecognizer! deinit { zl_debugPrint("ZLThumbnailViewController deinit") self.cleanTimer() } init(albumList: ZLAlbumListModel) { self.albumList = albumList super.init(nibName: nil, bundle: nil) } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } override func viewDidLoad() { super.viewDidLoad() self.setupUI() if ZLPhotoConfiguration.default().allowSlideSelect { self.panGes = UIPanGestureRecognizer(target: self, action: #selector(slideSelectAction(_:))) self.panGes.delegate = self self.view.addGestureRecognizer(self.panGes) } NotificationCenter.default.addObserver(self, selector: #selector(deviceOrientationChanged(_:)), name: UIApplication.willChangeStatusBarOrientationNotification, object: nil) self.loadPhotos() // Register for the album change notification when the status is limited, because the photoLibraryDidChange method will be repeated multiple times each time the album changes, causing the interface to refresh multiple times. So the album changes are not monitored in other authority. if #available(iOS 14.0, *), PHPhotoLibrary.authorizationStatus(for: .readWrite) == .limited { PHPhotoLibrary.shared().register(self) } } override func viewWillAppear(_ animated: Bool) { super.viewWillAppear(animated) self.navigationController?.navigationBar.isHidden = true self.collectionView.reloadItems(at: self.collectionView.indexPathsForVisibleItems) self.resetBottomToolBtnStatus() } override func viewDidAppear(_ animated: Bool) { super.viewDidAppear(animated) self.isLayoutOK = true self.isPreviewPush = false } override func viewDidLayoutSubviews() { super.viewDidLayoutSubviews() let navViewNormalH: CGFloat = 44 var insets = UIEdgeInsets(top: 20, left: 0, bottom: 0, right: 0) var collectionViewInsetTop: CGFloat = 20 if #available(iOS 11.0, *) { insets = self.view.safeAreaInsets collectionViewInsetTop = navViewNormalH } else { collectionViewInsetTop += navViewNormalH } let navViewFrame = CGRect(x: 0, y: 0, width: self.view.frame.width, height: insets.top + navViewNormalH) self.externalNavView?.frame = navViewFrame self.embedNavView?.frame = navViewFrame self.embedAlbumListView?.frame = CGRect(x: 0, y: navViewFrame.maxY, width: self.view.bounds.width, height: self.view.bounds.height-navViewFrame.maxY) var showBottomToolBtns = true let config = ZLPhotoConfiguration.default() let condition1 = config.editAfterSelectThumbnailImage && config.maxSelectCount == 1 && (config.allowEditImage || config.allowEditVideo) let condition2 = config.allowPreviewPhotos && config.maxSelectCount == 1 && !config.showSelectBtnWhenSingleSelect if condition1 || condition2 { showBottomToolBtns = false } let bottomViewH: CGFloat if self.showLimitAuthTipsView, showBottomToolBtns { bottomViewH = ZLLayout.bottomToolViewH + ZLLimitedAuthorityTipsView.height } else if self.showLimitAuthTipsView { bottomViewH = ZLLimitedAuthorityTipsView.height } else if showBottomToolBtns { bottomViewH = ZLLayout.bottomToolViewH } else { bottomViewH = 0 } let totalWidth = self.view.frame.width - insets.left - insets.right self.collectionView.frame = CGRect(x: insets.left, y: 0, width: totalWidth, height: self.view.frame.height) self.collectionView.contentInset = UIEdgeInsets(top: collectionViewInsetTop, left: 0, bottom: bottomViewH, right: 0) self.collectionView.scrollIndicatorInsets = UIEdgeInsets(top: insets.top, left: 0, bottom: bottomViewH, right: 0) if !self.isLayoutOK { self.scrollToBottom() } else if self.isSwitchOrientation { self.isSwitchOrientation = false if let ip = self.firstVisibleIndexPathBeforeRotation { self.collectionView.scrollToItem(at: ip, at: .top, animated: false) } } guard showBottomToolBtns || self.showLimitAuthTipsView else { return } let btnH = ZLLayout.bottomToolBtnH self.bottomView.frame = CGRect(x: 0, y: self.view.frame.height-insets.bottom-bottomViewH, width: self.view.bounds.width, height: bottomViewH+insets.bottom) self.bottomBlurView?.frame = self.bottomView.bounds if self.showLimitAuthTipsView { self.limitAuthTipsView?.frame = CGRect(x: 0, y: 0, width: self.bottomView.bounds.width, height: ZLLimitedAuthorityTipsView.height) } if showBottomToolBtns { let btnY = self.showLimitAuthTipsView ? ZLLimitedAuthorityTipsView.height + ZLLayout.bottomToolBtnY : ZLLayout.bottomToolBtnY let previewTitle = localLanguageTextValue(.preview) let previewBtnW = previewTitle.boundingRect(font: ZLLayout.bottomToolTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 30)).width self.previewBtn.frame = CGRect(x: 15, y: btnY, width: previewBtnW, height: btnH) let originalTitle = localLanguageTextValue(.originalPhoto) let originBtnW = originalTitle.boundingRect(font: ZLLayout.bottomToolTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 30)).width + 30 self.originalBtn.frame = CGRect(x: (self.bottomView.bounds.width-originBtnW)/2-5, y: btnY, width: originBtnW, height: btnH) self.refreshDoneBtnFrame() } } func setupUI() { self.automaticallyAdjustsScrollViewInsets = true self.edgesForExtendedLayout = .all self.view.backgroundColor = .thumbnailBgColor let layout = UICollectionViewFlowLayout() layout.sectionInset = UIEdgeInsets(top: 3, left: 0, bottom: 3, right: 0) self.collectionView = UICollectionView(frame: .zero, collectionViewLayout: layout) self.collectionView.backgroundColor = .thumbnailBgColor self.collectionView.dataSource = self self.collectionView.delegate = self self.collectionView.alwaysBounceVertical = true if #available(iOS 11.0, *) { self.collectionView.contentInsetAdjustmentBehavior = .always } self.view.addSubview(self.collectionView) ZLCameraCell.zl_register(self.collectionView) ZLThumbnailPhotoCell.zl_register(self.collectionView) ZLAddPhotoCell.zl_register(self.collectionView) self.bottomView = UIView() self.bottomView.backgroundColor = .bottomToolViewBgColor self.view.addSubview(self.bottomView) if let effect = ZLPhotoConfiguration.default().bottomToolViewBlurEffect { self.bottomBlurView = UIVisualEffectView(effect: effect) self.bottomView.addSubview(self.bottomBlurView!) } if self.showLimitAuthTipsView { self.limitAuthTipsView = ZLLimitedAuthorityTipsView(frame: CGRect(x: 0, y: 0, width: self.view.bounds.width, height: ZLLimitedAuthorityTipsView.height)) self.bottomView.addSubview(self.limitAuthTipsView!) } func createBtn(_ title: String, _ action: Selector) -> UIButton { let btn = UIButton(type: .custom) btn.titleLabel?.font = ZLLayout.bottomToolTitleFont btn.setTitle(title, for: .normal) btn.setTitleColor(.bottomToolViewBtnNormalTitleColor, for: .normal) btn.setTitleColor(.bottomToolViewBtnDisableTitleColor, for: .disabled) btn.addTarget(self, action: action, for: .touchUpInside) return btn } self.previewBtn = createBtn(localLanguageTextValue(.preview), #selector(previewBtnClick)) self.previewBtn.isHidden = !ZLPhotoConfiguration.default().showPreviewButtonInAlbum self.bottomView.addSubview(self.previewBtn) self.originalBtn = createBtn(localLanguageTextValue(.originalPhoto), #selector(originalPhotoClick)) self.originalBtn.setImage(getImage("zl_btn_original_circle"), for: .normal) self.originalBtn.setImage(getImage("zl_btn_original_selected"), for: .selected) self.originalBtn.titleEdgeInsets = UIEdgeInsets(top: 0, left: 10, bottom: 0, right: 0) self.originalBtn.isHidden = !(ZLPhotoConfiguration.default().allowSelectOriginal && ZLPhotoConfiguration.default().allowSelectImage) self.originalBtn.isSelected = (self.navigationController as! ZLImageNavController).isSelectedOriginal self.bottomView.addSubview(self.originalBtn) self.doneBtn = createBtn(localLanguageTextValue(.done), #selector(doneBtnClick)) self.doneBtn.layer.masksToBounds = true self.doneBtn.layer.cornerRadius = ZLLayout.bottomToolBtnCornerRadius self.bottomView.addSubview(self.doneBtn) self.setupNavView() } func setupNavView() { if ZLPhotoConfiguration.default().style == .embedAlbumList { self.embedNavView = ZLEmbedAlbumListNavView(title: self.albumList.title) self.embedNavView?.selectAlbumBlock = { [weak self] in if self?.embedAlbumListView?.isHidden == true { self?.embedAlbumListView?.show(reloadAlbumList: self?.hasTakeANewAsset ?? false) self?.hasTakeANewAsset = false } else { self?.embedAlbumListView?.hide() } } self.embedNavView?.cancelBlock = { [weak self] in let nav = self?.navigationController as? ZLImageNavController nav?.cancelBlock?() nav?.dismiss(animated: true, completion: nil) } self.view.addSubview(self.embedNavView!) self.embedAlbumListView = ZLEmbedAlbumListView(selectedAlbum: self.albumList) self.embedAlbumListView?.isHidden = true self.embedAlbumListView?.selectAlbumBlock = { [weak self] (album) in guard self?.albumList != album else { return } self?.albumList = album self?.embedNavView?.title = album.title self?.loadPhotos() self?.embedNavView?.reset() } self.embedAlbumListView?.hideBlock = { [weak self] in self?.embedNavView?.reset() } self.view.addSubview(self.embedAlbumListView!) } else if ZLPhotoConfiguration.default().style == .externalAlbumList { self.externalNavView = ZLExternalAlbumListNavView(title: self.albumList.title) self.externalNavView?.backBlock = { [weak self] in self?.navigationController?.popViewController(animated: true) } self.externalNavView?.cancelBlock = { [weak self] in let nav = self?.navigationController as? ZLImageNavController nav?.cancelBlock?() nav?.dismiss(animated: true, completion: nil) } self.view.addSubview(self.externalNavView!) } } func loadPhotos() { let nav = self.navigationController as! ZLImageNavController if self.albumList.models.isEmpty { let hud = ZLProgressHUD(style: ZLPhotoConfiguration.default().hudStyle) hud.show() DispatchQueue.global().async { self.albumList.refetchPhotos() DispatchQueue.main.async { self.arrDataSources.removeAll() self.arrDataSources.append(contentsOf: self.albumList.models) markSelected(source: &self.arrDataSources, selected: &nav.arrSelectedModels) hud.hide() self.collectionView.reloadData() self.scrollToBottom() } } } else { self.arrDataSources.removeAll() self.arrDataSources.append(contentsOf: self.albumList.models) markSelected(source: &self.arrDataSources, selected: &nav.arrSelectedModels) self.collectionView.reloadData() self.scrollToBottom() } } // MARK: btn actions @objc func previewBtnClick() { let nav = self.navigationController as! ZLImageNavController let vc = ZLPhotoPreviewController(photos: nav.arrSelectedModels, index: 0) self.show(vc, sender: nil) } @objc func originalPhotoClick() { self.originalBtn.isSelected = !self.originalBtn.isSelected (self.navigationController as? ZLImageNavController)?.isSelectedOriginal = self.originalBtn.isSelected } @objc func doneBtnClick() { let nav = self.navigationController as? ZLImageNavController nav?.selectImageBlock?() } @objc func deviceOrientationChanged(_ notify: Notification) { let pInView = self.collectionView.convert(CGPoint(x: 100, y: 100), from: self.view) self.firstVisibleIndexPathBeforeRotation = self.collectionView.indexPathForItem(at: pInView) self.isSwitchOrientation = true } @objc func slideSelectAction(_ pan: UIPanGestureRecognizer) { let point = pan.location(in: self.collectionView) guard let indexPath = self.collectionView.indexPathForItem(at: point) else { return } let config = ZLPhotoConfiguration.default() let nav = self.navigationController as! ZLImageNavController let cell = self.collectionView.cellForItem(at: indexPath) as? ZLThumbnailPhotoCell let asc = config.sortAscending if pan.state == .began { self.beginPanSelect = (cell != nil) if self.beginPanSelect { let index = asc ? indexPath.row : indexPath.row - self.offset let m = self.arrDataSources[index] self.panSelectType = m.isSelected ? .cancel : .select self.beginSlideIndexPath = indexPath if !m.isSelected, nav.arrSelectedModels.count < config.maxSelectCount, canAddModel(m, currentSelectCount: nav.arrSelectedModels.count, sender: self) { if self.shouldDirectEdit(m) { self.panSelectType = .none return } else { m.isSelected = true nav.arrSelectedModels.append(m) } } else if m.isSelected { m.isSelected = false nav.arrSelectedModels.removeAll { $0 == m } } cell?.btnSelect.isSelected = m.isSelected self.refreshCellIndexAndMaskView() self.resetBottomToolBtnStatus() self.lastSlideIndex = indexPath.row } } else if pan.state == .changed { self.autoScrollWhenSlideSelect(pan) if !self.beginPanSelect || indexPath.row == self.lastSlideIndex || self.panSelectType == .none || cell == nil { return } guard let beginIndexPath = self.beginSlideIndexPath else { return } self.lastPanUpdateTime = CACurrentMediaTime() let visiblePaths = self.collectionView.indexPathsForVisibleItems self.slideCalculateQueue.async { self.lastSlideIndex = indexPath.row let minIndex = min(indexPath.row, beginIndexPath.row) let maxIndex = max(indexPath.row, beginIndexPath.row) let minIsBegin = minIndex == beginIndexPath.row var i = beginIndexPath.row while (minIsBegin ? i <= maxIndex : i >= minIndex) { if i != beginIndexPath.row { let p = IndexPath(row: i, section: 0) if !self.arrSlideIndexPaths.contains(p) { self.arrSlideIndexPaths.append(p) let index = asc ? i : i - self.offset let m = self.arrDataSources[index] self.dicOriSelectStatus[p] = m.isSelected } } i += (minIsBegin ? 1 : -1) } var selectedArrHasChange = false for path in self.arrSlideIndexPaths { if !visiblePaths.contains(path) { continue } let index = asc ? path.row : path.row - self.offset // 是否在最初和现在的间隔区间内 let inSection = path.row >= minIndex && path.row <= maxIndex let m = self.arrDataSources[index] if self.panSelectType == .select { if inSection, !m.isSelected, canAddModel(m, currentSelectCount: nav.arrSelectedModels.count, sender: self, showAlert: false) { m.isSelected = true } } else if self.panSelectType == .cancel { if inSection { m.isSelected = false } } if !inSection { // 未在区间内的model还原为初始选择状态 m.isSelected = self.dicOriSelectStatus[path] ?? false } if !m.isSelected { if let index = nav.arrSelectedModels.firstIndex(where: { $0 == m }) { nav.arrSelectedModels.remove(at: index) selectedArrHasChange = true } } else { if !nav.arrSelectedModels.contains(where: { $0 == m }) { nav.arrSelectedModels.append(m) selectedArrHasChange = true } } DispatchQueue.main.async { let c = self.collectionView.cellForItem(at: path) as? ZLThumbnailPhotoCell c?.btnSelect.isSelected = m.isSelected } } if selectedArrHasChange { DispatchQueue.main.async { self.refreshCellIndexAndMaskView() self.resetBottomToolBtnStatus() } } } } else if pan.state == .ended || pan.state == .cancelled { self.cleanTimer() self.panSelectType = .none self.arrSlideIndexPaths.removeAll() self.dicOriSelectStatus.removeAll() self.resetBottomToolBtnStatus() } } func autoScrollWhenSlideSelect(_ pan: UIPanGestureRecognizer) { guard ZLPhotoConfiguration.default().autoScrollWhenSlideSelectIsActive else { return } let arrSel = (self.navigationController as? ZLImageNavController)?.arrSelectedModels ?? [] guard arrSel.count < ZLPhotoConfiguration.default().maxSelectCount else { // Stop auto scroll when reach the max select count. self.cleanTimer() return } let top = ((self.embedNavView?.frame.height ?? self.externalNavView?.frame.height) ?? 44) + 30 let bottom = self.bottomView.frame.minY - 30 let point = pan.location(in: self.view) var diff: CGFloat = 0 var direction: AutoScrollDirection = .none if point.y < top { diff = top - point.y direction = .top } else if point.y > bottom { diff = point.y - bottom direction = .bottom } else { self.autoScrollInfo = (.none, 0) self.cleanTimer() return } guard diff > 0 else { return } let s = min(diff, 60) / 60 * ZLPhotoConfiguration.default().autoScrollMaxSpeed self.autoScrollInfo = (direction, s) if self.autoScrollTimer == nil { self.cleanTimer() self.autoScrollTimer = CADisplayLink(target: ZLWeakProxy(target: self), selector: #selector(autoScrollAction)) self.autoScrollTimer?.add(to: RunLoop.current, forMode: .common) } } func cleanTimer() { self.autoScrollTimer?.remove(from: RunLoop.current, forMode: .common) self.autoScrollTimer?.invalidate() self.autoScrollTimer = nil } @objc func autoScrollAction() { guard self.autoScrollInfo.direction != .none else { return } let duration = CGFloat(self.autoScrollTimer?.duration ?? 1 / 60) if CACurrentMediaTime() - self.lastPanUpdateTime > 0.2 { // Finger may be not moved in slide selection mode self.slideSelectAction(self.panGes) } let distance = self.autoScrollInfo.speed * duration let offset = self.collectionView.contentOffset let inset = self.collectionView.contentInset if self.autoScrollInfo.direction == .top, offset.y + inset.top > distance { self.collectionView.contentOffset = CGPoint(x: 0, y: offset.y - distance) } else if self.autoScrollInfo.direction == .bottom, offset.y + self.collectionView.bounds.height + distance - inset.bottom < self.collectionView.contentSize.height { self.collectionView.contentOffset = CGPoint(x: 0, y: offset.y + distance) } } func resetBottomToolBtnStatus() { let nav = self.navigationController as! ZLImageNavController if nav.arrSelectedModels.count > 0 { self.previewBtn.isEnabled = true self.doneBtn.isEnabled = true let doneTitle = localLanguageTextValue(.done) + "(" + String(nav.arrSelectedModels.count) + ")" self.doneBtn.setTitle(doneTitle, for: .normal) self.doneBtn.backgroundColor = .bottomToolViewBtnNormalBgColor } else { self.previewBtn.isEnabled = false self.doneBtn.isEnabled = false self.doneBtn.setTitle(localLanguageTextValue(.done), for: .normal) self.doneBtn.backgroundColor = .bottomToolViewBtnDisableBgColor } self.originalBtn.isSelected = nav.isSelectedOriginal self.refreshDoneBtnFrame() } func refreshDoneBtnFrame() { let selCount = (self.navigationController as? ZLImageNavController)?.arrSelectedModels.count ?? 0 var doneTitle = localLanguageTextValue(.done) if selCount > 0 { doneTitle += "(" + String(selCount) + ")" } let doneBtnW = doneTitle.boundingRect(font: ZLLayout.bottomToolTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 30)).width + 20 let btnY = self.showLimitAuthTipsView ? ZLLimitedAuthorityTipsView.height + ZLLayout.bottomToolBtnY : ZLLayout.bottomToolBtnY self.doneBtn.frame = CGRect(x: self.bottomView.bounds.width-doneBtnW-15, y: btnY, width: doneBtnW, height: ZLLayout.bottomToolBtnH) } func scrollToBottom() { guard ZLPhotoConfiguration.default().sortAscending, self.arrDataSources.count > 0 else { return } let index = self.arrDataSources.count - 1 + self.offset self.collectionView.scrollToItem(at: IndexPath(row: index, section: 0), at: .centeredVertically, animated: false) } func showCamera() { let config = ZLPhotoConfiguration.default() if config.useCustomCamera { let camera = ZLCustomCamera() camera.takeDoneBlock = { [weak self] (image, videoUrl) in self?.save(image: image, videoUrl: videoUrl) } self.showDetailViewController(camera, sender: nil) } else { if UIImagePickerController.isSourceTypeAvailable(.camera) { let picker = UIImagePickerController() picker.delegate = self picker.allowsEditing = false picker.videoQuality = .typeHigh picker.sourceType = .camera picker.cameraFlashMode = config.cameraFlashMode.imagePickerFlashMode var mediaTypes = [String]() if config.allowTakePhoto { mediaTypes.append("public.image") } if config.allowRecordVideo { mediaTypes.append("public.movie") } picker.mediaTypes = mediaTypes picker.videoMaximumDuration = TimeInterval(config.maxRecordDuration) self.showDetailViewController(picker, sender: nil) } else { showAlertView(localLanguageTextValue(.cameraUnavailable), self) } } } func save(image: UIImage?, videoUrl: URL?) { let hud = ZLProgressHUD(style: ZLPhotoConfiguration.default().hudStyle) if let image = image { hud.show() ZLPhotoManager.saveImageToAlbum(image: image) { [weak self] (suc, asset) in if suc, let at = asset { let model = ZLPhotoModel(asset: at) self?.handleDataArray(newModel: model) } else { showAlertView(localLanguageTextValue(.saveImageError), self) } hud.hide() } } else if let videoUrl = videoUrl { hud.show() ZLPhotoManager.saveVideoToAlbum(url: videoUrl) { [weak self] (suc, asset) in if suc, let at = asset { let model = ZLPhotoModel(asset: at) self?.handleDataArray(newModel: model) } else { showAlertView(localLanguageTextValue(.saveVideoError), self) } hud.hide() } } } func handleDataArray(newModel: ZLPhotoModel) { self.hasTakeANewAsset = true self.albumList.refreshResult() let nav = self.navigationController as? ZLImageNavController let config = ZLPhotoConfiguration.default() var insertIndex = 0 if config.sortAscending { insertIndex = self.arrDataSources.count self.arrDataSources.append(newModel) } else { // 保存拍照的照片或者视频,说明肯定有camera cell insertIndex = self.offset self.arrDataSources.insert(newModel, at: 0) } var canSelect = true // If mixed selection is not allowed, and the newModel type is video, it will not be selected. if !config.allowMixSelect, newModel.type == .video { canSelect = false } if canSelect, canAddModel(newModel, currentSelectCount: nav?.arrSelectedModels.count ?? 0, sender: self, showAlert: false) { if !self.shouldDirectEdit(newModel) { newModel.isSelected = true nav?.arrSelectedModels.append(newModel) } } let insertIndexPath = IndexPath(row: insertIndex, section: 0) self.collectionView.performBatchUpdates({ self.collectionView.insertItems(at: [insertIndexPath]) }) { (_) in self.collectionView.scrollToItem(at: insertIndexPath, at: .centeredVertically, animated: true) self.collectionView.reloadItems(at: self.collectionView.indexPathsForVisibleItems) } self.resetBottomToolBtnStatus() } func showEditImageVC(model: ZLPhotoModel) { let nav = self.navigationController as! ZLImageNavController let hud = ZLProgressHUD(style: ZLPhotoConfiguration.default().hudStyle) hud.show() hud.show() ZLPhotoManager.fetchImage(for: model.asset, size: model.previewSize) { [weak self, weak nav] (image, isDegraded) in if !isDegraded { if let image = image { ZLEditImageViewController.showEditImageVC(parentVC: self, image: image, editModel: model.editImageModel) { [weak nav] (ei, editImageModel) in model.isSelected = true model.editImage = ei model.editImageModel = editImageModel nav?.arrSelectedModels.append(model) nav?.selectImageBlock?() } } else { showAlertView(localLanguageTextValue(.imageLoadFailed), self) } hud.hide() } } } func showEditVideoVC(model: ZLPhotoModel) { let nav = self.navigationController as! ZLImageNavController let hud = ZLProgressHUD(style: ZLPhotoConfiguration.default().hudStyle) var requestAvAssetID: PHImageRequestID? hud.show(timeout: 20) hud.timeoutBlock = { [weak self] in showAlertView(localLanguageTextValue(.timeout), self) if let _ = requestAvAssetID { PHImageManager.default().cancelImageRequest(requestAvAssetID!) } } func inner_showEditVideoVC(_ avAsset: AVAsset) { let vc = ZLEditVideoViewController(avAsset: avAsset) vc.editFinishBlock = { [weak self, weak nav] (url) in if let u = url { ZLPhotoManager.saveVideoToAlbum(url: u) { [weak self, weak nav] (suc, asset) in if suc, asset != nil { let m = ZLPhotoModel(asset: asset!) m.isSelected = true nav?.arrSelectedModels.append(m) nav?.selectImageBlock?() } else { showAlertView(localLanguageTextValue(.saveVideoError), self) } } } else { nav?.arrSelectedModels.append(model) nav?.selectImageBlock?() } } vc.modalPresentationStyle = .fullScreen self.showDetailViewController(vc, sender: nil) } // 提前fetch一下 avasset requestAvAssetID = ZLPhotoManager.fetchAVAsset(forVideo: model.asset) { [weak self] (avAsset, _) in hud.hide() if let _ = avAsset { inner_showEditVideoVC(avAsset!) } else { showAlertView(localLanguageTextValue(.timeout), self) } } } } // MARK: Gesture delegate extension ZLThumbnailViewController: UIGestureRecognizerDelegate { func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool { let config = ZLPhotoConfiguration.default() if config.maxSelectCount == 1, !config.showSelectBtnWhenSingleSelect { return false } return true } } // MARK: CollectionView Delegate & DataSource extension ZLThumbnailViewController: UICollectionViewDataSource, UICollectionViewDelegateFlowLayout { func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, minimumInteritemSpacingForSectionAt section: Int) -> CGFloat { return ZLLayout.thumbCollectionViewItemSpacing } func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, minimumLineSpacingForSectionAt section: Int) -> CGFloat { return ZLLayout.thumbCollectionViewLineSpacing } func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize { let defaultCount = CGFloat(ZLPhotoConfiguration.default().columnCount) var columnCount: CGFloat = deviceIsiPad() ? (defaultCount+2) : defaultCount if UIApplication.shared.statusBarOrientation.isLandscape { columnCount += 2 } let totalW = collectionView.bounds.width - (columnCount - 1) * ZLLayout.thumbCollectionViewItemSpacing let singleW = totalW / columnCount return CGSize(width: singleW, height: singleW) } func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int { return self.arrDataSources.count + self.offset } func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell { let config = ZLPhotoConfiguration.default() if self.showCameraCell && ((config.sortAscending && indexPath.row == self.arrDataSources.count) || (!config.sortAscending && indexPath.row == 0)) { // camera cell let cell = collectionView.dequeueReusableCell(withReuseIdentifier: ZLCameraCell.zl_identifier(), for: indexPath) as! ZLCameraCell if config.showCaptureImageOnTakePhotoBtn { cell.startCapture() } return cell } if #available(iOS 14, *) { if self.showAddPhotoCell && ((config.sortAscending && indexPath.row == self.arrDataSources.count - 1 + self.offset) || (!config.sortAscending && indexPath.row == self.offset - 1)) { guard let cell = collectionView.dequeueReusableCell(withReuseIdentifier: ZLAddPhotoCell.zl_identifier(), for: indexPath) as? ZLAddPhotoCell else { return UICollectionViewCell() } return cell } } let cell = collectionView.dequeueReusableCell(withReuseIdentifier: ZLThumbnailPhotoCell.zl_identifier(), for: indexPath) as! ZLThumbnailPhotoCell let model: ZLPhotoModel if !config.sortAscending { model = self.arrDataSources[indexPath.row - self.offset] } else { model = self.arrDataSources[indexPath.row] } let nav = self.navigationController as? ZLImageNavController cell.selectedBlock = { [weak self, weak nav, weak cell] (isSelected) in if !isSelected { let currentSelectCount = nav?.arrSelectedModels.count ?? 0 guard canAddModel(model, currentSelectCount: currentSelectCount, sender: self) else { return } if self?.shouldDirectEdit(model) == false { model.isSelected = true nav?.arrSelectedModels.append(model) cell?.btnSelect.isSelected = true self?.refreshCellIndexAndMaskView() } } else { cell?.btnSelect.isSelected = false model.isSelected = false nav?.arrSelectedModels.removeAll { $0 == model } self?.refreshCellIndexAndMaskView() } self?.resetBottomToolBtnStatus() } cell.indexLabel.isHidden = true if ZLPhotoConfiguration.default().showSelectedIndex { for (index, selM) in (nav?.arrSelectedModels ?? []).enumerated() { if model == selM { self.setCellIndex(cell, showIndexLabel: true, index: index + 1) break } } } self.setCellMaskView(cell, isSelected: model.isSelected, model: model) cell.model = model return cell } func collectionView(_ collectionView: UICollectionView, willDisplay cell: UICollectionViewCell, forItemAt indexPath: IndexPath) { guard let c = cell as? ZLThumbnailPhotoCell else { return } var index = indexPath.row if !ZLPhotoConfiguration.default().sortAscending { index -= self.offset } let model = self.arrDataSources[index] self.setCellMaskView(c, isSelected: model.isSelected, model: model) } func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) { let c = collectionView.cellForItem(at: indexPath) if c is ZLCameraCell { self.showCamera() return } if #available(iOS 14, *) { if c is ZLAddPhotoCell { PHPhotoLibrary.shared().presentLimitedLibraryPicker(from: self) return } } guard let cell = c as? ZLThumbnailPhotoCell else { return } if !ZLPhotoConfiguration.default().allowPreviewPhotos { cell.btnSelectClick() return } if !cell.enableSelect, ZLPhotoConfiguration.default().showInvalidMask { return } let config = ZLPhotoConfiguration.default() var index = indexPath.row if !config.sortAscending { index -= self.offset } let m = self.arrDataSources[index] if self.shouldDirectEdit(m) { return } let vc = ZLPhotoPreviewController(photos: self.arrDataSources, index: index) self.show(vc, sender: nil) } func shouldDirectEdit(_ model: ZLPhotoModel) -> Bool { let config = ZLPhotoConfiguration.default() let canEditImage = config.editAfterSelectThumbnailImage && config.allowEditImage && config.maxSelectCount == 1 && model.type.rawValue < ZLPhotoModel.MediaType.video.rawValue let canEditVideo = (config.editAfterSelectThumbnailImage && config.allowEditVideo && model.type == .video && config.maxSelectCount == 1) || (config.allowEditVideo && model.type == .video && !config.allowMixSelect && config.cropVideoAfterSelectThumbnail) //当前未选择图片 或已经选择了一张并且点击的是已选择的图片 let nav = self.navigationController as? ZLImageNavController let arrSelectedModels = nav?.arrSelectedModels ?? [] let flag = arrSelectedModels.isEmpty || (arrSelectedModels.count == 1 && arrSelectedModels.first?.ident == model.ident) if canEditImage, flag { self.showEditImageVC(model: model) } else if canEditVideo, flag { self.showEditVideoVC(model: model) } return flag && (canEditImage || canEditVideo) } func setCellIndex(_ cell: ZLThumbnailPhotoCell?, showIndexLabel: Bool, index: Int) { guard ZLPhotoConfiguration.default().showSelectedIndex else { return } cell?.index = index cell?.indexLabel.isHidden = !showIndexLabel } func refreshCellIndexAndMaskView() { let showIndex = ZLPhotoConfiguration.default().showSelectedIndex let showMask = ZLPhotoConfiguration.default().showSelectedMask || ZLPhotoConfiguration.default().showInvalidMask guard showIndex || showMask else { return } let visibleIndexPaths = self.collectionView.indexPathsForVisibleItems visibleIndexPaths.forEach { (indexPath) in guard let cell = self.collectionView.cellForItem(at: indexPath) as? ZLThumbnailPhotoCell else { return } var row = indexPath.row if !ZLPhotoConfiguration.default().sortAscending { row -= self.offset } let m = self.arrDataSources[row] let arrSel = (self.navigationController as? ZLImageNavController)?.arrSelectedModels ?? [] var show = false var idx = 0 var isSelected = false for (index, selM) in arrSel.enumerated() { if m == selM { show = true idx = index + 1 isSelected = true break } } if showIndex { self.setCellIndex(cell, showIndexLabel: show, index: idx) } if showMask { self.setCellMaskView(cell, isSelected: isSelected, model: m) } } } func setCellMaskView(_ cell: ZLThumbnailPhotoCell, isSelected: Bool, model: ZLPhotoModel) { cell.coverView.isHidden = true cell.enableSelect = true let arrSel = (self.navigationController as? ZLImageNavController)?.arrSelectedModels ?? [] let config = ZLPhotoConfiguration.default() if isSelected { cell.coverView.backgroundColor = .selectedMaskColor cell.coverView.isHidden = !config.showSelectedMask if config.showSelectedBorder { cell.layer.borderWidth = 4 } } else { let selCount = arrSel.count if selCount < config.maxSelectCount { if config.allowMixSelect { let videoCount = arrSel.filter { $0.type == .video }.count if videoCount >= config.maxVideoSelectCount, model.type == .video { cell.coverView.backgroundColor = .invalidMaskColor cell.coverView.isHidden = !config.showInvalidMask cell.enableSelect = false } else if (config.maxSelectCount - selCount) <= (config.minVideoSelectCount - videoCount), model.type != .video { cell.coverView.backgroundColor = .invalidMaskColor cell.coverView.isHidden = !config.showInvalidMask cell.enableSelect = false } } else if selCount > 0 { cell.coverView.backgroundColor = .invalidMaskColor cell.coverView.isHidden = (!config.showInvalidMask || model.type != .video) cell.enableSelect = model.type != .video } } else if selCount >= config.maxSelectCount { cell.coverView.backgroundColor = .invalidMaskColor cell.coverView.isHidden = !config.showInvalidMask cell.enableSelect = false } if config.showSelectedBorder { cell.layer.borderWidth = 0 } } } } extension ZLThumbnailViewController: UIImagePickerControllerDelegate, UINavigationControllerDelegate { func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) { picker.dismiss(animated: true) { let image = info[.originalImage] as? UIImage let url = info[.mediaURL] as? URL self.save(image: image, videoUrl: url) } } } extension ZLThumbnailViewController: PHPhotoLibraryChangeObserver { func photoLibraryDidChange(_ changeInstance: PHChange) { guard let changes = changeInstance.changeDetails(for: self.albumList.result) else { return } DispatchQueue.main.sync { // 变化后再次显示相册列表需要刷新 self.hasTakeANewAsset = true self.albumList.result = changes.fetchResultAfterChanges let nav = (self.navigationController as! ZLImageNavController) if changes.hasIncrementalChanges { for sm in nav.arrSelectedModels { let isDelete = changeInstance.changeDetails(for: sm.asset)?.objectWasDeleted ?? false if isDelete { nav.arrSelectedModels.removeAll { $0 == sm } } } if (!changes.removedObjects.isEmpty || !changes.insertedObjects.isEmpty) { self.albumList.models.removeAll() } self.loadPhotos() } else { for sm in nav.arrSelectedModels { let isDelete = changeInstance.changeDetails(for: sm.asset)?.objectWasDeleted ?? false if isDelete { nav.arrSelectedModels.removeAll { $0 == sm } } } self.albumList.models.removeAll() self.loadPhotos() } self.resetBottomToolBtnStatus() } } } // MARK: embed album list nav view class ZLEmbedAlbumListNavView: UIView { static let titleViewH: CGFloat = 32 static let arrowH: CGFloat = 20 var title: String { didSet { self.albumTitleLabel.text = title self.refreshTitleViewFrame() } } var navBlurView: UIVisualEffectView? var titleBgControl: UIControl! var albumTitleLabel: UILabel! var arrow: UIImageView! var cancelBtn: UIButton! var selectAlbumBlock: ( () -> Void )? var cancelBlock: ( () -> Void )? init(title: String) { self.title = title super.init(frame: .zero) self.setupUI() } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } override func layoutSubviews() { super.layoutSubviews() var insets = UIEdgeInsets(top: 20, left: 0, bottom: 0, right: 0) if #available(iOS 11.0, *) { insets = self.safeAreaInsets } self.refreshTitleViewFrame() let cancelBtnW = localLanguageTextValue(.cancel).boundingRect(font: ZLLayout.navTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 44)).width self.cancelBtn.frame = CGRect(x: insets.left+20, y: insets.top, width: cancelBtnW, height: 44) } func refreshTitleViewFrame() { var insets = UIEdgeInsets(top: 20, left: 0, bottom: 0, right: 0) if #available(iOS 11.0, *) { insets = self.safeAreaInsets } self.navBlurView?.frame = self.bounds let albumTitleW = min(self.bounds.width / 2, self.title.boundingRect(font: ZLLayout.navTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 44)).width) let titleBgControlW = albumTitleW + ZLEmbedAlbumListNavView.arrowH + 20 UIView.animate(withDuration: 0.25) { self.titleBgControl.frame = CGRect(x: (self.frame.width-titleBgControlW)/2, y: insets.top+(44-ZLEmbedAlbumListNavView.titleViewH)/2, width: titleBgControlW, height: ZLEmbedAlbumListNavView.titleViewH) self.albumTitleLabel.frame = CGRect(x: 10, y: 0, width: albumTitleW, height: ZLEmbedAlbumListNavView.titleViewH) self.arrow.frame = CGRect(x: self.albumTitleLabel.frame.maxX+5, y: (ZLEmbedAlbumListNavView.titleViewH-ZLEmbedAlbumListNavView.arrowH)/2.0, width: ZLEmbedAlbumListNavView.arrowH, height: ZLEmbedAlbumListNavView.arrowH) } } func setupUI() { self.backgroundColor = .navBarColor if let effect = ZLPhotoConfiguration.default().navViewBlurEffect { self.navBlurView = UIVisualEffectView(effect: effect) self.addSubview(self.navBlurView!) } self.titleBgControl = UIControl() self.titleBgControl.backgroundColor = .navEmbedTitleViewBgColor self.titleBgControl.layer.cornerRadius = ZLEmbedAlbumListNavView.titleViewH / 2 self.titleBgControl.layer.masksToBounds = true self.titleBgControl.addTarget(self, action: #selector(titleBgControlClick), for: .touchUpInside) self.addSubview(titleBgControl) self.albumTitleLabel = UILabel() self.albumTitleLabel.textColor = .navTitleColor self.albumTitleLabel.font = ZLLayout.navTitleFont self.albumTitleLabel.text = self.title self.albumTitleLabel.textAlignment = .center self.titleBgControl.addSubview(self.albumTitleLabel) self.arrow = UIImageView(image: getImage("zl_downArrow")) self.arrow.clipsToBounds = true self.arrow.contentMode = .scaleAspectFill self.titleBgControl.addSubview(self.arrow) self.cancelBtn = UIButton(type: .custom) self.cancelBtn.titleLabel?.font = ZLLayout.navTitleFont self.cancelBtn.setTitle(localLanguageTextValue(.cancel), for: .normal) self.cancelBtn.setTitleColor(.navTitleColor, for: .normal) self.cancelBtn.addTarget(self, action: #selector(cancelBtnClick), for: .touchUpInside) self.addSubview(self.cancelBtn) } @objc func titleBgControlClick() { self.selectAlbumBlock?() if self.arrow.transform == .identity { UIView.animate(withDuration: 0.25) { self.arrow.transform = CGAffineTransform(rotationAngle: .pi) } } else { UIView.animate(withDuration: 0.25) { self.arrow.transform = .identity } } } @objc func cancelBtnClick() { self.cancelBlock?() } func reset() { UIView.animate(withDuration: 0.25) { self.arrow.transform = .identity } } } // MARK: external album list nav view class ZLExternalAlbumListNavView: UIView { let title: String var navBlurView: UIVisualEffectView? var backBtn: UIButton! var albumTitleLabel: UILabel! var cancelBtn: UIButton! var backBlock: ( () -> Void )? var cancelBlock: ( () -> Void )? init(title: String) { self.title = title super.init(frame: .zero) self.setupUI() } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } override func layoutSubviews() { super.layoutSubviews() var insets = UIEdgeInsets(top: 20, left: 0, bottom: 0, right: 0) if #available(iOS 11.0, *) { insets = self.safeAreaInsets } self.navBlurView?.frame = self.bounds self.backBtn.frame = CGRect(x: insets.left, y: insets.top, width: 60, height: 44) let albumTitleW = min(self.bounds.width / 2, self.title.boundingRect(font: ZLLayout.navTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 44)).width) self.albumTitleLabel.frame = CGRect(x: (self.frame.width-albumTitleW)/2, y: insets.top, width: albumTitleW, height: 44) let cancelBtnW = localLanguageTextValue(.cancel).boundingRect(font: ZLLayout.navTitleFont, limitSize: CGSize(width: CGFloat.greatestFiniteMagnitude, height: 44)).width + 40 self.cancelBtn.frame = CGRect(x: self.frame.width-insets.right-cancelBtnW, y: insets.top, width: cancelBtnW, height: 44) } func setupUI() { self.backgroundColor = .navBarColor if let effect = ZLPhotoConfiguration.default().navViewBlurEffect { self.navBlurView = UIVisualEffectView(effect: effect) self.addSubview(self.navBlurView!) } self.backBtn = UIButton(type: .custom) self.backBtn.setImage(getImage("zl_navBack"), for: .normal) self.backBtn.imageEdgeInsets = UIEdgeInsets(top: 0, left: -10, bottom: 0, right: 0) self.backBtn.addTarget(self, action: #selector(backBtnClick), for: .touchUpInside) self.addSubview(self.backBtn) self.albumTitleLabel = UILabel() self.albumTitleLabel.textColor = .navTitleColor self.albumTitleLabel.font = ZLLayout.navTitleFont self.albumTitleLabel.text = self.title self.albumTitleLabel.textAlignment = .center self.addSubview(self.albumTitleLabel) self.cancelBtn = UIButton(type: .custom) self.cancelBtn.titleLabel?.font = ZLLayout.navTitleFont self.cancelBtn.setTitle(localLanguageTextValue(.cancel), for: .normal) self.cancelBtn.setTitleColor(.navTitleColor, for: .normal) self.cancelBtn.addTarget(self, action: #selector(cancelBtnClick), for: .touchUpInside) self.addSubview(self.cancelBtn) } @objc func backBtnClick() { self.backBlock?() } @objc func cancelBtnClick() { self.cancelBlock?() } } class ZLLimitedAuthorityTipsView: UIView { static let height: CGFloat = 70 var icon: UIImageView! var tipsLabel: UILabel! var arrow: UIImageView! override init(frame: CGRect) { super.init(frame: frame) self.icon = UIImageView(image: getImage("zl_warning")) self.addSubview(self.icon) self.tipsLabel = UILabel() self.tipsLabel.font = getFont(14) self.tipsLabel.text = localLanguageTextValue(.unableToAccessAllPhotos) self.tipsLabel.textColor = .bottomToolViewBtnDisableTitleColor self.tipsLabel.numberOfLines = 2 self.tipsLabel.lineBreakMode = .byTruncatingTail self.tipsLabel.adjustsFontSizeToFitWidth = true self.tipsLabel.minimumScaleFactor = 0.5 self.addSubview(self.tipsLabel) self.arrow = UIImageView(image: getImage("zl_right_arrow")) self.addSubview(self.arrow) let tap = UITapGestureRecognizer(target: self, action: #selector(tapAction)) self.addGestureRecognizer(tap) } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } override func layoutSubviews() { super.layoutSubviews() self.icon.frame = CGRect(x: 18, y: (ZLLimitedAuthorityTipsView.height - 25) / 2, width: 25, height: 25) self.tipsLabel.frame = CGRect(x: 55, y: (ZLLimitedAuthorityTipsView.height - 40) / 2, width: self.frame.width-55-30, height: 40) self.arrow.frame = CGRect(x: self.frame.width-25, y: (ZLLimitedAuthorityTipsView.height - 12) / 2, width: 12, height: 12) } @objc func tapAction() { guard let url = URL(string: UIApplication.openSettingsURLString) else { return } if UIApplication.shared.canOpenURL(url) { UIApplication.shared.open(url, options: [:], completionHandler: nil) } } }