c91dba9fd36188fabf2a95aa9db2f6f139885deb
2 /********************************************************************************
3 * include/misc.inc.php : Misc functions
4 * -------------------------
6 * This file is part of the banana distribution
7 * Copyright: See COPYING files that comes with this distribution
8 ********************************************************************************/
10 /********************************************************************************
14 function _b_($str) { return utf8_decode(dgettext('banana', utf8_encode($str))); }
16 function to_entities($str) {
17 require_once dirname(__FILE__
).'/utf8.php';
18 return utf8entities(htmlentities($str, ENT_NOQUOTES
, 'UTF-8'));
21 function is_utf8($s) { return iconv('utf-8', 'utf-8', $s) == $s; }
23 function textFormat_translate($format)
25 switch (strtolower($format)) {
26 case 'plain': return _b_('Texte brut');
27 case 'richtext': return _b_('Texte enrichi');
28 case 'html': return _b_('HTML');
29 default: return $format;
33 /** Redirect to the page with the given parameter
36 function redirectInBanana($params)
38 header('Location: ' . makeLink($params));
41 /** Make a link using the given parameters
42 * @param ARRAY params, the parameters with
45 * - group = group name
46 * - artid/first = article id the the group
47 * - subscribe = to show the subscription page
48 * - action = action to do (new, cancel, view)
49 * - part = to show the given MIME part of the article
50 * - pj = to get the given attachment
51 * - xface = to make a link to an xface
53 * Can be overloaded by defining a hook_makeLink function
55 function makeLink($params)
57 if (function_exists('hook_makeLink')
58 && $res = hook_makeLink($params)) {
61 $proto = empty($_SERVER['HTTPS']) ?
'http://' : 'https://';
62 $host = $_SERVER['HTTP_HOST'];
63 $file = $_SERVER['PHP_SELF'];
65 if (isset($params['xface'])) {
66 $file = dirname($file) . '/xface.php';
67 $get = 'face=' . urlencode(base64_encode($params['xface']));
68 } else if (count($params) != 0) {
70 foreach ($params as $key=>$value) {
71 if (strlen($get) != 1) {
74 $get .= $key . '=' . $value;
80 return $proto . $host . $file . $get;
83 /** Format a link to be use in a link
86 function makeHREF($params, $text = null
, $popup = null
, $class = null
)
88 $link = makeLink($params);
92 if (!is_null($popup)) {
93 $popup = ' title="' . $popup . '"';
95 if (!is_null($class)) {
96 $class = ' class="' . $class . '"';
99 if (isset($params['action']) && $params['action'] == 'view') {
100 $target = ' target="_blank"';
102 return '<a href="' . htmlentities($link) .'"' . $target . $popup . $class . '>' . $text . '</a>';
105 /** Format tree images links
106 * @param img STRING Image name (without extension)
107 * @param alt STRING alternative string
108 * @param width INT to force width of the image (null if not defined)
110 * This function can be overloaded by defining hook_makeImg()
112 function makeImg($img, $alt, $height = null
, $width = null
)
114 if (function_exists('hook_makeImg')
115 && $res = hook_makeImg($img, $alt, $height, $width)) {
119 if (!is_null($width)) {
120 $width = ' width="' . $width . '"';
122 if (!is_null($height)) {
123 $height = ' height="' . $height . '"';
126 $proto = empty($_SERVER['HTTPS']) ?
'http://' : 'https://';
127 $host = $_SERVER['HTTP_HOST'];
128 $file = dirname($_SERVER['PHP_SELF']) . '/img/' . $img;
129 $url = $proto . $host . $file;
131 return '<img src="' . $url . '"' . $height . $width . ' alt="' . $alt . '" />';
134 /** Make a link using an image
136 function makeImgLink($params, $img, $alt, $height = null
, $width = null
, $class = null
)
138 return makeHREF($params,
139 makeImg($img, ' [' . $alt . ']', $height, $width),
144 /********************************************************************************
152 * @desc Strip forbidden tags and delegate tag-source check to removeEvilAttributes()
154 function removeEvilTags($source)
156 $allowedTags = '<h1><b><i><a><ul><li><pre><hr><blockquote><img><br><font><p><small><big><sup><sub><code><em>';
157 $source = preg_replace('|</div>|i', '<br />', $source);
158 $source = strip_tags($source, $allowedTags);
159 return preg_replace('/<(.*?)>/ie', "'<'.removeEvilAttributes('\\1').'>'", $source);
165 * @desc Strip forbidden attributes from a tag
167 function removeEvilAttributes($tagSource)
169 $stripAttrib = 'javascript:|onclick|ondblclick|onmousedown|onmouseup|onmouseover|'.
170 'onmousemove|onmouseout|onkeypress|onkeydown|onkeyup';
171 return stripslashes(preg_replace("/$stripAttrib/i", '', $tagSource));
174 /** Convert html to plain text
176 function htmlToPlainText($res)
178 $res = trim(html_entity_decode(strip_tags($res, '<div><br><p>')));
179 $res = preg_replace("@</?(br|p|div)[^>]*>@i", "\n", $res);
180 if (!is_utf8($res)) {
181 $res = utf8_encode($res);
186 /** Match **, // and __ to format plain text
188 function formatPlainText($text)
190 $formatting = Array('\*' => 'strong',
193 foreach ($formatting as $limit=>$mark) {
194 $text = preg_replace('@(^|\W)' . $limit . '(\w+)' . $limit . '(\W|$)@'
195 ,'\1<' . $mark . '>\2</' . $mark . '>\3'
201 /********************************************************************************
205 /** Convert richtext to html
207 function richtextToHtml($source)
209 $tags = Array('bold' => 'b',
211 'smaller' => 'small',
214 'subscript' => 'sub',
215 'superscript' => 'sup',
216 'excerpt' => 'blockquote',
221 // clean unsupported tags
222 $protectedTags = '<signature><lt><comment><'.join('><', array_keys($tags)).'>';
223 $source = strip_tags($source, $protectedTags);
225 // convert richtext tags to html
226 foreach (array_keys($tags) as $tag) {
227 $source = preg_replace('@(</?)'.$tag.'([^>]*>)@i', '\1'.$tags[$tag].'\2', $source);
230 // some special cases
231 $source = preg_replace('@<signature>@i', '<br>-- <br>', $source);
232 $source = preg_replace('@</signature>@i', '', $source);
233 $source = preg_replace('@<lt>@i', '<', $source);
234 $source = preg_replace('@<comment[^>]*>((?:[^<]|<(?!/comment>))*)</comment>@i', '<!-- \1 -->', $source);
235 return removeEvilAttributes($source);
238 /********************************************************************************
242 function _headerdecode($charset, $c, $str) {
243 $s = ($c == 'Q' ||
$c == 'q') ?
quoted_printable_decode($str) : base64_decode($str);
244 $s = iconv($charset, 'iso-8859-15', $s);
245 return str_replace('_', ' ', $s);
248 function headerDecode($value) {
249 $val = preg_replace('/(=\?[^?]*\?[BQbq]\?[^?]*\?=) (=\?[^?]*\?[BQbq]\?[^?]*\?=)/', '\1\2', $value);
250 return preg_replace('/=\?([^?]*)\?([BQbq])\?([^?]*)\?=/e', '_headerdecode("\1", "\2", "\3")', $val);
253 function headerEncode($value, $trim = 0) {
255 if (strlen($value) > $trim) {
256 $value = substr($value, 0, $trim) . "[...]";
259 return "=?UTF-8?B?".base64_encode($value)."?=";
262 function header_translate($hdr) {
264 case 'from': return _b_('De');
265 case 'subject': return _b_('Sujet');
266 case 'newsgroups': return _b_('Forums');
267 case 'followup-to': return _b_('Suivi-Ã ');
268 case 'date': return _b_('Date');
269 case 'organization': return _b_('Organisation');
270 case 'references': return _b_('Références');
271 case 'x-face': return _b_('Image');
273 if (function_exists('hook_headerTranslate')
274 && $res = hook_headerTranslate($hdr)) {
281 function formatDisplayHeader($_header,$_text) {
283 if (function_exists('hook_formatDisplayHeader')
284 && $res = hook_formatDisplayHeader($_header, $_text)) {
290 return formatDate($_text);
295 $groups = preg_split("/[\t ]*,[\t ]*/",$_text);
296 foreach ($groups as $g) {
297 $res .= makeHREF(Array('group' => $g), $g) . ', ';
299 return substr($res,0, -2);
302 return formatFrom($_text);
307 $text = str_replace("><","> <",$_text);
308 $text = preg_split("/[ \t]/",strtr($text,$banana->spool
->ids
));
309 $parents = preg_grep("/^\d+$/",$text);
310 $p = array_pop($parents);
315 $p = $banana->spool
->overview
[$p]->parent
;
317 foreach (array_reverse($par_ok) as $p) {
318 $rsl .= makeHREF(Array('group' => $banana->spool
->group
,
326 return '<img src="' . makeLink(Array('xface' => headerDecode($_text))) .'" alt="x-face" />';
330 if (function_exists('hook_getSubject')) {
331 $link = hook_getSubject($_text);
333 return formatPlainText($_text) . $link;
336 return htmlentities($_text);
340 /********************************************************************************
344 function formatDate($_text) {
345 return strftime("%A %d %B %Y, %H:%M (fuseau serveur)", strtotime($_text));
348 function fancyDate($stamp) {
349 $today = intval(time() / (24*3600));
350 $dday = intval($stamp / (24*3600));
352 if ($today == $dday) {
354 } elseif ($today == 1 +
$dday) {
355 $format = _b_('hier')." %H:%M";
356 } elseif ($today < 7 +
$dday) {
357 $format = '%a %H:%M';
359 $format = '%a %e %b';
361 return strftime($format, $stamp);
364 function formatFrom($text) {
365 # From: mark@cbosgd.ATT.COM
366 # From: mark@cbosgd.ATT.COM (Mark Horton)
367 # From: Mark Horton <mark@cbosgd.ATT.COM>
368 $mailto = '<a href="mailto:';
370 $result = htmlentities($text);
371 if (preg_match("/^([^ ]+)@([^ ]+)$/",$text,$regs)) {
372 $result="$mailto{$regs[1]}@{$regs[2]}\">".htmlentities($regs[1]."@".$regs[2])."</a>";
374 if (preg_match("/^([^ ]+)@([^ ]+) \((.*)\)$/",$text,$regs)) {
375 $result="$mailto{$regs[1]}@{$regs[2]}\">".htmlentities($regs[3])."</a>";
377 if (preg_match("/^\"?([^<>\"]+)\"? +<(.+)@(.+)>$/",$text,$regs)) {
378 $result="$mailto{$regs[2]}@{$regs[3]}\">".htmlentities($regs[1])."</a>";
380 return preg_replace("/\\\(\(|\))/","\\1",$result);
383 function makeTab($link, $text)
385 return Array(makeHREF($link, $text),
389 function displayTabs()
392 extract($banana->state
);
393 if (function_exists('hook_shortcuts') && $cstm = hook_shortcuts()) {
397 $res['subscribe'] = makeTab(Array('subscribe' => 1), _b_('Abonnements'));
398 $res['forums'] = makeTab(Array(), _b_('Les forums'));
400 if (!is_null($group)) {
401 $res['group'] = makeTab(Array('group' => $group), $group);
402 if (is_null($artid)) {
403 if (@$action == 'new') {
404 $res['action'] = makeTab(Array('group' => $group,
406 _b_('Nouveau Message'));
409 $res['message'] = makeTab(Array('group' => $group,
412 if (!is_null($action)) {
413 if ($action == 'new') {
414 $res['action'] = makeTab(Array('group' => $group,
418 } elseif ($action == 'cancel') {
419 $res['action'] = makeTab(Array('group' => $group,
421 'action' => 'cancel'),
427 $ret = '<ul id="onglet">';
428 foreach ($res as $name=>$onglet) {
429 if ($name != $page) {
430 $ret .= '<li>' . $onglet[0] . '</li>';
432 $ret .= '<li class="actif">' . $onglet[1] . '</li>';
439 function displayPages($first = -1)
442 extract($banana->state
);
444 if (!is_null($group) && is_null($artid)
445 && sizeof($banana->spool
->overview
)>$banana->tmax
) {
446 $res .= '<div class="pages">';
447 $n = intval(log(count($banana->spool
->overview
), 10))+
1;
449 for ($ndx = 1 ; $ndx <= sizeof($banana->spool
->overview
) ; $ndx +
= $banana->tmax
) {
451 $res .= '<strong>' . $i . '</strong> ';
453 $res .= makeHREF(Array('group' => $group,
456 $ndx . '-' . min($ndx+
$banana->tmax
-1,sizeof($banana->spool
->overview
)))
466 function makeTable($text)
468 return '<table class="cadre_a_onglet" cellpadding="0" cellspacing="0" width="100%">'
472 . '<tr><td class="conteneur_tab">'
478 /********************************************************************************
479 * FORMATTING STUFF : BODY
482 function autoformat($text, $part = false
)
485 $length = $banana->wrap
;
491 $cmd = "echo ".escapeshellarg($text)." | perl -MText::Autoformat -e 'autoformat {left=>1, right=>$length$all };'";
492 exec($cmd, $result, $ret);
494 $result = split("\n", $text);
499 function wrap($text, $_prefix="", $_force=false
)
501 $parts = preg_split("/\n-- ?\n/", $text);
502 if (count($parts) >1) {
503 $sign = "\n-- \n" . array_pop($parts);
504 $text = join("\n-- \n", $parts);
510 $url = $banana->url_regexp
;
511 $length = $banana->wrap
;
512 $max = $length +
($length/10);
513 $splits = split("\n", $text);
517 foreach ($splits as $line) {
518 if ($_force ||
strlen($line) > $max) {
519 if (preg_match("!^(.*)($url)(.*)!i", $line, $matches)
520 && strlen($matches[2]) > $length && strlen($matches) < 900) {
521 if (strlen($matches[1]) != 0) {
522 array_push($next, rtrim($matches[1]));
523 if (strlen($matches[1]) > $max) {
529 $result = array_merge($result, autoformat(join("\n", $next)));
531 $result = array_merge($result, $next);
535 array_push($result, $matches[2]);
537 if (strlen($matches[6]) != 0) {
538 array_push($next, ltrim($matches[6]));
539 if (strlen($matches[6]) > $max) {
544 if (strlen($line) > 2 * $max) {
545 $next = array_merge($next, autoformat($line, true
));
548 array_push($next, $line);
552 array_push($next, $line);
556 $result = array_merge($result, autoformat(join("\n", $next)));
558 $result = array_merge($result, $next);
561 return $_prefix.join("\n$_prefix", $result).($_prefix ?
'' : $sign);
564 function cutlink($link)
568 if (strlen($link) > $banana->wrap
) {
569 $link = substr($link, 0, $banana->wrap
- 3)."...";
574 function cleanurl($url)
576 $url = str_replace('@', '%40', $url);
577 return '<a href="'.$url.'" title="'.$url.'">'.cutlink($url).'</a>';
580 function formatbody($_text, $format='plain', $flowed=false
)
582 if ($format == 'html') {
583 $res = html_entity_decode(to_entities(removeEvilTags($_text)));
584 } else if ($format == 'richtext') {
585 $res = html_entity_decode(to_entities(richtextToHtml($_text)));
587 $res = to_entities(wrap($_text, "", $flowed));
588 $res = formatPlainText($res);
591 if ($format != 'html') {
593 $url = $banana->url_regexp
;
594 $res = preg_replace("/(<|>|")/", " \\1 ", $res);
595 $res = preg_replace("!$url!ie", "'\\1'.cleanurl('\\2').'\\3'", $res);
596 $res = preg_replace('/(["\[])?(?:mailto:)?([a-z0-9.\-+_]+@[a-z0-9.\-+_]+)(["\]])?/i', '\1<a href="mailto:\2">\2</a>\3', $res);
597 $res = preg_replace("/ (<|>|") /", "\\1", $res);
599 if ($format == 'richtext') {
604 if ($format == 'html') {
605 $res = preg_replace("@(</p>)\n?-- ?\n?(<p[^>]*>|<br[^>]*>)@", "\\1<br/>-- \\2", $res);
606 $res = preg_replace("@<br[^>]*>\n?-- ?\n?(<p[^>]*>)@", "<br/>-- <br/>\\2", $res);
607 $res = preg_replace("@(<pre[^>]*>)\n?-- ?\n@", "<br/>-- <br/>\\1", $res);
608 $parts = preg_split("@(:?<p[^>]*>\n?-- ?\n?</p>|<br[^>]*>\n?-- ?\n?<br[^>]*>)@", $res);
609 $sign = '<hr style="width: 100%; margin: 1em 0em; " />';
611 while (preg_match("@(^|<pre>|\n)>@i", $res)) {
612 $res = preg_replace("@(^|<pre>|\n)((>[^\n]*\n)+)@ie",
613 "'\\1</pre><blockquote><pre>'"
614 .".stripslashes(preg_replace('@(^|<pre>|\n)>[ \\t\\r]*@i', '\\1', '\\2'))"
615 .".'</pre></blockquote><pre>'",
618 $res = preg_replace("@<pre>-- ?\n@", "<pre>\n-- \n", $res);
619 $parts = preg_split("/\n-- ?\n/", $res);
620 $sign = '</pre><hr style="width: 100%; margin: 1em 0em; " /><pre>';
623 return join($sign, $parts);
626 // vim:set et sw=4 sts=4 ts=4