<?php

$plugin='a:8:{s:7:"version";s:3:"0.6";s:6:"author";s:15:"Threshold State";s:10:"author_uri";s:26:"http://thresholdstate.com/";s:11:"description";s:36:"Flexible email contact/feedback form";s:4:"name";s:11:"zem_contact";s:4:"help";s:8692:"
<code>&lt;txp:zem_contact&gt;</code> produces a flexible, customizable email contact form.  It&#8217;s intended for use as an enquiry form for commercial sites, and includes several features to help reduce common problems with such forms (invalid email addresses, missing information).

	<h3>Features:</h3>

	<ul>
	<li>Arbitrary text fields can be specified, with min/max/required settings for validation</li>
		<li>Email address validation, including a check for a valid MX record (Unix only)</li>
		<li>Safe escaping of input data</li>
		<li>Accessible form layout, including <code>&lt;label&gt;</code> tags</li>
		<li>Spam prevention</li>
	</ul>

	<h3>Examples:</h3>

	<p>The simplest form is this:</p>

	<blockquote>
		<p><code>&lt;txp:zem_contact mailto="dest@example.com" /&gt;</code></p>
	</blockquote>

	<p>..which produces a default form with Name, Email and Message fields.  Email will be delivered to dest@example.com, with the user&#8217;s supplied email as the <code>From:</code> address.</p>

	<p>To specify fields explicitly, use something like this:</p>

	<blockquote>
		<p><code>&lt;txp:zem_contact mailto="dest@example.com"&gt;<br />
&lt;txp:zem_contact_email /&gt;&lt;br /&gt;<br />
&lt;txp:zem_contact_text label="Phone" min=7 /&gt;&lt;br /&gt;<br />
&lt;txp:zem_contact_textarea label="Enquiry" /&gt;&lt;br /&gt;<br />
&lt;txp:zem_contact_submit label="Enquire" /&gt;<br />
&lt;/txp:zem_contact&gt;<br />
</code></p>
	</blockquote>

	<p>Alternatively, place the field specifications in a Textpattern form, and call it like this:</p>

<pre><code>
&lt;txp:zem_contact mailto="dest@example.com" form="mycontactform" /&gt;
</code></pre>

	<h3>Tags</h3>

	<blockquote>
		<p><code>&lt;txp:zem_contact /&gt;</code></p>
	</blockquote>

	<p>May be used as a self-closing or container tag.  Place this where you want the input form to go.  Status and error messages, if any, will be displayed before the form.  Attributes:</p>

	<ul>
	<li><strong>mailto</strong> &#8211; destination address.  Required.</li>
		<li><strong>form</strong> &#8211; form containing the field layout.  Optional.</li>
		<li><strong>mailfrom</strong> &#8211; use the specified email address in the &#8220;From:&#8221; field when sending email.  If no <code>mailfrom</code> is specified, the sender&#8217;s email address will be used.  When <code>mailfrom</code> is specified, the sender&#8217;s email address will be placed in the &#8220;Reply-To:&#8221; field instead.</li>
		<li><strong>thanksform</strong> &#8211; The name of a form containing a thankyou message.  Optional.</li>
		<li><strong>showinput</strong> &#8211; Use <code>showinput=0</code> to disable display of the input form.  Optional.</li>
		<li><strong>showerror</strong> &#8211; Use <code>showerror=0</code> to disable display of error and status messages.  Optional.</li>
	</ul>

<code>showinput</code> and <code>showerror</code> can be used to display the form and error/thankyou messages on different parts of a page.  For example:

	<blockquote>
		<p><code>&lt;div id="error"&gt;&lt;txp:zem_contact form="contact_form" showinput=0 /&gt;&lt;/div&gt;</code><br />
