Основы PHP
  Что такое PHP?
  Возможности PHP
  Преимущества PHP
  История развития
  Что нового в PHP5?
  «Движок» PHP
  Переход на PHP 5.3
New Переход на PHP 5.6
  Введение в PHP
  Изучение PHP
  Основы CGI
  Синтаксис PHP
  Типы данных PHP
  Переменные в PHP
  Константы PHP
  Выражения PHP
  Операторы PHP
  Конструкции PHP
  Ссылки в PHP
  PHP и ООП
  Безопасность
  Функции PHP
  Функции по категориям
  Функции по алфавиту
  Стандартные функции
  Пользовательские
  PHP и HTTP
  Работа с формами
  PHP и Upload
  PHP и Cookies
  PHP и базы данных
  PHP и MySQL
  Документация MySQL
  Учебники
  Учебники по PHP
  Учебники по MySQL
  Другие учебники
  Уроки PHP
  Введение
  Самые основы
  Управление
  Функции
  Документация
  Математика
  Файлы
  Основы SQL
  Дата и время
  CURL
  Изображения
  Стили
  Безопасность
  Установка
  Проектирование БД
  Регулярные выражения
  Подготовка к работе
  Быстрый старт
  Установка PHP
  Установка MySQL
  Конфигурация PHP
  Download / Скачать
  Скачать Apache
  Скачать PHP
  Скачать PECL
  Скачать PEAR
  Скачать MySQL
  Редакторы PHP
  Полезные утилиты
  Документация
  PHP скрипты
  Скачать скрипты
  Инструменты
  PHP в примерах
  Новости портала
 Главная   »  Сборник статей
 
 

Заливка сложной области

Автор: mike (www.codenet.ru)

Для заливки сложной замкнутой области удобно использовать функцию imagefilledpolygon:

int imagefilledpolygon ( resource image, array points, int num_points, int color )

где,

  • image - идентификатор изображения;
  • points - массив точек;
  • num_points - количество точек в полигоне;
  • color - цвет заливки.

Массив точек содержит X и Y координату каждой точки. Таким образом, для трех точек, массив содержит шесть элементов: Array( X1, Y1, X2, Y2, X3, Y3)

Пример 1. Вывод самого простого полигона (треугольник):

<?php
header ("Content-type: image/png");
$im = imagecreatetruecolor(320, 240);
$ink = imagecolorallocate($im, 255, 255, 255);

imagefilledpolygon($im, Array(
	100,100,
	120,180,
	210,160,
	), 3, $ink);

imagepng($im);
imagedestroy($im);
?>

Результат работы этой программы выглядит следующим образом:

1

Эта функция удобна тем, что линии полигона могут пересекаться, и заливка при этом работает корректно:

2

Карта России

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

Задача на первый взгляд сложная, так как контуры некоторых регионов России имеют неправильную форму и использовать функцию imagefill нельзя. По этому, я использовал функцию imagepolygon для прорисовки контуров, и функцию imagefilledpolygon для закрашивания регионов.

Метод, которым я нарисовал карту рассматривался в уроке 6.

Пример 2. Рисование карты России:

<?php
// Ширина и высота изображения
$W=500;
$H=375;

// Функция выводит контуры региона и закрашивает его внутреннюю часть

// $im       - идентификатор изображения
// $filename - имя файла содержащего контур региона (в формате
//             Adobe Illustrator  
// $maxw     - максимальное значение координаты X в файле с контурами
//             России (необходимо для того, чтобы все регионы
//             рисовались с одним масштабом)
// $maxh     - максимальное значение координаты Y в файле с контурами
//             России
// $color    - цвет, в который будет закрашиваться регион

function DrawIllustratorFile($im,$filename,$maxw,$maxh,$color) {
	GLOBAL $W,$H;

	// Чтения файла
	$d=file($filename);

	// Если массив $d содержит только один элемент,
	// то в качестве переноса строк используется символ
	// возврата каретки, и нам необходимо разбить текст
	// на строку вручную
	if (count($d)==1) $d=explode("\r",$d[0]);

	// С помощью регулярного выражение выберем координаты
	// всех точек.

	$poly=Array();
	for ($i=0;$i<count($d);$i++)
		if (eregi("([0-9.]+) ([0-9.]+) [lm]",$d[$i],$r)) {
			$poly[]=$r[1];
			$poly[]=$r[2];
			}

	for ($i=0;$i<count($poly);$i+=2) {
		// Нормализуем координаты
		$poly[$i]/=$maxw;
		$poly[$i+1]/=$maxh;

		// Отмасштабируем координаты
		$poly[$i]*=($W-10);
		$poly[$i+1]*=($H-10);

		// Перевернем изображение по вертикали
		$poly[$i+1]=$H-$poly[$i+1];
		}

	// Вывод полигона
	imagefilledpolygon($im, $poly, count($poly)/2, $color);
	imagepolygon($im, $poly, count($poly)/2, $color["black"]);
	}

