//The build will inline common dependencies into this file. //For any third party dependencies, like jQuery, place them in the lib folder. //Configure loading modules from the lib directory, //except for 'app' ones, which are in a sibling directory. require.config({ // The shim config allows us to configure dependencies for // scripts that do not call define() to register a module shim: { underscore: { exports: '_' }, backbone: { deps: [ 'underscore', 'jquery' ], exports: 'Backbone' }, 'backbone.paginator': { deps: [ 'backbone' ], exports: 'BackbonePaginator' } }, paths: { 'jquery': 'lib/jquery.min', 'jquery.ui': 'lib/jquery-ui.min', // TODO: it uses deprecated methods in jquery 3 // for fileupload 'jquery.ui.widget': 'lib/jquery.ui.widget.1.11.1', 'tmpl': 'lib/tmpl.min', 'jquery.iframe-transport': 'lib/jquery.iframe-transport', 'jquery.fileupload': 'lib/jquery.fileupload', 'jquery.fileupload-process': 'lib/jquery.fileupload-process', 'jquery.fileupload-validate': 'lib/jquery.fileupload-validate', 'jquery.fileupload-ui': 'lib/jquery.fileupload-ui', 'jquery.magnific-popup': 'lib/jquery.magnific-popup.min', // TODO: it uses deprecated methods in jquery 3 'js.cookie': 'lib/js.cookie', simplemodal: 'lib/jquery.simplemodal', // TODO: it uses deprecated methods in jquery 3 jstree: 'lib/jstree.min', // TODO: it uses deprecated methods in jquery 3 select2: 'lib/select2-3.5.2', // TODO moment: 'lib/moment-with-locales.min', marked: 'lib/marked.min', underscore: 'lib/underscore', backbone: 'lib/backbone', 'backbone.paginator': 'lib/backbone.paginator', text: 'lib/text' } }); define([ 'jquery', 'underscore', 'text', // Workaround for r.js, otherwise text.js will not be included 'pinyin-by-unicode', 'moment', 'js.cookie' ], function($, _, text, PinyinByUnicode, Moment, Cookies) { return { INFO_TIMEOUT: 10000, // 10 secs for info msg SUCCESS_TIMEOUT: 3000, // 3 secs for success msg ERROR_TIMEOUT: 3000, // 3 secs for error msg strChineseFirstPY: PinyinByUnicode.strChineseFirstPY, getUrl: function(options) { var siteRoot = app.config.siteRoot; var fileServerRoot = app.config.fileServerRoot; switch (options.name) { // File Operations case 'list_lib_dir': return siteRoot + 'ajax/lib/' + options.repo_id + '/dir/'; case 'del_dir': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/dir/'; case 'del_file': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/file/'; case 'get_file_info': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/file/'; case 'download_dir_zip_url': return fileServerRoot + 'zip/' + options.zip_token; case 'zip_task': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/zip-task/'; case 'query_zip_progress': return siteRoot + 'api/v2.1/query-zip-progress/'; case 'cancel_zip_task': return siteRoot + 'api/v2.1/cancel-zip-task/'; case 'copy_move_task': return siteRoot + 'api/v2.1/copy-move-task/'; case 'query_copy_move_progress': return siteRoot + 'api/v2.1/query-copy-move-progress/'; case 'rename_dir': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/dir/'; case 'rename_file': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/file/'; case 'lock_or_unlock_file': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/file/'; case 'new_dir': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/dir/'; case 'new_file': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/file/'; case 'del_dirents': return siteRoot + 'ajax/repo/' + options.repo_id + '/dirents/delete/'; case 'mv_dirents': return siteRoot + 'ajax/repo/' + options.repo_id + '/dirents/move/'; case 'cp_dirents': return siteRoot + 'ajax/repo/' + options.repo_id + '/dirents/copy/'; case 'get_dirents': return siteRoot + 'ajax/repo/' + options.repo_id + '/dirents/'; case 'dir-details': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/dir/detail/'; case 'tags': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/tags/'; case 'file-comments': return siteRoot + 'api2/repos/' + options.repo_id + '/file/comments/'; // Repos case 'repos': return siteRoot + 'api2/repos/'; case 'deleted_repos': return siteRoot + 'api/v2.1/deleted-repos/'; case 'pub_repos': return siteRoot + 'api2/repos/public/'; case 'unenc_rw_repos': return siteRoot + 'ajax/unenc-rw-repos/'; case 'api_v2.1_repo_set_password': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/set-password/'; case 'get_history_changes': return siteRoot + 'ajax/repo/' + options.repo_id + '/history/changes/'; case 'beshared_repo': return siteRoot + 'api2/beshared-repos/' + options.repo_id + '/'; case 'dir_shared_items': return siteRoot + 'api2/repos/' + options.repo_id + '/dir/shared_items/'; case 'shared_repos': return siteRoot + 'api2/shared-repos/' + options.repo_id + '/'; case 'repo': return siteRoot + 'api2/repos/' + options.repo_id + '/'; case 'repo_v2.1': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/'; case 'repo_owner': return siteRoot + 'api2/repos/' + options.repo_id + '/owner/'; case 'repo_history_limit': return siteRoot + 'api2/repos/' + options.repo_id + '/history-limit/'; case 'repo_shared_download_links': return siteRoot + 'api2/repos/' + options.repo_id + '/download-shared-links/'; case 'repo_shared_download_link': return siteRoot + 'api2/repos/' + options.repo_id + '/download-shared-links/' + options.token + '/'; case 'repo_shared_upload_links': return siteRoot + 'api2/repos/' + options.repo_id + '/upload-shared-links/'; case 'repo_shared_upload_link': return siteRoot + 'api2/repos/' + options.repo_id + '/upload-shared-links/' + options.token + '/'; case 'repo_user_folder_perm': return siteRoot + 'api2/repos/' + options.repo_id + '/user-folder-perm/'; case 'repo_group_folder_perm': return siteRoot + 'api2/repos/' + options.repo_id + '/group-folder-perm/'; case 'repo_upload_link': return siteRoot + 'api2/repos/' + options.repo_id + '/upload-link/'; case 'repo_update_link': return siteRoot + 'api2/repos/' + options.repo_id + '/update-link/'; case 'repo_file_uploaded_bytes': return siteRoot + 'api/v2.1/repos/' + options.repo_id + '/file-uploaded-bytes/'; case 'repo_labels': return siteRoot + 'api/v2.1/revision-tags/tagged-items/'; case 'user_repo_labels': return siteRoot + 'api/v2.1/revision-tags/tag-names/'; case 'group-owned-library-user-folder-permission': return siteRoot + 'api/v2.1/group-owned-libraries/' + options.repo_id + '/user-folder-permission/'; case 'group-owned-library-group-folder-permission': return siteRoot + 'api/v2.1/group-owned-libraries/' + options.repo_id + '/group-folder-permission/'; case 'group-owned-library-user-share-in-library': return siteRoot + 'api/v2.1/group-owned-libraries/user-share-in-libraries/' + options.repo_id + '/'; // Share admin case 'share_admin_repos': return siteRoot + 'api/v2.1/shared-repos/'; case 'share_admin_repo': return siteRoot + 'api/v2.1/shared-repos/' + options.repo_id + '/'; case 'share_admin_folders': return siteRoot + 'api/v2.1/shared-folders/'; case 'share_admin_share_links': return siteRoot + 'api/v2.1/share-links/'; case 'share_admin_share_link': return siteRoot + 'api/v2.1/share-links/' + options.token + '/'; case 'share_admin_upload_links': return siteRoot + 'api/v2.1/upload-links/'; case 'share_admin_upload_link': return siteRoot + 'api/v2.1/upload-links/' + options.token + '/'; // Links case 'send_shared_download_link': return siteRoot + 'share/link/send/'; case 'send_shared_upload_link': return siteRoot + 'share/upload_link/send/'; case 'smart_link': return siteRoot + 'api/v2.1/smart-link/'; // Group case 'groups': return siteRoot + 'api/v2.1/groups/'; case 'all_groups': return siteRoot + 'api/v2.1/all-groups/'; case 'shareable_groups': return siteRoot + 'api/v2.1/shareable-groups/'; case 'search_group': return siteRoot + 'api/v2.1/search-group/'; case 'group': return siteRoot + 'api/v2.1/groups/' + options.group_id + '/'; case 'group_members': return siteRoot + 'api/v2.1/groups/' + options.group_id + '/members/'; case 'group_member': return siteRoot + 'api/v2.1/groups/' + options.group_id + '/members/' + options.email + '/'; case 'group_member_bulk': return siteRoot + 'api/v2.1/groups/' + options.group_id + '/members/bulk/'; case 'group_import_members': return siteRoot + 'ajax/group/' + options.group_id + '/members/import/'; case 'group_repos': return siteRoot + 'api2/groups/' + options.group_id + '/repos/'; case 'group_owned_repos': return siteRoot + 'api/v2.1/groups/' + options.group_id + '/group-owned-libraries/'; case 'group_owned_repo': return siteRoot + 'api/v2.1/groups/' + options.group_id + '/group-owned-libraries/' + options.repo_id + '/'; case 'group_owned_repo_user_share': return siteRoot + 'api/v2.1/group-owned-libraries/' + options.repo_id + '/user-share/'; case 'group_owned_repo_group_share': return siteRoot + 'api/v2.1/group-owned-libraries/' + options.repo_id + '/group-share/'; case 'address_book_sub_groups': return siteRoot + 'api/v2.1/address-book/groups/' + options.group_id + '/sub-groups/'; case 'address_book_group_search_members': return siteRoot + 'api/v2.1/address-book/groups/' + options.group_id + '/search-member/'; case 'group_discussions': return siteRoot + 'api2/groups/' + options.group_id + '/discussions/'; case 'group_discussion': return siteRoot + 'api2/groups/' + options.group_id + '/discussions/' + options.discussion_id + '/'; // thumbnail case 'thumbnail_create': return siteRoot + 'thumbnail/' + options.repo_id + '/create/'; case 'thumbnail_get': return siteRoot + 'thumbnail/' + options.repo_id + '/' + options.size + options.path; // Misc case 'view_raw_file': return siteRoot + 'repo/' + options.repo_id + '/raw' + options.path; case 'get_popup_notices': return siteRoot + 'ajax/get_popup_notices/'; case 'notifications': return siteRoot + 'api/v2.1/notifications/'; case 'notification': return siteRoot + 'api/v2.1/notification/'; case 'user_enabled_modules': return siteRoot + 'api/v2.1/user-enabled-modules/'; case 'starred_files': return siteRoot + 'api2/starredfiles/'; case 'events': return siteRoot + 'api2/events/'; case 'devices': return siteRoot + 'api2/devices/'; case 'invitations': return siteRoot + 'api/v2.1/invitations/'; case 'invitations_batch': return siteRoot + 'api/v2.1/invitations/batch/'; case 'invitation': return siteRoot + 'api/v2.1/invitations/' + options.token + '/'; case 'search_user': return siteRoot + 'api2/search-user/'; case 'user_profile': return siteRoot + 'profile/' + options.username + '/'; case 'space_and_traffic': return siteRoot + 'ajax/space_and_traffic/'; // sysadmin case 'sysinfo': return siteRoot + 'api/v2.1/admin/sysinfo/'; case 'admin-devices': return siteRoot + 'api/v2.1/admin/devices/'; case 'admin-device-errors': return siteRoot + 'api/v2.1/admin/device-errors/'; case 'admin-device-trusted-ip': return siteRoot + 'api/v2.1/admin/device-trusted-ip/'; case 'admin-libraries': return siteRoot + 'api/v2.1/admin/libraries/'; case 'admin-library': return siteRoot + 'api/v2.1/admin/libraries/' + options.repo_id + '/'; case 'admin-library-history-limit': return siteRoot + 'api/v2.1/admin/libraries/' + options.repo_id + '/history-limit/'; case 'admin-library-dirents': return siteRoot + 'api/v2.1/admin/libraries/' + options.repo_id + '/dirents/'; case 'admin-groups': return siteRoot + 'api/v2.1/admin/groups/'; case 'admin-group': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/'; case 'admin-group-libraries': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/libraries/'; case 'admin-group-library': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/libraries/' + options.repo_id + '/'; case 'admin-group-owned-libraries': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/group-owned-libraries/'; case 'admin-group-owned-library': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/group-owned-libraries/' + options.repo_id + '/'; case 'admin-group-members': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/members/'; case 'admin-group-member': return siteRoot + 'api/v2.1/admin/groups/' + options.group_id + '/members/' + encodeURIComponent(options.email) + '/'; case 'admin-system-library': return siteRoot + 'api/v2.1/admin/system-library/'; case 'admin-system-library-upload-link': return siteRoot + 'api/v2.1/admin/system-library/upload-link/'; case 'admin-trash-libraries': return siteRoot + 'api/v2.1/admin/trash-libraries/'; case 'admin-trash-library': return siteRoot + 'api/v2.1/admin/trash-libraries/' + options.repo_id + '/'; case 'admin_shares': return siteRoot + 'api/v2.1/admin/shares/'; case 'sys_group_admin_export_excel': return siteRoot + 'sys/groupadmin/export-excel/'; case 'admin-operation-logs': return siteRoot + 'api/v2.1/admin/admin-logs/'; case 'admin-login-logs': return siteRoot + 'api/v2.1/admin/admin-login-logs/'; case 'admin-address-book-groups': return siteRoot + 'api/v2.1/admin/address-book/groups/'; case 'admin-address-book-group': return siteRoot + 'api/v2.1/admin/address-book/groups/' + options.group_id + '/'; // org admin case 'org-admin-address-book-groups': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/address-book/groups/'; case 'org-admin-address-book-group': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/address-book/groups/' + options.group_id + '/'; case 'org-admin-group': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/groups/' + options.group_id + '/'; case 'org-admin-group-members': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/groups/' + options.group_id + '/members/'; case 'org-admin-group-member': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/groups/' + options.group_id + '/members/' + encodeURIComponent(options.email) + '/'; case 'org-admin-group-libraries': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/groups/' + options.group_id + '/libraries/'; case 'org-admin-group-owned-libraries': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/groups/' + options.group_id + '/group-owned-libraries/'; case 'org-admin-group-owned-library': return siteRoot + 'api/v2.1/org/' + options.org_id + '/admin/groups/' + options.group_id + '/group-owned-libraries/' + options.repo_id + '/'; case 'license': return siteRoot + 'api/v2.1/admin/license/'; } }, FILEEXT_ICON_MAP: { // text file 'md': 'txt.png', 'txt': 'txt.png', // pdf file 'pdf' : 'pdf.png', // document file 'doc' : 'word.png', 'docx' : 'word.png', 'odt' : 'word.png', 'fodt' : 'word.png', 'ppt' : 'ppt.png', 'pptx' : 'ppt.png', 'odp' : 'ppt.png', 'fodp' : 'ppt.png', 'xls' : 'excel.png', 'xlsx' : 'excel.png', 'ods' : 'excel.png', 'fods' : 'excel.png', // video 'mp4': 'video.png', 'ogv': 'video.png', 'webm': 'video.png', 'mov': 'video.png', 'flv': 'video.png', 'wmv': 'video.png', 'rmvb': 'video.png', // music file 'mp3' : 'music.png', 'oga' : 'music.png', 'ogg' : 'music.png', 'flac' : 'music.png', 'aac' : 'music.png', 'ac3' : 'music.png', 'wma' : 'music.png', // image file 'jpg' : 'pic.png', 'jpeg' : 'pic.png', 'png' : 'pic.png', 'svg' : 'pic.png', 'gif' : 'pic.png', 'bmp' : 'pic.png', 'ico' : 'pic.png', // default 'default' : 'file.png' }, getFileIconUrl: function(filename, size) { if (size > 24) { size = 192; } else { size = 24; } var file_ext; if (filename.lastIndexOf('.') == -1) { return app.config.mediaUrl + "img/file/" + size + "/" + this.FILEEXT_ICON_MAP['default']; } else { file_ext = filename.substr(filename.lastIndexOf('.') + 1).toLowerCase(); } if (_.has(this.FILEEXT_ICON_MAP, file_ext)) { return app.config.mediaUrl + "img/file/" + size + "/" + this.FILEEXT_ICON_MAP[file_ext]; } else { return app.config.mediaUrl + "img/file/" + size + "/" + this.FILEEXT_ICON_MAP['default']; } }, getDirIconUrl: function(is_readonly, size) { if (size > 24) { if (is_readonly) { return app.config.mediaUrl + "img/folder-read-only-192.png"; } else { return app.config.mediaUrl + "img/folder-192.png"; } } else { if (is_readonly) { return app.config.mediaUrl + "img/folder-read-only-24.png"; } else { return app.config.mediaUrl + "img/folder-24.png"; } } }, getLibIconUrl: function(is_encrypted, is_readonly, size) { // icon name var icon_name = 'lib.png'; if (is_encrypted) { icon_name = 'lib-encrypted.png'; } if (is_readonly) { icon_name = 'lib-readonly.png'; } // icon size var icon_size = size || 256; // 'size' can be 24, 48, or undefined. (2017.7.31) return app.config.mediaUrl + 'img/lib/' + icon_size + '/' + icon_name; }, getLibIconTitle: function(is_encrypted, is_readonly) { if (is_encrypted) { return gettext("Encrypted library"); } else if (is_readonly) { return gettext("Read-Only library"); } else { return gettext("Read-Write library"); } }, isHiDPI: function() { var pixelRatio = window.devicePixelRatio ? window.devicePixelRatio : 1; if (pixelRatio > 1) { return true; } else { return false; } }, showConfirm: function(title, content, yesCallback) { var $popup = $("#confirm-popup"); var $cont = $('#confirm-con'); var $yesBtn = $('#confirm-yes'); $cont.html('

