3 способа добавить watermark (водяной знак) на изображение — php

Сентябрь 25, 2007

Немного порылся в своих старых проектах, нашел целых три способа, как я в свое время реализовывал добавление водяного знака на картинки.

Первое, что хотелось бы сказать: сохраняйте классы и функции, велосипед — это клево, но несколько раз — слишком (в моем случае дело упрощает то, что один из них — трехколесный, еще один — чужой).

Второе: опишу по-подробнее, что да как:

Исходные данные:

замечательный рисунок работы Вила Мюрэя:

прозрачный png8:

прозрачный png24:

И какой-нибудь файл шрифта ttf.

Способ №1

Вывод по диагонали строки на изображение.


class watermark1
{
  function create_watermark( $main_img_obj, $text, $font, $r = 128, $g = 128, $b = 128, $alpha_level = 100 )
  {
   $width = imagesx($main_img_obj);
   $height = imagesy($main_img_obj);
   $angle =  -rad2deg(atan2((-$height),($width)));

   $text = " ".$text." ";

   $c = imagecolorallocatealpha($main_img_obj, $r, $g, $b, $alpha_level);
   $size = (($width+$height)/2)*2/strlen($text);
   $box  = imagettfbbox ( $size, $angle, $font, $text );
   $x = $width/2 - abs($box[4] - $box[0])/2;
   $y = $height/2 + abs($box[5] - $box[1])/2;

   imagettftext($main_img_obj,$size ,$angle, $x, $y, $c, $font, $text);
   return $main_img_obj;
  }
}

$main_img_obj — идентификатор изображения, на которое добавляется надпись

$text — текст надписи

$font — имя файла шрифта .ttf

$r,$g,$b — цвет надписи

$alpha_level — прозрачность (0 — не прозрачная, 128 — полностью прозрачная)

Вызываем:

$watermark = new watermark1();
$img = imagecreatefromjpeg(«image.jpg»);
$im=$watermark->create_watermark($img,»jeka911.wordpress.com»,»1.ttf»,0,0,255,120);
imagejpeg($im,»result.jpg»);

Получаем:

Все просчеты в функции получены методом тыка, так что не бойтесь экспериментировать.

Способ №2

Добавление в качестве вотермарка 8-битного png


class watermark2
{
function create_watermark( $main_img_obj, $watermark_img_obj, $alpha_level = 100 )
{
$watermark_width = imagesx($watermark_img_obj);
$watermark_height = imagesy($watermark_img_obj);

$dest_x = imagesx($main_img_obj) - $watermark_width - 5;
$dest_y = imagesy($main_img_obj) - $watermark_height - 5;
imagecopymerge($main_img_obj, $watermark_img_obj, $dest_x, $dest_y, 0, 0, $watermark_width, $watermark_height, $alpha_level);

return $main_img_obj;
}
}

$main_img_obj — идентификатор изображения, на которое добавляется надпись

$watermark_img_obj — ид. изображения прозрачного png8

$alpha_level — прозрачность (0 — прозрачное, 100 — полностью непрозрачное)

Вызываем:

$watermark = new watermark2();
$img = imagecreatefromjpeg(«image.jpg»);
$water = imagecreatefrompng(«watermark8.png»);
$im=$watermark->create_watermark($img,$water,10);
imagejpeg($im,»result.jpg»);

Получаем:

Способ №3

Добавление водяным знаком 24х битного png (видимо, брал отсюда)


class watermark3{

