2 // ScreenshotEditorViewController.swift
5 // Created by Hori,Masaki on 2016/12/29.
6 // Copyright © 2016年 Hori,Masaki. All rights reserved.
11 private struct URLImage {
16 init(image: NSImage, url: URL) {
24 final class TrimRectInformation: NSObject {
26 @objc private(set) var name: String
27 private(set) var rect: NSRect
29 fileprivate init(name: String, rect: NSRect) {
36 final class ScreenshotEditorViewController: BridgeViewController {
38 @objc let trimInfo = [
39 TrimRectInformation(name: "Status", rect: NSRect(x: 328, y: 13, width: 470, height: 365)),
40 TrimRectInformation(name: "List", rect: NSRect(x: 362, y: 15, width: 438, height: 368)),
41 TrimRectInformation(name: "AirplaneBase", rect: NSRect(x: 575, y: 13, width: 225, height: 358))
44 override init(nibName: NSNib.Name?, bundle: Bundle?) {
46 currentTrimInfo = trimInfo[0]
48 super.init(nibName: ScreenshotEditorViewController.nibName, bundle: nil)
51 required init?(coder: NSCoder) {
53 fatalError("init(coder:) has not been implemented")
57 @IBOutlet private weak var tiledImageView: TiledImageView!
58 @IBOutlet private weak var doneButton: NSButton!
60 @objc var columnCount: Int {
62 get { return tiledImageView.columnCount }
64 tiledImageView.columnCount = newValue
65 UserDefaults.standard[.screenshotEditorColumnCount] = newValue
71 return tiledImageView.image
74 @objc dynamic var currentTrimInfoIndex: Int {
76 get { return realiesCurrentTrimInforIndex }
78 guard case 0..<trimInfo.count = newValue else { return }
80 realiesCurrentTrimInforIndex = newValue
81 currentTrimInfo = trimInfo[newValue]
85 private var editedImage: NSImage?
86 private var currentSelection: [ScreenshotInformation] = []
87 private var originalImages: [URLImage] = []
88 private var realiesCurrentTrimInforIndex = UserDefaults.standard[.scrennshotEditorType]
89 private var currentTrimInfo: TrimRectInformation {
96 if $0.name != currentTrimInfo.name { return false }
97 return $0.rect == currentTrimInfo.rect
99 .map { UserDefaults.standard[.scrennshotEditorType] = $0 }
103 private var selectionObservation: NSKeyValueObservation?
105 var completeHandler: ((NSImage?) -> Void)?
107 override func viewDidLoad() {
111 selectionObservation = arrayController.observe(\NSArrayController.selectionIndexes) { [weak self] (_, _) in
113 self?.updateSelections()
116 currentTrimInfoIndex = UserDefaults.standard[.scrennshotEditorType]
120 override func viewWillAppear() {
122 doneButton.action = #selector(ScreenshotEditorViewController.done(_:))
125 private func updateSelections() {
127 guard let selection = arrayController.selectedObjects as? [ScreenshotInformation] else { return }
129 if selection == currentSelection { return }
131 let removed: [ScreenshotInformation] = currentSelection.compactMap {
133 selection.contains($0) ? nil : $0
136 let appended: [ScreenshotInformation] = selection.compactMap {
138 currentSelection.contains($0) ? nil : $0
143 removeOriginalImage(url: $0.url)
148 appendOriginalImage(url: $0.url)
151 currentSelection = selection
155 private func removeOriginalImage(url: URL) {
158 .index { $0.url == url }
159 .map { originalImages.remove(at: $0) }
162 private func appendOriginalImage(url: URL) {
164 NSImage(contentsOf: url)
165 .flatMap { URLImage(image: $0, url: url) }
166 .map { originalImages.append($0) }
169 private func makeTrimedImage() {
171 guard !originalImages.isEmpty else {
173 tiledImageView.images = []
177 DispatchQueue(label: "makeTrimedImage queue").async {
179 let images: [NSImage] = self.originalImages.compactMap {
181 let trimedImage = NSImage(size: self.currentTrimInfo.rect.size)
183 trimedImage.lockFocus()
184 $0.image.draw(at: .zero,
185 from: self.currentTrimInfo.rect,
188 trimedImage.unlockFocus()
193 DispatchQueue.main.async {
195 self.tiledImageView.images = images
200 @IBAction func done(_ sender: AnyObject?) {
202 completeHandler?(self.image)
206 extension ScreenshotEditorViewController: NibLoadable {}