Vocês provavelmente ja ouviram falar do Adobe Flex.
O Flex é um SDK para desenvolver aplicações Ricas para Internet, estas aplicações irão rodar no Flash Player que ja esta presente em quase todos os browsers que acessam a internet hoje.
Uma aplicação Flex, gera um arquivo SWF e este arquivo se comunica com um servidor, e este servidor pode ser um Adobe Lifecycle Data Server, mas vocês também podem utilizar Java e Open Source
Estes 2/3 exemplos rodam sem nenhum software pago, as únicas coisas necessárias são:
Flex SDK – O Flex SDK é gratuito e a versão 3 vai ser Open Source também.
Um Servlet Container – Eu estou utilizando o Tomcat
– Para o primeiro exemplo é só isto ![]()
E para os outros dois exemplos, também será necessário baixar os seguintes softwares:
OpenAMF – Uma implementação do protocolo AMF0 (Flex Remoting) em Java
RemoteObjectAMF0 – Uma implementação da tag RemoteObject que suporta a versão 0 do AMF (OpenAMF ainda não suporta a versão 3 e o Flex 2 por padrão utiliza a versão 3)
Então, comecemos criando uma aplicação Web Java na sua IDE favorita, crie um servlet de nome TestServlet, eu vou utilizar JAXB para renderizar XML, mas vocês podem utilizar qualquer outra coisa, se você quiser utilizar JAXB também, o schema que eu utilizei para este exemplo esta abaixo:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | <?xml version="1.0" encoding="UTF-8"?> <schema xmlns="http://www.w3.org/2001/XMLSchema" targetNamespace="http://www.example.com/example" xmlns:tns="http://www.example.com/example" elementFormDefault="qualified"> <element name="TestSVo" type="tns:TestList"></element> <complexType name="TestSVoType"> <attribute name="id" type="int"></attribute> <attribute name="name" type="string"></attribute> <attribute name="other" type="string"></attribute> </complexType> <complexType name="TestList"> <sequence minOccurs="1" maxOccurs="unbounded"> <element name="all" type="tns:TestSVoType"></element> </sequence> </complexType> </schema> |
Este Schema é para um XML parecido com este:
1 2 3 4 5 | <TestSVo> <all id="0" name="foo" other="bar"/> <all id="1" name="foo1" other="bar1"/> ... </TestSVo> |
Agora é preciso gerar os stubs para o JaxB ou utilize qualquer outra ferramenta que gere um XML parecido com este.
Generate the stubs for the XML generation using JaxB or use any other tool to generate a XML with this schema.
Como em qualquer outra aplicação Java WEB, eu vou começar poluindo meu código com XML (o web.xml neste caso):
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 | <?xml version="1.0" encoding="UTF-8"?> <web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd "> <servlet> <servlet-name>testServlet</servlet-name> <servlet-class>....servlet.TestServlet</servlet-class> </servlet> <servlet> <servlet-name>AdvancedGateway</servlet-name> <servlet-class>org.openamf.AdvancedGateway</servlet-class> <init-param> <description> Location of the OpenAMF config file. </description> <param-name>OPENAMF_CONFIG</param-name> <param-value>/WEB-INF/openamf-config.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>AdvancedGateway</servlet-name> <url-pattern>/gateway2</url-pattern> </servlet-mapping> <servlet-mapping> <servlet-name>testServlet</servlet-name> <url-pattern>/TestServlet</url-pattern> </servlet-mapping> <session-config> <session-timeout>30</session-timeout> </session-config> <welcome-file-list> <welcome-file>teste.html</welcome-file> </welcome-file-list> </web-app> |
Para o primeiro exemplo, apenas o TestServlet é utilizado, para os outros o AdvancedGateway que será utilizado …
Agora vamos começar codificando o Servlet:
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 | import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.xml.bind.JAXBContext; import javax.xml.bind.JAXBElement; import javax.xml.bind.JAXBException; import javax.xml.bind.Marshaller; import javax.xml.bind.Unmarshaller; /** * Servlet implementation class for Servlet: TestServlet * */ public class TestServlet extends javax.servlet.http.HttpServlet implements javax.servlet.Servlet { static final long serialVersionUID = 1L; private JAXBContext jc; private Marshaller marc; private Unmarshaller unmarc; private ObjectFactory factory; public TestServlet() { super(); factory = new ObjectFactory(); try { jc = JAXBContext.newInstance("....servlet"); marc = jc.createMarshaller(); unmarc = jc.createUnmarshaller(); } catch (JAXBException e) { e.printStackTrace(); } } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { float v = Float.parseFloat(request.getParameter("v")); float v1 = Float.parseFloat(request.getParameter("v1")); response.getWriter().print(v * v1); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("List called"); TestList res = new TestList(); for(int i=0;i<40;i++){ res.getAll().add(new TestSVoType(i,"Téste ã " + i)); } JAXBElement<TestList> elem = factory.createTestSVo(res); System.out.println(elem.toString()); try { marc.marshal(elem, response.getWriter()); } catch (JAXBException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } |
O método doGet deste servlet é uma calculadora, ele apenas multiplica dois parâmetros recebidos e imprime o resultado.
Já o doPost imprime um XML como o mostrado anteriormente, o conteúdo é praticamente estático, mas isto é só um exemplo
Agora vamos começar a brincar com Flex …
Crie um arquivo de nome xmltest.mxml na pasta raiz da sua aplicação WEB.
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 | <?xml version="1.0" encoding="utf-8"?> <mx:Application xmlns:mx="http://www.adobe.com/2006/mxml" layout="vertical" backgroundColor="#F4F4F4"> <mx:HTTPService id="calc" method="GET" url="http://localhost:8080/teste/TestServlet" > </mx:HTTPService> <mx:HTTPService id="list" method="POST" url="http://localhost:8080/teste/TestServlet" > </mx:HTTPService> <mx:Script> <![CDATA[ import mx.controls.DataGrid; import mx.controls.Alert; ]]> </mx:Script> <mx:Panel width="80%" height="531" title="Teste"> <mx:Label width="100%" color="blue" text="Type two numbers and press calculate." /> <mx:TextInput id="v1" text="3" /> <mx:TextInput id="v2" text="5" /> <mx:Label id="lbl" text="{calc.lastResult}"/> <mx:Button label="Calculate" click="calc.send({v:v1.text,v1:v2.text});" /> <mx:DataGrid dataProvider="{list.lastResult.TestSVo.all}" width="100%" height="195" change="Alert.show(DataGrid(event.currentTarget).selectedItem.other)"> <mx:columns> <mx:DataGridColumn headerText="Id" dataField="id" /> <mx:DataGridColumn headerText="Nome" dataField="name" /> </mx:columns> </mx:DataGrid> <mx:Button label="List" click="list.send({v:v1.text,v1:v2.text})" /> </mx:Panel> </mx:Application> |
A interface com o código java neste exemplo é feita via HTTP padrão, utilizando as tags mx:HTTPService, a primeira faz um GET para a URL e por tanto chama a nossa calculadora, a segunda faz um post, buscando o XML do exemplo Java …
Quando o primeiro botão (Calculate) é clicado, o primeiro serviço é chamado [calc.send(parameters)] e o texto na primeira label é atualizado de acordo com o binding configurado.
Quando o segundo botão é clicadl (List) o segundo serviço é chamado [list.send(params)] e os dados da grid serão atualizados automaticamente.
É só isto ![]()
Dica: se você utilizar list.send sem parâmetros, a tag HTTPservice dispara um GET e não um POST, ou pelo menos foi o que aconteceu com o Flex 2, eu não testei com FLex 3 ainda …
Esta abordagem é bastante simples e não necessita de muito código, mas não parece muito com OOP …
Então vamos escrever a classe TestService com as duas mesmas operações (calc e list)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | import java.util.ArrayList; import java.util.List; public class TestService { public float calc(float v, float v1){ System.out.format("%f X %f\n", v, v1); return v * v1; } public List<TestVo> list(){ System.out.println("List called"); ArrayList<TestVo> res = new ArrayList<TestVo>(); for(int i=0;i<40;i++){ res.add(new TestVo(i,"Testé ã " + i)); } return res; } } |
E agora não sera necessária toda aquela parafernalha XML (bom, pelo menos a parte de código que criamos antes), podemos utilizar um Value Object simples
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 | public class TestVo { private String name; private int id; public TestVo(int id, String name) { this.id = id; this.name = name; } public String getName() { return name; } public int getId() { return id; } public void setName(String name) { this.name = name; } public void setId(int id) { this.id = id; } } |
Para usar este serviço a partir do Flash, nos precisamos configurar o OpenAMF usando o arquivo WEB-INF/openamf-config.xml, o nome e local onde este arquivo se encontra foram configurados no web.xml anteriormente.
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 | <?xml version="1.0" encoding="UTF-8"?> <config> <amf-serializer> <force-lower-case-keys>false</force-lower-case-keys> </amf-serializer> <invoker> <name>Java</name> <class>org.openamf.invoker.JavaServiceInvoker</class> </invoker> <custom-class-mapping> <java-class>....TestVo</java-class> <custom-class>TestVo</custom-class> </custom-class-mapping> <service> <name>TestService</name> <service-location>....TestService</service-location> <invoker-ref>Java</invoker-ref> <method> <name>calc</name> <parameter> <type>*</type> </parameter> </method> <method> <name>list</name> <parameter> <type>*</type> </parameter> </method> </service> </config> |
Eu limpei quase todo o código do arquivo, removi tudo o que não seria utilizado neste exemplo ![]()
O OpenAMF tem diversas opções de configuração, é possível chamar diretamente EJBs, WebServices, qualquer classe Java, objetos JMX e Beans do Spring Framework a partir do seu código Flex sem problema algum …
Não é possível utilizar a tag padrão mx:RemoteObject para chamar este serviço, por que o OpenAMF não suporta o AMF3, e eu não consegui encontrar uma forma de configurar a versão do AMF para a tag mx:RemoteObject …
Então, para o primeiro exemplo, vamos utilizar apenas uma instância da classe NetConnection:
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 | <?xml version="1.0" encoding="utf-8"?> <mx:Application xmlns:mx="http://www.adobe.com/2006/mxml" layout="vertical" backgroundColor="#F4F4F4" initialize="init()"> <mx:Script> <![CDATA[ import mx.controls.Alert; import mx.rpc.events.ResultEvent; import mx.collections.ArrayCollection; [Bindable] public var lista:ArrayCollection; public var nc : NetConnection; function init() : void{ nc = new NetConnection(); nc.objectEncoding = ObjectEncoding.AMF3; nc.addEventListener(NetStatusEvent.NET_STATUS,netStatus); nc.connect("http://localhost:8080/teste/gateway2"); } function netStatus(event : NetStatusEvent) : void { Alert.show(event.info.code); } function onRetornaLista(event):void{ var e : Array = event; lista = new ArrayCollection(e); } function onRetornaCalc(event):void{ lbl.text = String(event); } function calc(v1 : Number, v2 : Number) : void { var r : Responder = new Responder(onRetornaCalc); nc.call("TestService.calc",r,v1,v2); } function list() : void{ var r : Responder = new Responder(onRetornaLista); nc.call("TestService.list",r); } ]]> </mx:Script> <mx:Panel width="80%" height="531" title="Teste"> <mx:Label width="100%" color="blue" text="Type two numbers and press calculate." /> <mx:TextInput id="v1" text="3" /> <mx:TextInput id="v2" text="5" /> <mx:Label id="lbl" /> <mx:Button label="Calculate" click="calc(Number(v1.text),Number(v2.text))" /> <mx:DataGrid dataProvider="{lista}" width="100%" height="195"> <mx:columns> <mx:DataGridColumn headerText="Id" dataField="id" /> <mx:DataGridColumn headerText="Nome" dataField="name" /> </mx:columns> </mx:DataGrid> <mx:Button label="List" click="list()" /> </mx:Panel> </mx:Application> |
O método init() apenas criou e configurou a instância NetConnection da aplicação (nc).
Os botões realizam as mesmas operações do exemplo anterior, mas agora é utilizada a sintaxe Classe.metodo, por exemplo TestService.calc para chamar o método de um serviço via NetConnection.
A novidade aqui é que precisamos dos Responders para tratar os retornos de cada método.
Eu acho esta abordagem um pouco melhor, mas ainda tenho esta sensação de estar fazendo alguma coisa errada …
Então, procurando pela internet eu encontrei a biblioteca RemoteObjectAMF0 …
Esta é uma pequena biblioteca escrita em ActionScript que cria uma tag nova para chamar serviços AMF0 a partir de uma aplicação Flex2.
Para utiliza-la basta fazer estas pequenas alterações na aplicação anterior …
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 | <?xml version="1.0" encoding="utf-8"?> <mx:Application xmlns:mx="http://www.adobe.com/2006/mxml" xmlns:renaun="com.renaun.rpc.*" layout="vertical" backgroundColor="#F4F4F4"> <mx:Script> <![CDATA[ import mx.controls.Alert; import mx.rpc.events.ResultEvent; import mx.collections.ArrayCollection; [Bindable] public var lista:ArrayCollection; NetConnection.defaultObjectEncoding = ObjectEncoding.AMF0; function onRetornaLista(event:ResultEvent):void{ lista = ArrayCollection(event.result); } function onRetornaCalc(event:ResultEvent):void{ lbl.text = String(event.result); } ]]> </mx:Script> <renaun:RemoteObjectAMF0 endpoint="http://localhost:8080/teste/gateway2" id="TST" source="....TestService" showBusyCursor="true" makeObjectsBindable="true" fault="Alert.show(String(event.fault)), 'Error'"> <renaun:methods> <renaun:method name="list" result="onRetornaLista(event)"></renaun:method> <renaun:method name="calc" result="onRetornaCalc(event)"></renaun:method> </renaun:methods> </renaun:RemoteObjectAMF0> <mx:Panel width="80%" height="531" title="Teste"> <mx:Label width="100%" color="blue" text="Type two numbers and press calculate." /> <mx:TextInput id="v1" text="3" /> <mx:TextInput id="v2" text="5" /> <mx:Label id="lbl" /> <mx:Button label="Calculate" click="TST.calc(Number(v1.text),Number(v2.text))" /> <mx:DataGrid dataProvider="{lista}" width="100%" height="195"> <mx:columns> <mx:DataGridColumn headerText="Id" dataField="id" /> <mx:DataGridColumn headerText="Nome" dataField="name" /> </mx:columns> </mx:DataGrid> <mx:Button label="List" click="TST.list()" /> </mx:Panel> </mx:Application> |
Ainda não ficou perfeito, esta quase tão bom quanto com as tags de remoting nativas, mas não tão bom quanto, mesmo assim é a melhor possibilidade deste post ![]()
Eu não gosto da idéia de precisar informar o nome completo da classe Java no flex para chamar um serviço Java, acho que a minha solução final vai ser algo parecido com um proxy usando uma NetConnection como backend, mas esta ja esta suficiente por enquanto
Eu espero que este post ajude algum de vocês, eu passei dois dias procurando uma solução aceitável para este tipo de comunicação, ja que infelizmente no meu projeto atual, o Lifecycle Data Services não á uma opção viável …
PS.: para compilar os arquivos .mxml você pode utilizar o comando mxmlc do Flex 2 SDK
PS2.: Você precisara corrigir as URLs para o caminho completo da aplicação
PS3.: os é e ã não são problemas com a codificação da página, eu adicionei eles propositalmente para ver como seria tratado o encoding diferente no flex e no java, e tudo correu muito bem
English Version here
Antes de iniciar a implementação do chat, nos precisamos instalar as dependências do juggernaut, ele precisa das gems json e eventmachine instaladas, para instalar ambas basta executar o seguinte comando:
Agora estamos prontos para começar a brincar …
Primeiro, vamos criar uma aplicação rails e instalar o plugin juggernaut com os seguitnes comandos:
O Juggernaut usa um servidor externo implementando Flash XML Push, e precisamos configurar este servidor, então vamos editar o arquivo: config/juggernaut.yml
Coloquei aqui no blog apenas as configurações que precisei alterar
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | PUSH_PORT: 8080 ... DEFAULT_CHANNELS: - "chat" ... PUSH_HELPER_HOST: "localhost" ... SECRET: "481516232342edededededed" ... LOGIN_GET_URL: "http://localhost:3000/session/login" LOGOUT_GET_URL: "http://localhost:3000/session/logout" ... SESSION_ID: "_chattest_session_id" ... BASE64: true |
Precisei alterar PUSH_PORT Por que no linux apenas o root pode ouvir em portas inferiores a 1024, e mesmo assim eu não acho que a porta 443 padrão seja uma boa escolha, ja que esta é a porta padrão do protocolo HTTPS.
A linhe com PUSH_HELPER_HOST precisa ser alterada para o mesmo nome do servidor que for utilizado para a acessar a aplicação, em desenvolvimento, localhost deve resolver, mas não esqueça de alterar esta configuração quando for fazer o deploy do seu chat.
LOGIN_GET_URL e LOGOUT_GET_URL são utilizadas para notificar a aplicação quando um usuário conecta ou desconecta, apenas o desconectar é realmente importante para nós.
A SESSION_ID precisa ser igual ao nome do cookie utilizado como marcador de sessão da aplicação, no rails 1.2.x fica no application controller, no rails 2.0 esta configuração vai para o enviroment.rb
e BASE64 é a configuração mágica que habilita o uso dos helpers do rails para geração do Javascript que vai ser utilizado.
Agora vamos começar a criar o layout da aplicação.
Crie um arquivo de nome pubic/stylesheets/public.css com o seguinte conteúdo:
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 | body { background-color: white; } #users { float: left; width: 200px; height: 400px; border-style: inset; overflow: auto; color: white; background-color: gray; } #dasd { height: 400px; margin-left: 5px; border-style: inset; overflow: auto; color: white; background-color: gray; } #controls { clear: both; padding: 0 0 0 0; height: 55px; vertical-align: top; border-style: inset; overflow: auto; color: white; background-color: gray; } |
e o arquivo: app/views/layouts/application.rhtml com o seguinte conteúdo.
1 2 3 4 5 6 7 8 9 10 | <html> <head> <title>Chat Test</title> <%= stylesheet_link_tag 'public' %> <%= javascript_include_tag :defaults %> </head> <body> <%= yield %> </body> </html> |
Com o layout pronto (sim, esta bem feio, mas eu não sou designer eu sou um desenvolvedor, então solicite ao designer da sua equipe a criação de um novo layout para este chat
) vamos criar o código fonte necessário e o banco de dados com estes 4 comandos:
Layout pronto, arquivos criados, agora só precisamos escrever o código da aplicação ![]()
Vamos começar editando o model OnlineUser (app/model/online_user.rb) para adicionar algumas validações, elas não são realmente necessárias, são resquícios da primeira tentativa de implementação do chat, mas mesmo assim mantive elas no código
1 2 3 4 | class OnlineUser < ActiveRecord::Base validates_presence_of :username, :session_id, :last_seen validates_uniqueness_of :username, :if => Proc.new {|user| user.online } end |
Agora a view principal da aplicação no arquivo:app/views/chat/index.rhtml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | <!-- Register with Juggernault --> <%= listen_to_juggernaut_channels [:generic],session.session_id %> <!-- The Users List --> <div id="users"> <ul id="users_list"></ul> </div> <!-- The messages pane --> <div id="dasd"></div> <!-- The controls pane (login and send messages) --> <div id="controls"><%= render :partial => 'login' %></div> <!-- An util javascript to scroll the messages window --> <script type="text/javascript"> function scrollMessages(){ $('dasd').scrollTop = $('dasd').scrollHeight; } </script> |
Como podem ver é bem simples, apenas 3 DIVs, um javascript para fazer o scroll da tela de mensagens e um comando para inicializar o juggernaut no canal “generic” e utilizando o session_id como identificador individual do usuário.
A DIV de mensagens esta com o id dasd por que eu estava testando algumas possibilidades de conflito e esqueci de mudar o ID depois
Como pode ser visto na página index, precisaremos de um partial de nome “login” e vamos também precisar de um de nome “controls”, então vamos começar codificando o partial “controls”: app/views/chat/_controls.rhtml
1 2 3 4 5 6 | <% form_remote_tag( :url => { :action => :say }, :complete => "$('message').value = '';$('message').focus();" ) do %> <%= text_field_tag( 'message', '', { :size => 90, :id => 'message'} ) %> <%= submit_tag "Send" %> <% end %> |
Bastante simples, possui apenas uma tag remote_form_tag que fara um post para a action “say”, um campo de texto para a mensagem e um botão de submit, quando o formulário é enviado o foco volta para o campo message para que o usuário possa digitar a próxima mensagem …
E o partial de login: app/views/chat/_login.rhtml
1 2 3 4 5 6 7 | <%= "#{@message}<br/>" if @message %><% form_remote_tag( :url => { :action => :login }, :complete => "$('username').value = ''", :after => "$('login').disabled = true" ) do %> <%= text_field_tag( 'username', '', { :size => 90, :id => 'username'} ) %> <%= submit_tag "Join", :id => 'login' %> <% end %> |
Bastante parecido com o anterior, mas faz o post para uma action de nome “login”, e o javascript agora desabilita o botão quando o formulário é enviado, e também possui uma mensagem que deve aparecer para o usuário informando que o apelido ja esta sendo utilizado.
Agora todas as views ja estão prontas, podemos passar para a codificação da lógica da aplicação.
Como podemos ver nas views até agora, precisaremos de um controller de nome “chat” com dois métodos: login e say
vamos analisar um pouco o código do chat controller: app/controllers/chat_controller.rb
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 | class ChatController < ApplicationController #this method does not need to exist, but I like to see it here, it only needs to render the index.rhtml view def index end def login #creates a new OnlineUser record, this is used to store who are the users that are online now @user = OnlineUser.new @user.username = Juggernaut.html_and_string_escape params[:username] @user.session_id = session.session_id @user.online = true @user.last_seen = Time.now #if we can save, it means that there is no other user with the same nick online, so this user can join the chat if @user.save #let's save the username in the session for future reference session[:username] = @user.username #if there are online users, fill the users box for the new user know who is online @users = OnlineUser.find(:all, :conditions => ["online = true and id != ?", @user.id]) if @users.size >0 data = render_to_string(:update) do |page| @users.each {|u| page.insert_html :bottom, :users_list, %Q{<li id="user_#{u.username}">#{u.username}</li>} } end #send the javascript only to the new user Juggernaut.send_to(@user.session_id, data) end #create a javascript call to add the new user to the end of the online users list data = render_to_string(:update) do |page| page.insert_html :bottom, :users_list, %Q{<li id="user_#{@user.username}">#{@user.username}</li>} page.insert_html :bottom, :dasd, "<b>user #{@user.username} just joined the chat</b><br/>" end #add the new user to the chat channel Juggernaut.add_channel(@user.session_id, 'chat') #send the javascript to all users in the chat channel Juggernaut.send_data(data, 'chat') render(:update) do |page| page.replace_html 'controls', :partial => "controls" end else @message = 'This nick name is already in use, please choose another' render(:update) do |page| page.replace_html 'controls', :partial => "login" end end end def say #escape the message, that way the user can not harm others sending HTML ot JavaScript commands message = "#{session[:username]}: #{Juggernaut.html_and_string_escape(params[:message])}" #create a javascript to add the new message to the end of the messages screen and scroll the div to the bottom data = render_to_string(:update) do |page| page.insert_html :bottom, :dasd, "#{message}<br/>" page.call "scrollMessages" end #send the message to all users Juggernaut.send_data(data, 'chat') render :nothing => true end end |
O método say é bastante simples, então vamos começar analisando este:
Primeiro a mensagem é reconstruída adicionando o nome do usuário no inicio da mensagem enviada removendo todos os possíveis códigos HTML ou Javascript para garantir a segurança dos outros usuários.
Então é construido um Javascript utilizando o JavaScriptBuilder do rails e o método render_to_string para adicionar a nova mensagem no final da div “dasd” e fazer o scroll caso necessário.
Então este Javascript é enviado para todos os clientes que estão ouvindo o canal “chat”.
Agora um pouco sobre o método login:
Esta é quase toda a lógica que precisamos, a única parte que falta é remover um usuário da lista de usuários online de todos os participantes quando este fechar o browser ou deixar o chat por qualquer motivo, para isto vamos ao método logout do controller session: app/controllers/session_controller.rb
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 | class SessionController < ApplicationController #Called when a user disconnect (a refresh in the browser causes this to be called too) def logout #search for the user record using the session_id @u = OnlineUser.find_by_session_id(session.session_id) reset_session #if a user was found if @u username = @u.username #remove it from the database @u.destroy #remove from the online users list from all users, and tell others this user left the chat data = render_to_string(:update) do |page| page.remove "user_#{username}" page.insert_html :bottom, :dasd, "<b>User #{username} left the chat</b><br/>" end Juggernaut.send_data(data,'chat') end render :nothing => true end def login render :nothing => true end end |
O método login não faz nada, pois não precisamos desta notificação para o chat, então vamos a descrição do método logout …
E isto é tudo pessoal! (isto só fica engraçado em ingles
)
Agora só falta executar a aplicação.
Para executar esta aplicação, precisamos iniciar o servidor do Rails como padrão, e então iniciar o push_server, para fazer isto basta executar os seguintes dois comandos.
E acessar o seu chat novo em folha no endereço: http://localhost:3000/chat
Eu criei este exemplo emquanto estudava um pouco sobre o Juggernaut, é possível que existam maneiras mais fáceis de implementar isto, mas eu acredito que seja um bom ponto de partida.
Quaisquer dicas para melhorar o exemplo são muito bem vindas.
Tags: ajax, chat, example, flash, howto, push, rails, reverse, Ruby, simple, xml