2 * We do not want the CSRF protection enabled for the AJAX post requests, it causes only trouble.
\r
3 * Get the csrftoken cookie and pass it to the X-CSRFToken HTTP request property.
\r
5 $('html').ajaxSend(function(event, xhr, settings) {
\r
6 function getCookie(name) {
\r
7 var cookieValue = null;
\r
8 if (document.cookie && document.cookie != '') {
\r
9 var cookies = document.cookie.split(';');
\r
10 for (var i = 0; i < cookies.length; i++) {
\r
11 var cookie = jQuery.trim(cookies[i]);
\r
12 // Does this cookie string begin with the name we want?
\r
13 if (cookie.substring(0, name.length + 1) == (name + '=')) {
\r
14 cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
\r
21 if (!(/^http:.*/.test(settings.url) || /^https:.*/.test(settings.url))) {
\r
22 // Only send the token to relative URLs i.e. locally.
\r
23 xhr.setRequestHeader("X-CSRFToken", getCookie('csrftoken'));
\r
27 var response_commands = {
\r
28 refresh_page: function() {
\r
29 window.location.reload(true)
\r
32 update_post_score: function(id, inc) {
\r
33 var $score_board = $('#post-' + id + '-score');
\r
34 var current = parseInt($score_board.html())
\r
35 if (isNaN(current)){
\r
38 $score_board.html(current + inc)
\r
41 update_user_post_vote: function(id, vote_type) {
\r
42 var $upvote_button = $('#post-' + id + '-upvote');
\r
43 var $downvote_button = $('#post-' + id + '-downvote');
\r
45 $upvote_button.removeClass('on');
\r
46 $downvote_button.removeClass('on');
\r
48 if (vote_type == 'up') {
\r
49 $upvote_button.addClass('on');
\r
50 } else if (vote_type == 'down') {
\r
51 $downvote_button.addClass('on');
\r
55 update_favorite_count: function(inc) {
\r
56 var $favorite_count = $('#favorite-count');
\r
57 var count = parseInt($favorite_count.html());
\r
67 $favorite_count.html(count);
\r
70 update_favorite_mark: function(type) {
\r
72 $('#favorite-mark').addClass('on');
\r
74 $('#favorite-mark').removeClass('on');
\r
78 mark_accepted: function(id) {
\r
79 var $answer = $('#answer-container-' + id);
\r
80 $answer.addClass('accepted-answer');
\r
81 $answer.find('.accept-answer').addClass('on');
\r
82 $answer.find('.accept-answer').attr('title', $answer.find('.accept-answer').attr('bn:on'));
\r
85 unmark_accepted: function(id) {
\r
86 var $answer = $('#answer-container-' + id);
\r
87 $answer.removeClass('accepted-answer');
\r
88 $answer.find('.accept-answer').removeClass('on');
\r
89 $answer.find('.accept-answer').attr('title', $answer.find('.accept-answer').attr('bn:off'));
\r
92 remove_comment: function(id) {
\r
93 var $comment = $('#comment-' + id);
\r
94 $comment.css('background', 'red')
\r
95 $comment.fadeOut('slow', function() {
\r
100 award_points: function(id) {
\r
104 insert_comment: function(post_id, comment_id, comment, username, profile_url, delete_url, edit_url, convert_url, can_convert) {
\r
105 var $container = $('#comments-container-' + post_id);
\r
106 var skeleton = $('#new-comment-skeleton-' + post_id).html().toString();
\r
108 skeleton = skeleton.replace(new RegExp('%ID%', 'g'), comment_id)
\r
109 .replace(new RegExp('%COMMENT%', 'g'), comment)
\r
110 .replace(new RegExp('%USERNAME%', 'g'), username)
\r
111 .replace(new RegExp('%PROFILE_URL%', 'g'), profile_url)
\r
112 .replace(new RegExp('%DELETE_URL%', 'g'), delete_url)
\r
113 .replace(new RegExp('%EDIT_URL%', 'g'), edit_url)
\r
114 .replace(new RegExp('%CONVERT_URL%', 'g'), convert_url);
\r
116 $container.append(skeleton);
\r
118 // Show the convert comment to answer tool only if the current comment can be converted
\r
119 if (can_convert == true) {
\r
120 $('#comment-' + comment_id + '-convert').show();
\r
123 $('#comment-' + comment_id).slideDown('slow');
\r
126 update_comment: function(comment_id, comment_text) {
\r
127 var $comment = $('#comment-' + comment_id);
\r
128 $comment.find('.comment-text').html(comment_text);
\r
130 $comment.slideDown('slow');
\r
133 mark_deleted: function(post_type, post_id) {
\r
134 if (post_type == 'question') {
\r
135 var $container = $('#question-table');
\r
136 $container.addClass('deleted');
\r
138 var $el = $('#' + post_type + '-container-' + post_id);
\r
139 $el.addClass('deleted');
\r
143 unmark_deleted: function(post_type, post_id) {
\r
144 if (post_type == 'answer') {
\r
145 var $answer = $('#answer-container-' + post_id);
\r
146 $answer.removeClass('deleted');
\r
148 var $container = $('#question-table');
\r
149 $container.removeClass('deleted');
\r
153 set_subscription_button: function(text) {
\r
154 $('.subscription_switch').html(text);
\r
157 set_subscription_status: function(text) {
\r
158 $('.subscription-status').html(text);
\r
161 copy_url: function(url) {
\r
166 function show_dialog (extern) {
\r
167 var default_close_function = function($diag) {
\r
168 $diag.fadeOut('fast', function() {
\r
176 x: ($(window).width() / 2) + $(window).scrollLeft(),
\r
177 y: ($(window).height() / 2) + $(window).scrollTop()
\r
180 yes_text: messages.ok,
\r
181 yes_callback: default_close_function,
\r
182 no_text: messages.cancel,
\r
184 close_on_clickoutside: false
\r
187 $.extend(options, extern);
\r
189 if (options.event != undefined) {
\r
190 options.pos = {x: options.event.pageX, y: options.event.pageY};
\r
193 var html = '<div class="dialog ' + options.extra_class + '" style="display: none;">'
\r
194 + '<div class="dialog-content">' + options.html + '</div><div class="dialog-buttons">';
\r
196 if (options.show_no) {
\r
197 html += '<button class="dialog-no">' + options.no_text + '</button>';
\r
200 html += '<button class="dialog-yes">' + options.yes_text + '</button>'
\r
204 $('body').append($dialog);
\r
205 var message = $('.dialog-content')[0];
\r
206 message.style.visibility = "hidden";
\r
208 if (options.dim === false) {
\r
210 visibility: 'hidden',
\r
214 options.dim = {w: $dialog.width(), h: $dialog.height()};
\r
219 visibility: 'visible'
\r
224 top: options.pos.y,
\r
225 left: options.pos.x
\r
228 top_position_change = (options.dim.h / 2)
\r
229 left_position_change = (options.dim.w / 2)
\r
231 new_top_position = options.pos.y - top_position_change
\r
232 new_left_position = options.pos.x - left_position_change
\r
234 if (new_left_position < 0) {
\r
235 left_position_change = 0
\r
237 if (($(window).scrollTop() - new_top_position) > 0) {
\r
238 top_position_change = 0
\r
240 if ((options.event.pageY + options.dim.h) > ($(window).height() + $(window).scrollTop())) {
\r
241 top_position_change = options.dim.h
\r
243 if ((options.event.pageX + options.dim.w) > ($(window).width() + $(window).scrollLeft())) {
\r
244 left_position_change = options.dim.w
\r
248 top: "-=" + top_position_change,
\r
249 left: "-=" + left_position_change,
\r
250 width: options.dim.w,
\r
251 height: options.dim.h
\r
252 }, 200, function() {
\r
253 message.style.visibility = "visible";
\r
256 $dialog.find('.dialog-no').click(function() {
\r
257 default_close_function($dialog);
\r
260 $dialog.find('.dialog-yes').click(function() {
\r
261 options.yes_callback($dialog);
\r
264 if (options.close_on_clickoutside) {
\r
265 $dialog.one('clickoutside', function() {
\r
266 default_close_function($dialog);
\r
273 function show_message(evt, msg, callback) {
\r
274 var $dialog = show_dialog({
\r
276 extra_class: 'warning',
\r
278 yes_callback: function() {
\r
279 $dialog.fadeOut('fast', function() {
\r
286 close_on_clickoutside: true
\r
290 function load_prompt(evt, el, url) {
\r
291 $.get(url, function(data) {
\r
294 extra_class: 'prompt',
\r
295 yes_callback: function() {
\r
297 $dialog.find('input, textarea, select').each(function() {
\r
298 postvars[$(this).attr('name')] = $(this).val();
\r
300 $.post(url, postvars, function(data) {
\r
301 $dialog.fadeOut('fast', function() {
\r
304 process_ajax_response(data, evt);
\r
310 if (!el.is('.centered')) {
\r
311 doptions.event = evt;
\r
314 var $dialog = show_dialog(doptions);
\r
318 function process_ajax_response(data, evt, callback) {
\r
319 if (!data.success && data['error_message'] != undefined) {
\r
320 show_message(evt, data.error_message, function() {if (callback) callback(true);});
\r
321 end_command(false);
\r
322 } else if (typeof data['commands'] != undefined){
\r
323 for (var command in data.commands) {
\r
324 response_commands[command].apply(null, data.commands[command])
\r
329 if (data['message'] != undefined) {
\r
330 show_message(evt, data.message, function() {if (callback) callback(false);})
\r
332 if (callback) callback(false);
\r
338 var running = false;
\r
340 function start_command() {
\r
341 $('body').append($('<div id="command-loader"></div>'));
\r
345 function end_command(success) {
\r
347 $('#command-loader').addClass('success');
\r
348 $('#command-loader').fadeOut("slow", function() {
\r
349 $('#command-loader').remove();
\r
353 $('#command-loader').remove();
\r
359 $('a.ajax-command').live('click', function(evt) {
\r
360 if (running) return false;
\r
364 var ajax_url = el.attr('href')
\r
365 ajax_url = ajax_url + "?nocache=" + new Date().getTime()
\r
367 $('.context-menu-dropdown').slideUp('fast');
\r
369 if (el.is('.withprompt')) {
\r
370 load_prompt(evt, el, ajax_url);
\r
371 } else if(el.is('.confirm')) {
\r
373 html: messages.confirm,
\r
374 extra_class: 'confirm',
\r
375 yes_callback: function() {
\r
377 $.getJSON(ajax_url, function(data) {
\r
378 process_ajax_response(data, evt);
\r
379 $dialog.fadeOut('fast', function() {
\r
384 yes_text: messages.yes,
\r
386 no_text: messages.no
\r
389 if (!el.is('.centered')) {
\r
390 doptions.event = evt;
\r
392 var $dialog = show_dialog(doptions);
\r
395 $.getJSON(ajax_url, function(data) {
\r
396 process_ajax_response(data, evt);
\r
403 $('.context-menu').each(function() {
\r
404 var $menu = $(this);
\r
405 var $trigger = $menu.find('.context-menu-trigger');
\r
406 var $dropdown = $menu.find('.context-menu-dropdown');
\r
408 $trigger.click(function() {
\r
409 $dropdown.slideToggle('fast', function() {
\r
410 if ($dropdown.is(':visible')) {
\r
411 $dropdown.one('clickoutside', function() {
\r
412 if ($dropdown.is(':visible'))
\r
413 $dropdown.slideUp('fast');
\r
420 $('div.comment-form-container').each(function() {
\r
421 var $container = $(this);
\r
422 var $comment_tools = $container.parent().find('.comment-tools');
\r
423 var $comments_container = $container.parent().find('.comments-container');
\r
425 var $form = $container.find('form');
\r
427 if ($form.length) {
\r
428 var $textarea = $container.find('textarea');
\r
429 var textarea = $textarea.get(0);
\r
430 var $csrf = $container.find('[name="csrfmiddlewaretoken"]');
\r
431 var $button = $container.find('.comment-submit');
\r
432 var $cancel = $container.find('.comment-cancel');
\r
433 var $chars_left_message = $container.find('.comments-chars-left-msg');
\r
434 var $chars_togo_message = $container.find('.comments-chars-togo-msg');
\r
435 var $chars_counter = $container.find('.comments-char-left-count');
\r
437 var $add_comment_link = $comment_tools.find('.add-comment-link');
\r
439 var chars_limits = $chars_counter.html().split('|');
\r
441 var min_length = parseInt(chars_limits[0]);
\r
442 var max_length = parseInt(chars_limits[1]);
\r
444 var warn_length = max_length - 30;
\r
445 var current_length = 0;
\r
446 var comment_in_form = false;
\r
447 var interval = null;
\r
449 var hcheck = !($.browser.msie || $.browser.opera);
\r
451 $textarea.css("padding-top", 0).css("padding-bottom", 0).css("resize", "none");
\r
452 textarea.style.overflow = 'hidden';
\r
455 function cleanup_form() {
\r
457 $textarea.css('height', 80);
\r
458 $chars_counter.html(max_length);
\r
459 $chars_left_message.removeClass('warn');
\r
460 comment_in_form = false;
\r
461 current_length = 0;
\r
463 $chars_left_message.hide();
\r
464 $chars_togo_message.show();
\r
466 $chars_counter.removeClass('warn');
\r
467 $chars_counter.html(min_length);
\r
468 $button.attr("disabled","disabled");
\r
475 function process_form_changes() {
\r
476 var length = $textarea.val().replace(/[ ]{2,}/g," ").length;
\r
478 if (current_length == length)
\r
481 if (length < warn_length && current_length >= warn_length) {
\r
482 $chars_counter.removeClass('warn');
\r
483 } else if (current_length < warn_length && length >= warn_length){
\r
484 $chars_counter.addClass('warn');
\r
487 if (length < min_length) {
\r
488 $chars_left_message.hide();
\r
489 $chars_togo_message.show();
\r
490 $chars_counter.html(min_length - length);
\r
492 length = $textarea.val().length;
\r
493 $chars_togo_message.hide();
\r
494 $chars_left_message.show();
\r
495 $chars_counter.html(max_length - length);
\r
498 if (length > max_length || length < min_length) {
\r
499 $button.attr("disabled","disabled");
\r
501 $button.removeAttr("disabled");
\r
504 var current_height = textarea.style.height;
\r
506 textarea.style.height = "0px";
\r
508 var h = Math.max(80, textarea.scrollHeight);
\r
509 textarea.style.height = current_height;
\r
510 $textarea.animate({height: h + 'px'}, 50);
\r
512 current_length = length;
\r
515 function show_comment_form() {
\r
516 $container.slideDown('slow');
\r
517 $add_comment_link.fadeOut('slow');
\r
519 interval = window.setInterval(function() {
\r
520 process_form_changes();
\r
524 function hide_comment_form() {
\r
525 if (interval != null) {
\r
526 window.clearInterval(interval);
\r
529 $container.slideUp('slow');
\r
530 $add_comment_link.fadeIn('slow');
\r
533 $add_comment_link.click(function(){
\r
535 show_comment_form();
\r
539 $('#' + $comments_container.attr('id') + ' .comment-edit').live('click', function() {
\r
540 var $link = $(this);
\r
541 var comment_id = /comment-(\d+)-edit/.exec($link.attr('id'))[1];
\r
542 var $comment = $('#comment-' + comment_id);
\r
544 comment_in_form = comment_id;
\r
546 $.get($link.attr('href'), function(data) {
\r
547 $textarea.val(data);
\r
550 $comment.slideUp('slow');
\r
551 show_comment_form();
\r
555 $button.click(function(evt) {
\r
556 if (running) return false;
\r
559 comment: $textarea.val(),
\r
560 csrfmiddlewaretoken: $csrf.val()
\r
563 if (comment_in_form) {
\r
564 post_data['id'] = comment_in_form;
\r
568 $.post($form.attr('action'), post_data, function(data) {
\r
569 process_ajax_response(data, evt, function(error) {
\r
572 hide_comment_form();
\r
581 $cancel.click(function(event) {
\r
582 if (confirm("You will lose all of your changes in this comment. Do you still wish to proceed?")){
\r
583 if (comment_in_form) {
\r
584 $comment = $('#comment-' + comment_in_form).slideDown('slow');
\r
586 hide_comment_form();
\r
593 $comment_tools.find('.show-all-comments-link').click(function() {
\r
594 $comments_container.find('.not_top_scorer').slideDown('slow');
\r
595 $(this).fadeOut('slow');
\r
596 $comment_tools.find('.comments-showing').fadeOut('slow');
\r
601 if ($('#editor').length) {
\r
602 var $editor = $('#editor');
\r
603 var $previewer = $('#previewer');
\r
604 var $container = $('#editor-metrics');
\r
606 var initial_whitespace_rExp = /^[^A-Za-zА-Яа-я0-9]+/gi;
\r
607 var non_alphanumerics_rExp = rExp = /[^A-Za-zА-Яа-я0-9]+/gi;
\r
608 var editor_interval = null;
\r
610 $editor.focus(function() {
\r
611 if (editor_interval == null) {
\r
612 editor_interval = window.setInterval(function() {
\r
618 function recalc_metrics() {
\r
619 var text = $previewer.text();
\r
621 var char_count = text.length;
\r
622 var fullStr = text + " ";
\r
623 var left_trimmedStr = fullStr.replace(initial_whitespace_rExp, "");
\r
624 var cleanedStr = left_trimmedStr.replace(non_alphanumerics_rExp, " ");
\r
625 var splitString = cleanedStr.split(" ");
\r
626 var word_count = splitString.length - 1;
\r
628 var metrics = char_count + " " + (char_count == 1 ? messages.character : messages.characters);
\r
629 metrics += " / " + word_count + " " + (word_count == 1 ? messages.word : messages.words);
\r
630 $container.html(metrics);
\r
635 //var scriptUrl, interestingTags, ignoredTags, tags, $;
\r
636 function pickedTags(){
\r
638 var sendAjax = function(tagname, reason, action, callback){
\r
639 var url = scriptUrl;
\r
640 if (action == 'add'){
\r
641 url += $.i18n._('mark-tag/');
\r
642 if (reason == 'good'){
\r
643 url += $.i18n._('interesting/');
\r
646 url += $.i18n._('ignored/');
\r
650 url += $.i18n._('unmark-tag/');
\r
652 url = url + tagname + '/';
\r
654 var call_settings = {
\r
659 if (callback !== false){
\r
660 call_settings.success = callback;
\r
662 $.ajax(call_settings);
\r
666 var unpickTag = function(from_target ,tagname, reason, send_ajax){
\r
667 //send ajax request to delete tag
\r
668 var deleteTagLocally = function(){
\r
669 from_target[tagname].remove();
\r
670 delete from_target[tagname];
\r
673 sendAjax(tagname,reason,'remove',deleteTagLocally);
\r
676 deleteTagLocally();
\r
681 var setupTagDeleteEvents = function(obj,tag_store,tagname,reason,send_ajax){
\r
682 obj.unbind('mouseover').bind('mouseover', function(){
\r
683 $(this).attr('src', mediaUrl('media/images/close-small-hover.png'));
\r
685 obj.unbind('mouseout').bind('mouseout', function(){
\r
686 $(this).attr('src', mediaUrl('media/images/close-small-dark.png'));
\r
688 obj.click( function(){
\r
689 unpickTag(tag_store,tagname,reason,send_ajax);
\r
693 var handlePickedTag = function(obj,reason){
\r
694 var tagname = $.trim($(obj).prev().attr('value'));
\r
695 var to_target = interestingTags;
\r
696 var from_target = ignoredTags;
\r
697 var to_tag_container;
\r
698 if (reason == 'bad'){
\r
699 to_target = ignoredTags;
\r
700 from_target = interestingTags;
\r
701 to_tag_container = $('div .tags.ignored');
\r
703 else if (reason != 'good'){
\r
707 to_tag_container = $('div .tags.interesting');
\r
710 if (tagname in from_target){
\r
711 unpickTag(from_target,tagname,reason,false);
\r
714 if (!(tagname in to_target)){
\r
715 //send ajax request to pick this tag
\r
717 sendAjax(tagname,reason,'add',function(){
\r
718 var new_tag = $('<span></span>');
\r
719 new_tag.addClass('deletable-tag');
\r
720 var tag_link = $('<a></a>');
\r
721 tag_link.attr('rel','tag');
\r
722 tag_link.attr('href', scriptUrl + $.i18n._('tags/') + tagname + '/');
\r
723 tag_link.html(tagname);
\r
724 var del_link = $('<img></img>');
\r
725 del_link.addClass('delete-icon');
\r
726 del_link.attr('src', mediaUrl('media/images/close-small-dark.png'));
\r
728 setupTagDeleteEvents(del_link, to_target, tagname, reason, true);
\r
730 new_tag.append(tag_link);
\r
731 new_tag.append(del_link);
\r
732 to_tag_container.append(new_tag);
\r
734 to_target[tagname] = new_tag;
\r
739 var collectPickedTags = function(){
\r
740 var good_prefix = 'interesting-tag-';
\r
741 var bad_prefix = 'ignored-tag-';
\r
742 var good_re = RegExp('^' + good_prefix);
\r
743 var bad_re = RegExp('^' + bad_prefix);
\r
744 interestingTags = {};
\r
746 $('.deletable-tag').each(
\r
748 var item_id = $(item).attr('id');
\r
749 var tag_name, tag_store;
\r
750 if (good_re.test(item_id)){
\r
751 tag_name = item_id.replace(good_prefix,'');
\r
752 tag_store = interestingTags;
\r
755 else if (bad_re.test(item_id)){
\r
756 tag_name = item_id.replace(bad_prefix,'');
\r
757 tag_store = ignoredTags;
\r
763 tag_store[tag_name] = $(item);
\r
764 setupTagDeleteEvents($(item).find('img'),tag_store,tag_name,reason,true);
\r
769 var setupHideIgnoredQuestionsControl = function(){
\r
770 $('#hideIgnoredTagsCb').unbind('click').click(function(){
\r
775 url: scriptUrl + $.i18n._('command/'),
\r
776 data: {command:'toggle-ignored-questions'}
\r
782 collectPickedTags();
\r
783 setupHideIgnoredQuestionsControl();
\r
784 $("#interestingTagInput, #ignoredTagInput").autocomplete(messages.matching_tags_url, {
\r
786 matchContains: true,
\r
788 /*multiple: false, - the favorite tags and ignore tags don't let you do multiple tags
\r
789 multipleSeparator: " "*/
\r
791 formatItem: function(row, i, max, value) {
\r
792 return row[1] + " (" + row[2] + ")";
\r
795 formatResult: function(row, i, max, value){
\r
800 $("#interestingTagAdd").click(function(){handlePickedTag(this,'good');});
\r
801 $("#ignoredTagAdd").click(function(){handlePickedTag(this,'bad');});
\r
806 Hilite={elementid:"content",exact:true,max_nodes:1000,onload:true,style_name:"hilite",style_name_suffix:true,debug_referrer:""};Hilite.search_engines=[["local","q"],["cnprog\\.","q"],["google\\.","q"],["search\\.yahoo\\.","p"],["search\\.msn\\.","q"],["search\\.live\\.","query"],["search\\.aol\\.","userQuery"],["ask\\.com","q"],["altavista\\.","q"],["feedster\\.","q"],["search\\.lycos\\.","q"],["alltheweb\\.","q"],["technorati\\.com/search/([^\\?/]+)",1],["dogpile\\.com/info\\.dogpl/search/web/([^\\?/]+)",1,true]];Hilite.decodeReferrer=function(d){var g=null;var e=new RegExp("");for(var c=0;c<Hilite.search_engines.length;c++){var f=Hilite.search_engines[c];e.compile("^http://(www\\.)?"+f[0],"i");var b=d.match(e);if(b){var a;if(isNaN(f[1])){a=Hilite.decodeReferrerQS(d,f[1])}else{a=b[f[1]+1]}if(a){a=decodeURIComponent(a);if(f.length>2&&f[2]){a=decodeURIComponent(a)}a=a.replace(/\'|"/g,"");a=a.split(/[\s,\+\.]+/);return a}break}}return null};Hilite.decodeReferrerQS=function(f,d){var b=f.indexOf("?");var c;if(b>=0){var a=new String(f.substring(b+1));b=0;c=0;while((b>=0)&&((c=a.indexOf("=",b))>=0)){var e,g;e=a.substring(b,c);b=a.indexOf("&",c)+1;if(e==d){if(b<=0){return a.substring(c+1)}else{return a.substring(c+1,b-1)}}else{if(b<=0){return null}}}}return null};Hilite.hiliteElement=function(f,e){if(!e||f.childNodes.length==0){return}var c=new Array();for(var b=0;b<e.length;b++){e[b]=e[b].toLowerCase();if(Hilite.exact){c.push("\\b"+e[b]+"\\b")}else{c.push(e[b])}}c=new RegExp(c.join("|"),"i");var a={};for(var b=0;b<e.length;b++){if(Hilite.style_name_suffix){a[e[b]]=Hilite.style_name+(b+1)}else{a[e[b]]=Hilite.style_name}}var d=function(m){var j=c.exec(m.data);if(j){var n=j[0];var i="";var h=m.splitText(j.index);var g=h.splitText(n.length);var l=m.ownerDocument.createElement("SPAN");m.parentNode.replaceChild(l,h);l.className=a[n.toLowerCase()];l.appendChild(h);return l}else{return m}};Hilite.walkElements(f.childNodes[0],1,d)};Hilite.hilite=function(){var a=Hilite.debug_referrer?Hilite.debug_referrer:document.referrer;var b=null;a=Hilite.decodeReferrer(a);if(a&&((Hilite.elementid&&(b=document.getElementById(Hilite.elementid)))||(b=document.body))){Hilite.hiliteElement(b,a)}};Hilite.walkElements=function(d,f,e){var a=/^(script|style|textarea)/i;var c=0;while(d&&f>0){c++;if(c>=Hilite.max_nodes){var b=function(){Hilite.walkElements(d,f,e)};setTimeout(b,50);return}if(d.nodeType==1){if(!a.test(d.tagName)&&d.childNodes.length>0){d=d.childNodes[0];f++;continue}}else{if(d.nodeType==3){d=e(d)}}if(d.nextSibling){d=d.nextSibling}else{while(f>0){d=d.parentNode;f--;if(d.nextSibling){d=d.nextSibling;break}}}}};if(Hilite.onload){if(window.attachEvent){window.attachEvent("onload",Hilite.hilite)}else{if(window.addEventListener){window.addEventListener("load",Hilite.hilite,false)}else{var __onload=window.onload;window.onload=function(){Hilite.hilite();__onload()}}}};
\r
808 var mediaUrl = function(resource){
\r
809 return scriptUrl + 'm/' + osqaSkin + '/' + resource;
\r
813 * jQuery i18n plugin
\r
814 * @requires jQuery v1.1 or later
\r
816 * Examples at: http://recurser.com/articles/2008/02/21/jquery-i18n-translation-plugin/
\r
817 * Dual licensed under the MIT and GPL licenses:
\r
818 * http://www.opensource.org/licenses/mit-license.php
\r
819 * http://www.gnu.org/licenses/gpl.html
\r
821 * Based on 'javascript i18n that almost doesn't suck' by markos
\r
822 * http://markos.gaivo.net/blog/?p=100
\r
825 * Version: 1.0.0 Feb-10-2008
\r
829 * i18n provides a mechanism for translating strings using a jscript dictionary.
\r
835 * i18n property list
\r
841 * Initialise the dictionary and translate nodes
\r
843 * @param property_list i18n_dict : The dictionary to use for translation
\r
845 setDictionary: function(i18n_dict) {
\r
846 i18n_dict = i18n_dict;
\r
851 * The actual translation function. Looks the given string up in the
\r
852 * dictionary and returns the translation if one exists. If a translation
\r
853 * is not found, returns the original word
\r
855 * @param string str : The string to translate
\r
856 * @param property_list params : params for using printf() on the string
\r
857 * @return string : Translated word
\r
860 _: function (str, params) {
\r
862 if (i18n_dict&& i18n_dict[str]) {
\r
863 transl = i18n_dict[str];
\r
865 return this.printf(transl, params);
\r
870 * Change non-ASCII characters to entity representation
\r
872 * @param string str : The string to transform
\r
873 * @return string result : Original string with non-ASCII content converted to entities
\r
876 toEntity: function (str) {
\r
878 for (var i=0;i<str.length; i++) {
\r
879 if (str.charCodeAt(i) > 128)
\r
880 result += "&#"+str.charCodeAt(i)+";";
\r
882 result += str.charAt(i);
\r
890 * @param string str : The string to strip
\r
891 * @return string result : Stripped string
\r
894 stripStr: function(str) {
\r
895 return str.replace(/^\s*/, "").replace(/\s*$/, "");
\r
901 * @param string str : The multi-line string to strip
\r
902 * @return string result : Stripped string
\r
905 stripStrML: function(str) {
\r
906 // Split because m flag doesn't exist before JS1.5 and we need to
\r
907 // strip newlines anyway
\r
908 var parts = str.split('\n');
\r
909 for (var i=0; i<parts.length; i++)
\r
910 parts[i] = stripStr(parts[i]);
\r
912 // Don't join with empty strings, because it "concats" words
\r
914 return stripStr(parts.join(" "));
\r
919 * C-printf like function, which substitutes %s with parameters
\r
920 * given in list. %%s is used to escape %s.
\r
922 * Doesn't work in IE5.0 (splice)
\r
924 * @param string S : string to perform printf on.
\r
925 * @param string L : Array of arguments for printf()
\r
927 printf: function(S, L) {
\r
931 var tS = S.split("%s");
\r
933 for(var i=0; i<L.length; i++) {
\r
934 if (tS[i].lastIndexOf('%') == tS[i].length-1 && i != L.length-1)
\r
935 tS[i] += "s"+tS.splice(i+1,1)[0];
\r
936 nS += tS[i] + L[i];
\r
938 return nS + tS[tS.length-1];
\r
949 'insufficient privilege':'??????????',
\r
950 'cannot pick own answer as best':'??????????????',
\r
951 'anonymous users cannot select favorite questions':'?????????????',
\r
952 'please login':'??????',
\r
953 'anonymous users cannot vote':'????????',
\r
954 '>15 points requried to upvote':'??+15?????????',
\r
955 '>100 points required to downvote':'??+100?????????',
\r
956 'please see': '??',
\r
957 'cannot vote for own posts':'??????????',
\r
958 'daily vote cap exhausted':'????????????????',
\r
959 'cannot revoke old vote':'??????????????',
\r
960 'please confirm offensive':"??????????????????????",
\r
961 'anonymous users cannot flag offensive posts':'???????????',
\r
962 'cannot flag message as offensive twice':'???????',
\r
963 'flag offensive cap exhausted':'?????????????5?
\91??
\92???',
\r
964 'need >15 points to report spam':"??+15??????
\91???
\92?",
\r
965 'confirm delete':"?????/????????",
\r
966 'anonymous users cannot delete/undelete':"???????????????",
\r
967 'post recovered':"?????????????",
\r
968 'post deleted':"????????????",
\r
969 'add comment':'????',
\r
970 'community karma points':'????',
\r
971 'to comment, need':'????',
\r
972 'delete this comment':'?????',
\r
973 'hide comments':"????",
\r
974 'add a comment':"????",
\r
976 'confirm delete comment':"?????????",
\r
979 'click to close':'???????',
\r
980 'loading...':'???...',
\r
981 'tags cannot be empty':'???????',
\r
982 'tablimits info':"??5????????????20????",
\r
983 'content cannot be empty':'???????',
\r
984 'content minchars': '????? {0} ???',
\r
985 'please enter title':'??????',
\r
986 'title minchars':"????? {0} ???",
\r
993 'preformatted text':'??',
\r
995 'numbered list':'??????',
\r
996 'bulleted list':'??????',
\r
998 'horizontal bar':'???',
\r
1001 'enter image url':'<b>??????</b></p><p>???<br />http://www.example.com/image.jpg \"????\"',
\r
1002 'enter url':'<b>??Web??</b></p><p>???<br />http://www.cnprog.com/ \"????\"</p>"',
\r
1003 'upload image':'?????????'
\r
1007 'need >15 points to report spam':'need >15 points to report spam ',
\r
1008 '>15 points requried to upvote':'>15 points required to upvote ',
\r
1009 'tags cannot be empty':'please enter at least one tag',
\r
1010 'anonymous users cannot vote':'sorry, anonymous users cannot vote ',
\r
1011 'anonymous users cannot select favorite questions':'sorry, anonymous users cannot select favorite questions ',
\r
1012 'to comment, need': '(to comment other people\'s posts, karma ',
\r
1013 'please see':'please see ',
\r
1014 'community karma points':' or more is necessary) - ',
\r
1015 'upload image':'Upload image:',
\r
1016 'enter image url':'enter URL of the image, e.g. http://www.example.com/image.jpg \"image title\"',
\r
1017 'enter url':'enter Web address, e.g. http://www.example.com \"page title\"',
\r
1018 'daily vote cap exhausted':'sorry, you\'ve used up todays vote cap',
\r
1019 'cannot pick own answer as best':'sorry, you cannot accept your own answer',
\r
1020 'cannot revoke old vote':'sorry, older votes cannot be revoked',
\r
1021 'please confirm offensive':'are you sure this post is offensive, contains spam, advertising, malicious remarks, etc.?',
\r
1022 'flag offensive cap exhausted':'sorry, you\'ve used up todays cap of flagging offensive messages ',
\r
1023 'confirm delete':'are you sure you want to delete this?',
\r
1024 'anonymous users cannot delete/undelete':'sorry, anonymous users cannot delete or undelete posts',
\r
1025 'post recovered':'your post is now restored!',
\r
1026 'post deleted':'your post has been deleted',
\r
1027 'confirm delete comment':'do you really want to delete this comment?',
\r
1028 'can write':'have ',
\r
1029 'tablimits info':'up to 5 tags, no more than 20 characters each',
\r
1030 'content minchars': 'please enter more than {0} characters',
\r
1031 'title minchars':"please enter at least {0} characters",
\r
1032 'characters':'characters left',
\r
1033 'cannot vote for own posts':'sorry, you cannot vote for your own posts',
\r
1034 'cannot flag message as offensive twice':'cannot flag message as offensive twice ',
\r
1035 '>100 points required to downvote':'>100 points required to downvote '
\r
1039 'insufficient privilege':'privilegio insuficiente',
\r
1040 'cannot pick own answer as best':'no puede escoger su propia respuesta como la mejor',
\r
1041 'anonymous users cannot select favorite questions':'usuarios anonimos no pueden seleccionar',
\r
1042 'please login':'por favor inicie sesión',
\r
1043 'anonymous users cannot vote':'usuarios anónimos no pueden votar',
\r
1044 '>15 points requried to upvote': '>15 puntos requeridos para votar positivamente',
\r
1045 '>100 points required to downvote':'>100 puntos requeridos para votar negativamente',
\r
1046 'please see': 'por favor vea',
\r
1047 'cannot vote for own posts':'no se puede votar por sus propias publicaciones',
\r
1048 'daily vote cap exhausted':'cuota de votos diarios excedida',
\r
1049 'cannot revoke old vote':'no puede revocar un voto viejo',
\r
1050 'please confirm offensive':"por favor confirme ofensiva",
\r
1051 'anonymous users cannot flag offensive posts':'usuarios anónimos no pueden marcar publicaciones como ofensivas',
\r
1052 'cannot flag message as offensive twice':'no puede marcar mensaje como ofensivo dos veces',
\r
1053 'flag offensive cap exhausted':'cuota para marcar ofensivas ha sido excedida',
\r
1054 'need >15 points to report spam':"necesita >15 puntos para reportar spam",
\r
1055 'confirm delete':"¿Está seguro que desea borrar esto?",
\r
1056 'anonymous users cannot delete/undelete':"usuarios anónimos no pueden borrar o recuperar publicaciones",
\r
1057 'post recovered':"publicación recuperada",
\r
1058 'post deleted':"publicación borrada?",
\r
1059 'add comment':'agregar comentario',
\r
1060 'community karma points':'reputación comunitaria',
\r
1061 'to comment, need':'para comentar, necesita reputación',
\r
1062 'delete this comment':'borrar este comentario',
\r
1063 'hide comments':"ocultar comentarios",
\r
1064 'add a comment':"agregar comentarios",
\r
1065 'comments':"comentarios",
\r
1066 'confirm delete comment':"¿Realmente desea borrar este comentario?",
\r
1067 'characters':'caracteres faltantes',
\r
1068 'can write':'tiene ',
\r
1069 'click to close':'haga click para cerrar',
\r
1070 'loading...':'cargando...',
\r
1071 'tags cannot be empty':'las etiquetas no pueden estar vacías',
\r
1072 'tablimits info':"hasta 5 etiquetas de no mas de 20 caracteres cada una",
\r
1073 'content cannot be empty':'el contenido no puede estar vacío',
\r
1074 'content minchars': 'por favor introduzca mas de {0} caracteres',
\r
1075 'please enter title':'por favor ingrese un título',
\r
1076 'title minchars':"por favor introduzca al menos {0} caracteres",
\r
1077 'delete':'borrar',
\r
1078 'undelete': 'recuperar',
\r
1079 'bold': 'negrita',
\r
1080 'italic':'cursiva',
\r
1083 'preformatted text':'texto preformateado',
\r
1085 'numbered list':'lista numerada',
\r
1086 'bulleted list':'lista no numerada',
\r
1088 'horizontal bar':'barra horizontal',
\r
1089 'undo':'deshacer',
\r
1091 'enter image url':'introduzca la URL de la imagen, por ejemplo?<br />http://www.example.com/image.jpg \"titulo de imagen\"',
\r
1092 'enter url':'introduzca direcciones web, ejemplo?<br />http://www.cnprog.com/ \"titulo del enlace\"</p>"',
\r
1093 'upload image':'cargar imagen?',
\r
1094 'questions/' : 'preguntas/',
\r
1095 'vote/' : 'votar/'
\r
1104 var i18n_dict = i18n[i18nLang];
\r
1107 jQuery TextAreaResizer plugin
\r
1108 Created on 17th January 2008 by Ryan O'Dell
\r
1110 */(function($){var textarea,staticOffset;var iLastMousePos=0;var iMin=32;var grip;$.fn.TextAreaResizer=function(){return this.each(function(){textarea=$(this).addClass('processed'),staticOffset=null;$(this).wrap('<div class="resizable-textarea"><span></span></div>').parent().append($('<div class="grippie"></div>').bind("mousedown",{el:this},startDrag));var grippie=$('div.grippie',$(this).parent())[0];grippie.style.marginRight=(grippie.offsetWidth-$(this)[0].offsetWidth)+'px'})};function startDrag(e){textarea=$(e.data.el);textarea.blur();iLastMousePos=mousePosition(e).y;staticOffset=textarea.height()-iLastMousePos;textarea.css('opacity',0.25);$(document).mousemove(performDrag).mouseup(endDrag);return false}function performDrag(e){var iThisMousePos=mousePosition(e).y;var iMousePos=staticOffset+iThisMousePos;if(iLastMousePos>=(iThisMousePos)){iMousePos-=5}iLastMousePos=iThisMousePos;iMousePos=Math.max(iMin,iMousePos);textarea.height(iMousePos+'px');if(iMousePos<iMin){endDrag(e)}return false}function endDrag(e){$(document).unbind('mousemove',performDrag).unbind('mouseup',endDrag);textarea.css('opacity',1);textarea.focus();textarea=null;staticOffset=null;iLastMousePos=0}function mousePosition(e){return{x:e.clientX+document.documentElement.scrollLeft,y:e.clientY+document.documentElement.scrollTop}}})(jQuery);
\r
1112 * Autocomplete - jQuery plugin 1.0.3
\r
1114 * Copyright (c) 2007 Dylan Verheul, Dan G. Switzer, Anjesh Tuladhar, Jörn Zaefferer
\r
1116 * Dual licensed under the MIT and GPL licenses:
\r
1117 * http://www.opensource.org/licenses/mit-license.php
\r
1118 * http://www.gnu.org/licenses/gpl.html
\r
1121 (function(a){a.fn.extend({autocomplete:function(b,c){var d=typeof b=="string";c=a.extend({},a.Autocompleter.defaults,{url:d?b:null,data:d?null:b,delay:d?a.Autocompleter.defaults.delay:10,max:c&&!c.scroll?10:150},c);c.highlight=c.highlight||function(e){return e};c.formatMatch=c.formatMatch||c.formatItem;return this.each(function(){new a.Autocompleter(this,c)})},result:function(b){return this.bind("result",b)},search:function(b){return this.trigger("search",[b])},flushCache:function(){return this.trigger("flushCache")},setOptions:function(b){return this.trigger("setOptions",[b])},unautocomplete:function(){return this.trigger("unautocomplete")}});a.Autocompleter=function(l,g){var c={UP:38,DOWN:40,DEL:46,TAB:9,RETURN:13,ESC:27,COMMA:188,PAGEUP:33,PAGEDOWN:34,BACKSPACE:8};var b=a(l).attr("autocomplete","off").addClass(g.inputClass);var j;var p="";var m=a.Autocompleter.Cache(g);var e=0;var u;var x={mouseDownOnSelect:false};var r=a.Autocompleter.Select(g,l,d,x);var w;a.browser.opera&&a(l.form).bind("submit.autocomplete",function(){if(w){w=false;return false}});b.bind((a.browser.opera?"keypress":"keydown")+".autocomplete",function(y){u=y.keyCode;switch(y.keyCode){case c.UP:y.preventDefault();if(r.visible()){r.prev()}else{t(0,true)}break;case c.DOWN:y.preventDefault();if(r.visible()){r.next()}else{t(0,true)}break;case c.PAGEUP:y.preventDefault();if(r.visible()){r.pageUp()}else{t(0,true)}break;case c.PAGEDOWN:y.preventDefault();if(r.visible()){r.pageDown()}else{t(0,true)}break;case g.multiple&&a.trim(g.multipleSeparator)==","&&c.COMMA:case c.TAB:case c.RETURN:if(d()){y.preventDefault();w=true;return false}break;case c.ESC:r.hide();break;default:clearTimeout(j);j=setTimeout(t,g.delay);break}}).focus(function(){e++}).blur(function(){e=0;if(!x.mouseDownOnSelect){s()}}).click(function(){if(e++>1&&!r.visible()){t(0,true)}}).bind("search",function(){var y=(arguments.length>1)?arguments[1]:null;function z(D,C){var A;if(C&&C.length){for(var B=0;B<C.length;B++){if(C[B].result.toLowerCase()==D.toLowerCase()){A=C[B];break}}}if(typeof y=="function"){y(A)}else{b.trigger("result",A&&[A.data,A.value])}}a.each(h(b.val()),function(A,B){f(B,z,z)})}).bind("flushCache",function(){m.flush()}).bind("setOptions",function(){a.extend(g,arguments[1]);if("data" in arguments[1]){m.populate()}}).bind("unautocomplete",function(){r.unbind();b.unbind();a(l.form).unbind(".autocomplete")});function d(){var z=r.selected();if(!z){return false}var y=z.result;p=y;if(g.multiple){var A=h(b.val());if(A.length>1){y=A.slice(0,A.length-1).join(g.multipleSeparator)+g.multipleSeparator+y}y+=g.multipleSeparator}b.val(y);v();b.trigger("result",[z.data,z.value]);return true}function t(A,z){if(u==c.DEL){r.hide();return}var y=b.val();if(!z&&y==p){return}p=y;y=i(y);if(y.length>=g.minChars){b.addClass(g.loadingClass);if(!g.matchCase){y=y.toLowerCase()}f(y,k,v)}else{n();r.hide()}}function h(z){if(!z){return[""]}var A=z.split(g.multipleSeparator);var y=[];a.each(A,function(B,C){if(a.trim(C)){y[B]=a.trim(C)}});return y}function i(y){if(!g.multiple){return y}var z=h(y);return z[z.length-1]}function q(y,z){if(g.autoFill&&(i(b.val()).toLowerCase()==y.toLowerCase())&&u!=c.BACKSPACE){b.val(b.val()+z.substring(i(p).length));a.Autocompleter.Selection(l,p.length,p.length+z.length)}}function s(){clearTimeout(j);j=setTimeout(v,200)}function v(){var y=r.visible();r.hide();clearTimeout(j);n();if(g.mustMatch){b.search(function(z){if(!z){if(g.multiple){var A=h(b.val()).slice(0,-1);b.val(A.join(g.multipleSeparator)+(A.length?g.multipleSeparator:""))}else{b.val("")}}})}if(y){a.Autocompleter.Selection(l,l.value.length,l.value.length)}}function k(z,y){if(y&&y.length&&e){n();r.display(y,z);q(z,y[0].value);r.show()}else{v()}}function f(z,B,y){if(!g.matchCase){z=z.toLowerCase()}var A=m.load(z);if(A&&A.length){B(z,A)}else{if((typeof g.url=="string")&&(g.url.length>0)){var C={timestamp:+new Date()};a.each(g.extraParams,function(D,E){C[D]=typeof E=="function"?E():E});a.ajax({mode:"abort",port:"autocomplete"+l.name,dataType:g.dataType,url:g.url,data:a.extend({q:i(z),limit:g.max},C),success:function(E){var D=g.parse&&g.parse(E)||o(E);m.add(z,D);B(z,D)}})}else{r.emptyList();y(z)}}}function o(B){var y=[];var A=B.split("\n");for(var z=0;z<A.length;z++){var C=a.trim(A[z]);if(C){C=C.split("|");y[y.length]={data:C,value:C[0],result:g.formatResult&&g.formatResult(C,C[0])||C[0]}}}return y}function n(){b.removeClass(g.loadingClass)}};a.Autocompleter.defaults={inputClass:"ac_input",resultsClass:"ac_results",loadingClass:"ac_loading",minChars:1,delay:400,matchCase:false,matchSubset:true,matchContains:false,cacheLength:10,max:100,mustMatch:false,extraParams:{},selectFirst:true,formatItem:function(b){return b[0]},formatMatch:null,autoFill:false,width:0,multiple:false,multipleSeparator:", ",highlight:function(c,b){return c.replace(new RegExp("(?![^&;]+;)(?!<[^<>]*)("+b.replace(/([\^\$\(\)\[\]\{\}\*\.\+\?\|\\])/gi,"\\$1")+")(?![^<>]*>)(?![^&;]+;)","gi"),"<strong>$1</strong>")},scroll:true,scrollHeight:180};a.Autocompleter.Cache=function(c){var f={};var d=0;function h(l,k){if(!c.matchCase){l=l.toLowerCase()}var j=l.indexOf(k);if(j==-1){return false}return j==0||c.matchContains}function g(j,i){if(d>c.cacheLength){b()}if(!f[j]){d++}f[j]=i}function e(){if(!c.data){return false}var k={},j=0;if(!c.url){c.cacheLength=1}k[""]=[];for(var m=0,l=c.data.length;m<l;m++){var p=c.data[m];p=(typeof p=="string")?[p]:p;var o=c.formatMatch(p,m+1,c.data.length);if(o===false){continue}var n=o.charAt(0).toLowerCase();if(!k[n]){k[n]=[]}var q={value:o,data:p,result:c.formatResult&&c.formatResult(p)||o};k[n].push(q);if(j++<c.max){k[""].push(q)}}a.each(k,function(r,s){c.cacheLength++;g(r,s)})}setTimeout(e,25);function b(){f={};d=0}return{flush:b,add:g,populate:e,load:function(n){if(!c.cacheLength||!d){return null}if(!c.url&&c.matchContains){var m=[];for(var j in f){if(j.length>0){var o=f[j];a.each(o,function(p,k){if(h(k.value,n)){m.push(k)}})}}return m}else{if(f[n]){return f[n]}else{if(c.matchSubset){for(var l=n.length-1;l>=c.minChars;l--){var o=f[n.substr(0,l)];if(o){var m=[];a.each(o,function(p,k){if(h(k.value,n)){m[m.length]=k}});return m}}}}}return null}}};a.Autocompleter.Select=function(e,j,l,p){var i={ACTIVE:"ac_over"};var k,f=-1,r,m="",s=true,c,o;function n(){if(!s){return}c=a("<div/>").hide().addClass(e.resultsClass).css("position","absolute").appendTo(document.body);o=a("<ul/>").appendTo(c).mouseover(function(t){if(q(t).nodeName&&q(t).nodeName.toUpperCase()=="LI"){f=a("li",o).removeClass(i.ACTIVE).index(q(t));a(q(t)).addClass(i.ACTIVE)}}).click(function(t){a(q(t)).addClass(i.ACTIVE);l();j.focus();return false}).mousedown(function(){p.mouseDownOnSelect=true}).mouseup(function(){p.mouseDownOnSelect=false});if(e.width>0){c.css("width",e.width)}s=false}function q(u){var t=u.target;while(t&&t.tagName!="LI"){t=t.parentNode}if(!t){return[]}return t}function h(t){k.slice(f,f+1).removeClass(i.ACTIVE);g(t);var v=k.slice(f,f+1).addClass(i.ACTIVE);if(e.scroll){var u=0;k.slice(0,f).each(function(){u+=this.offsetHeight});if((u+v[0].offsetHeight-o.scrollTop())>o[0].clientHeight){o.scrollTop(u+v[0].offsetHeight-o.innerHeight())}else{if(u<o.scrollTop()){o.scrollTop(u)}}}}function g(t){f+=t;if(f<0){f=k.size()-1}else{if(f>=k.size()){f=0}}}function b(t){return e.max&&e.max<t?e.max:t}function d(){o.empty();var u=b(r.length);for(var v=0;v<u;v++){if(!r[v]){continue}var w=e.formatItem(r[v].data,v+1,u,r[v].value,m);if(w===false){continue}var t=a("<li/>").html(e.highlight(w,m)).addClass(v%2==0?"ac_even":"ac_odd").appendTo(o)[0];a.data(t,"ac_data",r[v])}k=o.find("li");if(e.selectFirst){k.slice(0,1).addClass(i.ACTIVE);f=0}if(a.fn.bgiframe){o.bgiframe()}}return{display:function(u,t){n();r=u;m=t;d()},next:function(){h(1)},prev:function(){h(-1)},pageUp:function(){if(f!=0&&f-8<0){h(-f)}else{h(-8)}},pageDown:function(){if(f!=k.size()-1&&f+8>k.size()){h(k.size()-1-f)}else{h(8)}},hide:function(){c&&c.hide();k&&k.removeClass(i.ACTIVE);f=-1},visible:function(){return c&&c.is(":visible")},current:function(){return this.visible()&&(k.filter("."+i.ACTIVE)[0]||e.selectFirst&&k[0])},show:function(){var v=a(j).offset();c.css({width:typeof e.width=="string"||e.width>0?e.width:a(j).width(),top:v.top+j.offsetHeight,left:v.left}).show();if(e.scroll){o.scrollTop(0);o.css({maxHeight:e.scrollHeight,overflow:"auto"});if(a.browser.msie&&typeof document.body.style.maxHeight==="undefined"){var t=0;k.each(function(){t+=this.offsetHeight});var u=t>e.scrollHeight;o.css("height",u?e.scrollHeight:t);if(!u){k.width(o.width()-parseInt(k.css("padding-left"))-parseInt(k.css("padding-right")))}}}},selected:function(){var t=k&&k.filter("."+i.ACTIVE).removeClass(i.ACTIVE);return t&&t.length&&a.data(t[0],"ac_data")},emptyList:function(){o&&o.empty()},unbind:function(){c&&c.remove()}}};a.Autocompleter.Selection=function(d,e,c){if(d.setSelectionRange){d.setSelectionRange(e,c)}else{if(d.createTextRange){var b=d.createTextRange();b.collapse(true);b.moveStart("character",e);b.moveEnd("character",c);b.select()}else{if(d.selectionStart){d.selectionStart=e;d.selectionEnd=c}}}d.focus()}})(jQuery);
\r
1123 var notify = function() {
\r
1124 var visible = false;
\r
1126 show: function(html) {
\r
1128 $("body").css("margin-top", "2.2em");
\r
1129 $(".notify span").html(html);
\r
1131 $(".notify").fadeIn("slow");
\r
1134 close: function(doPostback) {
\r
1135 $(".notify").fadeOut("fast");
\r
1136 $("body").css("margin-top", "0");
\r
1139 isVisible: function() { return visible; }
\r
1144 * jQuery outside events - v1.1 - 3/16/2010
\r
1145 * http://benalman.com/projects/jquery-outside-events-plugin/
\r
1147 * Copyright (c) 2010 "Cowboy" Ben Alman
\r
1148 * Dual licensed under the MIT and GPL licenses.
\r
1149 * http://benalman.com/about/license/
\r
1151 (function($,c,b){$.map("click dblclick mousemove mousedown mouseup mouseover mouseout change select submit keydown keypress keyup".split(" "),function(d){a(d)});a("focusin","focus"+b);a("focusout","blur"+b);$.addOutsideEvent=a;function a(g,e){e=e||g+b;var d=$(),h=g+"."+e+"-special-event";$.event.special[e]={setup:function(){d=d.add(this);if(d.length===1){$(c).bind(h,f)}},teardown:function(){d=d.not(this);if(d.length===0){$(c).unbind(h)}},add:function(i){var j=i.handler;i.handler=function(l,k){l.target=k;j.apply(this,arguments)}}};function f(i){$(d).each(function(){var j=$(this);if(this!==i.target&&!j.has(i.target).length){j.triggerHandler(e,[i.target])}})}}})(jQuery,document,"outside");
\r
1153 $(document).ready( function(){
\r
1154 pickedTags().init();
\r
1156 $('input#bnewaccount').click(function() {
\r
1157 $('#bnewaccount').disabled=true;
\r
1161 function yourWorkWillBeLost(e) {
\r
1162 if(browserTester('chrome')) {
\r
1163 return "Are you sure you want to leave? Your work will be lost.";
\r
1164 } else if(browserTester('safari')) {
\r
1165 return "Are you sure you want to leave? Your work will be lost.";
\r
1167 if(!e) e = window.event;
\r
1168 e.cancelBubble = true;
\r
1169 e.returnValue = 'If you leave, your work will be lost.';
\r
1171 if (e.stopPropagation) {
\r
1172 e.stopPropagation();
\r
1173 e.preventDefault();
\r
1179 function browserTester(browserString) {
\r
1180 return navigator.userAgent.toLowerCase().indexOf(browserString) > -1;
\r
1183 // Add missing IE functionality
\r
1184 if (!window.addEventListener) {
\r
1185 if (window.attachEvent) {
\r
1186 window.addEventListener = function (type, listener, useCapture) {
\r
1187 window.attachEvent('on' + type, listener);
\r
1189 window.removeEventListener = function (type, listener, useCapture) {
\r
1190 window.detachEvent('on' + type, listener);
\r
1193 window.addEventListener = function (type, listener, useCapture) {
\r
1194 window['on' + type] = listener;
\r
1196 window.removeEventListener = function (type, listener, useCapture) {
\r
1197 window['on' + type] = null;
\r