forked from cztomczak/cefpython
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathframe.pyx
More file actions
234 lines (195 loc) · 9.12 KB
/
frame.pyx
File metadata and controls
234 lines (195 loc) · 9.12 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
# Copyright (c) 2012-2014 The CEF Python authors. All rights reserved.
# License: New BSD License.
# Website: http://code.google.com/p/cefpython/
cdef dict g_pyFrames = {}
cdef object GetUniqueFrameId(int browserId, object frameId):
return str(browserId) +"#"+ str(frameId)
cdef PyFrame GetPyFrameById(int browserId, object frameId):
cdef object uniqueFrameId = GetUniqueFrameId(browserId, frameId)
if uniqueFrameId in g_pyFrames:
return g_pyFrames[uniqueFrameId]
return None
cdef PyFrame GetPyFrame(CefRefPtr[CefFrame] cefFrame):
global g_pyFrames
if <void*>cefFrame == NULL or not cefFrame.get():
Debug("GetPyFrame(): returning None")
return
cdef PyFrame pyFrame
# long long
cdef object frameId = cefFrame.get().GetIdentifier()
cdef int browserId = cefFrame.get().GetBrowser().get().GetIdentifier()
assert (frameId and browserId), "frameId or browserId empty"
cdef object uniqueFrameId = GetUniqueFrameId(browserId, frameId)
if uniqueFrameId in g_pyFrames:
return g_pyFrames[uniqueFrameId]
for uFid, pyFrame in g_pyFrames.items():
if not pyFrame.cefFrame.get():
Debug("GetPyFrame(): removing an empty CefFrame reference, " \
"uniqueFrameId = %s" % uniqueFrameId)
del g_pyFrames[uFid]
# Debug("GetPyFrame(): creating new PyFrame, frameId=%s" % frameId)
pyFrame = PyFrame(browserId, frameId)
pyFrame.cefFrame = cefFrame
g_pyFrames[uniqueFrameId] = pyFrame
return pyFrame
cdef void RemovePyFrame(int browserId, object frameId) except *:
# Called from V8ContextHandler_OnContextReleased().
global g_pyFrames
cdef object uniqueFrameId = GetUniqueFrameId(browserId, frameId)
if uniqueFrameId in g_pyFrames:
Debug("del g_pyFrames[%s]" % uniqueFrameId)
del g_pyFrames[uniqueFrameId]
else:
Debug("RemovePyFrame() FAILED: uniqueFrameId = %s" % uniqueFrameId)
cdef void RemovePyFramesForBrowser(int browserId) except *:
# Called from LifespanHandler_BeforeClose().
cdef list toRemove = []
cdef object uniqueFrameId
cdef PyFrame pyFrame
global g_pyFrames
for uniqueFrameId, pyFrame in g_pyFrames.iteritems():
if pyFrame.GetBrowserIdentifier() == browserId:
toRemove.append(uniqueFrameId)
for uniqueFrameId in toRemove:
Debug("del g_pyFrames[%s]" % uniqueFrameId)
del g_pyFrames[uniqueFrameId]
cdef class PyFrame:
cdef CefRefPtr[CefFrame] cefFrame
cdef int browserId
cdef object frameId
cdef CefRefPtr[CefFrame] GetCefFrame(self) except *:
# Do not call IsValid() here, if the frame does not exist
# then no big deal, no reason to crash the application.
# The CEF calls will fail, but they also won't cause crash.
if <void*>self.cefFrame != NULL and self.cefFrame.get():
return self.cefFrame
raise Exception("PyFrame.GetCefFrame() failed: CefFrame was destroyed")
def __init__(self, int browserId, int frameId):
self.browserId = browserId
self.frameId = frameId
IF CEF_VERSION == 3:
cpdef py_bool IsValid(self):
if <void*>self.cefFrame != NULL and self.cefFrame.get() \
and self.cefFrame.get().IsValid():
return True
return False
def CallFunction(self, *args):
# DEPRECATED - keep for backwards compatibility.
self.ExecuteFunction(*args)
cpdef py_void Copy(self):
self.GetCefFrame().get().Copy()
cpdef py_void Cut(self):
self.GetCefFrame().get().Cut()
cpdef py_void Delete(self):
self.GetCefFrame().get().Delete()
def ExecuteFunction(self, *args):
# No need to enter V8 context as we're calling javascript
# asynchronously using ExecuteJavascript() function.
funcName = args[0]
code = funcName+"("
for i in range(1, len(args)):
if i != 1:
code += ", "
code += json.dumps(args[i])
code += ")"
self.ExecuteJavascript(code)
cpdef py_void ExecuteJavascript(self, py_string jsCode,
py_string scriptUrl="", int startLine=0):
self.GetCefFrame().get().ExecuteJavaScript(PyToCefStringValue(jsCode),
PyToCefStringValue(scriptUrl), startLine)
cpdef object GetIdentifier(self):
# It is better to save browser and frame identifiers during
# browser instantiation. When freeing PyBrowser and PyFrame
# we need these identifiers. CefFrame and CefBrowser may already
# be freed and calling methods on these objects may fail, this
# may or may not include GetIdentifier() method, but let's be sure.
return self.frameId
cpdef int GetBrowserIdentifier(self) except *:
return self.browserId
cpdef str GetName(self):
return CefToPyString(self.GetCefFrame().get().GetName())
IF CEF_VERSION == 1:
cpdef object GetProperty(self, py_string name):
assert IsThread(TID_UI), (
"Frame.GetProperty() may only be called on the UI thread")
cdef CefRefPtr[CefV8Context] v8Context = self.GetCefFrame().get().GetV8Context()
cdef CefRefPtr[CefV8Value] window = v8Context.get().GetGlobal()
cdef CefString cefPropertyName
PyToCefString(name, cefPropertyName)
cdef CefRefPtr[CefV8Value] v8Value = window.get().GetValue(cefPropertyName)
return V8ToPyValue(v8Value, v8Context)
cpdef str GetSource(self):
assert IsThread(TID_UI), (
"Frame.GetSource() may only be called on the UI thread")
return CefToPyString(self.GetCefFrame().get().GetSource())
cpdef str GetText(self):
assert IsThread(TID_UI), (
"Frame.GetText() may only be called on the UI thread")
return CefToPyString(self.GetCefFrame().get().GetText())
IF CEF_VERSION == 3:
cpdef py_void GetSource(self, object userStringVisitor):
self.GetCefFrame().get().GetSource(CreateStringVisitor(userStringVisitor))
cpdef py_void GetText(self, object userStringVisitor):
self.GetCefFrame().get().GetText(CreateStringVisitor(userStringVisitor))
cpdef str GetUrl(self):
return CefToPyString(self.GetCefFrame().get().GetURL())
cpdef py_bool IsFocused(self):
IF CEF_VERSION == 1:
assert IsThread(TID_UI), (
"Frame.IsFocused() may only be called on the UI thread")
return self.GetCefFrame().get().IsFocused()
cpdef py_bool IsMain(self):
return self.GetCefFrame().get().IsMain()
cpdef py_void LoadRequest(self):
pass
cpdef py_void LoadString(self, py_string value, py_string url):
cdef CefString cefValue
cdef CefString cefUrl
PyToCefString(value, cefValue)
PyToCefString(url, cefUrl)
self.GetCefFrame().get().LoadString(cefValue, cefUrl)
cpdef py_void LoadUrl(self, py_string url):
url = GetNavigateUrl(url)
cdef CefString cefUrl
PyToCefString(url, cefUrl)
self.GetCefFrame().get().LoadURL(cefUrl)
cpdef py_void Paste(self):
self.GetCefFrame().get().Paste()
IF CEF_VERSION == 1:
cpdef py_void Print(self):
self.GetCefFrame().get().Print()
cpdef py_void Redo(self):
self.GetCefFrame().get().Redo()
cpdef py_void SelectAll(self):
self.GetCefFrame().get().SelectAll()
IF CEF_VERSION == 1:
cpdef py_void SetProperty(self, py_string name, object value):
assert IsThread(TID_UI), (
"Frame.SetProperty() may only be called on the UI thread")
if not JavascriptBindings.IsValueAllowed(value):
valueType = JavascriptBindings.__IsValueAllowed(value)
raise Exception("Frame.SetProperty() failed: name=%s, "
"not allowed type: %s (this may be a type of a nested value)"
% (name, valueType))
cdef CefRefPtr[CefV8Context] v8Context = self.GetCefFrame().get().GetV8Context()
cdef CefRefPtr[CefV8Value] window = v8Context.get().GetGlobal()
cdef CefString cefPropertyName
PyToCefString(name, cefPropertyName)
cdef cpp_bool sameContext = v8Context.get().IsSame(cef_v8_static.GetCurrentContext())
if not sameContext:
Debug("Frame.SetProperty(): inside a different context, calling v8Context.Enter()")
assert v8Context.get().Enter(), "v8Context.Enter() failed"
window.get().SetValue(
cefPropertyName,
PyToV8Value(value, v8Context),
V8_PROPERTY_ATTRIBUTE_NONE)
if not sameContext:
assert v8Context.get().Exit(), "v8Context.Exit() failed"
cpdef py_void Undo(self):
self.GetCefFrame().get().Undo()
cpdef py_void ViewSource(self):
self.GetCefFrame().get().ViewSource()
cpdef PyFrame GetParent(self):
return GetPyFrame(self.GetCefFrame().get().GetParent())
cpdef PyBrowser GetBrowser(self):
return GetPyBrowser(self.GetCefFrame().get().GetBrowser())