' + title + '

' + content + '

'); $popup.modal(); $('#simplemodal-container').css({'height':'auto'}); $yesBtn.on('click', yesCallback); }, showConfirmWithExtraOption: function(title, content, extraOption, yesCallback) { var $popup = $("#confirm-popup"); var $cont = $('#confirm-con'); var $yesBtn = $('#confirm-yes'); var confirm_with_extra_option_template = _.template($('#confirm-dialog-with-extra-option-tmpl').html()); var html = confirm_with_extra_option_template({ 'is_pro': app.pageOptions.is_pro, 'title': title, 'content': content, 'extraOption': extraOption }); $cont.html(html); $popup.modal(); $('#simplemodal-container').css({'height':'auto'}); $yesBtn.on('click', function() { var extraOptionChecked = $('#confirm-extra-option:checked').val() === 'on'; yesCallback(extraOptionChecked); }); }, closeModal: function() { $.modal.close(); }, feedback: function(con, type, time) { var _this = this; var time = time || 5000; var $el; var hide_pos_top, show_pos_top = '15px'; var $con, str = ''; if (typeof con == 'string') { // most of the time $con = $('
  • ' + this.HTMLescape(con) + '
  • '); } else { // [{'msg':'', 'type':''}] $(con).each(function(index, item) { str += '
  • ' + _this.HTMLescape(item.msg) + '
  • '; }); $con = $(str); } if ($('.messages').length > 0) { $el = $('.messages').html($con); } else { $el = $('').html($con); $('#main').append($el); } hide_pos_top = '-' + ($el.outerHeight() + parseInt(show_pos_top)) + 'px'; // add transition: from 'hide' to 'show'. the transition effect is offered by CSS. $el.css({'left':($(window).width() - $el.width())/2, 'top': hide_pos_top}); setTimeout(function() { $el.css({'top': show_pos_top}); }, 10); setTimeout(function() { $el.css({'top': hide_pos_top}); }, time); }, showFormError: function(formid, error_msg) { $("#" + formid + " .error").html(error_msg).removeClass('hide'); $("#simplemodal-container").css({'height':'auto'}); }, prepareAjaxErrorMsg: function(xhr) { var error_msg; if (xhr.responseText) { var parsed_resp = JSON.parse(xhr.responseText); // use `HTMLescape` for msgs which contain variable like 'path' error_msg = this.HTMLescape(parsed_resp.error || parsed_resp.error_msg || parsed_resp.detail); } else { error_msg = gettext("Failed. Please check the network."); } return error_msg; }, ajaxErrorHandler: function(xhr, textStatus, errorThrown) { var error_msg = this.prepareAjaxErrorMsg(xhr); this.feedback(error_msg, 'error'); }, prepareCollectionFetchErrorMsg: function(collection, response, opts) { var err_msg; if (response.responseText) { if (response['status'] == 401) { err_msg = gettext("Permission error"); } else if (response['status'] == 403) { err_msg = gettext("Permission error"); location.href = app.config.loginUrl + '?next=' + encodeURIComponent(location.pathname + location.hash); } else { err_msg = this.HTMLescape(JSON.parse(response.responseText).error_msg); } } else { err_msg = gettext('Please check the network.'); } return err_msg; }, enableButton: function(btn) { btn.prop('disabled', false).removeClass('btn-disabled'); }, disableButton: function(btn) { btn.prop('disabled', true).addClass('btn-disabled'); }, setCaretPos: function(inputor, pos) { var range; if (document.selection) { range = inputor.createTextRange(); range.move("character", pos); return range.select(); } else { return inputor.setSelectionRange(pos, pos); } }, prepareApiCsrf: function() { /* alias away the sync method */ Backbone._sync = Backbone.sync; /* define a new sync method */ Backbone.sync = function(method, model, options) { /* only need a token for non-get requests */ if (method == 'create' || method == 'update' || method == 'delete') { // CSRF token value is in an embedded meta tag // var csrfToken = $("meta[name='csrf_token']").attr('content'); var csrfToken = app.pageOptions.csrfToken; options.beforeSend = function(xhr){ xhr.setRequestHeader('X-CSRFToken', csrfToken); }; } /* proxy the call to the old sync method */ return Backbone._sync(method, model, options); }; }, prepareCSRFToken: function(xhr, settings) { function getCookie(name) { var cookieValue = null; if (document.cookie && document.cookie != '') { var cookies = document.cookie.split(';'); for (var i = 0; i < cookies.length; i++) { var cookie = jQuery.trim(cookies[i]); // Does this cookie string begin with the name we want? if (cookie.substring(0, name.length + 1) == (name + '=')) { cookieValue = decodeURIComponent(cookie.substring(name.length + 1)); break; } } } return cookieValue; } if (!(/^http:.*/.test(settings.url) || /^https:.*/.test(settings.url))) { // Only send the token to relative URLs i.e. locally. xhr.setRequestHeader("X-CSRFToken", getCookie(app.config.csrfCookieName)); } }, ajaxPost: function(params) { // usually used for form ajax post in modal popup var _this = this, form = params.form, form_id = params.form_id, post_url = params.post_url, post_data = params.post_data, after_op_success = params.after_op_success, after_op_error; var submit_btn = form.children('[type="submit"]'); this.disableButton(submit_btn); if (params.after_op_error) { after_op_error = params.after_op_error; } else { after_op_error = function(xhr, textStatus, errorThrown) { var error_msg = _this.prepareAjaxErrorMsg(xhr); _this.showFormError(form_id, error_msg); _this.enableButton(submit_btn); }; } $.ajax({ url: post_url, type: 'POST', dataType: 'json', beforeSend: this.prepareCSRFToken, data: post_data, success: after_op_success, error: after_op_error }); }, ajaxGet: function(params) { var _this = this, get_url = params.get_url, data = params.data, after_op_success = params.after_op_success, after_op_error; if (params.after_op_error) { after_op_error = params.after_op_error; } else { after_op_error = function(xhr, textStatus, errorThrown) { }; } $.ajax({ url: get_url, cache: false, dataType: 'json', data: data, success: after_op_success, error: after_op_error }); }, HTMLescape: function(html) { return document.createElement('div') .appendChild(document.createTextNode(html)) .parentNode .innerHTML; }, pathJoin: function(array) { var result = array[0]; for (var i = 1; i < array.length; i++) { if (result[result.length-1] == '/' || array[i][0] == '/') result += array[i]; else result += '/' + array[i]; } return result; }, encodePath: function(path) { // IE8 does not support 'map()' /* return path.split('/').map(function(e) { return encodeURIComponent(e); }).join('/'); */ var path_arr = path.split('/'), path_arr_ = []; for (var i = 0, len = path_arr.length; i < len; i++) { path_arr_.push(encodeURIComponent(path_arr[i])); } return path_arr_.join('/'); }, initLocale: function() { var language_code; if (app.pageOptions.language_code == 'en') { language_code = 'en-gb'; } else if (app.pageOptions.language_code == 'es-ar' || app.pageOptions.language_code == 'es-mx') { language_code = 'es'; } else { language_code = app.pageOptions.language_code; } Moment.locale(language_code); }, getRelativeTimeStr: function(m) { var now = new Date(); if (m - now > 0) { return gettext("Just now"); } else { return m.fromNow(); } }, closeTopNoticeBar: function () { if (!app.pageOptions.cur_note) { return false; } var new_info_id = app.pageOptions.cur_note.id; $('#info-bar').addClass('hide'); if (navigator.cookieEnabled) { var date = new Date(), cookies = document.cookie.split('; '), info_id_exist = false; date.setTime(date.getTime() + 14*24*60*60*1000); new_info_id += '; expires=' + date.toGMTString() + '; path=' + app.config.siteRoot; for (var i = 0, len = cookies.length; i < len; i++) { if (cookies[i].split('=')[0] == 'info_id') { info_id_exist = true; document.cookie = 'info_id=' + cookies[i].split('=')[1] + new_info_id; break; } } if (!info_id_exist) { document.cookie = 'info_id=' + new_info_id; } } }, contactInputOptionsForSelect2: function(options) { var _this = this; var url; if (options && options.url) { url = options.url; } else { url = _this.getUrl({name: 'search_user'}); } return { placeholder: gettext("Search users or enter emails and press Enter"), // with 'tags', the user can directly enter, not just select // tags need ``, not ``, not `