2

少し背景

Symfony1.2で構築されたプロジェクトをあるサーバーから別のサーバーに移行しようとしています。プロジェクトの機能の1つは、グラフを作成することです(元々はJpGraph 2.3.5で実行されていました)。

グラフは、コードを変更しないと意図したとおりに表示されず、私が見落としている可能性のあるものについての洞察を探しています。 投稿するポイントが足りないため、画像がリンクされています。 グラフ画像ギャラリー

次のグラフは、以下のコードブロックによって生成されたものです

<?php
    public function Graph($section) {
        $report = $section->getReport();
        $this->crews = array();
        foreach ($section->getCrews() as $crew) {
            $this->crews[$crew->getId()] = $crew;
        };

        # get the data
        $nextDayValues = $section->getNextDayValues();
        $nextDayValueLabels = $section->getNextDayValueLabels();

        $max_y = max($nextDayValues) < 7 ? 7 : max($nextDayValues);
        $this->crew_order = array_keys($nextDayValues);
        $this->summary = $this->getSummary();
        $this->bar_count = count($this->crews) + count($this->summary);

        $left    = 200;
        $right   =  30;
        $top     =  60;
        $bottom  =  80;
        $width   = 640;
        $height  = $top + $bottom + ($this->bar_count * 30 );

        $x_unit = $this->bar_count / ($height - $top - $bottom);
        $y_unit = $max_y / ($width - $left - $right);

        $csim_targets = array();
        $csim_alts = array();
        $bar_data = array();
        $max_days = 0;

        foreach ($this->crew_order as $i => $crew_id) {
            $csim_targets[$i] = url_for('units/index?crew_id='.$crew_id);
            $csim_alts[$i] = sprintf("Units for %s",
              $this->crews[$crew_id]->getCrew());

            # figure out the maximum y value
            $nextDayUnitsList = $this->crews[$crew_id]->getNextDayUnitsList();
            $units_array[$crew_id] = $nextDayUnitsList;

            if (count($nextDayUnitsList) > $this->max_days) {
                $this->max_days = count($nextDayUnitsList);
            };
        };

        $bg_values = array_values($nextDayValues);
        foreach ($this->summary as $summary) {
            array_push ($bg_values, $summary['value']);
        };

        $bg_bar = new BarPlot($bg_values);
        $bg_bar->SetCSIMTargets($csim_targets, $csim_alts);
        $bg_bar->SetNoFill(true);

        $fg_bars = $this->getFgBars($units_array);
        $fg_bar = new AccBarPlot($fg_bars);
        $fg_bar->SetFillColor('black');

        # initialize the graph
        $graph = new Graph($width, $height, 'auto');
        $graph->SetScale('textlin', 0, $max_y, 0, $this->bar_count);
        $graph->Set90AndMargin($left, $right, $top, $bottom);
        $graph->SetMarginColor('white');
        $graph->SetFrame(false);

        $graph->Add($fg_bar);
        $graph->Add($bg_bar);

        # add text labels
        foreach ($this->crew_order as $i => $crew_id) {
            $label = $this->value_label(
                $nextDayValueLabels[$crew_id],
                $i, $nextDayValues[$crew_id],
                10 * $x_unit, 5 * $y_unit
            );
            $graph->AddText($label);
        };

        foreach ($this->summary as $i => $summary) {
            $label = $this->value_label(
                $summary['value'],
                $i, $summary['value'],
                10 * $x_unit, 5 * $y_unit
            );
            $graph->AddText($label);
        };

        # add title
        $graph->title->Set(sprintf("%s - %s", $report->getName(),
          $section->getName()));
        $graph->title->SetFont(FF_VERDANA,FS_BOLD, 12);
        $graph->title->SetMargin(10);

        # add subtitle
        $graph->subtitle->Set(date('d-M-Y g:ia'));
        $graph->subtitle->SetFont(FF_VERDANA,FS_BOLD, 8);

        # configure x-axis
        $graph->xaxis->SetFont(FF_VERDANA, FS_NORMAL, 8);
        $graph->xaxis->SetLabelAlign('right', 'center');
        $graph->xaxis->SetLabelFormatCallback(array($this, 'x_axis_label'));
        $graph->xaxis->scale->ticks->Set(1, 0);

        # configure y-axis
        $graph->yaxis->SetFont(FF_VERDANA, FS_NORMAL, 8);
        $graph->yaxis->SetLabelAlign('center', 'top');
        $graph->yaxis->SetLabelAngle(45);
        $graph->yaxis->SetLabelFormatCallback(array($this, 'y_axis_label'));
        $graph->yaxis->SetPos('max');
        $graph->yaxis->SetLabelSide(SIDE_RIGHT);
        $graph->yaxis->SetTickSide(SIDE_LEFT);

        if (max($nextDayValues) > 28) {
            $graph->yaxis->scale->ticks->Set(7, 1);
        } else {
            $graph->yaxis->scale->ticks->Set(1, 1);
        };

        # configure legend
        $graph->legend->SetAbsPos(5, $height - 5, "left", "bottom");
        $graph->legend->SetColumns(count($this->legend));
        $graph->legend->SetFillColor('white');
        $graph->legend->SetShadow(false);

        $graph->SetImgFormat('png');

        return $graph;
    }

    private function getFgBars($units_array) {
        # initialize fg_bar data
        $empty_crews = array_fill_keys(array_keys($this->crew_order),0);

        # add segment bars
        foreach ($this->summary as $summary) {
            $empty_crews[] = 0;
        };

        $empty_segment = array();
        foreach (array_keys($this->legend_colors) as $status) {
            $empty_segment[$status] = $empty_crews;
        };

        $segments = array();
        for ($day = 0; $day < $this->max_days; $day++) {
            foreach (array_keys($empty_segment) as $status) {
                $segment = $empty_segment;
                foreach ($this->crew_order as $i => $crew_id) {
                    $nextDayUnitsList = $units_array[$crew_id];
                    if ($day + 1 < count($nextDayUnitsList)) {
                        $units = $nextDayUnitsList[$day];
                        $units_status = $units->getNextDayStatus();
                        $segment[$units_status][$i] = 1;
                    } elseif ($day + 1 == count($nextDayUnitsList)) {
                        $units = $nextDayUnitsList[$day];
                        $units_status = $units->getNextDayStatus();
                        $avail = $units->getUsedRatio();
                        $segment[$units_status][$i] = $avail;
                    } elseif ($day + 1 > count($nextDayUnitsList)) {
                        $segment[$units_status][$i] = 0;
                    };
                };
            };

            foreach ($this->summary as $i => $summary) {
                $diff = $summary['value'] - $day;
                if ($diff >= 1) {
                    $segment['summary'][$i] = 1;
                } elseif ($diff >= 0) {
                    $segment['summary'][$i] = $diff;
                } else {
                    $segment['summary'][$i] = 0;
                }
            };
            $segments[$day] = $segment;
        };

        # create legend
        $fg_bars = array();
        foreach (array_keys($empty_segment) as $status) {
            $fg_bar = new BarPlot($empty_crews);
            $fg_bar->setFillColor($this->legend_colors[$status]);

            if ($status <> 'summary') {
                $fg_bar->SetLegend($this->legend[$status]);
            };
            $fg_bars[] = $fg_bar;
        };

        # add segments
        foreach ($segments as $day => $segment) {
            foreach (array_keys($empty_segment) as $status) {
                $fg_bar = new BarPlot($segment[$status]);
                $fg_bar->setColor($this->legend_colors[$status]);
                $fg_bar->setFillColor($this->legend_colors[$status]);
                $fg_bars[] = $fg_bar;
            };
        };
        return $fg_bars;
    }
