2 // GameViewController.swift
5 // Created by Hori,Masaki on 2016/12/31.
6 // Copyright © 2016年 Hori,Masaki. All rights reserved.
13 fileprivate extension Selector {
15 static let reloadContent = #selector(GameViewController.reloadContent(_:))
16 static let deleteCacheAndReload = #selector(GameViewController.deleteCacheAndReload(_:))
17 static let screenShot = #selector(GameViewController.screenShot(_:))
20 final class GameViewController: NSViewController {
22 private static let gamePageURL = "http://www.dmm.com/netgame/social/-/gadgets/=/app_id=854854/"
23 private static let loginPageURLPrefix = "https://www.dmm.com/my/-/login/=/"
25 @IBOutlet var webView: WebView!
27 override var nibName: String! {
29 return "GameViewController"
32 fileprivate var flashTopLeft = NSPoint(x: 2600, y: 1445)
33 private var clipView: NSClipView {
35 return view as! NSClipView // swiftlint:disable:this force_cast
38 override func viewDidLoad() {
42 clipView.documentView = webView
46 webView.mainFrame.frameView.allowsScrolling = false
48 webView.applicationNameForUserAgent = AppDelegate.shared.appNameForUserAgent
49 webView.mainFrameURL = GameViewController.gamePageURL
54 webView.superview?.scroll(flashTopLeft)
57 @IBAction func reloadContent(_ sender: AnyObject?) {
59 guard let _ = webView.mainFrameURL
61 webView.mainFrameURL = GameViewController.gamePageURL
65 // ゲームページでない場合はゲームページを表示する
66 if webView.mainFrameURL != GameViewController.gamePageURL {
68 webView.mainFrameURL = GameViewController.gamePageURL
72 if webView.mainFrameURL.hasPrefix(GameViewController.loginPageURLPrefix) {
74 webView.reload(sender)
81 let prevDate = UserDefaults.standard.prevReloadDate
82 if let prevDate = prevDate {
84 let now = Date(timeIntervalSinceNow: 0.0)
85 if now.timeIntervalSince(prevDate) < 1 * 60 {
87 let untilDate = prevDate.addingTimeInterval(1 * 60)
88 let date = DateFormatter.localizedString(from: untilDate, dateStyle: .none, timeStyle: .medium)
90 alert.messageText = NSLocalizedString("Reload interval is too short?", comment: "")
91 let format = NSLocalizedString("Reload interval is too short.\nWait until %@.", comment: "")
92 alert.informativeText = String(format: format, date)
99 webView.reload(sender)
101 UserDefaults.standard.prevReloadDate = Date(timeIntervalSinceNow: 0.0)
104 @IBAction func deleteCacheAndReload(_ sender: AnyObject?) {
106 let panel = ProgressPanel()
108 guard let window = view.window,
109 let panelWindow = panel.window
113 panel.message = NSLocalizedString("Deleting caches...", comment: "Deleting caches...")
116 window.beginSheet(panelWindow) { _ in NSSound(named: "Submarine")?.play() }
118 AppDelegate.shared.clearCache()
120 window.endSheet(panelWindow)
123 @IBAction func screenShot(_ sender: AnyObject?) {
125 let frame = webView.visibleRect
126 let screenshotBorder = UserDefaults.standard.screenShotBorderWidth
127 let f = frame.insetBy(dx: -screenshotBorder, dy: -screenshotBorder)
129 guard let rep = webView.bitmapImageRepForCachingDisplay(in: f) else { return }
131 webView.cacheDisplay(in: frame, to: rep)
132 AppDelegate.shared.registerScreenshot(rep, fromOnScreen: .zero)
135 override func validateMenuItem(_ menuItem: NSMenuItem) -> Bool {
137 if menuItem.action == .reloadContent {
139 guard let _ = webView.mainFrame,
140 let frameURL = webView.mainFrameURL
144 case GameViewController.gamePageURL:
145 menuItem.title = NSLocalizedString("Reload", comment: "Reload menu, reload")
147 case let s where s.hasPrefix(GameViewController.loginPageURLPrefix):
148 menuItem.title = NSLocalizedString("Reload", comment: "Reload menu, reload")
151 menuItem.title = NSLocalizedString("Back To Game", comment: "Reload menu, back to game")
157 if menuItem.action == .deleteCacheAndReload {
162 if menuItem.action == .screenShot {
171 extension GameViewController: WebFrameLoadDelegate, WebUIDelegate {
173 private static let excludeMenuItemTag = [
174 WebMenuItemTagOpenLinkInNewWindow,
175 WebMenuItemTagDownloadLinkToDisk,
176 WebMenuItemTagOpenImageInNewWindow,
177 WebMenuItemTagOpenFrameInNewWindow,
178 WebMenuItemTagGoBack,
179 WebMenuItemTagGoForward,
184 func webView(_ sender: WebView!, didFinishLoadFor frame: WebFrame!) {
186 guard let path = frame.dataSource?.initialRequest.url?.path
189 let handler: (JSContext?, JSValue?) -> Void = { (_, exception) in
191 if let exception = exception {
193 print("Caught exception in evaluteScript -> \(exception)")
197 print("Caught exception in evaluteScript")
202 if path.hasSuffix("gadgets/ifr") {
204 guard let context = frame.javaScriptContext
207 context.exceptionHandler = handler
208 context.evaluateScript(
209 ["var emb = document.getElementById('flashWrap');",
210 "var rect = emb.getBoundingClientRect();",
211 "var atop = rect.top;",
212 "var aleft = rect.left;"]
215 let top = context.objectForKeyedSubscript("atop").toDouble()
216 let left = context.objectForKeyedSubscript("aleft").toDouble()
217 flashTopLeft = NSPoint(x: CGFloat(left), y: webView.frame.size.height - CGFloat(top) - 480)
220 if path.hasSuffix("app_id=854854") {
222 guard let context = frame.javaScriptContext
225 context.exceptionHandler = handler
226 context.evaluateScript(
227 ["var iframe = document.getElementById('game_frame');",
228 "var validIframe = 0;",
231 " var rect = iframe.getBoundingClientRect();",
232 " var atop = rect.top;",
233 " var aleft = rect.left;",
237 let validIframe = context.objectForKeyedSubscript("validIframe").toInt32()
239 guard validIframe != 0
242 let top = context.objectForKeyedSubscript("atop").toDouble()
243 let left = context.objectForKeyedSubscript("aleft").toDouble()
244 flashTopLeft = NSPoint(x: flashTopLeft.x + CGFloat(left), y: flashTopLeft.y - CGFloat(top))
250 func webView(_ sender: WebView!, contextMenuItemsForElement element: [AnyHashable: Any]!, defaultMenuItems: [Any]!) -> [Any]! {
252 guard let menuItems = defaultMenuItems as? [NSMenuItem]
255 return menuItems.filter { !GameViewController.excludeMenuItemTag.contains($0.tag) }