<code>&lt;div id="inputform"&gt;&lt;txp:zem_contact form="contact_form" showerror=0 /&gt;&lt;/div&gt;</code></p>
	</blockquote>

	<p>The remaining tags may only be used within a <code>&lt;txp:zem_contact&gt;</code> form or container:</p>

	<blockquote>
		<p><code>&lt;txp:zem_contact_text /&gt;</code></p>
	</blockquote>

	<p>Creates a text input field and corresponding <code>&lt;label&gt;</code> tag.  The input value will be included in the email, preceeded by the label.  Attributes:</p>

	<ul>
	<li><strong>label</strong> &#8211; Text label displayed to the user.  Default is &#8216;Text&#8217;.</li>
		<li><strong>name</strong> &#8211; Field name, as used in the HTML input tag.  Optional, defaults to the <code>label</code>.</li>
		<li><strong>break</strong> &#8211; Break tag between the label and field.  Default is <code>&lt;br /&gt;</code>.  Use <code>break=''</code> to put the label and input field on the same line.</li>
		<li><strong>size</strong> &#8211; Size of the input field as displayed to the user.  Leave empty for the browser default.</li>
		<li><strong>required</strong> &#8211; Use <code>required=1</code> to make the field mandatory.  The text label will be displayed in bold, and the form will display an error message if no input is provided.  Optional.</li>
		<li><strong>min</strong> &#8211; Minimum input length in characters.  An error message will be displayed if the input is less than <code>min</code>.  Optional, default is 0.</li>
		<li><strong>max</strong> &#8211; Maximum input length in characters.  Used for the <code>maxlength</code> parameter of the input field.  No error will be displayed if the length is exceeded, but the value will be truncated for the email.  Optional, default is 100.</li>
		<li><strong>default</strong> &#8211; Default value when no input is provided.  Optional.</li>
	</ul>

	<blockquote>
		<p><code>&lt;txp:zem_contact_email /&gt;</code></p>
	</blockquote>

	<p>Input field for user&#8217;s email address.  The input value will automatically be validated to make sure it is of the form &#8216;abc@xxx.yyy[.zzz]&#8217;.  If the Unix-only <code>getmxrr()</code> function is available, the form will verify that an A or MX record exists for the domain.  Attributes are the same as for <code>&lt;txp:zem_contact_text /&gt;</code>, except that the default label is &#8216;Email&#8217;, and <code>required=1</code>.</p>

	<blockquote>
		<p><code>&lt;txp:zem_contact_select /&gt;</code></p>
	</blockquote>

	<p>Creates a drop-down selection list.  Attributes are similar to <code>&lt;txp:zem_contact_text /&gt;</code>, with the following additions:</p>

	<ul>
	<li><strong>list</strong> &#8211; A comma separated list of items to show in the drop-down box; e.g. &#8220;Sales,Support,Accounts&#8221;.</li>
		<li><strong>delim</strong> &#8211; The separator used in parsing the <strong>list</strong> attribute.  Default is &#8221;,&#8221;</li>
		<li><strong>selected</strong> &#8211; The list item that is selected by default; e.g. &#8220;Support&#8221;.  Default is &#8220;&#8221;.</li>
	</ul>

	<p><strong>min</strong> and <strong>max</strong> have no effect.  If <strong>required</strong> is used, an empty item will not be accepted.  Use <code>&lt;txp:zem_contact_select list=",Sales,Support" required=1 /&gt;</code> to make the first item blank, and require that the user select either Sales or Support.</p>

	<blockquote>
		<p><code>&lt;txp:zem_contact_checkbox /&gt;</code></p>
	</blockquote>

	<p>Creates a checkbox.  Attributes are similar to <code>&lt;txp:zem_contact_text /&gt;</code>, except that <strong>min</strong> and <strong>max</strong> have no effect.</p>

	<ul>
	<li><strong>checked</strong> &#8211; Use <code>checked=1</code> to make the box checked when the form is first displayed.  Default is 0.</li>
	</ul>

	<p>If <strong>required</strong> is used, the form won&#8217;t be accepted unless the box is checked.  Use <code>&lt;txp:zem_contact_checkbox label="I accept the terms and conditions" required=1 /&gt;</code> to make a crude shrinkwrap agreement.</p>

	<blockquote>
		<p><code>&lt;txp:zem_contact_textarea /&gt;</code></p>
	</blockquote>

	<p>Textarea input field.  Supports the same attributes as <code>&lt;txp:zem_content_text /&gt;</code>, with the exception of <code>size</code>.  The default label is &#8216;Message&#8217;, and the default <code>max</code> is 10000.  Additional attributes:</p>

	<ul>
	<li><strong>cols</strong> &#8211; column width.  Optional, default is 50.</li>
		<li><strong>rows</strong> &#8211; row height.  Optional, default is 6.</li>
	</ul>

	<blockquote>
		<p><code>&lt;txp:zem_contact_submit /&gt;</code></p>
	</blockquote>

	<p>Submit button.  Attributes:</p>

	<ul>
	<li><strong>label</strong> &#8211; Button text.  Default is &#8216;Send&#8217;.</li>
	</ul>

	<blockquote>
		<p><code>&lt;txp:zem_contact_serverinfo /&gt;</code></p>
	</blockquote>

	<p>This tag has no effect on the form or HTML output, but will include additional information in the email based on the PHP <code>$_SERVER</code> variable.  For example, <code>&lt;txp:zem_contact_serverinfo name="REMOTE_ADDR" label="IP" /&gt;</code> will report the user&#8217;s IP address.  Attributes:</p>

	<ul>
	<li><strong>name</strong> &#8211; name of the server variable, e.g. REQUEST_URI, HTTP_USER_AGENT.  See the <a href="http://www.php.net/manual/en/reserved.variables.php#reserved.variables.server">PHP manual</a> for a full list.  Required.</li>
		<li><strong>label</strong> &#8211; if supplied, this will be used to identify the field in the email.  Optional.</li>
	</ul>




 ";s:4:"code";s:8965:"