	# given two images, return a blended watermarked image
	function create_watermark( $main_img_obj, $watermark_img_obj, $alpha_level = 100 ) {
		$alpha_level	/= 100;	# convert 0-100 (%) alpha to decimal

		# calculate our images dimensions
		$main_img_obj_w	= imagesx( $main_img_obj );
		$main_img_obj_h	= imagesy( $main_img_obj );
		$watermark_img_obj_w	= imagesx( $watermark_img_obj );
		$watermark_img_obj_h	= imagesy( $watermark_img_obj );

		# determine center position coordinates
		$main_img_obj_min_x	= floor( ( $main_img_obj_w / 2 ) - ( $watermark_img_obj_w / 2 ) );
		$main_img_obj_max_x	= ceil( ( $main_img_obj_w / 2 ) + ( $watermark_img_obj_w / 2 ) );
		$main_img_obj_min_y	= floor( ( $main_img_obj_h / 2 ) - ( $watermark_img_obj_h / 2 ) );
		$main_img_obj_max_y	= ceil( ( $main_img_obj_h / 2 ) + ( $watermark_img_obj_h / 2 ) ); 

		# create new image to hold merged changes
		$return_img	= imagecreatetruecolor( $main_img_obj_w, $main_img_obj_h );

		# walk through main image
		for( $y = 0; $y < $main_img_obj_h; $y++ ) {
			for( $x = 0; $x < $main_img_obj_w; $x++ ) {
				$return_color	= NULL;

				# determine the correct pixel location within our watermark
				$watermark_x	= $x - $main_img_obj_min_x;
				$watermark_y	= $y - $main_img_obj_min_y;

				# fetch color information for both of our images
				$main_rgb = imagecolorsforindex( $main_img_obj, imagecolorat( $main_img_obj, $x, $y ) );

				# if our watermark has a non-transparent value at this pixel intersection
				# and we're still within the bounds of the watermark image
				if (	$watermark_x >= 0 && $watermark_x < $watermark_img_obj_w &&
							$watermark_y >= 0 && $watermark_y < $watermark_img_obj_h ) {
					$watermark_rbg = imagecolorsforindex( $watermark_img_obj, imagecolorat( $watermark_img_obj, $watermark_x, $watermark_y ) );

					# using image alpha, and user specified alpha, calculate average
					$watermark_alpha	= round( ( ( 127 - $watermark_rbg['alpha'] ) / 127 ), 2 );
					$watermark_alpha	= $watermark_alpha * $alpha_level;

					# calculate the color 'average' between the two - taking into account the specified alpha level
					$avg_red		= $this->_get_ave_color( $main_rgb['red'],		$watermark_rbg['red'],		$watermark_alpha );
					$avg_green	= $this->_get_ave_color( $main_rgb['green'],	$watermark_rbg['green'],	$watermark_alpha );
					$avg_blue		= $this->_get_ave_color( $main_rgb['blue'],	$watermark_rbg['blue'],		$watermark_alpha );

					# calculate a color index value using the average RGB values we've determined
					$return_color	= $this->_get_image_color( $return_img, $avg_red, $avg_green, $avg_blue );

				# if we're not dealing with an average color here, then let's just copy over the main color
				} else {
					$return_color	= imagecolorat( $main_img_obj, $x, $y );

				} # END if watermark

				# draw the appropriate color onto the return image
				imagesetpixel( $return_img, $x, $y, $return_color );

			} # END for each X pixel
		} # END for each Y pixel

		# return the resulting, watermarked image for display
		return $return_img;

	} # END create_watermark()

	# average two colors given an alpha
	function _get_ave_color( $color_a, $color_b, $alpha_level ) {
		return round( ( ( $color_a * ( 1 - $alpha_level ) ) + ( $color_b	* $alpha_level ) ) );
	} # END _get_ave_color()

	# return closest pallette-color match for RGB values
	function _get_image_color($im, $r, $g, $b) {
		$c=imagecolorexact($im, $r, $g, $b);
		if ($c!=-1) return $c;
		$c=imagecolorallocate($im, $r, $g, $b);
		if ($c!=-1) return $c;
		return imagecolorclosest($im, $r, $g, $b);
	} # EBD _get_image_color()

} # END watermark API

$main_img_obj — идентификатор изображения, на которое добавляется надпись

$watermark_img_obj — ид. изображения прозрачного png8

$alpha_level — прозрачность (0 — прозрачное, 100 — полностью непрозрачное)

Вызываем:

$watermark = new watermark3();
$img = imagecreatefromjpeg(«image.jpg»);
$water = imagecreatefrompng(«watermark24.png»);
$im=$watermark->create_watermark($img,$water,10);
imagejpeg($im,»result.jpg»);

Получаем:

Спасибо за внимание.

RSS

Реклама

imagegrabwindow

Сентябрь 19, 2007

Вот какая мощная функция есть в >= 5.2.2 под виндой

imagegrabwindow — делает скриншот окна приложения. возвращает идентификатор изображения или false при неудачной попытке.

$browser = new COM("InternetExplorer.Application");
$handle = $browser->HWND;
$browser->Visible = true;
$browser->Navigate("https://jeka911.wordpress.com");

