当前位置:首页 > CMS教程 > 其它CMS > 列表

Laravel5.5 手动分页和自定义分页样式的简单实现

发布:smiling 来源: PHP粉丝网  添加日期:2022-01-04 20:17:46 浏览: 评论:0 

今天小编就为大家分享一篇Laravel5.5 手动分页和自定义分页样式的简单实现,具有很好的参考价值,希望对大家有所帮助,一起跟随小编过来看看吧。

基于Laravel5.5 在项目实施过程中,需要对从接口中获取的数据(或者通过搜索工具查询出来的数据)进行分页

一、创建手动分页

在laravel自带的分页中,一般是通过数据库查询访问paginate()方法来达到分页的效果 ,like this:

class IndexControllerextends Controller

  1. {   
  2.   publicfunctionindex() 
  3.   { 
  4.     $person = DB::table('person')->paginate(15); 
  5.    
  6.     return view('index.pagTest',['person'=> $person]); 
  7.   } 

查看框架的分页源代码

#vender/laravel/framework/src/Illuminate/Database/Eloquent/Builder.php

  1. /** 
  2.    * Paginate the given query. 
  3.    * 
  4.    * @param int $perPage 
  5.    * @param array $columns 
  6.    * @param string $pageName 
  7.    * @param int|null $page 
  8.    * @return \Illuminate\Contracts\Pagination\LengthAwarePaginator 
  9.    * 
  10.    * @throws \InvalidArgumentException 
  11.    */ 
  12.   public function paginate($perPage = null, $columns = ['*'], $pageName = 'page'$page = null) 
  13.   { 
  14.     $page = $page ?: Paginator::resolveCurrentPage($pageName); 
  15.    
  16.     $perPage = $perPage ?: $this->model->getPerPage(); 
  17.    
  18.     $results = ($total = $this->toBase()->getCountForPagination()) 
  19.                   ? $this->forPage($page$perPage)->get($columns
  20.                   : $this->model->newCollection(); 
  21.    
  22.     return $this->paginator($results$total$perPage$page, [ 
  23.       'path' => Paginator::resolveCurrentPath(), 
  24.       'pageName' => $pageName
  25.     ]); 
  26.   } 

发现,分页用了 \Illuminate\Contracts\Pagination\LengthAwarePaginator 构造方法,查看这个构造方法

  1. <?php 
  2.    
  3. namespace Illuminate\Pagination; 
  4.    
  5. use Countable; 
  6. use ArrayAccess; 
  7. use JsonSerializable; 
  8. use IteratorAggregate; 
  9. use Illuminate\Support\Collection; 
  10. use Illuminate\Support\HtmlString; 
  11. use Illuminate\Contracts\Support\Jsonable; 
  12. use Illuminate\Contracts\Support\Arrayable; 
  13. use Illuminate\Contracts\Pagination\LengthAwarePaginator as LengthAwarePaginatorContract; 
  14.    
  15. class LengthAwarePaginator extends AbstractPaginator implements Arrayable, ArrayAccess, Countable, IteratorAggregate, JsonSerializable, Jsonable, LengthAwarePaginatorContract 
  16.   /** 
  17.    * The total number of items before slicing. 
  18.    * 
  19.    * @var int 
  20.    */ 
  21.   protected $total
  22.    
  23.   /** 
  24.    * The last available page. 
  25.    * 
  26.    * @var int 
  27.    */ 
  28.   protected $lastPage
  29.    
  30.   /** 
  31.    * Create a new paginator instance. 
  32.    * 
  33.    * @param mixed $items 
  34.    * @param int $total 
  35.    * @param int $perPage 
  36.    * @param int|null $currentPage 
  37.    * @param array $options (path, query, fragment, pageName) 
  38.    * @return void 
  39.    */ 
  40.   public function __construct($items$total$perPage$currentPage = null, array $options = []) 
  41.   { 
  42.     foreach ($options as $key => $value) { 
  43.       $this->{$key} = $value
  44.     } 
  45.    
  46.     $this->total = $total
  47.     $this->perPage = $perPage
  48.     $this->lastPage = max((int) ceil($total / $perPage), 1); 
  49.     $this->path = $this->path !== '/' ? rtrim($this->path, '/') : $this->path; 
  50.     $this->currentPage = $this->setCurrentPage($currentPage$this->pageName); 
  51.     $this->items = $items instanceof Collection ? $items : Collection::make($items); 
  52.   } 

如果要实现手动分页,只需要使用这个构造方法,给定参数,就能达到分页的效果

贴代码:

  1. public function setPage2(Request $request,$data,$prepage,$total){ 
  2.  
  3.   #每页显示记录 
  4.   $prePage = $prepage
  5.   //$total =count($data); 
  6.   $allitem = $prepage *100; 
  7.   $total > $allitem ? $total = $allitem : $total
  8.   if(isset($request->page)){ 
  9.     $current_page =intval($request->page); 
  10.     $current_page =$current_page<=0?1:$current_page
  11.   }else
  12.     $current_page = 1; 
  13.   } 
  14.   #url操作 
  15.   $url = $url='http://'.$_SERVER['SERVER_NAME'].$_SERVER["REQUEST_URI"]; 
  16.   if(strpos($url,'&page')) $url=str_replace('&page='.$request->page, '',$url); 
  17.  
  18.   # $data must be array 
  19.   $item =array_slice($data,($current_page-1)*$prePage,$prePage); 
  20.   $paginator = new LengthAwarePaginator($item,$total,$prePage,$current_page,[ 
  21.     'path'=>$url
  22.     'pageName'=>'page' 
  23.   ]); 
  24.  
  25.   return $paginator

($data 为需要进行分页的数据)

说明:

1、在考虑到代码的复用性,我将分页代码封装到app/Controllers/Controller.php中的一个方法里面,这样在其他控制器里只需要$this->setPage(Request $request,$data,$prepage,$total) 就能使用了,(前提:其他控制器继承了Controller.php)

2、分页的URL,因为我的项目的url一定会携带一个kw参数,所以我直接用str_replace替换"&page",如果是存在不携参分页的话,需要判断,到底是"?page"还是"&page"。(url的逻辑可以自己写)

#分页 php

$paginator = $this->setPage2($request,$data,25,$sum);

$data =$paginator->toArray()['data'];

在模板中:{{$paginator->render()}}即能输出分页HTML,样式如下:

Laravel5.5手动分页 自定义分页

二、自定义分页样式

在实际开发中,不希望用户在浏览时直接浏览最后几页,只想用户从前往后依次的浏览,如百度搜索分页,这时候,就想修改分页的样式,经过一个下午的奋战,贴出解决过程

在上一环节中,手动创建了分页,了解HTML的模板生成是render()方法,

#\Illuminate\Contracts\Pagination\LengthAwarePaginator

  1. /** 
  2.    * Render the paginator using the given view. 
  3.    * 
  4.    * @param string|null $view 
  5.    * @param array $data 
  6.    * @return \Illuminate\Support\HtmlString 
  7.    */ 
  8.   public function render($view = null, $data = []) 
  9.   { 
  10.     return new HtmlString(static::viewFactory()->make($view ?: static::$defaultViewarray_merge($data, [ 
  11.       'paginator' => $this
  12.       'elements' => $this->elements(), 
  13.     ]))->render()); 
  14.   } 

经过思考,我们不去改laravel框架的源代码,可以通过重构render方法或者重新定义一个生成HTML模板的方法来实现自定义HTML模板

因为我们只需要自定义HTML模板,所以,可以创建一个文件,继承\Illuminate\Contracts\Pagination\LengthAwarePaginator 类

看代码:

  1. <?php 
  2.    
  3. namespace App\Helpers; 
  4.    
  5. use Illuminate\Pagination\LengthAwarePaginator; 
  6. /** 
  7.  * Created by PhpStorm. 
  8.  * User: 1 
  9.  * Date: 2018/4/9 
  10.  * Time: 9:08 
  11.  */ 
  12. class Newpage extends LengthAwarePaginator { 
  13.    
  14.   public $de_page = 10; //默认显示分页数 
  15.   public $pageHtml; 
  16.    
  17.   public function newrender(){ 
  18.     if($this->hasPages()) 
  19.     { 
  20.    
  21.       return sprintf("<ul class='pagination'>%s %s %s</ul>"
  22.         $this->pre_page(), 
  23.         $this->pages_num(), 
  24.         $this->next_page() 
  25.       ); 
  26.     } 
  27.   } 
  28.    
  29.   #上一页 
  30.   public function pre_page(){ 
  31.     if($this->currentPage == 1){ 
  32.       //dd($this->currentPage); 
  33.       return "<li class='disabled'><span>《</span></li>"
  34.     }else
  35.    
  36.       $url = $this->path."&page=".($this->currentPage-1); 
  37.       //dd($url); 
  38.       return "<li><a href=".$url." rel="external nofollow" rel='prev'>《</a></li>"
  39.     } 
  40.   } 
  41.    
  42.   #页码 
  43.   public function pages_num(){ 
  44.     $pages = ''
  45.     if($this->currentPage <= 6){ 
  46.       for($i = 1; $i <= $this->de_page; $i++){ 
  47.         if($this->currentPage == $i){ 
  48.           $pages .= "<li class='active'><a href=".$this->path." rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" &page=".$i.">".$i."</a></li>"
  49.         }else
  50.           $pages .="<li><a href=".$this->path." rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" &page=".$i.">".$i."</a></li>"
  51.         } 
  52.       } 
  53.     }else
  54.       #当前页前边部分 
  55.       for($i = 5; $i >=1 ; $i--){ 
  56.         $url =$this->currentPage-$i; 
  57.         $pages .= "<li><a href=".$this->path." rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" &page=".$url.">".$url."</a></li>"
  58.       } 
  59.       #当前页 
  60.       $pages .= "<li class='active'><span>".$this->currentPage."</span></li>"
  61.       #当前页后边部分 
  62.       for($i = 1;$i < 5; $i++ ){ 
  63.         $nowpage =$this->currentPage+$i; 
  64.         $pages .= "<li><a href=".$this->path." rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" &page=".$nowpage.">".$nowpage."</a></li>"
  65.       } 
  66.     } 
  67.     return $pages; 
  68.    
  69.   } 
  70.   #下一页 
  71.   public function next_page(){ 
  72.     if($this->currentPage < $this->total){ 
  73.       $page =$this->currentPage+1; 
  74.       return "<li><a href=".$this->path." rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" rel="external nofollow" &page=".$page." rel='next'><span>》</span></a></li>"
  75.     }else
  76.       return "<li class='disabled'><span>》</span></li>"
  77.     } 
  78.    
  79.   } 
  80.    

我选择的方法是自定义新的方法生成HTML模板,模板中通过:{{$paginator->newrender()}}输出HTML

如果选择重构render()方法,只需要将上面的newrender()方法做一些小变动

  1. public function render($view=null,$data=[]){ 
  2.    
  3.     if($this->hasPages()) 
  4.     { 
  5.       return sprintf("<ul class='pagination'>%s %s %s</ul>"
  6.         $this->pre_page(), 
  7.         $this->pages_num(), 
  8.         $this->next_page() 
  9.       ); 
  10.     } 
  11.   } 

模板中通过:{{$paginator->render()}}输出HTML

最终效果如图:

Laravel5.5手动分页 自定义分页

注意:自定义HTML后因为新建了一个类继承了LengthAwarePaginator类,需要将第一步手动分页的方法中new LengthAwarePaginator 修改为 new Newpage 参数不变。

Tags: Laravel5 5手动分页 自定义分页

分享到: