python web 分頁組件

来源:https://www.cnblogs.com/wang-yi-shan/archive/2019/10/07/11629210.html
-Advertisement-
Play Games

閑來無事便寫了一個易使用,易移植的Python Web分頁組件。使用的技術棧是Python、Django、Bootstrap。 既然是易使用、易移植的組件,首先介紹一下其在django框架中的調用方式吧。我將組件封裝成了Django InclusionTag,在template模板中直接調用tag即 ...


閑來無事便寫了一個易使用,易移植的Python Web分頁組件。使用的技術棧是Python、Django、Bootstrap。


既然是易使用、易移植的組件,首先介紹一下其在django框架中的調用方式吧。我將組件封裝成了Django InclusionTag,在template模板中直接調用tag即可,代碼如下:

{% load pager_tags %}

{% pager request 100 10 %}

   其中 pager_tags是封裝的tag文件名,這個無需多說,不瞭解django框架custom filter、tag的可在官方文檔查看。

pager是註冊的tag名,後面是傳遞到pager的參數。request是django的請求對象,100是最大頁數,10是導航頁碼的個數。

可以看到,整個組件的調用是非常簡單的,比起django自帶的分頁器paginator要輕巧很多,這也是我決定開發這個組件的主要原因之一。當然組件使用的便利性帶來的是靈活度的犧牲,其擴展能力十分有限,而且依賴於前端框架Django。所以,想要定製一些特別的功能或者樣式的小伙伴還是應該選擇paginator自行開發。


接下來我們看看tag是怎麼寫的吧。

from django import template

from garra_rbac.services import page_service

register = template.Library()


@register.inclusion_tag('garra_pager.html')
def pager(request, max_index, show_count=5):
    _page = request.GET.get('page')
    page = 1
    if _page:
        page = int(_page)
    param = request.GET.urlencode()
    page_obj = page_service.PageObject(page, max_index, param, show_count)
    return {'page_obj': page_obj}

  我在tag中並沒有做太多的事情,只是處理了當前頁面的請求參數。其中當前頁預設為0,並把GET的參數傳遞下去,當點擊其他頁碼時要保留參數條件。


我們再看看garra_pager.html吧,html頁面是依賴於Bootstrap開發的,不過對於樣式的修改也很容易實現。

{% if page_obj.has_page_item %}
    <nav aria-label="Page navigation">
        <ul class="pagination">
            {% for page_index in page_obj.page_index_list %}
                <li {% if page_obj.current_index == page_index.index %}class="active"{% endif %}>
                    <a href="{{ page_index.url }}"
                       class="{{ page_obj.current_index }}">{{ page_index.text }}</a>
                </li>
            {% endfor %}
        </ul>
    </nav>

{% else %}
    <div class="alert alert-danger" role="alert">
        <span class="glyphicon glyphicon-exclamation-sign" aria-hidden="true"></span>
        <span class="sr-only">Error:</span>
        暫無數據
    </div>
{% endif %}

  html代碼很好理解,都是數據的展示,並添加了暫無數據的提示。


最後我們看看分頁組件功能的主要實現部分,page_service的代碼。

import math


def get_lhalf(show_count):
    """
    獲取當前頁的左側最多右多少個頁碼
    :param show_count:展示的頁碼個數
    :return:當前頁的左側最多右多少個頁碼
    """
    if show_count % 2 == 0:  # 如果為偶數,則讓當前頁的左側頁碼個數比右側少1
        return show_count / 2 - 1
    else:  # 如果為奇數,左右側頁碼個數相同
        return math.floor(show_count / 2)


def get_param(param, index):
    '''
    保留原搜索條件,並更新page當前頁條件
    :param param:原搜索條件
    :param index:當前頁
    :return:更新當前頁後的搜索條件
    '''
    if param:
        param = '?' + param
        if '?page' in param or '&page' in param:
            import re
            param = re.sub(r'page=\d+', 'page=%s' % index, param)
        else:
            param += '&page=%s' % index
    else:
        param = "?page=%s" % index
    return param


class PageObject(object):
    def __init__(self, current_index, max_index, param, show_count=5):
        '''
        分頁組件對象
        :param current_index:當前頁碼
        :param max_index: 最大頁碼
        :param param: 搜索條件
        :param show_count: 展示的頁碼個數
        '''
        self.current_index = current_index
        self.max_index = max_index
        self.show_count = show_count if show_count > 5 else 5
        self.page_index_list = list()
        self.__set_page_index_list(param)

    @property
    def has_page_item(self):
        """
        是否有頁碼數據
        :return:
        """
        return self.max_index > 0

    def __set_page_index_list(self, param):
        '''
        設置頁碼按鈕
        :param param:原搜索條件
        :return:
        '''
        lhalf = get_lhalf(self.show_count)  # 左半邊應有的數量

        l_temp_index = self.current_index - lhalf  # 臨時左頁碼
        _roffest = 0  # 右遷移量
        if l_temp_index < 1:  # 左頁碼需大於0,否則應向右偏移
            _roffest = -l_temp_index + 1
        lindex = l_temp_index + _roffest  # 左頁碼 = 臨時左頁碼向左偏移後的值

        r_temp_index = lindex + self.show_count - 1  # 臨時右頁碼
        _loffest = 0  # 左位移量
        if self.max_index - r_temp_index < 0:  # 右頁碼需大於最大頁碼,否則應向左偏移
            _loffest = r_temp_index - self.max_index
            lindex = lindex - _loffest  # 左頁碼向左偏移
            if lindex < 1:  # 左頁碼偏移後的最小值為1,當max_index<show_count時 多餘部分需捨棄
                lindex = 1
        rindex = r_temp_index - _loffest  # 右頁碼 = 臨時右頁碼向右偏移後的值

        # region 添加按鈕

        # 首頁按鈕
        page_first = PageIndex(1, '首頁', param)
        page_first.index = -1  # 此處設置是為了避免page=1時首頁按鈕被標記為選中狀態的情況,尾頁同理
        self.page_index_list.append(page_first)

        # 上一頁按鈕,如果當前頁是第一頁則不顯示
        if self.current_index > 1:
            page_prev = PageIndex(self.current_index - 1, '上一頁', param)
            self.page_index_list.append(page_prev)

        # 快速後退跳轉按鈕
        if lindex > lhalf:
            page_left_skip = PageIndex(lindex - lhalf, "‧‧‧", param)
            self.page_index_list.append(page_left_skip)

        # 頁碼按鈕
        for index in range(lindex, rindex + 1):
            page_index = PageIndex(index, str(index), param)
            self.page_index_list.append(page_index)

        # 快速前進跳轉按鈕
        if self.max_index - rindex > lhalf:
            page_left_skip = PageIndex(rindex + lhalf, "‧‧‧", param)
            self.page_index_list.append(page_left_skip)

        # 下一頁按鈕,如果當前頁是最後一頁則不顯示
        if self.current_index < self.max_index:
            page_next = PageIndex(self.current_index + 1, '下一頁', param)
            self.page_index_list.append(page_next)

        # 尾頁按鈕
        page_last = PageIndex(self.max_index, '尾頁', param)
        page_last.index = -1
        self.page_index_list.append(page_last)

        # endregion


