JfreeChart ограничивает количество отображаемых элементов легенды - PullRequest
0 голосов
/ 11 января 2012

Я отвечаю за поддержку приложения, которое может рисовать графики, используя JFreeChart. Приложение написано на Eclipse-RCP и SWT и использует ChartComposite для отображения графиков.

ChartComposite был частично переопределен для настройки контекстных меню в зависимости от выбора:

@Override
    public void createPartControl(Composite parent) {
    super.createPartControl(parent);

    chart = createChart(timeSeriesDataset);

    chartComposite = new MyChartComposite(this, parent, SWT.NONE, chart, true);
    chartComposite.setLayoutData(new GridData(GridData.FILL_BOTH));

    selectionProvider = new GenericObjectSelectionProvider();
    getSite().setSelectionProvider(selectionProvider);

    // add information to the status line:
    selectionProvider.addSelectionChangedListener(statusLineListener);

    addDropSupport();// add D'n D support for dropping TimeSeries 

}


protected JFreeChart createChart(TimeSeriesCollection ptimeSeriesDataset) {

        JFreeChart vChart = ChartFactory.createTimeSeriesChart(null, "time", "values", ptimeSeriesDataset, true,
                false, false);
        vChart.setBackgroundPaint(Color.white);

        XYPlot plot = vChart.getXYPlot();
        plot.setBackgroundPaint(Color.lightGray);
        plot.setDomainGridlinePaint(Color.white);
        plot.setRangeGridlinePaint(Color.white);
        // plot.setAxisOffset(new RectangleInsets(5.0, 5.0, 5.0, 5.0));
        plot.setDomainCrosshairVisible(true);
        plot.setRangeCrosshairVisible(true);

        plot.setRenderer(new /*OptimisedXYLineAndShapeRenderer()*/ StandardXYItemRendererFast());
        XYItemRenderer renderer = plot.getRenderer();
        renderer.setBaseToolTipGenerator(new MyXYSeriesToolTipGenerator());
        renderer.setBaseItemLabelGenerator(new MyXYSeriesItemLabelGenerator());
        renderer.setLegendItemLabelGenerator(new MyXYSeriesLegendItemLabelGenerator());
        if (renderer instanceof XYLineAndShapeRenderer) {
            XYLineAndShapeRenderer r = (XYLineAndShapeRenderer) renderer;
            r.setBaseShapesVisible(false);
            r.setBaseShapesFilled(true);
        }

        SimpleDateFormat dateFormat = getDateFormatAbscissa();
        if (dateFormat != null){
            DateAxis axis = (DateAxis) plot.getDomainAxis();
            axis.setDateFormatOverride(dateFormat);
        }   

        return vChart;
    }

Моя проблема в том, что, когда на диаграмму добавляется слишком много переменных (TimeSeriesChart), заголовок занимает слишком много места, и график исчезает из вида:

ChartComposite с 2 сериями

ДиаграммаСостав многих серий

Я попытался создать ScrollComposite для прокрутки в ChartComposite, и результат получился немного лучше; но это только позволяет добавить больше элементов в заголовок, прежде чем график снова исчезнет:

ScrolledComposite scrollableChart = new ScrolledComposite(parent, SWT.BORDER|SWT.V_SCROLL);

chartComposite = new MyChartComposite(this, scrollableChart, SWT.NONE, chart, true);
//chartComposite = new MyChartComposite(this, parent, SWT.NONE, chart, true);
//chartComposite.setLayoutData(new GridData(GridData.FILL_BOTH));

scrollableChart.setContent(chartComposite);
scrollableChart.setExpandVertical(true);
scrollableChart.setExpandHorizontal(true);
scrollableChart.setMinSize(ChartComposite.DEFAULT_MINIMUM_DRAW_WIDTH, ChartComposite.DEFAULT_MINIMUM_DRAW_WIDTH);

У меня такой вопрос: как предоставить реальную полосу прокрутки для ChartComposite, чтобы сохранить график, когда на нем изображено много рядов?

Ответы [ 2 ]

2 голосов
/ 12 января 2012

Мне удалось синхронизировать ползунок с XYSeries в SWT с помощью объектов ChartComosite и Slider с помощью FormData.И каждый раз, когда я перемещаю ползунок, я фиксирую это событие и обновляю график самостоятельно в соответствии с моими потребностями.

Мой вариант использования может отличаться от вашего, но стоит взглянуть на мой ответ здесь .

Если у вас есть вопросы относительно моей реализации, описанной вответьте, не стесняйтесь спрашивать подробности

1 голос
/ 24 января 2012

После нескольких неудачных попыток я решил ограничить число LegendItem, показанное на графике.

Чтобы изменить элементы легенды для отображения в LegendTitle, я использовал ползунок. Самым сложным было воссоздать новый LegendTitle при использовании ползунка; Я не уверен, что мое решение оптимально или изящно, но, по крайней мере, оно работает.

