1

我的网站有很多“页面”。为不同的主题创建页面并由用户管理(例如加勒比海盗的页面或贾斯汀比伯的页面)。该页面属于一个类别,例如电影或音乐。页面的默认 URL 是http://www.example.com/category-directory/page-id。或者,如果版主想要建立自己的页面链接作为更简单的导航方式,他们可以通过为“Page.link”列赋予值来实现(例如:http ://www.example.com/电影加勒比海盗的potc)。

我的路线都设置好了,这本身就花了一些时间。该路由检查“.com”之后的 URL 的第一个参数是否与任何页面的“链接”列匹配,如果匹配,则路由到该页面及其下的控制器/操作。如果没有,照常路线。

当用户查看http://www.example.com/potc/articles/99http://www.example.com/movies/106/articles/99时(假设加勒比海盗是带有 id 的页面= 106) 然后将用户定向到文章控制器,查看操作,id = 99。106 作为 $page_id 传递给控制器​​的操作,并在 AppController 中设置。

我在控制器中提供的 URL 数组可以正常工作,例如,如果 id 不存在我重定向到'controller' => 'articles', 'action' => 'index', 'page_id' = $page_id. 我被重定向回http://www.example.com/potc/articles。我实际上不必定义控制器,因为默认情况下它使用当前控制器。

但是,我遇到了分页链接的问题。例如,我在文章索引页面上,有 100 篇文章,但我只查看前 20 篇。我希望将 URL 输出为http://www.example.com/potc/articles/page :2

相反,如果我将 URL 数组设置为'controller' => 'articles', 'action' => 'index', 'page_id' => $page_id与我在控制器中所做的类似,我会得到链接http://www.example.com/articles/page_id:106/page:2我可以理解为什么这条路线不起作用,因为如果Page.link 存在,我可能必须传递值,如果不传递Category.directory 和Page.id。我尝试使用 URL array 执行后一个选项'controller' => 'articles', 'action' => 'index', 'category' => $page['Category']['directory'], 'page_id' => $page_id。我有一条与之匹配的路线,但分页链接仍然生成为http://www.example.com/articles/category:movies/page_id:106/page:2

以下是我的 routes.php 中的一些片段:https ://gist.github.com/9ba0a54a7f4d68803d14

我的问题是:“我可以为分页链接使用确切的当前 URL(减去对当前页面的引用)吗? ”例如,如果我正在查看http://www.example.com/dynamic-route/controller /id/page:2,分页链接会自动转到http://www.example.com/dynamic-route/controller/id/page:3

我将所有分页代码放入一个元素中,并且我想在需要分页的任何地方将该元素包含在我的视图文件中。

现在,如果我正在查看文章控制器,索引操作,分页链接的默认 URL 是http://www.example.com/articles/page:2,但我需要它是http://www .example.com/dynamic-route/articles/page:2

我尝试使用

   $this->Paginator->options = array(
                    'url' => 'http://www.example.com/dynamic-route/articles'
   ); 

但是链接的输出类似于http://www.example.com/articles/http://www.example.com/dynamic-route/articles/page:2

编辑: Jeztah,你提到放

'customlink' => '[a-z0-9]{1}([a-z0-9-]{2,}[a-z0-9]{1})?'

在我的路线上。好吧,我不能简单地这样做,因为路由会进行查询以检查 customlink 是否在数据库中,如果是则相应地路由。

我的路由器文件的一个例子是:

在获得 URL 的第一个参数http://www.domain.com/ CUSTOM-LINK后,执行以下操作:

