Файл: sys/themes/default_full/js.js
Строк: 563
<?php
/**
* Вставка текста по выделению
* @param {HTMLInputElement} node
* @param {String} Open Текст, вставляемый перед выделением
* @param {String} Close Текст, вставляемый после выделения
* @param {boolean = false} CursorEnd Флаг, указывающий на необходимость установки курсора после вставленного текста
* @returns {boolean}
*/
function InputInsert(node, Open, Close, CursorEnd) {
node.focus();
if (window.attachEvent && navigator.userAgent.indexOf('Opera') === -1) { // IE
var s = node.sel;
if (s) {
var l = s.text.length;
s.text = Open + s.text + Close;
s.moveEnd("character", -Close.length);
s.moveStart("character", -l);
s.select();
}
} else {
var ss = node.scrollTop;
var sel1 = node.value.substr(0, node.selectionStart);
var sel2 = node.value.substr(node.selectionEnd);
var sel = node.value.substr(node.selectionStart, node.selectionEnd - node.selectionStart);
node.value = sel1 + Open + sel + Close + sel2;
if (CursorEnd) {
node.selectionStart = sel1.length + Open.length + sel.length + Close.length;
node.selectionEnd = node.selectionStart;
} else {
node.selectionStart = sel1.length + Open.length;
node.selectionEnd = node.selectionStart + sel.length;
}
node.scrollTop = ss;
}
return false;
}
$(function () {
$('.DCMS_spoiler_title').on('mouseup', function (event) {
$(this).parent().toggleClass('collapsed');
event.preventDefault();
event.stopPropagation();
}).parent().addClass('collapsed');
});
$(function () {
$(document).on('click', '.DCMS_thumb_down', function (event) {
if (!confirm(translates.rating_down_message)) {
event.preventDefault();
event.stopPropagation();
}
});
});
angular.module('Dcms', ['monospaced.elastic', 'ngAnimate', 'dcmsApi'])
.directive('limitSize', function () {
"use strict";
return {
restrict: 'A',
scope: {
'limitSize': '@',
'limitFiles': '@'
},
link: function (scope, element, attrs) {
if (element[0].tagName !== 'FORM') {
return;
}
scope.inputs = element.find('input[type="file"]');
scope.inputs.on('change', scope.check);
},
controller: function ($scope) {
$scope.check = function () {
var all_size = 0;
var files = 0;
for (var i = 0; i < $scope.inputs.length; i++) {
files += $scope.inputs[i].files.length;
all_size += $scope.getInputSize($scope.inputs[i]);
}
if (all_size <= $scope.limitSize && files <= $scope.limitFiles) {
$scope.inputs.removeClass('error');
return;
}
if (all_size > $scope.limitSize) {
$scope.inputs.addClass('error');
alert('Общий размер выбранных файлов превысил ограничение');
}
if (files > $scope.limitFiles) {
$scope.inputs.addClass('error');
alert('Общее кол-во выбранных файлов превысило ограничение');
}
};
/**
*
* @param {input} input
*/
$scope.getInputSize = function (input) {
var size = 0;
for (var i = 0; i < input.files.length; i++) {
size += input.files[i].size;
}
return size;
};
}
};
})
.directive('ngInitial', function () { // инициализация модели по значению в инпуте
return {
restrict: 'A',
controller: [
'$scope', '$element', '$attrs', '$parse', function ($scope, $element, $attrs, $parse) {
var getter, setter, val;
val = $attrs.ngInitial || $attrs.value || $element.val();
getter = $parse($attrs.ngModel);
setter = getter.assign;
setter($scope, val);
}
]
};
})
.directive('origin', function () {
return {
compile: function ($templateElement, templateAttrs) {
if ($templateElement.prop("nodeName").toLowerCase() != 'img')
return;
if (!templateAttrs.origin)
return;
$templateElement.on('mouseup', function ($event) {
$event.stopPropagation();
$event.preventDefault();
window.open(templateAttrs.origin, '_blank');
});
}
}
})
.directive('bbcode', function () { // директива, добавляющая панель bbcode для textarea
return {
scope: true,
compile: function ($templateElement, templateAttrs) {
if ($templateElement.prop("nodeName").toLowerCase() != 'textarea')
return;
if ($templateElement.prop("bbcoded"))
return;
$templateElement.prop("bbcoded", true);
var $wrapper = $('<div class="textarea_wrapper"><div class="textarea_bbcode"></div></div>');
if ($templateElement.data('ctrl-enter-submit'))
$wrapper.addClass('with_ctrl_enter');
$templateElement.after($wrapper);
$wrapper.append($templateElement);
$wrapper.find('.textarea_bbcode')
.append(
'<span ng-repeat="code in bbcode.codes" ng-click="bbcode.insert(code)" ng-bind="code.Text" title="{{code.Title}}"></span>' +
'<span class="smiles" ng-click="bbcode.showSmiles = !bbcode.showSmiles">{{bbcode.translates.smiles}}' +
'<div ng-show="bbcode.showSmiles" class="smiles_drop_menu">' +
'<div class="smiles_drop_menu_container">{{bbcode.smilesContent}}' +
'<div ng-repeat="smile in bbcode.smiles" ng-click="bbcode.pasteSmile(smile.code)">' +
'<img ng-src="{{smile.image}}" alt="{{smile.title}}" />' +
'</div>' +
'</div>' +
'</div>' +
'</span>');
},
controller: function ($rootScope, $scope, $element, dcmsApi) {
var bbcode = $scope.bbcode = {
smiles: [],
smilesContent: '',
showSmiles: false,
smilesLoaded: false,
codes: codes, // из document.tpl
translates: translates, // из document.tpl
insert: function (code) {
InputInsert($element.find('textarea')[0], code.Prepend, code.Append);
$scope.$broadcast('elastic:adjust'); // обновление высоты textarea
},
pasteSmile: function (smile) {
InputInsert($element.find('textarea')[0], '', ' ' + smile + ' ', true);
$scope.$broadcast('elastic:adjust'); // обновление высоты textarea
}
};
$scope.$watch('bbcode.showSmiles', function () {
if (!bbcode.showSmiles || bbcode.smilesLoaded)
return;
bbcode.smilesContent = 'Загрузка смайлов';
dcmsApi.request('api_smiles', 'get', null)
.then(
function (data) {
bbcode.smiles = data;
bbcode.smilesLoaded = true;
bbcode.smilesContent = '';
},
function () {
bbcode.smilesContent = 'Не удалось загрузить смайлы';
});
});
}
};
})
.config(function ($httpProvider) { // [url]http://habrahabr.ru/post/181009/[/url]
$httpProvider.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded;charset=utf-8';
$httpProvider.defaults.headers.common['X-Requested-With'] = 'XMLHttpRequest';
$httpProvider.defaults.transformRequest = function (data) {
return angular.isObject(data) && String(data) !== '[object File]' ? $.param(data) : data;
};
})
.controller('FormCtrl', // контроллер форм. добавляет поддержку отправки AJAX-ом
['$rootScope', '$scope', '$element', '$http', '$timeout', '$compile',
function ($rootScope, $scope, $element, $http, $timeout, $compile) {
var form = $scope.form = {
msg: '',
err: '',
sending: false, // происходит отправка сообщения
values: {},
onSubmit: function (event) {
var formNode = event.target;
var url = $(formNode).data('url');
if (!url)
return;
if (form.sending)
return;
form.sending = true;
event.preventDefault();
var postData = {};
for (var i = 0; i < formNode.elements.length; i++) {
postData[formNode.elements[i].name] = formNode.elements[i].value;
}
$http.post(url, postData)
.success(function ($data) {
form.sending = false;
if ($data.msg)
form.showMessage($data.msg);
if ($data.err)
form.showError($data.err);
for (var k in $data.form) {
if (formNode.elements[k])
formNode.elements[k].value = $data.form[k];
//form.values[k] = $data.form[k];
}
$scope.$broadcast('elastic:adjust'); // обновление высоты textarea
$rootScope.$broadcast('dcms:form_sended', $element.attr('id')); // Уведомляем о том, что форма была отправлена. Это событие должен слушать листинг
})
.error(function () {
form.sending = false;
form.showError(translates.error);
});
},
showError: function (err) {
form.err = err;
$timeout(function () {
form.err = '';
}, 3000);
},
showMessage: function (msg) {
form.msg = msg;
$timeout(function () {
form.msg = '';
}, 3000);
},
keyDown: function (event) {
if ((event.keyCode == 10 || event.keyCode == 13) && event.ctrlKey) {
if ($(event.target).data('ctrl-enter-submit'))
form.onSubmit($.extend(event, {target: event.target.form}));
}
}
};
}])
.controller('ListingCtrl', // контроллер для списка постов. Добавляет поддержку автоматического обновления списка
['$rootScope', '$scope', '$http', '$interval', '$element', '$animate', '$compile',
function ($rootScope, $scope, $http, $interval, $element, $animate, $compile) {
var listing = $scope.listing = {
counter: 0, // счетчик запросов. необходим при принудительном обновлении данных, чтобы если ответ для более раннего запроса придет последним (маловероятно, но все же), то он не учитывался.
url: '',
id_form: '',
updating: false,
/**
* Обновление листинга
* @param {boolean} forcibly Принудительное обновление
*/
update: function (forcibly) {
if (!listing.url)
return;
if (listing.updating && !forcibly)
return;
listing.updating = true;
var counter = ++listing.counter;
var els = $element.children();
var skip_ids = [];
for (var i = 0; i < els.length; i++) {
skip_ids.push(els[i].id);
}
$http.post(listing.url, {skip_ids: skip_ids.join(',')})
.success(function ($data) {
if (counter != listing.counter)
return;
listing.updating = false;
if ($data.remove && $data.remove.length)
for (var i = 0; i < $data.remove.length; i++) {
$animate.leave($element.find('#' + $data.remove[i]));
}
if ($data.add && $data.add.length) {
for (var i = 0; i < $data.add.length; i++) {
var after_id = $data.add[i].after_id;
var $el = angular.element($data.add[i].html);
$el = $compile($el)($scope);
$animate.enter($el, $element, after_id ? $element.children('#' + after_id) : null);
}
if (!forcibly)
$rootScope.$broadcast('dcms:newMessage');
}
})
.error(function () {
if (counter != listing.counter)
return;
listing.updating = false;
});
}
};
$rootScope.$on('dcms:form_sended', function (event, id_form) {
if (id_form !== listing.id_form)
return;
listing.update(true);
});
$interval(angular.bind(listing, listing.update, false), 7000);
}])
.controller('ListingPostCtrl', function ($scope, $element) {
var post = $scope.post = {
diff: 5,
selection: true,
pageX: null,
pageY: null,
mdown: false,
href: $element.data('postUrl'),
mousedown: function (event) {
post.pageX = event.pageX;
post.pageY = event.pageY;
post.selection = false;
post.mdown = !$(event.target).hasClass('DCMS_phpcode');
if (event.which == 2)
event.preventDefault();
},
mouseup: function (event) {
if (!post.mdown)
return;
post.mdown = false;
if (post.selection) {
$element.removeClass('selection');
return;
}
if ($(event.target).prop("nodeName").toLowerCase() == 'a')
return;
if (event.which == 1)
window.location.href = post.href;
if (event.which == 2)
window.open(post.href, '_blank');
event.stopPropagation();
},
mousemove: function (event) {
if (!post.mdown)
return;
if (post.pageX > event.pageX + post.diff ||
post.pageX < event.pageX - post.diff ||
post.pageY > event.pageY + post.diff ||
post.pageY < event.pageY - post.diff) {
post.selection = true;
$element.addClass('selection');
}
}
};
if (post.href)
$element.on({
mousedown: post.mousedown,
mousemove: post.mousemove,
mouseup: post.mouseup
});
})
.controller('DcmsCtrl', // общий контроллер DCMS
['$scope', '$http', '$timeout', '$rootScope', 'dcmsApi',
function ($scope, $http, $timeout, $rootScope, dcmsApi) {
var scope = {
online: true,
interval: null,
requesting: false,
user: {}, // данные пользователя
translates: translates, // из document.tpl
URL: encodeURI(window.location.pathname + window.location.search),// адрес текущей страницы
str: {
mail: translates.mail, // Почта +[count]
friends: translates.friends // Друзья +[count]
},
onNewMessage: function () {
}
};
scope.setUserData = function (data, sound) {
sound = typeof sound == "undefined" || sound;
if (!data || typeof data.id === "undefined")
return;
if (sound && data.mail_new_count > scope.user.mail_new_count)
$rootScope.$broadcast('dcms:newMessage');
scope.user = angular.extend(scope.user, data);
var cMail = +scope.user.mail_new_count;
var cFriends = +scope.user.friend_new_count;
scope.str.mail = scope.translates.mail + (cMail ? ' +' + cMail : '');
scope.str.friends = scope.translates.friends + (cFriends ? ' +' + cFriends : '');
};
scope.requestUserData = function () {
if (scope.requesting)
return;
scope.requesting = true;
dcmsApi.request('api_user', 'get', Object.keys(scope.user))
.then(
function ($data) {
scope.online = true;
scope.setUserData($data);
scope.requesting = false;
$timeout(scope.requestUserData, +scope.user.id ? 7000 : 60000);
}
, function () {
scope.online = false;
scope.requesting = false;
$timeout(scope.requestUserData, 30000);
});
};
$rootScope.$on('dcms:newMessage', function () {
var audio = document.querySelector("#audio_notify");
audio.pause();
audio.loop = false;
audio.currentTime = 0;
audio.play();
});
scope.setUserData(user, false); // user из document.tpl.php
scope.requestUserData();
angular.extend($scope, scope);
}]);
?>