From 9fca3e4929c71bbadcb921ad73f99f6635737572 Mon Sep 17 00:00:00 2001 From: zero Date: Thu, 22 Jul 2010 10:24:08 +0000 Subject: [PATCH] =?UTF-8?q?=EB=AA=A8=EB=B0=94=EC=9D=BC=20=EC=A0=84?= =?UTF-8?q?=EC=9A=A9=20=EC=BD=98=ED=85=90=ED=8A=B8=20=EC=9C=84=EC=A0=AF?= =?UTF-8?q?=EC=9D=B8=20mcontent=20=EC=9C=84=EC=A0=AF=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit git-svn-id: http://xe-core.googlecode.com/svn/sandbox@7613 201d5d3c-b55e-5fd7-737f-ddc643e51545 --- widgets/mcontent/conf/info.xml | 490 ++++++++++++ widgets/mcontent/mcontent.class.php | 748 ++++++++++++++++++ widgets/mcontent/skins/default/content.html | 11 + .../mcontent/skins/default/image_title.html | 52 ++ .../skins/default/image_title_content.html | 55 ++ widgets/mcontent/skins/default/mcontent.css | 14 + widgets/mcontent/skins/default/normal.html | 46 ++ widgets/mcontent/skins/default/skin.xml | 15 + 8 files changed, 1431 insertions(+) create mode 100644 widgets/mcontent/conf/info.xml create mode 100644 widgets/mcontent/mcontent.class.php create mode 100644 widgets/mcontent/skins/default/content.html create mode 100644 widgets/mcontent/skins/default/image_title.html create mode 100644 widgets/mcontent/skins/default/image_title_content.html create mode 100644 widgets/mcontent/skins/default/mcontent.css create mode 100644 widgets/mcontent/skins/default/normal.html create mode 100644 widgets/mcontent/skins/default/skin.xml diff --git a/widgets/mcontent/conf/info.xml b/widgets/mcontent/conf/info.xml new file mode 100644 index 000000000..0642daeb1 --- /dev/null +++ b/widgets/mcontent/conf/info.xml @@ -0,0 +1,490 @@ + + + 모바일 콘텐츠 위젯 + Mobile Content Widget + Nội dung + 全局内容控件 + 內容 + コンテンツウィジェット + 게시판, 코멘트, 첨부파일 등 Content를 출력하는 위젯입니다. + This widget displays Content such as articles, comments, or attached files. + Widget này sẽ hiển thị những bài viết, bình luận, File đính kèm. + 以各种样式显示主题,评论,附件等站内内容的全局控件。 + 這個 Widget 可輸出討論板、評論,附加檔案等內容。 + 掲示板の書き込み、コメント、添付ファイルなどコンテンツを出力するウィジェットです。 + 0.1 + 2009-03-16 + + + sol + sol + sol + sol + sol + sol + sol + sol + + + + + select + 추출대상 + Target + Khu vực + 对象 + 目標 + 抽出対象 + + document + 게시물 + Article + Bài viết + 主题 + 主題 + 書き込み + + + comment + 댓글 + Comment + Bình luận + 评论 + 評論 + コメント + + + image + 첨부이미지 + Attached Image + Hình đính kèm + 图片 + 圖片 + 添付イメージ + + + trackback + 트랙백 + Trackback + Liên kết Web + 引用 + 引用 + トラックバック + + + rss + 피드 (RSS/ATOM) + Feed (RSS/ATOM) + Feed (RSS/ATOM) + Feed (RSS/ATOM) + Feed (RSS/ATOM) + Feed (RSS/ATOM) + + + + select + 내용형태 + Content Type + Kiểu hiển thị + 内容样式 + 內容樣式 + 内容のタイプ + + normal + 제목 + Title + Tiêu đề + 标题 + 標題 + タイトル + + + image_title + 이미지+제목 + Image+Title + Hình ảnh+Tiêu đề + 图片+标题 + 圖片+標題 + 画像+タイトル + + + image_title_content + 이미지+제목+내용 + Image+Title+Content + Hình ảnh+Tiêu đề+Nội dung + 图片+标题+内容 + 圖片+標題+內容 + 画像+タイトル+内容 + + + + select-multi-order + 표시항목 + Display Target & Order + Hiển thị khu vực và thứ tự + 显示项及顺序 + 顯示項目順序 + 表示項目、および順番 + + title + 제목 + Title + Tiêu đề + 标题 + 標題 + タイトル + + + thumbnail + 섬네일 + Thumbnail + Hình nhỏ + 缩略图 + 縮圖 + サムネール + + + regdate + 등록일 + Regdate + Ngày gửi + 发布日期 + 發表日期 + 登録日 + + + nickname + 글쓴이 + Author + Người gửi + 发布者 + 作者 + 登録者 + + + content + 내용 + Content + Nội dung + 内容 + 內容 + 内容 + + + + select + 게시판 이름 표시 + Display Article Name + Tên bài viết + 显示版面名称 + 討論板名稱 + 掲示板名の表示 + + Y + 출력 + Display + Hiển thị + 显示 + 顯示 + 表示 + + + N + 출력하지 않음 + No Display + Không hiển thị + 不显示 + 不顯示 + 表示しない + + + + select + 댓글수 표시 + Comment Count + Số bình luận + 显示评论数 + 評論數 + コメント数 + + Y + 출력 + Display + Hiển thị + 显示 + 顯示 + 表示 + + + N + 출력하지 않음 + No Display + Không hiển thị + 不显示 + 不顯示 + 表示しない + + + + select + 엮인글수 표시 + Trackback + Liên kết Web + 显示引用数 + 引用數 + トラックバック数の表示 + + Y + 출력 + Display + Hiển thị + 显示 + 顯示 + 表示 + + + N + 출력하지 않음 + No Display + Không hiển thị + 不显示 + 不顯示 + 表示しない + + + + select + 분류 표시 + Category + Thể loại + 显示分类 + 分類 + カテゴリ表示 + + Y + 출력 + Display + Hiển thị + 显示 + 顯示 + 表示 + + + N + 출력하지 않음 + No Display + Không hiển thị + 不显示 + 不顯示 + 表示しない + + + + select + 정렬 대상 + 排序对象 + ソート対象 + Target to be sorted + Phân loại + Objetivo para ser ordenados + Назначение для сортировки + 排列順序 + 등록된 순서 또는 변경된 순서로 정렬을 할 수 있습니다. + 可以按照指定的顺序进行排序。 + 登録順、または変更順にソートします。 + The list of newewst articles may be sorted by submitted order or modified order. + Danh sách bài viết mới nhất có thể được được phân loại bởi danh sách đã gửi hay đã sửa. + La lista de los documentos recientes pueden ser ordenados en el orden del agregado o en el de modificados. + Список последних статей может быть отсортирован по дате размещения или изменения. + 按照指定的順序進行排列。 + + list_order + 최신 등록순 + 最新发表顺 + 最新登録順 + Newest Submitted Order + Bài mới gửi + Orden de agregados recientemente + Порядок размещенных статей + 最新發表 + + + update_order + 최근 변경순 + 最新修改顺 + 最新変更順 + Newest Modified Order + Bài mới sửa + Orden de modificados recientemente + Порядок измененных статей + 最新修改 + + + + select + 정렬 방법 + 排序方式 + ソートタイプ + Sorting Type + Kiểu sắp xếp + Tipo de ordenamiento + Тип сортировки + 排列方式 + 정렬대상을 내림차순 또는 올림차순으로 정렬할 수 있습니다. + 对其排序对象可进行升序/降序方式排序。 + ソートタイプを、降順、または昇順に設定します。 + You can sort target articles by ascending or descending order. + Bạn có thể phân loại bài viết theo kiểu tăng hay giảm. + Usted puede ordenar los documentos en orden acendente o en orden descendente. + Вы можете сортировать статьи в порядке возрастания или убывания. + 可選擇升冪/降冪的方式對所選目標進行排列。 + + desc + 내림차순 + 降序 + 降順 + Descending order + Giảm dần + Orden Descendente + По убыванию + 降冪 + + + asc + 올림차순 + 升序 + 昇順 + Ascending order + Tăng dần + Orden Acendente + По возрастанию + 升冪 + + + + text + 목록수 + 目录数 + リスト数 + The number of list + Số bài hiển thị + Número de la lista + Число списка + 目錄數 + 출력될 목록의 수를 정하실 수 있습니다. (기본 5개) + 可设置要显示的目录数。 (默认为5个) + 出力するリスト数を指定します(デフォルト5個)。 + You can set the number of articles to be displayed. (default is 5) + Bạn có thể đặt số bài hiển thị. (Mặc định là 5) + Usted puede definir el número de los documentos a mostrar. (predefinido: 5) + Вы можете выбрать число списка статей для отображения. (стандарт: 5) + 設置要顯示的目錄數。(預設是5個) + + + text + 제목 글자수 + タイトルの文字数 + 标题字数 + Length of Subject + Độ dài tiêu đề + Número de letras del título + Длина темы + 標題字數 + 제목 글자수를 지정할 수 있습니다. (0또는 비워주시면 자르지 않습니다) + タイトルの文字数を設定します(「0」または空欄の場合は、文字数を制限しません)。 + 可以设置标题的字数。(0或留空为不限) + Length of Subject can be assigned. (0 or blank value will not restrict the length) + Có thể tăng thêm chiều dài của tiêu đề. (Đặt là 0 hoặc để trống nếu không giới hạn tiêu đề). + El largo del título puede ser asignado. (valor 0 o en blanco no restringe el largo) + Длина темы может быть присвоена. (0 или пустое значение не будут ограничивать длину) + 可設置標題的字數。(0或留白為不限制) + + + text + 내용 글자수 + Content Length + Độ dài nội dung + 内容の文字数 + 内容摘要字数 + 內容字數 + + + module_srl_list + 대상 모듈 + 对象模块 + モジュール + Target Module + Chèn Module + Módulo Objetivo + Модуль назначения + 目標模組 + 선택하신 모듈에 등록된 글을 대상으로 합니다. + チェックされたモジュールに登録されたコンテンツ(書き込み)を対象とします。 + 将把所选模块当中的主题作为对象。 + The target articles to be sorted will be the ones submitted in the selected module. + Những bài viết sẽ được phân loại và hiển thị khi có bài mới gửi trong những Module đã chọn. + El objetivo de los documentos agregados serán los del módulo selccionado. + Статьи назначения для сортировки будут теми, что были размещены в выбранном модуле. + 把所選擇的模組作為目標。 + + + text + 피드(RSS/ATOM) 주소 + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + 피드 주소는 타입이 지원하는 문서 형식일 경우에만 등록 됩니다. + (지원 문서 형식 : RSS 2.0, RSS 1.0, ATOM 1.0) + 请输入RSS格式地址。 + (支持文档格式 : RSS 2.0, RSS 1.0, ATOM 1.0) + Chỉ hỗ trợ những định dạng được dăng kí. (RSS 2.0 RSS 1.0 ATOM 1.0). + RSS URLはタイプが RSSの時だけ登録出来ます。 + (サポートする文書形式 : RSS 2.0, RSS 1.0, ATOM 1.0) + 請輸入 RSS 位址。 + (支援檔案格式 : RSS 2.0, RSS 1.0, ATOM 1.0) + + + text + 피드(RSS/ATOM) 주소 + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + + + text + 피드(RSS/ATOM) 주소 + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + + + text + 피드(RSS/ATOM) 주소 + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + + + text + 피드(RSS/ATOM) 주소 + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + Feed(RSS/ATOM) URL + + + diff --git a/widgets/mcontent/mcontent.class.php b/widgets/mcontent/mcontent.class.php new file mode 100644 index 000000000..6ad8e4f5f --- /dev/null +++ b/widgets/mcontent/mcontent.class.php @@ -0,0 +1,748 @@ +order_target, array('list_order','update_order'))) $args->order_target = 'list_order'; + + // 정렬 순서 + if(!in_array($args->order_type, array('asc','desc'))) $args->order_type = 'asc'; + + // 출력된 목록 수 + $args->list_count = (int)$args->list_count; + if(!$args->list_count) $args->list_count = 5; + + // 제목 길이 자르기 + if(!$args->subject_cut_size) $args->subject_cut_size = 0; + + // 내용 길이 자르기 + if(!$args->content_cut_size) $args->content_cut_size = 100; + + // 보기 옵션 + $args->option_view_arr = explode(',',$args->option_view); + + // markup 옵션 + if(!$args->markup_type) $args->markup_type = 'list'; + + // 내부적으로 쓰이는 변수 설정 + $oModuleModel = &getModel('module'); + $module_srls = $args->modules_info = $args->module_srls_info = $args->mid_lists = array(); + $site_module_info = Context::get('site_module_info'); + + // rss 인 경우 URL정리 + if($args->content_type == 'rss'){ + $args->rss_urls = array(); + $rss_urls = array_unique(array($args->rss_url0,$args->rss_url1,$args->rss_url2,$args->rss_url3,$args->rss_url4)); + for($i=0,$c=count($rss_urls);$i<$c;$i++) { + if($rss_urls[$i]) $args->rss_urls[] = $rss_urls[$i]; + } + + // rss가 아닌 XE모듈일 경우 모듈 번호 정리 후 모듈 정보 구함 + } else { + + // 대상 모듈이 선택되어 있지 않으면 해당 사이트의 전체 모듈을 대상으로 함 + if(!$args->module_srls){ + unset($obj); + $obj->site_srl = (int)$site_module_info->site_srl; + $output = executeQueryArray('widgets.content.getMids', $obj); + if($output->data) { + foreach($output->data as $key => $val) { + $args->modules_info[$val->mid] = $val; + $args->module_srls_info[$val->module_srl] = $val; + $args->mid_lists[$val->module_srl] = $val->mid; + $module_srls[] = $val->module_srl; + } + } + + $args->modules_info = $oModuleModel->getMidList($obj); + // 대상 모듈이 선택되어 있으면 해당 모듈만 추출 + } else { + $obj->module_srls = $args->module_srls; + $output = executeQueryArray('widgets.content.getMids', $obj); + if($output->data) { + foreach($output->data as $key => $val) { + $args->modules_info[$val->mid] = $val; + $args->module_srls_info[$val->module_srl] = $val; + $module_srls[] = $val->module_srl; + } + $idx = explode(',',$args->module_srls); + for($i=0,$c=count($idx);$i<$c;$i++) { + $srl = $idx[$i]; + if(!$args->module_srls_info[$srl]) continue; + $args->mid_lists[$srl] = $args->module_srls_info[$srl]->mid; + } + } + } + + // 아무런 모듈도 검색되지 않았다면 종료 + if(!count($args->modules_info)) return Context::get('msg_not_founded'); + $args->module_srl = implode(',',$module_srls); + } + + /** + * 컨텐츠 추출, 게시글/댓글/엮인글/RSS등 다양한 요소가 있어서 각 method를 따로 만듬 + **/ + // tab 형태 + if($args->tab_type == 'none' || $args->tab_type == '') { + switch($args->content_type){ + case 'comment': + $content_items = $this->_getCommentItems($args); + break; + case 'image': + $content_items = $this->_getImageItems($args); + break; + case 'rss': + $content_items = $this->getRssItems($args); + break; + case 'trackback': + $content_items = $this->_getTrackbackItems($args); + break; + default: + $content_items = $this->_getDocumentItems($args); + break; + } + // tab 형태가 아닐 경우 + } else { + $content_items = array(); + + switch($args->content_type){ + case 'comment': + foreach($args->mid_lists as $module_srl => $mid){ + $args->module_srl = $module_srl; + $content_items[$module_srl] = $this->_getCommentItems($args); + } + break; + case 'image': + foreach($args->mid_lists as $module_srl => $mid){ + $args->module_srl = $module_srl; + $content_items[$module_srl] = $this->_getImageItems($args); + } + break; + case 'rss': + $content_items = $this->getRssItems($args); + break; + case 'trackback': + foreach($args->mid_lists as $module_srl => $mid){ + $args->module_srl = $module_srl; + $content_items[$module_srl] = $this->_getTrackbackItems($args); + } + break; + default: + foreach($args->mid_lists as $module_srl => $mid){ + $args->module_srl = $module_srl; + $content_items[$module_srl] = $this->_getDocumentItems($args); + } + break; + } + } + + $output = $this->_compile($args,$content_items); + return $output; + } + + /** + * @brief 댓글 목록을 추출하여 contentItem으로 return + **/ + function _getCommentItems($args) { + // CommentModel::getCommentList()를 이용하기 위한 변수 정리 + $obj->module_srl = $args->module_srl; + $obj->sort_index = $args->order_target; + $obj->list_count = $args->list_count; + + // comment 모듈의 model 객체를 받아서 getCommentList() method를 실행 + $oCommentModel = &getModel('comment'); + $output = $oCommentModel->getNewestCommentList($obj); + + $content_items = array(); + + if(!count($output)) return; + + foreach($output as $key => $oComment) { + $attribute = $oComment->getObjectVars(); + $title = $oComment->getSummary($args->content_cut_size); + $thumbnail = $oComment->getThumbnail(); + $url = sprintf("%s#comment_%s",getUrl('','document_srl',$oComment->get('document_srl')),$oComment->get('comment_srl')); + + $attribute->mid = $args->mid_lists[$attribute->module_srl]; + $browser_title = $args->module_srls_info[$attribute->module_srl]->browser_title; + $domain = $args->module_srls_info[$attribute->module_srl]->domain; + + $content_item = new contentItem($browser_title); + $content_item->adds($attribute); + $content_item->setTitle($title); + $content_item->setThumbnail($thumbnail); + $content_item->setLink($url); + $content_item->setDomain($domain); + $content_item->add('mid', $args->mid_lists[$attribute->module_srl]); + $content_items[] = $content_item; + } + return $content_items; + } + + function _getDocumentItems($args){ + // document 모듈의 model 객체를 받아서 결과를 객체화 시킴 + $oDocumentModel = &getModel('document'); + + // 분류 구함 + $obj->module_srl = $args->module_srl; + $output = executeQueryArray('widgets.content.getCategories',$obj); + if($output->toBool() && $output->data) { + foreach($output->data as $key => $val) { + $category_lists[$val->module_srl][$val->category_srl] = $val; + } + } + + // 글 목록을 구함 + $obj->module_srl = $args->module_srl; + $obj->sort_index = $args->order_target; + $obj->order_type = $args->order_type=="desc"?"asc":"desc"; + $obj->list_count = $args->list_count; + $output = executeQueryArray('widgets.content.getNewestDocuments', $obj); + if(!$output->toBool() || !$output->data) return; + + // 결과가 있으면 각 문서 객체화를 시킴 + $content_items = array(); + $first_thumbnail_idx = -1; + if(count($output->data)) { + foreach($output->data as $key => $attribute) { + $oDocument = new documentItem(); + $oDocument->setAttribute($attribute, false); + $GLOBALS['XE_DOCUMENT_LIST'][$oDocument->document_srl] = $oDocument; + $document_srls[] = $oDocument->document_srl; + } + $oDocumentModel->setToAllDocumentExtraVars(); + + for($i=0,$c=count($document_srls);$i<$c;$i++) { + $oDocument = $GLOBALS['XE_DOCUMENT_LIST'][$document_srls[$i]]; + $document_srl = $oDocument->document_srl; + $module_srl = $oDocument->get('module_srl'); + $category_srl = $oDocument->get('category_srl'); + $thumbnail = $oDocument->getThumbnail(); + $content_item = new contentItem( $args->module_srls_info[$module_srl]->browser_title ); + $content_item->adds($oDocument->getObjectVars()); + $content_item->setTitle($oDocument->getTitle()); + $content_item->setCategory( $category_lists[$module_srl][$category_srl]->title ); + $content_item->setDomain( $args->module_srls_info[$module_srl]->domain ); + $content_item->setContent($oDocument->getSummary($args->content_cut_size)); + $content_item->setLink( getSiteUrl($domain,'','document_srl',$document_srl) ); + $content_item->setThumbnail($thumbnail); + $content_item->add('mid', $args->mid_lists[$module_srl]); + if($first_thumbnail_idx==-1 && $thumbnail) $first_thumbnail_idx = $i; + $content_items[] = $content_item; + } + + $content_items[0]->setFirstThumbnailIdx($first_thumbnail_idx); + } + return $content_items; + } + + function _getImageItems($args) { + $oDocumentModel = &getModel('document'); + + $obj->module_srls = $obj->module_srl = $args->module_srl; + $obj->direct_download = 'Y'; + $obj->isvalid = 'Y'; + + // 분류 구함 + $output = executeQueryArray('widgets.content.getCategories',$obj); + if($output->toBool() && $output->data) { + foreach($output->data as $key => $val) { + $category_lists[$val->module_srl][$val->category_srl] = $val; + } + } + + // 정해진 모듈에서 문서별 파일 목록을 구함 + $obj->list_count = $args->list_count; + $files_output = executeQueryArray("file.getOneFileInDocument", $obj); + $files_count = count($files_output->data); + if(!$files_count) return; + + $content_items = array(); + + for($i=0;$i<$files_count;$i++) $document_srl_list[] = $files_output->data[$i]->document_srl; + + $tmp_document_list = $oDocumentModel->getDocuments($document_srl_list); + + if(!count($tmp_document_list)) return; + + foreach($tmp_document_list as $oDocument){ + $attribute = $oDocument->getObjectVars(); + $browser_title = $args->module_srls_info[$attribute->module_srl]->browser_title; + $domain = $args->module_srls_info[$attribute->module_srl]->domain; + $category = $category_lists[$attribute->module_srl]->text; + $content = $oDocument->getSummary($args->content_cut_size); + $url = sprintf("%s#%s",$oDocument->getPermanentUrl() ,$oDocument->getCommentCount()); + $thumbnail = $oDocument->getThumbnail(); + $content_item = new contentItem($browser_title); + $content_item->adds($attribute); + $content_item->setCategory($category); + $content_item->setContent($content); + $content_item->setLink($url); + $content_item->setThumbnail($thumbnail); + $content_item->setExtraImages($extra_images); + $content_item->setDomain($domain); + $content_item->add('mid', $args->mid_lists[$attribute->module_srl]); + $content_items[] = $content_item; + } + + return $content_items; + } + + function getRssItems($args){ + + $content_items = array(); + $args->mid_lists = array(); + + foreach($args->rss_urls as $key => $rss){ + $args->rss_url = $rss; + $content_item = $this->_getRssItems($args); + if(count($content_item) > 0){ + $browser_title = $content_item[0]->getBrowserTitle(); + $args->mid_lists[] = $browser_title; + $content_items[] = $content_item; + } + } + + if($args->tab_type == 'none' || $args->tab_type == ''){ + $items = array(); + foreach($content_items as $key => $val){ + foreach($val as $k => $v){ + $date = $v->get('regdate'); + $i=0; + while(array_key_exists(sprintf('%s%02d',$date,$i), $items)) $i++; + $items[sprintf('%s%02d',$date,$i)] = $v; + } + } + if($args->order_type =='asc') ksort($items); + else krsort($items); + $content_items = array_slice(array_values($items),0,$args->list_count); + } return $content_items; + } + + function _getRssBody($value) { + if(!$value || is_string($value)) return $value; + if(is_object($value)) $value = get_object_vars($value); + $body = null; + if(!count($value)) return; + foreach($value as $key => $val) { + if($key == 'body') { + $body = $val; + continue; + } + if(is_object($val)||is_array($val)) $body = $this->_getRssBody($val); + if($body !== null) return $body; + } + return $body; + } + + function _getSummary($content, $str_size = 50) + { + $content = preg_replace('!([\s]*)+!is', ' ', $content); + + //

