Showing posts with label Xml/Xsl. Show all posts
Showing posts with label Xml/Xsl. Show all posts

Sunday, November 30, 2008

Groovy as extension for Netbeans editor - source code

Hi!
You can download source code of my previous tutorial here.

Alex: Sorry for late.

Cheers! :)

Saturday, October 4, 2008

Groovy as extension for Netbeans editor

As you know I work a lot with xml files (see my Xml/Xsl posts). Xml files are perfect for storing informations in structured way. Sometimes there is need for refactoring those informations. By refactoring I mean nodes/attributes renaming, changing value or place of those. If file is too big, that task may be very-very boring doing it by hand. Of course you can use your favorite text editor's search-replace feature but for complex tasks you will end writing utility for processing xml.
I played a little bit with Groovy and GroovyShell and noticed that you can easily pass your (java) variables to it. It means that you can pass your editor's document and process it with Groovy.
I created simple module for processing xml files. It is based on XPathEvaluator module. It adds a new window to Netbeans where you can write Groovy scripts.


Code for "Run Script" button looks like this:
Binding binding = new Binding();
binding.setVariable("doc", xmlDoc);
binding.setVariable("output", jTextArea1);
GroovyShell shell = new GroovyShell(binding);
shell.evaluate(jEditorPane1.getText());

It binds xmlDoc, which is EditorCookie.getDocument(), to doc variable and evaluates the script which is in jEditorArea.

Maybe a Groovy could be a new way for extending Netbeans? Tell me what you think.

Monday, June 2, 2008

Why I haven't been blogging much lately

Hello everybody!

Well, I was occupied with school and work. Last two months I was programming my diploma work :) and next week I have final exam. I just wanted to say hello and show you what I did using Netbeans.



What you see is an interface of program for modeling the spatial and temporal distribution of concentration changes caused by the diffusion and chemical reactions. I'm specially proud on feature which shows the model in 3D while you type the code. Something like JavaFX editor.
Some technologies I struggled with:
  • Web Service
  • XML/XSD
  • OpenGL

Now I have to learn for next week.
Bye :)

Sunday, February 24, 2008

File opening: EditorCookie or OpenCookie?

Yesterday I wanted to create XML document in memory and open it. On Netbeans Wiki there is explanation and sample code for creating text file and opening it using OpenCookie. But the sample code does not work with XML documents. I debugged the code and noticed that XmlDataObject does not contain OpenCookie class. Instead we have to use EditorCookie. And EditorCookie also works with text files too. So maybe the sample code on Netbeans Wiki should be updated to following:
    FileSystem fs = FileUtil.createMemoryFileSystem();
    FileObject fob = fs.getRoot().createData(name, "txt");

    DataObject data = DataObject.find(fob);
    EditorCookie cookie = (EditorCookie)data.getCookie(EditorCookie.class);
    cookie.open();
Feel free to comment.

Tuesday, February 12, 2008

XPathEvaluator source code

You can download source code of my previous tutorial here. Cheers! :)

Monday, February 4, 2008

XPath evaluator module for Netbeans

Introduction

I do a lot XML and Xslt processing at my workplace. Altova XmlSpy has XPath tool for evaluating queries on edited XML file. I'm missing that (useful) feature in Netbeans so I decided to build something similar. Of course it's far away from Altova's version in features but it can be useful (and you can always extend it).

Creating module project

  1. New project: Choose File > New Project (Ctrl-Shift-N). Under Categories, select NetBeans Modules. Under projects, select Module and click Next.
  2. In the Name and Location panel, type XPathEvaluator in Project Name. Change the Project Location to any directory on your computer, such as /home/damir/Projects/netbeans. Leave the Standalone Module radio button selected.
  3. In the Basic Module Configuration panel, replace yourorghere in Code Name Base with netbeans.modules, so that the whole code name base is org.netbeans.modules.xpathevaluator. Leave the location of the localizing bundle and XML layer, so that they will be stored in a package with the name org/netbeans/modules/xpathevaluator.
  4. Click finish.

Add module dependencies

In project tree right click on Libraries Node > Add module dependency...

  1. Import Datasystems API by typing datasystems
    It contains XmlDataObject which we will listen for.

Create a TopComponent for evaluating queries and displaying results

  1. New file (Ctrl-N). Under Categories, select Module Development. Under file types, select Window Component and click Next.
  2. Window Position: output
  3. Leave Open on Application start unchecked
  4. Class Name Prefix: XPathEvaluator
  5. Package: org.netbeans.modules.xpathevaluator
  6. Click finish
Now lets build interface in design view. We have to add following controls:
  • 2x JLabel
  • 1x JTextField for XPath input
  • 1x JTextArea for XPath results
TopComponent should look like this:

Create action for displaying XPathEvaluatorTopComponent

  1. New file (Ctrl-N). Under Categories, select Module Development. Under file types, select Action and click Next.
  2. Check conditionally enabled and choose EditorCookie for cookie class
  3. GUI registration:
    • Category: XML
    • Uncheck Global Menu Item
    • Check Editor Context Menu and choose text/xml for content type, change position to Tools - HERE
    • Check Separator before and Separator after
  4. Name, Icon, and Location:
    • Class Name: ShowXPathEvaluatorAction
    • Display Name: XPath Evaluator
    • Package Name: org.netbeans.modules.xpathevaluator
  5. Click finish
When user choose XPathNavigator item from xml's context menu we will show him XPathEvaluatorTopComponent but before showing we will create org.w3c.dom.Document from editorCookie's input stream and pass it.
PerformAction method should look like this:

