-
Notifications
You must be signed in to change notification settings - Fork 41
/
Copy pathwidgets.js
115 lines (106 loc) · 3.5 KB
/
widgets.js
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
'use strict';
angular.module('angularWidgetInternal')
.provider('widgets', function () {
var manifestGenerators = [];
var eventsToForward = [];
var servicesToShare = {};
this.setManifestGenerator = function (fn) {
manifestGenerators.push(fn);
};
this.addEventToForward = function (name) {
eventsToForward = eventsToForward.concat(name);
};
this.addServiceToShare = function (name, description) {
servicesToShare[name] = description;
};
this.$get = function ($injector, $rootScope) {
var widgets = [];
var instancesToShare = {};
//this will wrap setters so that we can run a digest loop in main app
//after some shared service state is changed.
function decorate(service, method, count) {
var original = service[method];
service[method] = function () {
if (arguments.length >= count && !$rootScope.$$phase) {
$rootScope.$applyAsync();
}
return original.apply(service, arguments);
};
}
angular.forEach(servicesToShare, function (description, name) {
var service = $injector.get(name);
if (angular.isArray(description)) {
description.forEach(function (method) {
decorate(service, method, 0);
});
} else {
angular.forEach(description, function (count, method) {
decorate(service, method, count);
});
}
instancesToShare[name] = service;
});
function notifyInjector(injector, args) {
var scope = injector.get('$rootScope');
var event;
if (args.length) {
event = scope.$broadcast.apply(scope, args);
}
if (!scope.$$phase && injector !== $injector) {
scope.$digest();
}
return event;
}
manifestGenerators = manifestGenerators.map(function (generator) {
return $injector.invoke(generator);
});
return {
getWidgetManifest: function () {
var args = arguments;
return manifestGenerators.reduce(function (prev, generator) {
var result = generator.apply(this, args);
if (result && prev) {
//take the manifest with higher priority.
//if same priority, last generator wins.
return prev.priority > result.priority ? prev : result;
} else {
return result || prev;
}
}, undefined);
},
unregisterWidget: function (injector) {
var del = [];
if (injector) {
var i = widgets.indexOf(injector);
if (i !== -1) {
del = widgets.splice(i, 1);
}
} else {
del = widgets;
widgets = [];
}
del.forEach(function (injector) {
var $rootScope = injector.get('$rootScope');
$rootScope.$destroy();
$rootScope.$$childHead = $rootScope.$$childTail = null;
$rootScope.$$ChildScope = null;
});
},
registerWidget: function (injector) {
widgets.push(injector);
},
notifyWidgets: function () {
var args = arguments;
return widgets.map(function (injector) {
return notifyInjector(injector, args);
});
},
getEventsToForward: function () {
return eventsToForward;
},
getServicesToShare: function () {
return instancesToShare;
}
};
};
});