X Tutup
# An example of embedding CEF browser in wxPython on Mac. # Tested with wxPython3.0-osx-3.0.2.0-cocoa-py2.7.dmg which # was downloaded from the wxpython.org website. # IMPORTANT - importing CEF Python # -------------------------------------------------------------- # The cefpython library must be the very first library imported. # This is because CEF was compiled with the tcmalloc memory # allocator which hooks globally and replaces the default # malloc allocator. If memory was allocated using malloc and # then freed using tcmalloc then this would result in random # segmentation faults in an application. See Issue 155 which # is to provide CEF builds on Mac with tcmalloc disabled: # https://github.com/cztomczak/cefpython/issues/155 import ctypes, os, sys libcef_so = os.path.join(os.path.dirname(os.path.abspath(__file__)),\ 'libcef.dylib') if os.path.exists(libcef_so): # Import a local module ctypes.CDLL(libcef_so, ctypes.RTLD_GLOBAL) if 0x02070000 <= sys.hexversion < 0x03000000: import cefpython_py27 as cefpython else: raise Exception("Unsupported python version: %s" % sys.version) else: # Import an installed package from cefpython3 import cefpython import wx import time import re import uuid import platform import inspect import struct g_applicationSettings = None g_browserSettings = None g_switches = None g_countWindows = 0 # Which method to use for message loop processing. # EVT_IDLE - wx application has priority (default) # EVT_TIMER - cef browser has priority # It seems that Flash content behaves better when using a timer. # IMPORTANT! On Linux EVT_IDLE does not work, the events seems to # be propagated only when you move your mouse, which is not the # expected behavior, it is recommended to use EVT_TIMER on Linux, # so set this value to False. USE_EVT_IDLE = False def GetApplicationPath(file=None): import re, os, platform # On Windows after downloading file and calling Browser.GoForward(), # current working directory is set to %UserProfile%. # Calling os.path.dirname(os.path.realpath(__file__)) # returns for eg. "C:\Users\user\Downloads". A solution # is to cache path on first call. if not hasattr(GetApplicationPath, "dir"): if hasattr(sys, "frozen"): dir = os.path.dirname(sys.executable) elif "__file__" in globals(): dir = os.path.dirname(os.path.realpath(__file__)) else: dir = os.getcwd() GetApplicationPath.dir = dir # If file is None return current directory without trailing slash. if file is None: file = "" # Only when relative path. if not file.startswith("/") and not file.startswith("\\") and ( not re.search(r"^[\w-]+:", file)): path = GetApplicationPath.dir + os.sep + file if platform.system() == "Windows": path = re.sub(r"[/\\]+", re.escape(os.sep), path) path = re.sub(r"[/\\]+$", "", path) return path return str(file) def ExceptHook(excType, excValue, traceObject): import traceback, os, time, codecs # This hook does the following: in case of exception write it to # the "error.log" file, display it to the console, shutdown CEF # and exit application immediately by ignoring "finally" (os._exit()). errorMsg = "\n".join(traceback.format_exception(excType, excValue, traceObject)) errorFile = GetApplicationPath("error.log") try: appEncoding = cefpython.g_applicationSettings["string_encoding"] except: appEncoding = "utf-8" if type(errorMsg) == bytes: errorMsg = errorMsg.decode(encoding=appEncoding, errors="replace") try: with codecs.open(errorFile, mode="a", encoding=appEncoding) as fp: fp.write("\n[%s] %s\n" % ( time.strftime("%Y-%m-%d %H:%M:%S"), errorMsg)) except: print("[wxpython.py] WARNING: failed writing to error file: %s" % ( errorFile)) # Convert error message to ascii before printing, otherwise # you may get error like this: # | UnicodeEncodeError: 'charmap' codec can't encode characters errorMsg = errorMsg.encode("ascii", errors="replace") errorMsg = errorMsg.decode("ascii", errors="replace") print("\n"+errorMsg+"\n") cefpython.QuitMessageLoop() cefpython.Shutdown() os._exit(1) class MainFrame(wx.Frame): browser = None mainPanel = None clientHandler = None javascriptExternal = None def __init__(self, url=None): wx.Frame.__init__(self, parent=None, id=wx.ID_ANY, title='wxPython CEF 3 example', size=(800,600)) global g_countWindows g_countWindows += 1 if not url: url = "file://"+GetApplicationPath("wxpython.html") # Test hash in url. # url += "#test-hash" self.CreateMenu() # Cannot attach browser to the main frame as this will cause # the menu not to work. # -- # You also have to set the wx.WANTS_CHARS style for # all parent panels/controls, if it's deeply embedded. self.mainPanel = wx.Panel(self, style=wx.WANTS_CHARS) # Global client callbacks must be set before browser is created. self.clientHandler = ClientHandler() cefpython.SetGlobalClientCallback("OnCertificateError", self.clientHandler._OnCertificateError) cefpython.SetGlobalClientCallback("OnBeforePluginLoad", self.clientHandler._OnBeforePluginLoad) cefpython.SetGlobalClientCallback("OnAfterCreated", self.clientHandler._OnAfterCreated) windowInfo = cefpython.WindowInfo() (width, height) = self.mainPanel.GetClientSizeTuple() windowInfo.SetAsChild(self.mainPanel.GetHandle(), [0, 0, width, height]) # Linux requires adding "file://" for local files, # otherwise /home/some will be replaced as http://home/some self.browser = cefpython.CreateBrowserSync( windowInfo, # If there are problems with Flash you can disable it here, # by disabling all plugins. browserSettings=g_browserSettings, navigateUrl=url) self.clientHandler.mainBrowser = self.browser self.browser.SetClientHandler(self.clientHandler) jsBindings = cefpython.JavascriptBindings( bindToFrames=False, bindToPopups=True) jsBindings.SetFunction("PyPrint", PyPrint) jsBindings.SetProperty("pyProperty", "This was set in Python") jsBindings.SetProperty("pyConfig", ["This was set in Python", {"name": "Nested dictionary", "isNested": True}, [1,"2", None]]) self.javascriptExternal = JavascriptExternal(self.browser) jsBindings.SetObject("external", self.javascriptExternal) jsBindings.SetProperty("sources", GetSources()) self.browser.SetJavascriptBindings(jsBindings) self.Bind(wx.EVT_CLOSE, self.OnClose) if USE_EVT_IDLE: # Bind EVT_IDLE only for the main application frame. self.Bind(wx.EVT_IDLE, self.OnIdle) def CreateMenu(self): filemenu = wx.Menu() filemenu.Append(1, "Open") exit = filemenu.Append(2, "Exit") self.Bind(wx.EVT_MENU, self.OnClose, exit) aboutmenu = wx.Menu() aboutmenu.Append(1, "CEF Python") menubar = wx.MenuBar() menubar.Append(filemenu,"&File") menubar.Append(aboutmenu, "&About") self.SetMenuBar(menubar) def OnClose(self, event): # Remove all CEF browser references so that browser is closed # cleanly. Otherwise there may be issues for example with cookies # not being flushed to disk when closing app immediately # (Issue 158). del self.javascriptExternal.mainBrowser del self.clientHandler.mainBrowser del self.browser # Destroy wx frame, this will complete the destruction of CEF browser self.Destroy() # In wx.chromectrl calling browser.CloseBrowser and/or self.Destroy # may cause crashes when embedding multiple browsers in tab # (Issue 107). In such case instead of calling CloseBrowser/Destroy # try this code: # | self.browser.ParentWindowWillClose() # | event.Skip() global g_countWindows g_countWindows -= 1 if g_countWindows == 0: # On Win/Linux the call to cefpython.Shutdown() is after # app.MainLoop() returns, but on Mac it needs to be here. cefpython.Shutdown() print("[wxpython.py] OnClose: Exiting") wx.GetApp().Exit() def OnIdle(self, event): cefpython.MessageLoopWork() def PyPrint(message): print("[wxpython.py] PyPrint: "+message) class JavascriptExternal: mainBrowser = None stringVisitor = None def __init__(self, mainBrowser): self.mainBrowser = mainBrowser def GoBack(self): self.mainBrowser.GoBack() def GoForward(self): self.mainBrowser.GoForward() def CreateAnotherBrowser(self, url=None): """ TODO: There are errors in the console when closing the window: >> Check failed: window >> Gdk: _gdk_window_destroy_hierarchy: assertion `GDK_IS_WINDOW\ >> (window)' failed >> GLib-GObject: g_object_unref: assertion `G_IS_OBJECT (object)' failed """ frame = MainFrame(url=url) frame.Show() def Print(self, message): print("[wxpython.py] Print: "+message) def TestAllTypes(self, *args): print("[wxpython.py] TestAllTypes: "+str(args)) def ExecuteFunction(self, *args): self.mainBrowser.GetMainFrame().ExecuteFunction(*args) def TestJSCallback(self, jsCallback): print("[wxpython.py] jsCallback.GetFunctionName() = %s"\ % jsCallback.GetFunctionName()) print("[wxpython.py] jsCallback.GetFrame().GetIdentifier() = %s" % \ jsCallback.GetFrame().GetIdentifier()) jsCallback.Call("This message was sent from python using js callback") def TestJSCallbackComplexArguments(self, jsObject): jsCallback = jsObject["myCallback"]; jsCallback.Call(1, None, 2.14, "string", ["list", ["nested list", \ {"nested object":None}]], \ {"nested list next":[{"deeply nested object":1}]}) def TestPythonCallback(self, jsCallback): jsCallback.Call(self.PyCallback) def PyCallback(self, *args): message = "PyCallback() was executed successfully! "\ "Arguments: %s" % str(args) print("[wxpython.py] "+message) self.mainBrowser.GetMainFrame().ExecuteJavascript( "window.alert(\"%s\")" % message) def GetSource(self): # Must keep a strong reference to the StringVisitor object # during the visit. self.stringVisitor = StringVisitor() self.mainBrowser.GetMainFrame().GetSource(self.stringVisitor) def GetText(self): # Must keep a strong reference to the StringVisitor object # during the visit. self.stringVisitor = StringVisitor() self.mainBrowser.GetMainFrame().GetText(self.stringVisitor) def ShowDevTools(self): print("[wxpython.py] external.ShowDevTools called") self.mainBrowser.ShowDevTools() # ------------------------------------------------------------------------- # Cookies # ------------------------------------------------------------------------- cookieVisitor = None def VisitAllCookies(self): # Need to keep the reference alive. self.cookieVisitor = CookieVisitor() cookieManager = self.mainBrowser.GetUserData("cookieManager") if not cookieManager: print("\n[wxpython.py] Cookie manager not yet created! Visit"\ " the cookietester website first and create some cookies") return cookieManager.VisitAllCookies(self.cookieVisitor) def VisitUrlCookies(self): # Need to keep the reference alive. self.cookieVisitor = CookieVisitor() cookieManager = self.mainBrowser.GetUserData("cookieManager") if not cookieManager: print("\n[wxpython.py] Cookie manager not yet created! Visit"\ " the cookietester website first and create some cookies") return cookieManager.VisitUrlCookies( "http://www.html-kit.com/tools/cookietester/", False, self.cookieVisitor) # .www.html-kit.com def SetCookie(self): cookieManager = self.mainBrowser.GetUserData("cookieManager") if not cookieManager: print("\n[wxpython.py] Cookie manager not yet created! Visit"\ "the cookietester website first and create some cookies") return cookie = cefpython.Cookie() cookie.SetName("Created_Via_Python") cookie.SetValue("yeah really") cookieManager.SetCookie("http://www.html-kit.com/tools/cookietester/", cookie) print("\n[wxpython.py] Cookie created! Visit html-kit cookietester to"\ " see it") def DeleteCookies(self): cookieManager = self.mainBrowser.GetUserData("cookieManager") if not cookieManager: print("\n[wxpython.py] Cookie manager not yet created! Visit"\ " the cookietester website first and create some cookies") return cookieManager.DeleteCookies( "http://www.html-kit.com/tools/cookietester/", "Created_Via_Python") print("\n[wxpython.py] Cookie deleted! Visit html-kit cookietester "\ "to see the result") class StringVisitor: def Visit(self, string): print("\n[wxpython.py] StringVisitor.Visit(): string:") print("--------------------------------") print(string) print("--------------------------------") class CookieVisitor: def Visit(self, cookie, count, total, deleteCookie): if count == 0: print("\n[wxpython.py] CookieVisitor.Visit(): total cookies: %s"\ % total) print("\n[wxpython.py] CookieVisitor.Visit(): cookie:") print(" "+str(cookie.Get())) # True to continue visiting cookies return True class ClientHandler: mainBrowser = None # May be None for global client callbacks. def __init__(self): pass # ------------------------------------------------------------------------- # DisplayHandler # ------------------------------------------------------------------------- def OnAddressChange(self, browser, frame, url): print("[wxpython.py] DisplayHandler::OnAddressChange()") print(" url = %s" % url) def OnTitleChange(self, browser, title): print("[wxpython.py] DisplayHandler::OnTitleChange()") print(" title = %s" % title) def OnTooltip(self, browser, textOut): # OnTooltip not yet implemented (both Linux and Windows), # will be fixed in next CEF release, see Issue 783: # https://code.google.com/p/chromiumembedded/issues/detail?id=783 print("[wxpython.py] DisplayHandler::OnTooltip()") print(" text = %s" % textOut[0]) statusMessageCount = 0 def OnStatusMessage(self, browser, value): if not value: # Do not notify in the console about empty statuses. return self.statusMessageCount += 1 if self.statusMessageCount > 3: # Do not spam too much. return print("[wxpython.py] DisplayHandler::OnStatusMessage()") print(" value = %s" % value) def OnConsoleMessage(self, browser, message, source, line): print("[wxpython.py] DisplayHandler::OnConsoleMessage()") print(" message = %s" % message) print(" source = %s" % source) print(" line = %s" % line) # ------------------------------------------------------------------------- # KeyboardHandler # ------------------------------------------------------------------------- def OnPreKeyEvent(self, browser, event, eventHandle, isKeyboardShortcutOut): print("[wxpython.py] KeyboardHandler::OnPreKeyEvent()") def OnKeyEvent(self, browser, event, eventHandle): if event["type"] == cefpython.KEYEVENT_KEYUP: # OnKeyEvent is called twice for F5/Esc keys, with event # type KEYEVENT_RAWKEYDOWN and KEYEVENT_KEYUP. # Normal characters a-z should have KEYEVENT_CHAR. return False print("[wxpython.py] KeyboardHandler::OnKeyEvent()") print(" type=%s" % event["type"]) print(" modifiers=%s" % event["modifiers"]) print(" windows_key_code=%s" % event["windows_key_code"]) print(" native_key_code=%s" % event["native_key_code"]) print(" is_system_key=%s" % event["is_system_key"]) print(" character=%s" % event["character"]) print(" unmodified_character=%s" % event["unmodified_character"]) print(" focus_on_editable_field=%s"\ % event["focus_on_editable_field"]) if platform.system() == "Linux": # F5 if event["native_key_code"] == 71: print("[wxpython.py] F5 pressed, calling"\ " browser.ReloadIgnoreCache()") browser.ReloadIgnoreCache() return True # Escape if event["native_key_code"] == 9: print("[wxpython.py] Esc pressed, calling browser.StopLoad()") browser.StopLoad() return True # F12 if event["native_key_code"] == 96: print("[wxpython.py] F12 pressed, calling"\ " browser.ShowDevTools()") browser.ShowDevTools() return True elif platform.system() == "Windows": # F5 todo # Escape todo pass elif platform.system() == "Darwin": # Cmd+Opt+I if event["modifiers"] == 136 and event["character"] == 94: browser.ShowDevTools() return True # F5 if event["modifiers"] == 0 and event["character"] == 63240: browser.ReloadIgnoreCache() return True # Esc if event["modifiers"] == 0 and event["character"] == 27: browser.StopLoad() return True return False # ------------------------------------------------------------------------- # RequestHandler # ------------------------------------------------------------------------- def OnBeforeBrowse(self, browser, frame, request, isRedirect): print("[wxpython.py] RequestHandler::OnBeforeBrowse()") print(" url = %s" % request.GetUrl()[:100]) return False def OnBeforeResourceLoad(self, browser, frame, request): print("[wxpython.py] RequestHandler::OnBeforeResourceLoad()") print(" url = %s" % request.GetUrl()[:100]) return False def OnResourceRedirect(self, browser, frame, oldUrl, newUrlOut, request): print("[wxpython.py] RequestHandler::OnResourceRedirect()") print(" old url = %s" % oldUrl[:100]) print(" new url = %s" % newUrlOut[0][:100]) def GetAuthCredentials(self, browser, frame, isProxy, host, port, realm, scheme, callback): # This callback is called on the IO thread, thus print messages # may not be visible. print("[wxpython.py] RequestHandler::GetAuthCredentials()") print(" host = %s" % host) print(" realm = %s" % realm) callback.Continue(username="test", password="test") return True def OnQuotaRequest(self, browser, originUrl, newSize, callback): print("[wxpython.py] RequestHandler::OnQuotaRequest()") print(" origin url = %s" % originUrl) print(" new size = %s" % newSize) callback.Continue(True) return True def GetCookieManager(self, browser, mainUrl): # Create unique cookie manager for each browser. # You must set the "unique_request_context_per_browser" # application setting to True for the cookie manager # to work. # Return None to have one global cookie manager for # all CEF browsers. if not browser: # The browser param may be empty in some exceptional # case, see docs. return None cookieManager = browser.GetUserData("cookieManager") if cookieManager: return cookieManager else: print("[wxpython.py] RequestHandler::GetCookieManager():"\ " created cookie manager") cookieManager = cefpython.CookieManager.CreateManager("") if "cache_path" in g_applicationSettings: path = g_applicationSettings["cache_path"] # path = os.path.join(path, "cookies_browser_{}".format( # browser.GetIdentifier())) cookieManager.SetStoragePath(path) browser.SetUserData("cookieManager", cookieManager) return cookieManager def OnProtocolExecution(self, browser, url, allowExecutionOut): # There's no default implementation for OnProtocolExecution on Linux, # you have to make OS system call on your own. You probably also need # to use LoadHandler::OnLoadError() when implementing this on Linux. print("[wxpython.py] RequestHandler::OnProtocolExecution()") print(" url = %s" % url) if url.startswith("magnet:"): print("[wxpython.py] Magnet link allowed!") allowExecutionOut[0] = True def _OnBeforePluginLoad(self, browser, mimeType, pluginUrl, topOriginUrl, info): # This is a global callback set using SetGlobalClientCallback(). # Plugins are loaded on demand, only when website requires it, # the same plugin may be called multiple times. # This callback is called on various threads, thus print messages # may not be visible. print("[wxpython.py] RequestHandler::_OnBeforePluginLoad()") print(" mimeType = %s" % mimeType) print(" pluginUrl = %s" % pluginUrl) print(" topOriginUrl = %s" % topOriginUrl) print(" info.GetName() = %s" % info.GetName()) print(" info.GetPath() = %s" % info.GetPath()) print(" info.GetVersion() = %s" % info.GetVersion()) print(" info.GetDescription() = %s" % info.GetDescription()) # False to allow, True to block plugin. return False def _OnCertificateError(self, certError, requestUrl, callback): # This is a global callback set using SetGlobalClientCallback(). print("[wxpython.py] RequestHandler::_OnCertificateError()") print(" certError = %s" % certError) print(" requestUrl = %s" % requestUrl) if requestUrl.startswith( "https://tv.eurosport.com/do-not-allow"): print(" Not allowed!") return False if requestUrl.startswith( "https://tv.eurosport.com/"): print(" Allowed!") callback.Continue(True) return True return False def OnRendererProcessTerminated(self, browser, status): print("[wxpython.py] RequestHandler::OnRendererProcessTerminated()") statuses = { cefpython.TS_ABNORMAL_TERMINATION: "TS_ABNORMAL_TERMINATION", cefpython.TS_PROCESS_WAS_KILLED: "TS_PROCESS_WAS_KILLED", cefpython.TS_PROCESS_CRASHED: "TS_PROCESS_CRASHED" } statusName = "Unknown" if status in statuses: statusName = statuses[status] print(" status = %s" % statusName) def OnPluginCrashed(self, browser, pluginPath): print("[wxpython.py] RequestHandler::OnPluginCrashed()") print(" plugin path = %s" % pluginPath) # ------------------------------------------------------------------------- # LoadHandler # ------------------------------------------------------------------------- def OnLoadingStateChange(self, browser, isLoading, canGoBack, canGoForward): print("[wxpython.py] LoadHandler::OnLoadingStateChange()") print(" isLoading = %s, canGoBack = %s, canGoForward = %s" \ % (isLoading, canGoBack, canGoForward)) def OnLoadStart(self, browser, frame): print("[wxpython.py] LoadHandler::OnLoadStart()") print(" frame url = %s" % frame.GetUrl()[:100]) def OnLoadEnd(self, browser, frame, httpStatusCode): print("[wxpython.py] LoadHandler::OnLoadEnd()") print(" frame url = %s" % frame.GetUrl()[:100]) # For file:// urls the status code = 0 print(" http status code = %s" % httpStatusCode) # Tests for the Browser object methods self._Browser_LoadUrl(browser) def _Browser_LoadUrl(self, browser): if browser.GetUrl() == "data:text/html,Test#Browser.LoadUrl": browser.LoadUrl("file://"+GetApplicationPath("wxpython.html")) def OnLoadError(self, browser, frame, errorCode, errorTextList, failedUrl): print("[wxpython.py] LoadHandler::OnLoadError()") print(" frame url = %s" % frame.GetUrl()[:100]) print(" error code = %s" % errorCode) print(" error text = %s" % errorTextList[0]) print(" failed url = %s" % failedUrl) # Handle ERR_ABORTED error code, to handle the following cases: # 1. Esc key was pressed which calls browser.StopLoad() in OnKeyEvent # 2. Download of a file was aborted if errorCode == cefpython.ERR_ABORTED: print("[wxpython.py] LoadHandler::OnLoadError(): Ignoring load"\ " error: Esc was pressed or file download was aborted") return; customErrorMessage = "My custom error message!" frame.LoadUrl("data:text/html,%s" % customErrorMessage) # ------------------------------------------------------------------------- # LifespanHandler # ------------------------------------------------------------------------- # ** This callback is executed on the IO thread ** # Empty place-holders: popupFeatures, client. def OnBeforePopup(self, browser, frame, targetUrl, targetFrameName, targetDisposition, userGesture, popupFeatures, windowInfo, client, browserSettings, noJavascriptAccess): print("[wxpython.py] LifespanHandler::OnBeforePopup()") print(" targetUrl = %s" % targetUrl) # Custom browser settings for popups: # > browserSettings[0] = {"plugins_disabled": True} # Set WindowInfo object: # > windowInfo[0] = cefpython.WindowInfo() allowPopups = True return not allowPopups def _OnAfterCreated(self, browser): # This is a global callback set using SetGlobalClientCallback(). print("[wxpython.py] LifespanHandler::_OnAfterCreated()") print(" browserId=%s" % browser.GetIdentifier()) def DoClose(self, browser): print("[wxpython.py] LifespanHandler::DoClose()") print(" browserId=%s" % browser.GetIdentifier()) def OnBeforeClose(self, browser): print("[wxpython.py] LifespanHandler::OnBeforeClose") print(" browserId=%s" % browser.GetIdentifier()) # ------------------------------------------------------------------------- # JavascriptDialogHandler # ------------------------------------------------------------------------- def OnJavascriptDialog(self, browser, originUrl, dialogType, messageText, defaultPromptText, callback, suppressMessage): print("[wxpython.py] JavascriptDialogHandler::OnJavascriptDialog()") print(" originUrl="+originUrl) print(" dialogType="+str(dialogType)) print(" messageText="+messageText) print(" defaultPromptText="+defaultPromptText) # If you want to suppress the javascript dialog: # suppressMessage[0] = True return False def OnBeforeUnloadJavascriptDialog(self, browser, messageText, isReload, callback): print("[wxpython.py] OnBeforeUnloadJavascriptDialog()") print(" messageText="+messageText) print(" isReload="+str(isReload)) # Return True if the application will use a custom dialog: # callback.Continue(allow=True, userInput="") # return True return False def OnResetJavascriptDialogState(self, browser): print("[wxpython.py] OnResetDialogState()") def OnJavascriptDialogClosed(self, browser): print("[wxpython.py] OnDialogClosed()") class MyApp(wx.App): timer = None timerID = 1 timerCount = 0 def OnInit(self): if not USE_EVT_IDLE: self.CreateTimer() frame = MainFrame() self.SetTopWindow(frame) frame.Show() return True def CreateTimer(self): # See "Making a render loop": # http://wiki.wxwidgets.org/Making_a_render_loop # Another approach is to use EVT_IDLE in MainFrame, # see which one fits you better. self.timer = wx.Timer(self, self.timerID) self.timer.Start(10) # 10ms wx.EVT_TIMER(self, self.timerID, self.OnTimer) def OnTimer(self, event): self.timerCount += 1 # print("[wxpython.py] OnTimer() %d" % self.timerCount) cefpython.MessageLoopWork() def OnExit(self): # When app.MainLoop() returns, MessageLoopWork() should # not be called anymore. if not USE_EVT_IDLE: self.timer.Stop() def GetSources(): # Get sources of all python functions and methods from this file. # This is to provide sources preview to wxpython.html. # The dictionary of functions is binded to "window.sources". thisModule = sys.modules[__name__] functions = inspect.getmembers(thisModule, inspect.isfunction) classes = inspect.getmembers(thisModule, inspect.isclass) sources = {} for funcTuple in functions: sources[funcTuple[0]] = inspect.getsource(funcTuple[1]) for classTuple in classes: className = classTuple[0] classObject = classTuple[1] methods = inspect.getmembers(classObject) for methodTuple in methods: try: sources[methodTuple[0]] = inspect.getsource(\ methodTuple[1]) except: pass return sources if __name__ == '__main__': print('[wxpython.py] architecture=%s-bit' % (8 * struct.calcsize("P"))) print('[wxpython.py] wx.version=%s' % wx.version()) # Intercept python exceptions. Exit app immediately when exception # happens on any of the threads. sys.excepthook = ExceptHook # Application settings g_applicationSettings = { # Disk cache # "cache_path": "webcache/", # CEF Python debug messages in console and in log_file "debug": True, # Set it to LOGSEVERITY_VERBOSE for more details "log_severity": cefpython.LOGSEVERITY_INFO, # Set to "" to disable logging to a file "log_file": GetApplicationPath("debug.log"), # "resources_dir_path" must be set on Mac, "locales_dir_path" not. # You must also set "locale_pak" using command line switch. "resources_dir_path": cefpython.GetModuleDirectory()+"/Resources", # The "subprocess" executable that launches the Renderer # and GPU processes among others. You may rename that # executable if you like. "browser_subprocess_path": "%s/%s" % ( cefpython.GetModuleDirectory(), "subprocess"), # This option is required for the GetCookieManager callback # to work. It affects renderer processes, when this option # is set to True. It will force a separate renderer process # for each browser created using CreateBrowserSync. "unique_request_context_per_browser": True, # Downloads are handled automatically. A default SaveAs file # dialog provided by OS will be displayed. "downloads_enabled": True, # Remote debugging port, required for Developer Tools support. # A value of 0 will generate a random port. To disable devtools # support set it to -1. "remote_debugging_port": 0, # Mouse context menu "context_menu": { "enabled": True, "navigation": True, # Back, Forward, Reload "print": True, "view_source": True, "external_browser": True, # Open in external browser "devtools": True, # Developer Tools }, # See also OnCertificateError which allows you to ignore # certificate errors for specific websites. "ignore_certificate_errors": False, } # Browser settings. You may have different settings for each # browser, see the call to CreateBrowserSync. g_browserSettings = { # "plugins_disabled": True, # "file_access_from_file_urls_allowed": True, # "universal_access_from_file_urls_allowed": True, } # Command line switches set programmatically g_switches = { # On Mac it is required to provide path to a specific # locale.pak file. On Win/Linux you only specify the # ApplicationSettings.locales_dir_path option. "locale_pak": cefpython.GetModuleDirectory() +"/Resources/en.lproj/locale.pak", # "proxy-server": "socks5://127.0.0.1:8888", # "no-proxy-server": "", # "enable-media-stream": "", # "remote-debugging-port": "12345", # "disable-gpu": "", # "--invalid-switch": "" -> Invalid switch name } cefpython.Initialize(g_applicationSettings, g_switches) app = MyApp(False) app.MainLoop() # Let wx.App destructor do the cleanup before calling # cefpython.Shutdown(). This is to ensure reliable CEF shutdown. del app # On Mac cefpython.Shutdown() is called in MainFrame.OnClose, # followed by wx.GetApp.Exit().
X Tutup