/* Still working? */
while ($browser->Busy) {
com_message_pump(4000);
}
$im = imagegrabwindow($handle, 0);
$browser->Quit();
imagepng($im, "iesnap.png");

И получается скриншот сайта с php.

Опа, какой мощный комментарий от nopox’a:

Первый нах! :)

Субъективное мнение: COM суть зло, ибо неуниверсально и привязано к платформе. Хотя, с другой стороны, эта задача — автоматизированное создание скриншотов сайта — не имеет красивых решений.

Вообще, если кому вдруг понадобится куча скриншотов: для Windows можно использовать утилиту url2bmp (http://www.pixel-technology.com/freeware/url2bmp), которая помимо графического имеет мощный консольный интерфейс.

Для Unix придётся устанавливать X-server, запускать браузер и делать снимок окна посредством import из ImageMagick.

Вообще, наверное, надо взяться и написать универсальный интерфейс, максимально упрощающий и унифицирующий процедуру создания скринов сайта :)


Функция для получения pagerank’а

Сентябрь 4, 2007

На вопрос Григория отвечаю постом.

Когда-то нашел эту функцию, с тех пор с ней не расстаюсь.

function GooglePageRank($url){
$arr = parse_url($url);
$url = $arr['host'];
$url="info:".$url; $ch=GoogleCSum($url,0xE6359A60);
$host="toolbarqueries.google.com"; $hostip=gethostbyname($host);
$query ="GET /search?client=navclient-auto&ch=6".$ch."&ie=UTF-8&oe=UTF-8&features=Rank&q=".rawurlencode($url)." HTTP/1.0\r\n";
$query.="Host: $host\r\n"; $rank=-1;
$query.="User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1)\r\n";
$query.="Connection: Close\r\n\r\n";
$fp=fsockopen($hostip,80,$errno,$errstr,30);
if ($fp)
{ fputs($fp,$query); $data=""; while (!feof($fp)) $data.=fgets($fp,4096); fclose($fp);
$data=explode("\n",$data);
foreach ($data as $line)
if (!is_bool(strpos($line,"Rank_1")))
{ $rank=explode(":",trim($line)); $rank=$rank[2]; break; } }
return $rank;
}

function GoogleCSum($s,$key){
$v4=$len=strlen($s); $esi=$key; $ebx=$edi=0X9E3779B9; $p=0;
if ($len>=12)
for($i=0;$i<floor($len/12);$i++)
{ $edi=unsign($edi+ord($s[$p+4])+(ord($s[$p+5]) << 8)+(ord($s[$p+6]) << 16)+(ord($s[$p+7]) << 24));
$esi=unsign($esi+ord($s[$p+8])+(ord($s[$p+9]) << 8)+(ord($s[$p+10]) << 16)+(ord($s[$p+11]) << 24));
$edx=unsign(($ebx+ord($s[$p+0])+(ord($s[$p+1]) << 8)+(ord($s[$p+2]) << 16)+(ord($s[$p+3]) << 24)-$edi-$esi)^shr($esi,13));
$edi=unsign(($edi-$esi-$edx)^($edx << 8));
$esi=unsign(($esi-$edx-$edi)^shr($edi,13));
$edx=unsign(($edx-$edi-$esi)^shr($esi,12));
$edi=unsign(($edi-$esi-$edx)^($edx << 16));
$esi=unsign(($esi-$edx-$edi)^shr($edi,5));
$edx=unsign(($edx-$edi-$esi)^shr($esi,3)); $ebx=$edx;
$edi=unsign(($edi-$esi-$ebx)^($ebx << 10));
$esi=unsign(($esi-$ebx-$edi)^shr($edi,15));
$v4-=12; $p+=12; }
$esi=unsign($esi+$len);
if ($v4>=11) $esi=unsign($esi+(ord($s[$p+10]) << 24));
if ($v4>=10) $esi=unsign($esi+(ord($s[$p+9]) << 16));
if ($v4>=9) $esi=unsign($esi+(ord($s[$p+8]) << 8));
if ($v4>=8) $edi=unsign($edi+ord($s[$p+4])+(ord($s[$p+5]) << 8)+(ord($s[$p+6]) << 16)+(ord($s[$p+7]) << 24));
else
{ if ($v4>=7) $edi=unsign($edi+(ord($s[$p+6]) << 16));
if ($v4>=6) $edi=unsign($edi+(ord($s[$p+5]) << 8));
if ($v4>=5) $edi=unsign($edi+ord($s[$p+4])); }
if ($v4>=4) $ebx=unsign($ebx+ord($s[$p+0])+(ord($s[$p+1]) << 8)+(ord($s[$p+2]) << 16)+(ord($s[$p+3]) << 24));
else
{ if ($v4>=3) $ebx=unsign($ebx+(ord($s[$p+2]) << 16));
if ($v4>=2) $ebx=unsign($ebx+(ord($s[$p+1]) << 8));
if ($v4>=1) $ebx=unsign($ebx+ord($s[$p+0])); }
$ebx=unsign(($ebx-$edi-$esi)^shr($esi,13));
$edi=unsign(($edi-$esi-$ebx)^($ebx << 8));
$esi=unsign(($esi-$ebx-$edi)^shr($edi,13));
$ebx=unsign(($ebx-$edi-$esi)^shr($esi,12));
$edi=unsign(($edi-$esi-$ebx)^($ebx << 16));
$esi=unsign(($esi-$ebx-$edi)^shr($edi,5));
$ebx=unsign(($ebx-$edi-$esi)^shr($esi,3));
$edi=unsign(($edi-$esi-$ebx)^($ebx << 10));
$esi=unsign(($esi-$ebx-$edi)^shr($edi,15)); return $esi;
}
function shr($x,$y) {
$x=unsign($x);
for($i=0;$i<$y;$i++) $x=floor($x/2); return $x;
}
function unsign($l) {
$l=intval($l);
if ($l>=0){
return $l;
}else{
return 4294967296+$l;
}
}


