cakephp Непряка модификация на претоварено свойство PaginatorHelper

имам следната форма за търсене

<?php
echo $this->Form->create('Order', array('action' => 'search','type'=>'get'));?>
<?php echo $this->Form->input('SearchTerm',array('label' => 'Find:')); ?>
<?php $options=array('full_name'=>'By Name','code'=>'By Code','phone'=>'By Phone','email'=>'By Mail'); ?>
<?php echo $this->Form->input('options', array('type'=>'select', 'label'=>'Search:', 'options'=>$options)); ?>

<?php echo $this->Form->end('search'); ?>  

<?php if($rs!=0){?>
<?php if($rs!=null) { ?>
    results found : <?php print $results; //print_r ($result['Order']['full_name']); ?>
<h1 class="ico_mug">Results Matching Term: <?php print '"'.$term.'"' ;?></h1>
<table id="table">
    <tr>

        <th>Client Name</th>
        <th>Order Code</th>
        <th>Phone</th>
        <th>Received Order</th>
        <th>Working On Order </th>
        <th>Order Ready </th>
    </tr>

    <!-- Here is where we loop through our $posts array, printing out post info -->

    <?php foreach ($rs as $order): ?>
    <tr>

        <td>
            <?php echo $this->Html->link($order['Order']['full_name'],
            array('controller' => 'orders', 'action' => 'view', $order['Order']['id'])); ?>
        </td>
        <td><?php echo $order['Order']['code']; ?></td>
        <td><?php echo $order['Order']['phone']; ?></td>
        <td><?php echo $this->OrderStatus->getStatusString($order['Order']['receive_state']); ?></td>
        <td><?php echo $this->OrderStatus->getStatusString($order['Order']['working_state']); ?></td>
        <td><?php echo $this->OrderStatus->getStatusString($order['Order']['ready_state']); ?></td>
        <td> <?php //echo $this->Html->link($this->Html->image("cancel.jpg"), array('action' => 'index'), array('escape' => false));?><?php echo $this->Html->link($this->Html->image("edit.jpg"), array('action' => 'edit',$order['Order']['id']), array('escape' => false));/*echo $this->Html->link('Edit', array('action' => 'edit', $order['Order']['id']));*/?></td>
    </tr>
    <?php endforeach; ?>
    <tr ><?php //echo $this->Paginator->numbers(array('first' => 'First page')); ?></tr>
    <?php
    $urlParams = $this->params['url'];
    unset($urlParams['url']);
    $optionss=array('url' => array('?' => http_build_query($urlParams)));
    //$this->Paginator->options($optionss);
    $this->Paginator->settings['paramType'] = 'querystring';

    ?>

    <tr ><?php  //echo" << ".$this->Paginator->counter(
       // 'Page {:page} of {:pages}');
        ?></tr>

</table>

 <?php }else echo"no results found"; ?>


<?php } //endif?>

и това е моето действие на контролера

