2016-03-23 18:39:41 +01:00
|
|
|
/* angular-summernote v0.7.1 | (c) 2016 JeongHoon Byun | MIT license */
|
2015-05-13 18:30:35 +02:00
|
|
|
/* global angular */
|
|
|
|
angular.module('summernote', [])
|
|
|
|
|
|
|
|
.controller('SummernoteController', ['$scope', '$attrs', '$timeout', function($scope, $attrs, $timeout) {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var currentElement,
|
|
|
|
summernoteConfig = $scope.summernoteConfig || {};
|
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
if (angular.isDefined($attrs.height)) { summernoteConfig.height = +$attrs.height; }
|
|
|
|
if (angular.isDefined($attrs.minHeight)) { summernoteConfig.minHeight = +$attrs.minHeight; }
|
|
|
|
if (angular.isDefined($attrs.maxHeight)) { summernoteConfig.maxHeight = +$attrs.maxHeight; }
|
|
|
|
if (angular.isDefined($attrs.placeholder)) { summernoteConfig.placeholder = $attrs.placeholder; }
|
2015-05-13 18:30:35 +02:00
|
|
|
if (angular.isDefined($attrs.focus)) { summernoteConfig.focus = true; }
|
|
|
|
if (angular.isDefined($attrs.airmode)) { summernoteConfig.airMode = true; }
|
|
|
|
if (angular.isDefined($attrs.lang)) {
|
|
|
|
if (!angular.isDefined($.summernote.lang[$attrs.lang])) {
|
|
|
|
throw new Error('"' + $attrs.lang + '" lang file must be exist.');
|
|
|
|
}
|
|
|
|
summernoteConfig.lang = $attrs.lang;
|
|
|
|
}
|
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
var callbacks = {};
|
|
|
|
callbacks.onInit = $scope.init;
|
|
|
|
callbacks.onEnter = function(evt) { $scope.enter({evt:evt}); };
|
|
|
|
callbacks.onFocus = function(evt) { $scope.focus({evt:evt}); };
|
|
|
|
callbacks.onPaste = function(evt) { $scope.paste({evt:evt}); };
|
|
|
|
callbacks.onKeyup = function(evt) { $scope.keyup({evt:evt}); };
|
|
|
|
callbacks.onKeydown = function(evt) { $scope.keydown({evt:evt}); };
|
2015-05-13 18:30:35 +02:00
|
|
|
if (angular.isDefined($attrs.onImageUpload)) {
|
2016-03-23 18:39:41 +01:00
|
|
|
callbacks.onImageUpload = function(files) {
|
|
|
|
$scope.imageUpload({files:files, editable: $scope.editable});
|
2015-05-13 18:30:35 +02:00
|
|
|
};
|
|
|
|
}
|
2016-03-23 18:39:41 +01:00
|
|
|
if (angular.isDefined($attrs.onMediaDelete)) {
|
|
|
|
callbacks.onMediaDelete = function(target) {
|
|
|
|
// make new object that has information of target to avoid error:isecdom
|
|
|
|
var removedMedia = {attrs: {}};
|
|
|
|
removedMedia.tagName = target[0].tagName;
|
|
|
|
angular.forEach(target[0].attributes, function(attr) {
|
|
|
|
removedMedia.attrs[attr.name] = attr.value;
|
|
|
|
});
|
|
|
|
$scope.mediaDelete({target: removedMedia});
|
|
|
|
}
|
|
|
|
}
|
2015-05-13 18:30:35 +02:00
|
|
|
|
|
|
|
this.activate = function(scope, element, ngModel) {
|
|
|
|
var updateNgModel = function() {
|
2016-03-23 18:39:41 +01:00
|
|
|
var newValue = element.summernote('code');
|
|
|
|
if (element.summernote('isEmpty')) { newValue = ''; }
|
2015-05-13 18:30:35 +02:00
|
|
|
if (ngModel && ngModel.$viewValue !== newValue) {
|
|
|
|
$timeout(function() {
|
|
|
|
ngModel.$setViewValue(newValue);
|
|
|
|
}, 0);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
callbacks.onChange = function(contents) {
|
|
|
|
$timeout(function() {
|
|
|
|
if (element.summernote('isEmpty')) { contents = ''; }
|
|
|
|
updateNgModel();
|
|
|
|
}, 0);
|
2015-05-13 18:30:35 +02:00
|
|
|
$scope.change({contents:contents, editable: $scope.editable});
|
|
|
|
};
|
2016-03-23 18:39:41 +01:00
|
|
|
callbacks.onBlur = function(evt) {
|
|
|
|
(!summernoteConfig.airMode) && element.blur();
|
|
|
|
$scope.blur({evt:evt});
|
|
|
|
};
|
|
|
|
summernoteConfig.callbacks = callbacks;
|
2015-05-13 18:30:35 +02:00
|
|
|
element.summernote(summernoteConfig);
|
|
|
|
|
|
|
|
var editor$ = element.next('.note-editor'),
|
|
|
|
unwatchNgModel;
|
|
|
|
editor$.find('.note-toolbar').click(function() {
|
|
|
|
updateNgModel();
|
|
|
|
|
|
|
|
// sync ngModel in codeview mode
|
|
|
|
if (editor$.hasClass('codeview')) {
|
|
|
|
editor$.on('keyup', updateNgModel);
|
|
|
|
if (ngModel) {
|
|
|
|
unwatchNgModel = scope.$watch(function () {
|
|
|
|
return ngModel.$modelValue;
|
|
|
|
}, function(newValue) {
|
|
|
|
editor$.find('.note-codable').val(newValue);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
editor$.off('keyup', updateNgModel);
|
|
|
|
if (angular.isFunction(unwatchNgModel)) {
|
|
|
|
unwatchNgModel();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (ngModel) {
|
|
|
|
ngModel.$render = function() {
|
2016-03-23 18:39:41 +01:00
|
|
|
if (ngModel.$viewValue) {
|
|
|
|
element.summernote('code', ngModel.$viewValue);
|
|
|
|
} else {
|
|
|
|
element.summernote('empty');
|
|
|
|
}
|
2015-05-13 18:30:35 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// set editable to avoid error:isecdom since Angular v1.3
|
|
|
|
if (angular.isDefined($attrs.editable)) {
|
|
|
|
$scope.editable = editor$.find('.note-editable');
|
|
|
|
}
|
2016-03-23 18:39:41 +01:00
|
|
|
if (angular.isDefined($attrs.editor)) {
|
|
|
|
$scope.editor = element;
|
|
|
|
}
|
2015-05-13 18:30:35 +02:00
|
|
|
|
|
|
|
currentElement = element;
|
|
|
|
// use jquery Event binding instead $on('$destroy') to preserve options data of DOM
|
|
|
|
element.on('$destroy', function() {
|
2016-03-23 18:39:41 +01:00
|
|
|
element.summernote('destroy');
|
2015-05-13 18:30:35 +02:00
|
|
|
$scope.summernoteDestroyed = true;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.$on('$destroy', function () {
|
|
|
|
// when destroying scope directly
|
|
|
|
if (!$scope.summernoteDestroyed) {
|
2016-03-23 18:39:41 +01:00
|
|
|
currentElement.summernote('destroy');
|
2015-05-13 18:30:35 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}])
|
|
|
|
.directive('summernote', [function() {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
return {
|
|
|
|
restrict: 'EA',
|
2016-03-23 18:39:41 +01:00
|
|
|
transclude: 'element',
|
2015-05-13 18:30:35 +02:00
|
|
|
replace: true,
|
2016-03-23 18:39:41 +01:00
|
|
|
require: ['summernote', '?ngModel'],
|
2015-05-13 18:30:35 +02:00
|
|
|
controller: 'SummernoteController',
|
|
|
|
scope: {
|
|
|
|
summernoteConfig: '=config',
|
|
|
|
editable: '=',
|
2016-03-23 18:39:41 +01:00
|
|
|
editor: '=',
|
2015-05-13 18:30:35 +02:00
|
|
|
init: '&onInit',
|
|
|
|
enter: '&onEnter',
|
|
|
|
focus: '&onFocus',
|
|
|
|
blur: '&onBlur',
|
|
|
|
paste: '&onPaste',
|
|
|
|
keyup: '&onKeyup',
|
|
|
|
keydown: '&onKeydown',
|
|
|
|
change: '&onChange',
|
2016-03-23 18:39:41 +01:00
|
|
|
imageUpload: '&onImageUpload',
|
|
|
|
mediaDelete: '&onMediaDelete'
|
2015-05-13 18:30:35 +02:00
|
|
|
},
|
|
|
|
template: '<div class="summernote"></div>',
|
2016-03-23 18:39:41 +01:00
|
|
|
link: function(scope, element, attrs, ctrls, transclude) {
|
2015-05-13 18:30:35 +02:00
|
|
|
var summernoteController = ctrls[0],
|
|
|
|
ngModel = ctrls[1];
|
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
transclude(scope, function(clone, scope) {
|
|
|
|
// to prevent binding to angular scope (It require `tranclude: 'element'`)
|
|
|
|
element.append(clone.html());
|
|
|
|
});
|
|
|
|
|
2015-05-13 18:30:35 +02:00
|
|
|
summernoteController.activate(scope, element, ngModel);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}]);
|