私はAngular 4とwebsocketを使用してチャットアプリケーションを作成しています。このために、これに従いました Angular websocket tutorial
これがWebsocketService
ソースコードです:
_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);
}
}
_
そして、それは私のChatService
です。
_import { Injectable } from '@angular/core';
import { Observable, Subject } from 'rxjs/Rx';
import { WebsocketService } from './websocket.service';
const CHAT_URL = 'ws://echo.websocket.org/';
export interface Message {
author: string,
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 {
author: data.author,
message: data.message
}
});
}
}
_
正常に動作しますが、接続状態を検知したいのですが。接続が中断されたか、サーバーがダウンしているかを知りたい。
そのために、次のようにWebsocketService
クラス内にisServerOn()
関数を実装しようとしました:
_isServerOn(): Observable<boolean> {
return Observable.of(!!this.subject);
}
_
しかし、それは問題を解決していません。同じ問題を奨励した人はいますか?
前もって感謝します。
Angularアプリケーションで socket.io-client の型定義を使用することをお勧めします。次に、次のようにサービスを定義します。
import { Injectable } from '@angular/core';
import { Observable } from 'rxjs/Observable';
import { Observer } from 'rxjs/Observer';
import { Message } from '../model/message';
import { Event } from '../model/event';
import * as socketIo from 'socket.io-client';
const SERVER_URL = 'https://yourserverhost.com';
@Injectable()
export class SocketService {
private socket;
public initSocket(): void {
this.socket = socketIo(SERVER_URL);
}
public send(message: Message): void {
this.socket.emit('message', message);
}
public onEvent(event: Event): Observable<any> {
return new Observable<Event>(observer => {
this.socket.on(event, () => observer.next());
});
}
}
イベント列挙型を定義します。
export enum Event {
CONNECT = 'connect',
DISCONNECT = 'disconnect'
}
次にsubscribe
をサービス関数にAngularコンポーネントから:
export class ChatComponent implements OnInit {
constructor(private socketService: SocketService) { }
ngOnInit(): void {
this.initIoConnection();
}
private initIoConnection(): void {
this.socketService.initSocket();
this.ioConnection = this.socketService.onMessage()
.subscribe((message: Message) => {
this.messages.Push(message);
});
this.socketService.onEvent(Event.CONNECT)
.subscribe(() => {
console.log('Connected to the server');
});
this.socketService.onEvent(Event.DISCONNECT)
.subscribe(() => {
console.log('Disconnected');
});
}
}
Node.js、WebSocketsおよびAngular here: https://github.com/luixaviles/socket-io-TypeScript-chat を使用している完全なチャットプロジェクトを見つけます。 =
あなたが何を達成したいのか分かりません
return Observable.of(!!this.subject);
あなたが思っていることをやっているとは思いません。代わりに、独自のBehaviorSubjectを作成して、次のように対応するObservableを返す必要があります
isServerOn(): Observable<boolean> {
return this.myServerOnSubject.asObservable();
}
対応するオンライン/オフラインコードの位置で、次の値を出力できます
this.myServerOnSubject.next(true/false);