function search($options=null){
        $this->set('results',"");
        $this->set('term',"");
        $this->set('rs',0);    

        if ((isset($_GET["SearchTerm"]))||(isset($_GET["options"])) ) {
            $SearchTerm=$_GET["SearchTerm"];
            $options=$_GET["options"];

            if (!$options || !$SearchTerm) {
                $this->Session->setFlash('Please enter something to search for');
            }
            else {
                $SearchArray = array($options." LIKE " => "%".$SearchTerm."%");

                $this->paginate = array('conditions' =>  $SearchArray,'limit'=>2,'convertKeys' => array($options, $SearchTerm));
                $data=$this->paginate('Order');
                $this->set('rs', $data);
                $this->set('term',$SearchTerm);

            }
        }

както можете да видите, използвам опции за получаване на параметри и SearchTerm. Искам тези две да останат във връзките към страниците. Опитах различни корекции, които намерих в stackoverflow и на други сайтове (като например:

$urlParams = $this->params['url'];
    unset($urlParams['url']);
    $optionss=array('url' => array('?' => http_build_query($urlParams)));

но все още получавам следното съобщение за грешка:

Непряка промяна на претоварено свойство PaginatorHelper::$settings няма ефект [APP\View\orders\search.ctp, ред 75

Защо така? и какво ще кажете за решение на това? (дори ако използвам wxample от готварската книга

$this->Paginator->settings['paramType'] = 'querystring';

все още получавам същата грешка :S може ли да ми помогнете/обясните?


person George    schedule 09.02.2012    source източник
comment
Има както PaginatorComponent, така и PaginatorHelper. $settings е свойство на компонента, така че $this->Paginator->settings трябва да се премести във вашия контролер. От това идва грешката на претовареното свойство.   -  person Wylie    schedule 10.02.2012


Отговори (1)


Ако разбирам кода ви правилно, трябва да можете да замените:

<?php
    $urlParams = $this->params['url'];
    unset($urlParams['url']);
    $optionss=array('url' => array('?' => http_build_query($urlParams)));
    //$this->Paginator->options($optionss);
    $this->Paginator->settings['paramType'] = 'querystring';
?>

с

<?php $this->Paginator->options(array('url' => $this->passedArgs)); ?>

Полета на формуляра в пагинацията Ако всъщност се опитвате да поставите елементите на формуляра в пагинацията, ще трябва леко да промените кода си. Ще трябва да създадете именувани параметри за всеки елемент в опциите за търсене. След това, в действието на контролера, ще трябва да проверите както за заявката->данни, така и за params['named'] версиите на търсенето, за да сте сигурни, че ги използвате и в двата случая. Също така бих изчистил кода, така че да е малко по-четлив.

Първо, трябва да използвате методите за торта за получаване на данните. Той ще се увери, че изчиства заявките вместо вас и т.н. Освен това ще трябва да отчетете думата за търсене и опциите за филтриране, които се предават като наименована променлива и като изпратени във формуляр. Така че трябва да актуализирате вашия контролер, както следва:

function search($options=null){

    $this->set('results',""); // where is this used ??
    $this->set('rs', 0); 
    $this->set('SearchTerm', '');
    $this->set('FilterBy', '');   
    $this->set('options', array('full_name'=>'By Name','code'=>'By Code','phone'=>'By Phone','email'=>'By Mail'));

    if ($SearchTerm = $this->request->data['Order']['SearchTerm']) or $SearchTerm = $this->params['named']['SearchTerm']) {
        if ($FilterBy = $this->request->data['Order']['FilterBy'] or $FilterBy = $this->params['named']['FilterBy'])) {
            $this->paginate = array(
                'conditions' => array($FilterBy." LIKE " => "%".$SearchTerm."%"), 
                'limit' => 2, 
                'convertKeys' => array($FilterBy, $SearchTerm)
            );
            $this->set('rs', $this->paginate('Order'));
            $this->set('SearchTerm', $SearchTerm);
            $this->set('FilterBy', $FilterBy);
        } else {
            $this->Session->setFlash('Please enter something to search for');
        }
    } else {
        $this->Session->setFlash('Please enter something to search for');
    }
}

След това се фокусираме върху гледката. Премахни премести това:

$options=array('full_name'=>'By Name','code'=>'By Code','phone'=>'By Phone','email'=>'By Mail');

Не трябва да е в изгледа. Принадлежи към контролера (където е сега).

След това почистете формата:

<?php
  echo $this->Form->create('Order', array('action' => 'search','type'=>'get'));
  echo $this->Form->input('SearchTerm',array('label' => 'Find:'));
  echo $this->Form->input('FilterBy', array('type'=>'select', 'label'=>'Search:', 'options'=>$options));
  echo $this->Form->end('search'); 
?>  

* Имайте предвид, че промених името options на FilterBy, така че да е по-лесно за намиране в контролера. Освен това има други неща, които могат да бъдат изчистени в изгледа. Аз обаче ще се спра само на нещата, които отговарят на въпроса.

Сега трябва да замените този код:

<tr ><?php //echo $this->Paginator->numbers(array('first' => 'First page')); ?></tr>
    <?php
    $urlParams = $this->params['url'];
    unset($urlParams['url']);
    $optionss=array('url' => array('?' => http_build_query($urlParams)));
    //$this->Paginator->options($optionss);
    $this->Paginator->settings['paramType'] = 'querystring';

    ?>

    <tr ><?php  //echo" << ".$this->Paginator->counter(
       // 'Page {:page} of {:pages}');
        ?></tr>

</table>

С този код:

</table>
<p>
<?php
  $this->Paginator->options(array('url' => array_merge(array('SearchString' => $SearchString, 'FilterBy' => $FilterBy), $this->passedArgs)));

echo $this->Paginator->counter(array(
'format' => __('Page %page% of %pages%, showing %current% records out of %count% total, starting on record %start%, ending on %end%')
));
?>  
</p>

<div class="paging">
    <?php echo $this->Paginator->prev('<< ' . __('previous'), array(), null, array('class'=>'disabled'));?>
 |  <?php echo $this->Paginator->numbers();?>
 |  <?php echo $this->Paginator->next(__('next') . ' >>', array(), null, array('class' => 'disabled'));?>
</div>

Можете да го форматирате по различен начин, разбира се, за да отговаря на вашите нужди. Но кодът трябва да работи както се очаква с думата за търсене и опцията за филтър, приложена към пагинацията.

Успех и приятно кодиране!

person Chuck Burgess    schedule 10.02.2012
comment
благодаря :) всъщност съм нов в MVC и все още не съм разбрал много неща в cakephp, но отговорът ви беше наистина полезен и изчерпателен :) - person George; 10.02.2012
comment
Удоволствието е мое! Обичам торта и знам каква е бариерата за влизане. Въпреки това знам силата му и обичам да насърчавам хората да се придържат към него. Ще се отплати в дългосрочен план. - person Chuck Burgess; 10.02.2012