function zem_contact($atts, $thing='') {
	global $zem_contact_error, $sitename, $zem_contact_from, $zem_contact_nonce, $zem_contact_form;

	$form = (empty($atts['form']) ? '' : $atts['form']);
	$thanksform = (empty($atts['thanksform']) ? '' : $atts['thanksform']);
	$thanks = (empty($atts['thanks']) ? '<p>Email enquiry sent</p>' : $atts['thanks']);
	$mailto = (empty($atts['mailto']) ? '' : $atts['mailto']);
	$mailfrom = (empty($atts['mailfrom']) ? '' : $atts['mailfrom']);
	$showinput = isset($atts['showinput']) ? $atts['showinput'] : 1;
	$showerror = isset($atts['showerror']) ? $atts['showerror'] : 1;

	if (!$mailto)
		return 'No mailto address specified';

	if ($form)
		$Form = fetch("Form","txp_form","name",$form);
	else
		$form = $thing;

	if (empty($form)) {
		$form =<<<EOF
<txp:zem_contact_text label="Name" /><br />
<txp:zem_contact_email /><br />
<txp:zem_contact_textarea /><br />
<txp:zem_contact_submit />
EOF;
	}

	if ($thanksform)
		$thanks = fetch("Form","txp_form","name",$thanksform);

	$zem_contact_nonce = md5(uniqid(rand(), true));
	safe_insert("txp_discuss_nonce", "issue_time=now(), nonce='$zem_contact_nonce'");

	$f = parse($form);

	$out = '';
	if (is_array($zem_contact_error) and count($zem_contact_error)) {
		if ($showerror) {
			$out .= "<ul>\n";
			foreach($zem_contact_error as $err) {
				$out .= "<li>$err</li>\n";
			}
			$out .= "</ul>\n";
		}
	}
	elseif ($showinput and is_array($zem_contact_form) and ps('zem_contact_submit')) {

		$from = ($mailfrom ? $mailfrom : $zem_contact_from);
		$headers = (empty($from) ? '' : "From: $from\r\n");
		if ($mailfrom)
			$headers .= "Reply-To: $zem_contact_from\r\n";
 		$headers .= "Content-Transfer-Encoding: 8bit\r\n";
 		$headers .= "Content-Type: text/plain; charset=\"UTF-8\"\r\n";

		$msg = '';
		foreach ($zem_contact_form as $k=>$v)
			$msg .= "$k: ".htmlspecialchars($v)."\n";
		$r = mail($mailto, "Email enquiry for $sitename", $msg, $headers);

		if ($r) {
			$_POST = array();
			$out .= $thanks;
			$f = parse($form);
		}
		else {
			$out .= "<p>Unable to send email</p>";
		}
	}

	if ($showinput)
		$out .= '<form action="'.$_SERVER['REQUEST_URI'].'" method="post">'
			. $f
			. '</form>';

	return $out;
}

