一个好用的UBB类!

ubb

<?php
/*
如有转载,请注明作者

原作者: 何志强
改进:   SonyMusic[sonymusic@163.net]
文件:   ubb.php
备注:   说是改进,其实核心函数parse()已经完全重写了,而且思路也是不一样的。
        不过仍是受何志强的例子的启发,而且测试的例子还有URLCHECK等几个函数也是沿用的何志强的程序,谢谢何志强。
        目前还没有颜色的功能,但我会加入的。
        如果在程序上有什么BUG或不便的地方,请给我MAIL。
        谢谢!
改进功能:
    对字符串进行UBB编码,该类目前只支持下列几个简单且实用的编码:
    1. URL裢接
       http://phpuser.com/
       http://头可以不需要
       如phpuser.com也是可以的。
    2. Email裢接
       sonymusic@163.net
    3. 图片裢接
       
       同URL链接一样,前面的http也可以不要。
    4. 文字方面
       粗体字
       斜体字
       加下划线
       

1号标题字

...

6号标题字

       
       
       [tt][/tt]
       [s][/s]
       
       [em][/em]
       [strong][/strong]
       
       [samp][/samp]
       [kbd][/kbd]
       [var][/var]
       [dfn][/dfn]
       [cite][/cite]
       
       
       
    注意以下几点:
    1. url,email,img等标签是不分大小写的.
    2. 在标签中不允许有TAB键出现,但空格允许。
    3. 该类要调用htmlencode,htmlencode4textarea,emailcheck函数和urlcheck类.
    4. 修改后支持嵌套,但url,email,img这三个标签不是允许嵌套的。
技术资料:
    Ultimate Bulletin Board
        http://www.ultimatebb.com/
    What is UBB Code
        http://www.scriptkeeper.com/ubb/ubbcode.html
*/

include("urlcheck.php");
include("otherfunc.php"); //这两个文件的内容,附在最后。

//ubbcode类
class ubbcode{
    var $call_time=0;
   //可处理标签及处理函数对应表
   var $tags = array(  //小写的标签 => 对应的处理函数
                       'url' => '$this->url',
                       'email' => '$this->email',
                       'img' => '$this->img',
                       'b' => '$this->simple',
                       'i' => '$this->simple',
                       'u' => '$this->simple',
                       'tt' => '$this->simple',
                       's' => '$this->simple',
                       'strike' => '$this->simple',
                       'h1' => '$this->simple',
                       'h2' => '$this->simple',
                       'h3' => '$this->simple',
                       'h4' => '$this->simple',
                       'h5' => '$this->simple',
                       'h6' => '$this->simple',
                       'sup' => '$this->simple',
                       'sub' => '$this->simple',
                       'em' => '$this->simple',
                       'strong' => '$this->simple',
                       'code' => '$this->simple',
                       'samp' => '$this->simple',
                       'kbd' => '$this->simple',
                       'var' => '$this->simple',
                       'dfn' => '$this->simple',
                       'cite' => '$this->simple',
                       'small' => '$this->simple',
                       'big' => '$this->simple',
                       'blink' => '$this->simple'
                    );
   //url裢接属性
   var $attr_url;
   //url合法性检查对象
   var $urlcheck;

   function ubbcode($attr_url){
      $this->attr_url = ''.$attr_url;
      $this->urlcheck = new urlcheck();
   }

