source: branches/version-2_12-dev/data/class/pages/LC_Page_Sitemap.php @ 22567

Revision 22567, 10.9 KB checked in by shutta, 11 years ago (diff)

#2043 (typo修正・ソース整形・ソースコメントの改善 for 2.12.4)
Zend Framework PHP 標準コーディング規約のコーディングスタイルへ準拠。
classおよびfunctionの開始波括弧「{」のスタイルを修正。

  • Property svn:eol-style set to LF
  • Property svn:keywords set to Id
  • Property svn:mime-type set to text/x-httpd-php; charset=UTF-8
Line 
1<?php
2/*
3 * This file is part of EC-CUBE
4 *
5 * Copyright(c) 2000-2013 LOCKON CO.,LTD. All Rights Reserved.
6 *
7 * http://www.lockon.co.jp/
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22 */
23
24// {{{ requires
25require_once CLASS_EX_REALDIR . 'page_extends/LC_Page_Ex.php';
26
27/**
28 * Sitemapプロトコル ファイル生成モジュール.
29 * PHP versions 4 and 5
30 *
31 * <pre>
32 * このモジュールは Sitemapプロトコルに対応した XMLファイルを出力する.
33 * EC-CUBE インストールディレクトリの htmlディレクトリへ配置することにより動作する.
34 *
35 * このモジュールにより, 以下のページのサイトマップが生成される.
36 * 1. $staticURL で指定したページ
37 * 2. 管理画面のデザイン管理から生成したページ
38 * 3. 公開されているすべての商品一覧ページ
39 * 4. 公開されているすべての商品詳細ページ
40 *
41 * このモジュールを設置後, 各検索エンジンにサイトマップを登録することにより, 検索エンジンの
42 * インデックス化が促進される.
43 * </pre>
44 * @see https://www.google.com/webmasters/tools/siteoverview?hl=ja
45 * @see https://siteexplorer.search.yahoo.com/mysites
46 *
47 * @author Kentaro Ohkouchi
48 * @version $Id:sitemap.php 15532 2007-08-31 14:39:46Z nanasess
49 *
50 * :TODO: 各ページの changefreq や priority を指定できるようにする
51 * :TODO: filemtime 関数を使えば、静的なページの更新時間も取得できそう
52 */
53class LC_Page_Sitemap extends LC_Page_Ex
54{
55
56    // }}}
57    // {{{ properties
58
59    /** 動的に生成しないページの配列 */
60    var $staticURL;
61
62    /** ページリスト */
63    var $arrPageList;
64
65    // }}}
66    // {{{ functions
67
68    /**
69     * Page を初期化する.
70     *
71     * @return void
72     */
73    function init()
74    {
75        parent::init();
76
77        $this->staticURL = array();
78
79        $this->staticURL[] = HTTP_URL . 'rss/' . DIR_INDEX_PATH;
80    }
81
82    /**
83     * Page のプロセス.
84     *
85     * @return void
86     */
87    function process()
88    {
89        // ページのデータを取得
90        // FIXME PCサイトのみに限定している。ある程度妥当だとは思うが、よりベターな方法はないだろうか。
91        $this->arrPageList = $this->getPageData('device_type_id = ?', DEVICE_TYPE_PC);
92
93        //キャッシュしない(念のため)
94        header('Paragrama: no-cache');
95
96        //XMLテキスト
97        header('Content-type: application/xml; charset=utf-8');
98
99        // 必ず UTF-8 として出力
100        mb_http_output('UTF-8');
101        ob_start('mb_output_handler');
102
103        echo '<?xml version="1.0" encoding="UTF-8"?>' . "\n";
104        echo '<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">' . "\n";
105
106        // TOPページを処理
107        $arrTopPagesList = $this->getTopPage($this->arrPageList);
108        $this->createSitemap($arrTopPagesList[0]['url'],
109                             $this->date2W3CDatetime($arrTopPagesList[0]['update_date']),
110                             'daily', 1.0);
111
112        // 静的なページを処理
113        foreach ($this->staticURL as $url) {
114            $this->createSitemap($url, '', 'daily', 1.0);
115        }
116
117        // 編集可能ページを処理
118        $arrEditablePagesList = $this->getEditablePage($this->arrPageList);
119        foreach ($arrEditablePagesList as $arrEditablePage) {
120            $this->createSitemap($arrEditablePage['url'],
121                                 $this->date2W3CDatetime($arrEditablePage['update_date']));
122        }
123
124        // 商品一覧ページを処理
125        $arrProductPagesList = $this->getAllProducts();
126        foreach ($arrProductPagesList as $arrProductPage) {
127            $this->createSitemap($arrProductPage['url'], '', 'daily');
128        }
129
130        // 商品詳細ページを処理
131        $arrDetailPagesList = $this->getAllDetail();
132        foreach ($arrDetailPagesList as $arrDetailPage) {
133            $this->createSitemap($arrDetailPage['url'],
134                                 $this->date2W3CDatetime($arrDetailPage['update_date']));
135        }
136
137        echo '</urlset>' . "\n";
138    }
139
140    /**
141     * デストラクタ.
142     *
143     * @return void
144     */
145    function destroy()
146    {
147        parent::destroy();
148    }
149
150    /**
151     * Sitemap の <url /> を生成する.
152     *
153     * @param string $loc ページの URL ※必須
154     * @param string $lastmod ファイルの最終更新日 YYYY-MM-DD or W3C Datetime 形式
155     * @param string $changefreq ページの更新頻度
156     * @param double $priority URL の優先度
157     * @return Sitemap 形式の <url />
158     * @see https://www.google.com/webmasters/tools/docs/ja/protocol.html#xmlTagDefinitions
159     * TODO Smarty に移行すべき?
160     */
161    function createSitemap($loc, $lastmod = '', $changefreq = '', $priority = '')
162    {
163        printf("\t<url>\n");
164        printf("\t\t<loc>%s</loc>\n", htmlentities($loc, ENT_QUOTES, 'UTF-8'));
165        if (!empty($lastmod)) {
166            printf("\t\t<lastmod>%s</lastmod>\n", $lastmod);
167        }
168        if (!empty($changefreq)) {
169            printf("\t\t<changefreq>%s</changefreq>\n", $changefreq);
170        }
171        if (!empty($priority)) {
172            printf("\t\t<priority>%01.1f</priority>\n", $priority);
173        }
174        printf("\t</url>\n");
175    }
176
177    /**
178     * TOPページの情報を取得する.
179     *
180     * @param array $arrPageList すべてのページ情報の配列
181     * @return array TOPページの情報
182     */
183    function getTopPage($arrPageList)
184    {
185        $arrRet = array();
186        foreach ($arrPageList as $arrPage) {
187            if ($arrPage['page_id'] == '1') {
188                $arrRet[0] = $arrPage;
189                return $arrRet;
190            }
191        }
192    }
193
194    /**
195     * すべての編集可能ページの情報を取得する.
196     *
197     * @param array $arrPageList すべてのページ情報の配列
198     * @return array 編集可能ページ
199     */
200    function getEditablePage($arrPageList)
201    {
202        $arrRet = array();
203        foreach ($arrPageList as $arrPage) {
204            if ($arrPage['page_id'] > 4) {
205                $arrRet[] = $arrPage;
206            }
207        }
208        return $arrRet;
209    }
210
211    /**
212     * すべての商品一覧ページを取得する.
213     *
214     * @return array 検索エンジンからアクセス可能な商品一覧ページの情報
215     */
216    function getAllProducts()
217    {
218
219        // XXX: 商品登録の無いカテゴリは除外する方が良い気もする
220        $objQuery = SC_Query_Ex::getSingletonInstance();
221        $sql = 'SELECT category_id FROM dtb_category WHERE del_flg = 0';
222        $result = $objQuery->getAll($sql);
223
224        $arrRet = array();
225        foreach ($result as $row) {
226            // :TODO: カテゴリの最終更新日を取得できるようにする
227
228            $arrPage['url'] = HTTP_URL . 'products/list.php?category_id=' . $row['category_id'];
229            $arrRet[] = $arrPage;
230        }
231        return $arrRet;
232    }
233
234    /**
235     * すべての商品詳細ページを取得する.
236     *
237     * @return array 検索エンジンからアクセス可能な商品詳細ページの情報
238     */
239    function getAllDetail()
240    {
241        $objQuery = SC_Query_Ex::getSingletonInstance();
242        $sql = 'SELECT product_id, update_date FROM dtb_products WHERE del_flg = 0 AND status = 1';
243        $result = $objQuery->getAll($sql);
244
245        $arrRet = array();
246        foreach ($result as $row) {
247
248            $arrPage['update_date'] = $row['update_date'];
249
250            $arrPage['url'] = HTTP_URL . substr(P_DETAIL_URLPATH, strlen(ROOT_URLPATH)) . $row['product_id'];
251            $arrRet[] = $arrPage;
252        }
253        return $arrRet;
254    }
255
256    /**
257     * ブロック情報を取得する.
258     *
259     * @param string $where WHERE句
260     * @param array  $arrVal WHERE句の値を格納した配列
261     * @return array $arrPageList ブロック情報
262     */
263    function getPageData($where = '', $arrVal = '')
264    {
265        $objQuery = SC_Query_Ex::getSingletonInstance();     // DB操作オブジェクト
266        $sql = '';                      // データ取得SQL生成用
267        $arrRet = array();              // データ取得用
268
269        // SQL生成(url と update_date 以外は不要?)
270        $sql .= ' SELECT';
271        $sql .= ' page_id';             // ページID
272        $sql .= ' ,page_name';          // 名称
273        $sql .= ' ,url';                // URL
274        $sql .= ' ,filename';           // ファイル名称
275        $sql .= ' ,header_chk ';        // ヘッダー使用FLG
276        $sql .= ' ,footer_chk ';        // フッター使用FLG
277        $sql .= ' ,author';             // authorタグ
278        $sql .= ' ,description';        // descriptionタグ
279        $sql .= ' ,keyword';            // keywordタグ
280        $sql .= ' ,update_url';         // 更新URL
281        $sql .= ' ,create_date';        // データ作成日
282        $sql .= ' ,update_date';        // データ更新日
283        $sql .= ' FROM ';
284        $sql .= '     dtb_pagelayout';
285
286        // where句の指定があれば追加
287        if ($where != '') {
288            $sql .= ' WHERE ' . $where;
289        }
290
291        $sql .= ' ORDER BY page_id';
292
293        $arrPageList = $objQuery->getAll($sql, $arrVal);
294
295        // URL にプロトコルの記載が無い場合、HTTP_URL を前置する。
296        foreach ($arrPageList as $key => $value) {
297            $arrPage =& $arrPageList[$key];
298            if (!preg_match('|^https?://|i', $arrPage['url'])) {
299                $arrPage['url'] = HTTP_URL . $arrPage['url'];
300            }
301            $arrPage['url'] = preg_replace('|/' . preg_quote(DIR_INDEX_FILE) . '$|', '/' . DIR_INDEX_PATH, $arrPage['url']);
302        }
303        unset($arrPage);
304
305        return $arrPageList;
306    }
307
308    /**
309     * date形式の文字列を W3C Datetime 形式に変換して出力する.
310     *
311     * @param date $date 変換する日付
312     * @return void
313     */
314    function date2W3CDatetime($date)
315    {
316        $arr = array();
317        // 正規表現で文字列を抽出
318        preg_match('/^([0-9]{4})-([0-9]{2})-([0-9]{2}) ([0-9]{2}):([0-9]{2}):([0-9]{2})/', $date, $arr);
319        // :TODO: time zone も取得するべき...
320        return sprintf('%04d-%02d-%02dT%02d:%02d:%02d+09:00',
321                       $arr[1], $arr[2], $arr[3], $arr[4], $arr[5], $arr[6]);
322    }
323
324}
Note: See TracBrowser for help on using the repository browser.