upload and change avatars
code to upload and change avatars. combined some code in the settings area, too. darcs-hash:20080517122045-84dde-8e13994e627805f29679c9533c2f62db81dc0925.gz
This commit is contained in:
parent
fac522f4d7
commit
3803cf2153
208
actions/avatar.php
Normal file
208
actions/avatar.php
Normal file
@ -0,0 +1,208 @@
|
|||||||
|
<?php
|
||||||
|
/*
|
||||||
|
* Laconica - a distributed open-source microblogging tool
|
||||||
|
* Copyright (C) 2008, Controlez-Vous, Inc.
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program 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 Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (!defined('LACONICA')) { exit(1) }
|
||||||
|
|
||||||
|
class AvatarAction extends SettingsAction {
|
||||||
|
|
||||||
|
function show_form($msg=NULL, $success=false) {
|
||||||
|
common_show_header(_t('Avatar'));
|
||||||
|
$this->settings_menu();
|
||||||
|
$this->message($msg, $success);
|
||||||
|
|
||||||
|
$user = common_current_user();
|
||||||
|
$profile = $user->getProfile();
|
||||||
|
$original = $profile->getOriginal();
|
||||||
|
|
||||||
|
if ($original) {
|
||||||
|
common_element('img', array('src' => $original->url,
|
||||||
|
'class' => 'avatar original',
|
||||||
|
'width' => $original->width,
|
||||||
|
'height' => $original->height));
|
||||||
|
}
|
||||||
|
|
||||||
|
$avatar = $profile->getAvatar(AVATAR_PROFILE_SIZE);
|
||||||
|
|
||||||
|
if ($avatar) {
|
||||||
|
common_element('img', array('src' => $avatar->url,
|
||||||
|
'class' => 'avatar profile',
|
||||||
|
'width' => AVATAR_PROFILE_SIZE,
|
||||||
|
'height' => AVATAR_PROFILE_SIZE));
|
||||||
|
}
|
||||||
|
|
||||||
|
common_start_element('form', array('enctype' => 'multipart/form-data',
|
||||||
|
'method' => 'POST',
|
||||||
|
'id' => 'avatar',
|
||||||
|
'action' =>
|
||||||
|
common_local_url('avatar')));
|
||||||
|
common_element('input', array('name' => 'MAX_FILE_SIZE',
|
||||||
|
'type' => 'hidden',
|
||||||
|
'id' => 'MAX_FILE_SIZE',
|
||||||
|
'value' => MAX_AVATAR_SIZE));
|
||||||
|
common_element('input', array('name' => 'avatarfile',
|
||||||
|
'type' => 'file',
|
||||||
|
'id' => 'avatarfile'));
|
||||||
|
common_element('input', array('name' => 'submit',
|
||||||
|
'type' => 'submit',
|
||||||
|
'id' => 'submit'),
|
||||||
|
_t('Upload'));
|
||||||
|
}
|
||||||
|
|
||||||
|
function handle_post() {
|
||||||
|
|
||||||
|
switch ($_FILES['avatarfile']['error']) {
|
||||||
|
case UPLOAD_ERR_OK: # success, jump out
|
||||||
|
break;
|
||||||
|
case UPLOAD_ERR_INI_SIZE:
|
||||||
|
case UPLOAD_ERR_FORM_SIZE:
|
||||||
|
$this->show_form(_t('That file is too big.'));
|
||||||
|
return;
|
||||||
|
case UPLOAD_ERR_PARTIAL:
|
||||||
|
@unlink($_FILES['avatarfile']['tmp_name']);
|
||||||
|
$this->show_form(_t('Partial upload.'));
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
$this->show_form(_t('System error uploading file.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$info = @getimagesize($_FILES['avatarfile']['tmp_name']);
|
||||||
|
|
||||||
|
if (!$info) {
|
||||||
|
@unlink($_FILES['avatarfile']['tmp_name']);
|
||||||
|
$this->show_form(_t('Not an image or corrupt file.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch ($info[2]) {
|
||||||
|
case IMAGETYPE_GIF:
|
||||||
|
case IMAGETYPE_JPEG:
|
||||||
|
case IMAGETYPE_PNG:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$this->show_form(_t('Unsupported image file format.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = common_current_user();
|
||||||
|
|
||||||
|
$filename = common_avatar_filename($user, image_type_to_extension($info[2]));
|
||||||
|
$filepath = common_avatar_path($filename);
|
||||||
|
|
||||||
|
if (!move_uploaded_file($_FILES['avatarfile']['tmp_name'], $filepath)) {
|
||||||
|
@unlink($_FILES['avatarfile']['tmp_name']);
|
||||||
|
$this->show_form(_t('System error uploading file.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$avatar = DB_DataObject::factory('avatar');
|
||||||
|
|
||||||
|
$avatar->profile_id = $user->id;
|
||||||
|
$avatar->width = $info[0];
|
||||||
|
$avatar->height = $info[1];
|
||||||
|
$avatar->mediatype = image_type_to_mime_type($info[2]);
|
||||||
|
$avatar->filename = $filename;
|
||||||
|
$avatar->original = true;
|
||||||
|
$avatar->url = common_avatar_url($filename);
|
||||||
|
|
||||||
|
foreach (array(AVATAR_PROFILE_SIZE, AVATAR_STREAM_SIZE, AVATAR_MINI_SIZE) as $size) {
|
||||||
|
$scaled[] = $this->scale_avatar($user, $avatar, $size);
|
||||||
|
}
|
||||||
|
|
||||||
|
# XXX: start a transaction here
|
||||||
|
|
||||||
|
if (!$this->delete_old_avatars($user)) {
|
||||||
|
@unlink($filepath);
|
||||||
|
common_server_error(_t('Error deleting old avatars.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$avatar->insert()) {
|
||||||
|
@unlink($filepath);
|
||||||
|
common_server_error(_t('Error inserting avatar.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($scaled as $s) {
|
||||||
|
if (!$s->insert()) {
|
||||||
|
common_server_error(_t('Error inserting scaled avatar.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# XXX: end transaction here
|
||||||
|
|
||||||
|
$this->show_form(_t('Avatar updated.'), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function scale_avatar($user, $avatar, $size) {
|
||||||
|
$image_s = imagecreatetruecolor($size, $size);
|
||||||
|
$image_a = $this->avatar_to_image($avatar);
|
||||||
|
|
||||||
|
$square = min($avatar->width, $avatar->height);
|
||||||
|
|
||||||
|
imagecopyresampled($image_s, $image_a, 0, 0, 0, 0,
|
||||||
|
$size, $size, $square, $square);
|
||||||
|
|
||||||
|
$ext = ($avatar->mediattype == 'image/jpeg') ? ".jpg" : ".png";
|
||||||
|
|
||||||
|
$filename = common_avatar_filename($user, $ext, $size);
|
||||||
|
|
||||||
|
if ($avatar->mediatype == 'image/jpeg') {
|
||||||
|
imagejpeg($image_s, common_avatar_path($filename));
|
||||||
|
} else {
|
||||||
|
imagepng($image_s, common_avatar_path($filename));
|
||||||
|
}
|
||||||
|
|
||||||
|
$scaled = DB_DataObject::factory('avatar');
|
||||||
|
$scaled->profile_id = $avatar->profile_id;
|
||||||
|
$scaled->width = $size;
|
||||||
|
$scaled->height = $size;
|
||||||
|
$scaled->original = false;
|
||||||
|
$scaled->mediatype = ($avatar->mediattype == 'image/jpeg') ? 'image/jpeg' : 'image/png';
|
||||||
|
$scaled->filename = $filename;
|
||||||
|
$scaled->url = common_avatar_url($filename);
|
||||||
|
|
||||||
|
return $scaled;
|
||||||
|
}
|
||||||
|
|
||||||
|
function avatar_to_image($avatar) {
|
||||||
|
$filepath = common_avatar_path($avatar->filename);
|
||||||
|
if ($avatar->mediatype == 'image/gif') {
|
||||||
|
return imagecreatefromgif($filepath);
|
||||||
|
} else if ($avatar->mediatype == 'image/jpeg') {
|
||||||
|
return imagecreatefromjpeg($filepath);
|
||||||
|
} else if ($avatar->mediatype == 'image/png') {
|
||||||
|
return imagecreatefrompng($filepath);
|
||||||
|
} else {
|
||||||
|
common_server_error(_t('Unsupported image type:') . $avatar->mediatype);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function delete_old_avatars($user) {
|
||||||
|
$avatar = DB_DataObject::factory('avatar');
|
||||||
|
$avatar->profile_id = $user->id;
|
||||||
|
$avatar->find();
|
||||||
|
while ($avatar->fetch()) {
|
||||||
|
$avatar->delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -21,26 +21,10 @@ if (!defined('LACONICA')) { exit(1) }
|
|||||||
|
|
||||||
class PasswordAction extends SettingsAction {
|
class PasswordAction extends SettingsAction {
|
||||||
|
|
||||||
function handle($args) {
|
|
||||||
parent::handle($args);
|
|
||||||
if (!common_logged_in()) {
|
|
||||||
common_user_error(_t('Not logged in.'));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if ($this->arg('METHOD') == 'POST') {
|
|
||||||
$this->handle_post();
|
|
||||||
} else {
|
|
||||||
$this->show_form();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function show_form($msg=NULL, $success=false) {
|
function show_form($msg=NULL, $success=false) {
|
||||||
common_show_header(_t('Change password'));
|
common_show_header(_t('Change password'));
|
||||||
$this->settings_menu();
|
$this->settings_menu();
|
||||||
if ($msg) {
|
$this->message($msg, $success);
|
||||||
common_element('div', ($success) ? 'success' : 'error',
|
|
||||||
$msg);
|
|
||||||
}
|
|
||||||
common_start_element('form', array('method' => 'POST',
|
common_start_element('form', array('method' => 'POST',
|
||||||
'id' => 'password',
|
'id' => 'password',
|
||||||
'action' =>
|
'action' =>
|
||||||
|
@ -21,26 +21,10 @@ if (!defined('LACONICA')) { exit(1) }
|
|||||||
|
|
||||||
class ProfilesettingsAction extends SettingsAction {
|
class ProfilesettingsAction extends SettingsAction {
|
||||||
|
|
||||||
function handle($args) {
|
|
||||||
parent::handle($args);
|
|
||||||
if (!common_logged_in()) {
|
|
||||||
common_user_error(_t('Not logged in.'));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if ($this->arg('METHOD') == 'POST') {
|
|
||||||
$this->handle_post();
|
|
||||||
} else {
|
|
||||||
$this->show_form();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function show_form($msg=NULL, $success=false) {
|
function show_form($msg=NULL, $success=false) {
|
||||||
common_show_header(_t('Profile settings'));
|
common_show_header(_t('Profile settings'));
|
||||||
$this->settings_menu();
|
$this->settings_menu();
|
||||||
if ($msg) {
|
$this->message($msg, $success);
|
||||||
common_element('div', ($success) ? 'success' : 'error',
|
|
||||||
$msg);
|
|
||||||
}
|
|
||||||
common_start_element('form', array('method' => 'POST',
|
common_start_element('form', array('method' => 'POST',
|
||||||
'id' => 'profilesettings',
|
'id' => 'profilesettings',
|
||||||
'action' =>
|
'action' =>
|
||||||
|
@ -16,10 +16,11 @@ create table profile (
|
|||||||
|
|
||||||
create table avatar (
|
create table avatar (
|
||||||
profile_id integer not null comment 'foreign key to profile table' references profile (id),
|
profile_id integer not null comment 'foreign key to profile table' references profile (id),
|
||||||
|
original boolean default false comment 'uploaded by user or generated?',
|
||||||
width integer not null comment 'image width',
|
width integer not null comment 'image width',
|
||||||
height integer not null comment 'image height',
|
height integer not null comment 'image height',
|
||||||
original boolean default false comment 'uploaded by user or generated?',
|
|
||||||
mediatype varchar(32) not null comment 'file type',
|
mediatype varchar(32) not null comment 'file type',
|
||||||
|
filename varchar(255) null comment 'local filename, if local',
|
||||||
url varchar(255) unique key comment 'avatar location',
|
url varchar(255) unique key comment 'avatar location',
|
||||||
|
|
||||||
constraint primary key (profile_id, width, height),
|
constraint primary key (profile_id, width, height),
|
||||||
|
11
doc/TODO
11
doc/TODO
@ -1,7 +1,7 @@
|
|||||||
+ login
|
+ login
|
||||||
+ register
|
+ register
|
||||||
+ settings
|
+ settings
|
||||||
- upload avatar
|
+ upload avatar
|
||||||
- default avatar
|
- default avatar
|
||||||
+ change password
|
+ change password
|
||||||
+ settings menu
|
+ settings menu
|
||||||
@ -18,9 +18,10 @@
|
|||||||
+ header menu
|
+ header menu
|
||||||
+ footer menu
|
+ footer menu
|
||||||
+ disallow direct to PHP files
|
+ disallow direct to PHP files
|
||||||
|
- require valid nicknames
|
||||||
|
- store canonical username for comparison and fetch
|
||||||
- use only canonical usernames
|
- use only canonical usernames
|
||||||
- use only canonical email addresses
|
- use only canonical email addresses
|
||||||
- require valid nicknames
|
|
||||||
- common_local_url()
|
- common_local_url()
|
||||||
- configuration system ($config)
|
- configuration system ($config)
|
||||||
- RSS 1.0 feeds of a user's notices
|
- RSS 1.0 feeds of a user's notices
|
||||||
@ -28,10 +29,12 @@
|
|||||||
- RSS 1.0 feed of all public notices
|
- RSS 1.0 feed of all public notices
|
||||||
- RDF dump of entire site
|
- RDF dump of entire site
|
||||||
- FOAF dump for user
|
- FOAF dump for user
|
||||||
- delete a notice
|
- license on showstream
|
||||||
- licenses
|
- license on shownotice
|
||||||
- design from Open Source Web Designs
|
- design from Open Source Web Designs
|
||||||
|
- TOS checkbox on register
|
||||||
- release 0.1
|
- release 0.1
|
||||||
|
- delete a notice
|
||||||
- gettext
|
- gettext
|
||||||
- subscribe remote
|
- subscribe remote
|
||||||
- add subscriber remote
|
- add subscriber remote
|
||||||
|
@ -22,6 +22,7 @@ if (!defined('LACONICA')) { exit(1) }
|
|||||||
define('AVATAR_PROFILE_SIZE', 96);
|
define('AVATAR_PROFILE_SIZE', 96);
|
||||||
define('AVATAR_STREAM_SIZE', 48);
|
define('AVATAR_STREAM_SIZE', 48);
|
||||||
define('AVATAR_MINI_SIZE', 24);
|
define('AVATAR_MINI_SIZE', 24);
|
||||||
|
define('MAX_AVATAR_SIZE', 256 * 1024);
|
||||||
|
|
||||||
# global configuration object
|
# global configuration object
|
||||||
|
|
||||||
@ -30,6 +31,9 @@ define('AVATAR_MINI_SIZE', 24);
|
|||||||
$config =
|
$config =
|
||||||
array('site' =>
|
array('site' =>
|
||||||
array('name' => 'Just another µB'),
|
array('name' => 'Just another µB'),
|
||||||
|
'avatar' =>
|
||||||
|
array('directory' => INSTALLDIR . 'files',
|
||||||
|
'path' => '/files'),
|
||||||
'dsn' =>
|
'dsn' =>
|
||||||
array('phptype' => 'mysql',
|
array('phptype' => 'mysql',
|
||||||
'username' => 'stoica',
|
'username' => 'stoica',
|
||||||
@ -228,6 +232,28 @@ function common_render_content($text) {
|
|||||||
return htmlspecialchars($text);
|
return htmlspecialchars($text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// where should the avatar go for this user?
|
||||||
|
|
||||||
|
function common_avatar_filename($user, $extension, $size=NULL) {
|
||||||
|
global $config;
|
||||||
|
|
||||||
|
if ($size) {
|
||||||
|
return $user->id . '-' . $size . $extension;
|
||||||
|
} else {
|
||||||
|
return $user->id . '-original' . $extension;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function common_avatar_path($filename) {
|
||||||
|
global $config;
|
||||||
|
return $config['avatar']['directory'] . '/' . $filename;
|
||||||
|
}
|
||||||
|
|
||||||
|
function common_avatar_url($filename) {
|
||||||
|
global $config;
|
||||||
|
return $config['avatar']['path'] . '/' . $filename;
|
||||||
|
}
|
||||||
|
|
||||||
// XXX: set up gettext
|
// XXX: set up gettext
|
||||||
|
|
||||||
function _t($str) { $str }
|
function _t($str) { $str }
|
||||||
|
@ -23,6 +23,31 @@ class SettingsAction extends Action {
|
|||||||
|
|
||||||
function handle($args) {
|
function handle($args) {
|
||||||
parent::handle($args);
|
parent::handle($args);
|
||||||
|
if (!common_logged_in()) {
|
||||||
|
common_user_error(_t('Not logged in.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if ($this->arg('METHOD') == 'POST') {
|
||||||
|
$this->handle_post();
|
||||||
|
} else {
|
||||||
|
$this->show_form();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# override!
|
||||||
|
function handle_post() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function show_form($msg=NULL, $success=false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function show_message($msg, $success) {
|
||||||
|
if ($msg) {
|
||||||
|
common_element('div', ($success) ? 'success' : 'error',
|
||||||
|
$msg);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function settings_menu() {
|
function settings_menu() {
|
||||||
|
Loading…
Reference in New Issue
Block a user