protected void performAction(Node[] activatedNodes)
{
    EditorCookie editorCookie = activatedNodes[0].getLookup().lookup(EditorCookie.class);

    try
    {
        DocumentBuilderFactory domFactory = DocumentBuilderFactory.newInstance();
        domFactory.setNamespaceAware(true);
        DocumentBuilder builder = domFactory.newDocumentBuilder();
        Document doc = builder.parse(((org.openide.text.CloneableEditorSupport) editorCookie).getInputStream());

        XPathEvaluatorTopComponent win = XPathEvaluatorTopComponent.findInstance();
        win.setDocument(doc);
        win.open();
        win.requestActive();
    }
    catch (SAXException ex)
    {
    //Exceptions.printStackTrace(ex);
    }
    catch (ParserConfigurationException ex)
    {
    //Exceptions.printStackTrace(ex);
    }
    catch (IOException ex)
    {
    //Exceptions.printStackTrace(ex);
    }
}
      
Press Ctrl+I to fix imports. Make sure you select org.w3c.dom.Document from combo box.

Extending XPathEvaluatorTopComponent

We have to:

  • create private member which will hold org.w3c.dom.Document from selected editor and on which we will execute XPath.
  • create setDocument(org.w3c.dom.Document doc) method for setting current xml document.
  • create evaluate() method for executing and displaying XPath query.
  • add DocumentListener to jTextField1 for executing XPath query as we type.
  • listen for changes in TopComponent selection (when user opens/activates/closes other xml or non xml panel).
I will focus on last three points and let you struggle with other two :)

evaluate() method

This method is simple. It evaluates text from jTextField1 control and fills jTextArea1 control with results after clearing the same. If there were errors it shows them at bottom of window (jLabel2). xmlDoc is current xml document.

private void evaluate()
{
    if (xmlDoc != null)
    {
        try
        {
            XPathFactory factory = XPathFactory.newInstance();
            XPath xpath = factory.newXPath();
            XPathExpression expr = xpath.compile(jTextField1.getText());

            NodeList nodes = (NodeList) expr.evaluate(xmlDoc, XPathConstants.NODESET);
            jTextArea1.setText("");
            for (int i = 0; i < nodes.getLength(); i++)
            {
                jTextArea1.setText(jTextArea1.getText() + nodes.item(i).getNodeValue() + "\n");
            }
        }
        catch (XPathExpressionException ex)
        {
            jLabel2.setText(ex.getMessage());
        }
    }
}
      

DocumentListener for XPath input

Append following code at the end of XmlPathEvaluatorTopComponent's constructor:

jTextField1.getDocument().addDocumentListener(new DocumentListener()
{
    public void insertUpdate(DocumentEvent e)
    {
        evaluate();
    }

    public void removeUpdate(DocumentEvent e)
    {
        evaluate();
    }

    public void changedUpdate(DocumentEvent e)
    {
    }
});
      

Listen for changes in Lookup

Following code is based on Netbeans Selection Managment Tutorial I. We have to listen for global selection changes of "Object". In our case it is XmlDataObject. We have to extend our TopComponent to implement LookupListener.

final class XPathEvaluatorTopComponent extends TopComponent implements LookupListener
      
LookupListener provides a method for handling changes (resultChanged(LookupEvent ev)). We have to check if there are any instances of XmlDataObject selected and change xmlDoc value according to that.
  • If there are any XmlDataObject instances (in this case only one) then we have set xmlDoc member's value to selected XmlDataObject's XmlDocument and we have to enable input in window.
  • If there are not any instances then we have to set it to null and to disable input on window.
If we put only these two checks then we won't get waiting result. Supose that user selects xml file. If he change focus to XPathEvaluator window, Lookup event will be fired and there won't be any XmlDataObject selected and our module will be unusable.
To avoid this we have to check if our window (XPathEvaluator) is selected.
public void resultChanged(LookupEvent arg0)
{
    // if this window selected then do nothing
    if (TopComponent.getRegistry().getActivated().equals(this))
    {
    // do nothing
    }
    else
    {
        Lookup.Result r = (Lookup.Result) arg0.getSource();

        Collection c = r.allInstances();
        // if there are instances
        if (!c.isEmpty())
        {
            try
            {
                XMLDataObject o = (XMLDataObject) c.iterator().next();
                setDocument(o.getDocument());
                evaluate();
                jLabel2.setText(o.getName());
                jTextArea1.setEnabled(true);
                jTextField1.setEnabled(true);
            }
            catch (IOException ex)
            {
                jLabel2.setText(ex.getMessage());
            }
            catch (SAXException ex)
            {
                jLabel2.setText(ex.getMessage());
            }
        }
        else
        {
            xmlDoc = null;
            jTextArea1.setEnabled(false);
            jTextField1.setEnabled(false);
            jLabel2.setText("no selection");
        }
    }
}
      
Now we have to register LookupListener. We will do that when window is opened:
@Override
public void componentOpened()
{
    Lookup.Template tpl = new Lookup.Template(org.openide.loaders.XMLDataObject.class);
    result = Utilities.actionsGlobalContext().lookup(tpl);
    result.addLookupListener(this);
}
      
When we close window we have to unregister the listener:
@Override
public void componentClosed()
{
    result.removeLookupListener(this);
    result = null;
}
      
We have to create a member for holding the result where lookup listener is registered. You can find explanation for this in Netbeans Selection Managment Tutorial I.

Result