// // ThemeManagerVC.swift // ThemeManager // // Created by vector sigma on 07/01/2020. // Copyright © 2020 vectorsigma. All rights reserved. // import Cocoa import WebKit final class GradientView : NSView { override func draw(_ dirtyRect: NSRect) { super.draw(dirtyRect) /* let up = NSColor(red: 48 / 255, green: 35 / 255, blue: 255 / 255, alpha: 0.5) let down = NSColor(red: 150 / 255, green: 158 / 255, blue: 215 / 255, alpha: 0.5) let grad = NSGradient(colors: [down, up]) grad?.draw(in: dirtyRect, angle: 45)*/ } } final class ThemeManagerVC: NSViewController, NSTableViewDelegate, NSTableViewDataSource, WebFrameLoadDelegate, WebUIDelegate, NSComboBoxDelegate, NSComboBoxDataSource { var targetVolume : String? = nil var themes : [String] = [String]() var manager : ThemeManager? @IBOutlet var installedThemesCheckBox : NSButton! @IBOutlet var webView : WebView! @IBOutlet var sidebar : NSTableView! @IBOutlet var nameBox : NSComboBox! @IBOutlet var authorField : NSTextField! @IBOutlet var infoField : NSTextField! @IBOutlet var targetPop : FWPopUpButton! @IBOutlet var spinner : NSProgressIndicator! @IBOutlet var installButton : NSButton! @IBOutlet var unistallButton : NSButton! @IBOutlet var optimizeButton : NSButton! var isPngTheme : Bool = false var loaded : Bool = false var showInstalled : Bool = false var isBusy : Bool = false override func awakeFromNib() { super.awakeFromNib() if !self.loaded { if #available(OSX 10.10, *) {} else { self.viewDidLoad() } self.loaded = true } } override func viewDidLoad() { if #available(OSX 10.10, *) { super.viewDidLoad() } self.loaded = true self.view.window?.title = self.view.window!.title.locale let settingVC = AppSD.settingsWC?.contentViewController as? SettingsViewController settingVC?.themeUserCBox.isEnabled = false settingVC?.themeRepoField.isEnabled = false self.webView.drawsBackground = false self.webView.uiDelegate = self self.webView.frameLoadDelegate = self self.webView.mainFrame.frameView.allowsScrolling = false localize(view: self.view) AppSD.isInstallerOpen = true settingVC?.disksPopUp.isEnabled = false settingVC?.updateCloverButton.isEnabled = false settingVC?.unmountButton.isEnabled = false self.nameBox.completes = true self.nameBox.delegate = self self.nameBox.dataSource = self self.nameBox.usesDataSource = true self.sidebar.delegate = self self.sidebar.dataSource = self self.sidebar.backgroundColor = NSColor.clear self.sidebar.enclosingScrollView?.contentView.drawsBackground = false if #available(OSX 10.10, *) { self.sidebar.usesStaticContents = true } self.targetPop.removeAllItems() self.populateTargets() if let bootDevice = findBootPartitionDevice() { if let mountPoint = getMountPoint(from: bootDevice) { for i in self.targetPop.itemArray { if let target = i.representedObject as? String { if bootDevice == target { self.targetPop.select(i) self.targetVolume = mountPoint break } } } } } self.reloadThemes() } func reloadThemes() { self.themes.removeAll() self.sidebar.reloadData() self.showIndexing() let themeManagerIndexDir = NSHomeDirectory().addPath("Library/Application Support/CloverApp/Themeindex/\(AppSD.themeUser)_\(AppSD.themeRepo)") self.manager = ThemeManager(user: AppSD.themeUser, repo: AppSD.themeRepo, basePath: themeManagerIndexDir, indexDir: themeManagerIndexDir, delegate: self) /* 1) immediately load indexed themes (if any). Fast if thumbnails already exists. */ for t in self.manager!.getIndexedThemes() { self.add(theme: t) } /* 2) download thumbnails. Slower if thumbnails didn't exist or themes never indexed. */ self.manager?.getThemes { (t) in let sorted = t.sorted() for t in sorted { DispatchQueue.main.async { self.add(theme: t) } } } } func dataSource() -> [String] { if showInstalled { return AppSD.installedThemes } return self.themes } func numberOfItems(in comboBox: NSComboBox) -> Int { return self.dataSource().count } func comboBox(_ comboBox: NSComboBox, indexOfItemWithStringValue string: String) -> Int { return self.dataSource().firstIndex(of: string) ?? -1 } func comboBox(_ comboBox: NSComboBox, objectValueForItemAt index: Int) -> Any? { return self.dataSource()[index] } func comboBox(_ comboBox: NSComboBox, completedString string: String) -> String? { for theme in dataSource() { if theme.lowercased().hasPrefix(string.lowercased()) { return theme } } return nil } @IBAction func refreshIndexedThemesPressed(_ sender: NSButton!) { if let repoDir = self.manager?.themeManagerIndexDir { // delete old index if fm.fileExists(atPath: repoDir) { try? fm.removeItem(atPath: repoDir) } // re index the repository self.reloadThemes() } else { NSSound.beep() } } @IBAction func showInstalledThemes(_ sender: NSButton!) { DispatchQueue.main.asyncAfter(deadline: .now() + 0.01) { let copy = self.manager!.getIndexedThemes().sorted() // in cases of low downloads self.sidebar.noteNumberOfRowsChanged() AppSD.installedThemes.removeAll() self.themes.removeAll() if sender.state == .on { if (self.targetVolume != nil && fm.fileExists(atPath: self.targetVolume!)) { let themeDir = self.targetVolume!.addPath("EFI/CLOVER/themes") var installed = [String]() for theme in copy { if fm.fileExists(atPath: themeDir.addPath(theme)) { installed.append(theme) } } if fm.fileExists(atPath: themeDir) { do { let userTheme = try fm.contentsOfDirectory(atPath: themeDir) for theme in userTheme { if (fm.fileExists(atPath: themeDir.addPath(theme).addPath("theme.plist")) && fm.fileExists(atPath: themeDir.addPath(theme).addPath("screenshot.png"))) { if !installed.contains(theme) { installed.append(theme) } } else if fm.fileExists(atPath: themeDir.addPath(theme).addPath("theme.svg")) { if !installed.contains(theme) { installed.append(theme) } } } } catch { } } AppSD.installedThemes = installed.sorted() if installed.count == 0 { self.showNoThemes() } } self.showInstalled = true } else { self.themes = copy self.showInstalled = false if copy.count == 0 { self.showNoThemes() } } self.sidebar.reloadData() } } @IBAction func optimizeThemePressed(_ sender: NSButton!) { DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) { var beep : Bool = true let sr = self.sidebar.selectedRow if sr >= 0 { if let v = self.sidebar.view(atColumn: 0, row: sr, makeIfNecessary: false) as? ThemeView { if v.isInstalled { if self.targetPop.indexOfSelectedItem > 0 && self.targetVolume == nil || fm.fileExists(atPath: self.targetVolume!) { let themePath = self.targetVolume!.addPath("EFI/CLOVER/themes").addPath(v.name) if fm.fileExists(atPath: themePath) { beep = false self.isBusy = true self.spinner.startAnimation(nil) self.optimizeButton.isEnabled = false self.installButton.isEnabled = false self.unistallButton.isEnabled = false self.targetPop.isEnabled = false self.nameBox.isEnabled = false self.installedThemesCheckBox.isEnabled = false //return self.manager?.optimizeTheme(at: themePath, completion: { (error) in DispatchQueue.main.async { self.isBusy = false self.spinner.stopAnimation(nil) self.optimizeButton.isEnabled = true self.installButton.isEnabled = true self.unistallButton.isEnabled = true self.targetPop.isEnabled = true self.nameBox.isEnabled = true self.installedThemesCheckBox.isEnabled = true } if (error != nil) { DispatchQueue.main.async { NSSound(named: "Basso")?.play() let alert = NSAlert() alert.messageText = "😱" alert.informativeText = error!.localizedDescription alert.alertStyle = .critical alert.addButton(withTitle: "Ok".locale) alert.beginSheetModal(for: self.view.window!) { (reponse) in } } } else { DispatchQueue.main.async { NSSound(named: "Glass")?.play() } } }) } } } } } if beep { NSSound.beep() } } } func showNoThemes() { self.webView.mainFrame.loadHTMLString( """