if($param1 != "users" && $param1){
    $pagelink = str_replace('-', ' ', $param1);

    $pagesModel = ClassRegistry::init('Page');
    $matchedpage = $pagesModel->find('first', array(
        'conditions' => array('Page.link LIKE' => "$pagelink"), 'recursive' => '-1'
    ));

    if($matchedpage['Page']['id']){
        Router::connect('/' . $param1 . '/:controller', array('action' => 'index', 'page_id' => $matchedpage['Page']['id']), array(
            'pass' => array('page_id'),
            'page_id' => '[0-9]+',
            'persist' => array('page_id'),
        ));

        Router::connect('/' . $param1 . '/:controller/:action/:id', array('page_id' => $matchedpage['Page']['id']), array(
            'pass' => array('page_id', 'id'),
            'page_id' => '[0-9]+',
            'persist' => array('page_id'),
        ));


        Router::connect('/' . $param1 . '/:controller/:id', array('action' => 'view', 'page_id' => $matchedpage['Page']['id']), array(
            'pass' => array('page_id', 'id'),
            'page_id' => '[0-9]+',
            'id' => '[0-9]+',
            'persist' => array('page_id'),
        ));


        Router::connect('/' . $param1 . '/:controller/:action', array('page_id' => $matchedpage['Page']['id']), array(
            'pass' => array('page_id'),
            'page_id' => '[0-9]+',
            'persist' => array('page_id'),
        ));

    } // end if page matches
} // end if param1 is not users

如果我去http://www.domain.com/custom-link/articleshttp://www.domain.com/custom-link/articles/1http://www.domain.com/custom- link/articles/edit/1等显示正确的页面。如果我尝试导航到自定义链接不存在的页面,请继续查看我的路由器文件,如果没有其他路由匹配,则抛出错误消息。那里一切都好。

在我的 ArticlesController.php 中,重定向方法按预期工作。例如,我尝试访问http://www.domain.com/custom-link/articles/99但 id 99 不存在,我被正确重定向回http://www.domain.com/custom - 链接/文章使用

$this->redirect(array('action' => 'index', 'page_id' => $page_id), null, true);

但是,在分页上,我将 url 选项设置为

array('controller' => 'articles', 'action' => 'index', 'page_id' => $page_id);

其中 $page_id 在 AppController 中设置为由路由器传递。虽然我不需要设置控制器和动作,因为将使用电流。同样,我的分页链接显示为http://www.domain.com/articles/page_id:3/page:2而不是我想要的http://www.domain.com/custom-link/articles/page :2

编辑:杰兹塔

我刚刚将我的整个路由文件更改为具有自定义链接参数的文件(我重命名为“链接”),然后创建了一个组件以在应用程序控制器中使用以从链接中获取页面 ID。

示例路线

Router::connect('/:link/:controller/:action/:id', array(), array(
    'pass' => array('link', 'id'),
    'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
    'persist' => array('link'),
));


Router::connect('/:link/:controller/:id', array('action' => 'view'), array(
    'pass' => array('link', 'id'),
    'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
    'id' => '[0-9]+',
    'persist' => array('link'),
));

但是,使用 url 数组作为array('link' => 'test-example')url 进行分页输出为http://www.domain.com/articles/link:test-example/page:2. 我的代码与您的代码完全相同,减去将“自定义链接”重命名为“链接”。此外,现在我的路线无法区分我是否尝试访问http://www.domain.com/custom-link,这将是 'controller' => 'pages', 'action' => 'view' (他的页面基本上会使用 News 模型并显示该页面的最新消息)如果我试图访问http://www.domain.com/articles这将是 'controller' => 'articles', ' action' => 'index' 将显示所有页面的最新文章。注意,文章和新闻是不同的模型。

Router::connect('/:link', array('controller' => 'pages', 'action' => 'view'), array(
    'pass' => array('link'),
    'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
    'persist' => array('link'),
));

是相同的

Router::connect('/:controller', array('action' => 'index'));

因为不再在路由中检查“链接”是否属于特定页面。

编辑:好的,我正在取得进展。我发现下一个和上一个分页链接不太适合 paginator->options。我删除了 paginator->options 并将下一个按钮的 url 数组留空。该链接输出为http://www.domain.com/articles/custom-link/articles/page:2。进步!但是有什么办法可以驾驭第一个“文章/”吗?