   //对$str进行UBB编码解析
   function parse($str){
           $this->call_time++;
      $parse = ''.htmlencode($str);

      $ret = '';
      while(true){
          $eregi_ret=eregi("\[[#]{0,1}[[:alnum:]]{1,7}\]",$parse,$eregi_arr); //查找[xx]
        if(!$eregi_ret){
            $ret .= $parse;
            break;  //如果没有,返回
         }
         $pos = @strpos($parse,$eregi_arr[0]);
         $tag_len=strlen($eregi_arr[0])-2;//标记长度
        $tag_start=substr($eregi_arr[0],1,$tag_len);
        $tag=strtolower($tag_start);
        
         if((($tag=="url") or ($tag=="email") or ($tag=="img")) and ($this->call_time>1)){
             echo $this->call_time."<br>";
             return $parse;//如果不能是不能嵌套的标记,直接返回
        }
        
         $parse2 = substr($parse,0,$pos);//标记之前
         $parse = substr($parse,$pos+$tag_len+2);//标记之后
         if(!isset($this->tags[$tag])){
             echo "$tag_start<br>";
            $ret .= $parse2.'['.$tag_start.']';
            continue;//如果是不支持的标记
         }
        
         //查找对对应的结束标记
         $eregi_ret=eregi("\[\/".$tag."\]",$parse,$eregi_arr);
         if(!$eregi_ret){
            $ret .= $parse2.'['.$tag_start.']';
            continue;//如果没有对应该的结束标记
         }
         $pos=strpos($parse,$eregi_arr[0]);
         $value=substr($parse,0,$pos);//这是起止标记之间的内容
         $tag_end=substr($parse,$pos+2,$tag_len);
         $parse=substr($parse,$pos+$tag_len+3);//结束标记之后的内容
         
         if(($tag!="url") and ($tag!="email") and ($tag!="img")){
             $value=$this->parse($value);
        }

         $ret .= $parse2;
         eval('$ret .= '.$this->tags[$tag].'("'.$tag_start.'","'.$tag_end.'","'.$value.'");');
      }
        $this->call_time--;
      return $ret;
   }

   function simple($start,$end,$value){
      return '<'.$start.'>'.$value.'</'.$end.'>';
   }

   function url($start,$end,$value){
           $trim_value=trim($value);
        if (strtolower(substr($trim_value,0,7))!="http://")
            $trim_value="http://".$trim_value;
      if($this->urlcheck->check($trim_value)) return '<a href="'.$trim_value.'" '.$this->attr_url.'>'.$value.'</a>';
      else return '['.$start.']'.$value.'[/'.$end.']';
   }

   function email($start,$end,$value){
      if(emailcheck($value)) return '<a href="mailto:'.$value.'">'.$value.'</a>';
      else return '['.$start.']'.$value.'[/'.$end.']';
   }

   function img($start,$end,$value){
           $trim_value=trim($value);
        if ((strtolower(substr($trim_value,0,7))!="http://") or ($this->urlcheck->check($trim_value)))
               return '<img src="'.$trim_value.'"></img>';
      else return '['.$start.']'.$value.'[/'.$end.']';
   }
}

//测试
echo '<html>';
echo '<head><title>测试</title></head>';
echo '<body>';
echo '<form action="'.str2url($PATH_INFO).'" method="post">';
echo '<textarea cols="100" rows="10" name="ubb">'.htmlencode4textarea($ubb).'</textarea><br>';
echo '<input type="submit" value="转换">';
echo '</form>';

if(isset($ubb)){
   $ubbcode = new ubbcode('target="_blank"');
   echo '<hr>'.$ubbcode->parse($ubb);
}

echo '</body>';
echo '</html>';

?>

文件urlcheck.php的内容
<?php
//urlcheck.php
class urlcheck{
   var $regex = array(//协议名(注意在这里必须写成小写) => 对应的正则表达式
                      'ftp' => '$this->ftpurl',
                      'file' => '$this->fileurl',
                      'http' => '$this->httpurl',
                      'https' => '$this->httpurl',
                      'gopher' => '$this->gopherurl',
                      'news' => '$this->newsurl',
                      'nntp' => '$this->nntpurl',
                      'telnet' => '$this->telneturl',
                      'wais' => '$this->waisurl'
                     );

   var $lowalpha;
   var $hialpha;
   var $alpha;
   var $digit;
   var $safe;
   var $extra;
   var $national;
   var $punctuation;
   var $reserved;
   var $hex;
   var $escape;
   var $unreserved;
   var $uchar;
   var $xchar;
   var $digits;

   var $urlpath;
   var $password;
   var $user;
   var $port;
   var $hostnumber;
   var $alphadigit;
   var $toplabel;
   var $domainlabel;
   var $hostname;
   var $host;
   var $hostport;
   var $login;

   //ftp
   var $ftptype;
   var $fsegment;
   var $fpath;
   var $ftpurl;

   //file
   var $fileurl;