function zem_contact_text($atts) {
	global $zem_contact_error, $zem_contact_form;
	$label = (empty($atts['label']) ? 'Text' : $atts['label']);
	$name = (empty($atts['name']) ? preg_replace('/\W/', '', $label) : $atts['name']);
	$break = (!isset($atts['break']) ? '<br />' : $atts['break']);
	$size = (empty($atts['size']) ? '' : $atts['size']);
	$min = (empty($atts['min']) ? 0 : $atts['min']);
	$max = (!isset($atts['max']) ? 100 : $atts['max']);
	$default = (empty($atts['default']) ? '' : $atts['default']);
	$required = (empty($atts['required']) ? false : true);

	$size = ($size ? 'size="'.(int)$size.'"' : '');
	$maxlength = ($max ? 'maxlength="'.(int)$max.'"' : '');

	$default = ($default ? 'value="'.$default.'"' : '');
	$v = (ps($name) ? ps($name) : $default);

	if ($v and $max)
		$v = substr($v, 0, $max);

	if (ps('zem_contact_submit') and $required and empty($v))
		$zem_contact_error[] = "Required field $label is missing";
	elseif (ps($name) and $min and strlen($v) < $min)
		$zem_contact_error[] = "$label must be at least $min characters";
	elseif (ps($name))
		$zem_contact_form[$label] = $v;

	$l = ($required ? "<b>$label</b>" : $label);

	return '<label for="'.$name.'">'.$l.':</label>'.
		$break.
		'<input type="text" name="'.$name.'" id="'.$name.'" value="'.htmlentities($v).'" '.$size.' '.$maxlength.' />';
}

function zem_contact_textarea($atts) {
	global $zem_contact_error, $zem_contact_form;
	$label = (empty($atts['label']) ? 'Message' : $atts['label']);
	$name = (empty($atts['name']) ? preg_replace('/\W/', '', $label) : $atts['name']);
	$break = (!isset($atts['break']) ? '<br />' : $atts['break']);
	$cols = (empty($atts['cols']) ? '50' : $atts['cols']);
	$rows = (empty($atts['rows']) ? '6' : $atts['rows']);
	$min = (empty($atts['min']) ? 0 : $atts['min']);
	$max = (!isset($atts['max']) ? 10000 : $atts['max']);
	$default = (empty($atts['default']) ? '' : $atts['default']);
	$required = (empty($atts['required']) ? false : true);

	$v = (ps($name) ? ps($name) : $default);
	if (ps($name) and $max)
		$v = substr($v, 0, $max);

	if (ps('zem_contact_submit') and $required and empty($v))
		$zem_contact_error[] = "Required field $label is missing";
	elseif (ps($name) and $min and strlen($v) < $min)
		$zem_contact_error[] = "$label must be at least $min characters";
	elseif (ps($name))
		$zem_contact_form[$label] = $v;

	$l = ($required ? "<b>$label</b>" : $label);
	return '<label for="'.$name.'">'.$l.':</label>'.
		$break.
		'<'.'textarea name="'.$name.'" id="'.$name.'" rows="'.$rows.'" cols="'.$cols.'">'.htmlentities($v).'<'.'/textarea>';
}

function zem_contact_email($atts) {
	global $zem_contact_error, $zem_contact_form, $zem_contact_from;
	$atts = array_merge(array('label'=>'Email', 'required'=>'1'), $atts);
	$label = (empty($atts['label']) ? 'Email' : $atts['label']);
	$name = (empty($atts['name']) ? preg_replace('/\W/', '', $label) : $atts['name']);

	if (ps('zem_contact_submit')) {
		$email = ps($name);
		if (!preg_match('/^[\w._-]+@([\w-]+\.)+[\w-]+$/', $email))
			$zem_contact_error[] = "'$email' is not a valid email address";
		elseif (preg_match("/@(.+)$/", $email, $match)) {
			$domain = $match[1];
			$mx = 0;
			if (is_callable('getmxrr')) {
				$dummy = array();
				$mx = @getmxrr($domain, $mx);
			}

			if (!$mx and (@gethostbyname($domain) == $domain))
				$zem_contact_error[] = "Domain '$domain' is not a valid email host";
			else
				$zem_contact_from = $email;
		}
		else
			$zem_contact_from = $email;
	}

	return zem_contact_text($atts);
}