Image Tag Cloud (Облако картинок)

Август 29, 2007

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

Мой друг Romiz пару раз подгонял мне идею про, так называемое, облако картинок. Т.е. как облако тэгов, только вместо слов — изображения.

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

Итак:

  • phpImageCloud — главная страница
  • Static demo — картинка из предыдущего поста, только рабочая
  • Client-side demo — это опять таки по напрягу Romiz’а, анимация с jquery. Попробуйте, выгребать свой блог из кучи соседей — довольно занятное дело, имхо.

Все чуствует себя GNU GPL, живет на sourceforge (кстати, еще один мастевой способ меньше переживать за жесткие диски).

Скачать, если кому интересно.


php скрипт — скачать видео с youtube [updated]

Июнь 22, 2007

Функциональная и более мастевая версия

Вроде работает. Функция get_youtube_video_link принимает url на youtube, а возвращает url самого flv видео. Требует curl.

function get_youtube_video_link($url)
{
$crl = curl_init();
curl_setopt($crl, CURLOPT_URL, $url);
curl_setopt($crl, CURLOPT_RETURNTRANSFER, 1);
$data = curl_exec($crl);
curl_close($crl);
if (preg_match(‘/\/player2\.swf\?(.*)», «movie_/’, $data, $match))
$url = ‘http://youtube.com/get_video.php?&#8217;.$match[1];
else return false;
$crl = curl_init();
curl_setopt($crl, CURLOPT_URL, $url);
curl_setopt($crl, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($crl, CURLOPT_HEADER, 1);
$data = curl_exec($crl);
curl_close($crl);
$a = explode(«\n»,$data);
foreach($a as $ae)
if(eregi(«location»,$ae)) $location = $ae;
if (!isset($location)) return false;
$location = explode(«Location:»,$location);
$location = trim($location[1]);
return $location;
}

Использование — что-то вроде:

$flv = get_youtube_video_link(«http://www.youtube.com/watch?v=je1H-57AaVg&#187;);

$movie = file_get_contents($flv);


Валидатор форм для лентяев

Май 26, 2007

Где-то живет дядя по имени Фабио Нагао. Все халявщики и лентяи ему глубоко признательны и бесконечно благодарны. После знакомства с mootools его пробило такое вдохновение, что он накатал яваскрипт библиотеку для валидации форм.

fvalidator.png

Пользоваться ею — чуть тяжелее, чем дверью:

<input id=«exA_Phone» class=«fValidate[‘required’,’phone’]» name=«phone» type=«text» />

Спасибо, дядя Фабио.


Графики, чарты, пироги и свечи

Май 2, 2007

Да, последнее время это стало моей любимой темой. Лозунг нашей эпохи: «Визуализируем все!». Вот список различных библиотек для создания графиков на стороне клиента, которые я накопал: Читать далее…