   //http,https
   var $search;
   var $hsegment;
   var $hpath;
   var $httpurl;

   //gopher
   var $gopher_string;
   var $selector;
   var $gtype;
   var $gopherurl;

   //news
   var $article;
   var $group;
   var $grouppart;
   var $newsurl;

   //nntp
   var $nntpurl;

   //telnet
   var $telneturl;

   //wais
   var $wpath;
   var $wtype;
   var $database;
   var $waisdoc;
   var $waisindex;
   var $waisdatabase;
   var $waisurl;

   function check($url){
      $pos = @strpos($url,':',1);
      if($pos<1) return false;
      $prot = substr($url,0,$pos);
      if(!isset($this->regex[$prot])) return false;
      eval('$regex = '.$this->regex[$prot].';');
      return ereg('^'.$regex.'$',$url);
   }

   function urlcheck(){
      $this->lowalpha = '[a-z]';
      $this->hialpha = '[A-Z]';
      $this->alpha = '('.$this->lowalpha.'|'.$this->hialpha.')';
      $this->digit = '[0-9]';
      $this->safe = '[$.+_-]';
      $this->extra = '[*()\'!,]';
      $this->national = '([{}|\^~`]|\\[|\\])';
      $this->punctuation = '[<>#%"]';
      $this->reserved = '[?;/:@&=]';
      $this->hex = '('.$this->digit.'|[a-fA-F])';
      $this->escape = '(%'.$this->hex.'{2})';
      $this->unreserved = '('.$this->alpha.'|'.$this->digit.'|'.$this->safe.'|'.$this->extra.')';
      $this->uchar = '('.$this->unreserved.'|'.$this->escape.')';
      $this->xchar = '('.$this->unreserved.'|'.$this->reserved.'|'.$this->escape.')';
      $this->digits = '('.$this->digit.'+)';

      $this->urlpath = '('.$this->xchar.'*)';
      $this->password = '(('.$this->uchar.'|[?;&=]'.')*)';
      $this->user = '(('.$this->uchar.'|[?;&=]'.')*)';
      $this->port = $this->digits;
      $this->hostnumber = '('.$this->digits.'.'.$this->digits.'.'.$this->digits.'.'.$this->digits.')';
      $this->alphadigit = '('.$this->alpha.'|'.$this->digit.')';
      $this->toplabel = '('.$this->alpha.'|('.$this->alpha.'('.$this->alphadigit.'|-)*'.$this->alphadigit.'))';
      $this->domainlabel = '('.$this->alphadigit.'|('.$this->alphadigit.'('.$this->alphadigit.'|-)*'.$this->alphadigit.'))';
      $this->hostname = '(('.$this->domainlabel.'\\.)*'.$this->toplabel.')';
      $this->host = '('.$this->hostname.'|'.$this->hostnumber.')';
      $this->hostport = '('.$this->host.'(:'.$this->port.')?)';
      $this->login = '(('.$this->user.'(:'.$this->password.')?@)?'.$this->hostport.')';

      $this->ftptype = '[aidAID]';
      $this->fsegment = '(('.$this->uchar.'|[?:@&=])*)';
      $this->fpath = '('.$this->fsegment.'(/'.$this->fsegment.')*)';
      $this->ftpurl = '([fF][tT][pP]://'.$this->login.'(/'.$this->fpath.'(;[tT][yY][pP][eE]='.$this->ftptype.')?)?)';

      $this->fileurl = '([fF][iI][lL][eE]://('.$this->host.'|[lL][oO][cC][aA][lL][hH][oO][sS][tT])?/'.$this->fpath.')';

      $this->search = '(('.$this->uchar.'|[;:@&=])*)';
      $this->hsegment = '(('.$this->uchar.'|[;:@&=])*)';
      $this->hpath = '('.$this->hsegment.'(/'.$this->hsegment.')*)';
      $this->httpurl = '([hH][tT][tT][pP][sS]?://'.$this->hostport.'(/'.$this->hpath.'([?]'.$this->search.')?)?)';

      $this->gopher_string = '('.$this->xchar.'*)';
      $this->selector = '('.$this->xchar.'*)';
      $this->gtype = $this->xchar;
      $this->gopherurl = '([gG][oO][pP][hH][eE][rR]://'.$this->hostport.'(/('.$this->gtype.'('.$this->selector.'(%09'.$this->search.'(%09'.$this->gopher_string.')?)?)?)?)?)';

      $this->article = '(('.$this->uchar.'|[;/?:&=])+@'.$this->host.')';
      $this->group = '('.$this->alpha.'('.$this->alpha.'|'.$this->digit.'|[-.+_])*)';
      $this->grouppart = '([*]|'.$this->group.'|'.$this->article.')';
      $this->newsurl = '([nN][eE][wW][sS]:'.$this->grouppart.')';

      $this->nntpurl = '([nN][nN][tT][pP]://'.$this->hostport.'/'.$this->group.'(/'.$this->digits.')?)';

      $this->telneturl = '([tT][eE][lL][nN][eE][tT]://'.$this->login.'/?)';

      $this->wpath = '('.$this->uchar.'*)';
      $this->wtype = '('.$this->uchar.'*)';
      $this->database = '('.$this->uchar.'*)';
      $this->waisdoc = '([wW][aA][iI][sS]://'.$this->hostport.'/'.$this->database.'/'.$this->wtype.'/'.$this->wpath.')';
      $this->waisindex = '([wW][aA][iI][sS]://'.$this->hostport.'/'.$this->database.'[?]'.$this->search.')';
      $this->waisdatabase = '([wW][aA][iI][sS]://'.$this->hostport.'/'.$this->database.')';
      $this->waisurl = '('.$this->waisdatabase.'|'.$this->waisindex.'|'.$this->waisdoc.')';
   }
}

