/jEdit/tags/jedit-4-0-pre8/doc/users-guide/plugin-implement.xml

# · XML · 1232 lines · 942 code · 218 blank · 72 comment · 0 complexity · bcacf1bd2db778f9554de6cae963797f MD5 · raw file

  1. <!-- jEdit 4.0 Plugin Guide, (C) 2001, 2002 John Gellene -->
  2. <!-- jEdit buffer-local properties: -->
  3. <!-- :indentSize=1:tabSize=2:noTabs=true:maxLineLen=72: -->
  4. <!-- Sat Jun 23 08:54:21 EDT 2001 @579 /Internet Time/ -->
  5. <!-- This chapter of the jEdit 3.2 Plugin Guide -->
  6. <!-- discusses how to implement a plugin -->
  7. <chapter id="plugin-implement"><title>Writing a Plugin</title>
  8. <para>
  9. One way to organize a plugin project is to design the software
  10. as if it were a <quote>stand alone</quote> application, with three
  11. exceptions:
  12. </para>
  13. <itemizedlist>
  14. <listitem>
  15. <para>
  16. The plugin can access the <classname>View</classname> object with
  17. which it is associated, as well as static methods of the
  18. <classname>jEdit</classname> class, to obtain and manipulate various
  19. data and host application objects;
  20. </para>
  21. </listitem>
  22. <listitem>
  23. <para>
  24. If the plugin has visible components, they are ultimately contained in a
  25. <classname>JPanel</classname> object instead of a top-level frame
  26. window; and
  27. </para>
  28. </listitem>
  29. <listitem>
  30. <para>
  31. The plugin implements the necessary elements of the jEdit plugin API
  32. that were outlined in the last chapter: a plugin core class,
  33. perhaps a number of plugin
  34. window classes, maybe a plugin option pane class, and a set of
  35. required plugin resources.
  36. </para>
  37. <para>
  38. Not every plugin has configurable options; some do not have a
  39. visible window. However, all will need a plugin
  40. core class and a minimum set of other resources.
  41. </para>
  42. </listitem>
  43. </itemizedlist>
  44. <para>
  45. We will now illustrate this approach by introducing an example plugin.
  46. </para>
  47. <sect1 id="plugin-example"><title>QuickNotepad: An Example Plugin</title>
  48. <para>
  49. There are many applications for the leading operating systems that
  50. provide a <quote>scratch-pad</quote> or <quote>sticky note</quote>
  51. facility for the desktop display. A similar type of facility operating
  52. within the jEdit display would be a convenience. The use of docking
  53. windows would allow the notepad to be displayed or hidden with a single
  54. mouse click or keypress (if a keyboard shortcut were defined). The
  55. contents of the notepad could be saved at program exit (or, if earlier,
  56. deactivation of the plugin) and retrieved at program startup or plugin
  57. activation.
  58. </para>
  59. <para>
  60. We will keep the capabilities of this plugin modest, but a few
  61. other features would be worthwhile. The user should be able to write
  62. the contents of the notepad to storage on demand. It should also be
  63. possible to choose the name and location of the file that will be
  64. used to hold the notepad text. This would allow the user to load
  65. other files into the notepad display. The path of the notepad file
  66. should be displayed in the plugin window, but will give the user the
  67. option to hide the file name. Finally, there should be an action
  68. by which a single click or keypress would cause the contents of the
  69. notepad to be written to the new text buffer for further processing.
  70. </para>
  71. <para>
  72. The full source code for QuickNotepad is contained in jEdit's
  73. source code distribution. We will provide excerpts in this discussion
  74. where it is helpful to illustrate specific points. You are invited
  75. to obtain the source code for further study or to use as a starting point
  76. for your own plugin.
  77. </para>
  78. </sect1>
  79. <sect1 id="plugin-implement-class"><title>Writing a Plugin Core Class</title>
  80. <para>
  81. The major issues encountered when writing a plugin core class arise
  82. from the developer's decisions on what features the plugin will make
  83. available. These issues have implications for other plugin elements
  84. as well.
  85. </para>
  86. <itemizedlist>
  87. <listitem><para>
  88. Will the plugin provide for actions that the user can trigger using
  89. jEdit's menu items, toolbar buttons and keyboard shortcuts?
  90. </para></listitem>
  91. <listitem><para>
  92. Will the plugin have its own visible interface?
  93. </para></listitem>
  94. <listitem><para>
  95. Will the plugin
  96. respond to any EditBus messages reflecting changes in the host
  97. application's state?
  98. </para></listitem>
  99. <listitem><para>
  100. Will the plugin have settings that the user can configure?
  101. </para></listitem>
  102. </itemizedlist>
  103. <sect2 id="implement-plugin-choose-base"><title>Choosing a Base Class</title>
  104. <para>
  105. If the plugin will respond to EditBus
  106. messages, it must be derived from <classname>EBPlugin</classname>.
  107. Otherwise, <classname>EditPlugin</classname> will suffice as a
  108. base class.
  109. </para>
  110. <para>
  111. Knowing what types of messages are made available by the plugin API is
  112. obviously helpful is determining both the base plugin class and the
  113. contents of a <filename>handleMessage()</filename> method. The message
  114. classes derived from <classname>EBMessage</classname> cover the opening
  115. and closing of the application, changes in the status of text buffers
  116. and their container and changes in user settings, as well as changes in
  117. the state of other program features. Specific message classes of potential
  118. interest to a plugin include the following:
  119. </para>
  120. <itemizedlist>
  121. <listitem>
  122. <para>
  123. <classname>EditorStarted</classname>, sent during the
  124. application's startup routine, just prior to the creation of the
  125. initial <classname>View</classname>;
  126. </para>
  127. </listitem>
  128. <listitem>
  129. <para>
  130. <classname>EditorExitRequested</classname>, sent when a request to
  131. exit has been made, but before saving open buffers and writing user
  132. settings to storage;
  133. </para>
  134. </listitem>
  135. <listitem>
  136. <para>
  137. <classname>EditorExiting</classname>, sent just before jEdit
  138. actually exits;
  139. </para>
  140. </listitem>
  141. <listitem>
  142. <para>
  143. <classname>EditPaneUpdate</classname>, sent when an edit pane
  144. containing a text area (including a pane created by splitting
  145. an existing window) has been created or destroyed, or when a
  146. buffer displayed in an edit pane has been changed;
  147. </para>
  148. </listitem>
  149. <listitem>
  150. <para>
  151. <classname>BufferUpdate</classname>, sent when a text buffer is
  152. created, loaded, or being saved, or when its editing mode or
  153. markers have changed;
  154. </para>
  155. </listitem>
  156. <listitem>
  157. <para>
  158. <classname>ViewUpdate</classname>, sent when a <classname>View</classname>
  159. is created or closed; and
  160. </para>
  161. </listitem>
  162. <listitem>
  163. <para>
  164. <classname>PropertiesChanged</classname>, sent when the properties
  165. of the application or a plugin has been changed through the
  166. <guimenuitem>General Options</guimenuitem> dialog;
  167. </para>
  168. </listitem>
  169. </itemizedlist>
  170. <para>
  171. Detailed documentation for each message class can be found in <xref
  172. linkend="api-message" />.
  173. </para>
  174. <!-- TODO: include data members of derived message classes in appendix -->
  175. </sect2>
  176. <sect2 id="plugin-implement-base-methods">
  177. <title>Implementing Base Class Methods</title>
  178. <sect3 id="plugin-implement-base-general">
  179. <title>General Considerations</title>
  180. <para>
  181. Whether <classname>EditPlugin</classname> or <classname>EBPlugin</classname>
  182. is selected as the base of the plugin core
  183. class, the implementations of <function>start()</function> and
  184. <function>stop()</function> in the plugin's derived class are likely to
  185. be trivial, or not present at all (in which case they will be
  186. <quote>no-ops</quote> inherited from <classname>EditPlugin</classname>).
  187. </para>
  188. <!--
  189. <para>
  190. If the plugin is to use the dockable window API, the
  191. <quote> internal names</quote> of any dockable windows must be
  192. registered with the EditBus component.
  193. The EditBus stores such information in one of a number of
  194. <quote>named lists</quote>. Here is how the QuickNotepad plugin
  195. registers its dockable window:
  196. </para>
  197. <informalexample>
  198. <programlisting>EditBus.addToNamedList(DockableWindow.DOCKABLE_WINDOW_LIST, NAME);</programlisting>
  199. </informalexample>
  200. <para>
  201. The first parameter is a <classname>String</classname> constant
  202. identifying the dockable window list. The second is a static
  203. <classname>String</classname> constant which is initialized in the
  204. plugin core class as the dockable window's internal name.
  205. </para>
  206. -->
  207. <para>
  208. The plugin core class can include <literal>static
  209. final String</literal> data members containing information to be registered
  210. with the EditBus or key names for certain types of plugin properties.
  211. This makes it easier to refer to the
  212. information when a method such as <function>handleMessage()</function>
  213. examines the contents of a message. The kind of data that can be handled
  214. in this fashion include the following:
  215. </para>
  216. <itemizedlist>
  217. <listitem>
  218. <para>the name of the plugin;</para>
  219. </listitem>
  220. <listitem>
  221. <para>
  222. a label for identifying the plugin's menu;
  223. </para>
  224. </listitem>
  225. <listitem>
  226. <para>
  227. a prefix for labeling properties required by the plugin API; and
  228. </para>
  229. </listitem>
  230. <listitem>
  231. <para>
  232. a prefix to be used in labeling items used in the plugin's option pane
  233. </para>
  234. </listitem>
  235. </itemizedlist>
  236. </sect3>
  237. <sect3 id="plugin-implement-base-example">
  238. <title>Example Plugin Core Class</title>
  239. <para>
  240. We will derive the plugin core class for QuickNotepad from
  241. <classname>EditPlugin</classname>, since there are no EditBus
  242. messages to which the plugin core class need respond. There are no special
  243. initialization or shut down chores to perform, so we will not need
  244. a <filename>start()</filename> or <filename>stop()</filename> method.
  245. We will define a few static
  246. <classname>String</classname> data members to enforce consistent syntax
  247. for the name of properties we will use throughout the plugin.
  248. Finally, we will use a standalone plugin
  249. window class to separate the functions of that class from the visible
  250. component class we will create.
  251. </para>
  252. <para>
  253. The resulting plugin core class is lightweight and straightforward to implement:
  254. </para>
  255. <informalexample><programlisting>public class QuickNotepadPlugin extends EBPlugin {
  256. public static final String NAME = "quicknotepad";
  257. public static final String MENU = "quicknotepad.menu";
  258. public static final String PROPERTY_PREFIX
  259. = "plugin.QuickNotepadPlugin.";
  260. public static final String OPTION_PREFIX
  261. = "options.quicknotepad.";
  262. public void createMenuItems(Vector menuItems) {
  263. menuItems.addElement(GUIUtilities.loadMenu(MENU));
  264. }
  265. public void createOptionPanes(OptionsDialog od) {
  266. od.addOptionPane(new QuickNotepadOptionPane());
  267. }
  268. }</programlisting></informalexample>
  269. <para>
  270. The implementations of <function>createMenuItems()</function> and
  271. <function>createOptionPane()</function>
  272. are typically trivial, because the real work will be done using other
  273. plugin elements. Menu creation is performed by a utility function in
  274. jEdit's API, using properties defined in the plugin's properties file.
  275. The option pane is constructed in its own class.
  276. </para>
  277. <para>
  278. If the plugin only had a single menu item (for example, a checkbox item
  279. that toggled activation of a dockable window), we would call
  280. <function>GUIUtilities.loadMenuItem()</function> instead of
  281. <function>loadMenu()</function>. We will explain the use of both methods
  282. in the next section.
  283. </para>
  284. <para>
  285. The constructor for <classname>QuickNotepadDockable</classname> takes
  286. the values of the <classname>View</classname> object and the docking
  287. position contained in the <classname>CreateDockableWindow</classname>
  288. message. This will enable the plugin to <quote>know</quote> where it is
  289. located and modify its behavior accordingly. In another plugin, it could
  290. enable the plugin to obtain and manipulate various data that are
  291. available through a <classname>View</classname> object.
  292. </para>
  293. </sect3>
  294. </sect2>
  295. <sect2 id="plugin-implement-resources">
  296. <title>Resources for the Plugin Core Class</title>
  297. <sect3 id="plugin-implement-actions"><title>Actions</title>
  298. <para>
  299. The plugin's user action catalog, <filename>actions.xml</filename>, is
  300. the resource used by the plugin API to get the names and definitions of
  301. user actions. The following <filename>actions.xml</filename>
  302. file from the <application>QuickNotepad</application> plugin can
  303. provide a model:
  304. </para>
  305. <informalexample><programlisting>&lt;!DOCTYPE ACTIONS SYSTEM "actions.dtd"&gt;
  306. &lt;ACTIONS&gt;
  307. &lt;ACTION NAME="quicknotepad.choose-file"&gt;
  308. &lt;CODE&gt;
  309. view.getDockableWindowManager()
  310. .getDockable(QuickNotepadPlugin.NAME).chooseFile();
  311. &lt;/CODE&gt;
  312. &lt;/ACTION&gt;
  313. &lt;ACTION NAME="quicknotepad.save-file"&gt;
  314. &lt;CODE&gt;
  315. view.getDockableWindowManager()
  316. .getDockable(QuickNotepadPlugin.NAME).saveFile();
  317. &lt;/CODE&gt;
  318. &lt;/ACTION&gt;
  319. &lt;ACTION NAME="quicknotepad.copy-to-buffer"&gt;
  320. &lt;CODE&gt;
  321. view.getDockableWindowManager()
  322. .getDockable(QuickNotepadPlugin.NAME).copyToBuffer();
  323. &lt;/CODE&gt;
  324. &lt;/ACTION&gt;
  325. &lt;/ACTIONS&gt;</programlisting></informalexample>
  326. <para>
  327. This file defines three actions. They use the current view's
  328. <classname>DockableWindowManager</classname> object and the method
  329. <filename>getDockable()</filename> to find the QuickNotepad plugin
  330. window and call the desired method.
  331. </para>
  332. <para>
  333. If you are unfamiliar with BeanShell code, you may nevertheless notice
  334. that the code statements bear a strong resemblance to Java code, with
  335. one exception: the
  336. variable <varname>view</varname> is never assigned any value.
  337. </para>
  338. <para>
  339. For complete answers to this and other BeanShell
  340. mysteries, see <xref linkend="writing-macros-part" />; two
  341. observations will suffice here. First, the variable
  342. <varname>view</varname> is predefined by jEdit's implementation of
  343. BeanShell to refer to the current <classname>View</classname> object.
  344. Second, the
  345. BeanShell scripting language is based upon Java syntax, but allows
  346. variables to be typed at run time, so explicit types for variables
  347. need not be declared.
  348. </para>
  349. <para>
  350. A formal description of each element of the
  351. <filename>actions.xml</filename> file can be found in
  352. <xref linkend="api-resources-action" />.
  353. </para>
  354. </sect3>
  355. <sect3 id="plugin-implement-menu"><title>Action Labels and Menu Items</title>
  356. <para>
  357. Now that we have named and defined actions for the plugin, we have to
  358. put them to work. To do so, we must first give them labels that can be
  359. used in menu items and in the sections of jEdit's options dialog that
  360. deal with toolbar buttons, keyboard shortcuts and context menu items. We
  361. supply this information to jEdit through entries in the plugin's
  362. properties file. A call to
  363. <function>GUIUtilities.loadMenu()</function> or
  364. <function>GUIUtilities.loadMenuItem()</function> will read and extract
  365. the necessary labels from the contents of a properties file.
  366. </para>
  367. <para>
  368. The following excerpt from <filename>QuickNotepad.props</filename>
  369. illustrates the format required for action labels and menu items:
  370. </para>
  371. <informalexample><programlisting># action labels
  372. quicknotepad.toggle.label=QuickNotepad
  373. quicknotepad-to-front.label=Bring QuickNotepad to front
  374. quicknotepad.choose-file.label=Choose notepad file
  375. quicknotepad.save-file.label=Save notepad file
  376. quicknotepad.copy-to-buffer.label=Copy notepad to buffer
  377. # application menu items
  378. quicknotepad.menu.label=QuickNotepad
  379. quicknotepad.menu=quicknotepad.toggle - quicknotepad.choose-file \
  380. quicknotepad.save-file quicknotepad.copy-to-buffer
  381. </programlisting></informalexample>
  382. <para>
  383. <function>GUIUtilities.loadMenuItem()</function> and
  384. <function>GUIUtilites.loadMenu()</function> use special conventions
  385. for the value of a menu property to specify menu layout. In
  386. <function>loadMenu()</function>, the use of the dash, as in the second
  387. item in the example menu list, signifies the placement of a separator.
  388. In addition,
  389. the character <userinput>'%'</userinput> used as a prefix on a
  390. label causes <function>loadMenu()</function> to call itself recursively
  391. with the prefixed label as the source of submenu data. Most plugins
  392. will not need to define menus that contain other submenus.
  393. </para>
  394. <para>
  395. Note also that <function>quicknotepad-to-front</function> is not
  396. included in the menu listing. It will appear, however, on the
  397. <guilabel>Shortcuts</guilabel> pane of the <guimenuitem>Global
  398. Options</guimenuitem> dialog,
  399. so that the action can be associated with a keyboard shortcut.
  400. </para>
  401. </sect3>
  402. </sect2>
  403. </sect1>
  404. <sect1 id="window-implement"><title>Implementing a Dockable Window Class</title>
  405. <para>
  406. The <application>QuickNotepad</application> plugin uses the dockable
  407. window API and provides one dockable window. Dockable window classes
  408. must implement the
  409. <classname>DockableWindow</classname> interface. There are basically two
  410. approaches to doing this. One is to have the top-level visible component
  411. also serve as the plugin window. The other is to derive a lightweight
  412. class that will create and hold the top-level window component. We will
  413. illustrate both approaches.
  414. </para>
  415. <sect2 id="window-implement-single"><title>Using a Single Window Class</title>
  416. <para>
  417. A single window class must implement the
  418. <classname>DockableWindow</classname> interface as well as provide for
  419. the creation and layout of the plugin's visible components, and
  420. execution of user actions. The window for QuickNotepad will also
  421. implement the <classname>EBComponent</classname> so it can receive
  422. messages from the EditBus whenever the user has changed the plugin's
  423. settings in the <guimenuitem>Global Options</guimenuitem> dialog. Here
  424. is an excerpt from a class
  425. definition showing the implementation of both interfaces:
  426. </para>
  427. <informalexample><programlisting>public class QuickNotepad extends JPanel
  428. implements EBComponent, QuickNotepadActions
  429. {
  430. private View view;
  431. private String position;
  432. ...
  433. public QuickNotepad(View view, String position) {
  434. this.view = view;
  435. this.position = position;
  436. ...
  437. }
  438. ...
  439. public void handleMessage(EBMessage message) {
  440. if (message instanceof PropertiesChanged) {
  441. propertiesChanged();
  442. }
  443. }
  444. ...
  445. }</programlisting></informalexample>
  446. <para>
  447. This excerpt does not set forth the layout of the plugin's visible
  448. components, nor does it show how our user actions will be implemented.
  449. Both these matters are covered in the full source code.
  450. </para>
  451. </sect2>
  452. <sect2 id="window-implement-actions"><title>An Action Interface</title>
  453. <para>
  454. When an action is invoked, program control must pass to the
  455. component responsible for executing the action. The use of an internal
  456. table of BeanShell scripts that implement actions avoids the need for
  457. plugins to implement <classname>ActionListener</classname> or similar
  458. objects to respond to actions. Instead, the BeanShell scripts address
  459. the plugin through static methods, or if instance data is needed, the
  460. current <classname>View</classname>, its
  461. <classname>DockableWindowManager</classname>, and the plugin
  462. object return by the <filename>getDockable()</filename> method.
  463. </para>
  464. <para>
  465. As an organizational device, we will employ a
  466. <classname>QuickNotepadActions</classname> to define the
  467. actions that QuickNotepad will implement:
  468. </para>
  469. <informalexample><programlisting>public interface QuickNotepadActions {
  470. void chooseFile();
  471. void saveFile();
  472. void copyToBuffer();
  473. }</programlisting></informalexample>
  474. </sect2>
  475. </sect1>
  476. <sect1 id="window-visible-implement">
  477. <title>The Plugin's Visible Window</title>
  478. <sect2 id="example-window-class"><title>Class QuickNotepad</title>
  479. <para>
  480. Here is where most of the features of the plugin will be implemented.
  481. To work with the dockable window API, the top level window will be a
  482. <classname>JPanel</classname>. The visible components reflect a
  483. simple layout. Inside the top-level panel we will place a scroll pane with
  484. a text area. Above the scroll pane we will place a panel containing a small
  485. tool bar and a label displaying the path of the current notepad file.
  486. </para>
  487. <para>
  488. We have identified three user actions in the
  489. <classname>QuickNotepadActions</classname> interface that need
  490. implementation here: <function>chooseFile()</function>,
  491. <function>saveFile()</function>, and
  492. <function>copyToBuffer()</function>. As noted earlier, we also want the
  493. text area to change its appearance in immediate response to a change in
  494. user options settings. In order to do that, the window class must
  495. respond to a <classname>PropertiesChanged</classname> message from
  496. the EditBus.
  497. </para>
  498. <!-- <para>
  499. We could have the plugin core class receive and delegate
  500. <classname>PropertiesChanged</classname> messages to the window class.
  501. However, this would require the plugin core class to hold a reference
  502. to either the plugin window class or the visible window class and to
  503. update that reference when the user activates or deactivates the
  504. plugin. It is simpler to have the plugin window class subscribe to the
  505. EditBus directly; many plugins take this approach. This means that
  506. <classname>QuickNotepad</classname> must implement the
  507. <classname>EBComponent</classname> interface.
  508. </para> -->
  509. <para>
  510. Unlike the <classname>EBPlugin</classname> class, the
  511. <classname>EBComponent</classname> interface does not deal with the
  512. component's actual subscribing and unsubscribing to the EditBus. To
  513. accomplish this, we use a pair of methods inherited from the
  514. Java platform's <classname>JComponent</classname> that are called when
  515. the visible window becomes is assigned and unassigned to its
  516. <classname>DockableWindowContainer</classname>. These two methods,
  517. <function>addNotify()</function> and
  518. <function>removeNotify()</function>, are overridden to add and remove
  519. the visible window from the list of EditBus subscribers.
  520. </para>
  521. <para>
  522. We will provide for two minor features when the notepad is
  523. displayed in the floating window. First, when a floating plugin window
  524. is created, we will give the notepad text area input focus. Second,
  525. when the notepad if floating and has input focus, we will have the
  526. <keycap>Escape</keycap> key dismiss the notepad window. An
  527. <classname>AncestorListener</classname> and a
  528. <classname>KeyListener</classname> will implement these details.
  529. </para>
  530. <para>
  531. Here is the listing for the data members, the constructor, and the
  532. implementation of the <classname>EBComponent</classname> interface:
  533. </para>
  534. <informalexample><programlisting>public class QuickNotepad extends JPanel
  535. implements EBComponent, QuickNotepadActions
  536. {
  537. private String filename;
  538. private String defaultFilename;
  539. private View view;
  540. private boolean floating;
  541. private QuickNotepadTextArea textArea;
  542. private QuickNotepadToolPanel toolPanel;
  543. //
  544. // Constructor
  545. //
  546. public QuickNotepad(View view, String position)
  547. {
  548. super(new BorderLayout());
  549. this.view = view;
  550. this.floating = position.equals(
  551. DockableWindowManager.FLOATING);
  552. this.filename = jEdit.getProperty(
  553. QuickNotepadPlugin.OPTION_PREFIX
  554. + "filepath");
  555. if(this.filename == null || this.filename.length() == 0)
  556. {
  557. this.filename = new String(jEdit.getSettingsDirectory()
  558. + File.separator + "qn.txt");
  559. jEdit.setProperty(QuickNotepadPlugin.OPTION_PREFIX
  560. + "filepath",this.filename);
  561. }
  562. this.defaultFilename = new String(this.filename);
  563. this.toolPanel = new QuickNotepadToolPanel(this);
  564. add(BorderLayout.NORTH, this.toolPanel);
  565. if(floating)
  566. this.setPreferredSize(new Dimension(500, 250));
  567. textArea = new QuickNotepadTextArea();
  568. textArea.setFont(QuickNotepadOptionPane.makeFont());
  569. textArea.addKeyListener(new KeyHandler());
  570. textArea.addAncestorListener(new AncestorHandler());
  571. JScrollPane pane = new JScrollPane(textArea);
  572. add(BorderLayout.CENTER, pane);
  573. readFile();
  574. }
  575. //
  576. // Attribute methods
  577. //
  578. // for toolBar display
  579. public String getFilename()
  580. {
  581. return filename;
  582. }
  583. //
  584. // EBComponent implementation
  585. //
  586. public void handleMessage(EBMessage message)
  587. {
  588. if (message instanceof PropertiesChanged)
  589. {
  590. propertiesChanged();
  591. }
  592. }
  593. private void propertiesChanged()
  594. {
  595. String propertyFilename = jEdit.getProperty(
  596. QuickNotepadPlugin.OPTION_PREFIX + "filepath");
  597. if(!defaultFilename.equals(propertyFilename))
  598. {
  599. saveFile();
  600. toolPanel.propertiesChanged();
  601. defaultFilename = propertyFilename.clone();
  602. filename = defaultFilename.clone();
  603. readFile();
  604. }
  605. Font newFont = QuickNotepadOptionPane.makeFont();
  606. if(!newFont.equals(textArea.getFont()))
  607. {
  608. textArea.setFont(newFont);
  609. textArea.invalidate();
  610. }
  611. }
  612. // These JComponent methods provide the appropriate points
  613. // to subscribe and unsubscribe this object to the EditBus
  614. public void addNotify()
  615. {
  616. super.addNotify();
  617. EditBus.addToBus(this);
  618. }
  619. public void removeNotify()
  620. {
  621. saveFile();
  622. super.removeNotify();
  623. EditBus.removeFromBus(this);
  624. }
  625. ...
  626. }</programlisting></informalexample>
  627. <para>
  628. This listing refers to a <classname>QuickNotebookTextArea</classname>
  629. object. It is currently implemented as a <classname>JTextArea</classname> with
  630. word wrap and tab sizes hard-coded. Placing the object in a separate
  631. class will simply future modifications.
  632. </para>
  633. </sect2>
  634. <sect2 id="example-window-toolbar"><title>Class QuickNotepadToolBar</title>
  635. <para>
  636. There is nothing remarkable about the toolbar panel that is placed
  637. inside the <classname>QuickNotepad</classname> object. The constructor
  638. shows the continued use of items from the plugin's properties file.
  639. </para>
  640. <informalexample><programlisting>public class QuickNotepadToolPanel extends JPanel
  641. {
  642. private QuickNotepad pad;
  643. private JLabel label;
  644. public QuickNotepadToolPanel(QuickNotepad qnpad)
  645. {
  646. pad = qnpad;
  647. JToolBar toolBar = new JToolBar();
  648. toolBar.setFloatable(false);
  649. toolBar.add(makeCustomButton("quicknotepad.choose-file",
  650. new ActionListener() {
  651. public void actionPerformed(ActionEvent evt) {
  652. QuickNotepadToolPanel.this.pad.chooseFile();
  653. }
  654. }));
  655. toolBar.add(makeCustomButton("quicknotepad.save-file",
  656. new ActionListener() {
  657. public void actionPerformed(ActionEvent evt) {
  658. QuickNotepadToolPanel.this.pad.saveFile();
  659. }
  660. }));
  661. toolBar.add(makeCustomButton("quicknotepad.copy-to-buffer",
  662. new ActionListener() {
  663. public void actionPerformed(ActionEvent evt) {
  664. QuickNotepadToolPanel.this.pad.copyToBuffer();
  665. }
  666. }));
  667. label = new JLabel(pad.getFilename(),
  668. SwingConstants.RIGHT);
  669. label.setForeground(Color.black);
  670. label.setVisible(jEdit.getProperty(
  671. QuickNotepadPlugin.OPTION_PREFIX
  672. + "show-filepath").equals("true"));
  673. this.setLayout(new BorderLayout(10, 0));
  674. this.add(BorderLayout.WEST, toolBar);
  675. this.add(BorderLayout.CENTER, label);
  676. this.setBorder(BorderFactory.createEmptyBorder(0, 0, 3, 10));
  677. }
  678. ...
  679. }</programlisting></informalexample>
  680. <para>
  681. The method <classname>makeCustomButton()</classname> provides uniform
  682. attributes for the three toolbar buttons corresponding to three of the
  683. plugin's use actions. The menu titles for the user actions serve double
  684. duty as tooltip text for the buttons. There is also a
  685. <function>propertiesChanged()</function> method for the toolbar that
  686. sets the text and visibility of the label containing the notepad file path.
  687. </para>
  688. </sect2>
  689. </sect1>
  690. <sect1 id="option-implement"><title>Designing an Option Pane</title>
  691. <para>
  692. Using the default implementation provided by
  693. <classname>AbstractOptionPane</classname> reduces the preparation of an
  694. option pane to two principal tasks: writing a
  695. <function>_init()</function> method to layout and initialize the pane,
  696. and writing a <function>_save()</function> method to commit any settings
  697. changed by user input. If a button on the option pane should trigger
  698. another dialog, such as a <classname>JFileChooser</classname> or jEdit's
  699. own enhanced <classname>VFSFileChooserDialog</classname>, the option
  700. pane will also have to implement the
  701. <classname>ActionListener</classname> interface to display additional
  702. components.
  703. </para>
  704. <para>
  705. The QuickNotepad plugin has only three options to set: the path name of
  706. the file that will store the notepad text, the visibility of the
  707. path name on the tool bar, and the notepad's display font.
  708. Using the shortcut methods of the plugin API, the implementation of
  709. <function>_init()</function> looks like this:
  710. </para>
  711. <informalexample><programlisting>public class QuickNotepadOptionPane extends AbstractOptionPane
  712. implements ActionListener
  713. {
  714. private JTextField pathName;
  715. private JButton pickPath;
  716. private FontSelector font;
  717. ...
  718. public void _init()
  719. {
  720. showPath = new JCheckBox(jEdit.getProperty(
  721. QuickNotepadPlugin.OPTION_PREFIX
  722. + "show-filepath.title"),
  723. jEdit.getProperty(
  724. QuickNotepadPlugin.OPTION_PREFIX + "show-filepath")
  725. .equals("true"));
  726. addComponent(showPath);
  727. pathName = new JTextField(jEdit.getProperty(
  728. QuickNotepadPlugin.OPTION_PREFIX
  729. + "filepath"));
  730. JButton pickPath = new JButton(jEdit.getProperty(
  731. QuickNotepadPlugin.OPTION_PREFIX
  732. + "choose-file"));
  733. pickPath.addActionListener(this);
  734. JPanel pathPanel = new JPanel(new BorderLayout(0, 0));
  735. pathPanel.add(pathName, BorderLayout.CENTER);
  736. pathPanel.add(pickPath, BorderLayout.EAST);
  737. addComponent(jEdit.getProperty(
  738. QuickNotepadPlugin.OPTION_PREFIX + "file"),
  739. pathPanel);
  740. font = new FontSelector(makeFont());
  741. addComponent(jEdit.getProperty(
  742. QuickNotepadPlugin.OPTION_PREFIX + "choose-font"),
  743. font);
  744. }
  745. ...
  746. }</programlisting></informalexample>
  747. <para>
  748. Here we adopt the vertical arrangement offered by use of the
  749. <function>addComponent()</function> method with one embellishment.
  750. We want the first <quote>row</quote> of the option pane to contain
  751. a text field with the current notepad file path and a button that will
  752. trigger a file chooser dialog when pressed. To place both of them on
  753. the same line (along with an identifying label for the file option),
  754. we create a <classname>JPanel</classname> to contain both components and
  755. pass the configured panel to <function>addComponent()</function>.
  756. </para>
  757. <para>
  758. The <function>_init()</function> method uses properties from the plugin's
  759. property file to provide the names of label for the components placed
  760. in the option pane. It also uses a property whose name begins with
  761. <function>PROPERTY_PREFIX</function> as a persistent data item - the
  762. path of the current notepad file. The elements of the notepad's font
  763. are also extracted from properties using a static method of the option
  764. pane class.
  765. </para>
  766. <para>
  767. The <function>_save()</function> method extracts data from the user
  768. input components and
  769. assigns them to the plugin's properties. The implementation is
  770. straightforward:
  771. </para>
  772. <informalexample><programlisting>public void _save()
  773. {
  774. jEdit.setProperty(QuickNotepadPlugin.OPTION_PREFIX
  775. + "filepath", pathName.getText());
  776. Font _font = font.getFont();
  777. jEdit.setProperty(QuickNotepadPlugin.OPTION_PREFIX
  778. + "font", _font.getFamily());
  779. jEdit.setProperty(QuickNotepadPlugin.OPTION_PREFIX
  780. + "fontsize", String.valueOf(_font.getSize()));
  781. jEdit.setProperty(QuickNotepadPlugin.OPTION_PREFIX
  782. + "fontstyle", String.valueOf(_font.getStyle()));
  783. jEdit.setProperty(QuickNotepadPlugin.OPTION_PREFIX
  784. + "show-filepath", String.valueOf(showPath.isSelected()));
  785. }</programlisting></informalexample>
  786. <para>
  787. The class has only two other methods, one to display a file chooser
  788. dialog in response to user action, and the other
  789. to construct a <classname>Font</classname> object from the plugin's font
  790. properties. They do not require discussion here.
  791. </para>
  792. </sect1>
  793. <sect1 id="resources-implement"><title>Creating Other Plugin Resources</title>
  794. <para>
  795. We have already covered in some detail one of the three types of resources
  796. that plugins include with their class files - the user action catalog - and
  797. the need for help documentation does not require extended discussion. The
  798. remaining resource is the properties file.
  799. </para>
  800. <para>
  801. The first type of property data is information about the plugin itself.
  802. The first few entries from the QuickNotepad plugin's properties file
  803. fulfills this requirement:
  804. </para>
  805. <informalexample><programlisting># general plugin information
  806. plugin.QuickNotepadPlugin.name=QuickNotepad
  807. plugin.QuickNotepadPlugin.author=John Gellene
  808. plugin.QuickNotepadPlugin.version=2.0
  809. plugin.QuickNotepadPlugin.docs=QuickNotepad.html
  810. plugin.QuickNotepadPlugin.depend.0=jedit 04.00.01.00</programlisting></informalexample>
  811. <para>
  812. These properties are described in detail in <xref
  813. linkend="api-resource-properties" /> and do not require further
  814. discussion here.
  815. </para>
  816. <para>
  817. Next in the file comes a property that sets the title of the
  818. plugin in docked or frame windows. The use of the suffix
  819. <literal>.title</literal> in the property's key name is
  820. required by the plugin API.
  821. </para>
  822. <informalexample><programlisting># dockable window name
  823. quicknotepad.title=QuickNotepad</programlisting></informalexample>
  824. <para>
  825. The next sections, consisting of the action label and menu item
  826. properties, have been discussed earlier in
  827. <xref linkend="plugin-implement-menu" />.
  828. </para>
  829. <informalexample><programlisting># action labels
  830. quicknotepad.toggle.label=QuickNotepad
  831. quicknotepad-to-front.label=Bring QuickNotepad to front
  832. quicknotepad.choose-file.label=Choose notepad file
  833. quicknotepad.save-file.label=Save notepad file
  834. quicknotepad.copy-to-buffer.label=Copy notepad to buffer
  835. # application menu items
  836. quicknotepad.menu=quicknotepad.toggle - quicknotepad.choose-file \
  837. quicknotepad.save-file quicknotepad.copy-to-buffer</programlisting></informalexample>
  838. <para>
  839. We have created a small toolbar as a component of QuickNotepad, so
  840. file names for the button icons follow:
  841. </para>
  842. <informalexample><programlisting># plugin toolbar buttons
  843. quicknotepad.choose-file.icon=Open.gif
  844. quicknotepad.save-file.icon=Save.gif
  845. quicknotepad.copy-to-buffer.icon=Edit.gif</programlisting></informalexample>
  846. <para>
  847. The menu labels corresponding to these icons will also serve as tooltip
  848. text.
  849. </para>
  850. <para>
  851. Finally, the properties file set forth the labels and settings
  852. used by the option pane:
  853. </para>
  854. <informalexample><programlisting># Option pane labels
  855. options.quicknotepad.label=QuickNotepad
  856. options.quicknotepad.file=File:
  857. options.quicknotepad.choose-file=Choose
  858. options.quicknotepad.choose-file.title=Choose a notepad file
  859. options.quicknotepad.choose-font=Font:
  860. options.quicknotepad.show-filepath.title=Display notepad file path
  861. # Initial default font settings
  862. options.quicknotepad.show-filepath=true
  863. options.quicknotepad.font=Monospaced
  864. options.quicknotepad.fontstyle=0
  865. options.quicknotepad.fontsize=14
  866. # Setting not defined but supplied for completeness
  867. options.quicknotepad.filepath=</programlisting></informalexample>
  868. <para>
  869. We do not define a default setting for the <literal>filepath</literal>
  870. property because of
  871. differences among operating systems. We will define a default file
  872. programatically that will reside in the directory jEdit designates for
  873. user settings.
  874. </para>
  875. </sect1>
  876. <sect1 id="example-building"><title>Compiling the Plugin</title>
  877. <para>
  878. We have already outlined the contents of the user action catalog, the
  879. properties file and the documentation file in our earlier discussion.
  880. The final step is to compile the source file and build the archive file
  881. that will hold the class files and the plugin's other resources.
  882. </para>
  883. <para>
  884. Publicly released plugins include with their source a makefile
  885. in XML format for the
  886. <application>Ant</application> utility. The format for this file
  887. requires few changes from plugin to plugin. Here is the version of
  888. <filename>build.xml</filename> used by QuickNotepad and many other
  889. plugins:
  890. </para>
  891. <informalexample><programlisting><![CDATA[<?xml version="1.0"?>
  892. <!--
  893. This is a build.xml file for building the QuickNotepad plugin.
  894. The 'dist' target compiles the plugin and creates the JAR file.
  895. Before running the 'dist' target, you will need to generate the
  896. documentation using one of these two targets:
  897. - 'docs-xalan': Creates documentation using the Xalan XSLT processor
  898. - 'docs-xsltproc': Creates documentation using the xsltproc tool
  899. To use it for building your own plugin, make these changes:
  900. - Change definition of 'jedit.install.dir' to point to the directory
  901. containing jedit.jar
  902. - Change definition of 'jar.name' to the name of your plugin's JAR file
  903. - If necessary, add any dependencies to the 'project.class.path'
  904. definition
  905. - If necessary, change the list of files in the 'dist' targtet
  906. - If your plugin has documentation generated using the DocBook XSL
  907. stylesheets, change the 'docs-xalan' and 'docs-xsltproc' targets
  908. accordingly.
  909. -->
  910. <project name="QuickNotepad" default="dist" basedir=".">
  911. <property name="jedit.install.dir" value="G:\\Program Files\\jEdit 4.0pre1"/>
  912. <property name="jar.name" value="QuickNotepad.jar"/>
  913. <property name="src.dir" value="."/>
  914. <property name="build.dir" value="build"/>
  915. <property name="install.dir" value=".."/>
  916. <path id="project.class.path">
  917. <pathelement location="${jedit.install.dir}/jedit.jar"/>
  918. <pathelement location="."/>
  919. </path>
  920. <target name="init">
  921. <mkdir dir="${build.dir}"/>
  922. </target>
  923. <target name="compile" depends="init">
  924. <javac
  925. srcdir="${src.dir}"
  926. destdir="${build.dir}"
  927. deprecation="on"
  928. includeJavaRuntime="yes"
  929. >
  930. <classpath refid="project.class.path"/>
  931. </javac>
  932. </target>
  933. <target name="dist" depends="compile">
  934. <mkdir dir="${install.dir}"/>
  935. <jar jarfile="${install.dir}/${jar.name}">
  936. <fileset dir="${build.dir}"/>
  937. <fileset dir="${src.dir}">
  938. <include name="actions.xml"/>
  939. <include name="dockables.xml"/>
  940. <include name="**/*.props"/>
  941. <include name="**/*.html"/>
  942. <include name="**/*.gif"/>
  943. </fileset>
  944. </jar>
  945. </target>
  946. <!-- Generate docs with xsltproc tool from www.xmlsoft.org -->
  947. <!-- NOTE: the "o" or "output" options do not appear to be working. -->
  948. <!-- To customize the title of an HTML output file, set the -->
  949. <!-- 'use.id.as.filename' variable in your XSL customization file -->
  950. <!-- and provide the file name (without extension) as the 'id' -->
  951. <!-- attribute of the element that constitutes a 'chunk' of output. For -->
  952. <!-- a plugin help file, there should usually be only one chunk, so the -->
  953. <!-- top-level element (<article> or <book>) should have the 'id' -->
  954. <!-- attribute. -->
  955. <target name="docs-xsltproc">
  956. <exec executable="xsltproc">
  957. <arg value="--catalogs"/>
  958. <arg value="--nonet"/>
  959. <arg value="users-guide.xsl"/>
  960. <arg value="users-guide.xml"/>
  961. </exec>
  962. </target>
  963. <!-- Generate docs with Xalan tool from xml.apache.org -->
  964. <!-- NOTE: build target does not work -->
  965. <target name="docs-xalan">
  966. <style
  967. processor="xalan"
  968. style="users-guide.xsl"
  969. in="users-guide.xml"
  970. out="QuickNotepad.html"
  971. destdir="."/>
  972. </target>
  973. <target name="clean">
  974. <delete dir="${build.dir}"/>
  975. <delete>
  976. <fileset dir="." includes="**/*~" defaultexcludes="no"/>
  977. <fileset dir="." includes="**/*.html" defaultexcludes="no"/>
  978. </delete>
  979. </target>
  980. </project>
  981. ]]></programlisting></informalexample>
  982. <para>
  983. For a full discussion of the <filename>Ant</filename> file format and
  984. command syntax, you should consult the <ulink
  985. url="http://jakarta.apache.org/ant/manual/index.html">Ant
  986. documentation site</ulink>. Modifying this makefile for a different
  987. plugin will likely only require three changes:
  988. </para>
  989. <itemizedlist>
  990. <listitem><para>
  991. the name of the plugin;
  992. </para></listitem>
  993. <listitem><para>
  994. the choice of compiler (made by inserting and deleting the comment character
  995. <userinput>'#'</userinput>); and
  996. </para> </listitem>
  997. <listitem><para>
  998. the classpath variables for <filename>jedit.jar</filename>
  999. any plugins this one depends on.
  1000. </para></listitem>
  1001. </itemizedlist>
  1002. <para>
  1003. If you have reached this point in the text, you are probably serious
  1004. about writing a plugin for jEdit. Good luck with your efforts, and
  1005. thank you for contributing to the jEdit project.
  1006. </para>
  1007. </sect1>
  1008. </chapter>