* @fixme can't submit file uploads
*
* @param {jQuery} form: jQuery object whose first element is a form
+ * @param function onSuccess: something extra to do on success
*
* @access public
*/
- FormXHR: function(form) {
+ FormXHR: function(form, onSuccess) {
$.ajax({
type: 'POST',
dataType: 'xml',
errorReported = $('#error', xhr.responseXML).text();
}
alert(errorReported || errorThrown || textStatus);
-
+
// Restore the form to original state.
// Hopefully. :D
form
},
success: function(data, textStatus) {
if (typeof($('form', data)[0]) != 'undefined') {
- form_new = document._importNode($('form', data)[0], true);
+ var form_new = $('form', $(data).children());
form.replaceWith(form_new);
+ if (onSuccess) {
+ onSuccess();
+ }
}
else if (typeof($('p', data)[0]) != 'undefined') {
- form.replaceWith(document._importNode($('p', data)[0], true));
+ var p_new = $('p', $(data).children());
+ form.replaceWith(p_new);
+ if (onSuccess) {
+ onSuccess();
+ }
}
else {
alert('Unknown error.');
success: function(data, textStatus) {
var results_placeholder = $('#profile_search_results');
if (typeof($('ul', data)[0]) != 'undefined') {
- var list = document._importNode($('ul', data)[0], true);
+ var list = $('ul', $(data).children());
results_placeholder.replaceWith(list);
}
else {
- var _error = $('<li/>').append(document._importNode($('p', data)[0], true));
+ var _error = $('<li/>').append($('p', $(data).children()));
results_placeholder.html(_error);
}
form
url: form.attr('action'),
data: form.serialize() + '&ajax=1',
beforeSend: function(xhr) {
- form.addClass(SN.C.S.Processing)
- .find('.submit')
+ form.find('.submit')
+ .addClass(SN.C.S.Processing)
.addClass(SN.C.S.Disabled)
.attr(SN.C.S.Disabled, SN.C.S.Disabled);
},
success: function(data, textStatus) {
var results_placeholder = form.parents('.entity_tags');
if (typeof($('.entity_tags', data)[0]) != 'undefined') {
- var tags = document._importNode($('.entity_tags', data)[0], true);
+ var tags = $('.entity_tags', $(data).children());
$(tags).find('.editable').append($('<button class="peopletags_edit_button"/>'));
results_placeholder.replaceWith(tags);
} else {
results_placeholder.find('p').remove();
- results_placeholder.append(document._importNode($('p', data)[0], true));
+ results_placeholder.append($('p', $(data).children()));
form.removeClass(SN.C.S.Processing)
.find('.submit')
.removeClass(SN.C.S.Disabled)
$('#content .notice_reply').live('click', function(e) {
e.preventDefault();
var notice = $(this).closest('li.notice');
- var nickname = ($('.author .nickname', notice).length > 0) ? $($('.author .nickname', notice)[0]) : $('.author .nickname.uid');
- SN.U.NoticeInlineReplyTrigger(notice, '@' + nickname.text());
+ SN.U.NoticeInlineReplyTrigger(notice);
return false;
});
},
// and we'll add on the end of it. Will add if needed.
list = $('ul.threaded-replies', notice);
if (list.length == 0) {
- list = $('<ul class="notices threaded-replies xoxo"></ul>');
- notice.append(list);
+ SN.U.NoticeInlineReplyPlaceholder(notice);
+ list = $('ul.threaded-replies', notice);
+ } else {
+ var placeholder = $('li.notice-reply-placeholder', notice);
+ if (placeholder.length == 0) {
+ SN.U.NoticeInlineReplyPlaceholder(notice);
+ }
}
}
var nextStep = function() {
// Override...?
replyForm.find('input[name=inreplyto]').val(id);
+ replyForm.find('#notice_to').attr('disabled', 'disabled').hide();
+ replyForm.find('#notice_private').attr('disabled', 'disabled').hide();
+ replyForm.find('label[for=notice_to]').hide();
+ replyForm.find('label[for=notice_private]').hide();
// Set focus...
var text = replyForm.find('textarea');
replyItem = $('<li class="notice-reply"></li>');
var intermediateStep = function(formMaster) {
- var formEl = document._importNode(formMaster, true);
- replyItem.append(formEl);
+ replyItem.append(formMaster);
list.append(replyItem); // *after* the placeholder
- var form = replyForm = $(formEl);
+ var form = replyForm = formMaster;
SN.Init.NoticeFormSetup(form);
nextStep();
// @fixme this fallback may or may not work
var url = $('#form_notice').attr('action');
$.get(url, {ajax: 1}, function(data, textStatus, xhr) {
- intermediateStep($('form', data)[0]);
- });
+ intermediateStep($('form', $(data).children()));
+ }, 'xml');
}
}
}
NoticeInlineReplyPlaceholder: function(notice) {
var list = notice.find('ul.threaded-replies');
+ if (list.length == 0) {
+ list = $('<ul class="notices threaded-replies xoxo"></ul>');
+ notice.append(list);
+ list = notice.find('ul.threaded-replies');
+ }
var placeholder = $('<li class="notice-reply-placeholder">' +
'<input class="placeholder">' +
'</li>');
var url = $(this).attr('href');
var area = $(this).closest('.threaded-replies');
$.get(url, {ajax: 1}, function(data, textStatus, xhr) {
- var replies = $('.threaded-replies', data);
+ var replies = $('.threaded-replies', $(data).children());
if (replies.length) {
- area.replaceWith(document._importNode(replies[0], true));
+ area.replaceWith(replies);
}
- });
+ }, 'xml');
return false;
});
},
* popout before submitting.
*
* Uses 'live' rather than 'bind', so applies to future as well as present items.
+ *
*/
NoticeRepeat: function() {
$('.form_repeat').live('click', function(e) {
if (NDMF.length === 0) {
$(this).addClass(SN.C.S.Processing);
$.get(NDM.attr('href'), null, function(data) {
- $('.entity_send-a-message').append(document._importNode($('form', data)[0], true));
+ $('.entity_send-a-message').append($('form', $(data).children()));
NDMF = $('.entity_send-a-message .form_notice');
SN.U.FormNoticeXHR(NDMF);
SN.U.FormNoticeEnhancements(NDMF);
return false;
});
NDM.removeClass(SN.C.S.Processing);
- });
+ }, 'xml');
}
else {
NDMF.show();
$('#input_form_nav_'+tag).addClass('current');
}
+ // Don't remove 'current' if we also have the "nonav" class.
+ // An example would be the message input form. removing
+ // 'current' will cause the form to vanish from the page.
+ var nonav = $('.input_form.current.nonav');
+ if (nonav.length > 0) {
+ return;
+ }
+
$('.input_form.current').removeClass('current');
$('#input_form_'+tag)
.addClass('current')
var form = $(this);
SN.Init.NoticeFormSetup(form);
})
- .find('textarea:first').focus();
+ .find('.notice_data-text').focus();
}
},
});
}
});
+
+ // Infield labels for notice form inputs.
+ $('.input_forms fieldset fieldset label').inFieldLabels({ fadeOpacity:0 });
+
}
},
if ($('body.user_in').length > 0) {
var masterForm = $('.form_notice:first');
if (masterForm.length > 0) {
- SN.C.I.NoticeFormMaster = document._importNode(masterForm[0], true);
+ SN.C.I.NoticeFormMaster = masterForm;
}
SN.U.NoticeRepeat();
SN.U.NoticeReply();
});
},
- PeopletagAutocomplete: function() {
- $('.form_tag_user #tags').tagInput({
- tags: SN.C.PtagACData,
- tagSeparator: " ",
- animate: false,
- formatLine: function (i, e, search, matches) {
- var tag = "<b>" + e.tag.substring(0, search.length) + "</b>" + e.tag.substring(search.length);
-
- var line = $("<div/>").addClass('mode-' + e.mode);
- line.append($("<div class='tagInputLineTag'>" + tag
- + " <em class='privacy_mode'>" + e.mode + "</em></div>"));
- if (e.freq)
- line.append("<div class='tagInputLineFreq'>" + e.freq + "</div>");
- return line;
+ /**
+ * Called when a people tag edit box is shown in the interface
+ *
+ * - loads the jQuery UI autocomplete plugin
+ * - sets event handlers for tag completion
+ *
+ */
+ PeopletagAutocomplete: function(txtBox) {
+ var split = function(val) {
+ return val.split( /\s+/ );
+ }
+ var extractLast = function(term) {
+ return split(term).pop();
+ }
+
+ // don't navigate away from the field on tab when selecting an item
+ txtBox.live( "keydown", function( event ) {
+ if ( event.keyCode === $.ui.keyCode.TAB &&
+ $(this).data( "autocomplete" ).menu.active ) {
+ event.preventDefault();
}
- });
+ }).autocomplete({
+ minLength: 0,
+ source: function(request, response) {
+ // delegate back to autocomplete, but extract the last term
+ response($.ui.autocomplete.filter(
+ SN.C.PtagACData, extractLast(request.term)));
+ },
+ focus: function() {
+ return false;
+ },
+ select: function(event, ui) {
+ var terms = split(this.value);
+ terms.pop();
+ terms.push(ui.item.value);
+ terms.push("");
+ this.value = terms.join(" ");
+ return false;
+ }
+ }).data('autocomplete')._renderItem = function(ul, item) {
+ // FIXME: with jQuery UI you cannot have it highlight the match
+ var _l = '<a class="ptag-ac-line-tag">' + item.tag
+ + ' <em class="privacy_mode">' + item.mode + '</em>'
+ + '<span class="freq">' + item.freq + '</span></a>'
+
+ return $("<li/>")
+ .addClass('mode-' + item.mode)
+ .addClass('ptag-ac-line')
+ .data("item.autocomplete", item)
+ .append(_l)
+ .appendTo(ul);
+ }
},
+ /**
+ * Run setup for the ajax people tags editor
+ *
+ * - show edit button
+ * - set event handle for click on edit button
+ * - loads people tag autocompletion data if not already present
+ * or if it is stale.
+ *
+ */
PeopleTags: function() {
$('.user_profile_tags .editable').append($('<button class="peopletags_edit_button"/>'));
$('.peopletags_edit_button').live('click', function() {
var form = $(this).parents('dd').eq(0).find('form');
// We can buy time from the above animation
- if (typeof SN.C.PtagACData === 'undefined') {
- $.getJSON(_peopletagAC + '?token=' + $('#token').val(), function(data) {
+
+ $.ajax({
+ url: _peopletagAC,
+ dataType: 'json',
+ data: {token: $('#token').val()},
+ ifModified: true,
+ success: function(data) {
+ // item.label is used to match
+ for (i=0; i < data.length; i++) {
+ data[i].label = data[i].tag;
+ }
+
SN.C.PtagACData = data;
- _loadTagInput(SN.Init.PeopletagAutocomplete);
- });
- } else { _loadTagInput(SN.Init.PeopletagAutocomplete); }
+ SN.Init.PeopletagAutocomplete(form.find('#tags'));
+ }
+ });
$(this).parents('ul').eq(0).fadeOut(200, function() {form.fadeIn(200).find('input#tags')});
- })
+ });
$('.user_profile_tags form .submit').live('click', function() {
SN.U.FormPeopletagsXHR($(this).parents('form')); return false;
SN.U.FormXHR($(this));
return false;
});
+ $('form.ajax input[type=submit]').live('click', function() {
+ // Some forms rely on knowing which submit button was clicked.
+ // Save a hidden input field which'll be picked up during AJAX
+ // submit...
+ var button = $(this);
+ var form = button.closest('form');
+ form.find('.hidden-submit-button').remove();
+ $('<input class="hidden-submit-button" type="hidden" />')
+ .attr('name', button.attr('name'))
+ .val(button.val())
+ .appendTo(form);
+ });
},
/**
}
}
});
- }
+ },
+
+ CheckBoxes: function() {
+ $("span[class='checkbox-wrapper']").addClass("unchecked");
+ $(".checkbox-wrapper").click(function(){
+ if($(this).children("input").attr("checked")){
+ // uncheck
+ $(this).children("input").attr({checked: ""});
+ $(this).removeClass("checked");
+ $(this).addClass("unchecked");
+ $(this).children("label").text("Private?");
+ }else{
+ // check
+ $(this).children("input").attr({checked: "checked"});
+ $(this).removeClass("unchecked");
+ $(this).addClass("checked");
+ $(this).children("label").text("Private");
+ }
+ });
+ }
}
};
$(document).ready(function(){
SN.Init.AjaxForms();
SN.Init.UploadForms();
+ SN.Init.CheckBoxes();
if ($('.'+SN.C.S.FormNotice).length > 0) {
SN.Init.NoticeForm();
}
SN.Init.PeopleTags();
}
});
+