2 // ScreenshotEditorViewController.swift
5 // Created by Hori,Masaki on 2016/12/29.
6 // Copyright © 2016年 Hori,Masaki. All rights reserved.
11 private struct EditedImage {
13 var editedImage: NSImage
16 init(image: NSImage, url: URL) {
23 final class TrimRectInformation: NSObject {
25 @objc private(set) var name: String
26 private(set) var rect: NSRect
28 fileprivate init(name: String, rect: NSRect) {
35 private extension Selector {
37 static let done = #selector(ScreenshotEditorViewController.done(_:))
40 final class ScreenshotEditorViewController: BridgeViewController {
42 @objc let trimInfo: [TrimRectInformation]
44 override init(nibName: NSNib.Name?, bundle: Bundle?) {
47 TrimRectInformation(name: "Status", rect: NSRect(x: 328, y: 13, width: 470, height: 365)),
48 TrimRectInformation(name: "List", rect: NSRect(x: 362, y: 15, width: 438, height: 368)),
49 TrimRectInformation(name: "AirplaneBase", rect: NSRect(x: 575, y: 13, width: 225, height: 358))
51 currentTrimInfo = trimInfo[0]
53 super.init(nibName: ScreenshotEditorViewController.nibName, bundle: nil)
56 required init?(coder: NSCoder) {
58 fatalError("init(coder:) has not been implemented")
62 @IBOutlet weak var tiledImageView: TiledImageView!
63 @IBOutlet weak var doneButton: NSButton!
65 @objc var columnCount: Int {
67 get { return tiledImageView.columnCount }
69 tiledImageView.columnCount = newValue
70 UserDefaults.standard[.screenshotEditorColumnCount] = newValue
76 return tiledImageView.image
79 @objc dynamic var currentTrimInfoIndex: Int {
81 get { return realiesCurrentTrimInforIndex }
83 guard 0..<trimInfo.count ~= newValue else { return }
85 realiesCurrentTrimInforIndex = newValue
86 currentTrimInfo = trimInfo[newValue]
90 private var editedImage: NSImage?
91 private var currentSelection: [ScreenshotInformation] = []
92 private var editedImages: [EditedImage] = []
93 private var realiesCurrentTrimInforIndex = UserDefaults.standard[.scrennshotEditorType]
94 private var currentTrimInfo: TrimRectInformation {
101 if $0.name != currentTrimInfo.name { return false }
102 return $0.rect == currentTrimInfo.rect
104 .map { UserDefaults.standard[.scrennshotEditorType] = $0 }
108 private var selectionObservation: NSKeyValueObservation?
110 var completeHandler: ((NSImage?) -> Void)?
112 override func viewDidLoad() {
116 selectionObservation = arrayController.observe(\NSArrayController.selectionIndexes) { [weak self] (_, _) in
118 self?.updateSelections()
121 currentTrimInfoIndex = UserDefaults.standard[.scrennshotEditorType]
125 override func viewWillAppear() {
127 doneButton.action = .done
130 private func updateSelections() {
132 guard let selection = arrayController.selectedObjects as? [ScreenshotInformation] else { return }
134 if selection == currentSelection { return }
136 let removed: [ScreenshotInformation] = currentSelection.flatMap {
138 selection.contains($0) ? nil : $0
141 let appended: [ScreenshotInformation] = selection.flatMap {
143 currentSelection.contains($0) ? nil : $0
148 removeEditedImage(url: $0.url)
153 appendEditedImage(url: $0.url)
156 currentSelection = selection
160 private func removeEditedImage(url: URL) {
163 .index { $0.url == url }
164 .map { editedImages.remove(at: $0) }
167 private func appendEditedImage(url: URL) {
169 NSImage(contentsOf: url)
170 .flatMap { EditedImage(image: $0, url: url) }
171 .map { editedImages.append($0) }
174 private func makeEditedImage() {
176 guard !editedImages.isEmpty else {
178 tiledImageView.images = []
182 DispatchQueue(label: "makeTrimedImage queue").async {
184 let images: [NSImage] = self.editedImages.flatMap {
186 guard let originalImage = NSImage(contentsOf: $0.url) else { return nil }
188 let trimedImage = NSImage(size: self.currentTrimInfo.rect.size)
190 trimedImage.lockFocus()
191 originalImage.draw(at: .zero,
192 from: self.currentTrimInfo.rect,
195 trimedImage.unlockFocus()
200 DispatchQueue.main.async {
202 self.tiledImageView.images = images
207 @IBAction func done(_ sender: AnyObject?) {
209 completeHandler?(self.image)
213 extension ScreenshotEditorViewController: NibLoadable {}