Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Right align text in an image with imagettftext(), PHP

I am setting up dynamic forum signature images for my users and I want to be able to put their username on the image. I am able to do this just fine, but since usernames are different lengths and I want to right align the username, how can I go about doing this when I have to set x & y coordinates.

$im = imagecreatefromjpeg("/path/to/base/image.jpg");
$text = "Username";
$font = "Font.ttf";
$black = imagecolorallocate($im, 0, 0, 0);

imagettftext($im, 10, 0, 217, 15, $black, $font, $text);
imagejpeg($im, null, 90);
like image 729
James Simpson Avatar asked Mar 14 '10 22:03

James Simpson


5 Answers

Use the imagettfbbox function to get the width of the string and then subtract that from the width of the image to get the starting x-coordinate.

$dimensions = imagettfbbox($fontSize, $angle, $font, $text);
$textWidth = abs($dimensions[4] - $dimensions[0]);
$x = imagesx($im) - $textWidth;
like image 115
Andy Shea Avatar answered Nov 20 '22 18:11

Andy Shea


You can use stil/gd-text class. Disclaimer: I am the author.

<?php
use GDText\Box;
use GDText\Color;

$im = imagecreatefromjpeg("/path/to/base/image.jpg");

$textbox = new Box($im);
$textbox->setFontSize(12);
$textbox->setFontFace("Font.ttf");
$textbox->setFontColor(new Color(0, 0, 0)); // black
$textbox->setBox(
    50,  // distance from left edge
    50,  // distance from top edge
    200, // textbox width
    100  // textbox height
);

// text will be aligned inside textbox to right horizontally and to top vertically
$textbox->setTextAlign('right', 'top');

$textbox->draw("Username");

You can also draw multilined text. Just use \n in the string passed to draw() method. Example generated with this class:

right aligned text demo

like image 32
stil Avatar answered Nov 20 '22 18:11

stil


Pre-calculate the size of the user's name using imagettfbbox().

From the width you get from there, you can then deduct the x position at which your text needs to start.

like image 35
Pekka Avatar answered Nov 20 '22 19:11

Pekka


This will work...............

                  $s = split("[\n]+", $text);
                  $top=20;
                  $left=30;
                  $font_file="yourfont.ttf";
                  $fontsize=20;
               $__H=$top;
               foreach($s as $key=>$val){
                    $_b = imageTTFBbox($fontsize,0,$font_file,$val);
                    $_W = abs($_b[2]-$_b[0]); 
                    $_X = ($left+$text_box_width)-$_W;
                    $_H = abs($_b[5]-$_b[3]); 
                    $_H +=1;  
                    $__H += $_H;              
                    $res=imagettftext($image, $this->_fontsize, 0, $_X, $__H, $color, $font_file, $val);
                    $__H += 1;
like image 22
sujithayur Avatar answered Nov 20 '22 18:11

sujithayur


I enhanced sujithayur code, and created function which allows all aligns (left, center, right) & (top, center, middle) and its combinations. It also uses text shadow.

// $x is margin from left, in case of left align, and margin from right, in case of right horizontal align
// $alignHorizontal values can be 'left', 'center', 'right'
// $alignVertical values can be 'top', 'center', 'bottom'
function imagettftext_aligned($image, $fontSize, $x, $y, $color, $colorShadow, $fontPath, $text, $alignHorizontal, $alignVertical) {

      $s = explode("\n", $text);
      $imageWidth = imagesx($image);
      $imageHeight = imagesy($image);

      $top=$y;
      $left=$imageWidth - $x;
      $__H=$top; // default - top
      $lineHeight = $fontSize + 14;

      if ($alignVertical == 'bottom')
        $__H = $imageHeight - $y - (count($s) * $lineHeight);
      elseif ($alignVertical == 'center')
        $__H = $imageHeight/2 - (count($s) * $lineHeight)/2;

      foreach($s as $key=>$val){
            $_b = imageTTFBbox($fontSize,0,$fontPath,$val);
            $_W = abs($_b[2]-$_b[0]); 
            $_H = abs($_b[5]-$_b[3]); 
            $_H +=1;  

            if ($alignHorizontal == 'right')
              $_X = $left - $_W;
            elseif ($alignHorizontal == 'center')
              $_X = $imageWidth/2 - $_W/2;
            else // default - left
              $_X = $x;

            imagettftextblur($image, $fontSize, 0, $_X + 2, $__H + 2, $colorShadow, $fontPath, $val, 4); // 1 can be higher to increase blurriness of the shadow
            imagettftextblur($image, $fontSize, 0, $_X, $__H, $color, $fontPath, $val);

            $__H += $lineHeight + 1;   
       } 

  return ['bottom' => $__H];

}

// https://github.com/andrewgjohnson/imagettftextblur
if (!function_exists('imagettftextblur'))
{
    function imagettftextblur(&$image,$size,$angle,$x,$y,$color,$fontfile,$text,$blur_intensity = null)
    {
        $blur_intensity = !is_null($blur_intensity) && is_numeric($blur_intensity) ? (int)$blur_intensity : 0;
        if ($blur_intensity > 0)
        {
            $text_shadow_image = imagecreatetruecolor(imagesx($image),imagesy($image));
            imagefill($text_shadow_image,0,0,imagecolorallocate($text_shadow_image,0x00,0x00,0x00));
            imagettftext($text_shadow_image,$size,$angle,$x,$y,imagecolorallocate($text_shadow_image,0xFF,0xFF,0xFF),$fontfile,$text);
            for ($blur = 1;$blur <= $blur_intensity;$blur++)
                imagefilter($text_shadow_image,IMG_FILTER_GAUSSIAN_BLUR);
            for ($x_offset = 0;$x_offset < imagesx($text_shadow_image);$x_offset++)
            {
                for ($y_offset = 0;$y_offset < imagesy($text_shadow_image);$y_offset++)
                {
                    $visibility = (imagecolorat($text_shadow_image,$x_offset,$y_offset) & 0xFF) / 255;
                    if ($visibility > 0)
                        imagesetpixel($image,$x_offset,$y_offset,imagecolorallocatealpha($image,($color >> 16) & 0xFF,($color >> 8) & 0xFF,$color & 0xFF,(1 - $visibility) * 127));
                }
            }
            imagedestroy($text_shadow_image);
        }
        else
            return imagettftext($image,$size,$angle,$x,$y,$color,$fontfile,$text);
    }
}
like image 1
luky Avatar answered Nov 20 '22 20:11

luky