?>

文件otherfunc.php的内容
<?php
//otherfunc.php
function htmlencode($str){
   $str = (string)$str;

   $ret = '';
   $len = strlen($str);
   $nl = false;
   for($i=0;$i<$len;$i++){
      $chr = $str[$i];
      switch($chr){
         case '<':
            $ret .= '<';
            $nl = false;
            break;
         case '>':
            $ret .= '>';
            $nl = false;
            break;
         case '"':
            $ret .= '"';
            $nl = false;
            break;
         case '&':
            $ret .= '&';
            $nl = false;
            break;
/*
         case ' ':
            $ret .= ' ';
            $nl = false;
            break;
*/        
         case chr(9):
            $ret .= '    ';
            $nl = false;
            break;
         case chr(10):
            if($nl) $nl = false;
            else{
               $ret .= '<br>';
               $nl = true;
            }
            break;
         case chr(13):
            if($nl) $nl = false;
            else{
               $ret .= '<br>';
               $nl = true;
            }
            break;
         default:
            $ret .= $chr;
            $nl = false;
            break;
      }
   }

   return $ret;
}

function htmlencode4textarea($str){
   $str = (string)$str;

   $ret = '';
   $len = strlen($str);
   for($i=0;$i<$len;$i++){
      $chr = $str[$i];
      switch($chr){
         case '<':
            $ret .= '<';
            break;
         case '>':
            $ret .= '>';
            break;
         case '"':
            $ret .= '"';
            break;
         case '&':
            $ret .= '&';
            break;
         case ' ':
            $ret .= ' ';
            break;
         case chr(9):
            $ret .= '    ';
            break;
         default:
            $ret .= $chr;
            break;
      }
   }

   return $ret;
}

function emailcheck($email){
   $ret=false;
   if(strstr($email, '@') && strstr($email, '.')){
      if(eregi("^([_a-z0-9]+([\\._a-z0-9-]+)*)@([a-z0-9]{2,}(\\.[a-z0-9-]{2,})*\\.[a-z]{2,3})$", $email)){
         $ret=true;
      }
   }
   return $ret;
}

function str2url($path){
   return eregi_replace("%2f","/",urlencode($path));
}
?>

时间: 2024-11-01 12:26:30

一个好用的UBB类!的相关文章

PHP实现的一个简单的数据库操作类