我的路线是

Router::connect('/:link/:controller/*', array(), array(
    'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
));
4

3 回答 3

0

最好为此使用相对路径:

   $this->Paginator->options = array(
                    'url' => '/dynamic-route'
   );
Or:
   $this->Paginator->options = array(
                    'url' => '/dynamic-route/articles'
   ); 
Or:
   $this->Paginator->options = array(
                    'url' => array('controller'=>'dynamic-route')
   );

或开始在您的路由器文件http://api.cakephp.org/class/router#method-Routerurl中构建自定义路由

于 2012-12-09T09:39:26.660 回答
0

如果您正确构建路线,这一切都应该自动运行:

在你的 routes.php

/**
 * Movies view/details page
 */
Router::connect(
    '/:link',
    array(
         'controller'  => 'movies',
         'action'      => 'view',
    ),
    array(
         /**
          * Custom Link:
          * lowercase alphanumeric and dashes, but NO leading/trailing dash
          * should be at least 4 characters long
          *
          * IMPORTANT:
          * custom links should *NOT* overlap with actual controller
          * names, like 'movies, articles' etc!
          */
         'link'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',

         // Define what should be passed to the 'view' action as arguments
         'pass'         => array('link'),

         /**
          * Define what parameters should be automatically preserved
          * when creating URLs/links
          */
         'persist' => array('link'),
    )
);

/**
 * Article view/details page
 */
Router::connect(
    '/:link/:controller/:page_id',
    array(
         'controller'  => 'articles',
         'action'      => 'view',
    ),
    array(
         /**
          * Specify the 'controllers' that are allowed to be put
          * after link. You can add more, for example
          * 'reviews' to show reviews for a movie. this
          * will result in Reviews::view([page_id]) to be shown
          */
         'controller' => 'articles',
         'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
         'page_id'  => '[1-9]{1}[0-9]{0,}',

         // Define what should be passed to the 'view' action as arguments
         'pass'       => array('page_id', 'link'),

         /**
          * Define what parameters should be automatically preserved
          * when creating URLs/links
          */
         'persist' => array('page_id', 'link'),
    )
);

/**
 * Article overview/index page
 */
Router::connect(
    '/:link/:controller/*',
    array(
         'controller'  => 'articles',
         'action'      => 'index',
    ),
    array(
         'controller' => 'articles',
         'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',

         // Define what should be passed to the 'index' action as arguments
         'pass'       => array('link'),

         /**
          * Define what parameters should be automatically maintained
          * when creating URLs/links
          */
         'persist' => array('page_id', 'link'),
    )
);

打开http://example.com/potc时,您将被带到 Movies::view('potc')

打开http://example.com/potc/articles时,您将被带到 Articles::index('potc')

打开http://example.com/potc/articles/page:2时,您还将被带到 Articles::index('potc')。“页面”将作为命名参数传递

打开http://example.com/potc/articles/99时,您将被带到 Articles::view(99, 'potc')

在您的 Articles/index.ctp 视图中:

debug($this->Html->link('current page', array()));
// outputs:
// <a href="/potc/articles/">current page</a>

debug($this->Html->link('next page', array('page' => 2)));
// outputs:
// <a href="/potc/articles/page:2">next page</a>

debug($this->Html->link('view article', array('action' => 'view', 'page_id' => 99)));
// outputs:
// <a href="/potc/articles/99">view article</a>

