【问题标题】:How can I send my message using websockets to backend如何使用 websockets 将消息发送到后端
【发布时间】:2017-07-29 11:26:35
【问题描述】:

我正在尝试使用 PostgreSQL 数据库在 Django 后端和 Angular 4 前端之间创建实时聊天。假设我想创建聊天机器人,例如A.L.I.C.E。我不确定,但在我看来,最好的解决方案是使用 websockets?我想在前端写消息,按回车并将我的消息发送到后端。那我想得到回应。我正在尝试按照下面显示的方式执行此操作,但是出现错误。我想在前端输入自己的文字,按回车键发送。

app.component.html:

<button (click)="sendMsg()">Send Message</button>

<h4>Type away! Press [enter] when done.</h4>
<div><key-up3 (keyup7)="send($event)"></key-up3></div>

app.component.ts:

import { Component } from '@angular/core';
import { WebsocketService } from './websocket.service';
import { ChatService } from './chat.service';
import { Output, EventEmitter } from '@angular/core';

@Component({
  selector: 'app-root',
  templateUrl: './app.component.html',
  styleUrls: ['./app.component.css'],
  providers: [ WebsocketService, ChatService ]
})
export class AppComponent {

    constructor(private chatService: ChatService) {
        chatService.messages.subscribe(msg => {
      console.log("Response from websocket: " + msg);
        });
    }

    private message = {
        message: 'this is a test message'
    }

  sendMsg() {
        console.log('new message from client to websocket: ', this.message);
        this.chatService.messages.next(this.message);
        this.message.message = '';
    }

  send(msg) {
     this.message = msg;
     this.sendMsg();
  }

}

@Component({
  selector: 'key-up3',
  template: `
    <input #box (keyup.enter)="keyup7.emit(box.value)">
    <p>{{value}}</p>
  `
})
export class KeyUpComponent_v3 {
   @Output() keyup7 = new EventEmitter<string>();
}

websocket.service.ts:

import { Injectable } from '@angular/core';
import * as Rx from 'rxjs/Rx';

@Injectable()
export class WebsocketService {
  constructor() { }

  private subject: Rx.Subject<MessageEvent>;

  public connect(url): Rx.Subject<MessageEvent> {
    if (!this.subject) {
      this.subject = this.create(url);
      console.log("Successfully connected: " + url);
    }
    return this.subject;
  }

  private create(url): Rx.Subject<MessageEvent> {
    let ws = new WebSocket(url);

    let observable = Rx.Observable.create(
    (obs: Rx.Observer<MessageEvent>) => {
        ws.onmessage = obs.next.bind(obs);
        ws.onerror = obs.error.bind(obs);
        ws.onclose = obs.complete.bind(obs);
        return ws.close.bind(ws);
    })
let observer = {
        next: (data: Object) => {
            if (ws.readyState === WebSocket.OPEN) {
                ws.send(JSON.stringify(data));
            }
        }
    }
    return Rx.Subject.create(observer, observable);
  }

}

chat.service.ts:

import { Injectable } from '@angular/core';
import { Observable, Subject } from 'rxjs/Rx';
import { WebsocketService } from './websocket.service';

const CHAT_URL = 'ws://localhost:8000/';

export interface Message {
    message: string
}

@Injectable()
export class ChatService {
    public messages: Subject<Message>;

    constructor(wsService: WebsocketService) {
        this.messages = <Subject<Message>>wsService
            .connect(CHAT_URL)
            .map((response: MessageEvent): Message => {
                let data = JSON.parse(response.data);
                return {
                    message: data.message
                }
            });
    }
}

更新 模板中的 KeyUp.emit(box.value) 为大写 K。组件中为小写。然而,此时当我写东西并按下回车时,什么也没有发生。有什么建议吗?

更新后的错误:

【问题讨论】:

    标签: javascript angular typescript websocket


    【解决方案1】:

    您可以将 EventEmitter 添加到您的 keyUp 组件中

    import { Component, Output, EventEmitter } from '@angular/core';
    
    @Component({
      selector: 'key-up3',
      template: `
        <input #box (keyup.enter)="keyUp.emit(box.value)">
        <p>{{value}}</p>
      `
    })
    export class KeyUpComponent_v3 {
       @Output() keyUp = new EventEmitter<string>();
    }
    

    然后在您的主组件中侦听该事件并相应地设置消息值。您需要公开您的 message 属性,以便可以设置模板。

    <h4>Type away! Press [enter] when done.</h4>
    <div><key-up3 (keyUp)="message = $event"></key-up3></div>
    

    另外:

    RxJS 有一个名为 WebSocketSubject 的类,您可以使用它轻松地从您的服务器发送和接收消息,这是一个服务的快速示例。

    import { Injectable } from '@angular/core';
    import { Observable } from 'rxjs/Rx';
    
    @Injectable()
    export class SocketServer {
    
        private webSocket$ = Observable.webSocket('ws://localhost:3000/');
    
        public asObservable(): Observable<any> {
            return this.webSocket$.retry();
        }
    
        public sendData(value) {
            this.webSocket$.next(value);
        }
    }
    

    .retry() 如果断开连接,将尝试重新连接到 WebSocket

    【讨论】:

    • 你能告诉我为什么你的方法比我的好吗?
    • 我觉得它更简洁。从长远来看,这将使其更容易维护。我希望这会有所帮助。
    • 好的,但是我怎样才能在前端写文本,按回车并将其发送到后端?
    • 这里是与上述服务挂钩的组件示例。 github.com/TheOriginalJosh/electron-angular-websockets-seed/…你需要监听进入按键按键事件
    • 我更新了我的帖子。我有一个组件,我可以在其中编写自己的文本并保存在变量值中,还可以通过单击按钮Send Message 发送消息。您能告诉我如何结合这两个组件来编写此消息并将其发送到后端吗?如果您可以将建议的解决方案与WebSocketSubject 一起使用,我将不胜感激。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2018-06-03
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多