\("No themes found".locale)

""", baseURL: Bundle.main.bundleURL) } func showIndexing() { self.webView.drawsBackground = true // https://www.w3schools.com/howto/howto_css_loader.asp self.webView.mainFrame.loadHTMLString("""

Indexing..

""", baseURL: Bundle.main.bundleURL) } @IBAction func unInstallPressed(_ sender: NSButton!) { if AppSD.isInstalling || self.isBusy { NSSound.beep() return } if self.targetPop.indexOfSelectedItem == 0 || self.targetVolume == nil || (self.targetVolume != nil && !fm.fileExists(atPath: self.targetVolume!)) { NSSound.beep() return // this should not happen } let theme = self.nameBox.stringValue let themePath = self.targetVolume!.addPath("EFI/CLOVER/themes").addPath(theme) let sr = self.sidebar.selectedRow if sr >= 0 && sr < self.dataSource().count { self.spinner.startAnimation(nil) do { if fm.fileExists(atPath: themePath) { try fm.removeItem(atPath: themePath) self.unistallButton.isEnabled = false self.unistallButton.animator().isHidden = true NSSound(contentsOfFile: "/System/Library/Components/CoreAudio.component/Contents/SharedSupport/SystemSounds/finder/empty trash.aif", byReference: false)?.play() } self.spinner.stopAnimation(nil) self.showInstalledThemes(self.installedThemesCheckBox) } catch { NSSound(named: "Basso")?.play() let alert = NSAlert() alert.messageText = "Can't remove the theme".locale alert.informativeText = "\(error.localizedDescription)" alert.addButton(withTitle: "OK") alert.runModal() self.spinner.stopAnimation(nil) self.showInstalledThemes(self.installedThemesCheckBox) } } } @IBAction func InstallPressed(_ sender: NSButton!) { if AppSD.isInstalling || self.isBusy { NSSound.beep() return } if self.targetPop.indexOfSelectedItem == 0 || self.targetVolume == nil || (self.targetVolume != nil && !fm.fileExists(atPath: self.targetVolume!)) { NSSound.beep() DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) { self.targetPop.performClick(self.targetPop) } return } let theme = self.nameBox.stringValue let themeDest = self.targetVolume!.addPath("EFI/CLOVER/themes").addPath(theme) let sr = self.sidebar.selectedRow if sr >= 0 && sr < self.dataSource().count { self.isBusy = true self.spinner.startAnimation(nil) self.optimizeButton.isEnabled = false self.installButton.isEnabled = false self.unistallButton.isEnabled = false self.targetPop.isEnabled = false self.nameBox.isEnabled = false self.installedThemesCheckBox.isEnabled = false self.manager?.download(theme: theme, down: .complete, completion: { (path) in if self.manager?.statusError != nil { NSSound(named: "Basso")?.play() DispatchQueue.main.async { let alert = NSAlert() alert.messageText = "Installation failed".locale alert.informativeText = self.manager!.statusError!.localizedDescription alert.addButton(withTitle: "OK") alert.beginSheetModal(for: self.view.window!, completionHandler: { (modalResponse) -> Void in }) } } else { if let themePath = path { try? fm.removeItem(atPath: themeDest) do { try fm.moveItem(atPath: themePath, toPath: themeDest) let ti = ThemeInfo(user: self.manager!.user, repo: self.manager!.repo, optimized: false).archive() try ti?.write(to: URL(fileURLWithPath: themeDest.addPath(kThemeInfoFile))) NSSound(named: "Glass")?.play() } catch { NSSound(named: "Basso")?.play() DispatchQueue.main.async { let alert = NSAlert() alert.messageText = "Installation failed".locale alert.informativeText = "\(error.localizedDescription)" alert.addButton(withTitle: "OK") alert.beginSheetModal(for: self.view.window!, completionHandler: { (modalResponse) -> Void in }) } } } else { NSSound(named: "Basso")?.play() DispatchQueue.main.async { let alert = NSAlert() let advice = "Try to refresh the list by pressing the Refresh button below." alert.messageText = "Installation failed".locale alert.informativeText = "Theme \"\(theme)\" cannot be downloaded.\n\n\(advice)" alert.addButton(withTitle: "OK") alert.beginSheetModal(for: self.view.window!, completionHandler: { (modalResponse) -> Void in }) } } } DispatchQueue.main.async { self.isBusy = false AppSD.isInstalling = false self.spinner.stopAnimation(nil) self.targetPop.isEnabled = true self.nameBox.isEnabled = true self.installedThemesCheckBox.isEnabled = true self.onSelection() } }) } } func addRow(for theme: String) { let index = self.themes.count self.themes.append(theme) self.sidebar.beginUpdates() self.sidebar.insertRows(at: IndexSet(integer: index), withAnimation: .slideUp) self.sidebar.endUpdates() } func add(theme: String) { if self.themes.contains(theme) { return } if let sha = self.manager?.getSha() { let themePath = self.manager!.basePath.addPath(sha).addPath(theme) if fm.fileExists(atPath: themePath.addPath("theme.svg")) { self.addRow(for: theme) } else if (fm.fileExists(atPath: themePath.addPath("screenshot.png")) && fm.fileExists(atPath: themePath.addPath("theme.plist"))) { self.addRow(for: theme) } else { self.manager?.download(theme: theme, down: .thumbnail, completion: { (path) in if path != nil { DispatchQueue.main.async { self.addRow(for: theme) } } }) } } } @IBAction func nameBoxSelected(_ sender: NSComboBox!) { let theme = sender.stringValue if theme.count > 0 && self.dataSource().contains(theme) { let index = self.dataSource().firstIndex(of: theme) self.sidebar.selectRowIndexes(IndexSet(integer: index!), byExtendingSelection: false) self.sidebar.scrollRowToVisible(index!) } else { NSSound.beep() } } func tableView(_ tableView: NSTableView, didAdd rowView: NSTableRowView, forRow row: Int) { if self.sidebar.selectedRow < 0 { self.sidebar.selectRowIndexes(IndexSet(integer: 0), byExtendingSelection: false) } } func numberOfRows(in tableView: NSTableView) -> Int { return self.dataSource().count } func tableView(_ tableView: NSTableView, heightOfRow row: Int) -> CGFloat { return 82 } func tableViewSelectionDidChange(_ notification: Notification) { self.webView.drawsBackground = false if self.isBusy { NSSound.beep() return } self.onSelection() } func onSelection() { self.isPngTheme = false self.optimizeButton.isEnabled = false self.optimizeButton.state = .off let sr = self.sidebar.selectedRow if sr >= 0 { if let v = self.sidebar.view(atColumn: 0, row: sr, makeIfNecessary: false) as? ThemeView { self.installButton.title = v.isUpToDate ? "Install".locale : "Update".locale if v.isInstalled { self.unistallButton.animator().isHidden = false self.unistallButton.isEnabled = true } else { self.unistallButton.animator().isHidden = true self.unistallButton.isEnabled = false } if (self.manager?.exist(theme: v.name))! { self.installButton.isEnabled = true } else { self.installButton.isEnabled = false } if let path = v.imagePath { self.nameBox.stringValue = v.name self.authorField.stringValue = v.author ?? "" self.infoField.stringValue = v.info ?? "" self.webView.mainFrame.loadHTMLString("""
""", baseURL: Bundle.main.bundleURL) if path.hasSuffix(".svg") { let svgStr = try? String(contentsOfFile: path) var author : String? = nil var version : String? = nil var description : String? = nil if let lines = svgStr?.components(separatedBy: .newlines) { /* Version="0.87" Year="2018-2019" Author="Blackosx" Description="Vector version of BGM (Based on Clovy theme file structure)" */ for line in lines { if (line.range(of: "Version=\"") != nil) { version = line.components(separatedBy: "Version=\"")[1].components(separatedBy: "\"")[0] } } for line in lines { if (line.range(of: "Author=\"") != nil) { author = line.components(separatedBy: "Author=\"")[1].components(separatedBy: "\"")[0] } } for line in lines { if (line.range(of: "Description=\"") != nil) { description = line.components(separatedBy: "Description=\"")[1].components(separatedBy: "\"")[0] } } } self.authorField.stringValue = author ?? "" self.infoField.stringValue = "\(description ?? "?"), v\(version ?? "?")" } else if path.hasSuffix(".png") { // get theme.plist self.isPngTheme = true let plistPath = "\((path as NSString).deletingLastPathComponent)/theme.plist" let plist = NSDictionary(contentsOfFile: plistPath) self.authorField.stringValue = (plist?.object(forKey: "Author") as? String) ?? "" self.infoField.stringValue = (plist?.object(forKey: "Description") as? String) ?? "" self.optimizeButton.isEnabled = v.isInstalled && self.isPngTheme } } else { v.load() } } } else { self.installButton.title = "Install".locale } } func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? { if let identifier = tableColumn?.identifier { if identifier.rawValue == "column1" { let count = self.dataSource().count if count == 0 { return nil } if row > (count - 1) { return nil } let tv = ThemeView(manager: self.manager!, name: self.dataSource()[row], row: row) tv.manager.delegate = self return tv } } return nil } func tableView(_ tableView: NSTableView, rowViewForRow row: Int) -> NSTableRowView? { return ThemeTableRowView() } // MARK: find suitable disks func populateTargets() { if AppSD.isInstalling { return } let selected : String? = self.targetPop.selectedItem?.representedObject as? String self.targetPop.removeAllItems() self.targetPop.addItem(withTitle: "Select a disk..".locale) let disks : NSDictionary = getAlldisks() let bootPartition = findBootPartitionDevice() let diskSorted : [String] = (disks.allKeys as! [String]).sorted() for d in diskSorted { let disk : String = d if isWritable(diskOrMtp: disk) && !kBannedMedia.contains(getVolumeName(from: disk) ?? "") { let fs : String = getFS(from: disk)?.lowercased() ?? kNotAvailable.locale let psm : String = getPartitionSchemeMap(from: disk) ?? kNotAvailable.locale let name : String = getVolumeName(from: disk) ?? kNotAvailable.locale let mp : String = getMountPoint(from: disk) ?? kNotAvailable.locale let parentDiskName : String = getMediaName(from: getBSDParent(of: disk) ?? "") ?? kNotAvailable.locale let supportedFS = ["msdos", "fat16", "fat32", "exfat", "hfs"] if supportedFS.contains(fs) { self.targetPop.addItem(withTitle: "\(disk)\t\(name), \("mount point".locale): \(mp), \(fs.uppercased()), \(psm): (\(parentDiskName))") self.targetPop.invalidateIntrinsicContentSize() // get the image if disk == bootPartition { let image : NSImage = NSImage(named: "NSApplicationIcon")!.copy() as! NSImage image.size = NSMakeSize(16, 16) self.targetPop.lastItem?.image = image } else if let image : NSImage = getIconFor(volume: disk) { image.size = NSMakeSize(16, 16) self.targetPop.lastItem?.image = image } self.targetPop.lastItem?.representedObject = disk } } } if (selected != nil) { for item in self.targetPop.itemArray { if let d = item.representedObject as? String { if selected == d { self.targetPop.select(item) break } } } } } @IBAction func targetPopPressed(_ sender: FWPopUpButton!) { if let disk = sender?.selectedItem?.representedObject as? String { if !isMountPoint(path: disk) { self.installButton.isEnabled = false DispatchQueue.global(priority: .background).async(execute: { () -> Void in let cmd = "diskutil mount \(disk)" let msg = String(format: "Clover wants to mount %@", disk) let script = "do shell script \"\(cmd)\" with prompt \"\(msg)\" with administrator privileges" let task = Process() if #available(OSX 10.12, *) { task.launchPath = "/usr/bin/osascript" task.arguments = ["-e", script] } else { task.launchPath = "/usr/sbin/diskutil" task.arguments = ["mount", disk] } task.terminationHandler = { t in DispatchQueue.main.async { self.view.window?.level = .floating self.view.window?.makeKeyAndOrderFront(nil) self.view.window?.level = .normal } if t.terminationStatus == 0 { if isMountPoint(path: disk) { DispatchQueue.main.async { self.targetVolume = getMountPoint(from: disk) self.populateTargets() self.showInstalledThemes(self.installedThemesCheckBox) } } DispatchQueue.main.async { self.installButton.isEnabled = true } } else { DispatchQueue.main.async { NSSound.beep() self.installButton.isEnabled = true self.showInstalledThemes(self.installedThemesCheckBox) } } } task.launch() }) //} } else { self.targetVolume = getMountPoint(from: disk) self.populateTargets() self.showInstalledThemes(self.installedThemesCheckBox) } } else { self.targetVolume = nil self.unistallButton.isEnabled = false self.unistallButton.isHidden = true self.showInstalledThemes(self.installedThemesCheckBox) } } } // MARK: ThemeManager Window controller final class ThemeManagerWC: NSWindowController, NSWindowDelegate { var viewController : NSViewController? = nil override var contentViewController: NSViewController? { get { self.viewController } set { self.viewController = newValue } } class func loadFromNib() -> ThemeManagerWC? { var topLevel: NSArray? = nil Bundle.main.loadNibNamed("ThemeManager", owner: self, topLevelObjects: &topLevel) if (topLevel != nil) { var wc : ThemeManagerWC? = nil for o in topLevel! { if o is ThemeManagerWC { wc = o as? ThemeManagerWC } } for o in topLevel! { if o is ThemeManagerVC { wc?.contentViewController = o as! ThemeManagerVC } } return wc } return nil } func windowShouldClose(_ sender: NSWindow) -> Bool { if AppSD.isInstalling { return false } let settingVC = AppSD.settingsWC?.contentViewController as? SettingsViewController settingVC?.disksPopUp.isEnabled = true settingVC?.updateCloverButton.isEnabled = true settingVC?.searchESPDisks() AppSD.isInstallerOpen = false settingVC?.themeUserCBox.isEnabled = true settingVC?.themeRepoField.isEnabled = true self.window = nil self.close() AppSD.themeManagerWC = nil // remove a strong reference AppSD.setActivationPolicy() return true } } final class ThemeTableRowView: NSTableRowView { override func draw(_ dirtyRect: NSRect) { super.draw(dirtyRect) self.wantsLayer = true self.layer?.backgroundColor = NSColor.clear.cgColor } override var isEmphasized: Bool { set {} get { return false } } override var selectionHighlightStyle: NSTableView.SelectionHighlightStyle { set {} get { return .regular } } override func drawSelection(in dirtyRect: NSRect) { if self.selectionHighlightStyle != .none { let selectionRect = NSInsetRect(self.bounds, 2.5, 2.5) NSColor.green.setFill() //NSColor(calibratedWhite: 0.85, alpha: 0.6).setFill() let selectionPath = NSBezierPath.init(rect: selectionRect) selectionPath.fill() } } }