var TracksForm = { toggle: function(toggleLinkId, formContainerId, formId, hideLinkText, hideLinkTitle, showLinkText, showLinkTitle) { form=$('#'+formContainerId) form.toggle(); toggleLink = $('#'+toggleLinkId); if (!form.is(':visible')) { toggleLink.text(showLinkText).attr('title', showLinkTitle); } else { toggleLink.text(hideLinkText).attr('title', hideLinkTitle); $('#'+formId+' input:text:first').focus(); } toggleLink.parent().toggleClass('hide_form'); }, set_project_name: function (name) { $('input#todo_project_name').val(name); }, set_context_name_and_default_context_name: function (name) { $('input#todo_context_name').val(name); $('input[name=default_context_name]').val(name); }, set_project_name_and_default_project_name: function (name) { $('#default_project_name_id').val(name); $('input#todo_project_name').val(); $('#project_name').html(name); }, set_tag_list: function (name) { $('input#tag_list').val(name); }, setup_behavior: function() { /* toggle new todo form for single todo */ $('#toggle_action_new').click(function(){ if ($("#todo_multi_add").is(':visible')) { /* hide multi next action form first */ $('#todo_new_action').show(); $('#todo_multi_add').hide(); $('a#toggle_multi').text("Add multiple next actions"); } TracksForm.toggle('toggle_action_new', 'todo_new_action', 'todo-form-new-action', '« Hide form', 'Hide next action form', 'Add a next action »', 'Add a next action'); }); /* toggle new todo form for multi edit */ $('#toggle_multi').click(function(){ if ($("#todo_multi_add").is(':visible')) { $('#todo_new_action').show(); $('#todo_multi_add').hide(); $('a#toggle_multi').text("Add multiple next actions"); } else { $('#todo_new_action').hide(); $('#todo_multi_add').show(); $('a#toggle_multi').text("Add single next action"); $('a#toggle_action_new').text('« Hide form'); } }); /* add behavior to clear the date both buttons for show_from and due */ $(".date_clear").live('click', function() { $(this).prev().val(''); }); /* behavior for delete icon */ $('.item-container a.delete_icon').live('click', function(evt){ evt.preventDefault(); params = {}; if(typeof(TAG_NAME) !== 'undefined'){ params._tag_name = TAG_NAME; } if(confirm("Are you sure that you want to "+this.title+"?")){ itemContainer = $(this).parents(".item-container"); itemContainer.block({ message: null }); params._method = 'delete'; $.post(this.href, params, function(){ itemContainer.unblock(); }, 'script'); } }); /* behavior for edit icon */ $('.item-container a.edit_icon').live('click', function(evt){ evt.preventDefault(); params = {}; if(typeof(TAG_NAME) !== 'undefined'){ params._tag_name = TAG_NAME; } itemContainer = $(this).parents(".item-container"); $(this).effect('pulsate', { times: 1 }, 800); $.get(this.href, params, function(){ }, 'script'); }); } } var TracksPages = { show_errors: function (html) { $('div#error_status').html(html); $('div#error_status').show(); }, hide_errors: function() { $('div#error_status').hide(); }, setup_nifty_corners: function() { Nifty("div#recurring_new_container","normal"); Nifty("div#context_new_container","normal"); Nifty("div#feedlegend","normal"); Nifty("div#feedicons-project","normal"); Nifty("div#feedicons-context","normal"); Nifty("div#todo_new_action_container","normal"); Nifty("div#project_new_project_container","normal"); }, setup_behavior: function () { /* main menu */ $('ul.sf-menu').superfish({ delay: 250, animation: { opacity:'show', height:'show' }, autoArrows: false, dropShadows: false, speed: 'fast' }); /* context menu */ $('ul.sf-item-menu').superfish({ delay: 100, animation: { opacity:'show', height:'show' }, autoArrows: false, dropShadows: false, speed: 'fast', onBeforeShow: function() { /* highlight todo */ $(this.parent().parent().parent()).addClass("sf-item-selected"); }, onHide: function() { /* remove hightlight from todo */ $(this.parent().parent().parent()).removeClass("sf-item-selected"); } }); /* for toggle notes link in mininav */ $("#toggle-notes-nav").click(function () { $(".todo_notes").toggle(); }); /* fade flashes and alerts in automatically */ $(".alert").fadeOut(8000); /* for edit project form and edit todo form * TODO: refactor to separate calls from project and todo */ $('.edit-form a.negative').live('click', function(){ $(this).parents('.edit-form').fadeOut(200, function () { $(this).parents('.list').find('.project').fadeIn(500); $(this).parents('.container').find('.item-show').fadeIn(500); }) }); } } var TodoItemsContainer = { // public ensureVisibleWithEffectAppear: function(elemId){ $('#'+elemId).fadeIn(400); }, expandNextActionListing: function(itemsElem, skipAnimation) { itemsElem = $(itemsElem); if(skipAnimation == true) { itemsElem.show(); } else { itemsElem.show('blind', 400); } TodoItems.showContainer(itemsElem.parentNode); }, collapseNextActionListing: function(itemsElem, skipAnimation) { itemsElem = $(itemsElem); if(skipAnimation == true) { itemsElem.hide(); } else { itemsElem.hide('blind', 400); } TodoItems.hideContainer(itemsElem.parentNode); }, ensureContainerHeight: function(itemsElem) { $(itemsElem).css({ height: '', overflow: '' }); }, expandNextActionListingByContext: function(itemsElemId, skipAnimation){ TodoItems.expandNextActionListing($('#'+itemsElemId).get(), skipAnimation); }, setup_container_toggles: function(){ // bind handlers $('.container_toggle').click(function(evt){ toggle_target = $(this.parentNode.parentNode).find('.toggle_target'); if(toggle_target.is(':visible')){ // hide it imgSrc = $(this).find('img').attr('src'); $(this).find('img').attr('src', imgSrc.replace('collapse', 'expand')); $.cookie(TodoItemsContainer.buildCookieName(this.parentNode.parentNode), true); } else { // show it imgSrc = $(this).find('img').attr('src'); $(this).find('img').attr('src', imgSrc.replace('expand', 'collapse')); $.cookie(TodoItemsContainer.buildCookieName(this.parentNode.parentNode), null); } toggle_target.toggle('blind'); }); // set to cookied state $('.container.context').each(function(){ if($.cookie(TodoItemsContainer.buildCookieName(this))=="true"){ imgSrc = $(this).find('.container_toggle img').attr('src'); if (imgSrc) { $(this).find('.container_toggle img').attr('src', imgSrc.replace('collapse', 'expand')); $(this).find('.toggle_target').hide(); } } }); }, // private buildCookieName: function(containerElem) { tracks_login = $.cookie('tracks_login'); return 'tracks_'+tracks_login+'_context_' + containerElem.id + '_collapsed'; }, showContainer: function(containerElem) { imgSrc = $(containerElem).find('.container_toggle img').attr('src'); $(containerElem).find('.container_toggle img').attr('src', imgSrc.replace('expand', 'collapse')); }, hideContainer: function (containerElem) { imgSrc = $(containerElem).find('.container_toggle img').attr('src'); $(containerElem).find('.container_toggle img').attr('src', imgSrc.replace('collapse', 'expand')); } } var TodoItems = { setup_behavior: function() { /* show the notes of a todo */ $(".show_notes").live('click', function () { $(this).next().toggle("fast"); return false; }); $(".show_successors").live('click', function () { $(this).next().toggle("fast"); return false; }); /* set behavior for star icon */ $(".item-container a.star_item").live('click', function (ev){ $.post(this.href, { _method: 'put' }, null, 'script'); return false; }); /* set behavior for toggle checkboxes */ $(".item-container input.item-checkbox").live('click', function(ev){ params = { _method: 'put' }; if(typeof(TAG_NAME) !== 'undefined') params._tag_name = TAG_NAME; $.post(this.value, params, null, 'script'); }); /* set behavior for edit icon */ $(".item-container a.edit_item").live('click', function (ev){ get_with_ajax_and_block_element(this.href, $(this).parents(".item-container")); return false; }); } } var ProjectListPage = { update_state_count: function(state, count) { $('#'+state+'-projects-count').html(count); }, update_all_states_count: function (active_count, hidden_count, completed_count) { $(["active", "hidden", "completed"]).each(function() { ProjectListPage.update_state_count(this, eval(this+'_count')); }); }, show_or_hide_all_state_containers: function (show_active, show_hidden, show_completed) { $(["active", "hidden", "completed"]).each(function() { ProjectListPage.set_state_container_visibility(this, eval('show_'+this)); }); }, set_state_container_visibility: function (state, set_visible) { if (set_visible) { $('#list-'+state+'-projects-container').slideDown("fast"); } else { $('#list-'+state+'-projects-container').slideUp("fast"); } }, save_project_name: function(value, settings){ project_id = $(this).parents('.container').children('div').get(0).id.split('_')[2]; highlight = function(){ $('h2#project_name').effect('highlight', {}, 500); }; $.post(relative_to_root('projects/update/'+project_id), { 'project[name]': value, 'update_project_name': 'true' }, highlight, 'script'); return(value); }, setup_behavior: function() { /* in-place edit of project name */ $('h2#project_name').editable(ProjectListPage.save_project_name, { style: 'padding:0px', submit: "OK", cancel: "CANCEL" }); /* alphabetize project list */ $('.alphabetize_link').live('click', function(evt){ if(confirm('Are you sure that you want to sort these projects alphabetically? This will replace the existing sort order.')){ post_with_ajax_and_block_element(this.href, $(this).parents('.alpha_sort')); } return false; }); /* sort by number of actions */ $('.actionize_link').click(function(evt){ if(confirm('Are you sure that you want to sort these projects by the number of tasks? This will replace the existing sort order.')){ post_with_ajax_and_block_element(this.href, $(this).parents('.tasks_sort')); } return false; }); /* delete button to delete a project from the list */ $('a.delete_project_button').live('click', function(evt){ if(confirm("Are you sure that you want to "+this.title+"?")){ delete_with_ajax_and_block_element(this.href, $(this).parents('.project')); } return false; }); /* set behavior for edit project settings link in both projects list page and project page */ $("a.project_edit_settings").live('click', function (evt) { get_with_ajax_and_block_element(this.href, $(this).parent().parent()); return false; }); /* submit project form after edit */ $("form.edit-project-form button.positive").live('click', function (ev) { submit_with_ajax_and_block_element('form.edit-project-form', $(this)); return false; }); /* submit project form after entering new project */ $("form#project_form button.positive").live('click', function (ev) { submit_with_ajax_and_block_element('form.#project_form', $(this)); return false; }); /* toggle new project form */ $('#toggle_project_new').click(function(evt){ TracksForm.toggle('toggle_project_new', 'project_new', 'project-form', '« Hide form', 'Hide new project form', 'Create a new project »', 'Add a project'); }); /* make the three lists of project sortable */ $(['active', 'hidden', 'completed']).each(function() { $("#list-"+this+"-projects").sortable({ handle: '.handle', update: update_order }); }); } } var ContextListPage = { update_state_count: function(state, count) { $('#'+state+'-contexts-count').html(count); }, update_all_states_count: function (active_count, hidden_count, completed_count) { $(["active", "hidden"]).each(function() { ContextListPage.update_state_count(this, eval(this+'_count')); }); }, show_or_hide_all_state_containers: function (show_active, show_hidden, show_completed) { $(["active", "hidden"]).each(function() { ContextListPage.set_state_container_visibility(this, eval('show_'+this)); }); }, set_state_container_visibility: function (state, set_visible) { if (set_visible) { $('#list-'+state+'-contexts-container').slideDown("fast"); } else { $('#list-'+state+'-contexts-container').slideUp("fast"); } }, save_context_name: function(value, settings) { context_id = $(this).parents('.container.context').get(0).id.split('c')[1]; highlight = function(){ $('div.context span#context_name').effect('highlight', {}, 500); }; $.post(relative_to_root('contexts/update/'+context_id), { 'context[name]': value }, highlight); return value; }, setup_behavior: function() { /* in place edit of context name */ $('div.context span#context_name').editable(ContextListPage.save_context_name, { style: 'padding:0px', submit: "OK", cancel: "CANCEL" }); /* delete a context using the x button */ $('a.delete_context_button').live('click', function(evt){ if(confirm("Are you sure that you want to "+this.title+"? Be aware that this will also delete all (repeating) actions in this context!")){ delete_with_ajax_and_block_element(this.href, $(this).parents('.context')); } return false; }); /* set behavior for edit context settings link in projects list page and project page */ $("a.context_edit_settings").live('click', function (ev) { get_with_ajax_and_block_element(this.href, $(this).parent().parent()); return false; }); /* submit form when editing a context */ $("form.edit-context-form button.positive").live('click', function (ev) { submit_with_ajax_and_block_element('form.edit-context-form', $(this)); return false; }); /* submit form for new context in sidebar */ $("form#context-form button.positive").live('click', function (ev) { submit_with_ajax_and_block_element('form.#context-form', $(this)); return false; }); /* Contexts behavior */ $('#toggle_context_new').click(function(evt){ TracksForm.toggle('toggle_context_new', 'context_new', 'context-form', '« Hide form', 'Hide new context form', 'Create a new context »', 'Add a context'); }); /* make the two state lists of context sortable */ $(['active', 'hidden']).each(function() { $("#list-contexts-"+this).sortable({ handle: '.handle', update: update_order }) }); } } var IntegrationsPage = { setup_behavior: function() { $('#applescript1-contexts').live('change', function(){ IntegrationsPage.get_script_for_context("#applescript1", "get_applescript1", this.value); }); $('#applescript2-contexts').live('change', function(){ IntegrationsPage.get_script_for_context("#applescript2", "get_applescript2", this.value); }); $('#quicksilver-contexts').live('change', function(){ IntegrationsPage.get_script_for_context("#quicksilver", "get_quicksilver_applescript", this.value) }); }, get_script_for_context: function(element, getter, context){ generic_get_script_for_list(element, "integrations/"+getter, "context_id="+context); } } var FeedsPage = { setup_behavior: function() { /* TODO: blocking of dropdown */ $("#feed-contexts").change(function(){ FeedsPage.get_script_for_context("#feeds-for-context", "get_feeds_for_context", this.value ); }); $("#feed-projects").change(function(){ FeedsPage.get_script_for_project("#feeds-for-project", "get_feeds_for_project", this.value ); }); }, get_script_for_context: function(element, getter, context){ generic_get_script_for_list(element, "feedlist/"+getter, "context_id="+context); }, get_script_for_project: function(element, getter, project){ generic_get_script_for_list(element, "feedlist/"+getter, "project_id="+project); } } var NotesPage = { setup_behavior: function() { /* Add note */ $(".add_note_link a").live('click', function(){ $('#new-note').show(); $('textarea#note_body').val(''); $('textarea#note_body').focus(); }); /* delete button for note */ $('a.delete_note_button').live('click', function(){ if(confirm("Are you sure that you want to "+this.title+"?")){ delete_with_ajax_and_block_element(this.href, $(this).parents('.project_notes')); } return false; }); /* edit button for note */ $('a.note_edit_settings').live('click', function(){ dom_id = this.id.substr(10); $('#'+dom_id).toggle(); $('#edit_'+dom_id).show(); $('#edit_form_'+dom_id+' textarea').focus(); return false; }); /* cancel button when editing a note */ $('.edit-note-form a.negative').live('click', function(){ dom_id = this.id.substr(14); /* dom_id == 'note_XX' on notes page and just 'note' on project page */ if (dom_id == 'note') { $('#new-note').hide(); } else { $('#'+dom_id).toggle(); $('#edit_'+dom_id).hide(); } return false; }); /* update button when editing a note */ $("form.edit-note-form button.positive").live('click', function (ev) { submit_with_ajax_and_block_element('form.edit-note-form', $(this)); return false; }); } } var RecurringTodosPage = { hide_all_recurring: function () { $.each(['daily', 'weekly', 'monthly', 'yearly'], function(){ $('#recurring_'+this).hide(); }); }, hide_all_edit_recurring: function () { $.each(['daily', 'weekly', 'monthly', 'yearly'], function(){ $('#recurring_edit_'+this).hide(); }); }, toggle_overlay: function () { el = document.getElementById("overlay"); el.style.visibility = (el.style.visibility == "visible") ? "hidden" : "visible"; }, setup_behavior: function() { $("#recurring_todo_new_action_cancel").click(function(){ $('#recurring-todo-form-new-action input:text:first').focus(); RecurringTodosPage.hide_all_recurring(); $('#recurring_daily').show(); RecurringTodosPage.toggle_overlay(); }); $("#recurring_todo_edit_action_cancel").live('click', function(){ $('#recurring-todo-form-edit-action input:text:first').focus(); RecurringTodosPage.hide_all_recurring(); $('#recurring_daily').show(); RecurringTodosPage.toggle_overlay(); }); $("#recurring_edit_period input").live('click', function(){ RecurringTodosPage.hide_all_edit_recurring(); $('#recurring_edit_'+this.id.split('_')[5]).show(); }); $("#recurring_period input").live('click', function(){ RecurringTodosPage.hide_all_recurring(); $('#recurring_'+this.id.split('_')[4]).show(); }); } } var SearchPage = { setup_behavior: function() { $('#search-form #search').focus(); } } function generic_get_script_for_list(element, getter, param){ $(element).load(relative_to_root(getter+'?'+param)); } function default_ajax_options(ajax_type, the_url, element_to_block) { return { url: the_url, type: ajax_type, async: true, blocked_elem: element_to_block, dataType: 'script', beforeSend: function() { this.blocked_elem.block({ message: null }); }, complete:function() { this.blocked_elem.unblock(); enable_rich_interaction(); } } } function submit_with_ajax_and_block_element(form, element_to_block) { $(form).ajaxSubmit({ type: 'POST', async: true, blocked_elem: element_to_block, beforeSend: function() { this.blocked_elem.block({ message: null }); }, complete: function() { this.blocked_elem.unblock(); enable_rich_interaction(); } }); } function get_with_ajax_and_block_element(the_url, element_to_block) { $.ajax(default_ajax_options('GET', the_url, element_to_block)); } function post_with_ajax_and_block_element(the_url, element_to_block) { $.ajax(default_ajax_options('POST', the_url, element_to_block)); } function delete_with_ajax_and_block_element(the_url, element_to_block) { $.ajax(default_ajax_options('DELETE', the_url, element_to_block)); } /**************************************** * Unobtrusive jQuery written by Eric Allen ****************************************/ /* Set up authenticity token properly */ $(document).ajaxSend(function(event, request, settings) { if ( settings.type == 'POST' || settings.type == 'post' ) { if(typeof(AUTH_TOKEN) != 'undefined'){ settings.data = (settings.data ? settings.data + "&" : "") + "authenticity_token=" + encodeURIComponent( AUTH_TOKEN ) + "&" + "_source_view=" + encodeURIComponent( SOURCE_VIEW ); } else { settings.data = (settings.data ? settings.data + "&" : "") + "_source_view=" + encodeURIComponent( SOURCE_VIEW ); } request.setRequestHeader("Content-Type", "application/x-www-form-urlencoded"); } request.setRequestHeader("Accept", "text/javascript"); }); $.fn.clearForm = function() { return this.each(function() { var type = this.type, tag = this.tagName.toLowerCase(); if (tag == 'form') return $(':input',this).clearForm(); if (type == 'text' || type == 'password' || tag == 'textarea') this.value = ''; else if (type == 'checkbox' || type == 'radio') this.checked = false; else if (tag == 'select') this.selectedIndex = -1; }); }; function redirect_to(path) { window.location.href = path; } function askIfNewContextProvided(source) { var givenContextName = $('#'+source+'todo_context_name').val(); var contextNames = []; var contextNamesRequest = $.ajax({ url: relative_to_root('contexts.autocomplete'), async: false, dataType: "text", data: "q="+givenContextName, success: function(result){ lines = result.split("\n"); for(var i = 0; i < lines.length; i++){ contextNames.push(lines[i].split("|")[0]); } } }); if (givenContextName.length == 0) return true; // do nothing and depend on rails validation error for (var i = 0; i < contextNames.length; ++i) { if (contextNames[i] == givenContextName) return true; } return confirm('New context "' + givenContextName + '" will be also created. Are you sure?'); } function update_order(event, ui){ container = $(ui.item).parent(); row = $(ui.item).children('.sortable_row'); url = ''; if(row.hasClass('context')) url = relative_to_root('contexts/order'); else if(row.hasClass('project')) url = relative_to_root('projects/order'); else { console.log("Bad sortable list"); return; } $.post(url, container.sortable("serialize"), function(){ row.effect('highlight', {}, 1000) }, 'script'); } /* Unobtrusive jQuery behavior */ function project_defaults(){ if($('body').hasClass('contexts')){ // don't change the context // see ticket #934 } else { if(defaultContexts[$(this).val()] !== undefined) { context_name = $(this).parents('form').find('input[name=context_name]'); if(context_name.attr('edited') === undefined){ context_name.val(defaultContexts[$(this).val()]); } } } if(defaultTags[$(this).val()] !== undefined) { tag_list = $(this).parents('form').find('input[name=tag_list]'); if(tag_list.attr('edited') === undefined){ tag_list.val(defaultTags[$(this).val()]); } } } function enable_rich_interaction(){ /* fix for #1036 where closing a edit form before the autocomplete was filled * resulted in a dropdown box that could not be removed. We remove all * autocomplete boxes the hard way */ $('.ac_results').remove(); $('input.Date').datepicker({ 'dateFormat': dateFormat, 'firstDay': weekStart, 'showAnim': 'fold' }); /* Autocomplete */ $('input[name=context_name]').autocomplete({ source: relative_to_root('contexts.autocomplete') }); /* $('input[name=project[default_context_name]]').autocomplete( relative_to_root('contexts.autocomplete'), {matchContains: true}); $('input[name=project_name]').autocomplete( relative_to_root('projects.autocomplete'), {matchContains: true}); $('input[name=tag_list]:not(.ac_input)').autocomplete( relative_to_root('tags.autocomplete'), {multiple: true,multipleSeparator:',',matchContains:true}); $('input[name=predecessor_list]:not(.ac_input)').autocomplete( relative_to_root('auto_complete_for_predecessor'), {multiple: true,multipleSeparator:','}); /* have to bind on keypress because of limitataions of live() */ $('input[name=project_name]').live('keypress', function(){ $(this).bind('blur', project_defaults); }); $('input[name=context_name]').live('keypress', function(){ $(this).attr('edited', 'true'); }); $('input[name=tag_list]').live('keypress', function(){ $(this).attr('edited', 'true'); }); /* Drag & Drop for successor/predecessor */ function drop_todo(evt, ui) { dragged_todo = ui.draggable[0].id.split('_')[2]; dropped_todo = this.id.split('_')[2]; ui.draggable.remove(); $('.drop_target').hide(); // IE8 doesn't call stop() in this situation $(this).block({ message: null }); $.post(relative_to_root('todos/add_predecessor'), { successor: dragged_todo, predecessor: dropped_todo }, null, 'script'); } function drag_todo(){ $('.drop_target').show(); $(this).parents(".container").find(".context_target").hide(); } $('.item-show').draggable({ handle: '.grip', revert: 'invalid', start: drag_todo, stop: function() { $('.drop_target').hide(); } }); $('.item-show').droppable({ drop: drop_todo, tolerance: 'pointer', hoverClass: 'hover' }); /* Drag & drop for changing contexts */ function drop_todo_on_context(evt, ui) { target = $(this); dragged_todo = ui.draggable[0].id.split('_')[2]; context_id = this.id.split('_')[1]; ui.draggable.remove(); target.block({ message: null }); setTimeout(function() { target.show() }, 0); $.post(relative_to_root('todos/change_context'), { "todo[id]": dragged_todo, "todo[context_id]": context_id }, function(){ target.unblock(); target.hide(); }, 'script'); } $('.context_target').droppable({ drop: drop_todo_on_context, tolerance: 'pointer', hoverClass: 'hover' }); /* Reset auto updater */ field_touched = false; /* shrink the notes on the project pages. This is not live(), so this needs * to be run after ajax adding of a new note */ $('.note_wrapper').truncate({ max_length: 90, more: '', less: '' }); } /* Auto-refresh */ function setup_auto_refresh(interval){ field_touched = false; function refresh_page() { if(!field_touched){ window.location.reload(); } } setTimeout(refresh_page, interval); $(function(){ $("input").live('keydown', function(){ field_touched = true; }); }); } function page_notify(type, message, fade_duration_in_sec) { flash = $('h4#flash'); flash.html("