parent
cdf99cf68b
commit
601fd3e305
|
@ -67,7 +67,7 @@ export class ClientMessageBroker_ extends ClientMessageBroker {
|
||||||
this._serializer = _serializer;
|
this._serializer = _serializer;
|
||||||
const source = messageBus.from(channel);
|
const source = messageBus.from(channel);
|
||||||
|
|
||||||
source.subscribe({next: (message: MessageData) => this._handleMessage(message)});
|
source.subscribe({next: (message: ResponseMessageData) => this._handleMessage(message)});
|
||||||
}
|
}
|
||||||
|
|
||||||
private _generateMessageId(name: string): string {
|
private _generateMessageId(name: string): string {
|
||||||
|
@ -102,31 +102,33 @@ export class ClientMessageBroker_ extends ClientMessageBroker {
|
||||||
this._pending.set(id, completer);
|
this._pending.set(id, completer);
|
||||||
|
|
||||||
promise.catch((err) => {
|
promise.catch((err) => {
|
||||||
if (console && console.log) {
|
if (console && console.error) {
|
||||||
// tslint:disable-next-line:no-console
|
// tslint:disable-next-line:no-console
|
||||||
console.log(err);
|
console.error(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
completer.reject(err);
|
completer.reject(err);
|
||||||
});
|
});
|
||||||
|
|
||||||
promise = promise.then(
|
promise = promise.then(
|
||||||
(value: any) =>
|
(v: any) => this._serializer ? this._serializer.deserialize(v, returnType) : v);
|
||||||
this._serializer ? value : this._serializer.deserialize(value, returnType));
|
|
||||||
} else {
|
} else {
|
||||||
promise = null;
|
promise = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const message = {'method': args.method, 'args': fnArgs};
|
const message: RequestMessageData = {
|
||||||
|
'method': args.method,
|
||||||
|
'args': fnArgs,
|
||||||
|
};
|
||||||
if (id != null) {
|
if (id != null) {
|
||||||
(message as any)['id'] = id;
|
message['id'] = id;
|
||||||
}
|
}
|
||||||
this._sink.emit(message);
|
this._sink.emit(message);
|
||||||
|
|
||||||
return promise;
|
return promise;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleMessage(message: MessageData): void {
|
private _handleMessage(message: ResponseMessageData): void {
|
||||||
if (message.type === 'result' || message.type === 'error') {
|
if (message.type === 'result' || message.type === 'error') {
|
||||||
const id = message.id;
|
const id = message.id;
|
||||||
if (this._pending.has(id)) {
|
if (this._pending.has(id)) {
|
||||||
|
@ -141,7 +143,13 @@ export class ClientMessageBroker_ extends ClientMessageBroker {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
interface MessageData {
|
interface RequestMessageData {
|
||||||
|
method: string;
|
||||||
|
args?: any[];
|
||||||
|
id?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ResponseMessageData {
|
||||||
type: 'result'|'error';
|
type: 'result'|'error';
|
||||||
value?: any;
|
value?: any;
|
||||||
id?: string;
|
id?: string;
|
||||||
|
|
|
@ -17,18 +17,21 @@ export class RenderStore {
|
||||||
allocateId(): number { return this._nextIndex++; }
|
allocateId(): number { return this._nextIndex++; }
|
||||||
|
|
||||||
store(obj: any, id: number): void {
|
store(obj: any, id: number): void {
|
||||||
|
if (id == null) return;
|
||||||
this._lookupById.set(id, obj);
|
this._lookupById.set(id, obj);
|
||||||
this._lookupByObject.set(obj, id);
|
this._lookupByObject.set(obj, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
remove(obj: any): void {
|
remove(obj: any): void {
|
||||||
const index = this._lookupByObject.get(obj);
|
const index = this._lookupByObject.get(obj);
|
||||||
|
if (index != null) {
|
||||||
this._lookupByObject.delete(obj);
|
this._lookupByObject.delete(obj);
|
||||||
this._lookupById.delete(index);
|
this._lookupById.delete(index);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
deserialize(id: number): any {
|
deserialize(id: number): any {
|
||||||
return id == null || !this._lookupById.has(id) ? null : this._lookupById.get(id);
|
return this._lookupById.has(id) ? this._lookupById.get(id) : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
serialize(obj: any): number { return obj == null ? null : this._lookupByObject.get(obj); }
|
serialize(obj: any): number { return obj == null ? null : this._lookupByObject.get(obj); }
|
||||||
|
|
|
@ -19,7 +19,7 @@ import {RenderStore} from './render_store';
|
||||||
* @experimental WebWorker support in Angular is currently experimental.
|
* @experimental WebWorker support in Angular is currently experimental.
|
||||||
* @deprecated in v4. Use SerializerTypes.PRIMITIVE instead
|
* @deprecated in v4. Use SerializerTypes.PRIMITIVE instead
|
||||||
*/
|
*/
|
||||||
export const PRIMITIVE: Type<any> = String;
|
export const PRIMITIVE = SerializerTypes.PRIMITIVE;
|
||||||
|
|
||||||
export class LocationType {
|
export class LocationType {
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -45,7 +45,7 @@ export class Serializer {
|
||||||
constructor(private _renderStore: RenderStore) {}
|
constructor(private _renderStore: RenderStore) {}
|
||||||
|
|
||||||
serialize(obj: any, type: Type<any>|SerializerTypes = SerializerTypes.PRIMITIVE): Object {
|
serialize(obj: any, type: Type<any>|SerializerTypes = SerializerTypes.PRIMITIVE): Object {
|
||||||
if (obj == null || type === PRIMITIVE || type === SerializerTypes.PRIMITIVE) {
|
if (obj == null || type === SerializerTypes.PRIMITIVE) {
|
||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
if (Array.isArray(obj)) {
|
if (Array.isArray(obj)) {
|
||||||
|
@ -68,7 +68,7 @@ export class Serializer {
|
||||||
|
|
||||||
deserialize(map: any, type: Type<any>|SerializerTypes = SerializerTypes.PRIMITIVE, data?: any):
|
deserialize(map: any, type: Type<any>|SerializerTypes = SerializerTypes.PRIMITIVE, data?: any):
|
||||||
any {
|
any {
|
||||||
if (map == null || type === PRIMITIVE || type === SerializerTypes.PRIMITIVE) {
|
if (map == null || type === SerializerTypes.PRIMITIVE) {
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
if (Array.isArray(map)) {
|
if (Array.isArray(map)) {
|
||||||
|
|
|
@ -265,24 +265,21 @@ export class MessageBasedRendererV2 {
|
||||||
const methods: any[][] = [
|
const methods: any[][] = [
|
||||||
['createRenderer', this.createRenderer, RSO, CRT, P],
|
['createRenderer', this.createRenderer, RSO, CRT, P],
|
||||||
['createElement', this.createElement, RSO, P, P, P],
|
['createElement', this.createElement, RSO, P, P, P],
|
||||||
['createComment', this.createComment, RSO, P, P],
|
['createComment', this.createComment, RSO, P, P], ['createText', this.createText, RSO, P, P],
|
||||||
['createText', this.createText, RSO, P, P],
|
|
||||||
['appendChild', this.appendChild, RSO, RSO, RSO],
|
['appendChild', this.appendChild, RSO, RSO, RSO],
|
||||||
['insertBefore', this.insertBefore, RSO, RSO, RSO, RSO],
|
['insertBefore', this.insertBefore, RSO, RSO, RSO, RSO],
|
||||||
['removeChild', this.removeChild, RSO, RSO, RSO],
|
['removeChild', this.removeChild, RSO, RSO, RSO],
|
||||||
['selectRootElement', this.selectRootElement, RSO, P, P],
|
['selectRootElement', this.selectRootElement, RSO, P, P],
|
||||||
['parentNode', this.parentNode, RSO, RSO, P],
|
['parentNode', this.parentNode, RSO, RSO, P], ['nextSibling', this.nextSibling, RSO, RSO, P],
|
||||||
['nextSibling', this.nextSibling, RSO, RSO, P],
|
|
||||||
['setAttribute', this.setAttribute, RSO, RSO, P, P, P],
|
['setAttribute', this.setAttribute, RSO, RSO, P, P, P],
|
||||||
['removeAttribute', this.removeAttribute, RSO, RSO, P, P],
|
['removeAttribute', this.removeAttribute, RSO, RSO, P, P],
|
||||||
['addClass', this.addClass, RSO, RSO, P],
|
['addClass', this.addClass, RSO, RSO, P], ['removeClass', this.removeClass, RSO, RSO, P],
|
||||||
['removeClass', this.removeClass, RSO, RSO, P],
|
|
||||||
['setStyle', this.setStyle, RSO, RSO, P, P, P, P],
|
['setStyle', this.setStyle, RSO, RSO, P, P, P, P],
|
||||||
['removeStyle', this.removeStyle, RSO, RSO, P, P],
|
['removeStyle', this.removeStyle, RSO, RSO, P, P],
|
||||||
['setProperty', this.setProperty, RSO, RSO, P, P],
|
['setProperty', this.setProperty, RSO, RSO, P, P], ['setValue', this.setValue, RSO, RSO, P],
|
||||||
['setValue', this.setValue, RSO, RSO, P],
|
['listen', this.listen, RSO, RSO, P, P, P], ['unlisten', this.unlisten, RSO, RSO],
|
||||||
['listen', this.listen, RSO, RSO, P, P, P],
|
['destroy', this.destroy, RSO], ['destroyNode', this.destroyNode, RSO, P]
|
||||||
['unlisten', this.unlisten, RSO, RSO],
|
|
||||||
];
|
];
|
||||||
|
|
||||||
methods.forEach(([name, method, ...argTypes]: any[]) => {
|
methods.forEach(([name, method, ...argTypes]: any[]) => {
|
||||||
|
@ -290,6 +287,15 @@ export class MessageBasedRendererV2 {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private destroy(r: RendererV2) { r.destroy(); }
|
||||||
|
|
||||||
|
private destroyNode(r: RendererV2, node: any) {
|
||||||
|
if (r.destroyNode) {
|
||||||
|
r.destroyNode(node);
|
||||||
|
}
|
||||||
|
this._renderStore.remove(node);
|
||||||
|
}
|
||||||
|
|
||||||
private createRenderer(el: any, type: RendererTypeV2, id: number) {
|
private createRenderer(el: any, type: RendererTypeV2, id: number) {
|
||||||
this._renderStore.store(this._rendererFactory.createRenderer(el, type), id);
|
this._renderStore.store(this._rendererFactory.createRenderer(el, type), id);
|
||||||
}
|
}
|
||||||
|
|
|
@ -48,10 +48,10 @@ export class NamedEventEmitter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const globalEvents = new NamedEventEmitter();
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class WebWorkerRootRenderer implements RootRenderer {
|
export class WebWorkerRootRenderer implements RootRenderer {
|
||||||
|
globalEvents = new NamedEventEmitter();
|
||||||
|
|
||||||
private _messageBroker: ClientMessageBroker;
|
private _messageBroker: ClientMessageBroker;
|
||||||
private _componentRenderers = new Map<string, WebWorkerRenderer>();
|
private _componentRenderers = new Map<string, WebWorkerRenderer>();
|
||||||
|
|
||||||
|
@ -79,7 +79,7 @@ export class WebWorkerRootRenderer implements RootRenderer {
|
||||||
const target = message['eventTarget'];
|
const target = message['eventTarget'];
|
||||||
const event = message['event'];
|
const event = message['event'];
|
||||||
if (target) {
|
if (target) {
|
||||||
globalEvents.dispatchEvent(eventNameWithTarget(target, eventName), event);
|
this.globalEvents.dispatchEvent(eventNameWithTarget(target, eventName), event);
|
||||||
} else {
|
} else {
|
||||||
element.events.dispatchEvent(eventName, event);
|
element.events.dispatchEvent(eventName, event);
|
||||||
}
|
}
|
||||||
|
@ -277,7 +277,7 @@ export class WebWorkerRenderer implements Renderer {
|
||||||
}
|
}
|
||||||
|
|
||||||
listenGlobal(target: string, name: string, callback: Function): Function {
|
listenGlobal(target: string, name: string, callback: Function): Function {
|
||||||
globalEvents.listen(eventNameWithTarget(target, name), callback);
|
this._rootRenderer.globalEvents.listen(eventNameWithTarget(target, name), callback);
|
||||||
const unlistenCallbackId = this._rootRenderer.allocateId();
|
const unlistenCallbackId = this._rootRenderer.allocateId();
|
||||||
this._runOnService('listenGlobal', [
|
this._runOnService('listenGlobal', [
|
||||||
new FnArg(target),
|
new FnArg(target),
|
||||||
|
@ -285,7 +285,7 @@ export class WebWorkerRenderer implements Renderer {
|
||||||
new FnArg(unlistenCallbackId),
|
new FnArg(unlistenCallbackId),
|
||||||
]);
|
]);
|
||||||
return () => {
|
return () => {
|
||||||
globalEvents.unlisten(eventNameWithTarget(target, name), callback);
|
this._rootRenderer.globalEvents.unlisten(eventNameWithTarget(target, name), callback);
|
||||||
this._runOnService('listenDone', [new FnArg(unlistenCallbackId)]);
|
this._runOnService('listenDone', [new FnArg(unlistenCallbackId)]);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -322,6 +322,8 @@ function eventNameWithTarget(target: string, eventName: string): string {
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class WebWorkerRendererFactoryV2 implements RendererFactoryV2 {
|
export class WebWorkerRendererFactoryV2 implements RendererFactoryV2 {
|
||||||
|
globalEvents = new NamedEventEmitter();
|
||||||
|
|
||||||
private _messageBroker: ClientMessageBroker;
|
private _messageBroker: ClientMessageBroker;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -359,6 +361,8 @@ export class WebWorkerRendererFactoryV2 implements RendererFactoryV2 {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
freeNode(node: any) { this.renderStore.remove(node); }
|
||||||
|
|
||||||
allocateId(): number { return this.renderStore.allocateId(); }
|
allocateId(): number { return this.renderStore.allocateId(); }
|
||||||
|
|
||||||
private _dispatchEvent(message: {[key: string]: any}): void {
|
private _dispatchEvent(message: {[key: string]: any}): void {
|
||||||
|
@ -370,7 +374,7 @@ export class WebWorkerRendererFactoryV2 implements RendererFactoryV2 {
|
||||||
const event = message['event'];
|
const event = message['event'];
|
||||||
|
|
||||||
if (target) {
|
if (target) {
|
||||||
globalEvents.dispatchEvent(eventNameWithTarget(target, eventName), event);
|
this.globalEvents.dispatchEvent(eventNameWithTarget(target, eventName), event);
|
||||||
} else {
|
} else {
|
||||||
element.events.dispatchEvent(eventName, event);
|
element.events.dispatchEvent(eventName, event);
|
||||||
}
|
}
|
||||||
|
@ -380,13 +384,16 @@ export class WebWorkerRendererFactoryV2 implements RendererFactoryV2 {
|
||||||
|
|
||||||
export class WebWorkerRendererV2 implements RendererV2 {
|
export class WebWorkerRendererV2 implements RendererV2 {
|
||||||
constructor(private _rendererFactory: WebWorkerRendererFactoryV2) {}
|
constructor(private _rendererFactory: WebWorkerRendererFactoryV2) {}
|
||||||
destroyNode: (node: any) => void | null = null;
|
|
||||||
|
|
||||||
private asFnArg = new FnArg(this, SerializerTypes.RENDER_STORE_OBJECT);
|
private asFnArg = new FnArg(this, SerializerTypes.RENDER_STORE_OBJECT);
|
||||||
|
|
||||||
// TODO(vicb): destroy the allocated nodes
|
|
||||||
destroy(): void { this.callUIWithRenderer('destroy'); }
|
destroy(): void { this.callUIWithRenderer('destroy'); }
|
||||||
|
|
||||||
|
destroyNode(node: any) {
|
||||||
|
this.callUIWithRenderer('destroyNode', [new FnArg(node, SerializerTypes.RENDER_STORE_OBJECT)]);
|
||||||
|
this._rendererFactory.freeNode(node);
|
||||||
|
}
|
||||||
|
|
||||||
createElement(name: string, namespace?: string): any {
|
createElement(name: string, namespace?: string): any {
|
||||||
const node = this._rendererFactory.allocateNode();
|
const node = this._rendererFactory.allocateNode();
|
||||||
this.callUIWithRenderer('createElement', [
|
this.callUIWithRenderer('createElement', [
|
||||||
|
@ -543,7 +550,7 @@ export class WebWorkerRendererV2 implements RendererV2 {
|
||||||
[target, null, null];
|
[target, null, null];
|
||||||
|
|
||||||
if (fullName) {
|
if (fullName) {
|
||||||
globalEvents.listen(fullName, listener);
|
this._rendererFactory.globalEvents.listen(fullName, listener);
|
||||||
} else {
|
} else {
|
||||||
targetEl.events.listen(eventName, listener);
|
targetEl.events.listen(eventName, listener);
|
||||||
}
|
}
|
||||||
|
@ -557,7 +564,7 @@ export class WebWorkerRendererV2 implements RendererV2 {
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
if (fullName) {
|
if (fullName) {
|
||||||
globalEvents.unlisten(fullName, listener);
|
this._rendererFactory.globalEvents.unlisten(fullName, listener);
|
||||||
} else {
|
} else {
|
||||||
targetEl.events.unlisten(eventName, listener);
|
targetEl.events.unlisten(eventName, listener);
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,7 +42,7 @@ export function createPairedMessageBuses(): PairedMessageBuses {
|
||||||
export function expectBrokerCall(
|
export function expectBrokerCall(
|
||||||
broker: SpyMessageBroker, methodName: string, vals?: Array<any>,
|
broker: SpyMessageBroker, methodName: string, vals?: Array<any>,
|
||||||
handler?: (..._: any[]) => Promise<any>| void): void {
|
handler?: (..._: any[]) => Promise<any>| void): void {
|
||||||
broker.spy('callUI').and.callFake((args: UiArguments, returnType: Type<any>) => {
|
broker.spy('runOnService').and.callFake((args: UiArguments, returnType: Type<any>) => {
|
||||||
expect(args.method).toEqual(methodName);
|
expect(args.method).toEqual(methodName);
|
||||||
if (isPresent(vals)) {
|
if (isPresent(vals)) {
|
||||||
expect(args.args.length).toEqual(vals.length);
|
expect(args.args.length).toEqual(vals.length);
|
||||||
|
|
|
@ -47,7 +47,7 @@ export declare const platformWorkerApp: (extraProviders?: Provider[]) => Platfor
|
||||||
export declare const platformWorkerUi: (extraProviders?: Provider[]) => PlatformRef;
|
export declare const platformWorkerUi: (extraProviders?: Provider[]) => PlatformRef;
|
||||||
|
|
||||||
/** @experimental */
|
/** @experimental */
|
||||||
export declare const PRIMITIVE: Type<any>;
|
export declare const PRIMITIVE: SerializerTypes;
|
||||||
|
|
||||||
/** @experimental */
|
/** @experimental */
|
||||||
export interface ReceivedMessage {
|
export interface ReceivedMessage {
|
||||||
|
|
Loading…
Reference in New Issue