PHP实现的一个简单的数据库操作类 实现的功能: - 在实例化的时候能设置连接字符集 - 在实例化的时候能连接数据库 - 在实例化的时候能选择默认数据库 - 销毁对象时关闭数据库 代码如下: <?php // 数据库操作类MySQLDB class MySQLDB { // 声明属性 private $server; private $username; private $password; public $default_db; public $link; // 声明构造函数 public f

表情发布(类似QQ表情)和UBB类(c#)

ubb 最近由于项目的需要,特别做了个表情的发布(类似QQ表情)和UBB的类^_^ 大家看下图: 表情: UBB类的应用: 其实都很简单的,如果大家有兴趣的话可以给我发邮件,我会提供源代码. ps:由于没有空间,只能靠邮件了^_^

贴一个偶写的分页类

分页 抽象类==========================================================import java.util.ArrayList;import java.sql.Connection;import java.sql.ResultSet; import com.xxx.util.DBTool; /** * <pre> * 分页类.默认页面大小为20 * 这是一个抽象类.子类需要重构方法selResult() * </pre> */p

PHP跳转函数和一个通用的操作提示类的编写

PHP 跳转,即重定向浏览器到指定的 URL,是一个很常见的功能.这种功能也有一些细节性的要求,比如等待多少秒以后跳转,用不用JavaScript实现跳转,等等.下面的跳转方法考虑到很多,并参数化,可以用到具体的项目当中. <?php /** * 重定向浏览器到指定的 URL * * @param string $url 要重定向的 url * @param int $delay 等待多少秒以后跳转 * @param bool $js 指示是否返回用于跳转的 JavaScript 代码 * @p

根据模板文件生成一个静态html文件的类

一般我们用PHP输出一个html文件,总是用$head="<head>--</head>"这样一个长串来完成.本类主要提供一个简便的用PHP输出html文件的方法.避免了在程序中出现过多的带有长字串. 类定义文件 createhtml.class.php 如下: <?php //------------------- // TCreateHTML //根据模板文件生成一个静态html文件的类 // 作者:sharetop // email:ycshowto

运算符重载-用C++编程:定义一个四进制的类,重定义“+”号实现四进制数的累加。

问题描述 用C++编程:定义一个四进制的类,重定义"+"号实现四进制数的累加. 定义一个四进制的类,重定义"+"号实现四进制数的累加. 输入第一行输入所需要的四进制数的个数第二行开始,依次输入四进制数 输出所有输入四进制数累加的和 解决方案 又是作业贴啊,要是做的过程中有问题,LZ直接说遇到的问题吧 解决方案二: 我写了个简单的,你试试好用不,没有做输入判断,你输入一定不能输4或者4以上的数123+321+333=2103 #include ""

代码-一个JAVA苦手的关于类的问题

问题描述 一个JAVA苦手的关于类的问题 public class Animal{ public Animal(){ } public void cry(){ System.out.print("crying"); } } 其中为什么还要再加上public Animal(){ }这一行代码? 解决方案 创建一个子类对象的实例的时候,必先调用父类的无参构造函数(默认构造函数),假如父类有带参数的构造函数,那么系统不会给他创建无参数的构造函数,这时,子类在实例化的时候,因为找不到父类的默认

定制并发类(三)实现一个基于优先级的Executor类

声明:本文是< Java 7 Concurrency Cookbook>的第七章,作者: Javier Fernández González     译者:许巧辉 实现一个基于优先级的Executor类 在Java并发API的第一个版本中,你必须创建和运行应用程序中的所有线程.在Java版本5中,随着执行者框架(Executor framework)的出现,对于并发任务的执行,一个新的机制被引进. 使用执行者框架(Executor framework),你只要实现你的任务并把它们提交给执行者.

ios-iOS:请问一个线程涉及2个类,如何向该线程performSelecto?

问题描述 iOS:请问一个线程涉及2个类,如何向该线程performSelecto? 我有一个线程,这线程首先是在a类中创建,并调用a类的a1方法,而a1方法又调用了b类的b1方法. 我在c类中,想向该线程发通知,希望其调用b类的b2方法, 我的代码: [self performSelector:@selector(b2) onThread:m_thread withObject:nil waitUnitilDone:NO]; 提示:unrecognized selector to instan