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
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
|
nsplugins
=========
The 'nsplugins' directory tqcontains various tools used to utilize Netscape(R)
browser plugins in KDE applications.
- ./ - common headers and utilities
- viewer/ - provides means to embed a Netscape(R) browser plugin
into a KDE application.
- sdk/ - headers from the Netscape(R) browser plugin SDK.
- control/ - KDE Control Center applet for managing plugins
- test/ - simple applications to test nsplugin
- wrapper/ - currently unused.
At the time of writing, nsplugins supports Navigator 4 API plugins
and doesn't support new Mozilla XPCOM components.
Parts
=====
The Netscape(R) browser plugin support consists of 3 parts:
- "PluginPart" KPart which is embedded into KHTML pages
(alone, it looks no different than a gray rectangle)
- "nspluginviewer", a separate application which is capable of
actually loading and playing the various Netscape(R) plugins.
- "pluginscan", a small utility to scan predefined directories
for Netscape(R) browser plugins.
PluginPart's responsibility is to run "nspluginviewer" and instruct
it to display the plugin inside its' rectangle.
* KParts is KDE's standard for embeddable objects, much like
Microsoft's OLE.
PluginPart / nspluginviewer separation
======================================
Plugins contain 3rd-party code which might crash if badly written
(especially inside our "Netscape emulation" environment).
The PluginPart runs inside the memory space of the embedding application,
making the embedding application susceptible to a crash in the PluginPart.
In our case, the "embedding application" would be the browser, and having
the whole browser shut down just because some plugin crashed is bad.
By loading and talking to the plugin is a separate process
("nspluginviewer"), we prevent such crashes from taking
the browser down with them -- the worst that could happen
is the plugin suddenly disappearing from the page, revealing
PluginPart's gray background.
nspluginviewer to PluginPart communications
===========================================
The communications between the PluginPart and nspluginviewer
are done via DCOP. Messages which nspluginviewer could send
to PluginPart include:
- requestURL(url, target) - make the browser load a URL
- statusMessage(msg) - change the status bar in the
embedding application
(For an updated list of messages, see NSPluginCallbackIface.h)
When received, those messages are mapped to functions in
NSPluginCallback, which in turn calls PluginPart functions.
* DCOP is KDE's standard for RPC, typically used by local
processes to pass messages.
PluginPart to nspluginviewer communications
===========================================
nspluginviewer's window is, all in all, a simple X11 window,
it can be resized, focused and unfocused with standard X11 functions
as long as we know its' window ID.
PluginPart gets a NSPluginInstance (from NSPluginLoader::newInstance),
which it sees as a regular QWidget. e.g. it can resize the QWidget and
nspluginviewer's window will resize.
This magic happens thanks to KJavaEmbed which NSPluginInstance inherits.
KJavaEmbed inherits QWidget and hooks up to QWidget's events, mapping
those events to the appropriate X11 window operations.
Resizing, focusing in and out is about all PluginPart could ever do
with the nspluginviewer.
How does KJavaEmbed know the window ID of the nspluginviewer window?
NSPluginInstance retrieves it from nspluginviewer with a DCOP call
("int winId()") and gives it to KJavaEmbed.
Main classes
============
- PluginPart class (plugin_part.h, plugin_part.cpp)
A KParts::ReadOnlyPart.
- PluginFactory class (plugin_part.h, plugin_part.cpp)
Creates PluginPart objects.
- NSPluginInstance class (nspluginloader.h, nspluginloader.cpp)
PluginPart's "looking glass" to "nspluginviewer".
- NSPluginLoader class (nspluginloader.h, nspluginloader.cpp)
Responsible for starting a new independant process of the
plugin viewer and initiating DCOP communications with it.
- NSPluginCallbackIface (NSPluginCallbackIface.h)
Defines functions which the nspluginviewer can call in the PluginPart.
Those function calls would be transfered over DCOP and implemented
in PluginPart's NSPluginCallback class.
Plugin Scanner
==============
Implemented in: pluginscan.cpp
Utility which scans for Netscape(R) browser plugins in pre-defined
directories (according to KDE's configuration) and writes down
information about the located plugins into the plugin information
cache. To locate all the plugins, it doesn't limit itself to files
with .so extension. A forking mechanism is implemented to prevent
faulty .so's from crashing the scanning process.
This utility is usually not invoked standalone, but used by the
KDE Control Center applet which invokes it on:
- explicit request (ie. pressing "Scan for new plugins" button)
- KDE startup (in the applet's initialization code)
- first time the applet is ran
Viewer
======
Implemented in: viewer/
The viewer is a KDE component used to embed Netscape(R) plugins
into KDE applications (such as KHTML / Konqueror). To the plugin, it
emulates a Netscape(R) browser, while outside - it offers a standard KParts
object.
NSPluginCallbackIface.h
Defines functions which the nspluginviewer can call in the PluginPart.
Those function calls would be transfered over DCOP and implemented
in PluginPart's NSPluginCallback class.
kxt.h, kxt.cpp
...
resolve.h
Some macros to resolve symbols.
nsplugin.h, nsplugin.cpp
The implementation of the interface between KDE and the Netscape(R)
browser plugin as a KParts object.
Note: This file implements its own NSPluginInstance, which should
not be confused with PluginPart's NSPluginInstance.
It implements:
- The Netscape(R) emulation side:
g_NPN_* functions, which implement Netscape(R) browser
plugin API NPN_* C methods accordingly. Our emulation
of a Netscape(R) browser is as good as those functions are.
By definition, all functions are passed with an NPN ("plugin
instance") struct, and each NPN has an 'ndata' pointer member.
That's good news to us, since we now can use it to point to
our NSPluginInstance object.
Those functions contain the minimum necessary code to
call the NSPluginInstance function for the requested operation
- The KDE side:
NSPluginInstance implements functions for various g_NPN_*
operations. Typically those operations will be passed via DCOP
to the PluginPart -- e.g. a operation which changes the window's
status bar.
viewer.cpp
A standalone application that executes Netscape plugins.
Control Center applet
=====================
The Control Center applet allows the user to:
- Add additional directories which "pluginscan" would scan for
Netscape(R) plugins.
- See which plugins "pluginscan" found (by reading the plugin cache).
- Force a "pluginscan" rescan.
The KDE Control Center API is well documented in the KDE SDK.
Compatibility notes
===================
Although nsplugins strive for full compatibility with the Netscape(R)
browser plugin execution environment, it still lacks the following features:
- LiveConnect (Netscape Navigator 3.0) Java / JavaScript connectivity
Support for LiveConnect (for JavaScript support) requires:
- Reimplementation of Netscape's JRI functions (see sdk/jri.h)
to communicate back (via DCOP?) with the PluginPart, which'll
signal them to KHTML's KJS interpretor. KParts::BrowserExtension
should be extended to support those signals. Of course, only
JRI calls in the "netscape.javascript" namespace should be handled.
- Implementation of g_NPN_GetJavaEnv and g_NPN_GetJavaPeer
(currently they're empty placeholders).
- Implementing g_NPN_GetValue(NPNVjavascriptEnabledBool, ...)
(currently always returns "false").
Should involve adding a slot to KParts::BrowserExtension
and making KHTML signal it whenever JS gets enabled or disabled.
LiveConnect will probably never be supported due to JRE ABI issues, and
the fact that we need to have a JRE with JNI inside the nspluginviewer.
LiveConnect is almost obsolete as well. XPConnect will not be integrated
either.
The current support is enough to run common plugins such as Macromedia(R)
Flash(R) and RealPlayer(R), but some of their functionality might be missing.
For instance, Flash(R) animations cannot call user-defined JavaScript
routines nor can the user control the Flash(R) animation via JavaScript.
Additional documentation
========================
Additional documentation on the Netscape(R) plugin API is available at:
- http://developer.netscape.com/docs/manuals/plugins.html
- http://www.mozilla.org/docs/plugin.html
Author
======
This README was authored Ilya Konstantinov <kde-devel@future.shiny.co.il>.
Thanks go to Matthias Hoelzer-Kluepfel and Stefan Schimanski for writing
the actual code and making it all possible.
Present maintainer of the nsplugin code is George Staikos <staikos@kde.org>
|