class PageIndex(object):
    def __init__(self, index, text, param):
        self.index = index
        self.text = text
        self.url = get_param(param, index)

  整個page_service主要分為兩個功能,第一是求出組件的最左側頁碼和最右側頁碼的數值,第二是添加全部按鈕給template渲染。

稍微解釋一下演算法,我首先取到了當前頁左側頁碼數的最大值,並求出此時最左側的頁碼值,如果其小於1 則整個頁碼組件需要向右側移動 直至最左側的頁碼為1。爾後計算出最右側的頁碼,並和最大頁碼進行比較,如果最右側頁碼大於最大頁碼,則整個頁碼組件需要向左側移動。註:由於最左側頁碼已經算出,如需左移組件,最左側頁碼也應該進行移動,並應註意 最左側頁碼不能小於1。

此時我們拿到了分頁組件的最左側頁碼和最右側頁碼,就可以添加按鈕併進行渲染了。


如此一來,分頁所需要的主要功能就都被我們實現了,而且調用簡單,源碼簡潔,為此我犧牲了擴展性,使其和paginator分頁器有一個極為明顯的差別,可根據不同場景進行選擇。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 要向HTML DOM添加新元素,必須首先創建元素(元素節點),然後將其附加到現有元素。 appendChild()上一個示例中的方法將新元素作為父項的最後一個子項附加。如果您不希望可以使用insertBefore()方法: 要將元素替換為HTML DOM,請使用以下replaceChild()方法: ...
  • 背景 事情是這樣的。一天下午4點42分左右。業務反饋我開發的服務在測試環境出現問題,返回資源數據是0。查日誌發現是ES訪問超時。相當於資料庫掛了。持續了20多分鐘自己恢復。咨詢了ES團隊,最終得到下麵的答覆: 調查 1.需要換成本地磁碟,測試環境也是我們的正式環境。是否能直接替換成物理機?多少台合適 ...
  • 聲明:本文是一個系列原創(作者在GIS+BIM行業已有從業15年有餘,還是個行業的小學生,文章內容不免有錯誤或者不當之處,敬請理解),旨在通過這個系列打造一個高性能,高可擴展的GIS+BIM框架,拋磚引玉,為國內GIS+BIM行業貢獻綿薄之力。 對於行業內的人說到GIS、BIM最先想到是:引擎,是的 ...
  • 觀察者模式(Observer): 指多個對象間存在一對多的依賴關係,當一個對象的狀態發生改變時,所有依賴於它的對象都得到通知並被自動更新。 觀察者模式的角色: 1)抽象目標(Subject):也叫抽象目標類,它提供了一個用於保存觀察者對象的聚集類和增加、刪除觀察者對象的方法,以及通知所有觀察者的抽象 ...
  • 責任鏈模式: 下圖為責任鏈 1、定義:為了避免請求發送者與多個請求處理者耦合在一起,將所有請求的處理者通過前一對象 記住其下一個對象的引用而連成一條鏈;當有請求發生時,可將請求沿著這條鏈傳遞,直到有對象處理它為止 2、模型結構: (1)抽象處理者(Handler):定義一個處理請求的介面,包含抽象處 ...
  • kafka的術語(Terminology) Topic 和Consumer Group Topic 每條發佈到 Kafka 集群的消息都有一個類別,這個類別被稱為 Topic。(物理上不同 Topic 的消息分開存儲,邏輯上一個 Topic 的消息雖然保存於一個或多個 broker 上但用戶只需指定 ...
  • 適配器模式: 類適配器: 對象適配器: 1、定義:將一個介面轉換成客戶希望的另一個介面,適配器模式使介面不相容的那些類可以一起工作 2、模型結構: (1)目標抽象類(Target):客戶所期待得到的介面 (2)適配器類(Adapter):通過包裝一個需要適配的對象,把原介面轉換成目標介面 (3)適配 ...
  • 單例模式: 1、定義:單例模式確保某一個類只有一個實例,而且自行實例化並向整個系統提供這個實例,這個類稱為單例類 2、實現過程要點: (1)單例類的構造函數為私有 (2)提供一個自身的靜態私有成員變數 (3)提供一個公有的靜態工廠方法 3、優點: (1)提供了對唯一實例的受控訪問 (2)由於在系統內 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...