import Cocoa import SwiftSoup // Show an informational alert public func infoAlert(_ message: String, _ extra: String? = nil, style: NSAlert.Style = NSAlert.Style.informational) { let alert = NSAlert() alert.messageText = message if extra != nil { alert.informativeText = extra! } alert.alertStyle = style alert.runModal() } // Show an informational alert and then quit public func quitAlert(_ message: String, _ extra: String? = nil) { infoAlert(message, "The application will now quit.", style: NSAlert.Style.critical) NSApp.terminate(nil) } @NSApplicationMain class AppDelegate: NSObject, NSApplicationDelegate { var mainStoryboard: NSStoryboard! var abtController: NSWindowController! // Where the official R installs go let macos_r_framework_dir = "/Library/Frameworks/R.framework/Versions" // Get the bar setup let statusItem = NSStatusBar.system.statusItem(withLength: NSStatusItem.variableLength) let statusMenu = NSMenu() let quitItem = NSMenuItem(title: NSLocalizedString("Quit", comment: "Quit menu item"), action: #selector(NSApp.terminate), keyEquivalent: "q") override init() { super.init() statusMenu.delegate = self // dial by IconMark from the Noun Project statusItem.button?.image = #imageLiteral(resourceName: "RSwitch") statusItem.menu = statusMenu mainStoryboard = NSStoryboard(name: "Main", bundle: nil) abtController = (mainStoryboard.instantiateController(withIdentifier: "aboutPanelController") as! NSWindowController) } func applicationDidFinishLaunching(_ aNotification: Notification) { } func applicationWillTerminate(_ aNotification: Notification) { // Insert code here to tear down app } // The core worker function. Receives the basename of the selected directory // then removes the current alias and creates the new one. @objc func handleSwitch(_ sender: NSMenuItem?) { let fm = FileManager.default; let title = sender?.title do { try fm.removeItem(atPath: macos_r_framework_dir + "/" + "Current") } catch { infoAlert("Failed to remove 'Current' alias", macos_r_framework_dir + "/" + "Current") } do { try fm.createSymbolicLink( at: NSURL(fileURLWithPath: macos_r_framework_dir + "/" + "Current") as URL, withDestinationURL: NSURL(fileURLWithPath: macos_r_framework_dir + "/" + title!) as URL ) } catch { infoAlert("Failed to create alias for " + macos_r_framework_dir + "/" + title!) } } // browse macOS dev page @objc func browse_r_macos_dev_page(_ sender: NSMenuItem?) { let url = URL(string: "https://mac.r-project.org/")! NSWorkspace.shared.open(url) } // browse macOS dev page @objc func browse_r_macos_cran_page(_ sender: NSMenuItem?) { let url = URL(string: "https://cran.rstudio.org/bin/macosx/")! NSWorkspace.shared.open(url) } // browse macOS dev page @objc func browse_r_sig_mac_page(_ sender: NSMenuItem?) { let url = URL(string: "https://stat.ethz.ch/pipermail/r-sig-mac/")! NSWorkspace.shared.open(url) } // browse RStudio macOS Dailies @objc func browse_rstudio_mac_dailies_page(_ sender: NSMenuItem?) { let url = URL(string: "https://dailies.rstudio.com/rstudio/oss/mac/")! NSWorkspace.shared.open(url) } // Show about dialog @objc func about(_ sender: NSMenuItem?) { abtController.showWindow(self) } // Show the framework dir in a new Finder window @objc func openFrameworksDir(_ sender: NSMenuItem?) { NSWorkspace.shared.openFile(macos_r_framework_dir, withApplication: "Finder") } // Download latest rstudio daily build @objc func download_latest_rstudio(_ sender: NSMenuItem?) { let url = URL(string: "https://dailies.rstudio.com/rstudio/oss/mac/") do { let html = try String.init(contentsOf: url!) let document = try SwiftSoup.parse(html) let link = try document.select("td > a").first! let href = try link.attr("href") let dlurl = URL(string: href)! let dldir = FileManager.default.urls(for: .downloadsDirectory, in: .userDomainMask).first! var dlfile = dldir dlfile.appendPathComponent(dlurl.lastPathComponent) let task = URLSession.shared.downloadTask(with: dlurl) { (tempURL, response, error) in if let tempURL = tempURL, error == nil { if ((response as? HTTPURLResponse)?.statusCode == 200) { do { try FileManager.default.copyItem(at: tempURL, to: dlfile) NSWorkspace.shared.openFile(dldir.path, withApplication: "Finder") DispatchQueue.main.async { infoAlert("Download of latest RStudio daily successful.") } } catch { DispatchQueue.main.async { infoAlert("Error downloading and saving latest RStudio daily.") } } } else { DispatchQueue.main.async { infoAlert("Latest RStudio daily not found.") } } } else { DispatchQueue.main.async { infoAlert("Error downloading latest RStudio daily.") } } } task.resume() } catch { DispatchQueue.main.async { infoAlert("Error downloading latrest RStudio daily.") } } } // Download latest r-devel tarball @objc func download_latest_tarball(_ sender: NSMenuItem?) { let url = URL(string: "https://mac.r-project.org/el-capitan/R-devel/R-devel-el-capitan-sa-x86_64.tar.gz")! let dldir = FileManager.default.urls(for: .downloadsDirectory, in: .userDomainMask).first! var dlfile = dldir dlfile.appendPathComponent("R-devel-el-capitan-sa-x86_64.tar.gz") let task = URLSession.shared.downloadTask(with: url) { (tempURL, response, error) in if let tempURL = tempURL, error == nil { if ((response as? HTTPURLResponse)?.statusCode == 200) { do { try FileManager.default.copyItem(at: tempURL, to: dlfile) NSWorkspace.shared.openFile(dldir.path, withApplication: "Finder") DispatchQueue.main.async { infoAlert("Download of latest r-devel successful.") } } catch { DispatchQueue.main.async { infoAlert("Error downloading and saving latest r-devel .") } } } else { DispatchQueue.main.async { infoAlert("Latest r-devel file not found.") } } } else { DispatchQueue.main.async { infoAlert("Error downloading latest r-devel .") } } } task.resume() } } extension AppDelegate: NSMenuDelegate { func menuWillOpen(_ menu: NSMenu) { if (menu != self.statusMenu) { return } // clear the menu menu.removeAllItems() // add selection to open frameworks dir in Finder menu.addItem(NSMenuItem(title: "Open R Frameworks Directory", action: #selector(openFrameworksDir), keyEquivalent: "" )) menu.addItem(NSMenuItem.separator()) // populate installed versions let fm = FileManager.default var targetPath:String? = nil do { // gets a directory listing let entries = try fm.contentsOfDirectory(atPath: macos_r_framework_dir) // retrieves all versions (excludes hidden files and the Current alias let versions = entries.sorted().filter { !($0.hasPrefix(".")) && !($0 == "Current") } let hasCurrent = entries.filter { $0 == "Current" } // if there was a Current alias (prbly shld alert if not) if (hasCurrent.count > 0) { // get where Current points to let furl = NSURL(fileURLWithPath: macos_r_framework_dir + "/" + "Current") if (furl.fileReferenceURL() != nil) { do { let fdat = try NSURL(resolvingAliasFileAt: furl as URL, options: []) targetPath = fdat.lastPathComponent! } catch { targetPath = furl.path } } // populate menu items with all installed R versions, ensuring we // put a checkbox next to the one that is Current var i = 1 for version in versions { let keynum = (i < 10) ? String(i) : "" let item = NSMenuItem(title: version, action: #selector(handleSwitch), keyEquivalent: keynum) item.isEnabled = true if (version == targetPath) { item.state = NSControl.StateValue.on } item.representedObject = version menu.addItem(item) i = i + 1 } } } catch { quitAlert("Failed to list contents of R framework directory. You either do not have R installed or have incorrect permissions set on " + macos_r_framework_dir) } // Add items to download latest r-devel tarball and latest macOS daily menu.addItem(NSMenuItem.separator()) menu.addItem(NSMenuItem(title: NSLocalizedString("Download latest R-devel tarball", comment: "Download latest tarball item"), action: #selector(download_latest_tarball), keyEquivalent: "")) menu.addItem(NSMenuItem(title: NSLocalizedString("Download latest RStudio daily build", comment: "Download latest RStudio item"), action: #selector(download_latest_rstudio), keyEquivalent: "")) // Add items to open variosu R for macOS pages menu.addItem(NSMenuItem.separator()) menu.addItem(NSMenuItem(title: NSLocalizedString("Open R for macOS Developers Page…", comment: "Open macOS Dev Page item"), action: #selector(browse_r_macos_dev_page), keyEquivalent: "")) menu.addItem(NSMenuItem(title: NSLocalizedString("Open R for macOS CRAN Page…", comment: "Open macOS CRAN Page item"), action: #selector(browse_r_macos_cran_page), keyEquivalent: "")) menu.addItem(NSMenuItem(title: NSLocalizedString("Open R-SIG-Mac Archives Page…", comment: "Open R-SIG-Mac Page item"), action: #selector(browse_r_sig_mac_page), keyEquivalent: "")) menu.addItem(NSMenuItem(title: NSLocalizedString("Open RStudio macOS Dailies Page…", comment: "Open RStudio macOS Dailies Page item"), action: #selector(browse_rstudio_mac_dailies_page), keyEquivalent: "")) // Add a About item menu.addItem(NSMenuItem.separator()) menu.addItem(NSMenuItem(title: NSLocalizedString("About RSwitch…", comment: "About menu item"), action: #selector(about), keyEquivalent: "")) // Add a Quit item menu.addItem(NSMenuItem.separator()) menu.addItem(quitItem) } }