访问分页链接(http://example.com/potc/articles/page:2/sort:title/dir:desc

debug($this->request);

输出:

object(CakeRequest) {
    params => array(
        'plugin' => null,
        'controller' => 'articles',
        'action' => 'index',
        'named' => array(
            'page' => '2',
            'sort' => 'title',
            'dir' => 'desc'
        ),
        'pass' => array(
            (int) 0 => 'potc'
        ),
        'link' => 'potc',
        (int) 0 => 'page:2/sort:title/dir:desc',
        'paging' => array(
            'Article' => array(
                'page' => (int) 1,
                'current' => (int) 0,
                'count' => (int) 3,
                'prevPage' => false,
                'nextPage' => false,
                'pageCount' => (int) 1,
                'order' => array(),
                'limit' => (int) 10,
                'options' => array(
                    'page' => (int) 2,
                    'sort' => 'title',
                    'order' => array(),
                    'conditions' => array()
                ),
                'paramType' => 'named'
            )
        ),
        'models' => array(
            'Article' => array(
                'plugin' => null,
                'className' => 'Article'
            )
        )
    )
    data => array()
    query => array()
    url => 'potc/articles/page:2/sort:title/dir:desc'
    base => ''
    webroot => '/'
    here => '/potc/articles/page:2/sort:title/dir:desc'
}

如您所见,您需要的所有信息都在请求对象中

[OP修改后更新]

我看到您在这条路线中使用了 ':id':

/:link/:controller/:action/:id

但是您没有为此包含正则表达式。另外,我认为您不想使用此路由,因为要匹配此路由,您必须在 URL 中包含该操作。例如:

/test-example/articles/view/1

这条路线的更好选择是我原来的例子中的这条路线:

/:link/:controller/:page_id

这将路由到:

Controller::view('page_id', 'link')

正如我之前提到的,添加路由的“顺序”决定了首先匹配的内容,因此如果您在包含 CakePHP 默认路由之前添加了路由,那么您的路由将是首选。

在这种情况下; /test-example/ /pages/

将始终与 '/:link' 匹配,因此 /pages/view/test-example,如果你传递一个有效的控制器名(!)为了​​防止这种情况,你应该在包含 CakePHP 默认路由之后添加你的路由。

如果添加路由 /:link/:controller/

然后这个网址:/test-example/something

当“某物”是已知控制器时才会匹配

/:link/:controller/:action/:id

如果“控制器”是已知控制器,则匹配,任何操作都将被“接受”,但如果您尝试访问它,则会导致“未找到异常”。另外(如上所述)您应该为“id”添加一个正则表达式

您添加的最后一条路线:

Router::connect('/:link/:controller/*', array(), array(
    'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
));

缺少默认的 'action' 参数,因此 CakePHP 可能无法匹配,这应该是:

Router::connect('/:link/:controller/*', 
    array(
        'action' => 'index',
    ), array(
    'link' => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
));

这样,它将匹配:

/any-link/controllername/pass1
/any-link/controllername/pass1/pass2
etc.

分别是 'pass1, pass2' 将作为参数发送到控制器的操作,例如:

/test-example/articles/some/thing/else

将被路由到:

ArticlesController::index('some', 'thing', 'else');

我认为您的路线的最佳顺序是:

  1. Cake 默认路由
  2. /:link (-> pages::view('link'))
  3. /:link/:controller/:id (-> controller::view('id', 'link))
  4. /:link/:controller/* (-> controller::index('link'))

注意最后两条路线的顺序;Route 3 仅在 'id' 为数值时才匹配,导致路由 'view' 动作 Route 4 将匹配控制器之后的 'anything',并显示控制器的 'index' 动作

于 2013-02-02T11:10:00.850 回答
0

编辑:在我意识到我的上一个答案在 ArticlesConntroller::redirect 方面存在缺陷之后,这是一个经过编辑的答案

对于任何设法阅读整个问题/评论并想知道答案的人。感谢 Jeztah 的帮助,我想出了一个解决方案,我将把这笔赏金授予他。让我很难过的一个问题是一个页面可能有 2 个 URL。例如:http://www.domain.com/potchttp://www.domain.com/movies/106。基本上我结合了我和 Jeztah 的想法。

首先,我必须检查routes.php以查看 .com 之后的第一个参数是否与页面的快捷方式“链接”匹配(例如: http: //www.domain.com/potc = potc)。

// get params of URL
$requestURI = explode('/', $_SERVER['REQUEST_URI']);
$scriptName = explode('/',$_SERVER['SCRIPT_NAME']);

for($i= 0;$i < sizeof($scriptName);$i++) {
      if ($requestURI[$i]     == $scriptName[$i]) {
                unset($requestURI[$i]);
        }
      }
 $param = array_values($requestURI);

// only concerned about first param
$param1 = $param[0];

// do the following routes if url != http://www.domain.com/users/* or http://www.domain.com
if($param1 != "users" && $param1){

    // convert http://www.domain.com/assassins-creed to "assassins creed"
    $pagelink = str_replace('-', ' ', $param1);
    $pagesModel = ClassRegistry::init('Page');
    // check if there is a page whose "link" matches converted link, if so, get the field "id"
    $matched_page_id = $pagesModel->field('id', array('Page.link LIKE' => "$pagelink"));
    // if there is a match, route as follows
    if($matched_page_id){

         // Jeztah's way with some modifications

        // domain.com/potc/articles/edit/1
        Router::connect(
            '/:link/:controller/:action/:id',
            array('page_id' => $matched_page_id, 'category' => 0),
            array(
                'pass' => array('id', 'link', 'category', 'page_id'),
                'link'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
                'persist' => array('link'),
                'id' => '[0-9]+'
            )
        );

        // domain.com/potc/articles/1
        Router::connect(
            '/:link/:controller/:id',
            array('action' => 'view', 'page_id' => $matched_page_id, 'category' => 0),
            array(
                'pass' => array('id', 'link', 'category', 'page_id'),
                'link'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
                'persist' => array('link'),
                'id' => '[0-9]+'
            )
        );

        // domain.com/potc/articles/add
        Router::connect(
            '/:link/:controller/:action',
            array('page_id' => $matched_page_id, 'category' => 0),
            array(
                'pass' => array('link', 'category', 'page_id'),
                'link'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
                'persist' => array('link'),
                'action' => 'add|uncategorized',
            )
        );

        // domain.com/potc/articles OR domain.com/articles/page:2
        Router::connect(
            '/:link/:controller/*',
            array('action' => 'index', 'page_id' => $matched_page_id, 'category' => 0),
            array(
                'pass' => array('link', 'category', 'page_id'),
                'link'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
                'persist' => array('link'),
            )
        );
    }  // end if first param matches a page's "link" shortcut
}  // end if first param is not users and is not empty

// do some general site routing
// do some general site routing

// now route for page's that do not have a link shortcut (example: http://www.domain.com/movies/106)

// domain.com/movies/106/articles/edit/1
Router::connect(
    '/:category/:page_id/:controller/:action/:id',
    array('link' => '0'),
    array(
        'pass' => array('id', 'link', 'category', 'page_id'),
        'page_id' => '[0-9]+',
        'id' => '[0-9]+'
    )
);

// domain.com/movies/106/articles/1
Router::connect(
    '/:category/:page_id/:controller/:id',
    array('action' => 'view', 'link' => '0'),
    array(
        'pass' => array('id', 'link', 'category', 'page_id'),
        'page_id' => '[0-9]+',
        'id' => '[0-9]+'
    )
);

// domain.com/movies/106/articles/add
Router::connect(
    '/:category/:page_id/:controller/:action',
    array('link' => '0'),
    array(
        'pass' => array('link', 'category', 'page_id'),
        'page_id' => '[0-9]+',
        'action' => 'add|uncategorized'
    )
);

// domain.com/movies/106/articles OR domain.com/movies/106/articles/page:2
Router::connect(
    '/:category/:page_id/:controller/*',
    array('action' => 'index', 'link' => '0'),
    array(
        'pass' => array('link', 'category', 'page_id', 'controller'),
        'page_id' => '[0-9]+'
    )
);

因为 page_id 在 :link 路由和 :category 路由中都被传递,所以我不必像建议的那样拥有一个获取页面 id 的组件。

现在在我的ArticlesController

public function index($link = null, $category = null, $page_id = null, $controller = null) {
    if($page_id){

         // display all articles for a particular page if on
         // http://www.domain.com/potc/articles or http://www.domain.com/movies/106/articles
        $this->set('articles', $this->paginate('Article', array('Article.page_id' => $page_id, 'Article.status <= 5'))); 
    }
    else{

        // display all articles if on http://www.domain.com/articles
        $this->set('articles', $this->paginate('Article', array('Article.status <= 5')));
    }    
} // end index function

public function view($id = null, $link = null, $category = null, $page_id = null) {
    $this->Article->id = $id;
    $article_page = $this->Article->field('page_id', array('id =' => $id));

    if (!$this->Article->exists()) {
        $this->Session->setFlash('This article does not exist');

        // using PageLinkRedirect function of my custom component,
        // determine where to redirect. Either to "potc/articles" or
        // "movies/106/articles";
        $this->redirect('../'.$this->CustomPage->PageLinkRedirect($link, $category, $page_id).'/articles', null, true);
    }
    elseif($article_page != $page_id) {
        $this->Session->setFlash('Invalid article for this page');

        // using PageLinkRedirect function of my custom component,
        // determine where to redirect. Either to "potc/articles" or
        // "movies/106/articles";
        $this->redirect('../'.$this->CustomPage->PageLinkRedirect($link, $category, $page_id).'/articles', null, true);
    }
    else{
        $this->set('title_for_layout', $this->Article->field('title', array('id =' => $id)));
        $this->set('article',  $this->Article->read(null, $id));
        $this->set('comments', $this->paginate($this->Article->Comment, array('Comment.module_id' => $this->viewVars['module_id'], 'Comment.post_id' => $id)));
    } // end else
} // end view function

我的分页存储在一个元素中,但我在文章 index.ctp 视图的底部加载了该元素

<?php echo $this->element('all/pagination', array('pagination_url' => 'articles')); ?>

我会为每个不同的控制器放置相同的代码,但更改“文章”一词(例如:“评论”、“新闻/主题”等)

我的 pagination.ctp 元素看起来像这样

<?php
if($this->params['paging'][key($this->params['paging'])]['pageCount'] > 1){
    $pagination_class = "pagination_enabled";
}
else{
    $pagination_class = "pagination_disabled";
}
 ?>
<div class="<?php echo $pagination_class; ?>">
<?php echo $this->Paginator->counter(array(
    'format' => __('Page {:page} of {:pages}, showing {:current} records out of {:count} total, starting on record {:start}, ending on {:end}')
)); 
?>
<ul>
<?php

// $link is set in AppController from the link param, same with category and page_id
if($link){
    $this->Paginator->options = array('url' => array('controller' => null, $link, $pagination_url));

    // This outputs http://www.domain.com/potc/articles where articles
    // is from the passed $pagination_url when I loaded the element
    // in my Articles index.ctp view. I have to set controller = null
    // or else the link will output as http://www.domain.com/articles/potc/articles
}
else{
    $this->Paginator->options = array('url' => array('controller' => null, 'link' => null, $category, $page_id, $pagination_url));

    // This outputs http://www.domain.com/movies/106/articles where articles
    // is from the passed $pagination_url when I loaded the element
    // in my Articles index.ctp view. I have to set controller = null AND
    // link = null or else the link will output as
    // http://www.domain.com/articles/0/movies/9/articles
} 
echo $this->Paginator->prev('< ' . __('previous'), array('tag' => 'li'), null, array('class' => 'prev disabled'));
if($this->Paginator->numbers){echo $this->Paginator->numbers(array('tag' => 'li', 'separator' => ''));}
echo $this->Paginator->next(__('next') . ' >', array('tag' => 'li'), null, array('class' => 'next disabled'));
?>
</ul></div>
于 2013-02-06T18:55:44.980 回答