2004-09-26 04:57:07 +00:00
|
|
|
<?php
|
|
|
|
|
2009-11-01 16:48:45 +00:00
|
|
|
// This file is part of Moodle - http://moodle.org/
|
|
|
|
//
|
2009-05-25 08:27:25 +00:00
|
|
|
// Moodle is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// Moodle is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
2009-11-01 16:48:45 +00:00
|
|
|
//
|
2009-05-25 08:27:25 +00:00
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2004-09-26 04:57:07 +00:00
|
|
|
/**
|
2007-08-15 14:00:35 +00:00
|
|
|
* gdlib.php - Collection of routines in Moodle related to
|
2004-09-26 04:57:07 +00:00
|
|
|
* processing images using GD
|
|
|
|
*
|
|
|
|
* @package moodlecore
|
2009-05-25 08:27:25 +00:00
|
|
|
* @copyright 1999 onwards Martin Dougiamas {@link http://moodle.com}
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
2004-09-26 04:57:07 +00:00
|
|
|
*/
|
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
/**
|
2004-09-26 04:57:07 +00:00
|
|
|
*
|
|
|
|
* long description
|
2009-05-25 08:27:25 +00:00
|
|
|
* @global object
|
|
|
|
* @param object $dst_img
|
|
|
|
* @param object $src_img
|
2009-11-01 16:48:45 +00:00
|
|
|
* @param int $dst_x
|
|
|
|
* @param int $dst_y
|
|
|
|
* @param int $src_x
|
|
|
|
* @param int $src_y
|
|
|
|
* @param int $dst_w
|
|
|
|
* @param int $dst_h
|
|
|
|
* @param int $src_w
|
|
|
|
* @param int $src_h
|
2009-05-25 08:27:25 +00:00
|
|
|
* @return bool
|
2004-09-26 04:57:07 +00:00
|
|
|
* @todo Finish documenting this function
|
|
|
|
*/
|
2004-01-10 16:38:18 +00:00
|
|
|
function ImageCopyBicubic ($dst_img, $src_img, $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h) {
|
|
|
|
|
|
|
|
global $CFG;
|
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
if (function_exists('ImageCopyResampled') and $CFG->gdversion >= 2) {
|
2004-01-10 16:38:18 +00:00
|
|
|
return ImageCopyResampled($dst_img, $src_img, $dst_x, $dst_y, $src_x, $src_y,
|
|
|
|
$dst_w, $dst_h, $src_w, $src_h);
|
|
|
|
}
|
|
|
|
|
|
|
|
$totalcolors = imagecolorstotal($src_img);
|
2007-08-15 14:00:35 +00:00
|
|
|
for ($i=0; $i<$totalcolors; $i++) {
|
2004-01-10 16:38:18 +00:00
|
|
|
if ($colors = ImageColorsForIndex($src_img, $i)) {
|
|
|
|
ImageColorAllocate($dst_img, $colors['red'], $colors['green'], $colors['blue']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
$scaleX = ($src_w - 1) / $dst_w;
|
|
|
|
$scaleY = ($src_h - 1) / $dst_h;
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
$scaleX2 = $scaleX / 2.0;
|
|
|
|
$scaleY2 = $scaleY / 2.0;
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
for ($j = 0; $j < $dst_h; $j++) {
|
|
|
|
$sY = $j * $scaleY;
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
for ($i = 0; $i < $dst_w; $i++) {
|
|
|
|
$sX = $i * $scaleX;
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
$c1 = ImageColorsForIndex($src_img,ImageColorAt($src_img,(int)$sX,(int)$sY+$scaleY2));
|
|
|
|
$c2 = ImageColorsForIndex($src_img,ImageColorAt($src_img,(int)$sX,(int)$sY));
|
|
|
|
$c3 = ImageColorsForIndex($src_img,ImageColorAt($src_img,(int)$sX+$scaleX2,(int)$sY+$scaleY2));
|
|
|
|
$c4 = ImageColorsForIndex($src_img,ImageColorAt($src_img,(int)$sX+$scaleX2,(int)$sY));
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
$red = (int) (($c1['red'] + $c2['red'] + $c3['red'] + $c4['red']) / 4);
|
|
|
|
$green = (int) (($c1['green'] + $c2['green'] + $c3['green'] + $c4['green']) / 4);
|
|
|
|
$blue = (int) (($c1['blue'] + $c2['blue'] + $c3['blue'] + $c4['blue']) / 4);
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
$color = ImageColorClosest ($dst_img, $red, $green, $blue);
|
|
|
|
ImageSetPixel ($dst_img, $i + $dst_x, $j + $dst_y, $color);
|
|
|
|
}
|
|
|
|
}
|
2004-01-10 16:38:18 +00:00
|
|
|
}
|
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
/**
|
|
|
|
* Delete profile images associated with user or group
|
2009-05-25 08:27:25 +00:00
|
|
|
*
|
|
|
|
* @global object
|
2007-08-15 14:00:35 +00:00
|
|
|
* @param int $id user or group id
|
|
|
|
* @param string $dir type of entity - 'groups' or 'users'
|
|
|
|
* @return boolean success
|
|
|
|
*/
|
|
|
|
function delete_profile_image($id, $dir='users') {
|
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
require_once $CFG->libdir.'/filelib.php';
|
|
|
|
$location = $CFG->dataroot .'/'. $dir .'/'. $id;
|
|
|
|
|
|
|
|
if (file_exists($location)) {
|
|
|
|
return fulldelete($location);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2004-09-26 04:57:07 +00:00
|
|
|
* Given an upload manager with the right settings, this function performs a virus scan, and then scales and crops
|
|
|
|
* it and saves it in the right place to be a "user" or "group" image.
|
|
|
|
*
|
2009-05-25 08:27:25 +00:00
|
|
|
* @global object
|
2007-08-15 14:00:35 +00:00
|
|
|
* @param int $id user or group id
|
2007-10-11 09:01:29 +00:00
|
|
|
* @param string $dir type of entity - groups, user, ...
|
2008-02-21 05:24:18 +00:00
|
|
|
* @return string $destination (profile image destination path) or false on error
|
2004-09-26 04:57:07 +00:00
|
|
|
*/
|
2008-02-21 05:24:18 +00:00
|
|
|
function create_profile_image_destination($id, $dir='user') {
|
2004-01-10 16:38:18 +00:00
|
|
|
global $CFG;
|
|
|
|
|
|
|
|
umask(0000);
|
|
|
|
|
2004-09-26 04:57:07 +00:00
|
|
|
if (!file_exists($CFG->dataroot .'/'. $dir)) {
|
2007-10-11 09:01:29 +00:00
|
|
|
if (!mkdir($CFG->dataroot .'/'. $dir, $CFG->directorypermissions)) {
|
2004-01-10 16:38:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2009-11-01 16:48:45 +00:00
|
|
|
|
2007-10-11 09:01:29 +00:00
|
|
|
if ($dir == 'user') {
|
|
|
|
$destination = make_user_directory($id, true);
|
|
|
|
} else {
|
|
|
|
$destination = "$CFG->dataroot/$dir/$id";
|
|
|
|
}
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-10-11 09:01:29 +00:00
|
|
|
if (!file_exists($destination)) {
|
2007-10-12 19:13:18 +00:00
|
|
|
if (!make_upload_directory(str_replace($CFG->dataroot . '/', '', $destination))) {
|
2004-01-10 16:38:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2008-02-21 05:24:18 +00:00
|
|
|
return $destination;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given an upload manager with the right settings, this function performs a virus scan, and then scales and crops
|
|
|
|
* it and saves it in the right place to be a "user" or "group" image.
|
|
|
|
*
|
|
|
|
* @param int $id user or group id
|
2008-07-31 22:15:30 +00:00
|
|
|
* @param object $userform with imagefile upload field
|
2008-02-21 05:24:18 +00:00
|
|
|
* @param string $dir type of entity - groups, user, ...
|
|
|
|
* @return boolean success
|
|
|
|
*/
|
2008-07-31 22:15:30 +00:00
|
|
|
function save_profile_image($id, $userform, $dir='user') {
|
2008-02-21 05:24:18 +00:00
|
|
|
|
|
|
|
$destination = create_profile_image_destination($id, $dir);
|
|
|
|
if ($destination === false) {
|
|
|
|
return false;
|
|
|
|
}
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2008-07-31 22:15:30 +00:00
|
|
|
$filename = $userform->get_new_filename('imagefile');
|
|
|
|
$pathname = $destination.'/'.$filename;
|
|
|
|
|
|
|
|
if (!$userform->save_file('imagefile', $pathname, true)) {
|
2004-01-10 16:38:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-07-31 22:15:30 +00:00
|
|
|
return process_profile_image($pathname, $destination);
|
2008-02-21 05:24:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given a path to an image file this function scales and crops it and saves it in
|
|
|
|
* the right place to be a "user" or "group" image.
|
|
|
|
*
|
2009-05-25 08:27:25 +00:00
|
|
|
* @global object
|
2008-02-21 05:24:18 +00:00
|
|
|
* @param string $originalfile the path of the original image file
|
|
|
|
* @param string $destination the final destination directory of the profile image
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
function process_profile_image($originalfile, $destination) {
|
2009-08-18 05:20:12 +00:00
|
|
|
global $CFG, $OUTPUT;
|
2008-02-21 05:24:18 +00:00
|
|
|
|
|
|
|
if(!(is_file($originalfile) && is_dir($destination))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($CFG->gdversion)) {
|
|
|
|
return false;
|
|
|
|
}
|
2004-01-10 16:38:18 +00:00
|
|
|
|
|
|
|
$imageinfo = GetImageSize($originalfile);
|
2007-08-15 14:00:35 +00:00
|
|
|
|
2004-01-10 16:38:18 +00:00
|
|
|
if (empty($imageinfo)) {
|
2006-02-11 16:17:06 +00:00
|
|
|
if (file_exists($originalfile)) {
|
|
|
|
unlink($originalfile);
|
|
|
|
}
|
2004-01-10 16:38:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
2007-08-15 14:00:35 +00:00
|
|
|
|
2004-01-10 16:38:18 +00:00
|
|
|
$image->width = $imageinfo[0];
|
|
|
|
$image->height = $imageinfo[1];
|
|
|
|
$image->type = $imageinfo[2];
|
|
|
|
|
|
|
|
switch ($image->type) {
|
2008-02-21 05:24:18 +00:00
|
|
|
case IMAGETYPE_GIF:
|
2004-09-26 04:57:07 +00:00
|
|
|
if (function_exists('ImageCreateFromGIF')) {
|
2007-08-15 14:00:35 +00:00
|
|
|
$im = ImageCreateFromGIF($originalfile);
|
2004-01-10 16:38:18 +00:00
|
|
|
} else {
|
2004-09-26 04:57:07 +00:00
|
|
|
notice('GIF not supported on this server');
|
2004-01-10 16:38:18 +00:00
|
|
|
unlink($originalfile);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
2008-02-21 05:24:18 +00:00
|
|
|
case IMAGETYPE_JPEG:
|
2004-09-26 04:57:07 +00:00
|
|
|
if (function_exists('ImageCreateFromJPEG')) {
|
2007-08-15 14:00:35 +00:00
|
|
|
$im = ImageCreateFromJPEG($originalfile);
|
2004-01-10 16:38:18 +00:00
|
|
|
} else {
|
2004-09-26 04:57:07 +00:00
|
|
|
notice('JPEG not supported on this server');
|
2004-01-10 16:38:18 +00:00
|
|
|
unlink($originalfile);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
2008-02-21 05:24:18 +00:00
|
|
|
case IMAGETYPE_PNG:
|
2004-09-26 04:57:07 +00:00
|
|
|
if (function_exists('ImageCreateFromPNG')) {
|
2007-08-15 14:00:35 +00:00
|
|
|
$im = ImageCreateFromPNG($originalfile);
|
2004-01-10 16:38:18 +00:00
|
|
|
} else {
|
2004-09-26 04:57:07 +00:00
|
|
|
notice('PNG not supported on this server');
|
2004-01-10 16:38:18 +00:00
|
|
|
unlink($originalfile);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
2007-08-15 14:00:35 +00:00
|
|
|
default:
|
2004-01-10 16:38:18 +00:00
|
|
|
unlink($originalfile);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
unlink($originalfile);
|
|
|
|
|
2004-09-26 04:57:07 +00:00
|
|
|
if (function_exists('ImageCreateTrueColor') and $CFG->gdversion >= 2) {
|
2004-01-10 16:38:18 +00:00
|
|
|
$im1 = ImageCreateTrueColor(100,100);
|
|
|
|
$im2 = ImageCreateTrueColor(35,35);
|
|
|
|
} else {
|
|
|
|
$im1 = ImageCreate(100,100);
|
|
|
|
$im2 = ImageCreate(35,35);
|
|
|
|
}
|
2007-08-15 14:00:35 +00:00
|
|
|
|
2004-01-10 16:38:18 +00:00
|
|
|
$cx = $image->width / 2;
|
|
|
|
$cy = $image->height / 2;
|
|
|
|
|
|
|
|
if ($image->width < $image->height) {
|
|
|
|
$half = floor($image->width / 2.0);
|
|
|
|
} else {
|
|
|
|
$half = floor($image->height / 2.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ImageCopyBicubic($im1, $im, 0, 0, $cx-$half, $cy-$half, 100, 100, $half*2, $half*2);
|
|
|
|
ImageCopyBicubic($im2, $im, 0, 0, $cx-$half, $cy-$half, 35, 35, $half*2, $half*2);
|
|
|
|
|
2005-09-12 07:57:16 +00:00
|
|
|
if (function_exists('ImageJpeg')) {
|
2007-10-11 09:01:29 +00:00
|
|
|
@touch($destination .'/f1.jpg'); // Helps in Safe mode
|
|
|
|
@touch($destination .'/f2.jpg'); // Helps in Safe mode
|
|
|
|
if (ImageJpeg($im1, $destination .'/f1.jpg', 90) and
|
|
|
|
ImageJpeg($im2, $destination .'/f2.jpg', 95) ) {
|
|
|
|
@chmod($destination .'/f1.jpg', 0666);
|
|
|
|
@chmod($destination .'/f2.jpg', 0666);
|
2005-09-12 07:57:16 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2004-01-10 16:38:18 +00:00
|
|
|
} else {
|
2009-08-18 05:20:12 +00:00
|
|
|
echo $OUTPUT->notification('PHP has not been configured to support JPEG images. Please correct this.');
|
2004-01-10 16:38:18 +00:00
|
|
|
}
|
2005-09-12 07:57:16 +00:00
|
|
|
return 0;
|
2004-01-10 16:38:18 +00:00
|
|
|
}
|
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
/**
|
|
|
|
* Given a user id this function scales and crops the user images to remove
|
2005-07-22 14:08:05 +00:00
|
|
|
* the one pixel black border.
|
|
|
|
*
|
2009-05-25 08:27:25 +00:00
|
|
|
* @global object
|
|
|
|
* @param int $id
|
|
|
|
* @param string $dir
|
2005-07-22 14:08:05 +00:00
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
function upgrade_profile_image($id, $dir='users') {
|
2009-08-18 05:20:12 +00:00
|
|
|
global $CFG, $OUTPUT;
|
2004-01-10 16:38:18 +00:00
|
|
|
|
2007-08-15 14:00:35 +00:00
|
|
|
$im = ImageCreateFromJPEG($CFG->dataroot .'/'. $dir .'/'. $id .'/f1.jpg');
|
2005-07-22 14:08:05 +00:00
|
|
|
|
|
|
|
if (function_exists('ImageCreateTrueColor') and $CFG->gdversion >= 2) {
|
|
|
|
$im1 = ImageCreateTrueColor(100,100);
|
|
|
|
$im2 = ImageCreateTrueColor(35,35);
|
|
|
|
} else {
|
|
|
|
$im1 = ImageCreate(100,100);
|
|
|
|
$im2 = ImageCreate(35,35);
|
|
|
|
}
|
2007-08-15 14:00:35 +00:00
|
|
|
|
|
|
|
if (function_exists('ImageCopyResampled') and $CFG->gdversion >= 2) {
|
2005-07-22 14:08:05 +00:00
|
|
|
ImageCopyBicubic($im1, $im, 0, 0, 2, 2, 100, 100, 96, 96);
|
|
|
|
} else {
|
|
|
|
imagecopy($im1, $im, 0, 0, 0, 0, 100, 100);
|
2007-08-15 14:00:35 +00:00
|
|
|
$c = ImageColorsForIndex($im1,ImageColorAt($im1,2,2));
|
|
|
|
$color = ImageColorClosest ($im1, $c['red'], $c['green'], $c['blue']);
|
|
|
|
ImageSetPixel ($im1, 0, 0, $color);
|
|
|
|
$c = ImageColorsForIndex($im1,ImageColorAt($im1,2,97));
|
|
|
|
$color = ImageColorClosest ($im1, $c['red'], $c['green'], $c['blue']);
|
|
|
|
ImageSetPixel ($im1, 0, 99, $color);
|
|
|
|
$c = ImageColorsForIndex($im1,ImageColorAt($im1,97,2));
|
|
|
|
$color = ImageColorClosest ($im1, $c['red'], $c['green'], $c['blue']);
|
|
|
|
ImageSetPixel ($im1, 99, 0, $color);
|
|
|
|
$c = ImageColorsForIndex($im1,ImageColorAt($im1,97,97));
|
|
|
|
$color = ImageColorClosest ($im1, $c['red'], $c['green'], $c['blue']);
|
|
|
|
ImageSetPixel ($im1, 99, 99, $color);
|
|
|
|
for ($x = 1; $x < 99; $x++) {
|
|
|
|
$c1 = ImageColorsForIndex($im1,ImageColorAt($im,$x,1));
|
|
|
|
$color = ImageColorClosest ($im, $c1['red'], $c1['green'], $c1['blue']);
|
|
|
|
ImageSetPixel ($im1, $x, 0, $color);
|
|
|
|
$c2 = ImageColorsForIndex($im1,ImageColorAt($im1,$x,98));
|
|
|
|
$color = ImageColorClosest ($im, $c2['red'], $c2['green'], $c2['blue']);
|
|
|
|
ImageSetPixel ($im1, $x, 99, $color);
|
|
|
|
}
|
|
|
|
for ($y = 1; $y < 99; $y++) {
|
|
|
|
$c3 = ImageColorsForIndex($im1,ImageColorAt($im, 1, $y));
|
|
|
|
$color = ImageColorClosest ($im, $c3['red'], $c3['green'], $c3['blue']);
|
|
|
|
ImageSetPixel ($im1, 0, $y, $color);
|
|
|
|
$c4 = ImageColorsForIndex($im1,ImageColorAt($im1, 98, $y));
|
|
|
|
$color = ImageColorClosest ($im, $c4['red'], $c4['green'], $c4['blue']);
|
|
|
|
ImageSetPixel ($im1, 99, $y, $color);
|
|
|
|
}
|
|
|
|
}
|
2005-07-22 14:08:05 +00:00
|
|
|
ImageCopyBicubic($im2, $im, 0, 0, 2, 2, 35, 35, 96, 96);
|
|
|
|
|
2005-09-12 07:57:16 +00:00
|
|
|
if (function_exists('ImageJpeg')) {
|
2007-08-15 14:00:35 +00:00
|
|
|
if (ImageJpeg($im1, $CFG->dataroot .'/'. $dir .'/'. $id .'/f1.jpg', 90) and
|
2005-09-12 07:57:16 +00:00
|
|
|
ImageJpeg($im2, $CFG->dataroot .'/'. $dir .'/'. $id .'/f2.jpg', 95) ) {
|
|
|
|
@chmod($CFG->dataroot .'/'. $dir .'/'. $id .'/f1.jpg', 0666);
|
|
|
|
@chmod($CFG->dataroot .'/'. $dir .'/'. $id .'/f2.jpg', 0666);
|
|
|
|
return 1;
|
|
|
|
}
|
2005-07-22 14:08:05 +00:00
|
|
|
} else {
|
2009-08-18 05:20:12 +00:00
|
|
|
echo $OUTPUT->notification('PHP has not been configured to support JPEG images. Please correct this.');
|
2005-07-22 14:08:05 +00:00
|
|
|
}
|
2005-09-12 07:57:16 +00:00
|
|
|
return 0;
|
2005-07-22 14:08:05 +00:00
|
|
|
}
|