我有一个我创建的电子邮件模板,我希望能够使用[NAME],[EMAIL],[PHONE]等内容,并在发送时替换它们。这些很容易,但我还有其他更复杂的标签,如[PRODUCT:12],[ARTICLE:23]等,我想在第一部分中读作'type',然后将第二部分作为值处理。
我猜我需要某种preg_match然后preg_replace,但我真的很茫然,因为我不擅长这些功能。
有什么建议吗?
--- --- EDIT
要清楚。问题在于两部分括号中的var:value对。我需要能够看到它是PRODUCT并查看id = 12返回产品,以便我可以建立链接。
---编辑2 ---
public function parse_message_tags($matches){
$keys = explode(':', $matches[1]);
$key = $keys[0];
$id = (array_key_exists(1, $keys)) ? $keys[1] : -1;
switch ($key) {
case 'NAME':
return $this->session->member->name();
break;
case 'FIRST_NAME':
return $this->session->member->first_name;
break;
case 'EMAIL':
return $this->session->member->email;
break;
case 'PRODUCT':
$product = Product::find_by_id($id);
return '<a href="/products/'.$product->seo_name.'">'.$product->name.'</a>';
break;
case 'ARTICLE':
$product = Article::find_by_id($id);
return '<a href="/blog/'.$article->seo_title.'">'.$article->name.'</a>';
break;
}
}
public function parse_message($input)
{
$pattern = '/\[(NAME|FIRST_NAME|EMAIL|PHONE|PRODUCT:\d+|ARTICLE:\d+)\]/';
return preg_replace_callback($pattern, create_function('$matches', 'return self::parse_message_tags;'), $input);
}
结果:
致命错误:当没有类范围处于活动状态时,无法访问self ::
OR
致命错误:不在对象上下文中时使用$ this
我猜测create_function()会将你带出范围/上下文。显然我在这里使用类方法,所以简单地调用'parse_message_tags'不是一种选择。
答案 0 :(得分:2)
在这种情况下使用的函数是preg_replace_callback(),因为您需要在第二种类型的标记的索引上执行数组查找。将代码包装在类中非常有用。我们需要以某种方式将数据传递给回调函数。使用全局变量会很难看。
class EmailTemplate {
protected $text;
protected $data;
public function __construct($text, $data) {
$this->text = $text;
$this->data = $data;
}
public function callback($matches) {
$replacement = null;
$name = $matches[1];
$index = (int) $matches[2];
if(isset($this->data[$name])) {
$value = $this->data[$name];
if(is_scalar($value)) {
$replacement = $value;
} else {
if(isset($value[$index])) {
$replacement = $value[$index];
}
}
}
return $replacement;
}
public function __toString() {
return preg_replace_callback('/\[(\w+)(?:\:(\d+))?\]/', array($this, 'callback'), $this->text);
}
}
$data = array(
"NAME" => "Bob",
"PRODUCT" => array("dog", "cat", "eggplant")
);
$email = new EmailTemplate("Dear [NAME], please buy my [PRODUCT:2]", $data);
echo $email;
这种模式可能看起来有点令人困惑。这里的关键是(?:)?部分。第二个问号表示其中的模式,\:(\ d +)是可选的。第一个问号以及冒号将字符分组而不捕获它们。它在这里使用,因为我们只需要数组查找的数字。
答案 1 :(得分:1)
此正则表达式与Id匹配,并应在文本中为您提供ID:\[([A-Z]*):(\d*)\]
新的正则表达式\[([A-Z]*:?\d*)\]
与两者匹配。
答案 2 :(得分:0)
你可以使用这样的函数:
function parse_tag($tag) {
preg_match('#\[(.*?)\]#', $tag, $matches);
return explode(':', $matches[1]);
}
要使用它,您可能会执行以下操作:
$tests = array(
'[NAME]' => array('NAME'),
'[PRODUCT:SHIRT]' => array('PRODUCT', 'SHIRT'),
'[ARTICLE:23]' => array('ARTICLE', 23)
);
foreach($tests as $search => $expected) {
$result = parse_tag($search);
assert($expected == $result);
}
查看工作示例here
答案 3 :(得分:0)
$pattern = '/\[(NAME|EMAIL|PHONE|PRODUCT:\d+|ARTICLE:\d+)\]/';
function callbackHandler($matches) {
$keys = explode(':', $matches[1]);
$key = $keys[0];
$id = (array_key_exists(1, $keys)) ? $keys[1] : -1;
switch ($key) {
case 'NAME':
return '##name##';
break;
case 'EMAIL':
return '##email##';
break;
case 'PHONE':
return '##phone##';
break;
case 'PRODUCT':
return '##product id:'.$id.'##';
break;
case 'ARTICLE':
return '##article id:'.$id.'##';
break;
}
}
$string = "Hello [NAME], have you checked out our latest product [PRODUCT:32]. We're sure you'll like it! If you don't we'll call you on [PHONE] and link you to [ARTICLE:12], which will put your email ([EMAIL]) on our hourly email newsletter";
echo preg_replace_callback($pattern, 'callbackHandler', $string);
返回:
您好## name ##,您查看了我们的最新产品##产品编号:32 ##。我们相信你会喜欢它!如果您不这样做,我们会在## phone ##上给您打电话,并将您链接到## article id:12 ##,这会将您的电子邮件(## email ##)发送到我们的每小时电子邮件简报