123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189 |
- /********************************************************************
- # Copyright 2014-2022 Daniel 'grindhold' Brendle, 2015 Daniel Espinosa <esodan@gmail.com>
- #
- # This file is part of libgtkflow.
- #
- # libgtkflow is free software: you can redistribute it and/or
- # modify it under the terms of the GNU Lesser General Public License
- # as published by the Free Software Foundation, either
- # version 3 of the License, or (at your option) any later
- # version.
- #
- # libgtkflow is distributed in the hope that it will be
- # useful, but WITHOUT ANY WARRANTY; without even the implied
- # warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
- # PURPOSE. See the GNU Lesser General Public License for more details.
- #
- # You should have received a copy of the GNU Lesser General Public
- # License along with libgtkflow.
- # If not, see http://www.gnu.org/licenses/.
- *********************************************************************/
- namespace GFlow {
- /**
- * A simple implementation of {@link GFlow.Sink}.
- */
- public class SimpleSink : Object, Dock, Sink {
- // Dock interface
- protected GLib.Type _type = GLib.Type.NONE;
- private string? _name = null;
- /**
- * This SimpleSink's displayname
- */
- public string? name {
- get { return this._name; }
- set { this._name = value; }
- }
- public string? _typename = null;
- /**
- * This SimpleSink's typestring
- */
- public string? typename {
- get { return this._typename; }
- set { this._typename = value; }
- }
- /**
- * Defines how many sources can be connected to this sink
- *
- * Setting this variable to a lower value than the current
- * amount of connected sources will have no further effects
- * than not allowing more connections.
- */
- public uint max_sources {get; set; default=1;}
- /**
- * Indicates whether this Sink should be rendered highlighted
- */
- public bool highlight { get; set; default = false; }
- /**
- * Indicates whether this Sink should be rendered active
- */
- public bool active {get; set; default=false;}
- /**
- * A reference to the {@link Node} that this SimpleSink resides in
- */
- public weak Node? node { get; set; }
- /**
- * The type that this SimpleSink was initialized with
- */
- public GLib.Type value_type { get { return _type; } }
- // Sink Interface
- private List<Source> _sources = new List<Source> ();
- /**
- * The {@link Source}s that this SimpleSink is currently connected to
- */
- public List<Source> sources { get { return _sources; } }
- /**
- * Connects this SimpleSink to the given {@link Source}. This will
- * only succeed if both {@link Dock}s are of the same type. If this
- * is not the case, an exception will be thrown
- */
- protected void add_source(Source s) throws Error
- {
- if (this.value_type != s.value_type) {
- throw new NodeError.INCOMPATIBLE_SINKTYPE(
- "Can't connect. Source has type %s while Sink has type %s".printf(
- s.value_type.name(), this.value_type.name()
- )
- );
- }
- this._sources.append(s);
- s.changed.connect(this.do_source_changed);
- }
- /**
- * Destroys the connection between this SimpleSink and the given {@link Source}
- */
- protected void remove_source (Source s) throws GLib.Error
- {
- if (this._sources.index(s) != -1)
- this._sources.remove(s);
- if (s.is_linked_to(this)) {
- s.unlink (this);
- this.unlinked(s, this._sources.length () == 0);
- }
- }
-
- /**
- * Creates a new SimpleSink with type of given value {@link GLib.Value}
- */
- public SimpleSink(GLib.Value value) {
- _type = value.type();
- }
- /**
- * Creates a new SimpleSink with given type {@link GLib.Type}
- */
- public SimpleSink.with_type(GLib.Type type) {
- _type = type;
- }
- /**
- * Returns true if this sink is connected to at least one source
- */
- public bool is_linked() {
- return this.sources.length() > 0;
- }
- /**
- * Returns true if this SimpleSink is connected to the given {@link Dock}
- */
- public bool is_linked_to (Dock dock) { // FIXME Use more logic to know Source type, value or name
- if (!(dock is Source)) return false;
- return this._sources.index((Source) dock) != -1;
- }
- /**
- * Disconnect from the given {@link Dock}
- */
- public new void unlink (Dock dock) throws GLib.Error {
- if (!this.is_linked_to (dock)) return;
- if (!this.before_unlinking(this, dock)) return;
- if (dock is Source) {
- this.remove_source((Source) dock);
- this.do_source_changed();
- dock.unlinked(this, this.sources.length() == 0);
- dock.changed.disconnect(this.do_source_changed);
- }
- }
- private void do_source_changed(Value? source_value = null, string? flow_id = null) {
- changed(source_value, flow_id);
- }
- /**
- * Connect to the given {@link Dock}
- */
- public new void link (Dock dock) throws GLib.Error {
- if (this.is_linked_to (dock)) return;
- if (!this.before_linking(this, dock)) return;
- if (this._sources.length()+1 > this.max_sources && this.sources.length() > 0) {
- this.unlink(this.sources.nth_data(this.sources.length()-1));
- }
- if (dock is Source) {
- var source = dock as Source;
- add_source(source);
- do_source_changed(source.get_last_value ());
- dock.link (this);
- linked (dock);
- }
- }
- /**
- * Disconnect from any {@link Dock} that this SimpleSink is connected to
- */
- public new void unlink_all() throws GLib.Error {
- foreach (Source s in this._sources)
- if (s != null)
- this.unlink(s);
- }
- }
- }
|