PHP hates me wird 500!
Heute hat DER deutschsprachige PHP Blog seinen 500sten (und inzwischen auch schon seinen 501ten) Eintrag veröffentlicht. Und das nur ca. 1 1/2 Jahre nach dem ersten Artikel. Das ist fast ein Artikel pro Tag! Gratulieren möchte ich Nils Langner und seinen Mitautoren aber eigentlich nicht zum 500 Eintrag, sondern zu den wahnsinnig interessanten Inhalten.
PHP hates me unterscheidet sich nämlich nicht nur durch die Frequenz der Beiträge von anderen PHP Blogs. Vor allem die Gedanken, die sich die Autoren,allen voran Nils, über alltägliche Probleme eines (Web-)Entwicklers machen, machen den Blog für mich so lesenswert. Er formuliert viele Gedanken, über die man ab und zu mal drüberfällt aber nie zu Ende denkt aus, und die überaus eifrige Leserschaft gibt ausführlich und gesittet ihren Senf dazu. So soll es sein, danke!
Bildbearbeitung mit PHP
Dateien hochladen
Dieser erste Teil hat noch nicht unbedingt was mit Bildverarbeitung zu tun, aber meistens kommen die Bilder ja über die PHP-Applikation erst auf den Server.
Um eine Datei auf den Webserver zu laden benötigt man ein Formular, das in seiner einfachsten Form ungefähr so aussehen könnte:
<form name="uploadform" method="post" enctype="multipart/form-data" action=""> <input type="file" name="thefile" /> <input name="thesubmit" type="submit" value="Upload image" /> </form>
Wichtig ist, dass als Methode POST angegeben wird und als MIME-Typ (enctype) multipart/form-data, da der Dateiupload sonst nicht funktioniert.
Im Ziel-Script (das im action-Attribut angegeben werden kann, wenn nichts angegeben wird, wird das selbe Script wieder geladen) kann man nun die hochgeladene Datei überprüfen und in ein Verzeichnis seiner Wahl kopieren:
<?php
//überprüfen, ob das Formular gesendet wurde
if(isset($_POST['thesubmit'])){
//Metadaten, die uns PHP über die Datei liefert ausgeben
echo $_FILES['thefile']['name'] . "<br />";
echo $_FILES['thefile']['type'] . "<br />";
echo $_FILES['thefile']['tmp_name'] . "<br />";
echo $_FILES['thefile']['error'] . "<br />";
echo $_FILES['thefile']['size'] . "<br />";
//überprüfen des MIME-Typs
$mime = $_FILES['thefile']['type'];
if($mime == 'image/gif' || $mime == 'image/jpeg' || $mime == 'image/png'){
//aktueller Pfad und Dateiname des Bildes
$oldfile = $_FILES['thefile']['tmp_name'];
//vorgesehener Pfad und Dateiname des Bildes
$newfile = 'uploads/'.$_FILES['thefile']['name'];
//datei kopieren
move_uploaded_file($oldfile, $newfile);
}
}
?>
Das Bild wird also hochgeladen, überprüft ob es sich um ein Bild handelt, dann wird es aus dem Temporären Verzeichnis in ein beliebiges neues kopiert, in unserem Fall uploads/, mit dem Originalnamen dahinter.
Achtung! Die Überprüfung des MIME-Typs (oder auch der Dateiendung) genügt noch nicht als sichere Überprüfung der Datei. Eine bessere Methode ist, mit den weiter unten vorgestellten Funktionen zu prüfen, ob das Bild tatsächlich eine Höhe und eine Breite hat. Ist auch keine 100% sichere Methode, aber schon mal besser als gar keine.
Auch der Dateiname sollte geändert werden, der Originalname sollte niemals übernommen werden.
Bild von Datei laden
Dafür gibts drei Funktionen für gif, jpeg und png:
- resource imagecreatefromgif ( string $filename )
- resource imagecreatefromjpeg ( string $filename )
- resource imagecreatefrompng( string $filename )
Dadurch erhält man einen Zeiger auf die Grafik, z.B.:
$image = imagecreatefrompng( 'einbild.png' );
Daten eines Bildes - Metadaten
Für viele Anwendungen in der Bildverarbeitung ist es wichtig, Daten über das Bild zu erhalten. Dazu gehören unter anderem Höhe, Breite, Typ, Farbtiefe usw. Die gängigste Funktion für diese Metadaten ist getimagesize(). Diese Funktion liefert ein Array mit 7 Elementen, die verschiedene Metainformationen beinhalten.
- Index 0 Breite des Bildes
- Index 1 Höhe des Bildes
- Index 2 ist eine der IMAGETYPE-Konstanten (IMAGETYPE_GIF, IMAGETYPE_JPEG, IMAGETYPE_PNG, IMAGETYPE_SWF, IMAGETYPE_PSD, IMAGETYPE_BMP, IMAGETYPE_WBMP, IMAGETYPE_XBM, IMAGETYPE_TIFF_II, IMAGETYPE_TIFF_MM, IMAGETYPE_IFF, IMAGETYPE_JB2, IMAGETYPE_JPC, IMAGETYPE_JP2, IMAGETYPE_JPX, IMAGETYPE_SWC, IMAGETYPE_ICO)
- Index 3 Textstring im Format 'height="yyy" width="xxx"'für ein IMG-Tag.
- Index mime Der MIME Typ des Bildes
- Index channels 3 für RGB und 4 für CMYK Bilder.
- Index bits Die Farbtiefe in Bit pro Farbe .
Größe eines Bildes ändern
Um ein Bild zu skalieren, braucht man mehrere Funktionen, die man am besten in einer Funktion zusammenfasst:
public static function scaleImage($image, $maxwidth, $maxheight){
list($image_width, $image_height) = getimagesize($file);
if($image_width > $image_height){
$new_image_width = $maxwidth;
$new_image_height = $image_height*($maxheight/$image_height);
}else{
$new_image_height = $maxheight;
$new_image_width = $image_width * ($maxwidth/$image_width);
}
$new_image = imagecreatetruecolor(round($new_image_width),
round($new_image_height));
imagecopyresized($new_image, $image, 0, 0, 0, 0, $new_image_width,
$new_image_height, $image_width, $image_height);
return $new_image;
}
Die Funktion nimmt als Parameter einen Zeiger und die gewünschte maximale Breite und maxiamle Höhe entgegen. Je nachdem, ob Höhe oder Breite größer ist, wird dann der andere Wert berechnet.
Danach wird ein neues Bild mit der berchneten Größe erstellt.
Schlußendlich kopiert die Funktion imagecopyresized() das alte Bild in das neue Bild.
Die verwendeten Funktionen:
resource imagecreatetruecolor ( int $width , int $height ) bool imagecopyresized ($to, $from, $to_x, $to_y, $from_x, $from_y, $to_w, $to_h, $from_w, $from_h)
Bild speichern bzw. ausgeben
Um ein Bild zu speichern oder im Dokument auszugeben gibt es wieder für jedes Format die entsprechenden Funktionen:
bool imagegif ( resource $image [, string $filename ] ) bool imagejpeg ( resource $image [, string $filename [, int $quality ]] ) bool imagepng ( resource $image [, string $filename [, int $quality [, int $filters ]]] )
Das Bild wird einfach im Skript ausgegeben, wenn kein Dateiname angegeben ist.
Die Qualität wird für JPEGs von 0-100 und bei PNGs von 0-9 angegeben.
$filters kann eine der Konstanten PNG_NO_FILTER, PNG_FILTER_NONE, PNG_FILTER_SUB , PNG_FILTER_UP, PNG_FILTER_AVG, PNG_FILTER_PAETH oder PNG_ALL_FILTERS sein.
Neues, leeres Bild erstellen
Das geschieht durch die Funktion
resource imagecreate ( int $width , int $height )
Einfach Breite und Höhe angeben und den zurückgegebenen Zeiger in einer Variable speichern.
Farben definieren
Zu Beginn ist die Farbpalette des Bildes leer, das heißt man muss jede neue Farbe vor jeder Operation definieren:
int imagecolorallocate ( resource $image , int $red , int $green , int $blue )
Also z.B.:
$white = imageColorAllocate($image, 255, 255, 255)
Diese Farbe kann dann manchen Funktionen mitgegeben werden.
Außerdem legt der erste Aufruf der Funktion den Hintergrund der Grafik fest, außer sie wurde mittels imagecreatetruecolor() erstellt.
Ausschnitt eines Bildes in ein anderes kopieren
bool imagecopy ( resource $dst_im , resource $src_im , int $dst_x , int $dst_y , int $src_x , int $src_y , int $src_w , int $src_h )
Text einfügen
Zeichenketten kann man mithilfe der Funktion
bool imagestring ( resource $image , int $font , int $x , int $y , string $string , int $color )
ausgeben, wobei die Parameter für das Bild, die Schriftart (1-5), die Position, die auszugebende Zeichenkette und die Farbe stehen.
Für weitere Schriftarten siehe die Funktion
int imageloadfont ( string $file )
Zeichnen
Man kann in ein Bild verschiedene Formen zeichnen, dafür gibt’s unter anderem die folgenden Funktionen:
bool imagesetpixel ( resource $image , int $x , int $y , int $color ) bool imageline ( resource $image , int $x1 , int $y1 , int $x2 , int $y2 , int $color ) bool imagerectangle ( resource $image , int $x1 , int $y1 , int $x2 , int $y2 , int $color ) bool imagefilledrectangle ( resource $image , int $x1 , int $y1 , int $x2 , int $y2 ,int $color ) bool imageellipse ( resource $image , int $cx , int $cy , int $width , int $height , int $color ) bool imagefilledellipse ( resource $image , int $cx , int $cy , int $width , int $height, int $color ) bool imagearc ( resource $image , int $cx , int $cy , int $width , int $height , int $start , int $end , int $color ) bool imagepolygon ( resource $image , array $points , int $num_points , int $color )
Upload Problem mit dem Internet Explorer
Schön, wieder mal eine Stunde mit dem IE verschi$$en...
Ich hab eine kleine PHP-Anwendung mit Bilderupload geschrieben, die auch ganz gut funktioniert hat - in allen Browsern ausser dem Liebling aller Webdesigner, dem Internet Explorer.
Das Problem war, dass der Internet Explorer beim Hochladen die MIME-Typen der Dateien verändert, siehe hier. Und diese MIME Type Detection macht aus image/png ein image/x-png und aus image/jpg ein image/pjpg, was meiner Ansicht nach äusserst hirnrissig ist.
Also Lösung: Wenn im Uploadskript (z.B. wie bei mir PHP) auf MIME-Typen überprüft werden soll, dann auch die Typen image/x-png und image/pjpg beachten!
Interner Link im Joomla-Template
Manchmal kann es notwendig sein, einen internen Link direkt ins Joomla Template einzufügen. Die naheliegendste Möglichkeit ist wohl, auf die entsprechende Seite im Browser zu gehen, die URL zu kopieren und im Template einzufügen. Allerdings kann sich der Name der Seite (und damit unter Umständen der Alias) ändern und das ergibt dann ungültige Links auf der Seite. Diese gilt es zu vermeiden.
Nun, die Lösung ist eigentlich ganz einfach und heißt JHTML::link(). Das ist die Funktion, die aus einer URL mit GET-Parametern eine schöne Joomla-URL macht.
Also, zuerst müssen wir den Link zusammenbauen:
Einen Artikel verlinken
Im Joomla-Backend auf die Beitragsübersicht gehen um die ID des Artikels herauszufinden (in der ganz rechten Spalte):