// Чтения файла
$d=file("russia.ai");

// Если массив $d содержит только один элемент,
// то в качестве переноса строк используется символ
// возврата каретки, и нам необходимо разбить текст
// на строку вручную
if (count($d)==1) $d=explode("\r",$d[0]);

// С помощью регулярного выражение выберем координаты
// всех точек и найдем точки с самыми большими координатами.
// Эти значения нам понадобятся для нормализации.
$maxw=$maxh=0;
for ($i=0;$i<count($d);$i++)
	if (eregi("([0-9.]+) ([0-9.]+) [lm]",$d[$i],$r)) {
		if ($maxw<$r[1]) $maxw=$r[1];
		if ($maxh<$r[2]) $maxh=$r[2];
		}

// В отличи от урока 6, в одном файле может быть несколько
// полигонов. Для хранения координат их вершин, будем использовать
// двухмерный массив

// С помощью регулярного выражение выберем координаты
// всех полигонов. Формат Adobe Illustrator, прост, 
// при условии использования прямых линий
$points=Array();
$num=-1;
for ($i=0;$i<count($d);$i++) {

	// Первая точка в полигоне
	if (eregi("([0-9.]+) ([0-9.]+) [m]",$d[$i],$r)) {
		$num++;
		$points[$num][]=$r[1];
		$points[$num][]=$r[2];
		}

	// Не первая точка в полигоне
	if (eregi("([0-9.]+) ([0-9.]+) [l]",$d[$i],$r)) {
		$points[$num][]=$r[1];
		$points[$num][]=$r[2];
		}
	}

// Создадим изображение и выделим цвета
header ("Content-type: image/png");
$im = imagecreatetruecolor($W, $H);
$bg = imagecolorallocate($im, 255, 255, 255);
imagefilledrectangle($im,0,0,imagesx($im),imagesy($im),$bg);

$ink=imagecolorallocate($im,0,0,0);

// Вывод полигонов
for ($j=0;$j<count($points);$j++) {

	// Перерассчитанные координаты полигона
	// будем хранить в массиве $poly
	$poly=Array();

	for ($i=0;$i<count($points[$j]);$i+=2) {
		$x=$points[$j][$i];
		$y=$points[$j][$i+1];

		// Нормализуем координаты
		$x=$x/$maxw;
		$y=$y/$maxh;

		// Отмасштабируем координаты
		$x=$x*($W-10);
		$y=$y*($H-10);

		// Перевернем изображение по вертикали
		$y=$H-$y;

		// Заносим координаты в массив, по которому 
		// будет построен полигон
		$poly[]=$x;
		$poly[]=$y;
		}

	// Вывод полигона
	imagepolygon($im, $poly, count($poly)/2, $ink);
	}

// Выделим цвета для заливки регионов
$color=Array();
for ($i=196;$i<256;$i+=6) $color[]=imagecolorallocate($im,$i,$i,$i);
$color["black"]=$ink;

// Прорисовка и закрашивание регионов
DrawIllustratorFile($im,"arhangelsk.ai",$maxw,$maxh,$color[7]);
DrawIllustratorFile($im,"murmansk.ai",$maxw,$maxh,$color[5]);
DrawIllustratorFile($im,"novgorod.ai",$maxw,$maxh,$color[0]);
DrawIllustratorFile($im,"pertrozavodsk.ai",$maxw,$maxh,$color[3]);
DrawIllustratorFile($im,"pskov.ai",$maxw,$maxh,$color[8]);
DrawIllustratorFile($im,"s-petersburg.ai",$maxw,$maxh,$color[9]);
DrawIllustratorFile($im,"syktyvkar.ai",$maxw,$maxh,$color[3]);
DrawIllustratorFile($im,"vologda.ai",$maxw,$maxh,$color[4]);

imagepng($im);
imagedestroy($im);
?>

Так выглядит карта России до закрашивания регионов:

3

А окончательный результат работы этой программы выглядит следующим образом: (Терпения мне хватило только на обводку областей северо-западного федерального округа)

4

Скачать PHP скрипт рисующий карту России можно здесь.

На самом деле, обводить контур страны не обязательно, так как его построят сложенные вместе регионы. Я сделал это лишь из-за того что не нашел времени обвести все регионы.

Этот метод, в сочетании со сглаживанием изображения, я использовал для рисования карты мира на сайте http://top.novgorod.ru/. Вот как она выглядит:

5

Ссылки по теме

 
 » Обсудить эту статью на форуме

 
 Сборник статей 
 Содержание раздела 
Есть еще вопросы или что-то непонятно - добро пожаловать на наш  форум портала PHP.SU 
 

 
Powered by PHP  Powered By MySQL  Powered by Nginx  Valid CSS