, , 등의 태그를 공백 문자로 치환 + $content = str_replace(array('

', '', ''), ' ', $content); + + // 태그 제거 + $content = preg_replace('!<([^>]*?)>!is','', $content); + + // < , > , " 를 치환 + $content = str_replace(array('<','>','"',' '), array('<','>','"',' '), $content); + + // 연속된 공백문자 삭제 + $content = preg_replace('/ ( +)/is', ' ', $content); + + // 문자열을 자름 + $content = trim(cut_str($content, $str_size, $tail)); + + // >, <, "를 다시 복구 + $content = str_replace(array('<','>','"'),array('<','>','"'), $content); + + // 영문이 연결될 경우 개행이 안 되는 문제를 해결 + $content = preg_replace('/([a-z0-9\+:\/\.\~,\|\!\@\#\$\%\^\&\*\(\)\_]){20}/is',"$0-",$content); + return $content; + } + + + /** + * @brief rss 주소로 부터 내용을 받아오는 함수 + * tistory 의 경우 원본 주소가 location 헤더를 뿜는다. (내용은 없음)이를 해결하기 위한 수정 - rss_reader 위젯과 방식 동일 + **/ + function requestFeedContents($rss_url) { + $rss_url = str_replace('&','&',Context::convertEncodingStr($rss_url)); + return FileHandler::getRemoteResource($rss_url, null, 3, 'GET', 'application/xml'); + } + + function _getRssItems($args){ + // 날짜 형태 + $DATE_FORMAT = $args->date_format ? $args->date_format : "Y-m-d H:i:s"; + + $buff = $this->requestFeedContents($args->rss_url); + + $encoding = preg_match("/<\?xml.*encoding=\"(.+)\".*\?>/i", $buff, $matches); + if($encoding && !preg_match("/UTF-8/i", $matches[1])) $buff = Context::convertEncodingStr($buff); + + $buff = preg_replace("/<\?xml.*\?>/i", "", $buff); + + $oXmlParser = new XmlParser(); + $xml_doc = $oXmlParser->parse($buff); + if($xml_doc->rss) { + $rss->title = $xml_doc->rss->channel->title->body; + $rss->link = $xml_doc->rss->channel->link->body; + + $items = $xml_doc->rss->channel->item; + + if(!$items) return; + if($items && !is_array($items)) $items = array($items); + + $content_items = array(); + + foreach ($items as $key => $value) { + if($key >= $args->list_count) break; + unset($item); + + foreach($value as $key2 => $value2) { + if(is_array($value2)) $value2 = array_shift($value2); + $item->{$key2} = $this->_getRssBody($value2); + } + + $content_item = new contentItem($rss->title); + $content_item->setContentsLink($rss->link); + $content_item->setTitle($item->title); + $content_item->setNickName(max($item->author,$item->{'dc:creator'})); + //$content_item->setCategory($item->category); + $item->description = preg_replace('!pubdate,$item->pubDate,$item->{'dc:date'}))); + $content_item->setRegdate($date); + + $content_items[] = $content_item; + } + } elseif($xml_doc->{'rdf:rdf'}) { + // rss1.0 지원 (Xml이 대소문자를 구분해야 하는데 XE의 XML파서가 전부 소문자로 바꾸는 바람에 생긴 case) by misol + $rss->title = $xml_doc->{'rdf:rdf'}->channel->title->body; + $rss->link = $xml_doc->{'rdf:rdf'}->channel->link->body; + + $items = $xml_doc->{'rdf:rdf'}->item; + + if(!$items) return; + if($items && !is_array($items)) $items = array($items); + + $content_items = array(); + + foreach ($items as $key => $value) { + if($key >= $args->list_count) break; + unset($item); + + foreach($value as $key2 => $value2) { + if(is_array($value2)) $value2 = array_shift($value2); + $item->{$key2} = $this->_getRssBody($value2); + } + + $content_item = new contentItem($rss->title); + $content_item->setContentsLink($rss->link); + $content_item->setTitle($item->title); + $content_item->setNickName(max($item->author,$item->{'dc:creator'})); + //$content_item->setCategory($item->category); + $item->description = preg_replace('!pubdate,$item->pubDate,$item->{'dc:date'}))); + $content_item->setRegdate($date); + + $content_items[] = $content_item; + } + } elseif($xml_doc->feed && $xml_doc->feed->attrs->xmlns == 'http://www.w3.org/2005/Atom') { + // Atom 1.0 spec 지원 by misol + $rss->title = $xml_doc->feed->title->body; + $links = $xml_doc->feed->link; + if(is_array($links)) { + foreach ($links as $value) { + if($value->attrs->rel == 'alternate') { + $rss->link = $value->attrs->href; + break; + } + } + } + elseif($links->attrs->rel == 'alternate') $rss->link = $links->attrs->href; + + $items = $xml_doc->feed->entry; + + if(!$items) return; + if($items && !is_array($items)) $items = array($items); + + $content_items = array(); + + foreach ($items as $key => $value) { + if($key >= $args->list_count) break; + unset($item); + + foreach($value as $key2 => $value2) { + if(is_array($value2)) $value2 = array_shift($value2); + $item->{$key2} = $this->_getRssBody($value2); + } + + $content_item = new contentItem($rss->title); + $links = $value->link; + if(is_array($links)) { + foreach ($links as $val) { + if($val->attrs->rel == 'alternate') { + $item->link = $val->attrs->href; + break; + } + } + } + elseif($links->attrs->rel == 'alternate') $item->link = $links->attrs->href; + + $content_item->setContentsLink($rss->link); + if($item->title) { + if(!preg_match("/html/i", $value->title->attrs->type)) $item->title = $value->title->body; + } + $content_item->setTitle($item->title); + $content_item->setNickName(max($item->author,$item->{'dc:creator'})); + $content_item->setAuthorSite($value->author->uri->body); + //$content_item->setCategory($item->category); + $item->description = preg_replace('!published,$item->updated,$item->{'dc:date'}))); + $content_item->setRegdate($date); + + $content_items[] = $content_item; + } + } + return $content_items; + } + + function _getTrackbackItems($args){ + // 분류 구함 + $output = executeQueryArray('widgets.content.getCategories',$obj); + if($output->toBool() && $output->data) { + foreach($output->data as $key => $val) { + $category_lists[$val->module_srl][$val->category_srl] = $val; + } + } + + $obj->module_srl = $args->module_srl; + $obj->sort_index = $args->order_target; + $obj->list_count = $args->list_count; + + // trackback 모듈의 model 객체를 받아서 getTrackbackList() method를 실행 + $oTrackbackModel = &getModel('trackback'); + $output = $oTrackbackModel->getNewestTrackbackList($obj); + + // 오류가 생기면 그냥 무시 + if(!$output->toBool() || !$output->data) return; + + // 결과가 있으면 각 문서 객체화를 시킴 + $content_items = array(); + foreach($output->data as $key => $item) { + $domain = $args->module_srls_info[$item->module_srl]->domain; + $category = $category_lists[$item->module_srl]->text; + $url = getSiteUrl($domain,'','document_srl',$item->document_srl); + $browser_title = $args->module_srls_info[$item->module_srl]->browser_title; + + $content_item = new contentItem($browser_title); + $content_item->adds($item); + $content_item->setTitle($item->title); + $content_item->setCategory($category); + $content_item->setNickName($item->blog_name); + $content_item->setContent($item->excerpt); ///<< + $content_item->setDomain($domain); ///<< + $content_item->setLink($url); + $content_item->add('mid', $args->mid_lists[$item->module_srl]); + $content_item->setRegdate($item->regdate); + $content_items[] = $content_item; + } + return $content_items; + } + + function _compile($args,$content_items){ + $oTemplate = &TemplateHandler::getInstance(); + + // 위젯에 넘기기 위한 변수 설정 + $widget_info->modules_info = $args->modules_info; + $widget_info->option_view_arr = $args->option_view_arr; + $widget_info->list_count = $args->list_count; + $widget_info->subject_cut_size = $args->subject_cut_size; + $widget_info->content_cut_size = $args->content_cut_size; + + $widget_info->thumbnail_type = $args->thumbnail_type; + $widget_info->thumbnail_width = $args->thumbnail_width; + $widget_info->thumbnail_height = $args->thumbnail_height; + $widget_info->mid_lists = $args->mid_lists; + + $widget_info->show_browser_title = $args->show_browser_title; + $widget_info->show_category = $args->show_category; + $widget_info->show_comment_count = $args->show_comment_count; + $widget_info->show_trackback_count = $args->show_trackback_count; + $widget_info->show_icon = $args->show_icon; + + $widget_info->list_type = $args->list_type; + $widget_info->tab_type = $args->tab_type; + + $widget_info->markup_type = $args->markup_type; + $widget_info->content_items = $content_items; + + unset($args->option_view_arr); + unset($args->modules_info); + + Context::set('colorset', $args->colorset); + Context::set('widget_info', $widget_info); + + $tpl_path = sprintf('%sskins/%s', $this->widget_path, $args->skin); + return $oTemplate->compile($tpl_path, "content"); + } + } + + class contentItem extends Object { + + var $browser_title = null; + var $has_first_thumbnail_idx = false; + var $first_thumbnail_idx = null; + var $contents_link = null; + var $domain = null; + + function contentItem($browser_title=''){ + $this->browser_title = $browser_title; + } + function setContentsLink($link){ + $this->contents_link = $link; + } + function setFirstThumbnailIdx($first_thumbnail_idx){ + if(is_null($this->first_thumbnail) && $first_thumbnail_idx>-1) { + $this->has_first_thumbnail_idx = true; + $this->first_thumbnail_idx= $first_thumbnail_idx; + } + } + function setExtraImages($extra_images){ + $this->add('extra_images',$extra_images); + } + function setDomain($domain) { + static $default_domain = null; + if(!$domain) { + if(is_null($default_domain)) $default_domain = Context::getDefaultUrl(); + $domain = $default_domain; + } + $this->domain = $domain; + } + function setLink($url){ + $this->add('url',$url); + } + function setTitle($title){ + $this->add('title',$title); + } + + function setThumbnail($thumbnail){ + $this->add('thumbnail',$thumbnail); + } + function setContent($content){ + $this->add('content',$content); + } + function setRegdate($regdate){ + $this->add('regdate',$regdate); + } + function setNickName($nick_name){ + $this->add('nick_name',$nick_name); + } + + // 글 작성자의 홈페이지 주소를 저장 by misol + function setAuthorSite($site_url){ + $this->add('author_site',$site_url); + } + function setCategory($category){ + $this->add('category',$category); + } + function getBrowserTitle(){ + return $this->browser_title; + } + function getDomain() { + return $this->domain; + } + function getContentsLink(){ + return $this->contents_link; + } + + function getFirstThumbnailIdx(){ + return $this->first_thumbnail_idx; + } + + function getLink(){ + return $this->get('url'); + } + function getModuleSrl(){ + return $this->get('module_srl'); + } + function getTitle($cut_size = 0, $tail='...'){ + $title = strip_tags($this->get('title')); + + if($cut_size) $title = cut_str($title, $cut_size, $tail); + + $attrs = array(); + if($this->get('title_bold') == 'Y') $attrs[] = 'font-weight:bold'; + if($this->get('title_color') && $this->get('title_color') != 'N') $attrs[] = 'color:#'.$this->get('title_color'); + + if(count($attrs)) $title = sprintf("%s", implode(';', $attrs), htmlspecialchars($title)); + + return $title; + } + function getContent(){ + return $this->get('content'); + } + function getCategory(){ + return $this->get('category'); + } + function getNickName(){ + return $this->get('nick_name'); + } + function getAuthorSite(){ + return $this->get('author_site'); + } + function getCommentCount(){ + $comment_count = $this->get('comment_count'); + return $comment_count>0 ? $comment_count : ''; + } + function getTrackbackCount(){ + $trackback_count = $this->get('trackback_count'); + return $trackback_count>0 ? $trackback_count : ''; + } + function getRegdate($format = 'Y.m.d H:i:s') { + return zdate($this->get('regdate'), $format); + } + function printExtraImages() { + return $this->get('extra_images'); + } + function haveFirstThumbnail() { + return $this->has_first_thumbnail_idx; + } + function getThumbnail(){ + return $this->get('thumbnail'); + } + function getMemberSrl() { + return $this->get('member_srl'); + } + } +?> diff --git a/widgets/mcontent/skins/default/content.html b/widgets/mcontent/skins/default/content.html new file mode 100644 index 000000000..0094232a9 --- /dev/null +++ b/widgets/mcontent/skins/default/content.html @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/widgets/mcontent/skins/default/image_title.html b/widgets/mcontent/skins/default/image_title.html new file mode 100644 index 000000000..c1b51036f --- /dev/null +++ b/widgets/mcontent/skins/default/image_title.html @@ -0,0 +1,52 @@ + diff --git a/widgets/mcontent/skins/default/image_title_content.html b/widgets/mcontent/skins/default/image_title_content.html new file mode 100644 index 000000000..112f68456 --- /dev/null +++ b/widgets/mcontent/skins/default/image_title_content.html @@ -0,0 +1,55 @@ + \ No newline at end of file diff --git a/widgets/mcontent/skins/default/mcontent.css b/widgets/mcontent/skins/default/mcontent.css new file mode 100644 index 000000000..4dbbe7f15 --- /dev/null +++ b/widgets/mcontent/skins/default/mcontent.css @@ -0,0 +1,14 @@ +@charset "utf-8"; +/* Mobile Content Widget */ +.mcw h2{position:relative;margin:-1px 0; padding:10px;background:#ddd;border:1px solid #999;border-left:0;border-right:0;text-shadow:1px 1px #fff;font-size:20px} +.mcw ul{margin:0;padding:0;list-style:none} +.mcw li{border-bottom:1px solid #999} +.mcw li:last-child{border:0} +.mcw ul a{display:block;text-decoration:none;padding:10px;color:#000;*zoom:1} +.mcw ul a:after{content:"";display:block;clear:both} +.mcw ul a:visited{color:#666} +.mcw .th{float:left;margin:0 10px 5px 0;border:1px solid #fff;background:#eee;color:#666;text-align:center;line-height:80px;width:80px;height:80px;font-size:14px;box-shadow:0 0 3px #333;-moz-box-shadow:0 0 3px #333;-webkit-box-shadow:0 0 3px #333} +.mcw .title{display:block;margin:0 0 8px 0} +.mcw .title em{color:#f60;font-size:12px} +.mcw .auth{font-size:12px;color:#666} +.mcw .text{color:#666} diff --git a/widgets/mcontent/skins/default/normal.html b/widgets/mcontent/skins/default/normal.html new file mode 100644 index 000000000..8f8ead5bc --- /dev/null +++ b/widgets/mcontent/skins/default/normal.html @@ -0,0 +1,46 @@ + diff --git a/widgets/mcontent/skins/default/skin.xml b/widgets/mcontent/skins/default/skin.xml new file mode 100644 index 000000000..cb744a97f --- /dev/null +++ b/widgets/mcontent/skins/default/skin.xml @@ -0,0 +1,15 @@ + + + 모바일 콘텐츠 위젯 스킨 + 모바일 콘텐츠 전용 심플한 스킨 입니다. + 0.1 + 2009-08-20 + + Jeong, Chan Myeong + + + + 흰색 바탕용 + + +