Na prostym przykładzie spróbuję pokazać jak można wykorzystać Netbeans Platform do tworzenia aplikacji desktopowej.
A więc zaczynamy. Będzie nam potrzebny Netbeans (6.0) ;)
Zaczynamy od utworzenia nowego projektu (typu NetBeans Module Suite).
Nazywamy nasz projekt SampleSuite i zapisujemy.
W ustawieniach dla nowo utworzonego projektu z sekcji Libraries wybieramy tylko moduły z klastra Platform7 (inne nie będą nam potrzebne). W sekcji Build wybieramy Create standalone application. Możemy wybrać jaki tytuł będzie nosiła nasza aplikacja, jaką będzie miała ikonę oraz jakim obrazkiem się z nami przywita (sekcja Splash Screen).
W tym momencie mamy już gotowy szkielet aplikacji, który można uruchomić.
Utworzymy teraz nowy moduł, który dodamy do aplikacji. W skład modułu wejdzie drzewko. Po zaznaczeniu jakiegoś liścia w drzewie, po prawej stronie w oknie Properites pojawią się parametry zaznaczonego obiektu.
Wprowadzamy nazwę projektu i nazwę pakietu głównego i dodajemy go do projektu aplikacji.
Wszystkie nowe klasy będziemy tworzyć w nowo utworzonym projekcie modułu. Utworzymy teraz okno z drzewkiem nawigatora. Aby nasze drzewko miało co pokazywać utwórzmy klasę MyDataObject, która będzie reprezentowała jakieś dane. W praktyce klasa ta może reprezentować np. dane z pliku (dla plików mamy gotową klasę FileObject) lub z bazy danych.
public class MyDataObject {
private static int count = 0;
private final int index;
private final String name;
public MyDataObject(){
index = count++;
name = "Nasz " + index + ". obiekt";
}
public String getName(){
return name;
}
public int getIndex(){
return index;
}
}
Każda nowo powołana instancja obiektu MyDataObject będzie posiadała kolejny numer.
Aby można było zbudować drzewo na podstawie MyDataObject potrzebujemy jeszcze klasy typu Node. Node jest warstwą prezentacji dla danych (MyDataObject).
public class MyDataNode extends AbstractNode{
public MyDataNode(DataObject obj) {
super(Children.LEAF, Lookups.singleton(obj));
setDisplayName(obj.getName());
}
public MyDataNode() {
super(Children.create(new MyDataChildFactory(), true));
setDisplayName("Korzeń");
}
}
Jak widzimy klasa MyDataNode rozszerza klasę AbstractNode. Po wklejeniu (lub wpisaniu) powyższego kodu klasy NetBeans najprawdopodobniej nie rozpozna klasy AbstractNode. Musimy do naszego projektu modułu dodać odpowiednie zależności. W tym celu we właściwościach projektu, w kategorii Libraries wybieramy Add dependencies i szukamy modułu zawierającego klasę AbstractNode.
Dodajmy od razu wszystkie moduły, które będą nam później potrzebne
Dalej nierozpoznana pozostaje klasa MyDataChildFactory. Jest to fabryka liści-potomków, którą właśnie utworzymy.
public class MyDataChildFactory extends ChildFactory{
@Override
protected boolean createKeys(Listlist) {
for (int i = 0; i < 10; i++) {
list.add(new MyDataObject());
}
return true;
}
@Override
protected Node createNodeForKey(MyDataObject obj) {
return new MyDataNode(obj);
}
}
Klasa MyDataNode posiada dwa konstruktory. Domyślny konstruktor tworzy korzeń drzewka z MyDataChildFactory jako fabryką dzieci dla niego. Drugi konstruktor tworzy obiekt MyDataNode jako liść bez dzieci (Children.LEAF). Mechanizm Lookup jest tematem na osobny wykład. W wielkim skrócie można powiedzieć, że Lookup jest mapą gdzie kluczem jest typ obiektu a wartością instancje obiektów tego typu. Lookups.singleton(obj) "dba" aby w mapie była tylko jedna instancja obiektu.
MyDataChildFactory posiada dwie metody:
createKeys tworzy listę kluczy (obiektów MyDataObject), dla których będą utworzone nody (węzły) reprezentujące je w strukturze drzewa
createNodeForKey jest wywoływana dla każdego stworzonego klucza w pierwszej metodzie.
Mając już wszystkie potrzebne klasy możemy przejść do stworzenia drzewka.
Tworzymy nowy plik typu Window Component z kategorii Module development.
Window position ustawiamy jako explorer i zaznaczymy otwieranie przy starcie aplikacji.
W następnym oknie kreatora podajemy Class name prefix (dla naszej aplikacji będzie to Tree). Po tym kreator utworzy dla nas 4 pliki TreeAction.java, TreeTopComponent.java, TreeTopComponentSettings.xml, TreeTopComponentWstcref.xml. Pliki xml są plikami konfiguracyjnymi. TreeTopComponent to obiekt klasy TopComponent (póki co możemy przyjąć, że jest to panel, na którym możemy poukładać różne kontrolki Swing), TreeAction to klasa akcji pozwalająca otwierać ten panel jako nowe okno w naszej aplikacji. Jeśli uruchomimy aplikację zobaczymy, że pojawiło się nowe okno ("Tree Window") oraz nowa pozycja w menu Windows (Windows -> Tree). Nazwy dla naszego okna i akcji możemy zmienić w pliku Bundle.properties w pakiecie, w którym te klasy się znajdują (w naszym przypadku org.myorg.samplemodule.tree.Bundle.properties).
Przejdźmy do edycji pliku TreeTopComponent.java. W trybie Design ustawiamy BorderLayout jako layout głównego komponentu (TopComponent w oknie inspektora). W „środek” tego layouta wstawiamy z palety komponentów JScrollPane.
We właściwościach jScrollPane1 w sekcji Code ustawiamy Custom Creation Code.
Klasa BeanTreeView jest rozszerzeniem klasy JScrollPane. Wyświetla ona węzły (Node) w postaci drzewka. Musimy jeszcze uzupełnić import dla klasy BeanTreeView. Przechodzimy do trybu edycji Source i dodajemy brakujący import.
Musimy jeszcze do naszego drzewa dodać obsługę zdarzeń użytkownika (zaznaczanie liścia, itp). W tym celu dodajemy interfejs ExplorerManager.Provider.
final class TreeTopComponent extends TopComponent implements ExplorerManager.Provider{
private ExplorerManager manager = new ExplorerManager();
który wprowadza tylko jedną metodę:
public ExplorerManager getExplorerManager() {
return manager;
}
W konstruktorze okna wskazujemy menadżerowi, jaki obiekt ma być pokazany w drzewie.
private TreeTopComponent() {
...
manager.setRootContext(new MyDataNode());
}
Po uruchomieniu naszej aplikacji, pokaże się nam już drzewko, jednak aplikacja nie będzie na razie w żaden sposób reagowała na zaznaczenie obiektu. Ponieważ chcemy, aby po zaznaczeniu liścia w drzewie, w oknie Properties pokazały się parametry zaznaczonego obiektu, musimy do definicji naszego węzła dodać obsługę properties.
@Override
protected Sheet createSheet() {
Sheet sheet = Sheet.createDefault();
Sheet.Set set = Sheet.createPropertiesSet();
MyDataObject data = getLookup().lookup(MyDataObject.class);
try {
Property indexProp = new PropertySupport.Reflection(data, Integer.class, "getIndex", null);
Property nameProp = new PropertySupport.Reflection(data, String.class, "getName", null);
indexProp.setName("Index");
nameProp.setName("Name");
set.put(indexProp);
set.put(nameProp);
} catch (NoSuchMethodException ex) {
Exceptions.printStackTrace(ex);
}
sheet.put(set);
return sheet;
}
Musimy nadpisać metodę createSheet() w MyDataNode. Aby okno properties wiedziało, że zmieniono zaznaczenie musimy jeszcze dodać do konstruktora TreeTopComponent linijkę:
associateLookup(ExplorerUtils.createLookup(manager, getActionMap()));
Uruchamiamy aplikację:
Aby drzewko wyglądało tak jak na powyższym rysunku (wiele poziomów) wystarczy w konstruktorze MyDataNode(MyDataObject object)
zmienić dzieci liścia z Children.LEAF na Children.create(new MyChildFactory(), true) tak jak w domyślnym konstruktorze.
Brak komentarzy:
Prześlij komentarz