[Kde-bindings] Qyoto: SIGNALS/SLOTS
Arno Rehn
arno at arnorehn.de
Sun Dec 11 14:39:35 UTC 2005
Am Freitag, 9. Dezember 2005 20:16 schrieb Richard Dale:
> On Friday 09 December 2005 16:47, Arno Rehn wrote:
> > Am Donnerstag, 8. Dezember 2005 21:58 schrieb Richard Dale:
> > > On Thursday 08 December 2005 19:32, Arno Rehn wrote:
> > > > Am Mittwoch, 7. Dezember 2005 17:43 schrieb Richard Dale:
> > > > > On Wednesday 07 December 2005 15:36, Arno Rehn wrote:
> > > > > > Hi,
> > > > > >
> > > > > > I experimented a bit and created a more complex Hello-World
> > > > > > sample app. I wanted to connect the "clicked()" signal of a
> > > > > > button to the "quit()" slot of QApplication. Got a SIGSEGV.
> > > > > > I looked into the Qt.cpp of qtruby and it seems as signals and
> > > > > > slots are a bit difficult.
> > > > >
> > > > > QtRuby dynamically constructs QMetaObjects, rather than it being
> > > > > done at compile time by the moc pre-processor. But the actual
> > > > > runtime is identical. We need a way of denoting which methods are
> > > > > slots in C#, and I assume we can use 'attributes' (or are they
> > > > > called
> > > > > 'annotations' I've forgotten). Maybe you could have a look at how
> > > > > we might mark slots, and then the runtime side should be very
> > > > > similar to QtRuby. We don't have to invent anything much new apart
> > > > > from that.
> > > >
> > > > OK, I found out how to mark Signals/Slots with Attributes. I attached
> > > > an example.
> > >
> > > That looks good. In Qt4 there is a macro 'Q_SLOTS' to use instead of
> > > the 'slots' moc keyword because of a clash with the Boost library. So
> > > maybe slots could be marked like this, to look more like that macro:
> > >
> > > [Q_SLOT()]
> > > public void mySlot()
> > > {
> > > }
> > >
> > > Maybe signals need to map onto delegates, rather than ordinary methods.
> >
> > Would be a good idea. Maybe signals need to be delegates, which are set
> > by the programmer or 'QObject.Connect()'. Could look like this:
> >
> > public class QApplication : ...
> > {
> > delegate void LastWindowClosedHandler();
> >
> > [Q_SIGNAL()]
> > LastWindowClosedHandler LastWindowClosed;
> >
> > ...
> > }
>
> Yes, and I think it would need to have a string with the C++ type
> signature, like the Q_SLOT example I give below:
>
> [Q_SIGNAL("void lastWindowClosed()")]
> LastWindowClosedHandler LastWindowClosed;
>
> > Now we could just say 'LastWindowClosed' is set by the programmer. This
> > would keep the delegate-system of Mono/.NET. Additionally it would be
> > possible to create a 'QObject.Connect(...)' which dynamically connects
> > 'LastWindowClosed' to another method. But I don't know how to check
> > whether the underlying C++-instance triggered the signal. Then it would
> > be quite easy to call the appropiate delegate.
>
> We don't want to dynamically connect the delegate to target slots because
> the Qt runtime will do that. But we would need to trap the LastWindowClosed
> delegate call and then raise the Qt signal inside the Qyoto runtime. I need
> to read up more on how delegates work. Should the Q_SIGNAL attribute be
> against the delegate declaration?
>
> [Q_SIGNAL("void lastWindowClosed()")]
> delegate void LastWindowClosedHandler();
>
Yeah, that's right, the attribute has to be against the delegate declaration.
But I don't get the other part. LastWindowClosed is a native Qt-signal. So we
would have to connect the native signal to a C#-slot ( a method or a
delegate/event ). But I don't know how to do that. Can we call a C#-Method
form C++ ?
> > > At the moment an interface is generated containing the type signatures
> > > of signals like this:
> > >
> > > public interface IQApplicationSignals {
> > > void LastWindowClosed();
> > > void AboutToQuit();
> > > void GuiThreadAwake();
> > > }
> > >
> > > And I was thinking of having a QObject.Emit() method which returned a
> > > transparent proxy, so you could invoke a signal with like this:
> > >
> > > (LastWindowClosed) myInstance.Emit()
>
> I got this wrong, it should look like this:
>
> ((IQApplicationSignals) myInstance.Emit()).LastWindowClosed();
>
> And the interface for a class would need to extend the signals interface
> for its superclass. So in the case above the IQApplicationSignals interface
> would need to extend the IQObjectSignals interface. But this doesn't really
> fit in with delegates.
>
> > > And it would be forwarded to SignalInvocation.Invoke(), in a similar
> > > manner to how ordinary method calls get diverted to
> > > SmokeInvocation.Invoke() at the moment.
We don't need a thing like SignalInvocation if all the signal are delegates.
With QObject.Emit() we would just call an delegate. I wouldn't use interaces
at all, just delegates which are declared in the class directly. The
delegates would be inherited by derived classes, too.
> > > But slots are easiest to get working first though, and marking them
> > > with attributes like you've done looks pretty good to me.
>
> I think the slots need to be marked as slots, along with the C++ type
> signature that would be used for constructing the QMetaObject entry for it.
> Like this:
>
> using System;
> using System.Reflection;
>
> [AttributeUsage( AttributeTargets.Method )]
> class Q_SLOT : Attribute
> {
> public string signature;
>
> public string Signature
> {
> get
> {
> return signature;
> }
> }
>
> public Q_SLOT(string signature)
> {
> this.signature = signature;
> }
> }
>
> public class main : Object
> {
> public static void Main(string[] args)
> {
> new main();
> }
>
> public main()
> {
> MethodInfo miSlot = this.GetType().GetMethod( "mySlot" );
> Console.WriteLine( GetAttributeName( miSlot ) );
>
> object[] attributes;
> attributes = miSlot.GetCustomAttributes(typeof(Q_SLOT), false);
> if (attributes.Length > 0) {
> Q_SLOT slotAttr = (Q_SLOT) attributes[0];
> Console.WriteLine( "Q_SLOT signature: {0}", slotAttr.Signature );
> }
>
> }
>
> [Q_SLOT("void mySlot(QString)")]
> public void mySlot(string arg)
> {
> }
>
> public string GetAttributeName( MethodInfo mi )
> {
> Attribute[] attr = ( Attribute[] ) mi.GetCustomAttributes( true );
>
> if( attr.Length < 1 )
> return null;
>
> Type t = attr[0].GetType();
> return t.Name;
> }
> }
>
Looks very good, but my question: How do we connect a native Qt-signal to a
C#-method ? Can we trap these signals and then call a C#-method? Or can we
connect a C++ signal to a C#-method directly?
--
MfG
Arno Rehn
arno at arnorehn.de
More information about the Kde-bindings
mailing list