Нарисуйте линию с толщиной на изображении?

Я хочу нарисовать линию с толщиной на изображении с помощью PHP, используя библиотеку GD. На этой странице я нашел несколько решений PHP: imageline - Manual, но ни одно из них, кажется, не работает правильно, когда изменяется положение (x, y) строки.

Есть 3 функции, которые я нашел на странице

function dickelinie($img, $start_x, $start_y, $end_x, $end_y, $color, $thickness)
{
    $angle = (atan2(($start_y - $end_y), ($end_x - $start_x)));

    $dist_x = $thickness * (sin($angle));
    $dist_y = $thickness * (cos($angle));

    $p1x = ceil(($start_x + $dist_x));
    $p1y = ceil(($start_y + $dist_y));
    $p2x = ceil(($end_x + $dist_x));
    $p2y = ceil(($end_y + $dist_y));
    $p3x = ceil(($end_x - $dist_x));
    $p3y = ceil(($end_y - $dist_y));
    $p4x = ceil(($start_x - $dist_x));
    $p4y = ceil(($start_y - $dist_y));

    $array = array(0 => $p1x, $p1y, $p2x, $p2y, $p3x, $p3y, $p4x, $p4y);
    imagefilledpolygon ($img,  $array, (count($array) / 2), $color);
}

function imagelinethick($image, $x1, $y1, $x2, $y2, $color, $thick = 1)
{
    if ($thick == 1)
    {
        return imageline($image, $x1, $y1, $x2, $y2, $color);
    }

    $t = $thick / 2 - 0.5;

    if ($x1 == $x2 || $y1 == $y2)
    {
        return imagefilledrectangle($image, round(min($x1, $x2) - $t), round(min($y1, $y2) - $t), round(max($x1, $x2) + $t), round(max($y1, $y2) + $t), $color);
    }

    $k = ($y2 - $y1) / ($x2 - $x1);
    $a = $t / sqrt(1 + pow($k, 2));

    $points = array(
        round($x1 - (1+$k)*$a), round($y1 + (1-$k)*$a),
        round($x1 - (1-$k)*$a), round($y1 - (1+$k)*$a),
        round($x2 + (1+$k)*$a), round($y2 - (1-$k)*$a),
        round($x2 + (1-$k)*$a), round($y2 + (1+$k)*$a),
    );

    imagefilledpolygon($image, $points, 4, $color);
    return imagepolygon($image, $points, 4, $color);
}

function imagelinethick1($image, $x1, $y1, $x2, $y2, $color, $thick = 1)
{
    imagesetthickness($image, $thick);
    imageline($image, $x1, $y1, $x2, $y2, $color);
}

И мой тестовый случай

header("Content-Type: image/png");
$image = @imagecreatetruecolor(500, 500) or die("Cannot initialize new GD image stream");
$color = imagecolorallocate($image, 255, 255, 255);

# Line thickness equals to 18 pixels
$thickness = 18;

# OK
dickelinie($image, 0, 0, 0, 500, $color, $thickness);

# Wrong: The thickness of the line is doubled
dickelinie($image, 200, 0, 200, 500, $color, $thickness);

# Wrong: The thickness of the line is halved
imagelinethick($image, 0, 0, 0, 500, $color, $thickness);

# OK
imagelinethick($image, 200, 0, 200, 500, $color, $thickness);

# Wrong: The thickness of the line is halved
imagelinethick1($image, 0, 0, 0, 500, $color, $thickness);

# OK
imagelinethick1($image, 200, 0, 200, 500, $color, $thickness);

imagepng($image);
imagedestroy($image);

Кто-нибудь может сказать мне, что не так, пожалуйста?

3 ответа

Это не самый быстрый способ решить вашу проблему, но я использую эту функцию в моем проекте:

function thickline( $img, $x1, $y1, $x2, $y2, $color, $thickness ) {
  $radius = $thickness * .5;
  $vx = $x2 - $x1;
  $vy = $y2 - $y1;
  $steps = ceil( .5 + max( abs($vx), abs($vy) ) );
  $vx /= $steps;
  $vy /= $steps;
  $x = $x1;
  $y = $y1;
  while( $steps --> 0 ) {
    imagefilledellipse( $img, $x, $y, $radius, $radius, $color );
    $x += $vx;
    $y += $vy;
  }
}

Идея состоит в том, чтобы нанести круг на каждую точку линии.

Некоторые из ваших тестов неверны. Каждый тестовый случай, который визуализируется с x = 0, будет разрезан пополам, поскольку подпрограммы будут центрировать новую более толстую линию по заданным координатам xy. Таким образом, половина отображаемой строки отображается за пределами растрового изображения и теряется.

Учитывая, что ваши тестовые примеры немного ошибочны, мы можем видеть, что метод "dickelinie" всегда удваивает ожидаемую толщину.

Я по-прежнему использовал метод "dickelinie" для рендеринга простых аналоговых часов, поскольку они давали наилучшие результаты для моих нужд. Я только что прошел половину толщины, чтобы убедиться, что это работает. Я также добавил следующий код в конец метода "dickelinie", чтобы гарантировать, что полученная строка будет сглажена, когда указано:

return imagepolygon($img,  $array, (count($array) / 2), $color);

Вы захотите этого, если вам нужно визуализировать линии под разными углами.

Пример # 1 для php.net imageline "Рисование толстой линии". Он предлагает использовать либо imagesetthickness или следующий код:

<?php
function imagelinethick($image, $x1, $y1, $x2, $y2, $color, $thick = 1)
{
    /* this way it works well only for orthogonal lines
    imagesetthickness($image, $thick);
    return imageline($image, $x1, $y1, $x2, $y2, $color);
    */
    if ($thick == 1) {
        return imageline($image, $x1, $y1, $x2, $y2, $color);
    }
    $t = $thick / 2 - 0.5;
    if ($x1 == $x2 || $y1 == $y2) {
        return imagefilledrectangle($image, round(min($x1, $x2) - $t), round(min($y1, $y2) - $t), round(max($x1, $x2) + $t), round(max($y1, $y2) + $t), $color);
    }
    $k = ($y2 - $y1) / ($x2 - $x1); //y = kx + q
    $a = $t / sqrt(1 + pow($k, 2));
    $points = array(
        round($x1 - (1+$k)*$a), round($y1 + (1-$k)*$a),
        round($x1 - (1-$k)*$a), round($y1 - (1+$k)*$a),
        round($x2 + (1+$k)*$a), round($y2 - (1-$k)*$a),
        round($x2 + (1-$k)*$a), round($y2 + (1+$k)*$a),
    );
    imagefilledpolygon($image, $points, 4, $color);
    return imagepolygon($image, $points, 4, $color);
}
?>
Другие вопросы по тегам