Domain eines WordPress-Blogs ändern

caticonslite_bm_alt

Möchte man die Domain eines WordPress-Blogs ändern, so sind hierfür Änderungen in der Datenbank erforderlich. Zum einen müssen in der Tabelle „wp_options“ die Einträge „home“ und „siteurl“ angepasst werden, zum anderen in „wp_posts“ alle „guid“-Einträge. Andernfalls ist ein „wp-admin“-Login nicht möglich bzw. wird WordPress z.B. Bilder aus der Mediathek weiterhin von der alten URL/Domain ausliefern. Die hierfür erforderlichen SQL-Anweisungen kann man z.B. mit phpMyAdmin oder ähnlichen Tools ausführen:

UPDATE `wp_options` SET `option_value` = REPLACE(`option_value`, 'dev.webentwickler.at', 'webentwickler.at');
UPDATE `wp_posts` SET `guid` = REPLACE(`guid`, 'dev.webentwickler.at', 'webentwickler.at');

„dev.webentwickler.at“ stelle in diesem Fall die alte Domain dar, während es sich bei „webentwickler.at“ um die neue Domain handelt. War der WordPress-Blog früher in einem Unterverzeichnis installiert, so ist die zu ersetzenden Zeichenkette um dieses zu erweitern – z.B. „dev.webentwickler.at/pfad/zum/unterverzeichnis“.

Individuelle Marker in Direct Mail

caticonslite_bm_alt

In einem Newsletter werden häufig auch benutzerspezifische Parameter abseits der standardmäßig vorhandenen benötigt, z.B. um die Anrede zu personalisieren (Sehr geehrte Frau bzw. Sehr geehrter Herr,…) oder um Felder aus marketingtechnischen Gründen (z.B. individueller Gewinncode) hinzuzufügen. Dies ist in der Direct Mail-Extension (Extensionkey: direct_mail), dank der Verwendung von Hooks, relativ einfach zu bewerkstelligen.

Zuerst muss man hierfür eine neue Extension erstellen, ich verwende hier den Extensionkey „direct_mail_markers“, um die benötigten Dateien irgendwo ablegen zu können. Als nächstes wird eine Klasse (samt entsprechender Methode), welche benutzerdefinierte Marker generiert, erstellt – im Ordner „Classes/Hooks/“ der Extension legt man eine Datei namens „DirectMailMarkers.php“ mit folgendem Inhalt an:

<?php
namespace WebentwicklerAt\DirectMailMarkers\Hooks;
 
class DirectMailMarkers
{
    /**
     * @param array $params
     * @return array
     */
    function mailMarkersHook(array $params)
    {
        // ###USER_salutation###
        $salutation = '';
        if (in_array($params['row']['gender'], array(0, 'm', 1, 'f')) && !empty($params['row']['last_name'])) {
            // gender
            if ($params['row']['gender'] === 0 || $params['row']['gender'] === 'm') {
                $salutation = 'Sehr geehrter Herr ';
            } else if ($params['row']['gender'] === 1 || $params['row']['gender'] === 'f') {
                $salutation = 'Sehr geehrte Frau ';
            } else {
                $salutation = 'Sehr geehrte(r) Herr/Frau ';
            }
 
            // title
            if (!empty($params['row']['title'])) {
                $salutation .= $params['row']['title'] . ' ';
            }
 
            // last_name
            $salutation .= $params['row']['last_name'];
        } else {
            $salutation = 'Sehr geehrte Damen und Herren';
        }
 
        $params['markers']['###USER_salutation###'] = $salutation;
 
        return $params;
    }
}

Diese Methode erzeugt abhängig vom Feld „gender“ die persönliche Anrede „Sehr geehrter Herr Nachname“„Sehr geehrte Frau Nachname“ oder wenn kein Nachname oder Geschlecht angegeben worden ist „Sehr geehrte Damen und Herren“. Zusätzlich wird der Titel – sofern angegeben – eingefügt.

Damit die soeben erstellte Hook-Methode verwendet wird und der generierte Marker ###USER_salutation### im Newsletter verwendet werden kann, muss diese in der„ext_localconf.php“ registriert werden:

<?php
 
if (!defined ('TYPO3_MODE')) die ('Access denied.');
 
// register directmail markers hook
$GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/direct_mail']['res/scripts/class.dmailer.php']['mailMarkersHook'][] = 'EXT:' . $_EXTKEY . '/Classes/Hooks/DirectMailMarkers.php:' . \WebentwicklerAt\DirectMailMarkers\Hooks\DirectMailMarkers::class . '->mailMarkersHook';

Nachdem die soeben erstellte Erweiterung über den Extensionmanager installiert worden ist, steht der generierte Marker im Newsletter zur Verfügung.

In der Datei composer.json der Installation ist folgender Eintrag für das Autoloading unter „autoload/psr-4“ zu ergänzen:

"autoload": {
  "psr-4": {
    "WebentwicklerAt\\DirectMailMarkers\\": "typo3conf/ext/direct_mail_markers/Classes/"
  }
}

Abschließend müssen die Autoloading-Informationen noch mit composer dump-autoload aktualisiert werden.

RSS-Feed generieren

caticonslite_bm_alt

Viele TYPO3-Erweiterungen zur Generierung von RSS-Feeds können lediglich Seiten und tt_news-Artikel verwenden – das ist häufig nicht ausreichend. Eine Ausnahme bildet die Extension „in2rss“, welche sich recht flexibel mittels TypoScript konfigurieren lässt und Datensätze aus jeder beliebigen Tabelle darstellen kann.

Nach der üblichen Vorgehensweise zur Installation des Plugins, müssen noch die zu verwendendenTabellen konfiguriert werden. Die Grundkonfiguration samt einem vorkonfigurierten Beispiel lässt sich mittels Static Template einfügen – dieses erscheint mir allerdings als unpraktikabel, daher lege ich alle notwendigen Parameter selbst im Setup fest:

includeLibs.in2rss_main = EXT:in2rss/lib/class.tx_in2rss_main.php
 
rss = PAGE
rss {
	typeNum = 100
 
	config {
		disableAllHeaderCode = 1
		disablePrefixComment = 1
		xhtml_cleaning = 0
		admPanel = 0
		no_cache = 1
	}
 
	10 = USER
	10 {
		userFunc = user_in2rss_main->showRSS
		userFunc.config {
			template = fileadmin/templates/webentwickler/in2rss/template.xml
			sorting = DESC
			limit = 20
			uft8_coding =
 
			10 {
				settings {
					table = pages
					fieldTitle = title
					fieldDescription = description
					fieldBody = description
					fieldImage = media
					fieldSorting = crdate
					fieldCrdate = crdate
					pid =
					pid_recursive = 0
					additionalWhereClause = AND doktype IN (1) AND nav_hide = 0
					cropDescription = 250
					limit = 100
				}
 
				fields {
					rssTitle = TEXT
					rssTitle {
						field = rssTitle
					}
 
					rssDescription = TEXT
					rssDescription {
						field = rssDescription
					}
 
					rssBody = TEXT
					rssBody {
						field = rssBody
					}
 
					rssSorting = TEXT
					rssSorting {
						field = rssSorting
					}
 
					rssCrdate = TEXT
					rssCrdate {
						field = rssCrdate
					}
 
					rssLink = TEXT
					rssLink {
						typolink {
							parameter.field = uid
							returnLast = url
						}
					}
				}
			}
		}
	}
}
 
[globalVar = GP:type = 100]
	config.absRefPrefix = {$baseUrl}
[end]

Die Konfiguration der Tabelle erfolgt in den Zeilen 24-74, in diesem Fall werden alle nicht im Menü versteckten (nav_hide = 0) Seiten (table = pages) vom Typ Standard (doktype IN (1)) verwendet. Die restlichen Zeilen dieses Bereichs enthalten Parameterzuweisungen, damit diese im in Zeile 19 angegebenen Template verwendet werden können:

<?xml version="1.0" encoding="utf-8"?>
<rss version="2.0">
	<channel>
		<title>webentwickler.at</title>
		<link>http://www.webentwickler.at/</link>
		<description>Blog für Webentwickler.</description>
		<language>de</language>
 
		<f:for each="{rssFeed}" as="rss"><f:if condition="{rss.rssTitle}">
			<item>
				<title>{rss.rssTitle}</title>
				<description>{rss.rssDescription}</description>
				<link>{rss.rssLink}</link>
				<pubDate>{rss.rssCrdate}</pubDate>
			</item>
		</f:if></f:for>
	</channel>
</rss>

Statische Dateien von cookie-free Domain bzw. externem Server oder CDN laden

caticonslite_bm_alt

TYPO3 setzt standardmäßig das Cookie „fe_typo_user“ für Frontend-Benutzer, selbst wenn es sich nur um einen Gast handelt und somit kein Login erfolgt ist. Das hat zur Folge, dass für jede statische Datei (z.B. CSS, Bilder und JavaScript) die Cookie-Informationen über den langsamen Upload an den Server übertragen werden müssen. Indem man diese Dateien von einer cookie-free Domain holt, lässt sich etwas Ladezeit einsparen. Außerdem kann man statische Dateien von einemleichtgewichtigeren Webserver (z.B. nginx, Squid, lighttpd) oder CDN (Content Distribution Network) ausliefern lassen und somit den Hauptwebserver von TYPO3 (z.B. Apache, nginx) entlasten.

Für die Umsetzung dieses Vorhabens findet man im TER einige Erweiterungen, welche alle nach dem selben Prinzip funktionieren: Ersetzen der lokalen/relativen Pfade von fileadmin, typo3temp und uploads mittels regulärem Ausdruck. Ich setze hier bewusst nicht auf ein ausschließlich für diesen Zweck zugeschnittenes Plugin, sondern auf die „regex“-Extension, mit der ich bei Bedarf auch noch andere Ersetzungen vornehmen kann.

Nach der üblichen Vorgehensweise zur Installation des Plugins, muss im TypoScript-Setup noch folgende Konfiguration vorgenommen werden:

config.regex {
	pattern1 = /"fileadmin\/([^"]+)"/
	pattern1.replacement = "http://static.webentwickler.at/fileadmin/$1"
 
	pattern2 = /"typo3temp\/([^"]+)"/
	pattern2.replacement = "http://static.webentwickler.at/typo3temp/$1"
 
	pattern3 = /"uploads\/([^"]+)"/
	pattern3.replacement = "http://static.webentwickler.at/uploads/$1"
 
	pattern4 = /"typo3conf\/([^"]+)"/
	pattern4.replacement = "http://static.webentwickler.at/typo3conf/$1"
}

Die Zusatzdomain „static.webentwickler.at“ muss natürlich noch eingerichtet werden und auf das selbe Verzeichnis wie die TYPO3-Hauptdomain „www.webentwickler.at“ verweisen. Setzt man einen leichtgewichtigen Webserver bzw. ein CDN für die Zusatzdomain ein, so muss auf diesem natürlich dasTYPO3-Verzeichnis eingebunden werden (z.B. mittels NFS-Share oder DRBD). In diesem Fall sollte darauf geachtet werden, dass keine PHP-Dateien Plaintext ausgeliefert werden.

„csc-default“-Wrap entfernen

caticonslite_bm_alt

TYPO3 wrapt standardmäßig alle Inhaltselemente mit einem div, die Ausgabe im Frontend sieht in etwa wie folgt aus:

&lt;div id="c1" class="csc-default"&gt;
	&lt;p class="bodytext"&gt;
		Lorem ipsum dolor sit amet.
	&lt;/p&gt;
&lt;/div&gt;

Interessant ist die eindeutige ID (aufgebaut aus der Datenbank-UID mit einem vorangestellten „c“) ausschließlich im Zusammenhang mit der Verwendung von Sprungmarken bzw. wenn man Rahmen (Frames) verwendet bzw. benutzerdefinierte Rahmen definiert. Entfernen lässt sich der div-Wrapper mit folgendem TypoScript-Setup:

tt_content.stdWrap.innerWrap &gt;

Als Ergebnis erhält man:

&lt;p class="bodytext"&gt;
	Lorem ipsum dolor sit amet.
&lt;/p&gt;
1 5 6 7 8  nach oben