define(['taskbar'], function(taskbar) { var composer = { active: undefined, posts: {} }; function allowed() { if(!(parseInt(app.uid, 10) > 0 || config.allowGuestPosting)) { app.alert({ type: 'danger', timeout: 5000, alert_id: 'post_error', title: 'Please Log In to Post', message: 'Posting is currently restricted to registered members only, click here to log in', clickfn: function() { ajaxify.go('login'); } }); return false; } return true; } composer.newTopic = function(cid) { if(allowed()) { push({ cid: cid, title: '', body: '', modified: false }); } } composer.addQuote = function(tid, pid, title, username, text){ if (allowed()) { var uuid = composer.active; if(uuid !== undefined){ var bodyEl = $('#cmp-uuid-'+uuid).find('textarea'); var prevText = bodyEl.val(); if(tid !== composer.posts[uuid].tid) { text = username + ' said in ['+title+'](/topic/'+tid+'#'+pid+'):\n'+text; }else { text = username + ' said:\n' + text; } composer.posts[uuid].body = (prevText.length ? prevText + '\n\n' : '') + text; bodyEl.val(composer.posts[uuid].body); }else{ composer.newReply(tid,title,username + ' said:\n' + text); } } }; composer.newReply = function(tid, title, text) { if(allowed()) { push({ tid: tid, title: title, body: text, modified: false }); } } composer.editPost = function(pid) { if(allowed()) { socket.emit('modules.composer.push', pid, function(err, threadData) { if(err) { return app.alertError(err.message); } push({ pid: pid, title: threadData.title, body: threadData.body, modified: false }); }); } } function push(post) { var uuid = utils.generateUUID(); translator.translate('[[topic:composer.new_topic]]', function(newTopicStr) { taskbar.push('composer', uuid, { title: post.title ? post.title : newTopicStr, icon: post.picture }); }); composer.posts[uuid] = post; composer.posts[uuid].uploadsInProgress = []; composer.load(uuid); } composer.load = function(post_uuid) { if($('#cmp-uuid-' + post_uuid).length) { composer.activateReposition(post_uuid); } else { composer.createNewComposer(post_uuid); } } composer.createNewComposer = function(post_uuid) { templates.preload_template('composer', function() { var composerTemplate = templates['composer'].parse({}); translator.translate(composerTemplate, function(composerTemplate) { composerTemplate = $(composerTemplate); composerTemplate.attr('id', 'cmp-uuid-' + post_uuid); $(document.body).append(composerTemplate); composer.activateReposition(post_uuid); var postContainer = $(composerTemplate[0]); if(config.allowFileUploads || config.hasImageUploadPlugin) { initializeDragAndDrop(post_uuid); } var postData = composer.posts[post_uuid], titleEl = postContainer.find('.title'), bodyEl = postContainer.find('textarea'); if (parseInt(postData.tid) > 0) { translator.translate('[[topic:composer.replying_to]]: ' + postData.title, function(newTitle) { titleEl.val(newTitle); }); titleEl.prop('disabled', true); } else if (parseInt(postData.pid) > 0) { titleEl.val(postData.title); titleEl.prop('disabled', true); socket.emit('modules.composer.editCheck', postData.pid, function(err, editCheck) { if (!err && editCheck.titleEditable) { titleEl.prop('disabled', false); } }); } else { titleEl.val(postData.title); titleEl.prop('disabled', false); } bodyEl.val(postData.body); postContainer.on('change', 'input, textarea', function() { composer.posts[post_uuid].modified = true; }); postContainer.on('click', '.action-bar button', function() { var action = $(this).attr('data-action'); switch(action) { case 'post': $(this).attr('disabled', true); composer.post(post_uuid); break; case 'discard': if (composer.posts[post_uuid].modified) { bootbox.confirm('Are you sure you wish to discard this post?', function(discard) { if (discard) { composer.discard(post_uuid); } }); } else { composer.discard(post_uuid); } break; } }); postContainer.on('click', '.formatting-bar span', function() { var postContentEl = postContainer.find('textarea'), iconClass = $(this).find('i').attr('class'), cursorEnd = postContentEl.val().length, selectionStart = postContentEl[0].selectionStart, selectionEnd = postContentEl[0].selectionEnd, selectionLength = selectionEnd - selectionStart; function insertIntoInput(element, value) { var start = postContentEl[0].selectionStart; element.val(element.val().slice(0, start) + value + element.val().slice(start, element.val().length)); postContentEl[0].selectionStart = postContentEl[0].selectionEnd = start + value.length; } switch(iconClass) { case 'fa fa-bold': if (selectionStart === selectionEnd) { // Nothing selected insertIntoInput(postContentEl, "**bolded text**"); } else { // Text selected postContentEl.val(postContentEl.val().slice(0, selectionStart) + '**' + postContentEl.val().slice(selectionStart, selectionEnd) + '**' + postContentEl.val().slice(selectionEnd)); postContentEl[0].selectionStart = selectionStart + 2; postContentEl[0].selectionEnd = selectionEnd + 2; } break; case 'fa fa-italic': if (selectionStart === selectionEnd) { // Nothing selected insertIntoInput(postContentEl, "*italicised text*"); } else { // Text selected postContentEl.val(postContentEl.val().slice(0, selectionStart) + '*' + postContentEl.val().slice(selectionStart, selectionEnd) + '*' + postContentEl.val().slice(selectionEnd)); postContentEl[0].selectionStart = selectionStart + 1; postContentEl[0].selectionEnd = selectionEnd + 1; } break; case 'fa fa-list': // Nothing selected insertIntoInput(postContentEl, "\n* list item"); break; case 'fa fa-link': if (selectionStart === selectionEnd) { // Nothing selected insertIntoInput(postContentEl, "[link text](link url)"); } else { // Text selected postContentEl.val(postContentEl.val().slice(0, selectionStart) + '[' + postContentEl.val().slice(selectionStart, selectionEnd) + '](link url)' + postContentEl.val().slice(selectionEnd)); postContentEl[0].selectionStart = selectionStart + selectionLength + 3; postContentEl[0].selectionEnd = selectionEnd + 11; } break; } }); postContainer.on('click', '.formatting-bar span .fa-picture-o, .formatting-bar span .fa-upload', function() { $('#files').click(); }); $('#files').on('change', function(e) { var files = e.target.files; if(files) { uploadSubmit(files, post_uuid, '/api/post/upload'); } }); postContainer.find('.nav-tabs a').click(function (e) { e.preventDefault(); $(this).tab('show'); var selector = $(this).attr('data-pane'); postContainer.find('.tab-content div').removeClass('active'); postContainer.find(selector).addClass('active'); if(selector === '.tab-write') { bodyEl.focus(); } return false; }); bodyEl.on('blur', function() { socket.emit('modules.composer.renderPreview', bodyEl.val(), function(err, preview) { preview = $(preview); preview.find('img').addClass('img-responsive'); postContainer.find('.preview').html(preview); }); }); var resizeActive = false, resizeCenterY = 0, resizeOffset = 0, resizeStart = function(e) { resizeRect = resizeEl.getBoundingClientRect(); resizeCenterY = resizeRect.top + (resizeRect.height/2); resizeOffset = resizeCenterY - e.clientY; resizeActive = true; $(window).on('mousemove', resizeAction); $(window).on('mouseup', resizeStop); document.body.addEventListener('touchmove', resizeTouchAction); }, resizeStop = function() { resizeActive = false; bodyEl.focus(); $(window).off('mousemove', resizeAction); $(window).off('mouseup', resizeStop); document.body.removeEventListener('touchmove', resizeTouchAction); }, resizeTouchAction = function(e) { e.preventDefault(); resizeAction(e.touches[0]); }, resizeAction = function(e) { if (resizeActive) { position = (e.clientY + resizeOffset); var newHeight = $(window).height() - position; var paddingBottom = parseInt(postContainer.css('padding-bottom'), 10); if(newHeight > $(window).height() - $('#header-menu').height() - 20) { newHeight = $(window).height() - $('#header-menu').height() - 20; } else if (newHeight < paddingBottom) { newHeight = paddingBottom; } postContainer.css('height', newHeight); $('body').css({'margin-bottom': newHeight}); resizeSavePosition(newHeight); } e.preventDefault(); return false; }, resizeSavePosition = function(px) { var percentage = px / $(window).height(); localStorage.setItem('composer:resizePercentage', percentage); }, resizeRect; var resizeEl = postContainer.find('.resizer')[0]; resizeEl.addEventListener('mousedown', resizeStart); resizeEl.addEventListener('touchstart', function(e) { e.preventDefault(); resizeStart(e.touches[0]); }); resizeEl.addEventListener('touchend', function(e) { e.preventDefault(); resizeStop(); }); // .on('mousedown touchstart', resizeStart) // .on('mouseup touchend', resizeStop) window.addEventListener('resize', function() { if (composer.active !== undefined) { composer.activateReposition(composer.active); } }); $(window).trigger('action:composer.loaded', { post_uuid: post_uuid }); }); }); } //http://stackoverflow.com/questions/14441456/how-to-detect-which-device-view-youre-on-using-twitter-bootstrap-api function findBootstrapEnvironment() { var envs = ['xs', 'sm', 'md', 'lg']; $el = $('