# smd_lib 0.23 # Shared function library used by smd_ plugins and others. # Stef Dawson # http://stefdawson.com/ # ..................................................................... # This is a plugin for Textpattern - http://textpattern.com/ # To install: textpattern > admin > plugins # Paste the following text into the 'Install plugin' box: # ..................................................................... a:9:{s:4:"name";s:7:"smd_lib";s:6:"author";s:11:"Stef Dawson";s:10:"author_uri";s:22:"http://stefdawson.com/";s:7:"version";s:4:"0.23";s:11:"description";s:56:"Shared function library used by smd_ plugins and others.";s:4:"help";s:6950:"	<h1>smd_ plugin library</h1>

	<p>Offers no public textpattern tags. It is simply a shared library of common functions used by smd_ plugins.</p>

<h1>smd_ plugin library</h1>

<p>Offers no public textpattern tags. It is simply a shared library of common functions used by smd_ plugins.</p>
<h2>Changelog</h2>

<ul>
<li>v0.1   07-02-25 Initial public release</li>
<li>v0.2   07-03-03 Added <span class="caps">MLP</span> (Multi-Lingual Pack) library support</li>
<li>v0.21  07-03-06 Added integer range functionality. getAtts now takes a regexp arg.</li>
<li>v0.21a 07-03-21 Fixed: Character ranges ignored (thanks mrdale)</li>
<li>v0.21b 07-04-02 Fixed: Sticky article support in getAtts (thanks pieman)</li>
<li>v0.21c 07-07-29 Fixed: Numeric ranges in categories (thanks wolle)</li>
<li>v0.21d 07-08-03 Fixed: Negation with multiple elements</li>
<li>v0.22  07-03-21 Added generic <span class="caps">MLP</span> class.</li>
<li>v0.23  07-03-25 Added the FuzzyFind class and getWord function. getAtts() now allows &#8216;?q&#8217;</li>
</ul>

	<p><h2>Function Reference</h2><br />
<b>smd_addQSVar</b><br />

