Skip to content

Commit

Permalink
Identicon plugin
Browse files Browse the repository at this point in the history
  • Loading branch information
alecpl committed Dec 2, 2016
1 parent 8fc488a commit bbab6a6
Show file tree
Hide file tree
Showing 6 changed files with 302 additions and 0 deletions.
1 change: 1 addition & 0 deletions CHANGELOG
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
CHANGELOG Roundcube Webmail
===========================

- Added identicon plugin
- Widescreen layout aka three column view (#5093)
- Unify automatic marking as \Seen in preview pane, full-page and extwin views (#5071)
- Disable double-click on the list when preview pane is on (#5199)
Expand Down
25 changes: 25 additions & 0 deletions plugins/identicon/composer.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
{
"name": "roundcube/identicon",
"type": "roundcube-plugin",
"description": "Displays Github-like identicons for contacts/addresses without photo specified.",
"license": "GPLv3+",
"version": "0.1",
"authors": [
{
"name": "Aleksander Machniak",
"email": "[email protected]",
"role": "Lead"
}
],
"repositories": [
{
"type": "composer",
"url": "http://plugins.roundcube.net"
}
],
"require": {
"php": ">=5.3.0",
"php-gd": "*",
"roundcube/plugin-installer": ">=0.1.3"
}
}
78 changes: 78 additions & 0 deletions plugins/identicon/identicon.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
<?php

/**
* Identicon
*
* Plugin to display a unique github-like identification icons
* for contacts/addresses that do not have a photo image.
*
* @todo: Make it optional and configurable via user preferences
* @todo: Make color palettes match the curren skin
* @todo: Implement optional SVG generator
*
* @license GNU GPLv3+
* @author Aleksander Machniak <[email protected]>
* @website http://roundcube.net
*/
class identicon extends rcube_plugin
{
public $task = 'addressbook';


/**
* Plugin initilization.
*/
function init()
{
$this->add_hook('contact_photo', array($this, 'contact_photo'));
}

/**
* 'contact_photo' hook handler to inject an identicon image
*/
function contact_photo($args)
{
// pre-conditions, exit if photo already exists or invalid input
if (!empty($args['url']) || !empty($args['data'])
|| (empty($args['record']) && empty($args['email']))
) {
return $args;
}

$rcmail = rcmail::get_instance();

// supporting edit/add action may be tricky, let's not do this
if ($rcmail->action == 'show' || $rcmail->action == 'photo') {
$email = $args['email'];
if (!$email && $args['record']) {
$addresses = rcube_addressbook::get_col_values('email', $args['record'], true);
if (!empty($addresses)) {
$email = $addresses[0];
}
}

if ($email) {
require_once __DIR__ . '/identicon_engine.php';

$identicon = new identicon_engine($email);

if ($rcmail->action == 'show') {
// set photo URL using data-uri
if (($icon = $identicon->getBinary()) && ($icon = base64_encode($icon))) {
$mimetype =$identicon->getMimetype();
$args['url'] = sprintf('data:%s;base64,%s', $mimetype, $icon);
}
}
else {
// send the icon to the browser
$identicon = new identicon_engine($email);
if ($identicon->sendOutput()) {
exit;
}
}
}
}

return $args;
}
}
175 changes: 175 additions & 0 deletions plugins/identicon/identicon_engine.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,175 @@
<?php

/**
* @license GNU GPLv3+
* @author Aleksander Machniak <[email protected]>
*/
class identicon_engine
{
private $ident;
private $width;
private $height;
private $margin;
private $binary;
private $color;
private $bgcolor = '#F9F9F9';
private $mimetype = 'image/png';
private $palette = array(
'#F44336', '#E91E63', '#9C27B0', '#673AB7', '#3F51B5', '#2196F3',
'#03A9F4', '#00BCD4', '#009688', '#4CAF50', '#8BC34A', '#CDDC39',
'#FFEB3B', '#FFC107', '#FF9800', '#FF5722', '#795548', '#607D8B',
);
private $grid = array(
0, 1, 2, 1, 0,
3, 4, 5, 4, 3,
6, 7, 8, 7, 6,
9, 10, 11, 10, 9,
12, 13, 14, 13, 12,
);

const GRID_SIZE = 5;
const ICON_SIZE = 150;


/**
* Class constructor
*
* @param string $ident Unique identifier (email address)
* @param int $size Icon size in pixels
*/
public function __construct($ident, $size = null)
{
if (!$size) {
$size = self::ICON_SIZE;
}

$this->ident = $ident;
$this->margin = (int) round($size / 10);
$this->width = (int) round(($size - $this->margin * 2) / self::GRID_SIZE) * self::GRID_SIZE + $this->margin * 2;
$this->height = $this->width;

$this->generate();
}

/**
* Returns image mimetype
*/
public function getMimetype()
{
return $this->mimetype;
}

/**
* Returns the image in binary form
*/
public function getBinary()
{
return $this->binary;
}

/**
* Sends the image to the browser
*/
public function sendOutput()
{
if ($this->binary) {
$rcmail = rcmail::get_instance();
$rcmail->output->future_expire_header(10 * 60);

header('Content-Type: ' . $this->mimetype);
header('Content-Size: ' . strlen($this->binary));
echo $this->binary;

return true;
}

return false;
}

/**
* Icon generator
*/
private function generate()
{
$ident = md5($this->ident, true);

// set icon color
$div = intval(255/count($this->palette));
$index = intval(ord($ident[0]) / $div);
$this->color = $this->palette[$index] ?: $this->palette[0];

// set cell size
$cell_width = ($this->width - $this->margin * 2) / self::GRID_SIZE;
$cell_height = ($this->height - $this->margin * 2) / self::GRID_SIZE;

// create a grid
foreach ($this->grid as $i => $idx) {
$row_num = intval($i / self::GRID_SIZE);
$cell_num_h = $i - $row_num * self::GRID_SIZE;

$this->grid[$i] = array(
'active' => ord($ident[$idx]) % 2 > 0,
'x1' => $cell_width * $cell_num_h + $this->margin,
'y1' => $cell_height * $row_num + $this->margin,
'x2' => $cell_width * ($cell_num_h + 1) + $this->margin,
'y2' => $cell_height * ($row_num + 1) + $this->margin,
);
}

// really generate the image using supported methods
if (function_exists('imagepng')) {
$this->generateGD();
}
else {
// log an error
$error = array(
'code' => 500,
'message' => "PHP-GD module not found. It's required by identicon plugin.",
);

rcube::raise_error($error, true, false);
}
}

/**
* GD-based icon generation worker
*/
private function generateGD()
{
$color = $this->toRGB($this->color);
$bgcolor = $this->toRGB($this->bgcolor);

// create an image, setup colors
$image = imagecreate($this->width, $this->height);
$color = imagecolorallocate($image, $color[0], $color[1], $color[2]);
$bgcolor = imagecolorallocate($image, $bgcolor[0], $bgcolor[1], $bgcolor[2]);

imagefilledrectangle($image, 0, 0, $this->width, $this->height, $bgcolor);

// draw the grid created in self::generate()
foreach ($this->grid as $item) {
if ($item['active']) {
imagefilledrectangle($image, $item['x1'], $item['y1'], $item['x2'], $item['y2'], $color);
}
}

// generate an image and save it to a variable
ob_start();
imagepng($image, null, 6, PNG_ALL_FILTERS);
$this->binary = ob_get_contents();
ob_end_clean();

// cleanup
imagedestroy($image);
}

/**
* Convert #FFFFFF color format to 3-value RGB
*/
private function toRGB($color)
{
preg_match('/^#?([A-F0-9]{2})([A-F0-9]{2})([A-F0-9]{2})/i', $color, $m);

return array(hexdec($m[1]), hexdec($m[2]), hexdec($m[3]));
}
}
22 changes: 22 additions & 0 deletions plugins/identicon/tests/Identicon.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
<?php

class Identicon_Plugin extends PHPUnit_Framework_TestCase
{

function setUp()
{
include_once __DIR__ . '/../identicon.php';
}

/**
* Plugin object construction test
*/
function test_constructor()
{
$rcube = rcube::get_instance();
$plugin = new identicon($rcube->api);

$this->assertInstanceOf('identicon', $plugin);
$this->assertInstanceOf('rcube_plugin', $plugin);
}
}
1 change: 1 addition & 0 deletions tests/phpunit.xml
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@
<file>./../plugins/help/tests/Help.php</file>
<file>./../plugins/hide_blockquote/tests/HideBlockquote.php</file>
<file>./../plugins/http_authentication/tests/HttpAuthentication.php</file>
<file>./../plugins/identicon/tests/Identicon.php</file>
<file>./../plugins/identity_select/tests/IdentitySelect.php</file>
<file>./../plugins/jqueryui/tests/Jqueryui.php</file>
<file>./../plugins/krb_authentication/tests/KrbAuthentication.php</file>
Expand Down

0 comments on commit bbab6a6

Please sign in to comment.