Die ID wird dann an den folgenden Link statt xyz angehängt.
index.php?option=com_content&view=article&id=xyz
Oft will man aber nicht auf den Beitrag, sondern auf einen bestimmten
Menüpunkt verlinken
Dazu geht man im Backend auf ein Menü, und wählt den entsprechenden Menüeintrag aus. In der Detailansicht sieht man dann die ID des Menüeintrags (Menü-Item), die man benötigt:

Auch hier wird die ID (in diesem Fall 54) an den Link angehängt:
index.php?Itemid=xyz
Ins Template einfügen
Nun muß der Link nur noch an der gewünschten Stelle ins Template eingefügt werden:
<?
echo JHTML::link('index.php?Itemid=xyz','Linktitel', $attribute);
?>Mögliche HTML Attribute des Link-Tags können über das assoziative Array $attribute übergeben werden, z.B.:
<?
$attribute = array('class'=>'menulink', 'title'=>'Zum Menü');
echo JHTML::link('index.php?Itemid=xyz','Linktitel', $attribute);
?>
Die Programmierer-Hierarchie
Eine Hierarchie, wie sich Programmierer bestimmter Sprachen selbst sehen, also wer sich wem überlegen fühlt.
Bemerkenswert der Kommentar zu Ruby (frei übersetzt):
Ruby Programmierer fühlen sich jedem überlegen, aber sie wissen nicht dass es auch Nicht-Web-Programmiersprachen gibt, darum sind sie in der Grafik über Perl.
Hm, hat was wahres...

