Merge pull request #6876 from nextcloud/always_img_avatar
Always generate avatar
This commit is contained in:
commit
ed7beb929e
4 changed files with 241 additions and 39 deletions
|
@ -111,6 +111,9 @@ class AvatarController extends Controller {
|
||||||
$this->timeFactory = $timeFactory;
|
$this->timeFactory = $timeFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @NoAdminRequired
|
* @NoAdminRequired
|
||||||
* @NoCSRFRequired
|
* @NoCSRFRequired
|
||||||
|
@ -133,19 +136,10 @@ class AvatarController extends Controller {
|
||||||
$resp = new FileDisplayResponse($avatar,
|
$resp = new FileDisplayResponse($avatar,
|
||||||
Http::STATUS_OK,
|
Http::STATUS_OK,
|
||||||
['Content-Type' => $avatar->getMimeType()]);
|
['Content-Type' => $avatar->getMimeType()]);
|
||||||
} catch (NotFoundException $e) {
|
|
||||||
$user = $this->userManager->get($userId);
|
|
||||||
$resp = new JSONResponse([
|
|
||||||
'data' => [
|
|
||||||
'displayname' => $user->getDisplayName(),
|
|
||||||
],
|
|
||||||
]);
|
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
$resp = new JSONResponse([
|
$resp = new Http\Response();
|
||||||
'data' => [
|
$resp->setStatus(Http::STATUS_NOT_FOUND);
|
||||||
'displayname' => $userId,
|
return $resp;
|
||||||
],
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Let cache this!
|
// Let cache this!
|
||||||
|
|
|
@ -124,20 +124,27 @@ class Avatar implements IAvatar {
|
||||||
$type = 'jpg';
|
$type = 'jpg';
|
||||||
}
|
}
|
||||||
if ($type !== 'jpg' && $type !== 'png') {
|
if ($type !== 'jpg' && $type !== 'png') {
|
||||||
throw new \Exception($this->l->t("Unknown filetype"));
|
throw new \Exception($this->l->t('Unknown filetype'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$img->valid()) {
|
if (!$img->valid()) {
|
||||||
throw new \Exception($this->l->t("Invalid image"));
|
throw new \Exception($this->l->t('Invalid image'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!($img->height() === $img->width())) {
|
if (!($img->height() === $img->width())) {
|
||||||
throw new NotSquareException($this->l->t("Avatar image is not square"));
|
throw new NotSquareException($this->l->t('Avatar image is not square'));
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->remove();
|
$this->remove();
|
||||||
$file = $this->folder->newFile('avatar.'.$type);
|
$file = $this->folder->newFile('avatar.'.$type);
|
||||||
$file->putContent($data);
|
$file->putContent($data);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$generated = $this->folder->getFile('generated');
|
||||||
|
$generated->delete();
|
||||||
|
} catch (NotFoundException $e) {
|
||||||
|
//
|
||||||
|
}
|
||||||
$this->user->triggerChange('avatar', $file);
|
$this->user->triggerChange('avatar', $file);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -146,16 +153,13 @@ class Avatar implements IAvatar {
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public function remove () {
|
public function remove () {
|
||||||
$regex = '/^avatar\.([0-9]+\.)?(jpg|png)$/';
|
|
||||||
$avatars = $this->folder->getDirectoryListing();
|
$avatars = $this->folder->getDirectoryListing();
|
||||||
|
|
||||||
$this->config->setUserValue($this->user->getUID(), 'avatar', 'version',
|
$this->config->setUserValue($this->user->getUID(), 'avatar', 'version',
|
||||||
(int)$this->config->getUserValue($this->user->getUID(), 'avatar', 'version', 0) + 1);
|
(int)$this->config->getUserValue($this->user->getUID(), 'avatar', 'version', 0) + 1);
|
||||||
|
|
||||||
foreach ($avatars as $avatar) {
|
foreach ($avatars as $avatar) {
|
||||||
if (preg_match($regex, $avatar->getName())) {
|
$avatar->delete();
|
||||||
$avatar->delete();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
$this->user->triggerChange('avatar', '');
|
$this->user->triggerChange('avatar', '');
|
||||||
}
|
}
|
||||||
|
@ -164,7 +168,16 @@ class Avatar implements IAvatar {
|
||||||
* @inheritdoc
|
* @inheritdoc
|
||||||
*/
|
*/
|
||||||
public function getFile($size) {
|
public function getFile($size) {
|
||||||
$ext = $this->getExtension();
|
try {
|
||||||
|
$ext = $this->getExtension();
|
||||||
|
} catch (NotFoundException $e) {
|
||||||
|
$data = $this->generateAvatar($this->user->getDisplayName(), 1024);
|
||||||
|
$avatar = $this->folder->newFile('avatar.png');
|
||||||
|
$avatar->putContent($data);
|
||||||
|
$ext = 'png';
|
||||||
|
|
||||||
|
$this->folder->newFile('generated');
|
||||||
|
}
|
||||||
|
|
||||||
if ($size === -1) {
|
if ($size === -1) {
|
||||||
$path = 'avatar.' . $ext;
|
$path = 'avatar.' . $ext;
|
||||||
|
@ -179,19 +192,26 @@ class Avatar implements IAvatar {
|
||||||
throw new NotFoundException;
|
throw new NotFoundException;
|
||||||
}
|
}
|
||||||
|
|
||||||
$avatar = new OC_Image();
|
if ($this->folder->fileExists('generated')) {
|
||||||
/** @var ISimpleFile $file */
|
$data = $this->generateAvatar($this->user->getDisplayName(), $size);
|
||||||
$file = $this->folder->getFile('avatar.' . $ext);
|
|
||||||
$avatar->loadFromData($file->getContent());
|
} else {
|
||||||
if ($size !== -1) {
|
$avatar = new OC_Image();
|
||||||
|
/** @var ISimpleFile $file */
|
||||||
|
$file = $this->folder->getFile('avatar.' . $ext);
|
||||||
|
$avatar->loadFromData($file->getContent());
|
||||||
$avatar->resize($size);
|
$avatar->resize($size);
|
||||||
|
$data = $avatar->data();
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$file = $this->folder->newFile($path);
|
$file = $this->folder->newFile($path);
|
||||||
$file->putContent($avatar->data());
|
$file->putContent($data);
|
||||||
} catch (NotPermittedException $e) {
|
} catch (NotPermittedException $e) {
|
||||||
$this->logger->error('Failed to save avatar for ' . $this->user->getUID());
|
$this->logger->error('Failed to save avatar for ' . $this->user->getUID());
|
||||||
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $file;
|
return $file;
|
||||||
|
@ -211,4 +231,166 @@ class Avatar implements IAvatar {
|
||||||
}
|
}
|
||||||
throw new NotFoundException;
|
throw new NotFoundException;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $userDisplayName
|
||||||
|
* @param int $size
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private function generateAvatar($userDisplayName, $size) {
|
||||||
|
$text = strtoupper(substr($userDisplayName, 0, 1));
|
||||||
|
$backgroundColor = $this->avatarBackgroundColor($userDisplayName);
|
||||||
|
|
||||||
|
$im = imagecreatetruecolor($size, $size);
|
||||||
|
$background = imagecolorallocate($im, $backgroundColor[0], $backgroundColor[1], $backgroundColor[2]);
|
||||||
|
$white = imagecolorallocate($im, 255, 255, 255);
|
||||||
|
imagefilledrectangle($im, 0, 0, $size, $size, $background);
|
||||||
|
|
||||||
|
$font = __DIR__ . '/../../core/fonts/OpenSans-Semibold.woff';
|
||||||
|
|
||||||
|
$fontSize = $size * 0.4;
|
||||||
|
$box = imagettfbbox($fontSize, 0, $font, $text);
|
||||||
|
|
||||||
|
$x = ($size - ($box[2] - $box[0])) / 2;
|
||||||
|
$y = ($size - ($box[1] - $box[7])) / 2;
|
||||||
|
$x += 1;
|
||||||
|
$y -= $box[7];
|
||||||
|
imagettftext($im, $fontSize, 0, $x, $y, $white, $font, $text);
|
||||||
|
|
||||||
|
ob_start();
|
||||||
|
imagepng($im);
|
||||||
|
$data = ob_get_contents();
|
||||||
|
ob_end_clean();
|
||||||
|
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param int $r
|
||||||
|
* @param int $g
|
||||||
|
* @param int $b
|
||||||
|
* @return double[] Array containing h s l in [0, 1] range
|
||||||
|
*/
|
||||||
|
private function rgbToHsl($r, $g, $b) {
|
||||||
|
$r /= 255.0;
|
||||||
|
$g /= 255.0;
|
||||||
|
$b /= 255.0;
|
||||||
|
|
||||||
|
$max = max($r, $g, $b);
|
||||||
|
$min = min($r, $g, $b);
|
||||||
|
|
||||||
|
|
||||||
|
$h = ($max + $min) / 2.0;
|
||||||
|
$l = ($max + $min) / 2.0;
|
||||||
|
|
||||||
|
if($max === $min) {
|
||||||
|
$h = $s = 0; // Achromatic
|
||||||
|
} else {
|
||||||
|
$d = $max - $min;
|
||||||
|
$s = $l > 0.5 ? $d / (2 - $max - $min) : $d / ($max + $min);
|
||||||
|
switch($max) {
|
||||||
|
case $r:
|
||||||
|
$h = ($g - $b) / $d + ($g < $b ? 6 : 0);
|
||||||
|
break;
|
||||||
|
case $g:
|
||||||
|
$h = ($b - $r) / $d + 2.0;
|
||||||
|
break;
|
||||||
|
case $b:
|
||||||
|
$h = ($r - $g) / $d + 4.0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$h /= 6.0;
|
||||||
|
}
|
||||||
|
return [$h, $s, $l];
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $text
|
||||||
|
* @return int[] Array containting r g b in the range [0, 255]
|
||||||
|
*/
|
||||||
|
private function avatarBackgroundColor($text) {
|
||||||
|
$hash = preg_replace('/[^0-9a-f]+/', '', $text);
|
||||||
|
|
||||||
|
$hash = md5($hash);
|
||||||
|
$hashChars = str_split($hash);
|
||||||
|
|
||||||
|
|
||||||
|
// Init vars
|
||||||
|
$result = ['0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '0'];
|
||||||
|
$rgb = [0, 0, 0];
|
||||||
|
$sat = 0.70;
|
||||||
|
$lum = 0.68;
|
||||||
|
$modulo = 16;
|
||||||
|
|
||||||
|
|
||||||
|
// Splitting evenly the string
|
||||||
|
foreach($hashChars as $i => $char) {
|
||||||
|
$result[$i % $modulo] .= intval($char, 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Converting our data into a usable rgb format
|
||||||
|
// Start at 1 because 16%3=1 but 15%3=0 and makes the repartition even
|
||||||
|
for($count = 1; $count < $modulo; $count++) {
|
||||||
|
$rgb[$count%3] += (int)$result[$count];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reduce values bigger than rgb requirements
|
||||||
|
$rgb[0] %= 255;
|
||||||
|
$rgb[1] %= 255;
|
||||||
|
$rgb[2] %= 255;
|
||||||
|
|
||||||
|
$hsl = $this->rgbToHsl($rgb[0], $rgb[1], $rgb[2]);
|
||||||
|
|
||||||
|
// Classic formula to check the brightness for our eye
|
||||||
|
// If too bright, lower the sat
|
||||||
|
$bright = sqrt(0.299 * ($rgb[0] ** 2) + 0.587 * ($rgb[1] ** 2) + 0.114 * ($rgb[2] ** 2));
|
||||||
|
if ($bright >= 200) {
|
||||||
|
$sat = 0.60;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->hslToRgb($hsl[0], $sat, $lum);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param double $h Hue in range [0, 1]
|
||||||
|
* @param double $s Saturation in range [0, 1]
|
||||||
|
* @param double $l Lightness in range [0, 1]
|
||||||
|
* @return int[] Array containing r g b in the range [0, 255]
|
||||||
|
*/
|
||||||
|
private function hslToRgb($h, $s, $l){
|
||||||
|
$hue2rgb = function ($p, $q, $t){
|
||||||
|
if($t < 0) {
|
||||||
|
$t += 1;
|
||||||
|
}
|
||||||
|
if($t > 1) {
|
||||||
|
$t -= 1;
|
||||||
|
}
|
||||||
|
if($t < 1/6) {
|
||||||
|
return $p + ($q - $p) * 6 * $t;
|
||||||
|
}
|
||||||
|
if($t < 1/2) {
|
||||||
|
return $q;
|
||||||
|
}
|
||||||
|
if($t < 2/3) {
|
||||||
|
return $p + ($q - $p) * (2/3 - $t) * 6;
|
||||||
|
}
|
||||||
|
return $p;
|
||||||
|
};
|
||||||
|
|
||||||
|
if($s === 0){
|
||||||
|
$r = $l;
|
||||||
|
$g = $l;
|
||||||
|
$b = $l; // achromatic
|
||||||
|
}else{
|
||||||
|
$q = $l < 0.5 ? $l * (1 + $s) : $l + $s - $l * $s;
|
||||||
|
$p = 2 * $l - $q;
|
||||||
|
$r = $hue2rgb($p, $q, $h + 1/3);
|
||||||
|
$g = $hue2rgb($p, $q, $h);
|
||||||
|
$b = $hue2rgb($p, $q, $h - 1/3);
|
||||||
|
}
|
||||||
|
|
||||||
|
return array(round($r * 255), round($g * 255), round($b * 255));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -134,9 +134,7 @@ class AvatarControllerTest extends \Test\TestCase {
|
||||||
$response = $this->avatarController->getAvatar('userId', 32);
|
$response = $this->avatarController->getAvatar('userId', 32);
|
||||||
|
|
||||||
//Comment out until JS is fixed
|
//Comment out until JS is fixed
|
||||||
//$this->assertEquals(Http::STATUS_NOT_FOUND, $response->getStatus());
|
$this->assertEquals(Http::STATUS_NOT_FOUND, $response->getStatus());
|
||||||
$this->assertEquals(Http::STATUS_OK, $response->getStatus());
|
|
||||||
$this->assertEquals('displayName', $response->getData()['data']['displayname']);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -167,9 +165,7 @@ class AvatarControllerTest extends \Test\TestCase {
|
||||||
$response = $this->avatarController->getAvatar('userDoesNotExist', 32);
|
$response = $this->avatarController->getAvatar('userDoesNotExist', 32);
|
||||||
|
|
||||||
//Comment out until JS is fixed
|
//Comment out until JS is fixed
|
||||||
//$this->assertEquals(Http::STATUS_NOT_FOUND, $response->getStatus());
|
$this->assertEquals(Http::STATUS_NOT_FOUND, $response->getStatus());
|
||||||
$this->assertEquals(Http::STATUS_OK, $response->getStatus());
|
|
||||||
$this->assertEquals('userDoesNotExist', $response->getData()['data']['displayname']);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -12,6 +12,8 @@ use OC\Files\SimpleFS\SimpleFolder;
|
||||||
use OC\User\User;
|
use OC\User\User;
|
||||||
use OCP\Files\File;
|
use OCP\Files\File;
|
||||||
use OCP\Files\Folder;
|
use OCP\Files\Folder;
|
||||||
|
use OCP\Files\NotFoundException;
|
||||||
|
use OCP\Files\SimpleFS\ISimpleFile;
|
||||||
use OCP\IConfig;
|
use OCP\IConfig;
|
||||||
use OCP\IL10N;
|
use OCP\IL10N;
|
||||||
use OCP\ILogger;
|
use OCP\ILogger;
|
||||||
|
@ -49,7 +51,35 @@ class AvatarTest extends \Test\TestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testGetNoAvatar() {
|
public function testGetNoAvatar() {
|
||||||
$this->assertEquals(false, $this->avatar->get());
|
$file = $this->createMock(ISimpleFile::class);
|
||||||
|
$this->folder->method('newFile')
|
||||||
|
->willReturn($file);
|
||||||
|
|
||||||
|
$this->folder->method('getFile')
|
||||||
|
->will($this->returnCallback(function($path) {
|
||||||
|
if ($path === 'avatar.64.png') {
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
$this->folder->method('fileExists')
|
||||||
|
->will($this->returnCallback(function($path) {
|
||||||
|
if ($path === 'generated') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}));
|
||||||
|
|
||||||
|
$data = NULL;
|
||||||
|
$file->method('putContent')
|
||||||
|
->with($this->callback(function ($d) use (&$data) {
|
||||||
|
$data = $d;
|
||||||
|
return true;
|
||||||
|
}));
|
||||||
|
|
||||||
|
$file->method('getContent')
|
||||||
|
->willReturn($data);
|
||||||
|
|
||||||
|
$this->assertEquals($data, $this->avatar->get()->data());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testGetAvatarSizeMatch() {
|
public function testGetAvatarSizeMatch() {
|
||||||
|
@ -161,13 +191,13 @@ class AvatarTest extends \Test\TestCase {
|
||||||
->willReturn('avatar.32.jpg');
|
->willReturn('avatar.32.jpg');
|
||||||
$resizedAvatarFile->expects($this->once())->method('delete');
|
$resizedAvatarFile->expects($this->once())->method('delete');
|
||||||
|
|
||||||
$nonAvatarFile = $this->createMock(File::class);
|
|
||||||
$nonAvatarFile->method('getName')
|
|
||||||
->willReturn('avatarX');
|
|
||||||
$nonAvatarFile->expects($this->never())->method('delete');
|
|
||||||
|
|
||||||
$this->folder->method('getDirectoryListing')
|
$this->folder->method('getDirectoryListing')
|
||||||
->willReturn([$avatarFileJPG, $avatarFilePNG, $resizedAvatarFile, $nonAvatarFile]);
|
->willReturn([$avatarFileJPG, $avatarFilePNG, $resizedAvatarFile]);
|
||||||
|
|
||||||
|
$generated = $this->createMock(File::class);
|
||||||
|
$this->folder->method('getFile')
|
||||||
|
->with('generated')
|
||||||
|
->willReturn($generated);
|
||||||
|
|
||||||
$newFile = $this->createMock(File::class);
|
$newFile = $this->createMock(File::class);
|
||||||
$this->folder->expects($this->once())
|
$this->folder->expects($this->once())
|
||||||
|
|
Loading…
Reference in a new issue