/* TIMELINE NAVIGATION ================================================== */ if(typeof VMM.Timeline != 'undefined' && typeof VMM.Timeline.TimeNav == 'undefined') { VMM.Timeline.TimeNav = function(parent, content_width, content_height) { trace("VMM.Timeline.TimeNav"); var events = {}, timespan = {}, layout = parent; var data = [], era_markers = [], markers = [], interval_array = [], interval_major_array = [], eras, content; var current_marker = 0; var _active = false; var timelookup = {day: 24, month: 12, year: 10, hour: 60, minute: 60, second: 1000, decade: 10, century: 100, millenium: 1000, week: 4.34812141, days_in_month: 30.4368499, days_in_week: 7, weeks_in_month:4.34812141, weeks_in_year:52.177457, days_in_year: 365.242199, hours_in_day: 24 }; var dateFractionBrowser = {day: 86400000, week: 7, month: 30.4166666667, year: 12, hour: 24, minute: 1440, second: 86400, decade: 10, century: 100, millenium: 1000 }; var interval = {type: "year", number: 10, first: 1970, last: 2011, multiplier: 100, classname:"_idd", interval_type:"interval"}; var interval_major = {type: "year", number: 10, first: 1970, last: 2011, multiplier: 100, classname:"major", interval_type:"interval major"}; var interval_macro = {type: "year", number: 10, first: 1970, last: 2011, multiplier: 100, classname:""}; var interval_calc = {day: {},month: {},year: {},hour: {},minute: {}, second: {},decade: {},century: {},millenium: {},week: {}}; /* ELEMENTS ================================================== */ var $timenav, $content, $time, $timeintervalminor, $timeinterval, $timeintervalmajor, $timebackground, $timeintervalbackground, $timenavline, $timeintervalminor_minor, $toolbar, $zoomin, $zoomout; /* ADD to Config ================================================== */ var config = VMM.Timeline.Config; config.nav.rows = [config.nav.marker.height, config.nav.marker.height*2, 1]; if (content_width != null && content_width != "") { config.nav.width = content_width; } if (content_height != null && content_height != "") { config.nav.height = content_height; } /* config.nav.density = 2; config.nav.multiplier = { current: 6, min: .1, max: 50 }; */ /* INIT ================================================== */ this.init = function(d,e) { trace('VMM.Timeline.TimeNav init'); // need to evaluate d // some function to determine type of data and prepare it if(typeof d != 'undefined') { this.setData(d, e); } else { trace("WAITING ON DATA"); } }; /* GETTERS AND SETTERS ================================================== */ this.setData = function(d,e) { if(typeof d != 'undefined') { data = d; eras = e; build(); } else{ trace("NO DATA"); } }; this.setSize = function(w, h) { if (w != null) {config.width = w}; if (h != null) {config.height = h}; if (_active) { reSize(); } } this.setMarker = function(n, ease, duration, fast) { goToMarker(n, ease, duration); } this.getCurrentNumber = function() { return current_marker; } /* ON EVENT ================================================== */ function onConfigSet() { trace("onConfigSet"); }; function reSize(firstrun) { VMM.Element.css($timenavline, "left", Math.round(config.width/2)+2); goToMarker(current_marker, config.ease, config.duration, true, firstrun); }; function upDate() { VMM.fireEvent(layout, "UPDATE"); } function onZoomIn() { VMM.DragSlider.cancelSlide(); if (config.nav.multiplier.current > config.nav.multiplier.min) { if (config.nav.multiplier.current <= 1) { config.nav.multiplier.current = config.nav.multiplier.current - .25; } else { if (config.nav.multiplier.current > 5) { if (config.nav.multiplier.current > 16) { config.nav.multiplier.current = Math.round(config.nav.multiplier.current - 10); } else { config.nav.multiplier.current = Math.round(config.nav.multiplier.current - 4); } } else { config.nav.multiplier.current = Math.round(config.nav.multiplier.current - 1); } } if (config.nav.multiplier.current <= 0) { config.nav.multiplier.current = config.nav.multiplier.min; } refreshTimeline(); } } function onZoomOut() { VMM.DragSlider.cancelSlide(); if (config.nav.multiplier.current < config.nav.multiplier.max) { if (config.nav.multiplier.current > 4) { if (config.nav.multiplier.current > 16) { config.nav.multiplier.current = Math.round(config.nav.multiplier.current + 10); } else { config.nav.multiplier.current = Math.round(config.nav.multiplier.current + 4); } } else { config.nav.multiplier.current = Math.round(config.nav.multiplier.current + 1); } if (config.nav.multiplier.current >= config.nav.multiplier.max) { config.nav.multiplier.current = config.nav.multiplier.max; } refreshTimeline(); } } function onBackHome(e) { VMM.DragSlider.cancelSlide(); goToMarker(0); upDate(); } var refreshTimeline = function() { trace("config.nav.multiplier " + config.nav.multiplier.current); positionMarkers(true); positionInterval(interval_array, true, true); positionInterval(interval_major_array, true); }; /* MARKER EVENTS ================================================== */ function onMarkerClick(e) { VMM.DragSlider.cancelSlide(); goToMarker(e.data.number); upDate(); }; function onMarkerHover(e) { VMM.Element.toggleClass(e.data.elem, "zFront"); }; var goToMarker = function(n, ease, duration, fast, firstrun) { current_marker = n; var _ease = config.ease; var _duration = config.duration; var is_last = false; var is_first = false; var _pos = VMM.Element.position(markers[current_marker].marker); if (current_marker == 0) { is_first = true; } if (current_marker +1 == markers.length) { is_last = true } if (ease != null && ease != "") {_ease = ease}; if (duration != null && duration != "") {_duration = duration}; // set marker style for(var i = 0; i < markers.length; i++) { VMM.Element.removeClass(markers[i].marker, "active"); } if (config.start_page && markers[current_marker].type == "start") { VMM.Element.visible(markers[current_marker].marker, false); VMM.Element.addClass(markers[current_marker].marker, "start"); } VMM.Element.addClass(markers[current_marker].marker, "active"); // ANIMATE MARKER VMM.Element.stop($timenav); VMM.Element.animate($timenav, _duration, _ease, {"left": (config.width/2) - (_pos.left)}); } /* TOUCH EVENTS ================================================== */ function onTouchUpdate(e, b) { VMM.Element.animate($timenav, b.time/2, config.ease, {"left": b.left}); }; /* CALCULATIONS ================================================== */ var averageMarkerPositionDistance = function() { var last_pos = 0; var pos = 0; var pos_dif = 0; var mp_diff = []; for(var i = 0; i < markers.length; i++) { if (data[i].type == "start") { } else { var _pos = positionOnTimeline(interval, data[i].startdate, data[i].enddate); last_pos = pos; pos = _pos.begin; pos_dif = pos - last_pos; mp_diff.push(pos_dif); } } return VMM.Util.average(mp_diff).mean; } var averageDateDistance = function() { var last_dd = 0; var dd = 0; var date_dif = 0; var date_diffs = []; var is_first_date = true; for(var i = 0; i < data.length; i++) { if (data[i].type == "start") { trace("DATA DATE IS START") } else { var _dd = data[i].startdate; last_dd = dd; dd = _dd; date_dif = dd - last_dd; date_diffs.push(date_dif); } } return VMM.Util.average(date_diffs); } var calculateMultiplier = function() { var temp_multiplier = config.nav.multiplier.current; for(var i = 0; i < temp_multiplier; i++) { if (averageMarkerPositionDistance() < 75) { if (config.nav.multiplier.current > 1) { config.nav.multiplier.current = config.nav.multiplier.current - 1; } } } } var calculateInterval = function() { var _first = getDateFractions(data[0].startdate); var _last = getDateFractions(data[data.length - 1].enddate); /* MILLENIUM ================================================== */ // NOT DONE interval_calc.millenium.type = "millenium"; interval_calc.millenium.first = _first.milleniums; interval_calc.millenium.base = Math.floor(_first.milleniums); interval_calc.millenium.last = _last.milleniums; interval_calc.millenium.number = timespan.milleniums; interval_calc.millenium.multiplier = timelookup.millenium; interval_calc.millenium.minor = timelookup.millenium; /* CENTURY ================================================== */ // NOT DONE interval_calc.century.type = "century"; interval_calc.century.first = _first.centuries; interval_calc.century.base = Math.floor(_first.centuries); interval_calc.century.last = _last.centuries; interval_calc.century.number = timespan.centuries; interval_calc.century.multiplier = timelookup.century; interval_calc.century.minor = timelookup.century; /* DECADE ================================================== */ interval_calc.decade.type = "decade"; interval_calc.decade.first = _first.decades; interval_calc.decade.base = Math.floor(_first.decades); interval_calc.decade.last = _last.decades; interval_calc.decade.number = timespan.decades; interval_calc.decade.multiplier = timelookup.decade; interval_calc.decade.minor = timelookup.decade; /* YEAR ================================================== */ interval_calc.year.type = "year"; interval_calc.year.first = _first.years; interval_calc.year.base = Math.floor(_first.years); interval_calc.year.last = _last.years; interval_calc.year.number = timespan.years; interval_calc.year.multiplier = 1; interval_calc.year.minor = timelookup.month; /* MONTH ================================================== */ interval_calc.month.type = "month"; interval_calc.month.first = _first.months; interval_calc.month.base = Math.floor(_first.months); interval_calc.month.last = _last.months; interval_calc.month.number = timespan.months; interval_calc.month.multiplier = 1; interval_calc.month.minor = Math.round(timelookup.week); /* WEEK ================================================== */ // NOT DONE interval_calc.week.type = "week"; interval_calc.week.first = _first.weeks; interval_calc.week.base = Math.floor(_first.weeks); interval_calc.week.last = _last.weeks; interval_calc.week.number = timespan.weeks; interval_calc.week.multiplier = 1; interval_calc.week.minor = 7; /* DAY ================================================== */ // NOT DONE interval_calc.day.type = "day"; interval_calc.day.first = _first.days; interval_calc.day.base = Math.floor(_first.days); interval_calc.day.last = _last.days; interval_calc.day.number = timespan.days; interval_calc.day.multiplier = 1; interval_calc.day.minor = 24; /* HOUR ================================================== */ // NOT DONE interval_calc.hour.type = "hour"; interval_calc.hour.first = _first.hours; interval_calc.hour.base = Math.floor(_first.hours); interval_calc.hour.last = _last.hours; interval_calc.hour.number = timespan.hours; interval_calc.hour.multiplier = 1; interval_calc.hour.minor = 60; /* MINUTE ================================================== */ // NOT DONE interval_calc.minute.type = "minute"; interval_calc.minute.first = _first.minutes; interval_calc.minute.base = Math.floor(_first.minutes); interval_calc.minute.last = _last.minutes; interval_calc.minute.number = timespan.minutes; interval_calc.minute.multiplier = 1; interval_calc.minute.minor = 60; /* SECOND ================================================== */ // NOT DONE interval_calc.second.type = "decade"; interval_calc.second.first = _first.seconds; interval_calc.second.base = Math.floor(_first.seconds); interval_calc.second.last = _last.seconds; interval_calc.second.number = timespan.seconds; interval_calc.second.multiplier = 1; interval_calc.second.minor = 10; } var getDateFractions = function(the_date, is_utc) { var _time = {}; _time.days = the_date / dateFractionBrowser.day; _time.weeks = _time.days / dateFractionBrowser.week; _time.months = _time.days / dateFractionBrowser.month; _time.years = _time.months / dateFractionBrowser.year; _time.hours = _time.days * dateFractionBrowser.hour; _time.minutes = _time.days * dateFractionBrowser.minute; _time.seconds = _time.days * dateFractionBrowser.second; _time.decades = _time.years / dateFractionBrowser.decade; _time.centuries = _time.years / dateFractionBrowser.century; _time.milleniums = _time.years / dateFractionBrowser.millenium; /* trace("MILLENIUMS " + _time.milleniums); trace("CENTURIES " + _time.centuries); trace("DECADES " + _time.decades); trace("YEARS " + _time.years); trace("MONTHS " + _time.months); trace("WEEKS " + _time.weeks); trace("DAYS " + _time.days); trace("HOURS " + _time.hours); trace("MINUTES " + _time.minutes); trace("SECONDS " + _time.seconds); */ return _time; } /* POSITION ================================================== */ var positionOnTimeline = function(the_interval, first, last) { var _type = the_interval.type; var _multiplier = the_interval.multiplier; var _first = getDateFractions(first); var _last; var tsd; var ted; /* CALCULATE POSITION ON TIMELINE ================================================== */ tsd = first.months; if (type.of(last) == "date") { /* LAST ================================================== */ _last = getDateFractions(last); ted = last.months; if (_type == "millenium") { tsd = first.milleniums; ted = last.milleniums; } else if (_type == "century") { tsd = _first.centuries; ted = _last.centuries; } else if (_type == "decade") { tsd = _first.decades; ted = _last.decades; } else if (_type == "year") { tsd = _first.years; ted = _last.years; } else if (_type == "month") { tsd = _first.months; ted = _last.months; } else if (_type == "week") { tsd = _first.weeks; ted = _last.weeks; } else if (_type == "day") { tsd = _first.days; ted = _last.days; } else if (_type == "hour") { tsd = _first.hours; ted = _last.hours; } else if (_type == "minute") { tsd = _first.minutes; ted = _last.minutes; } _pos = ( tsd - interval.base ) * (config.nav.interval_width / config.nav.multiplier.current); _pos_end = ( ted - interval.base ) * (config.nav.interval_width / config.nav.multiplier.current); } else { if (_type == "millenium") { tsd = first.milleniums; ted = first.milleniums; } else if (_type == "century") { tsd = _first.centuries; ted = _first.centuries; } else if (_type == "decade") { tsd = _first.decades; ted = _first.decades; } else if (_type == "year") { tsd = _first.years; ted = _first.years; } else if (_type == "month") { tsd = _first.months; ted = _first.months; } else if (_type == "week") { tsd = _first.weeks; ted = _first.weeks; } else if (_type == "day") { tsd = _first.days; ted = _first.days; } else if (_type == "hour") { tsd = _first.hours; ted = _first.hours; } else if (_type == "minute") { tsd = _first.minutes; ted = _first.minutes; } _pos = ( tsd - interval.base ) * (config.nav.interval_width / config.nav.multiplier.current); _pos_end = _pos; } return pos = {begin:_pos ,end:_pos_end}; } var positionMarkers = function(is_animated) { var _type = interval.type; var _multiplier = interval.multiplier; // ROWS var row = 2; //row var lpos = 0; // last marker pos; var row_depth = 0; var _line_last_height_pos = 150; var _line_height = 6; var cur_mark = 0; VMM.Element.removeClass(".flag", "row1"); VMM.Element.removeClass(".flag", "row2"); VMM.Element.removeClass(".flag", "row3"); for(var i = 0; i < markers.length; i++) { var _line; // EVENT LENGTH var _marker = markers[i].marker; var _marker_flag = markers[i].flag; var _marker_line_event = markers[i].lineevent; var _pos = positionOnTimeline(interval, data[i].startdate, data[i].enddate); var _pos_offset = -2; pos = _pos.begin; _pos_end = _pos.end; // COMPENSATE FOR DATES BEING POITIONED IN THE MIDDLE pos = Math.round(pos + _pos_offset); _pos_end = Math.round(_pos_end + _pos_offset); _line = Math.round(_pos_end - pos); // APPLY POSITION TO MARKER if (is_animated) { VMM.Element.stop(_marker); VMM.Element.animate(_marker, config.duration/2, config.ease, {"left": pos}); } else { VMM.Element.css(_marker, "left", pos); } if (i == current_marker) { cur_mark = pos; } // EVENT LENGTH LINE if (_line > 5) { VMM.Element.css(_marker_line_event, "height", _line_height); VMM.Element.css(_marker_line_event, "width", _line); VMM.Element.css(_marker_line_event, "top", _line_last_height_pos); } // CONTROL ROW POSITION if (pos - lpos < (config.nav.marker.width + config.spacing)) { if (row < config.nav.rows.length - 1) { row ++; } else { row = 0; row_depth ++; } } else { row_depth = 0; row = 0; } // SET LAST MARKER POSITION lpos = pos; if (is_animated) { VMM.Element.stop(_marker_flag); VMM.Element.animate(_marker_flag, config.duration, config.ease, {"top": config.nav.rows[row]}); } else { VMM.Element.css(_marker_flag, "top", config.nav.rows[row]); } // IS THE MARKER A REPRESENTATION OF A START SCREEN? if (config.start_page && markers[i].type == "start") { VMM.Element.visible(_marker, false); } } for(var j = 0; j < era_markers.length; j++) { var _line; var era = era_markers[j]; var era_elem = era.content; var pos = positionOnTimeline(interval, era.startdate, era.enddate); var era_length = pos.end - pos.begin; var era_height = 25; // APPLY POSITION TO MARKER VMM.Element.css(era_elem, "left", pos.begin); VMM.Element.css(era_elem, "width", era_length); } // ANIMATE THE TIMELINE TO ADJUST TO CHANGES VMM.Element.stop($timenav); VMM.Element.animate($timenav, config.duration/2, config.ease, {"left": (config.width/2) - (cur_mark)}); } var positionInterval = function(the_intervals, is_animated, is_minor) { var _type = interval.type; var _multiplier = interval.multiplier; var last_position = 0; var last_position_major = 0; for(var i = 0; i < the_intervals.length; i++) { var _interval = the_intervals[i].interval_element; var _interval_date = the_intervals[i].interval_date; var _interval_visible = the_intervals[i].interval_visible; var _pos = positionOnTimeline(interval, _interval_date); var pos = _pos.begin; var is_visible = true; var pos_offset = 50; // APPLY POSITION TO MARKER if (is_animated) { VMM.Element.animate(_interval, config.duration/2, config.ease, {"left": pos}); } else { VMM.Element.css(_interval, "left", pos); } // CONDENSE WHAT IS DISPLAYED if (config.nav.multiplier.current > 16 && is_minor) { is_visible = false; } else { if ((pos - last_position) < 65 ) { if ((pos - last_position) < 35 ) { if (i%4 == 0) { if (pos == 0) { is_visible = false; } } else { is_visible = false; } } else { if (!VMM.Util.isEven(i)) { is_visible = false; } } } } if (_interval_visible) { if (!is_visible) { if (is_animated) { VMM.Element.animate(_interval, config.duration*2, config.ease, {"opacity": 0}); } else { VMM.Element.css(_interval, "opacity", 0); } the_intervals[i].interval_visible = false; } } else { if (is_visible) { if (is_animated) { VMM.Element.animate(_interval, config.duration*2, config.ease, {"opacity": 100}); } else { VMM.Element.css(_interval, "opacity", 100); } the_intervals[i].interval_visible = true; } } last_position = pos; if (pos > config.nav.minor_width) { config.nav.minor_width = pos; } } VMM.Element.css($timeintervalminor_minor, "left", -(config.width/2)); VMM.Element.width($timeintervalminor_minor, (config.nav.minor_width)+(config.width) ); } var createIntervalElements = function(_interval, _array, _element_parent) { var inc_time = 0; var _first_run = true; var _last_pos = 0; var _largest_pos = 0; VMM.attachElement(_element_parent, ""); _interval.date = new Date(data[0].startdate.getFullYear(), 0, 1, 0,0,0); for(var i = 0; i < Math.ceil(_interval.number) + 1; i++) { var _idd; var _pos; var pos; var _element = VMM.appendAndGetElement(_element_parent, "