Generische konfigurierbare (Swing-)Actions

vorhergehende Artikel in: Java Komponenten
24.02.2016

Ich wollte eine Infrastruktur schaffen, die Aktionen ausführt als Reaktion auf beliebige Trigger-Ereignisse. Die Ergebnisse der Gedanken, die ich mir dazu gemacht habe, werden hier dargestellt.

Im Prinzip sind die Gedanken, die ich mir gemacht habe einfach zusammenzufassen: Man wähle eine Architektur, die der Swing-Action beim Erzeugen eine Instanz übergibt, die das Interface Configuration implementiert.

Werden die Actions in eine andere Klasse hinein kombiniert, muss diese Klasse das Interface implementieren. Ansonsten ist es natürlich auch möglich, dass eine eigene Konfigurations-Klasse geschaffen wird, die das Interface implementiert. Dadurch wird die Kette zu einer aus drei Gliedern: die Instanz, die die Action nutzen möchte, übergibt eine Instanz der Konfigurationsklasse an die Action. Sie kann das Verhalten der Action dann über Änderungen an der Konfigurationsinstanz beeinflussen.

Das Interface selbst ist zunächst einmal nur ein Marker-Interface:

public interface Configuration
{
	
}

Dazu habe ich eine generische Basisklasse geschrieben, die wie folgt aussieht:

public abstract class ConfigurableActionBase<T extends Configuration> extends javax.swing.AbstractAction
{
	private final T conf;

public ConfigurableActionBase(T conf, String name) { super(name); this.conf = conf; }

public ConfigurableActionBase(T conf, String name, Icon icon) { super(name, icon); this.conf = conf; }

public ConfigurableActionBase(T conf) { super(); this.conf=conf; } public T getConfiguration() { return conf; } }

Damit hat man in abgeleiteten Klassen einen typsicheren Zugriff auf die Konfiguration erreicht. Eine abgeleitete Action-Klasse könnte dann zum Beispiel so aussehen:

public class ConfigurableAction extends ConfigurableActionBase<ConfigurableAction.Configuration>
{
	public ConfigurableAction(Configuration conf)
	{
		super(conf);
	}
	public ConfigurableAction()
	{
		this(new ConfImpl());
	}
	public interface Configuration extends de.elbosso.util.pattern.command.Configuration
	{
		boolean isEnabled();
	}
	public void actionPerformed(java.awt.event.ActionEvent evt)
	{
		System.out.println(getConfiguration().isEnabled()?"enabled":"disabled");
	}
}

Man beachte das Inner-Interface, das das Marker-Interface erweitert. Der zweite Konstruktor dient der Bequemlichkeit: Wenn die Konfiguration über eine eigene Implementationsklasse erfolgt, muss die Klasse, die die Action benutzen möchte, diese Instanz nicht implizit erzeugen, sondern kann einfach den diesen Konstruktor benutzen, durch den eine Default-Implementations-Instanz erstellt wird, auf die wiederum mit der entsprechenden Getter-Methode zugegriffen werden kann. Im Beispiel könnte diese Default-Implementation wie folgt aussehen:

public class ConfImpl implements ConfigurableAction.Configuration
{
	private boolean enabled;

public boolean isEnabled() { return enabled; }

public void setEnabled(boolean enabled) { boolean old=isEnabled(); this.enabled = enabled; } }

Artikel, die hierher verlinken

dWb+ als Rule-Engine

03.03.2016

Eine einfache Rule-Engine - zusammengesetzt auf Basis von dWb+ und einigen weiteren Komponenten.

Alle Artikel rss Wochenübersicht Monatsübersicht Codeberg Repositories Mastodon Über mich home xmpp


Vor 5 Jahren hier im Blog

  • Günstiges Arm64-Board mit Raspi-Formfaktor

    12.12.2020

    Ich habe ein neues Gerät in meinem Hardware-Zoo: Nachdem ich bereits einigeErfahrungen mit Raspis sammeln konnte, machte mich ein Kollege neulich auf eine günstige Variante aufmerksam, Erfahrungen mit ARM64 zu sammeln...

    Weiterlesen

Neueste Artikel

  • Asymmetrische Kryptographie

    Ich habe mich mit der Idee schon länger getragen: Nochmal einen Rundumschlag zu asymmetrischer Kryptographie zu machen. Dabei werde ich mich auf Demonstrationen der einzelnen Konzepte und Operationen mit Beispielcode konzentrieren und zu jedem der vorgestellten Konzepte mehr oder weniger ausführlich bezüglich der Einsatzszenarien und Vor- und Nachteile Stellung beziehen

    Weiterlesen
  • Verschlüsseln und Signieren aus Java heraus nach RFC5652

    Wie bereits angekündigt werde ich in den nächsten Wochen einige Aspekte asymmetrischer Kryptographie beschreiben. Der erste Artikel in der Reihe - dieser hier - zeigt Verschlüsselung und Signieren von Botschaften mittels der Cryptographic Message Syntax (CMS) nach RFC5652.

    Weiterlesen
  • LinkCollections 2025 XI

    Nach der letzten losen Zusammenstellung (für mich) interessanter Links aus den Tiefen des Internet von 2025 folgt hier gleich die nächste:

    Weiterlesen

Manche nennen es Blog, manche Web-Seite - ich schreibe hier hin und wieder über meine Erlebnisse, Rückschläge und Erleuchtungen bei meinen Hobbies.

Wer daran teilhaben und eventuell sogar davon profitieren möchte, muss damit leben, daß ich hin und wieder kleine Ausflüge in Bereiche mache, die nichts mit IT, Administration oder Softwareentwicklung zu tun haben.

Ich wünsche allen Lesern viel Spaß und hin und wieder einen kleinen AHA!-Effekt...

PS: Meine öffentlichen Codeberg-Repositories findet man hier.