700,
"height"=>460,
"image" => array(
"width"=>200,
"height"=>200,
"top_offset"=>20,
),
"image_text" => array(
"width"=>400,
"height"=>30,
"top_offset"=>20,
),
"graph" => array(
"width"=>600,
"height"=>160,
"top_offset"=>30,
),
"title" => array(
"top_offset"=>50
),
"key" => array(
"top_offset"=>50
),
"axes" => array(
"x" => array(
"height" => 20,
),
"y" => array(
"width" => 30,
),
),
"grid" => array(
"x" => array(
"major" => array(
"max" => 12,
"min" => 4,
),
"minor" => array(
"max" => 48,
"min" => 12,
),
),
"y" => array(
"major" => array(
"max" => 8,
"min" => 1,
),
"minor" => array(
"max" => 0,
"min" => 0,
),
),
),
);
$monitors = array();
$monitors_sql = "select * from Monitors order by Sequence asc";
//srand( 97981 );
foreach( dbFetchAll( $monitors_sql ) as $row )
{
//if ( empty($row['WebColour']) )
//{
//$row['WebColour'] = sprintf( "#%02x%02x%02x", rand( 0, 255 ), rand( 0, 255), rand( 0, 255 ) );
//}
$monitors[$row['Id']] = $row;
}
$range_sql = "select min(E.StartTime) as MinTime, max(E.EndTime) as MaxTime from Events as E inner join Monitors as M on (E.MonitorId = M.Id) where not isnull(E.StartTime) and not isnull(E.EndTime)";
$events_sql = "select E.Id,E.Name,E.StartTime,E.EndTime,E.Length,E.Frames,E.MaxScore,E.Cause,E.Notes,E.Archived,E.MonitorId from Events as E inner join Monitors as M on (E.MonitorId = M.Id) where not isnull(StartTime)";
if ( !empty($user['MonitorIds']) )
{
$mon_filter_sql = " and M.Id in (".join( ",", preg_split( '/["\'\s]*,["\'\s]*/', $user['MonitorIds'] ) ).")";
$range_sql .= $mon_filter_sql;
$events_sql .= $mon_filter_sql;
}
if ( isset($filter) )
$tree = parseFilterToTree( $filter );
else
$tree = false;
if ( isset($range) )
{
$half_range = (int)($range/2);
if ( isset($mid_time) )
{
$mid_time_t = strtotime($mid_time);
$min_time_t = $mid_time_t-$half_range;
$max_time_t = $mid_time_t+$half_range;
if ( !($range%1) )
{
$max_time_t--;
}
$min_time = strftime( STRF_FMT_DATETIME_DB, $min_time_t );
$max_time = strftime( STRF_FMT_DATETIME_DB, $max_time_t );
}
elseif ( isset($min_time) )
{
$min_time_t = strtotime($min_time);
$max_time_t = $min_time_t + $range;
$mid_time_t = $min_time_t + $half_range;
$mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t );
$max_time = strftime( STRF_FMT_DATETIME_DB, $max_time_t );
}
elseif ( isset($max_time) )
{
$max_time_t = strtotime($max_time);
$min_time_t = $max_time_t - $range;
$mid_time_t = $min_time_t + $half_range;
$min_time = strftime( STRF_FMT_DATETIME_DB, $min_time_t );
$mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t );
}
}
elseif ( isset($min_time) && isset($max_time) )
{
$min_time_t = strtotime($min_time);
$max_time_t = strtotime($max_time);
$range = ($max_time_t - $min_time_t) + 1;
$half_range = (int)($range/2);
$mid_time_t = $min_time_t + $half_range;
$mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t );
}
if ( isset($min_time) && isset($max_time) )
{
$temp_min_time = $temp_max_time = $temp_expandable = false;
extractDatetimeRange( $tree, $temp_min_time, $temp_max_time, $temp_expandable );
$filter_sql = parseTreeToSQL( $tree );
if ( $filter_sql )
{
$filter_sql = " and $filter_sql";
$events_sql .= $filter_sql;
}
}
else
{
//$filter_query = parseTreeToQuery( $tree );
//echo $filter_query;
//echo '
';
$filter_sql = parseTreeToSQL( $tree );
$temp_min_time = $temp_max_time = $temp_expandable = false;
extractDatetimeRange( $tree, $temp_min_time, $temp_max_time, $temp_expandable );
//echo $filter_sql;
//echo '
';
if ( $filter_sql )
{
$filter_sql = " and $filter_sql";
$range_sql .= $filter_sql;
$events_sql .= $filter_sql;
}
if ( !isset($min_time) || !isset($max_time) )
{
// Dynamically determine range
$row = dbFetchOne( $range_sql );
if ( !isset($min_time) )
$min_time = $row['MinTime'];
if ( !isset($max_time) )
$max_time = $row['MaxTime'];
}
if ( empty($min_time) )
$min_time = $temp_min_time;
if ( empty($max_time) )
$max_time = $temp_max_time;
if ( empty($max_time) )
$max_time = "now";
$min_time_t = strtotime($min_time);
$max_time_t = strtotime($max_time);
$range = ($max_time_t - $min_time_t) + 1;
$half_range = (int)($range/2);
$mid_time_t = $min_time_t + $half_range;
$mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t );
}
//echo "MnT: $temp_min_time, MxT: $temp_max_time, ExP: $temp_expandable
";
appendDatetimeRange( $tree, $min_time, $max_time );
$filter_query = parseTreeToQuery( $tree );
if ( $filter_query )
{
$filter_query = '&'.$filter_query;
}
//echo $filter_query;
//echo '
';
$scales = array(
array( "name"=>"year", "factor"=>60*60*24*365, "align"=>1, "zoomout"=>2, "label"=>STRF_TL_AXIS_LABEL_YEAR ),
array( "name"=>"month", "factor"=>60*60*24*30, "align"=>1, "zoomout"=>12, "label"=>STRF_TL_AXIS_LABEL_MONTH ),
array( "name"=>"week", "factor"=>60*60*24*7, "align"=>1, "zoomout"=>4.25, "label"=>STRF_TL_AXIS_LABEL_WEEK, "label_check"=>"%W" ),
array( "name"=>"day", "factor"=>60*60*24, "align"=>1, "zoomout"=>7, "label"=>STRF_TL_AXIS_LABEL_DAY ),
array( "name"=>"hour", "factor"=>60*60, "align"=>1, "zoomout"=>24, "label"=>STRF_TL_AXIS_LABEL_HOUR, "label_check"=>"%H" ),
array( "name"=>"minute10", "factor"=>60, "align"=>10, "zoomout"=>6, "label"=>STRF_TL_AXIS_LABEL_10MINUTE, "label_check"=>"%M" ),
array( "name"=>"minute", "factor"=>60, "align"=>1, "zoomout"=>10, "label"=>STRF_TL_AXIS_LABEL_MINUTE, "label_check"=>"%M" ),
array( "name"=>"second10", "factor"=>1, "align"=>10, "zoomout"=>6, "label"=>STRF_TL_AXIS_LABEL_10SECOND ),
array( "name"=>"second", "factor"=>1, "align"=>1, "zoomout"=>10, "label"=>STRF_TL_AXIS_LABEL_SECOND ),
);
$maj_x_scale = getDateScale( $scales, $range, $chart['grid']['x']['major']['min'], $chart['grid']['x']['major']['max'] );
//print_r( $maj_x_scale );
// Adjust the range etc for scale
$min_time_t -= $min_time_t%($maj_x_scale['factor']*$maj_x_scale['align']);
$min_time = strftime( STRF_FMT_DATETIME_DB, $min_time_t );
$max_time_t += (($maj_x_scale['factor']*$maj_x_scale['align'])-$max_time_t%($maj_x_scale['factor']*$maj_x_scale['align']))-1;
if ( $max_time_t > time() )
$max_time_t = time();
$max_time = strftime( STRF_FMT_DATETIME_DB, $max_time_t );
$range = ($max_time_t - $min_time_t) + 1;
$half_range = (int)($range/2);
$mid_time_t = $min_time_t + $half_range;
$mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t );
//echo "R:$range
";
//echo "MnT:$min_time
";
//echo "MnTt:$min_time_t
";
//echo "MdT:$mid_time
";
//echo "MdTt:$mid_time_t
";
//echo "MxT:$max_time
";
//echo "MxTt:$max_time_t
";
if ( isset($min_time) && isset($max_time) )
{
$events_sql .= " and E.EndTime >= '$min_time' and E.StartTime <= '$max_time'";
}
$events_sql .= " order by Id asc";
//echo "ESQL: $events_sql
";
$chart['data'] = array(
"x" => array(
"lo" => strtotime( $min_time ),
"hi" => strtotime( $max_time ),
),
"y" => array(
"lo" => 0,
"hi" => 0,
)
);
$chart['data']['x']['range'] = ($chart['data']['x']['hi'] - $chart['data']['x']['lo']) + 1;
$chart['data']['x']['density'] = $chart['data']['x']['range']/$chart['graph']['width'];
$mon_event_slots = array();
$mon_frame_slots = array();
$monitor_ids = array();
foreach( dbFetchAll( $events_sql ) as $event )
{
if ( !isset($monitor_ids[$event['MonitorId']]) )
$monitor_ids[$event['MonitorId']] = true;
if ( !isset($mon_event_slots[$event['MonitorId']]) )
$mon_event_slots[$event['MonitorId']] = array();
if ( !isset($mon_frame_slots[$event['MonitorId']]) )
$mon_frame_slots[$event['MonitorId']] = array();
$curr_event_slots = &$mon_event_slots[$event['MonitorId']];
$curr_frame_slots = &$mon_frame_slots[$event['MonitorId']];
$start_time_t = strtotime($event['StartTime']);
$start_index = $raw_start_index = (int)(($start_time_t - $chart['data']['x']['lo']) / $chart['data']['x']['density']);
if ( $start_index < 0 )
$start_index = 0;
if ( isset($event['EndTime']) )
{
$end_time_t = strtotime($event['EndTime']);
}
else
{
$end_time_t = time();
}
$end_index = $raw_end_index = (int)(($end_time_t - $chart['data']['x']['lo']) / $chart['data']['x']['density']);
if ( $end_index >= $chart['graph']['width'] )
$end_index = $chart['graph']['width'] - 1;
for ( $i = $start_index; $i <= $end_index; $i++ )
{
if ( !isset($curr_event_slots[$i]) )
{
if ( $raw_start_index == $raw_end_index )
{
$offset = 1;
}
else
{
$offset = 1+ ($event['Frames']?((int)($event['Frames']*(($i-$raw_start_index)/($raw_end_index-$raw_start_index)))):0);
}
$curr_event_slots[$i] = array( "count"=>0, "width"=>1, "offset"=>$offset, "event"=>$event );
}
else
{
$curr_event_slots[$i]['count']++;
}
}
if ( $event['MaxScore'] > 0 )
{
if ( $start_index == $end_index )
{
$i = $start_index;
if ( !isset($curr_frame_slots[$i]) )
{
$curr_frame_slots[$i] = array( "count"=>1, "value"=>$event['MaxScore'], "event"=>$event );
}
else
{
$curr_frame_slots[$i]['count']++;
if ( $event['MaxScore'] > $curr_frame_slots[$i]['value'] )
{
$curr_frame_slots[$i]['value'] = $event['MaxScore'];
$curr_frame_slots[$i]['event'] = $event;
}
}
if ( $event['MaxScore'] > $chart['data']['y']['hi'] )
{
$chart['data']['y']['hi'] = $event['MaxScore'];
}
}
else
{
$frames_sql = "select F.FrameId,F.Delta,unix_timestamp(F.TimeStamp) as TimeT,F.Score from Frames as F where F.EventId = '".$event['Id']."' and F.Score > 0";
foreach( dbFetchAll( $frames_sql ) as $frame )
{
$frame_time_t = $frame['TimeT'];
$frame_time_t = $start_time_t + $frame['Delta'];
$frame_index = (int)(($frame_time_t - $chart['data']['x']['lo']) / $chart['data']['x']['density']);
if ( $frame_index < 0 )
continue;
if ( $frame_index >= $chart['graph']['width'] )
continue;
if ( !isset($curr_frame_slots[$frame_index]) )
{
$curr_frame_slots[$frame_index] = array( "count"=>1, "value"=>$frame['Score'], "event"=>$event, "frame"=>$frame );
}
else
{
$curr_frame_slots[$frame_index]['count']++;
if ( $frame['Score'] > $curr_frame_slots[$frame_index]['value'] )
{
$curr_frame_slots[$frame_index]['value'] = $frame['Score'];
$curr_frame_slots[$frame_index]['event'] = $event;
$curr_frame_slots[$frame_index]['frame'] = $frame;
}
}
if ( $frame['Score'] > $chart['data']['y']['hi'] )
{
$chart['data']['y']['hi'] = $frame['Score'];
}
}
}
}
}
ksort($monitor_ids,SORT_NUMERIC);
ksort($mon_event_slots,SORT_NUMERIC);
ksort($mon_frame_slots,SORT_NUMERIC);
// Add on missing frames
$xcount = 0;
foreach( array_keys($mon_frame_slots) as $monitor_id )
{
unset( $curr_frame_slots );
$curr_frame_slots = &$mon_frame_slots[$monitor_id];
for ( $i = 0; $i < $chart['graph']['width']; $i++ )
{
if ( isset($curr_frame_slots[$i]) )
{
if ( !isset($curr_frame_slots[$i]['frame']) )
{
$xcount++;
$frames_sql = "select F.FrameId,F.Score from Frames as F where F.EventId = '".$curr_frame_slots[$i]['event']['Id']."' and F.Score > 0 order by F.FrameId limit 0,1";
$curr_frame_slots[$i]['frame'] = dbFetchOne( $frames_sql );
}
}
}
}
$chart['data']['y']['range'] = ($chart['data']['y']['hi'] - $chart['data']['y']['lo']) + 1;
$chart['data']['y']['density'] = $chart['data']['y']['range']/$chart['graph']['height'];
$maj_y_scale = getYScale( $chart['data']['y']['range'], $chart['grid']['y']['major']['min'], $chart['grid']['y']['major']['max'] );
//print_r( $maj_y_scale );
$max_width = 0;
$max_height = 0;
foreach ( array_keys($monitor_ids) as $monitor_id )
{
if ( $max_width < $monitors[$monitor_id]['Width'] )
$max_width = $monitors[$monitor_id]['Width'];
if ( $max_height < $monitors[$monitor_id]['Height'] )
$max_height = $monitors[$monitor_id]['Height'];
}
// Optimise boxes
foreach( array_keys($mon_event_slots) as $monitor_id )
{
unset( $curr_event_slots );
$curr_event_slots = &$mon_event_slots[$monitor_id];
for ( $i = 0; $i < $chart['graph']['width']; $i++ )
{
if ( isset($curr_event_slots[$i]) )
{
//if ( isset($curr_slot) && (($curr_slot['width'] < $min_event_width) || (($curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id']) && ($curr_slot['frame']['FrameId'] == $curr_event_slots[$i]['frame']['FrameId'])) ) )
//if ( isset($curr_slot) && ($curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id']) )
if ( isset($curr_slot) )
{
if ( $curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id'] )
{
if ( $curr_slot['width'] < $max_event_width )
{
// Merge slots for the same long event
$curr_slot['width']++;
unset( $curr_event_slots[$i] );
continue;
}
elseif ( $curr_slot['offset'] < $curr_event_slots[$i]['offset'] )
{
// Split very long events
$curr_event_slots[$i]['frame'] = array( 'FrameId'=>$curr_event_slots[$i]['offset'] );
}
}
elseif ( $curr_slot['width'] < $min_event_width )
{
// Merge multiple small events
$curr_slot['width']++;
unset( $curr_event_slots[$i] );
continue;
}
}
$curr_slot = &$curr_event_slots[$i];
//if ( isset($curr_slot) && ($curr_slot['width'] < $min_event_width || ($curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id']) ) )
//{
//$curr_slot['width']++;
//unset( $curr_event_slots[$i] );
//}
//else
//{
//$curr_slot = &$curr_event_slots[$i];
//}
}
else
{
unset( $curr_slot );
}
}
if ( isset( $curr_slot ) )
unset( $curr_slot );
}
// Stack events
$frame_slots = array();
$frame_monitor_ids = array_keys($mon_frame_slots);
for ( $i = 0; $i < $chart['graph']['width']; $i++ )
{
foreach ( $frame_monitor_ids as $frame_monitor_id )
{
unset( $curr_frame_slots );
$curr_frame_slots = &$mon_frame_slots[$frame_monitor_id];
if ( isset($curr_frame_slots[$i]) )
{
if ( !isset($frame_slots[$i]) )
{
$frame_slots[$i] = array();
$frame_slots[$i][] = &$curr_frame_slots[$i];
}
else
{
$slot_count = count($frame_slots[$i]);
for ( $j = 0; $j < $slot_count; $j++ )
{
if ( $curr_frame_slots[$i]['value'] > $frame_slots[$i][$j]['value'] )
{
for ( $k = $slot_count; $k > $j; $k-- )
{
$frame_slots[$i][$k] = $frame_slots[$i][$k-1];
}
$frame_slots[$i][$j] = &$curr_frame_slots[$i];
break 2;
}
}
$frame_slots[$i][] = &$curr_frame_slots[$i];
}
}
}
}
//print_r( $mon_event_slots );
//print_r( $mon_frame_slots );
//print_r( $chart );
preg_match( '/^(\d+)-(\d+)-(\d+) (\d+):(\d+)/', $min_time, $start_matches );
preg_match( '/^(\d+)-(\d+)-(\d+) (\d+):(\d+)/', $max_time, $end_matches );
if ( $start_matches[1] != $end_matches[1] )
{
// Different years
$title = strftime( STRF_TL_AXIS_RANGE_YEAR1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_YEAR2, $chart['data']['x']['hi'] );
}
elseif ( $start_matches[2] != $end_matches[2] )
{
// Different months
$title = strftime( STRF_TL_AXIS_RANGE_MONTH1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_MONTH2, $chart['data']['x']['hi'] );
}
elseif ( $start_matches[3] != $end_matches[3] )
{
// Different dates
$title = strftime( STRF_TL_AXIS_RANGE_DAY1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_DAY2, $chart['data']['x']['hi'] );
}
else
{
// Different times
$title = strftime( STRF_TL_AXIS_RANGE_TIME1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_TIME2, $chart['data']['x']['hi'] );
}
function getDateScale( $scales, $range, $min_lines, $max_lines )
{
foreach ( $scales as $scale )
{
$align = isset($scale['align'])?$scale['align']:1;
$scale_range = (int)($range/($scale['factor']*$align));
//echo "S:".$scale['name'].", A:$align, SR:$scale_range
";
if ( $scale_range >= $min_lines )
{
$scale['range'] = $scale_range;
break;
}
}
if ( !isset($scale['range']) )
{
$scale['range'] = (int)($range/($scale['factor']*$align));
}
$scale['divisor'] = 1;
while ( ($scale['range']/$scale['divisor']) > $max_lines )
{
$scale['divisor']++;
}
$scale['lines'] = (int)($scale['range']/$scale['divisor']);
return( $scale );
}
function getYScale( $range, $min_lines, $max_lines )
{
$scale['range'] = $range;
$scale['divisor'] = 1;
while ( $scale['range']/$scale['divisor'] > $max_lines )
{
$scale['divisor']++;
}
$scale['lines'] = (int)(($scale['range']-1)/$scale['divisor'])+1;
return( $scale );
}
function drawXGrid( $chart, $scale, $label_class, $tick_class, $grid_class, $zoom_class=0 )
{
global $zmSlangZoomIn;
ob_start();
$label_count = 0;
$last_tick = 0;
unset( $last_label );
$label_check = isset($scale['label_check'])?$scale['label_check']:$scale['label'];
for ( $i = 0; $i < $chart['graph']['width']; $i++ )
{
$x = $i - 1;
$time_offset = (int)($chart['data']['x']['lo'] + ($i * $chart['data']['x']['density']));
if ( $scale['align'] > 1 )
{
$label = (int)(strftime( $label_check, $time_offset )/$scale['align']);
}
else
{
$label = strftime( $label_check, $time_offset );
}
if ( !isset($last_label) || ($last_label != $label) )
{
$label_count++;
}
if ( $label_count >= $scale['divisor'] )
{
$label_count = 0;
if ( isset($last_label) )
{
if ( $label_class )
{
?>