add _b_
[banana.git] / banana / misc.inc.php
CommitLineData
78cd27b3 1<?php
2/********************************************************************************
3 * include/misc.inc.php : Misc functions
4 * -------------------------
5 *
6 * This file is part of the banana distribution
7 * Copyright: See COPYING files that comes with this distribution
8 ********************************************************************************/
9
10/********************************************************************************
11 * MISC
12 */
13
14function _b_($str) { return utf8_decode(dgettext('banana', utf8_encode($str))); }
15
16function to_entities($str) {
39816f8b 17 require_once dirname(__FILE__).'/utf8.php';
78cd27b3 18 return utf8entities(htmlentities($str, ENT_NOQUOTES, 'UTF-8'));
19}
20
21function is_utf8($s) { return iconv('utf-8', 'utf-8', $s) == $s; }
22
1f75b135 23function textFormat_translate($format)
24{
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;
30 }
31}
32
8e519bd8 33/** Redirect to the page with the given parameter
34 * @ref makeLink
35 */
41b90d00 36function redirectInBanana($params)
fe8cbf1d 37{
f6df9eb2 38 header('Location: ' . makeLink($params));
fe8cbf1d 39}
40
8e519bd8 41/** Make a link using the given parameters
42 * @param ARRAY params, the parameters with
43 * key => value
44 * Known key are:
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
78f61fa3 51 * - xface = to make a link to an xface
8e519bd8 52 *
53 * Can be overloaded by defining a hook_makeLink function
54 */
fe8cbf1d 55function makeLink($params)
56{
f6df9eb2 57 if (function_exists('hook_makeLink')
58 && $res = hook_makeLink($params)) {
59 return $res;
60 }
8e519bd8 61 $proto = empty($_SERVER['HTTPS']) ? 'http://' : 'https://';
fe8cbf1d 62 $host = $_SERVER['HTTP_HOST'];
63 $file = $_SERVER['PHP_SELF'];
64
78f61fa3 65 if (isset($params['xface'])) {
66 $file = dirname($file) . '/xface.php';
67 $get = 'face=' . $params['xface'];
68 } else if (count($params) != 0) {
f6df9eb2 69 $get = '?';
70 foreach ($params as $key=>$value) {
71 if (strlen($get) != 1) {
72 $get .= '&';
73 }
74 $get .= $key . '=' . $value;
75 }
76 } else {
77 $get = '';
78 }
79
80 return $proto . $host . $file . $get;
8e519bd8 81}
82
83/** Format a link to be use in a link
84 * @ref makeLink
85 */
86function makeHREF($params, $text = null)
87{
f6df9eb2 88 $link = makeLink($params);
89 if (is_null($text)) {
90 $text = $link;
91 }
92 $target = null;
93 if (isset($params['action']) && $params['action'] == 'view') {
94 $target = ' target="_blank"';
95 }
96 return '<a href="' . htmlentities($link) . $target . '">' . $text . '</a>';
fe8cbf1d 97}
98
c14c86f4 99/** Format tree images links
100 * @param img STRING Image name (without extension)
101 * @param alt STRING alternative string
102 * @param width INT to force width of the image (null if not defined)
103 *
104 * This function can be overloaded by defining hook_makeImg()
105 */
5e200c3e 106function makeImg($img, $alt, $height = null, $width = null)
c14c86f4 107{
108 if (function_exists('hook_makeImg')
f944d187 109 && $res = hook_makeImg($img, $alt, $height, $width)) {
c14c86f4 110 return $res;
111 }
112
113 if (!is_null($width)) {
114 $width = ' width="' . $width . '"';
115 }
5e200c3e 116 if (!is_null($height)) {
117 $height = ' height="' . $height . '"';
118 }
119
c14c86f4 120 $proto = empty($_SERVER['HTTPS']) ? 'http://' : 'https://';
121 $host = $_SERVER['HTTP_HOST'];
122 $file = dirname($_SERVER['PHP_SELF']) . '/img/' . $img . '.gif';
123 $url = $proto . $host . $file;
124
5e200c3e 125 return '<img src="' . $url . '"' . $height . $width . ' alt="' . $alt . '" />';
c14c86f4 126}
127
1f75b135 128/********************************************************************************
129 * HTML STUFF
130 * Taken from php.net
131 */
132
aef14768 133/**
1f75b135 134 * @return string
135 * @param string
136 * @desc Strip forbidden tags and delegate tag-source check to removeEvilAttributes()
137 */
138function removeEvilTags($source)
139{
f12fdb59 140 $allowedTags = '<h1><b><i><a><ul><li><pre><hr><blockquote><img><br><font><p><small><big><sup><sub><code><em>';
16b17ba5 141 $source = preg_replace('|</div>|i', '<br />', $source);
1f75b135 142 $source = strip_tags($source, $allowedTags);
143 return preg_replace('/<(.*?)>/ie', "'<'.removeEvilAttributes('\\1').'>'", $source);
144}
145
146/**
147 * @return string
148 * @param string
149 * @desc Strip forbidden attributes from a tag
150 */
151function removeEvilAttributes($tagSource)
152{
153 $stripAttrib = 'javascript:|onclick|ondblclick|onmousedown|onmouseup|onmouseover|'.
154 'onmousemove|onmouseout|onkeypress|onkeydown|onkeyup';
155 return stripslashes(preg_replace("/$stripAttrib/i", '', $tagSource));
156}
157
aef14768 158/** Convert html to plain text
159 */
160function htmlToPlainText($res)
161{
16b17ba5 162 $res = trim(html_entity_decode(strip_tags($res, '<div><br><p>')));
163 $res = preg_replace("@</?(br|p|div)[^>]*>@i", "\n", $res);
4cc80c9a 164 if (!is_utf8($res)) {
165 $res = utf8_encode($res);
166 }
aef14768 167 return $res;
168}
169
170/********************************************************************************
171 * RICHTEXT STUFF
172 */
173
174/** Convert richtext to html
175 */
176function richtextToHtml($source)
177{
178 $tags = Array('bold' => 'b',
179 'italic' => 'i',
180 'smaller' => 'small',
181 'bigger' => 'big',
182 'underline' => 'u',
183 'subscript' => 'sub',
184 'superscript' => 'sup',
185 'excerpt' => 'blockquote',
186 'paragraph' => 'p',
187 'nl' => 'br'
188 );
189
190 // clean unsupported tags
191 $protectedTags = '<signature><lt><comment><'.join('><', array_keys($tags)).'>';
192 $source = strip_tags($source, $protectedTags);
193
194 // convert richtext tags to html
195 foreach (array_keys($tags) as $tag) {
196 $source = preg_replace('@(</?)'.$tag.'([^>]*>)@i', '\1'.$tags[$tag].'\2', $source);
197 }
198
199 // some special cases
200 $source = preg_replace('@<signature>@i', '<br>-- <br>', $source);
201 $source = preg_replace('@</signature>@i', '', $source);
202 $source = preg_replace('@<lt>@i', '&lt;', $source);
203 $source = preg_replace('@<comment[^>]*>((?:[^<]|<(?!/comment>))*)</comment>@i', '<!-- \1 -->', $source);
204 return removeEvilAttributes($source);
205}
206
78cd27b3 207/********************************************************************************
208 * HEADER STUFF
209 */
210
211function _headerdecode($charset, $c, $str) {
eeae3e3e 212 $s = ($c == 'Q' || $c == 'q') ? quoted_printable_decode($str) : base64_decode($str);
78cd27b3 213 $s = iconv($charset, 'iso-8859-15', $s);
214 return str_replace('_', ' ', $s);
215}
216
217function headerDecode($value) {
eeae3e3e 218 $val = preg_replace('/(=\?[^?]*\?[BQbq]\?[^?]*\?=) (=\?[^?]*\?[BQbq]\?[^?]*\?=)/', '\1\2', $value);
219 return preg_replace('/=\?([^?]*)\?([BQbq])\?([^?]*)\?=/e', '_headerdecode("\1", "\2", "\3")', $val);
78cd27b3 220}
221
222function headerEncode($value, $trim = 0) {
223 if ($trim) {
224 if (strlen($value) > $trim) {
225 $value = substr($value, 0, $trim) . "[...]";
226 }
227 }
228 return "=?UTF-8?B?".base64_encode($value)."?=";
229}
230
231function header_translate($hdr) {
232 switch ($hdr) {
233 case 'from': return _b_('De');
234 case 'subject': return _b_('Sujet');
235 case 'newsgroups': return _b_('Forums');
236 case 'followup-to': return _b_('Suivi-à');
237 case 'date': return _b_('Date');
238 case 'organization': return _b_('Organisation');
239 case 'references': return _b_('Références');
240 case 'x-face': return _b_('Image');
241 default:
242 if (function_exists('hook_headerTranslate')
243 && $res = hook_headerTranslate($hdr)) {
244 return $res;
245 }
246 return $hdr;
247 }
248}
249
250function formatDisplayHeader($_header,$_text) {
251 global $banana;
252 switch ($_header) {
253 case "date":
254 return formatDate($_text);
255
256 case "followup-to":
257 case "newsgroups":
258 $res = "";
259 $groups = preg_split("/[\t ]*,[\t ]*/",$_text);
260 foreach ($groups as $g) {
8e519bd8 261 $res .= makeHREF(Array('group' => $g), $g) . ', ';
78cd27b3 262 }
263 return substr($res,0, -2);
264
265 case "from":
266 return formatFrom($_text);
267
268 case "references":
269 $rsl = "";
270 $ndx = 1;
271 $text = str_replace("><","> <",$_text);
272 $text = preg_split("/[ \t]/",strtr($text,$banana->spool->ids));
273 $parents = preg_grep("/^\d+$/",$text);
274 $p = array_pop($parents);
275 $par_ok = Array();
276
277 while ($p) {
278 $par_ok[]=$p;
279 $p = $banana->spool->overview[$p]->parent;
280 }
281 foreach (array_reverse($par_ok) as $p) {
1146dd8a 282 $rsl .= makeHREF(Array('group' => $banana->spool->group), $ndx) . ' ';
78cd27b3 283 $ndx++;
284 }
285 return $rsl;
286
287 case "x-face":
78f61fa3 288 return '<img src="' . makeLink(Array('xface' => urlencode(base64_encode($_text)))) .'" alt="x-face" />';
78cd27b3 289
290 default:
291 if (function_exists('hook_formatDisplayHeader')
292 && $res = hook_formatDisplayHeader($_header, $_text))
293 {
294 return $res;
295 }
296 return htmlentities($_text);
297 }
298}
299
300/********************************************************************************
301 * FORMATTING STUFF
302 */
303
304function formatDate($_text) {
305 return strftime("%A %d %B %Y, %H:%M (fuseau serveur)", strtotime($_text));
306}
307
308function fancyDate($stamp) {
309 $today = intval(time() / (24*3600));
310 $dday = intval($stamp / (24*3600));
311
312 if ($today == $dday) {
313 $format = "%H:%M";
314 } elseif ($today == 1 + $dday) {
315 $format = _b_('hier')." %H:%M";
316 } elseif ($today < 7 + $dday) {
317 $format = '%a %H:%M';
318 } else {
319 $format = '%a %e %b';
320 }
321 return strftime($format, $stamp);
322}
323
324function formatFrom($text) {
325# From: mark@cbosgd.ATT.COM
326# From: mark@cbosgd.ATT.COM (Mark Horton)
327# From: Mark Horton <mark@cbosgd.ATT.COM>
328 $mailto = '<a href="&#109;&#97;&#105;&#108;&#116;&#111;&#58;';
329
330 $result = htmlentities($text);
331 if (preg_match("/^([^ ]+)@([^ ]+)$/",$text,$regs)) {
332 $result="$mailto{$regs[1]}&#64;{$regs[2]}\">".htmlentities($regs[1]."&#64;".$regs[2])."</a>";
333 }
334 if (preg_match("/^([^ ]+)@([^ ]+) \((.*)\)$/",$text,$regs)) {
335 $result="$mailto{$regs[1]}&#64;{$regs[2]}\">".htmlentities($regs[3])."</a>";
336 }
337 if (preg_match("/^\"?([^<>\"]+)\"? +<(.+)@(.+)>$/",$text,$regs)) {
338 $result="$mailto{$regs[2]}&#64;{$regs[3]}\">".htmlentities($regs[1])."</a>";
339 }
340 return preg_replace("/\\\(\(|\))/","\\1",$result);
341}
342
343function displayshortcuts($first = -1) {
344 global $banana;
345 extract($banana->state);
346
347 $res = '<div class="banana_scuts">';
8e519bd8 348 $res .= '[' . makeHREF(Array(), _b_('Liste des forums')) . '] ';
78cd27b3 349 if (is_null($group)) {
8e519bd8 350 return $res.'[' . makeHREF(Array('subscribe' => 1), _b_('Abonnements')) . ']</div>';
78cd27b3 351 }
352
8e519bd8 353 $res .= '[' . makeHREF(Array('group' => $group), $group) . '] ';
78cd27b3 354
355 if (is_null($artid)) {
8e519bd8 356 $res .= '[' . makeHREF(Array('group' => $group,
f6df9eb2 357 'action' => 'new'),
358 _b_('Nouveau message'))
359 . '] ';
78cd27b3 360 if (sizeof($banana->spool->overview)>$banana->tmax) {
361 $res .= '<br />';
362 $n = intval(log(count($banana->spool->overview), 10))+1;
363 for ($ndx=1; $ndx <= sizeof($banana->spool->overview); $ndx += $banana->tmax) {
364 if ($first==$ndx) {
365 $fmt = "[%0{$n}u-%0{$n}u] ";
366 } else {
8e519bd8 367 $fmt = '[' . makeHREF(Array('group' => $group,
f6df9eb2 368 'first' => $ndx),
369 '%0' . $n . 'u-%0' . $n . 'u')
370 . '] ';
78cd27b3 371 }
372 $res .= sprintf($fmt, $ndx, min($ndx+$banana->tmax-1,sizeof($banana->spool->overview)));
373 }
374 }
375 } else {
8e519bd8 376 $res .= '[' . makeHREF(Array('group' => $group,
f6df9eb2 377 'artid' => $artid,
378 'action' => 'new'),
379 _b_('Répondre'))
380 . '] ';
f2dab64a 381 if ($banana->post && $banana->post->checkcancel()) {
8e519bd8 382 $res .= '[' . makeHREF(Array('group' => $group,
f6df9eb2 383 'artid' => $artid,
384 'action' => 'cancel'),
385 _b_('Annuler ce message'))
386 . '] ';
78cd27b3 387 }
388 }
389 return $res.'</div>';
390}
391
392/********************************************************************************
393 * FORMATTING STUFF : BODY
394 */
395
9bd35988 396function autoformat($text)
397{
398 global $banana;
399 $length = $banana->wrap;
400
401 $cmd = "echo ".escapeshellarg($text)." | perl -MText::Autoformat -e 'autoformat {left=>1, right=>$length, all=>1 };'";
402 exec($cmd, $result, $ret);
403 if ($ret != 0) {
ebd93a2d 404 $result = split("\n", $text);
9bd35988 405 }
406 return $result;
407}
408
87514711 409function wrap($text, $_prefix="", $_force=false)
78cd27b3 410{
411 $parts = preg_split("/\n-- ?\n/", $text);
412 if (count($parts) >1) {
413 $sign = "\n-- \n" . array_pop($parts);
414 $text = join("\n-- \n", $parts);
415 } else {
416 $sign = '';
78cd27b3 417 }
418
419 global $banana;
87514711 420 $url = $banana->url_regexp;
78cd27b3 421 $length = $banana->wrap;
87514711 422 $max = $length + ($length/10);
423 $splits = split("\n", $text);
9bd35988 424 $result = array();
425 $next = array();
426 $format = false;
87514711 427 foreach ($splits as $line) {
428 if ($_force || strlen($line) > $max) {
9bd35988 429 if (preg_match("!^(.*)($url)(.*)!i", $line, $matches) && strlen($matches[2]) > $length && strlen($matches) < 900) {
430 if (strlen($matches[1]) != 0) {
431 array_push($next, rtrim($matches[1]));
432 if (strlen($matches[1]) > $max) {
433 $format = true;
434 }
435 }
436
437 if ($format) {
438 $result = array_merge($result, autoformat(join("\n", $next)));
439 } else {
440 $result = array_merge($result, $next);
441 }
442 $format = false;
443 $next = array();
444 array_push($result, $matches[2]);
445
446 if (strlen($matches[6]) != 0) {
447 array_push($next, ltrim($matches[6]));
448 if (strlen($matches[6]) > $max) {
449 $format = true;
450 }
451 }
452 } else {
453 $format = true;
454 array_push($next, $line);
87514711 455 }
9bd35988 456 } else {
457 array_push($next, $line);
87514711 458 }
459 }
9bd35988 460 if ($format) {
461 $result = array_merge($result, autoformat(join("\n", $next)));
462 } else {
463 $result = array_merge($result, $next);
a4c1ad46 464 }
78cd27b3 465
466 return $_prefix.join("\n$_prefix", $result).($_prefix ? '' : $sign);
467}
468
87514711 469function cutlink($link)
470{
471 global $banana;
472
473 if (strlen($link) > $banana->wrap) {
474 $link = substr($link, 0, $banana->wrap - 3)."...";
475 }
476 return $link;
477}
478
e7241984 479function cleanurl($url)
480{
481 $url = str_replace('@', '%40', $url);
482 return '<a href="'.$url.'" title="'.$url.'">'.cutlink($url).'</a>';
483}
484
87514711 485function formatbody($_text, $format='plain', $flowed=false)
1f75b135 486{
487 if ($format == 'html') {
16b17ba5 488 $res = '<br/>'.html_entity_decode(to_entities(removeEvilTags($_text))).'<br/>';
aef14768 489 } else if ($format == 'richtext') {
d67abd9d 490 $res = '<br/>'.html_entity_decode(to_entities(richtextToHtml($_text))).'<br/>';
1f75b135 491 } else {
87514711 492 $res = "\n\n" . to_entities(wrap($_text, "", $flowed))."\n\n";
1ff637a0 493 $formatting = Array('\*' => 'strong',
494 '_' => 'u',
495 '/' => 'em');
0b2475ce 496 $word = '-a-z0-9\'âäàãéêëêïîìöôòõüûùÿ';
1ff637a0 497 foreach ($formatting as $limit=>$mark) {
0b2475ce 498 $res = preg_replace('@([^' . $word . '])'
499 . $limit . '([' . $word . ']+)' . $limit
500 . '([^' . $word . '])@i'
501 ,'\1<' . $mark . '>\2</' . $mark . '>\3'
502 , $res);
1ff637a0 503 }
f6df9eb2 504 }
1f75b135 505
d67abd9d 506 if ($format != 'html') {
507 global $banana;
508 $url = $banana->url_regexp;
509 $res = preg_replace("/(&lt;|&gt;|&quot;)/", " \\1 ", $res);
e7241984 510 $res = preg_replace("!$url!ie", "'\\1'.cleanurl('\\2').'\\3'", $res);
d67abd9d 511 $res = preg_replace('/(["\[])?(?:mailto:)?([a-z0-9.\-+_]+@[a-z0-9.\-+_]+)(["\]])?/i', '\1<a href="mailto:\2">\2</a>\3', $res);
512 $res = preg_replace("/ (&lt;|&gt;|&quot;) /", "\\1", $res);
513
514 if ($format == 'richtext') {
515 $format = 'html';
516 }
517 }
87514711 518
1f75b135 519 if ($format == 'html') {
19e26ef7 520 $res = preg_replace("@(</p>)\n?-- ?\n?(<p[^>]*>|<br[^>]*>)@", "\\1<br/>-- \\2", $res);
521 $res = preg_replace("@<br[^>]*>\n?-- ?\n?(<p[^>]*>)@", "<br/>-- <br/>\\2", $res);
522 $res = preg_replace("@(<pre[^>]*>)\n?-- ?\n@", "<br/>-- <br/>\\1", $res);
523 $parts = preg_split("@(:?<p[^>]*>\n?-- ?\n?</p>|<br[^>]*>\n?-- ?\n?<br[^>]*>)@", $res);
11102461 524 } else {
ebd93a2d 525 while (preg_match("@(^|<pre>|\n)&gt;@i", $res)) {
6c7c1730 526 $res = preg_replace("@(^|<pre>|\n)((&gt;[^\n]*\n)+)@ie",
ebd93a2d 527 "'\\1</pre><blockquote><pre>'"
f6df9eb2 528 .".stripslashes(preg_replace('@(^|<pre>|\n)&gt;[ \\t\\r]*@i', '\\1', '\\2'))"
529 .".'</pre></blockquote><pre>'",
530 $res);
cfaab21d 531 }
f6df9eb2 532 $res = preg_replace("@<pre>-- ?\n@", "<pre>\n-- \n", $res);
11102461 533 $parts = preg_split("/\n-- ?\n/", $res);
1f75b135 534 }
535
78cd27b3 536 if (count($parts) > 1) {
11102461 537 $sign = array_pop($parts);
538 if ($format == 'html') {
539 $res = join('<br/>-- <br/>', $parts);
cfaab21d 540 $sign = '<hr style="width: 100%; margin: 1em 0em; " />'.$sign.'<br/>';
11102461 541 } else {
542 $res = join('\n-- \n', $parts);
543 $sign = '</pre><hr style="width: 100%; margin: 1em 0em; " /><pre>'.$sign;
544 }
545 return $res.$sign;
78cd27b3 546 } else {
547 return $res;
548 }
549}
550
f6df9eb2 551// vim:set et sw=4 sts=4 ts=4
78cd27b3 552?>