Result

Чтобы сделать это возможным, мне нужно было прослушать создание серии (ChartChangeEventType.DATASET_UPDATED), чтобы обновить LegendTitle и установить значения ползунка.

Так вот код:

public class MyExampleChartComposite extends ChartComposite {
    // snip
    /**
     * A slider to choose the legend items to display
     */
     private Slider legendSlider;
     /**
     * Number of legend items to display on the chart
     */
     private final static int NUMBER_OF_LEGENDITEMS_TO_DISPLAY = 10;

     private void createPartControl(Composite parent, int style) {
         JFreeChart chart = createChart();
         setChart(chart);
         legendSlider = new Slider(parent, SWT.NONE|SWT.H_SCROLL);
         legendSlider.addSelectionListener(new SelectionListener() {
             @Override
             public void widgetSelected(SelectionEvent arg0) {
                      refreshLegend();
             }
         });

     private JFreeChart createChart() {
         chart.addChangeListener(new ChartChangeListener() {    
             @Override
             public void chartChanged(ChartChangeEvent e) {
                  if (e.getType().equals(ChartChangeEventType.DATASET_UPDATED)) {
                      refreshLegend();
                  }
             }
         });
    }

    /**
     * Refresh the the LegendItems and the slider,
     * according to slider selection.
     */
    public void refreshLegend() {
        // Display LegendItems according to the selection
        // display the 10 nearest legend item (current selected element included)
        int begin = legendSlider.getSelection() - (NUMBER_OF_LEGENDITEMS_TO_DISPLAY/2);
        int end = legendSlider.getSelection() + (NUMBER_OF_LEGENDITEMS_TO_DISPLAY/2 -1);
        int seriesEndIndex = Math.max(getSeriesCount()-1, 0);
        // if begin is less than 0
        // set it to 0, and increase end to display 10 items
        if (begin < 0) {
            begin = 0; 
            end = NUMBER_OF_LEGENDITEMS_TO_DISPLAY - 1;
        }
        // if end is greater than the number of series plotted
        // set it to the max possible value and increase begin to
        // display 10 items
        if  (end > seriesEndIndex) {
            end = seriesEndIndex;  
            begin = seriesEndIndex - (NUMBER_OF_LEGENDITEMS_TO_DISPLAY - 1);
        }
        end = Math.min(seriesEndIndex, end);
        begin = Math.max(begin, 0);
        // Refresh only if begin != end
        if (end != begin) {
            refreshLegendItems(begin, end);
            refreshLegendSlider();
        } else {
            // in this case no more series are plotted on the chart
            // clear legend
            getChart().clearSubtitles();
        }
    }
    /**
     * Refresh the LegendTitle.
     * Display only LegendItems between beginIndex and toIndex,
     * to preserve space for the chart.
     * @param beginIndex index of the {@link LegendItemCollection} used as the beginning of the new {@link LegendTitle}
     * @param endIndex index of the {@link LegendItemCollection} used as the end of the new {@link LegendTitle}
     */
    private void refreshLegendItems(int beginIndex, int endIndex) {
        // Last 10 items
        final LegendItemCollection result = new LegendItemCollection();
        // get the renderer to retrieve legend items
        XYPlot plot = getChart().getXYPlot();
        XYItemRenderer renderer = plot.getRenderer();
        // Number of series displayed on the chart 
        // Construct the legend
        for (int i = beginIndex; i <= endIndex; i++) {
            LegendItem item = renderer.getLegendItem(0, i);
            result.add(item);
        }
        // Because the only way to create a new LegendTitle is to
        // create a LegendItemSource first
        LegendItemSource source = new LegendItemSource() {
            LegendItemCollection lic = new LegendItemCollection();
            {lic.addAll(result);}
            public LegendItemCollection getLegendItems() {  
                return lic;
            }
        };
        // clear previous legend title
        getChart().clearSubtitles();
        // Create the new LegendTitle and set its position
        LegendTitle legend = new LegendTitle(source);
        legend.setHorizontalAlignment(HorizontalAlignment.CENTER);
        legend.setVerticalAlignment(VerticalAlignment.CENTER);
        legend.setPosition(RectangleEdge.BOTTOM);
        legend.setBorder(1.0, 1.0, 1.0, 1.0);
        legend.setVisible(true);
        // Add the LegendTitle to the graph
        getChart().addLegend(legend);
    }

    /**
     * Set values of the slider according to the number of series
     * plotted on the graph
     */
    private void refreshLegendSlider() {
        int max = getSeriesCount() -1;
        int selection = Math.min(legendSlider.getSelection(), max);
        legendSlider.setValues(selection, 0, max, 1, 1, 1);
    }
}
...