/* * Copyright 2010 Nathan Gelderloos, Ethan Zonca, Nathan Phillip Brink * * This file is part of SlatePermutate. * * SlatePermutate is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * SlatePermutate is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with SlatePermutate. If not, see . */ //-------------------------------------------------- // General Notes //-------------------------------------------------- var classNum = 0; var sectionsOfClass = new Array(); //-------------------------------------------------- // Validation Functions //-------------------------------------------------- //-------------------------------------------------- // Default Error Message //-------------------------------------------------- jQuery.each(jQuery.validator.messages, function(i) { jQuery.validator.messages[i] = "

Please fill the field

"; }); //-------------------------------------------------- // Time Selection Validation //-------------------------------------------------- jQuery.validator.addMethod( "selectNone", function(value, element) { if (element.value == "none") { return false; } else return true; }, "

Please select a time

" ); //-------------------------------------------------- // Days of Week validation //-------------------------------------------------- jQuery.validator.addMethod( "daysRequired", function(value, element) { var checkedCount = 0; jQuery(element).parent().parent().children().children('.daysRequired:checked').each( function() { checkedCount++; }); if (checkedCount == 0) { return false; } else return true; }, "

Select a day!

" ); /** * Class name validation: only require a class name if it has at least * one section. Backend throws out empty classes and it's more * convenient if we can let the user have one extra, empty class. This * is because we automatically add a new class each time we do an * autofill to make the life of the user easier and less confusing. */ jQuery.validator.addMethod('classRequired', function(value, element) { if (value.length) return true; var css_classes = jQuery(element).attr('class'); var cnum_pos = css_classes.indexOf('className'); var cnum = css_classes.substr(cnum_pos + 'className'.length, css_classes.indexOf(' ', cnum_pos) - cnum_pos - 'className'.length) * 1; if (cnum < 0 || cnum > classNum) alert('JS error: ' + cnum + ' is an invalid class number.'); /* * ignore the class with no * sections. This only works when the * class was added and the user _never_ * clicked the Add Section button. Once * the user clicks that button, he has * to delete the class because of how * our numbering works. */ if (!sectionsOfClass[cnum]) return true; return false; }, '

Enter Class Name.

