#include "GraphStyle.hxx" #include "GraphData.hxx" #include "Graph.hxx" namespace systemtap { using namespace std; using namespace tr1; typedef pair TimeListPair; GraphStyleBar GraphStyleBar::instance; void GraphStyleBar::draw(std::tr1::shared_ptr graphData, Graph* graph, Cairo::RefPtr cr) { shared_ptr > realData = dynamic_pointer_cast >(graphData); if (!realData) return; int64_t left, right; double top, bottom; graph->getExtents(left, right, top, bottom); double horizScale = (graph->_zoomFactor * graph->_graphWidth / static_cast(right - left)); GraphDataBase::TimeList::iterator lower = lower_bound(graphData->times.begin(), graphData->times.end(), left); GraphDataBase::TimeList::iterator upper = upper_bound(graphData->times.begin(), graphData->times.end(), right); for (GraphDataBase::TimeList::iterator ditr = lower, de = upper; ditr != de; ++ditr) { size_t dataIndex = ditr - graphData->times.begin(); cr->set_source_rgba(graphData->color[0], graphData->color[1], graphData->color[2], 1.0); cr->move_to((*ditr - left) * horizScale, 0); cr->line_to((*ditr - left) * horizScale, realData->data[dataIndex] * graph->_graphHeight / graphData->scale); cr->stroke(); } } ssize_t GraphStyleBar::dataIndexAtPoint(double x, double y, shared_ptr graphData, shared_ptr graph) { shared_ptr > realData = dynamic_pointer_cast >(graphData); if (!realData || graphData->times.empty()) return -1; int64_t left, right; double top, bottom; graph->getExtents(left, right, top, bottom); double t = graph->getTimeAtPoint(x); TimeListPair range = equal_range(graphData->times.begin(), graphData->times.end(), t); if (range.first == graphData->times.end()) return -1; size_t dataIndex = distance(graphData->times.begin(), range.first); double val = realData->data[dataIndex]; double ycoord = val * graph->_graphHeight / graphData->scale; if (y >= graph->_yOffset + graph->_graphHeight - ycoord) return static_cast(dataIndex); else return -1; } GraphStyleDot GraphStyleDot::instance; void GraphStyleDot::draw(std::tr1::shared_ptr graphData, Graph* graph, Cairo::RefPtr cr) { shared_ptr > realData = dynamic_pointer_cast >(graphData); if (!realData) return; int64_t left, right; double top, bottom; graph->getExtents(left, right, top, bottom); double horizScale = (graph->_zoomFactor * graph->_graphWidth / static_cast(right - left)); GraphDataBase::TimeList::iterator lower = lower_bound(graphData->times.begin(), graphData->times.end(), left); GraphDataBase::TimeList::iterator upper = upper_bound(graphData->times.begin(), graphData->times.end(), right); cr->set_source_rgba(graphData->color[0], graphData->color[1], graphData->color[2], 1.0); for (GraphDataBase::TimeList::iterator ditr = lower, de = upper; ditr != de; ++ditr) { size_t dataIndex = ditr - graphData->times.begin(); cr->arc((*ditr - left) * horizScale, (realData->data[dataIndex] * graph->_graphHeight / graphData->scale), graph->_lineWidth / 2.0, 0.0, M_PI * 2.0); cr->fill(); } } GraphStyleEvent GraphStyleEvent::instance; void GraphStyleEvent::draw(std::tr1::shared_ptr graphData, Graph* graph, Cairo::RefPtr cr) { shared_ptr > stringData = dynamic_pointer_cast >(graphData); if (!stringData) return; int64_t left, right; double top, bottom; graph->getExtents(left, right, top, bottom); double horizScale = (graph->_zoomFactor * graph->_graphWidth / static_cast(right - left)); double eventHeight = graph->_graphHeight * (graphData->scale / 100.0); cr->save(); cr->set_line_width(3 * graph->_lineWidth); cr->set_source_rgba(graphData->color[0], graphData->color[1], graphData->color[2], .33); cr->move_to(0, eventHeight); cr->line_to(graph->_graphWidth, eventHeight); cr->stroke(); cr->restore(); GraphDataBase::TimeList::iterator lower = lower_bound(graphData->times.begin(), graphData->times.end(), left); GraphDataBase::TimeList::iterator upper = upper_bound(graphData->times.begin(), graphData->times.end(), right); for (GraphDataBase::TimeList::iterator ditr = lower, de = upper; ditr != de; ++ditr) { // size_t dataIndex = ditr - graphData->times.begin(); double eventHeight = graph->_graphHeight * (graphData->scale / 100.0); cr->save(); cr->rectangle((*ditr - left) * horizScale - 1.5 * graph->_lineWidth, eventHeight - 1.5 * graph->_lineWidth, 3.0 * graph->_lineWidth, 3.0 * graph->_lineWidth); cr->fill(); cr->restore(); } } ssize_t GraphStyleEvent::dataIndexAtPoint(double x, double y, shared_ptr graphData, shared_ptr graph) { shared_ptr > stringData = dynamic_pointer_cast >(graphData); if (!stringData || graphData->times.empty()) return -1; int64_t left, right; double top, bottom; graph->getExtents(left, right, top, bottom); double horizScale = (graph->_zoomFactor * graph->_graphWidth / static_cast(right - left)); double eventHeight = graph->_graphHeight * (graphData->scale / 100.0); GraphDataBase::TimeList::iterator lower = lower_bound(graphData->times.begin(), graphData->times.end(), left); GraphDataBase::TimeList::iterator upper = upper_bound(graphData->times.begin(), graphData->times.end(), right); // easier to transform x,y into graph coordinates double xgraph, ygraph; graph->window2GraphCoords(x, y, xgraph, ygraph); double yrect = eventHeight - 1.5 * graph->_lineWidth; for (GraphDataBase::TimeList::iterator ditr = lower, de = upper; ditr != de; ++ditr) { double xrect = (*ditr - left) * horizScale - 1.5 * graph->_lineWidth; if (xrect <= xgraph && xgraph < xrect + 3.0 * graph->_lineWidth && yrect <= ygraph && ygraph < yrect + 3.0 * graph->_lineWidth) return static_cast(distance(lower, ditr)); } return -1; } }