I know this has been asked a thousand time, but I think I\'ve tried every solution I\'ve read and I can\'t seem to get it to work.
I have an API from which I\'m getting
As others pointed out you need to wait until all that stuff is rendered which happens only after your api call returns with result. The signal for your directive would be the change of gallery.images
array. So, make your directive watch for it like the following.
Please pay attention to the newly introduced isolated scope - to me it sounds like it should be isolated, but I can't really say it for sure because I don't know the rest of your app code.
Anyway, it tells angular that whatever value is used in photoswipe
attribute in the template (see below) should be bound to the images
property of directive's scope. Using such a construct ensures that this will happen on each and every change of gallery.images
regardless of when and how it happens. All you need to do outside is just to change the collection when api call finishes and the rest will just work. No events (to me they are pretty cumbersome in angular 1.x), no connections between different components, no mixing of concerns. Pretty clean solution. However, you need to understand how $watchCollection() works, it's a bit different from regular $watch
in that it performs shallow collection scan and does not compare objects in collection themselves but only their references.
Another important moment is a cleanup. You must make sure that when something is getting removed from the collection then corresponding events and whatever else that plugin binds to elements is properly destroyed, otherwise you'll end up having memory leaks, unexpected events and poor performance.
App.directive('photoswipe', function ($timeout) {
return {
replace: false,
restrict: 'A',
scope: {
"images": "=photoswipe"
},
link: function photoSwipeLink(scope, element, attr) {
scope.$watchCollection('images', function(value){
// here $timeout() is necessary because this event will be fired
// immediately after collection change
// thus not giving angular time to render it
// so, jquery plugin will be fired on the next angular digest
// and everything will be rendered by then.
$timeout(function () {
angular.element('#gallery a').photoSwipe({
enableMouseWheel: false,
enableKeyboard: false
});
// and now figure out how to clean it up!
});
});
}
};
});
Now to the template. Please note that first time I use photoswipe="gallery.images"
attribute. This is what tells angular to bind gallery.images
to the scope of photoswipe
directive to images
property.
Since directive now introduces its own isolated scope everything inside it should account to this fact. That's why ng-repeat="image in images"
.
{{ gallery.headline }}
I honestly did not test this code but apart from possible syntax errors it should work. Also, I want to stress one more time that I don't know the structure of the rest of your application, so introducing isolated scope can break it if something inside your directive is bound to some higher scope - but to me it is bad idea by itself because it introduces hidden interconnections between components.