?>

これで、JpGraph2.3.5またはJpGraph3.5.0b1を使用した新しいサーバー上の同じグラフ

いくつかの問題が見られます:

  1. 色は適用されません
  2. x / y座標がオフセットされています(マージン?)
  3. スケールがオフ
  4. 目盛りも削除されます

プロパティを変更する前に、配置した色を修正することを認識して$graph->Add($x);います。そのため、内のコードをに移動する必要がありgetFgBars()ましたGraph()

これは、 /docs/chunkhtml/ch29s02.htmlの下のマニュアルに記載されています

コードを修正することで、FgBarを前景データ(色)と凡例バーに $fg_bars[]分割しました。$lg_bars[]

凡例バーを削除すると、グラフが意図したとおりに表示されます。

質問:

秒を追加するときにマージン/スケールが不安定になる原因は何AccBarPlot()ですか?

解決:

デフォルトのテーマを削除します。$graph->graph_theme = null;コードを書き直すことなく、色と余白の両方を復元します。

    # initialize the graph
    $graph = new Graph($width, $height, 'auto');
    $graph->SetScale('textlin', 0, $max_y, 0, $this->bar_count);
    $graph->Set90AndMargin($left, $right, $top, $bottom);
    $graph->SetMarginColor('white');
    $graph->SetFrame(false);
    $graph->graph_theme = null;

4

1 に答える 1

5

解決:

デフォルトのテーマを削除します。$graph->graph_theme = null;初期化後Graph()、コードを書き直すことなく、色と余白の両方を復元します。

    # initialize the graph
    $graph = new Graph($width, $height, 'auto');
    $graph->SetScale('textlin', 0, $max_y, 0, $this->bar_count);
    $graph->Set90AndMargin($left, $right, $top, $bottom);
    $graph->SetMarginColor('white');
    $graph->SetFrame(false);
    $graph->graph_theme = null;

資力:

/docs/chunkhtml/ch29.html#id2619634-SetTheme()の順序と設定 の変更/docs/chunkhtml/ch29s02.html-線/棒グラフの表示設定の変更

于 2012-05-11T17:02:43.003 に答える