<b>smd_removeQSVar</b><br />
<p>Add or remove a query string variable to the given <span class="caps">URL</span>, taking into account any existing variables that may be in the <span class="caps">URL</span> already. &#8216;Add&#8217; takes three arguments, &#8216;Remove&#8217; just takes the first two:</p><br />
<ol><br />
<li>The <span class="caps">URL</span> string to add to/remove from</li><br />
<li>The id of the querystring (the bit before the = sign)</li><br />
<li>The value of the new querystring (the bit after the = sign)</li><br />
</ol><br />
<p>e.g. <code>smd_addQSVar($thisarticle[&#39;url_title&#39;], &#39;tpg&#39;, 15);</code> would add <code>tpg=15</code> to the current article&#8217;s <span class="caps">URL</span>. If there are no other variables currently in the <span class="caps">URL</span>, it is added with a question-mark, otherwise it is appended with an ampersand.</p></p>

	<p><b>smd_getAtts</b><br />
<p>Searches the passed string for predetermined sequences of characters and, if that sequence is in the given $allowed array, replaces it as follows:</p><br />
<ul><br />
<li>?c = current global category (!c = not current category)</li><br />
<li>?s = current section (!s = not current section)</li><br />
<li>?t = current article title (!t = not current title)</li><br />
<li>?id = current article ID, prepended with $idprefix (!id = not current ID)</li><br />
<li>?q = current query term (!q = not current query term)</li><br />
<li>?field = contents of the current article&#8217;s field (could be a comma-separated list)</li><br />
<li>!field = not the contents of the current article&#8217;s field (could be a comma-separated list)</li><br />
</ul><br />
<p>Integer ranges (e.g. 1-5) will be expanded into their individual values; anything else is returned verbatim. It outputs two arrays; the 1st containing items for inclusion, the 2nd contains items for exclusion.</p></p>

	<p><b>smd_splitRange</b><br />
<p>Return an array of items from a string of (usually) comma-separated values. If any values contain ranges of numbers like 1-5 that need &#8216;expanding&#8217; first, they are dealt with. Takes one mandatory and one optional argument:</p><br />
<ol><br />
<li>The string to split up</li><br />
<li>The regular expression character classes to match. Each item in this string is treated as an entity and is fair game for splitting the string at. Defaults to &#8220;\s,&#8221; which is a &#8216;space&#8217; character or comma.</li><br />
</ol></p>

	<p><b>smd_MLP</b><br />
<p>Instantiate one of these to handle <span class="caps">MLP</span> in your plugin like this:<br />
<ol><br />
<li>Declare a unique global variable, e.g. global $myPlug</li><br />
<li>Define your default string replacement array (doesn&#8217;t need to be global), e.g:<br />

    $myStrings = array (&#8216;msg1&#8217; => &#8216;This is message 1&#8217;, &#8216;msg2&#8217; => &#8216;This is message 2&#8217;);</li><br />
<li>Create an <span class="caps">MLP</span> handler:</br />
    $myPlug = new smd_MLP(&#8216;plugin_name&#8217;, &#8216;plugin_prefix&#8217;, $myStrings);</li><br />
<li>That&#8217;s it! There are two optional args to smd_MLP:<br />

     a) the default (full) language to use, e.g &#8216;da-dk&#8217;. Defaults to &#8216;en-gb&#8217;.<br />

     b) the interface the strings are for. Choose from &#8216;public&#8217; (the default), &#8216;admin&#8217; or &#8216;common&#8217;</li><br />
<li>To use a replacement string in your code:<br />

     a) Make sure to import the unique global variable: e.g. global $myPlug;<br />

     b) Call $myPlug->gTxt(&#8216;messageID&#8217;); [ e.g. $myPlug->gTxt(&#8216;msg1&#8217;) ]<br />

     c) If you want to replace any args in your message string, pass an associative
         array as the 2nd arg to gTxt()</li><br />
</ol><br />
</p></p>

	<p><b>smd_FuzzyFind</b><br />
<p>A <span class="caps">PHP</span> class for approximate string searching of large text masses, adapted (*cough* borrowed) from http://elonen.iki.fi/code/misc-notes/appr-search-php/. Instantiate one of these and pass it the string pattern/word you are looking for and a number indiating how close that match has to be (the amount of error tolerable). 0=exact match; 10=pretty much every string in the world. Practical values are usually 1 or 2, sometimes 3.</p><br />
<p>Usage example:</p><br />
<pre>
  $finder = new smd_FuzzyFind($patt, $max_err);
  if ($finder->too_short_err)
    $error = &#8220;Unable to search &#8211; use longer pattern or reduce error tolerance.&#8221;;</p>

  while($text = get_next_page_of_text()) {
    $matches = $finder->search($text);
    while(list($idx,$rng) = each($matches))
      print &#8220;Match found ending at position $idx with a closeness of $val\n&#8221;;
  }
</pre>
<p>The code uses initial filtering to sort out possible match candidates and then applies a slower character-by-character search (search_short()) against them.</p>

	<p><b>smd_getWord</b><br />
<p>Useful with smd_FuzzyFind: it takes a string and an offset into that string and returns the nearest &#8220;word&#8221; before that offset position.</p> If the offset is not supplied it starts from the beginning of the string, thus returning the first word.</p></p>

<p>The optional third argument is a full regular expression pattern (including start and end slashes and switches) that determine what a &#8216;character&#8217; is. Anything that <strong>doesn&#8217;t</strong> match is treated as a word delimiter. Defaults to any (unicode) letter or number character. Note that although it uses a unicode pattern by default, it treats chars in the string as single-byte entities&#8230; this will be revisited when PHP6 becomes available on webhosts. So, maybe around 2042.</p>";s:4:"code";s:17994:"// smdMLP DEPRECATED: use smd_MLP class instead
global $smdMLP;
$smdMLP = array(
		'smd_slimbox' => 'smd_sbox',
		'smd_slimbox_inc' => 'smd_sbox',
		'smd_random_banner' => 'smd_rban',
);
class smd_FuzzyFind {
	// The last 3 parameters are for optimization only, to avoid the
	// surprisingly slow strlen() and substr() calls:
	//  - $start_index = from which character of $text to start the search
	//  - $max_len = maximum character to search (starting from $start_index)
	//  - $text_strlen =
	// The return value is an array of matches:
	//   Array( [<match-end-index>] => <error>, ... )
	// Note: <error> is generally NOT an exact edit distance but rather a
	// lower bound. This is unfortunate but the routine would be slower if
	// the exact error was calculate along with the matches.
	// The function is based on the non-deterministic automaton simulation
	// algorithm (without bit parallelism optimizations).
	function search_short($patt, $k, $text, $start_index=0, $max_len=-1, $text_strlen=-1) {
		if ( $text_strlen < 0 )
			$text_strlen = strlen( $text );

		if ( $max_len < 0 )
			$max_len = $text_strlen;

		$start_index = max( 0, $start_index );
		$n = min( $max_len, $text_strlen-$start_index );
		$m = strlen( $patt );
		$end_index = $start_index + $n;

		// If $text is shorter than $patt, use the built-in
		// levenshtein() instead:
		if ($n < $m)
		{
			$lev = levenshtein(substr($text, $start_index, $n), $patt);
			if ( $lev <= $k )
				return Array( $start_index+$n-1 => $lev );
			else
				return Array();
		}

		$s = Array();
		for ($i=0; $i<$m; $i++)
		{
			$c = $patt{$i};
			if ( isset($s[$c]))
				$s[$c] = min($i, $s[$c]);
			else
				$s[$c] = $i;
		}

		if ( $end_index < $start_index )
			return Array();

		$matches = Array();
		$da = $db = range(0, $m-$k+1);

		$mk = $m-$k;

		for ($t=$start_index; $t<$end_index; $t++)
		{
			$c = $text{$t};
			$in_patt = isset($s[$c]);

			if ($t&1) { $d=&$da; $e=&$db; }
			else { $d=&$db; $e=&$da; }

			for ($i=1; $i<=$mk; $i++)
			{
				$g = min( $k+1, $e[$i]+1, $e[$i+1]+1 );

				// TODO: optimize this with a look-up-table?
				if ( $in_patt )
					for ($j=$e[$i-1]; ($j<$g && $j<=$mk); $j++)
						if ( $patt{$i+$j-1} == $c )
							$g = $j;

				$d[$i] = $g;
			}

			if ( $d[$mk] <= $k )
			{
				$err = $d[$mk];
				$i = min( $t-$err+$k+1, $start_index+$n-1);
				if ( !isset($matches[$i]) || $err < $matches[$i])
					$matches[$i] = $err;
			}
		}

		unset( $da, $db );
		return $matches;
	}
	function test_short_search() {
		$test_text = "Olipa kerran jussi bj&xling ja kolme\n iloista ".
			"jussi bforling:ia mutta ei yhtaan jussi bjorling-nimista laulajaa.";
		$test_patt = "jussi bjorling";
		assert( $this->search_short($test_patt, 4, $test_text) == Array(27=>2, 60=>1, 94=>0));
		assert( $this->search_short($test_patt, 2, $test_text) == Array(27=>2, 60=>1, 94=>0));
		assert( $this->search_short($test_patt, 1, $test_text) == Array(60=>1, 94=>0));
		assert( $this->search_short($test_patt, 0, $test_text) == Array(94=>0));
		assert( $this->search_short("bjorling", 2, $test_text, 19, 7) == Array());
		assert( $this->search_short("bjorling", 2, $test_text, 19, 8) == Array(26=>2));
		assert( $this->search_short("bjorling", 2, $test_text, 20, 8) == Array());
	}

	var $patt, $patt_len, $max_err;
	var $parts, $n_parts, $unique_parts, $max_part_len;
	var $transf_patt;
	var $too_short_err;
	function smd_FuzzyFind( $pattern, $max_error ) {
		$this->patt = $pattern;
		$this->patt_len = strlen($this->patt);
		$this->max_err = $max_error;

		// Calculate pattern partition size
		$intpartlen = floor($this->patt_len/($this->max_err+2));
		if ($intpartlen < 1)
		{
			$this->too_short_err = True;
			return;
		}
		else $this->too_short_err = False;

		// Partition the pattern for pruning
		$this->parts = Array();
		for ($i=0; $i<$this->patt_len; $i+=$intpartlen)
		{
			if ( $i + $intpartlen*2 > $this->patt_len )
			{
				$this->parts[] = substr( $this->patt, $i );
				break;
			}
			else
				$this->parts[] = substr( $this->patt, $i, $intpartlen );
		}
		$this->n_parts = count($this->parts);

		// The intpartlen test above should have covered this:
		assert( $this->n_parts >= $this->max_err+1 );

		// Find maximum part length
		foreach( $this->parts as $p )
			$this->max_part_len = max( $this->max_part_len, strlen($p));

		// Make a new part array with duplicate strings removed
		$this->unique_parts = array_unique($this->parts);

		// Transform the pattern into a low resolution pruning string
		// by replacing parts with single characters
		$this->transf_patt = "";
		reset( $this->parts );
		while (list(,$p) = each($this->parts))
		   $this->transf_patt .= chr(array_search($p, $this->unique_parts)+ord("A"));

		// Self diagnostics
		$this->test_short_search();
	}
	function search( $text ) {
		// Find all occurences of unique parts in the
		// full text. The result is an array:
		//   Array( <index> => <part#>, .. )
		$part_map = Array();
		reset( $this->unique_parts );
		while (list($pi, $part_str) = each($this->unique_parts))
		{
			$pos = strpos($text, $part_str);
			while ( $pos !== False )
			{
				$part_map[$pos] = $pi;
				$pos = strpos($text, $part_str, $pos+1);
			}
		}
		ksort( $part_map ); // Sort by string index

		// The following code does several things simultaneously:
		//  1) Divide the indices into groups using gaps
		//	  larger than $this->max_err as boundaries.
		//  2) Translate the groups into strings so that
		//	  part# 0 = 'A', part# 1 = 'B' etc. to make
		//	  a low resolution approximate search possible later
		//  3) Save the string indices in the full string
		//	  that correspond to characters in the translated string.
		//  4) Discard groups (=part sequences) that are too
		//	  short to contain the approximate pattern.
		// The format of resulting array:
		//   Array(
		//	  Array( "<translate-string>",
		//			 Array( <translated-idx> => <full-index>, ... ) ),
		//	  ... )
		$transf = Array();
		$transf_text = "";
		$transf_pos = Array();
		$last_end = 0;
		$group_len = 0;
		reset( $part_map );
		while (list($i,$p) = each($part_map))
		{
			if ( $i-$last_end > $this->max_part_len+$this->max_err )
			{
				if ( $group_len >= ($this->n_parts-$this->max_err))
					$transf[] = Array( $transf_text, $transf_pos );

				$transf_text = "";
				$transf_pos = Array();
				$group_len = 0;
			}

			$transf_text .= chr($p + ord("A"));
			$transf_pos[] = $i;
			$group_len++;
			$last_end = $i + strlen($this->parts[$p]);
		}
		if ( strlen( $transf_text ) >= ($this->n_parts-$this->max_err))
			$transf[] = Array( $transf_text, $transf_pos );

		unset( $transf_text, $transf_pos );

		if ( current($transf) === False )
			return Array();

		// Filter the remaining groups ("approximate anagrams"
		// of the pattern) and leave only the ones that have enough
		// parts in correct order. You can think of this last step of the
		// algorithm as a *low resolution* approximate string search.
		// The result is an array of candidate text spans to be scanned:
		//   Array( Array(<full-start-idx>, <full-end-idx>), ... )
		$part_positions = Array();
		while (list(,list($str, $pos_map)) = each($transf))
		{
//			print "|$transf_patt| - |$str|\n";
			$lores_matches = $this->search_short( $this->transf_patt, $this->max_err, $str );
			while (list($tr_end, ) = each($lores_matches))
			{
				$tr_start = max(0, $tr_end - $this->n_parts);
				if ( $tr_end >= $tr_start )
				{
					$median_pos = $pos_map[ (int)(($tr_start+$tr_end)/2) ];
					$start = $median_pos - ($this->patt_len/2+1) - $this->max_err - $this->max_part_len;
					$end = $median_pos + ($this->patt_len/2+1) + $this->max_err + $this->max_part_len;

//					print "#" . strtr(substr( $text, $start, $end-$start ), "\n\r", "$$") . "#\n";
//					print_r( $this->search_short( &$this->patt, $this->max_err, &$text, $start, $end-$start ));

					$part_positions[] = Array($start, $end);
				}
			}
			unset( $lores_matches );
		}
		unset( $transf );

		if ( current($part_positions) === False )
			return Array();

		// Scan the final candidates and put the matches in a new array:
		$matches = Array();
		$text_len = strlen($text);
		while (list(, list($start, $end)) = each($part_positions))
		{
			$m = $this->search_short( $this->patt, $this->max_err, $text, $start, $end-$start, $text_len );
			while (list($i, $cost) = each($m))
				$matches[$i] = $cost;
		}
		unset($part_positions);

		return $matches;
	}
}

class smd_MLP {
	var $smd_strings;
	var $smd_owner;
	var $smd_prefix;
	var $smd_lang;
	var $smd_event;
	function smd_MLP($plug, $prefx, $strarray, $lng='en-gb', $ev='public') {
		$this->smd_owner = $plug;
		$this->smd_prefix = $prefx;
		$this->smd_strings = $strarray;
		$this->smd_lang = $lng;
		$this->smd_event = $ev;
		register_callback(array(&$this, 'smd_Callback'), 'l10n.enumerate_strings');
	}
	function smd_Callback($event='l10n.enumerate_strings', $step='', $pre=0) {
		$r = array(
			'owner' => $this->smd_owner,
			'prefix' => $this->smd_prefix,
			'lang' => $this->smd_lang,
			'event' => $this->smd_event,
			'strings' => $this->smd_strings,
		);
		return $r;
	}
	// Generic lookup
	//  $what = key to look up
	//  $args = any arguments the key is expecting for replacement
	function gTxt($what, $args = array()) {
		global $textarray;

		// Prepare the prefixed key for use
		$key = $this->smd_prefix . '-' . $what;
		$key = strtolower($key);

		// Grab from the global textarray (possibly edited by MLP) if we can
		if(isset($textarray[$key])) {
			$str = $textarray[$key];
		} else {
			// The string isn't in the localised textarray so fallback to using
			// the (non prefixed) string array in the plugin
			$key = strtolower($what);
			$str = (isset($this->smd_strings[$key])) ? $this->smd_strings[$key] : $what;
		}
		// Perform substitutions
		if(!empty($args)) {
			$str = strtr($str, $args);
		}

		return $str;
	}
}
function smd_addQSVar($url, $key, $value) {
	$url = smd_removeQSVar($url, $key);
	if (strpos($url, '?') === false) {
		return ($url . '?' . $key . '=' . $value);
	} else {
		return ($url . '&' . $key . '=' . $value);
	}
}
function smd_removeQSVar($url, $key) {
	$url = preg_replace('/(.*)(\?|&)' . $key . '=[^&]+?(&)(.*)/i', '$1$2$4', $url . '&');
	$url = substr($url, 0, -1);
	return ($url);
}
// return a list of categories under (and including) the given parent category
function smd_getSubCats($parent,$cattype) {
	$parent = doSlash($parent);
	$lextrem = 1; $rextrem = 1;
	extract(safe_row("lft as lextrem, rgt as rextrem", "txp_category", "name='$parent' and type = '$cattype'"));
	$rs = safe_rows_start("id,name,title","txp_category", "lft between $lextrem and $rextrem and type = '$cattype' order by lft asc");
	$cats = array();

	while ($row = nextRow($rs)) {
		$cats[] = array(
			'id' => $row['id'],
			'name' => $row['name'],
			'title' => $row['title']
		);
	}
	return $cats;
}
function smd_getAtts($str, $allowed, $idprefix="", $splitat=",\s") {
	global $pretext, $thisarticle;

	$out = array();
	$subout = array();
	$notout = array();
	$matches = smd_splitRange($str, $splitat);

	for ($idx = 0; $idx < count($matches); $idx++) {
		$thismatch = $matches[$idx];
		if (($thismatch === "?c") && in_array("?c",$allowed)) {
			// Use global article category, if it exists
			if ($pretext['c'] != "") {
				if (!in_array($pretext['c'], $out)) {
					$out[] = $pretext['c'];
				}
			}
		} else if (($thismatch === "!c") && in_array("!c",$allowed)) {
			// Don't use global article category
			if ($pretext['c'] != "") {
				if (!in_array($pretext['c'], $notout)) {
					$notout[] = $pretext['c'];
				}
			}
		} else if (($thismatch === "?s") && in_array("?s",$allowed)) {
			// Use article section
			if ($pretext['s'] != "") {
				if (!in_array($pretext['s'], $out)) {
					$out[] = $pretext['s'];
				}
			}
		} else if (($thismatch === "!s") && in_array("!s",$allowed)) {
			// Don't use article section
			if ($pretext['s'] != "") {
				if (!in_array($pretext['s'], $notout)) {
					$notout[] = $pretext['s'];
				}
			}
		} else if (($thismatch === "?q") && in_array("?q",$allowed)) {
			// Use query string
			if ($pretext['q'] != "") {
				if (!in_array($pretext['q'], $out)) {
					$out[] = $pretext['q'];
				}
			}
		} else if (($thismatch === "!q") && in_array("!q",$allowed)) {
			// Don't use query string
			if ($pretext['q'] != "") {
				if (!in_array($pretext['q'], $notout)) {
					$notout[] = $pretext['q'];
				}
			}
		} else if (($thismatch === "?t") && in_array("?t",$allowed)) {
			// Use article URL title if this is an article.
			if ($thisarticle != NULL && $thisarticle['url_title'] != "") {
				if (!in_array($thisarticle['url_title'], $out)) {
					$out[] = $thisarticle['url_title'];
				}
			}
		} else if (($thismatch === "!t") && in_array("!t",$allowed)) {
			// Don't use article URL title
			if ($thisarticle != NULL && $thisarticle['url_title'] != "") {
				if (!in_array($thisarticle['url_title'], $notout)) {
					$notout[] = $thisarticle['url_title'];
				}
			}
		} else if (($thismatch === "?id") && in_array("?id",$allowed)) {
			// Use article ID, prepended with $idprefix
			if ($thisarticle != NULL) {
				if (!in_array($idprefix . $pretext['id'], $out)) {
					$out[] = $idprefix . $pretext['id'];
				}
			}
		} else if (($thismatch === "!id") && in_array("!id",$allowed)) {
			// Don't use article ID
			if ($thisarticle != NULL) {
				if (!in_array($idprefix . $pretext['id'], $notout)) {
					$notout[] = $idprefix . $pretext['id'];
				}
			}
		} else if (($thismatch[0] === "?") && in_array("?field",$allowed)) {
			// Use the given field name; which may be a comma-separated sublist.
			// Split off the field name from the question mark
			$fieldname = substr($thismatch,1);
			if (($thisarticle != NULL) && (isset($thisarticle[$fieldname]))) {
				$fieldContents = $thisarticle[$fieldname];
			} else {
				$fieldContents = $fieldname;
			}
			if (!empty($fieldContents)) {
				$subout = smd_splitRange(strip_tags($fieldContents), $splitat);
				foreach ($subout as $subname) {
					if (!in_array($subname, $out)) {
						$out[] = $subname;
					}
				}
			}
		} else if (($thismatch[0] === "!") && in_array("!field",$allowed)) {
			// Negation. May either be a field name (and maybe another CSL) or a fixed term.
			// Split off the name from the exclamation mark
			$fieldname = substr($thismatch,1);
			if (($thisarticle != NULL) && (isset($thisarticle[$fieldname]))) {
				$fieldContents = $thisarticle[$fieldname];
			} else {
				$fieldContents = $fieldname;
			}
			if (!empty($fieldContents)) {
				$subout = smd_splitRange(strip_tags($fieldContents), $splitat);
				foreach ($subout as $subname) {
					if (!in_array($subname, $notout)) {
						$notout[] = $subname;
					}
				}
			}
		} else {
			if (!in_array($thismatch, $out)) {
				$out[] = $thismatch;
			}
		}
	}
	return array($out,$notout);
}
function smd_splitRange($str, $splitat=",\s") {
	$retarr = array();
	$elems = preg_split('/['.$splitat.']+/', $str, -1, PREG_SPLIT_NO_EMPTY);
	foreach ($elems as $item) {
		$negate = false;
		// Does the item start with a negation character
		if ($item[0] === "!") {
			$negate = true;
			$item = substr($item,1);
		}
		// Is the item an integer list range
		if (preg_match('/^(\d+)\-(\d+)$/', $item)) {
			list($lo, $hi) = explode("-", $item, 2);
			$rng = range($lo, $hi);
			// Reapply the negation if necessary
			for($idx = 0; $idx < count($rng); $idx++) {
				$rng[$idx] = (($negate) ? "!" : "") . $rng[$idx];
			}
			$retarr = array_merge($retarr, $rng);
		} else {
			$retarr[] = (($negate) ? "!" : "") . $item;
		}
	}
	return $retarr;
}
function smd_getWord($haystack,$searchterm,$offset=0) {
	$numwords = str_word_count($searchterm);
	$haystack = strrev(substr($haystack,0,$offset+1));
	$spacePos = 1;
	// If the algorithm has found the start of the next word instead of the end of the last word, skip it
	if (ctype_space($haystack[1]) && ctype_alnum($haystack[0])) {
		$startpos = 2;
		$spacePos = 2;
	}
	for ($idx = 0; $idx < $numwords; $idx++) {
		$spacePos = (strpos($haystack, " ", $spacePos))+1;
	}
	return trim(strrev(substr($haystack, $startpos, $spacePos-$startpos)));
}
//DEPRECATED: use smd_MLP->gTxt() instead
// Generic lookup function for interfacing with MLP to localise strings.
//  $stray = array of default strings
//  $what = key to look up
//  $args = any arguments the key is expecting for replacement
function smd_gTxt($stray, $what, $args = array()) {
	global $textarray, $smdMLP;
	$fn = smd_getCaller();

	// Drop out if the id isn't defined.
	// Would be nice to use reflection to find the name of the function and pass that,
	// but that's PHP5 territory only
	if (empty($fn)) {
		$str = $what;
	} else {
		// Prepare the prefixed key for use
		$key = $smdMLP[$fn] . '-' . $what;
		$key = strtolower($key);

		// Grab from the global textarray (possibly edited by MLP) if we can
		if(isset($textarray[$key])) {
			$str = $textarray[$key];
		} else {
			// The string isn't in the localised $textarray so fallback to using
			// the (non prefixed) string array in the plugin
			$key = strtolower($what);
			$str = (isset($stray[$key])) ? $stray[$key] : $what;
		}
	}
	// Perform substitutions
	if(!empty($args)) {
		$str = strtr($str, $args);
	}
	return $str;
}
//DEPRECATED: Will be removed in the next release
// Cheating way of finding a function's caller
function smd_getCaller() {
	$backtrace = debug_backtrace();
	return $backtrace[2]['function'];
}
";s:4:"type";s:1:"0";s:3:"md5";s:32:"7d16a4f3bc5456298b250e50cda85c0b";}