' ); //-------------------------------------------------- // Add validation rules //-------------------------------------------------- jQuery.validator.addClassRules("selectRequired", { selectNone: true }); jQuery.validator.addClassRules("daysRequired", { daysRequired: true }); jQuery.validator.addClassRules('classRequired', { classRequired: true }); //-------------------------------------------------- // General Input Functions //-------------------------------------------------- /** * \brief * Returns the common inputs for each new section. */ function genSectionHtml(cnum) { genSectionHtml_n(cnum, '', '', '', '', '', '', ''); } /* @TODO: This should select & set items based on args, if the args != '' */ function genSectionHtml_n(cnum, name, synonym, stime, etime, days, prof, location, type) { var snum = sectionsOfClass[cnum]; var result = ''; result = result + ''; result = result + ''; result = result + '\ \ \ \ \ \ '; result = result + '
' + '' + '' + ''; return result; } /** * Outputs an \n"; } /** Add tooltips for user guidance */ function addTips(id) { jQuery(id).qtip({ content: 'Type your class ID (such as PEF-1010)', style: { name: 'dark', tip: true }, show: { effect: { type: 'fade', length: 2000 } }, /* show: { ready: false }, */ /* hide: { when: { event: 'inactive' } }, */ corner: { target: 'topMiddle', tooltip: 'bottomMiddle' }, }); } /** * \brief * Add a section to a class. */ function add_section_n(cnum, name, synonym, stime, etime, days, prof, location, type) { jQuery('.pclass'+cnum).after(genSectionHtml_n(cnum, name, synonym, stime, etime, days, prof, location, type)); sectionsOfClass[cnum] ++; } function add_section(cnum) { return add_section_n(cnum, '', '', '', '', {'m':false, 't':false, 'w':false, 'h':false, 'f':false}, '', '', ''); } /** * Add a list of sections gotten via an AJAX call. */ function add_sections(cnum, data) { var i; if (!data.sections) return; /* * we get the sections in the correct order. For the user to see * them in the correct order, we must reverse the add_setion_n() * calls. */ for (i = data.sections.length - 1; i >= 0; i --) { section = data.sections[i]; add_section_n(cnum, section.section, section.synonym, section.time_start, section.time_end, section.days, section.prof, section.location, section.type); } } //-------------------------------------------------- // Adds a new class to the input. //-------------------------------------------------- function add_class_n(name) { sectionsOfClass[classNum] = 0; // Initialize at 0 jQuery('#jsrows').append('
'); jQuery('.className' + classNum).autocomplete({ source: "auto.php" }); jQuery('.className' + classNum).bind('autocompleteselect', {'class_num': classNum}, function(event, ui) { if (ui.item && ui.item.value.indexOf('-')) { jQuery.ajax( { url: 'auto.php', data: {'getsections': 1, 'term': ui.item.value}, context: {'class_num': event.data.class_num}, success: function(data, textStatus, reqobj) { if (data.sections) { add_sections(this.class_num, data); add_class(); } } } ); } }); addTips('.class'+classNum+ ' td:first'); classNum++; return (classNum - 1); }; function add_class() { return add_class_n(''); } /** * \brief * Render a slate_permutate-encoded time-of-day. * * \param time_str * A four-character representation of a time of day based on a * time's 24-hour representation. * \return * A string representing the specified time. */ function prettyTime(time_str) { var i_hour; var hour_str; var m; i_hour = time_str.substr(0, 2) * 1; if (i_hour <= 12) { m = 'a'; } else { m = 'p'; i_hour -= 12; } hour_str = new String(i_hour); /* uncomment to have 08:01 instead of 8:01 */ /* while (hour_str.length < 2) hour_str = '0' + hour_str; */ return hour_str + ':' + time_str.substr(2) + ' ' + m + 'm'; } //-------------------------------------------------- // Items bound to pageload/events //-------------------------------------------------- jQuery(document).ready(function() { //-------------------------------------------------- // Validates the form (pre-submission check) //-------------------------------------------------- jQuery('#scheduleForm').validate({ debug: false }); //-------------------------------------------------- // Bind the class-adding method //-------------------------------------------------- jQuery('#addclass').click(function() { add_class(); }); //-------------------------------------------------- // Deletes the selected class from input //-------------------------------------------------- jQuery('.deleteClass').live('click', function() { if(confirm('Delete class and all sections of this class?')) { jQuery('.class'+ jQuery(this).parent().parent().attr("title")).remove(); } }); //-------------------------------------------------- // Deletes the selected section from the input //-------------------------------------------------- jQuery('.deleteSection').live('click', function() { sectionsOfClass[jQuery(this).parent().parent().attr("title")]--; // Decreases the number of classes jQuery(this).parent().parent().remove(); }); //-------------------------------------------------- // Bind the section-adding method //-------------------------------------------------- jQuery('.addSection').live('click', function() { add_section(jQuery(this).parent().parent().attr("title"), sectionsOfClass[jQuery(this).parent().parent().attr("title")]); }); //-------------------------------------------------- // Default text //-------------------------------------------------- jQuery(".defText").focus(function(srcc) { if (jQuery(this).val() == jQuery(this)[0].title) { jQuery(this).removeClass("defaultTextActive"); jQuery(this).val(""); } }); jQuery(".defText").blur(function() { if (jQuery(this).val() == "") { jQuery(this).addClass("defaultTextActive"); jQuery(this).val($(this)[0].title); } }); jQuery(".defText").blur(); //-------------------------------------------------- // Show/Hide advanced items //-------------------------------------------------- jQuery('.advanced').hide(); jQuery('#showadvanced').click( function() { jQuery('#showadvanced').hide(); jQuery('.advanced').slideToggle(); }); //------------------------------------------------- // Show more saved schedules //------------------------------------------------- jQuery('#showMore').click( function() { jQuery('.hidden').show(); jQuery('#showMore').hide(); jQuery('#showLess').show(); }); jQuery('#showLess').click( function() { jQuery('.hidden').hide(); jQuery('#showMore').show(); jQuery('#showLess').hide(); }); });