function zem_contact_select($atts) {
	global $zem_contact_error, $zem_contact_form;
	$delim = empty($atts['delim']) ? ',' : $atts['delim'];
	$list = empty($atts['list']) ? 'General enquiry' : $atts['list'];
	$selected = empty($atts['selected']) ? '' : $atts['selected'];
	$label = (empty($atts['label']) ? 'Option' : $atts['label']);
	$name = (empty($atts['name']) ? preg_replace('/\W/', '', $label) : $atts['name']);
	$break = (!isset($atts['break']) ? '<br />' : $atts['break']);
	$required = (empty($atts['required']) ? false : true);

	$list = split($delim, $list);
	$list = array_map('trim', $list);

	$v = (ps($name) ? ps(name) : $selected);
	if (ps('zem_contact_submit') and $required and empty($v))
		$zem_contact_error[] = "Required field $label is missing";
	elseif (ps($name) and !in_array($v, $list))
		$zem_contact_error[] = "Unknown $label $v";
	elseif (ps($name))
		$zem_contact_form[$label] = $v;

	$out = '';
	for ($i=0; $i < count($list); $i++) {
		if ($list[$i] == $selected)
			$out .= '<option selected=1>'.$list[$i]."</option>\n";
		else
			$out .= '<option>'.$list[$i]."</option>\n";
	}

	$l = ($required ? "<b>$label</b>" : $label);
	return '<label for="'.$name.'">'.$l.':</label>'.
		$break.
		'<select name="'.$name.'">'.$out.'</select>';
}

function zem_contact_checkbox($atts) {
	global $zem_contact_error, $zem_contact_form;

	$label = (empty($atts['label']) ? 'Checkbox' : $atts['label']);
	$name = (empty($atts['name']) ? preg_replace('/\W/', '', $label) : $atts['name']);
	$break = (!isset($atts['break']) ? '<br />' : $atts['break']);
	$checked = empty($atts['checked']) ? 0 : 1;
	$required = (empty($atts['required']) ? false : true);

	$v = (ps('zem_contact_submit') ? (bool)ps($name) : '');

	if (ps('zem_contact_submit') and $required and empty($v))
		$zem_contact_error[] = "Required field $label is missing";
	elseif (ps('zem_contact_submit'))
		$zem_contact_form[$label] = ($v ? 'Yes' : 'No');

	$l = ($required ? "<b>$label</b>" : $label);
	return '<input type="checkbox" name="'.$name.'"'.
		($checked ? ' checked="true"' : '').
		'>'.
		'<label for="'.$name.'">'.$l.'</label>'.
		$break;
}

function zem_contact_serverinfo($atts) {
	global $zem_contact_error, $zem_contact_form;

	$name = (empty($atts['name']) ? '' : $atts['name']);
	$label = (empty($atts['label']) ? $name : $atts['label']);

	if ($name and ps('zem_contact_submit'))
		$zem_contact_form[$label] = serverSet($name);
}

function zem_contact_submit($atts) {
	global $zem_contact_nonce, $zem_contact_error;

	$label = (empty($atts['label']) ? 'Send' : $atts['label']);

	if (ps('zem_contact_submit')) {
		$nonce = ps('zem_contact_nonce');
		if (empty($nonce)) {
			$zem_contact_error[] = "Required field nonce is missing";
		}
		else {
			safe_delete("txp_discuss_nonce", "issue_time < date_sub(now(),interval 10 minute)");
			$rs = safe_row("*", "txp_discuss_nonce", "nonce='$nonce' and used='0'");
			if ($rs)
				safe_update("txp_discuss_nonce", "used='1'", "nonce='$nonce'");
			else
				$zem_contact_error[] = "Form expired, please try again.";
		}
	}

	return '<input type="hidden" name="zem_contact_nonce" value="'.$zem_contact_nonce.'" />'.
		'<input type="submit" name="zem_contact_submit" value="'.$label.'" />';
}
";s:3:"md5";s:32:"226af80cc79f99bfe801a4ede90d936f";}'
?>