Via informatik-forum.at.
Gratis eBook "MySQL mit PHP"
O'Reilly hat in seiner Serie OpenBooks ein neues eBook zum Download bereitgestellt. Heißen tut das gute Stück "MySQL mit PHP".Es beschreibt die Installation des "magischen Trios" Apache, PHP und MySQL, Grundlagen und Methoden des Datenbankentwurfs (Datentypen, Schlüssel, Entity-Relationship Diagramme, Normalisierung,..), und wie man per SQL mit der Datenbank kommuniziert. Auch erweiterte Funktionen wie Transaktionen, Views, Prepared Statements, Stored Procedures und Trigger werden durchgenommen.
Dann gibts eine kleine Einführung in PHP und eine Beschreibung der MySQL-Funktionen von PHP, garniert mit einem kleinem, ein bisschen deplazierten Ajax-Tutorial und einer umfangreichen Beispielanwendung.
Zum Schluß gibts auch noch eine Einführung in die Administration von MySQL.
Ich denke alles in allem ist es ein recht brauchbares Buch für Einsteiger oder auch zum Nachschlagen für erfahrenere Nutzer.
PHP Feedreader-Libraries: Zend_Feed vs. SimplePie
Ich hab mich in letzter Zeit ein wenig mit dem Zend Framework beschäftigt und bin nach kurzer zeit auch auf die PHP-Feedreader Komponente Zend_Feed gestoßen.Nach ein bisschen herumprobieren mit diversen Feeds, bin ich aber anscheinend schon an ihre Grenzen gestoßen. Das parsen von Atom-Feeds ist mir nicht gelungen, besser gesagt das extrahieren des link-Tags.Wobei ich jetzt nicht unbedingt die Schuld auf Zend_Feed schieben möchte, ich hab einfach nicht lange herumprobiert und Google hat nichts brauchbares zu dem Thema ausgespuckt.
Langer Rede kurzer Sinn, ich hab mich nach Alternativen umgesehen und bin auch sogleich fündig geworden: Eine oft empfohlene Library ist SimplePie, ein Open Source PHP Feed Reader. Der Vorteil, der sofort ins Auge sticht, ist die Fehlertoleranz und die Unterstüzung für diverse Feed-Formate. Momentan sind das (laut Website):
- RSS 0.90
- RSS 0.91 (Netscape)
- RSS 0.91 (Userland)
- RSS 0.92
- RSS 1.0
- RSS 2.0
- Atom 0.3
- Atom 1.0
Ein weiteres praktisches Feature ist das automatische Auffinden von Feeds einer Website. Man muss also nicht die genaue URL des Feeds eingeben, es reicht auch beispielsweise die Adresse eines Blogs. Das kann Zend_Feed zwar auch, macht es aber nicht automatisch (Was vielleicht auch performancetechnisch intelligenter ist, whatever).
Auch Caching der geladenen Einträge wird unterstützt, was die etwas trägere Performance etwas besser macht.
Zu guter letzt möchte ich ein kleines Beispiel bringen, um zu zeigen, wie einfach SimplePie angewandt werden kann:
<?php
require_once('simplepie/simplepie.inc');
$feed = new SimplePie('webcode.lemme.at');
$feed->handle_content_type();
if($feed->error()){
//handle Error
}else{
echo'<ul>';
foreach ($feed->get_items() as $item) {
echo '<li><ul>';
echo '<li>'.$item->get_title().'</li>';
echo '<li>'.$item->get_permalink().'</li>';
echo '</ul></li>';
}
echo '</ul>';
}
?>
Ich denke mal der Code ist halbwegs selbsterklärend, sollte er das nicht sein, darf natürlich in den Kommentaren gefragt werden ![]()
Enterprise PHP Patterns
Stefan Priebsch hat auf der DLW-Europe einen Vortrag zum Thema: "Beyond MVC: Enterprise PHP Patterns" gehalten. Die sehr interessanten Slides stellt er auf seinem Blog mittels Slideshare zur Verfügung:
"Enterprise PHP Patterns" vollständig lesen
MVC mit PHP - Ein neues Tutorial
Weil mich die Arbeit mit dem MVC-Joomla-Framework so fasziniert hat, habe ich ein MVC mit PHP-Tutorial geschrieben, dass beschreibt, wie man das MVC-Pattern ganz einfach mit PHP implementieren kann.
Feedback, Fragen und Diskussion zu dem Tutorial ist hier in den Kommentaren möglich und erwünscht ![]()
Natürlich habe ich mich von anderen Inspirieren lassen, vor allem von folgenden Blogs und Artikeln:
- Understanding MVC in PHP, von Joe Stump
- MVC-Pattern in PHP von Brati
- Building a simple MVC system with PHP5 von Dennis Pallett auf PHPit.net
Joomla-Tipps
Sprache Herausfinden
Oft ist es interessant, die Sprache des aktuellen Users festzustellen, etwa wenn man aus mehreren Sprachen auswählen kann. Das geht ganz einfach mittels:
$language = JFactory::getLanguage();
Nun bekommt man ein JLanguage-Objekt, durch den Aufruf der Methode 'getTag()' erhält man das Sprachkürzel in der Form 'de-DE':
$tag = $language->getTag();
Layout im Menü-Maager verstecken / Layout (um)benennen
Bei der Entwicklung von Komponenten ist es oft wichtig, im Menümanager des Backends ein Layout einer View zu verstecken, oder dem Layout einen ansprechenderen Namen zu geben.
Dazu legt man in dem 'tmpl'-Ordner der View ein neues XML-File an. Den Namen der Datei bestimmt der Name des Layouts. Ist das Layout-File beispielsweise 'show_item.php', muss die XML-Datei 'show_item.xml' heißen. Um das Layout zu verstecken muss die XML-Datei nun folgenden Aufbau haben:
<?xml version="1.0" encoding="UTF-8"?>
<metadata>
<layout hidden="true" />
</metadata>
Um dem Layout einen ansprechenden Namen zu geben, muss sie folgenden Inhalt haben:
<?xml version="1.0" encoding="UTF-8"?>
<metadata>
<layout title="Alle Items anzeigen"></layout>
</metadata>
Komponenten für Joomla entwickeln
Es gibt eigentlich 2 gute Quellen bzw. Tutorials für die Entwicklung von Komponenten für Joomla 1.5: Das Online-Buch von Hagen Graf und die Joomla Tutorials (Englisch).
