define(function() { var Topic = {}; Topic.init = function() { var expose_tools = templates.get('expose_tools'), tid = templates.get('topic_id'), postListEl = document.getElementById('post-container'), editBtns = document.querySelectorAll('#post-container .post-buttons .edit, #post-container .post-buttons .edit i'), thread_state = { locked: templates.get('locked'), deleted: templates.get('deleted'), pinned: templates.get('pinned') }, topic_name = templates.get('topic_name'), twitter_url = templates.get('twitter-intent-url'), facebook_url = templates.get('facebook-share-url'), google_url = templates.get('google-share-url'); jQuery('document').ready(function() { app.addCommasToNumbers(); var room = 'topic_' + tid, adminTools = document.getElementById('thread-tools'); app.enter_room(room); $('.twitter-share').on('click', function () { window.open(twitter_url, '_blank', 'width=550,height=420,scrollbars=no,status=no'); return false; }); $('.facebook-share').on('click', function () { window.open(facebook_url, '_blank', 'width=626,height=436,scrollbars=no,status=no'); return false; }); $('.google-share').on('click', function () { window.open(google_url, '_blank', 'width=500,height=570,scrollbars=no,status=no'); return false; }); // Resetting thread state if (thread_state.locked === '1') set_locked_state(true); if (thread_state.deleted === '1') set_delete_state(true); if (thread_state.pinned === '1') set_pinned_state(true); if (expose_tools === '1') { var moveThreadModal = $('#move_thread_modal'); adminTools.style.visibility = 'inherit'; // Add events to the thread tools $('#delete_thread').on('click', function(e) { if (thread_state.deleted !== '1') { bootbox.confirm('Are you sure you want to delete this thread?', function(confirm) { if (confirm) { socket.emit('api:topic.delete', { tid: tid }); } }); } else { bootbox.confirm('Are you sure you want to restore this thread?', function(confirm) { if (confirm) socket.emit('api:topic.restore', { tid: tid }); }); } return false; }); $('#lock_thread').on('click', function(e) { if (thread_state.locked !== '1') { socket.emit('api:topic.lock', { tid: tid }); } else { socket.emit('api:topic.unlock', { tid: tid }); } return false; }); $('#pin_thread').on('click', function(e) { if (thread_state.pinned !== '1') { socket.emit('api:topic.pin', { tid: tid }); } else { socket.emit('api:topic.unpin', { tid: tid }); } return false; }); $('#move_thread').on('click', function(e) { moveThreadModal.modal('show'); return false; }); moveThreadModal.on('shown.bs.modal', function() { var loadingEl = document.getElementById('categories-loading'); if (loadingEl) { socket.once('api:categories.get', function(data) { console.log(data); // Render categories var categoriesFrag = document.createDocumentFragment(), categoryEl = document.createElement('li'), numCategories = data.categories.length, modalBody = moveThreadModal.find('.modal-body'), categoriesEl = modalBody[0].getElementsByTagName('ul')[0], confirmDiv = document.getElementById('move-confirm'), confirmCat = confirmDiv.getElementsByTagName('span')[0], commitEl = document.getElementById('move_thread_commit'), cancelEl = document.getElementById('move_thread_cancel'), x, info, targetCid, targetCatLabel; categoriesEl.className = 'category-list'; for (x = 0; x < numCategories; x++) { info = data.categories[x]; categoryEl.className = info.blockclass + (info.disabled === '1' ? ' disabled' : ''); categoryEl.innerHTML = ' ' + info.name; categoryEl.setAttribute('data-cid', info.cid); categoriesFrag.appendChild(categoryEl.cloneNode(true)); } categoriesEl.appendChild(categoriesFrag); modalBody[0].removeChild(loadingEl); categoriesEl.addEventListener('click', function(e) { if (e.target.nodeName === 'LI') { confirmCat.innerHTML = e.target.innerHTML; confirmDiv.style.display = 'block'; targetCid = e.target.getAttribute('data-cid'); targetCatLabel = e.target.innerHTML; commitEl.disabled = false; } }, false); commitEl.addEventListener('click', function() { if (!commitEl.disabled && targetCid) { commitEl.disabled = true; $(cancelEl).fadeOut(250); $(moveThreadModal).find('.modal-header button').fadeOut(250); commitEl.innerHTML = 'Moving '; socket.once('api:topic.move', function(data) { moveThreadModal.modal('hide'); if (data.status === 'ok') { app.alert({ 'alert_id': 'thread_move', type: 'success', title: 'Topic Successfully Moved', message: 'This topic has been successfully moved to ' + targetCatLabel, timeout: 5000 }); } else { app.alert({ 'alert_id': 'thread_move', type: 'danger', title: 'Unable to Move Topic', message: 'This topic could not be moved to ' + targetCatLabel + '.
Please try again later', timeout: 5000 }); } }); socket.emit('api:topic.move', { tid: tid, cid: targetCid }); } }); }); socket.emit('api:categories.get'); } }); } // Fix delete state for this thread's posts var postEls = document.querySelectorAll('#post-container li[data-deleted]'); for (var x = 0, numPosts = postEls.length; x < numPosts; x++) { if (postEls[x].getAttribute('data-deleted') === '1') toggle_post_delete_state(postEls[x].getAttribute('data-pid')); postEls[x].removeAttribute('data-deleted'); } // Follow Thread State var followEl = $('.main-post .follow'), set_follow_state = function(state, quiet) { if (state && !followEl.hasClass('btn-success')) { followEl.addClass('btn-success'); followEl[0].title = 'You are currently receiving updates to this topic'; if (!quiet) { app.alert({ alert_id: 'topic_follow', timeout: 2500, title: 'Following Topic', message: 'You will now be receiving notifications when somebody posts to this topic.', type: 'success' }); } } else if (!state && followEl.hasClass('btn-success')) { followEl.removeClass('btn-success'); followEl[0].title = 'Be notified of new replies in this topic'; if (!quiet) { app.alert({ alert_id: 'topic_follow', timeout: 2500, title: 'Not Following Topic', message: 'You will no longer receive notifications from this topic.', type: 'success' }); } } }; socket.on('api:topic.followCheck', function(state) { set_follow_state(state, true); }); socket.on('api:topic.follow', function(data) { if (data.status && data.status === 'ok') set_follow_state(data.follow); else { app.alert({ type: 'danger', alert_id: 'topic_follow', title: 'Please Log In', message: 'Please register or log in in order to subscribe to this topic', timeout: 5000 }); } }); socket.emit('api:topic.followCheck', tid); if (followEl[0]) { followEl[0].addEventListener('click', function() { socket.emit('api:topic.follow', tid); }, false); } enableInfiniteLoading(); var bookmark = localStorage.getItem('topic:' + tid + ':bookmark'); if(bookmark) { app.scrollToPost(parseInt(bookmark, 10)); } $('#post-container').on('click', '.deleted', function(ev) { $(this).toggleClass('deleted-expanded'); }); }); function enableInfiniteLoading() { $(window).off('scroll').on('scroll', function() { var bottom = ($(document).height() - $(window).height()) * 0.9; if ($(window).scrollTop() > bottom && !app.infiniteLoaderActive && $('#post-container').children().length) { app.loadMorePosts(tid); } }); } var reply_fn = function() { var selectionText = '', selection = window.getSelection() || document.getSelection(); if ($(selection.baseNode).parents('.post-content').length > 0) { var snippet = selection.toString(); if (snippet.length > 0) selectionText = '> ' + snippet.replace(/\n/g, '\n> '); } if (thread_state.locked !== '1') { require(['composer'], function(cmp) { cmp.push(tid, null, null, selectionText.length > 0 ? selectionText + '\n\n' : ''); }); } }; $('#post-container').on('click', '.post_reply', reply_fn); $('#post_reply').on('click', reply_fn); $('#post-container').on('click', '.quote', function() { if (thread_state.locked !== '1') { var pid = $(this).parents('li').attr('data-pid'); socket.once('api:posts.getRawPost', function(data) { quoted = '> ' + data.post.replace(/\n/g, '\n> ') + '\n\n'; require(['composer'], function(cmp) { cmp.push(tid, null, null, quoted); }); }); socket.emit('api:posts.getRawPost', { pid: pid }); } }); $('#post-container').on('click', '.favourite', function() { var pid = $(this).parents('li').attr('data-pid'); var uid = $(this).parents('li').attr('data-uid'); var element = $(this).find('i'); if (element.attr('class') == 'icon-star-empty') { socket.emit('api:posts.favourite', { pid: pid, room_id: app.current_room }); } else { socket.emit('api:posts.unfavourite', { pid: pid, room_id: app.current_room }); } }); $('#post-container').on('click', '.link', function() { var pid = $(this).parents('li').attr('data-pid'); $('#post_' + pid + '_link').val(window.location.href + "#" + pid).stop(true, false).fadeIn().select(); $('#post_' + pid + '_link').off('blur').on('blur', function() { $(this).fadeOut(); }); }); $('#post-container').delegate('.edit', 'click', function(e) { var pid = $(this).parents('li').attr('data-pid'), main = $(this).parents('.main-post'); require(['composer'], function(cmp) { cmp.push(null, null, pid); }); }); $('#post-container').delegate('.delete', 'click', function(e) { var pid = $(this).parents('li').attr('data-pid'), postEl = $(document.querySelector('#post-container li[data-pid="' + pid + '"]')), deleteAction = !postEl.hasClass('deleted') ? true : false, confirmDel = confirm((deleteAction ? 'Delete' : 'Restore') + ' this post?'); if (confirmDel) { deleteAction ? socket.emit('api:posts.delete', { pid: pid }) : socket.emit('api:posts.restore', { pid: pid }); } }); $('#post-container').on('click', '.chat', function(e) { var username = $(this).parents('li.row').attr('data-username'); var touid = $(this).parents('li.row').attr('data-uid'); if (username === app.username || !app.username) return; app.openChat(username, touid); }); ajaxify.register_events([ 'event:rep_up', 'event:rep_down', 'event:new_post', 'api:get_users_in_room', 'event:topic_deleted', 'event:topic_restored', 'event:topic:locked', 'event:topic_unlocked', 'event:topic_pinned', 'event:topic_unpinned', 'event:topic_moved', 'event:post_edited', 'event:post_deleted', 'event:post_restored', 'api:posts.favourite' ]); socket.on('api:get_users_in_room', function(data) { if(data) { var activeEl = $('.thread_active_users'); function createUserIcon(uid, picture, userslug, username) { if(!activeEl.find("[href='/user/"+ data.users[i].userslug + "']").length) { var userIcon = $(''); var userLink = $('').append(userIcon); userLink.attr('data-uid', uid); userLink.tooltip({ placement: 'left', title: username }); return userLink; } } // remove users that are no longer here activeEl.children().each(function(index, element) { if(element) { var uid = $(element).attr('data-uid'); for(var i=0; i 8) { break; } } var remainingUsers = data.users.length - 9; remainingUsers = remainingUsers < 0 ? 0 : remainingUsers; var anonymousCount = parseInt(data.anonymousCount, 10); activeEl.find('.anonymous-box').remove(); if(anonymousCount || remainingUsers) { var anonLink = $(''); activeEl.append(anonLink); var title = ''; if(remainingUsers && anonymousCount) title = remainingUsers + ' more user(s) and ' + anonymousCount + ' guest(s)'; else if(remainingUsers) title = remainingUsers + ' more user(s)'; else title = anonymousCount + ' guest(s)'; anonLink.tooltip({ placement: 'left', title: title }); } } app.populate_online_users(); }); socket.on('event:rep_up', function(data) { adjust_rep(1, data.pid, data.uid); }); socket.on('event:rep_down', function(data) { adjust_rep(-1, data.pid, data.uid); }); socket.on('event:new_post', app.createNewPosts); socket.on('event:topic_deleted', function(data) { if (data.tid === tid && data.status === 'ok') { set_locked_state(true); set_delete_state(true); } }); socket.on('event:topic_restored', function(data) { if (data.tid === tid && data.status === 'ok') { set_locked_state(false); set_delete_state(false); } }); socket.on('event:topic_locked', function(data) { if (data.tid === tid && data.status === 'ok') { set_locked_state(true, 1); } }); socket.on('event:topic_unlocked', function(data) { if (data.tid === tid && data.status === 'ok') { set_locked_state(false, 1); } }); socket.on('event:topic_pinned', function(data) { if (data.tid === tid && data.status === 'ok') { set_pinned_state(true, 1); } }); socket.on('event:topic_unpinned', function(data) { if (data.tid === tid && data.status === 'ok') { set_pinned_state(false, 1); } }); socket.on('event:topic_moved', function(data) { if (data && data.tid > 0) ajaxify.go('topic/' + data.tid); }); socket.on('event:post_edited', function(data) { var editedPostEl = document.getElementById('content_' + data.pid); var editedPostTitle = $('#topic_title_' + data.pid); if (editedPostTitle.length > 0) { editedPostTitle.fadeOut(250, function() { editedPostTitle.html(data.title); editedPostTitle.fadeIn(250); }); } $(editedPostEl).fadeOut(250, function() { this.innerHTML = data.content; $(this).fadeIn(250); }); }); socket.on('api:posts.favourite', function(data) { if (data.status === 'ok' && data.pid) { var favEl = document.querySelector('.post_rep_' + data.pid).nextSibling; if (favEl) { favEl.className = 'icon-star'; $(favEl).parent().addClass('btn-warning'); } } }); socket.on('api:posts.unfavourite', function(data) { if (data.status === 'ok' && data.pid) { var favEl = document.querySelector('.post_rep_' + data.pid).nextSibling; if (favEl) { favEl.className = 'icon-star-empty'; $(favEl).parent().removeClass('btn-warning'); } } }); socket.on('event:post_deleted', function(data) { if (data.pid) toggle_post_delete_state(data.pid, true); }); socket.on('event:post_restored', function(data) { if (data.pid) toggle_post_delete_state(data.pid, true); }); socket.on('api:post.privileges', function(privileges) { if (privileges.editable) toggle_mod_tools(privileges.pid, true); }); function adjust_rep(value, pid, uid) { var post_rep = jQuery('.post_rep_' + pid), user_rep = jQuery('.user_rep_' + uid); var ptotal = parseInt(post_rep.html(), 10), utotal = parseInt(user_rep.html(), 10); ptotal += value; utotal += value; post_rep.html(ptotal + ' '); user_rep.html(utotal + ' '); } function set_locked_state(locked, alert) { var threadReplyBtn = document.getElementById('post_reply'), postReplyBtns = document.querySelectorAll('#post-container .post_reply'), quoteBtns = document.querySelectorAll('#post-container .quote'), editBtns = document.querySelectorAll('#post-container .edit'), deleteBtns = document.querySelectorAll('#post-container .delete'), numPosts = document.querySelectorAll('#post_container li[data-pid]').length, lockThreadEl = document.getElementById('lock_thread'), x; if (locked === true) { lockThreadEl.innerHTML = ' Unlock Thread'; threadReplyBtn.disabled = true; threadReplyBtn.innerHTML = 'Locked '; for (x = 0; x < numPosts; x++) { postReplyBtns[x].innerHTML = 'Locked '; quoteBtns[x].style.display = 'none'; editBtns[x].style.display = 'none'; deleteBtns[x].style.display = 'none'; } if (alert) { app.alert({ 'alert_id': 'thread_lock', type: 'success', title: 'Thread Locked', message: 'Thread has been successfully locked', timeout: 5000 }); } thread_state.locked = '1'; } else { lockThreadEl.innerHTML = ' Lock Thread'; threadReplyBtn.disabled = false; threadReplyBtn.innerHTML = 'Reply'; for (x = 0; x < numPosts; x++) { postReplyBtns[x].innerHTML = 'Reply '; quoteBtns[x].style.display = 'inline-block'; editBtns[x].style.display = 'inline-block'; deleteBtns[x].style.display = 'inline-block'; } if (alert) { app.alert({ 'alert_id': 'thread_lock', type: 'success', title: 'Thread Unlocked', message: 'Thread has been successfully unlocked', timeout: 5000 }); } thread_state.locked = '0'; } } function set_delete_state(deleted) { var deleteThreadEl = document.getElementById('delete_thread'), deleteTextEl = deleteThreadEl.getElementsByTagName('span')[0], threadEl = $('#post-container'), deleteNotice = document.getElementById('thread-deleted') || document.createElement('div'); if (deleted) { deleteTextEl.innerHTML = ' Restore Thread'; threadEl.addClass('deleted'); // Spawn a 'deleted' notice at the top of the page deleteNotice.setAttribute('id', 'thread-deleted'); deleteNotice.className = 'alert alert-warning'; deleteNotice.innerHTML = 'This thread has been deleted. Only users with thread management privileges can see it.'; threadEl.before(deleteNotice); thread_state.deleted = '1'; } else { deleteTextEl.innerHTML = ' Delete Thread'; threadEl.removeClass('deleted'); deleteNotice.parentNode.removeChild(deleteNotice); thread_state.deleted = '0'; } } function set_pinned_state(pinned, alert) { var pinEl = document.getElementById('pin_thread'); if (pinned) { pinEl.innerHTML = ' Unpin Thread'; if (alert) { app.alert({ 'alert_id': 'thread_pin', type: 'success', title: 'Thread Pinned', message: 'Thread has been successfully pinned', timeout: 5000 }); } thread_state.pinned = '1'; } else { pinEl.innerHTML = ' Pin Thread'; if (alert) { app.alert({ 'alert_id': 'thread_pin', type: 'success', title: 'Thread Unpinned', message: 'Thread has been successfully unpinned', timeout: 5000 }); } thread_state.pinned = '0'; } } function toggle_post_delete_state(pid) { var postEl = $(document.querySelector('#post-container li[data-pid="' + pid + '"]')); if (postEl[0]) { quoteEl = $(postEl[0].querySelector('.quote')), favEl = $(postEl[0].querySelector('.favourite')), replyEl = $(postEl[0].querySelector('.post_reply')); socket.once('api:post.privileges', function(privileges) { if (privileges.editable) { if (!postEl.hasClass('deleted')) { toggle_post_tools(pid, false); } else { toggle_post_tools(pid, true); } } if (privileges.view_deleted) { postEl.toggleClass('deleted'); } else { postEl.toggleClass('none'); } }); socket.emit('api:post.privileges', pid); } } function toggle_post_tools(pid, state) { var postEl = $(document.querySelector('#post-container li[data-pid="' + pid + '"]')), quoteEl = $(postEl[0].querySelector('.quote')), favEl = $(postEl[0].querySelector('.favourite')), replyEl = $(postEl[0].querySelector('.post_reply')); if (state) { quoteEl.removeClass('none'); favEl.removeClass('none'); replyEl.removeClass('none'); } else { quoteEl.addClass('none'); favEl.addClass('none'); replyEl.addClass('none'); } } function toggle_mod_tools(pid, state) { var postEl = $(document.querySelector('#post-container li[data-pid="' + pid + '"]')), editEl = postEl.find('.edit'), deleteEl = postEl.find('.delete'); if (state) { editEl.removeClass('none'); deleteEl.removeClass('none'); } else { editEl.addClass('none'); deleteEl.addClass('none'); } } var postAuthorImage, mobileAuthorOverlay, pagination; var postcount = templates.get('postcount'); function updateHeader() { if (pagination == null) { jQuery('.pagination-block i:first').on('click', function() { app.scrollToTop(); }); jQuery('.pagination-block i:last').on('click', function() { app.scrollToBottom(); }); } jQuery('.mobile-author-overlay').css('bottom', '0px'); postAuthorImage = postAuthorImage || document.getElementById('mobile-author-image'); mobileAuthorOverlay = mobileAuthorOverlay || document.getElementById('mobile-author-overlay'); pagination = pagination || document.getElementById('pagination'); pagination.parentNode.style.display = 'block'; var windowHeight = jQuery(window).height(); var scrollTop = jQuery(window).scrollTop(); var scrollBottom = scrollTop + windowHeight; if (scrollTop < 50 && postcount > 1) { localStorage.removeItem("topic:" + tid + ":bookmark"); postAuthorImage.src = (jQuery('.main-post .avatar img').attr('src')); mobileAuthorOverlay.innerHTML = 'Posted by ' + jQuery('.main-post').attr('data-username') + ', ' + jQuery('.main-post').find('.relativeTimeAgo').html(); pagination.innerHTML = '0 out of ' + postcount; return; } var count = 0, smallestNonNegative = 0; jQuery('.sub-posts').each(function() { count++; this.postnumber = count; var el = jQuery(this); var elTop = el.offset().top; var height = Math.floor(el.height()); var elBottom = elTop + (height < 300 ? height : 300); var inView = ((elBottom >= scrollTop) && (elTop <= scrollBottom) && (elBottom <= scrollBottom) && (elTop >= scrollTop)); if (inView) { if(elTop - scrollTop > smallestNonNegative) { localStorage.setItem("topic:" + tid + ":bookmark", el.attr('data-pid')); smallestNonNegative = Number.MAX_VALUE; } pagination.innerHTML = this.postnumber + ' out of ' + postcount; postAuthorImage.src = (jQuery(this).find('.profile-image-block img').attr('src')); mobileAuthorOverlay.innerHTML = 'Posted by ' + jQuery(this).attr('data-username') + ', ' + jQuery(this).find('.relativeTimeAgo').html(); } }); setTimeout(function() { if (scrollTop + windowHeight == jQuery(document).height()) { pagination.innerHTML = postcount + ' out of ' + postcount; } }, 100); } window.onscroll = updateHeader; window.onload = updateHeader; }; return Topic; });