<?php
class mod_pushpost{
var $mypage;
function mod_pushpost($PMS){
$PMS->hookModuleMethod('ModulePage', __CLASS__); // 向系統登記模組專屬獨立頁面
$this->mypage = $PMS->getModulePageURL(__CLASS__);
$this->PUSHPOST_SEPARATOR = '[MOD_PUSHPOST_USE]';
$this->PUSHPOST_DEF = 10; // 討論串最多顯示之推文筆數 (超過則自動隱藏,全部隱藏:0)
$this->PUSH_EMOTIONS = array("","(゚∀゚)","(´∀`)","(*´∀`)","(*゚∇゚)","(*´д`)","(*゚ー゚)","(゚3゚)","(´ー`)","(・_ゝ・)","(´,_ゝ`)","(´_ゝ`)","(・ー・)","(・∀・)","(ゝ∀・)",
"(〃∀〃)","(*゚∀゚*)","(゚∀。)","(`д´)","(`ε´)","(`ヮ´)","σ`∀´)","゚∀゚)σ","゚∀゚)ノ","(╬゚д゚)","(|||゚д゚)","(´゚д゚`)","(゚д゚)","Σ(゚д゚)","(;゚д゚)","(;´д`)",
"(д)゚゚","(☉д⊙)","(((゚д゚)))","(`・´)","(´д`)","(-д-)","(>д<)","(つд⊂)","・゚(ノд`゚)","(TдT)","(*´ω`*)","(・ω・)","(`・ω・)","(´・ω)","(`・ω)",
"(´・ω・`)","(`・ω・´)","(・_っ・)","(^_っ^)","(`_っ´)","(`ー´)","(´_っ`)","(´ρ`)","(゚ω゚)","(o゚ω゚o)","(^ω^)","(・ิ㉨・ิ)","(。◕∀◕。)","ヾ(´ε`ヾ)","(ノ・ω・)ノ",
"(ノ゚∀゚)ノ","(σ゚д゚)σ","(σ゚∀゚)σ","|д`)","|ー`)","|д゚)","|∀゚)","ヽ(●´∀`●)ノ","(゚д゚≡゚д゚)","。・゚・(つд`゚)・゚・","_,._","゚Å゚)","⊂彡☆))д`)","⊂彡☆))д´)",
"⊂彡☆))∀`)","(´∀((☆ミつ"); // 表情
$this->GESTURES = array("","<","。o0","ノシ","彡","o彡゚","ゝ","ㄏ","ノ","ノ゛","ノ","ノミ","ツ","っ","っ゛","つ","つ゛","o","凸","/","y","y━・~","y-~","≡3","~♪","♬","❤",
"~♥","σ","ノ彡┴─┴","⌒*","⌒☆","⌒★","≡☆"); //發言
AttachLanguage(array($this, '_loadLanguage')); // 載入語言檔
}
/* Get the name of module */
function getModuleName(){
return 'mod_pushpost : 文章推文機制';
}
/* Get the module version infomation */
function getModuleVersionInfo(){
return 'r811整合V板表情符號 (v120616)';
}
/* 生成識別ID */
function _getID(){
return substr(crypt(md5(getREMOTE_ADDR().IDSEED.gmdate('Ymd', time() + TIME_ZONE * 3600)), 'id'), -8);
}
function autoHookHead(&$txt, $isReply){
global $language;
$txt .= '<style type="text/css">.pushpost { background-color: #fff; font-size: 0.8em; padding: 10px; }</style>
<script type="text/javascript">
// <![CDATA[
var lastpushpost=0;
function mod_pushpostShow(pid){
$g("mod_pushpostID").value = pid;
$g("mod_pushpostName").value = getCookie("namec");
$("div#mod_pushpostBOX").insertBefore($("div#r"+pid+" .quote"));
if(lastpushpost!=pid) {
$("div#mod_pushpostBOX").show();
} else
$("div#mod_pushpostBOX").toggle();
lastpushpost = pid;
return false;
}
function mod_pushpostKeyPress(e){if(e.which==13){e.preventDefault();mod_pushpostSend();}}
function mod_pushpostSend(){
var o0 = $g("mod_pushpostID"), o1 = $g("mod_pushpostName"), o2 = $g("mod_pushpostComm"), o3 = $g("mod_pushpostSmb"), pp = $("div#r"+o0.value+" .quote"), o4 = $("#push_emotion option:selected").val(), o5 = $("#gesture option:selected").val();
if(o2.value===""){ alert("'._T('modpushpost_nocomment').'"); return false; }
o1.disabled = o2.disabled = o3.disabled = true;
$.ajax({
url: "'.str_replace('&', '&', $this->mypage).'&no="+o0.value,
type: "POST",
data: {ajaxmode: true, name: o1.value, push_emotion: o4, gesture: o5, comm: o2.value},
success: function(rv){
if(rv.substr(0, 4)!=="+OK "){ alert(rv); o3.disabled = false; return false; }
rv = rv.substr(4);
(pp.find(".pushpost").length===0)
? pp.append("<div class=\'pushpost\'>"+rv+"</div>")
: pp.children(".pushpost").append("<br />"+rv);
o0.value = o1.value = o2.value = o4 = o5 = ""; o1.disabled = o2.disabled = o3.disabled = false;
$("div#mod_pushpostBOX").hide();
},
error: function(){ alert("Network error."); o1.disabled = o2.disabled = o3.disabled = false; }
});
}
// ]]>
</script>';
}
function autoHookFoot(&$foot){
global $language;
$ecnt=count($this->PUSH_EMOTIONS);
$gcnt=count($this->GESTURES);
for($i=0;$i<$ecnt;$i++) {
$emot.="<option>".$this->PUSH_EMOTIONS[$i]."</option>\n";
}
for($i=0;$i<$gcnt;$i++) {
$gest.="<option>".$this->GESTURES[$i]."</option>\n";
}
$foot .= '
<div id="mod_pushpostBOX" style="display:none">
<input type="hidden" id="mod_pushpostID" />'._T('modpushpost_pushpost').' <ul>
<li>'._T('form_name').' <input type="text" id="mod_pushpostName" maxlength="20" onkeypress="mod_pushpostKeyPress(event)" /></li>
<li>'._T('form_comment').
'<select id="push_emotion" style="text-align:right;line-height:1.7em" class="push_area">'.$emot.'</select>'.
'<select id="gesture" class="push_area" style="line-height:1.7em">'.$gest.'</select>'.
'<input type="text" id="mod_pushpostComm" size="50" maxlength="50" onkeypress="mod_pushpostKeyPress(event)" /><input type="button" id="mod_pushpostSmb" value="'._T('form_submit_btn').'" onclick="mod_pushpostSend()" /></li></ul>
</div>
';
}
function autoHookThreadPost(&$arrLabels, $post, $isReply){
global $language, $PIO;
$pushcount = '';
if($post['status'] != ''){
$f = $PIO->getPostStatus($post['status']);
$pushcount = $f->value('mppCnt'); // 被推次數
}
$arrLabels['{$QUOTEBTN}'] .= ' <a href="'.$this->mypage.'&no='.$post['no'].'" onclick="return mod_pushpostShow('.$post['no'].')">'.$pushcount._T('modpushpost_pushbutton').'</a>';
if(strpos($arrLabels['{$COM}'], $this->PUSHPOST_SEPARATOR.'<br />') !== false){
if($isReply || $pushcount <= $this->PUSHPOST_DEF) // 回應模式
$arrLabels['{$COM}'] = str_replace($this->PUSHPOST_SEPARATOR.'<br />', '<div class="pushpost">', $arrLabels['{$COM}']).'</div>';
else{ // 頁面瀏覽
$delimiter = strpos($arrLabels['{$COM}'], $this->PUSHPOST_SEPARATOR.'<br />'); // 定位符號位置
if($this->PUSHPOST_DEF > 0){
$push_array = explode('<br />', substr($arrLabels['{$COM}'], $delimiter + strlen($this->PUSHPOST_SEPARATOR.'<br />')));
$pushs = '<div class="pushpost">……<br />'.implode('<br />', array_slice($push_array, 0 - $this->PUSHPOST_DEF)).'</div>';
}else{
$pushs = '';
}
$arrLabels['{$COM}'] = substr($arrLabels['{$COM}'], 0, $delimiter).$pushs;
$arrLabels['{$WARN_BEKILL}'] .= '<span class="warn_txt2">'._T('modpushpost_omitted').'<br /></span>'."\n";
}
}
}
function autoHookThreadReply(&$arrLabels, $post, $isReply){
$this->autoHookThreadPost($arrLabels, $post, $isReply);
}
function autoHookRegistBegin(&$name, &$email, &$sub, &$com, $upfileInfo, $accessInfo, $isReply){
if(adminAuthenticate('check')) return; // 登入權限允許標籤留存不轉換 (後端登入修改文章後推文仍有效)
if(strpos($com, $this->PUSHPOST_SEPARATOR."\r\n") !== false){ // 防止不正常的插入標籤形式
$com = str_replace($this->PUSHPOST_SEPARATOR."\r\n", "\r\n", $com);
}
}
function autoHookAdminList(&$modFunc, $post, $isres){
$modFunc .= '[<a href="'.$this->mypage.'&action=del&no='.$post['no'].'">刪推</a>]';
}
function ModulePage(){
global $PIO, $PTE, $PMS, $language;
if(!isset($_GET['no'])) die('[Error] not enough parameter.');
if(isset($_GET['action'])) {
if(adminAuthenticate('check')) {
$pushcount = ''; $puststart=0;
$post = $PIO->fetchPosts($_GET['no']);
if(!count($post)) die('[Error] Post does not exist.'); // 被推之文章不存在
extract($post[0]);
if($status != ''){
$f = $PIO->getPostStatus($status);
$pushcount = $f->value('mppCnt'); // 被推次數
}
if(($puststart=strpos($com, $this->PUSHPOST_SEPARATOR.'<br />'))===false) die('[Error] No pushpost.');
$ocom = substr($com,0,$puststart);
$pushpost = explode('<br />',substr($com,$puststart+strlen($this->PUSHPOST_SEPARATOR.'<br />')));
$com = $ocom;
if($_GET['action'] == 'del') { // list
$p_count = 1;
$com .= '<div class="pushpost">';
foreach($pushpost as $p)
$com .= '<input type="checkbox" name="'.($p_count++).'" value="delete" />'.$p.'<br />';
$com .= '</div>';
$dat = '';
head($dat);
$dat .= '<div class="bar_reply">'._T('modpushpost_deletepush').'</div>';
$dat .= '<form action="'.$this->mypage.'&action=delpush&no='.$_GET['no'].'" method="post">';
$dat .= $PTE->ParseBlock('SEARCHRESULT', array('{$NO}'=>$no, '{$SUB}'=>$sub, '{$NAME}'=>$name, '{$NOW}'=>$now, '{$COM}'=>$com, '{$CATEGORY}'=>$category, '{$NAME_TEXT}'=>_T('post_name'), '{$CATEGORY_TEXT}'=>_T('post_category')));
echo $dat, '<input type="submit" value="'._T('del_btn').'" /></form></body></html>';
return;
} elseif($_GET['action'] == 'delpush') { // delete
$delno = array();
reset($_POST);
while($item = each($_POST)){ if($item[1]=='delete' && $item[0] != 'func') array_push($delno, $item[0]); }
if(count($delno)) foreach($delno as $d) if(isset($pushpost[$d-1])) unset($pushpost[$d-1]);
$pushcount = count($pushpost);
if($pushcount) {
$f->update('mppCnt',$pushcount); // 更新推文次數
$com = $ocom.$this->PUSHPOST_SEPARATOR.'<br />'.implode('<br />',$pushpost);
} else {
$f->remove('mppCnt'); // 刪除推文次數
$com = $ocom;
}
$PIO->updatePost($_GET['no'], array('com'=>$com, 'status'=>$f->toString())); // 更新推文
$PIO->dbCommit();
echo '+OK ';
return;
} else die('[Error] unknown action.');
} else die('[Error] unauthenticated action.');
}
if(!isset($_POST['comm'])){
$post = $PIO->fetchPosts($_GET['no']);
if(!count($post)) die('[Error] Post does not exist.');
$dat = $PTE->ParseBlock('HEADER', array('{$TITLE}'=>TITLE, '{$RESTO}'=>''));
$dat .= '</head><body id="main">';
$dat .= '<form action="'.$this->mypage.'&no='.$_GET['no'].'" method="post">
'._T('modpushpost_pushpost').' <ul><li>'._T('form_name').' <input type="text" name="name" maxlength="20" /></li><li>'._T('form_comment').' <input type="text" name="comm" size="50" maxlength="50" /><input type="submit" value="'._T('form_submit_btn').'" /></li></ul>
</form>';
echo $dat, '</body></html>';
}else{
if($_SERVER['REQUEST_METHOD'] != 'POST') die(_T('regist_notpost')); // 傳送方法不正確
// 查IP
$baninfo = '';
$ip = getREMOTE_ADDR(); $host = gethostbyaddr($ip);
if(BanIPHostDNSBLCheck($ip, $host, $baninfo)) die(_T('regist_ipfiltered', $baninfo));
$name = CleanStr($_POST['name']); $comm = CleanStr($_POST['comm']); $push_emotion = CleanStr($_POST['push_emotion']); $gesture = CleanStr($_POST['gesture']);
if(strlen($name) > 30) die(_T('modpushpost_maxlength')); // 名稱太長
if(strlen($comm) > 160) die(_T('modpushpost_maxlength')); // 太多字
if(strlen($comm) == 0) die(_T('modpushpost_nocomment')); // 沒打字
$name = str_replace(array(_T('trip_pre'), _T('admin'), _T('deletor')), array(_T('trip_pre_fake'), '"'._T('admin').'"', '"'._T('deletor').'"'), $name);
$pushID = $this->_getID();
$pushtime = gmdate('y/m/d H:i', time() + intval(TIME_ZONE) * 3600);
if(preg_match('/(.*?)[##](.*)/u', $name, $regs)){
$cap = strtr($regs[2], array('&'=>'&'));
$salt = strtr(preg_replace('/[^\.-z]/', '.', substr($cap.'H.', 1, 2)), ':;<=>?@[\\]^_`', 'ABCDEFGabcdef');
$name = $regs[1]._T('trip_pre').substr(crypt($cap, $salt), -10);
}
if(!$name || preg_match("/^[ | |]*$/", $name)){
if(ALLOW_NONAME) $name = DEFAULT_NONAME;
else die(_T('regist_withoutname')); // 不接受匿名
}
if(ALLOW_NONAME==2){ // 強制砍名
$name = preg_match('/(\\'._T('trip_pre').'.{10})/', $name, $matches) ? $matches[1].':' : DEFAULT_NONAME.':';
}else{
$name .= ':';
}
$pushpost = "{$name} {$push_emotion}{$gesture}{$comm} ({$pushID} {$pushtime})"; // 推文主體
$post = $PIO->fetchPosts($_GET['no']);
if(!count($post)) die('[Error] Post does not exist.'); // 被推之文章不存在
$parentNo = $post[0]['resto'] ? $post[0]['resto'] : $post[0]['no'];
$threads = array_flip($PIO->fetchThreadList());
$threadPage = floor($threads[$parentNo] / PAGE_DEF);
$p = ($parentNo==$post[0]['no']) ? $post : $PIO->fetchPosts($parentNo); // 取出首篇
$flgh = $PIO->getPostStatus($p[0]['status']);
if($flgh->exists('TS')) die('[Error] '._T('regist_threadlocked')); // 首篇禁止回應/同時表示禁止推文
$post[0]['com'] .= ((strpos($post[0]['com'], $this->PUSHPOST_SEPARATOR.'<br />')===false) ? '<br />'.$this->PUSHPOST_SEPARATOR : '').'<br /> '.$pushpost;
$flgh2 = $PIO->getPostStatus($post[0]['status']);
$flgh2->plus('mppCnt'); // 推文次數+1
$PIO->updatePost($_GET['no'], array('com'=>$post[0]['com'], 'status'=>$flgh2->toString())); // 更新推文
$PIO->dbCommit();
// logcat
$PMS->callCHP('mod_audit_logcat',
array(sprintf('[%s] No.%d %s (%s)',
__CLASS__,
$_GET['no'],
str_cut($comm, 50),
$pushID)
)
);
if(STATIC_HTML_UNTIL == -1 || $threadPage <= STATIC_HTML_UNTIL) updatelog(0, $threadPage, true); // 僅更新討論串出現那頁
deleteCache(array($parentNo)); // 刪除討論串舊快取
if(isset($_POST['ajaxmode'])){
echo '+OK ', $pushpost;
}else{
header('HTTP/1.1 302 Moved Temporarily');
header('Location: '.fullURL().PHP_SELF2.'?'.time());
}
}
}
function _loadLanguage(){
global $language;
if(PIXMICAT_LANGUAGE != 'zh_TW' && PIXMICAT_LANGUAGE != 'ja_JP' && PIXMICAT_LANGUAGE != 'en_US') $lang = 'en_US';
else $lang = PIXMICAT_LANGUAGE;
if($lang=='zh_TW'){
$language['modpushpost_nocomment'] = '請輸入內文';
$language['modpushpost_pushpost'] = '[推文]';
$language['modpushpost_pushbutton'] = '推';
$language['modpushpost_maxlength'] = '你話太多了';
$language['modpushpost_omitted'] = '有部分推文被省略。要閱讀全部推文請按下回應連結。';
$language['modpushpost_deletepush'] = '刪除推文模式';
}elseif($lang=='ja_JP'){
$language['modpushpost_nocomment'] = '何か書いて下さい';
$language['modpushpost_pushpost'] = '[推文]';
$language['modpushpost_pushbutton'] = '推';
$language['modpushpost_maxlength'] = 'コメントが長すぎます';
$language['modpushpost_omitted'] = '推文省略。全て読むには返信ボタンを押してください。';
$language['modpushpost_deletepush'] = '削除推文モード';
}elseif($lang=='en_US'){
$language['modpushpost_nocomment'] = 'Please type your comment.';
$language['modpushpost_pushpost'] = '[Push this post]';
$language['modpushpost_pushbutton'] = 'PUSH';
$language['modpushpost_maxlength'] = 'You typed too many words';
$language['modpushpost_omitted'] = 'Some pushs omitted. Click Reply to view.';
$language['modpushpost_deletepush'] = 'Delete Push Post Mode';
}
}
}
?>