aboutsummaryrefslogtreecommitdiffstats
path: root/documentation/articles/SendingEventsFromTheClientToTheServerUsingRPC.asciidoc
blob: 67b16b0f5a967c99a6d420476e52235e86501869 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
---
title: Sending Events From The Client To The Server Using RPC
order: 20
layout: page
---

[[sending-events-from-the-client-to-the-server-using-RPC]]
= Sending events from the client to the server using RPC
An RPC mechanism can be used to communicate from the client to the
server. In effect, the client can call methods that are executed by the
server component. The server component can then take appropriate action
- e.g updating the shared state or calling event listeners.

To set up client-server RPC we need to create one interface defining the
RPC methods, and then make use of that interface on both the client and
the server. Place the `MyComponentServerRpc` interface in the client
package:

[source,java]
....
package com.example.mycomponent.client;

import com.vaadin.terminal.gwt.client.MouseEventDetails;
import com.vaadin.terminal.gwt.client.communication.ServerRpc;

public interface MyComponentServerRpc extends ServerRpc {
    public void clicked(MouseEventDetails mouseDetails);
}
....

Note that the RPC methods can not have return values. In this example,
we pass `MouseEventDetails` to get a more complete example, but you
could pass almost any (or no) parameters.

In the server side `MyComponent` we need to implement the interface, and
register it for use:

[source,java]
....
package com.example.mycomponent;

import com.example.mycomponent.client.MyComponentServerRpc;
import com.example.mycomponent.client.MyComponentState;
import com.vaadin.terminal.gwt.client.MouseEventDetails;
import com.vaadin.ui.AbstractComponent;

public class MyComponent extends AbstractComponent {

    private int clickCount = 0;

    private MyComponentServerRpc rpc = new MyComponentServerRpc() {
        public void clicked(MouseEventDetails mouseDetails) {
            clickCount++;
            setText("You have clicked " + clickCount + " times");
        }
    };

    public MyComponent() {
        registerRpc(rpc);
    }

/* Previous code commented out for clarity:
    @Override
    public MyComponentState getState() {
        return (MyComponentState) super.getState();
    }
    public void setText(String text) {
        getState().text = text;
    }
    public String getText() {
        return getState().text;
    }
*/
}
....

Here we react to the RPC call by incrementing a counter. We do not make
use of the `MouseEventDetails` (yet). Notice the *important call to
`registerRpc()`* in the added constructor.

In the client side `MyComponentConnector`, we use `RpcProxy` to get an
implementation of the RPC interface, and call the `clicked()` method
when the widget is clicked:

[source,java]
....
package com.example.mycomponent.client;

// imports removed for clarity
import com.vaadin.terminal.gwt.client.communication.RpcProxy;

@Connect(MyComponent.class)
public class MyComponentConnector extends AbstractComponentConnector {

    MyComponentServerRpc rpc = RpcProxy
            .create(MyComponentServerRpc.class, this);

    public MyComponentConnector() {
        getWidget().addClickHandler(new ClickHandler() {
            public void onClick(ClickEvent event) {

                final MouseEventDetails mouseDetails = MouseEventDetailsBuilder
                        .buildMouseEventDetails(event.getNativeEvent(),
                                getWidget().getElement());

                rpc.clicked(mouseDetails);
            }
        });
    }

/* Previous code commented out for clarity:
    @Override
    protected Widget createWidget() {
        return GWT.create(MyComponentWidget.class);
    }
    @Override
    public MyComponentWidget getWidget() {
        return (MyComponentWidget) super.getWidget();
    }
    @Override
    public MyComponentState getState() {
        return (MyComponentState) super.getState();
    }
    @OnStateChange("text")
    void updateText() {
        getWidget().setText(getState().text);
    }
*/
}
....

Notice that most of the code is for attaching the click handler and
creating the `MouseEventDetails`, the code for the actual RPC is quite
minimal.

Compile the widgetset, and the label text should be updated with the
click count whenever you click it (remember that the counting is done on
the server-side).

Finally, note that you can use multiple RPC interfaces in one component,
allowing for better code separation and reuse.

You can do the same thing in the other direction, see the article about
server to client RPC for details.