PHP / MySQL - разбиение на страницы с различными предложениями ORDER BY - PullRequest
2 голосов
/ 16 августа 2011

У меня есть следующие запросы: первый в качестве счетчика для нумерации страниц, второй для заполнения результирующей таблицы данными.Они оба работали очень хорошо, но я хочу добавить функциональность упорядочения и к столу, и приложил немало усилий в этом направлении.

По сути, я разбил запрос MySQL на части, хранящиеся в переменных, которые могут назначаться или не назначаться в зависимости от пожеланий пользователя, выраженных в отправке и получении форм.Я только на начальном уровне, и это то, что логично для меня.Вероятно, это неправильный метод!

Данные упорядочены по умолчанию, так как они вводятся в базу данных.Таблица заполняется 15 строками за раз.В идеале пользователь мог бы затем заказать данные о животных другим способом - например, по возрасту или цвету.Выбор делается путем отправки формы на саму страницу, запрос повторно отправляется, и результат снова выводится на 15-строчные страницы.

Итак, вот мой код:

PAGINATION:

$genderid = $_GET['groupid'];
$sortby = $_GET['sortby'];

##################

if (isset($_GET['pageno'])) { $pageno = $_GET['pageno']; }
    else { $pageno = 1; }

    $initialcount = "SELECT count(*) 
                FROM profiles
                WHERE ProfileName != 'Unknown'";

        if ($genderid > 0) {
            $genderquery = " AND ProfileGenderID = $genderid";
            }                 
        if ($sortby == 'age') {
            $orderby = " ORDER BY ProfileYearOfBirth ASC";
            } 
        elseif ($sortby == 'colour') {
            $orderby = " ORDER BY ProfileAdultColourID ASC";
            }

    $finalcount = ($initialcount . ' ' . $genderquery . ' ' . $orderby);

    $result = mysql_query($finalcount) or trigger_error();
    $query_data = mysql_fetch_row($result);
    $numrows = $query_data[0];

    $rows_per_page = 15;
    $lastpage = ceil($numrows/$rows_per_page);

    $pageno = (int)$pageno;
    if ($pageno > $lastpage) {
        $pageno = $lastpage;
    }
    if ($pageno < 1) {
        $pageno = 1;
    }

    $limit = 'LIMIT ' .($pageno - 1) * $rows_per_page .',' .$rows_per_page;


    if ($pageno == 1) 
        {
            echo '<p class="pagination">';
            echo '<span class="first"><< First</span><span class="previous">< Previous</span>';
        } 
    else    
        {
            echo '<p class="pagination">';
            echo "<span class='first'><a href='{$_SERVER['PHP_SELF']}?pageno=1'><< First</a></span>";
            $prevpage = $pageno-1;
            echo "<span id='class'><a href='{$_SERVER['PHP_SELF']}?pageno=$prevpage'>< Previous</a></span>";
        }

    echo '<span class="pagination-nav">' . "Page $pageno of $lastpage" . '</span>';

    if ($pageno == $lastpage) 
        {
            echo '<span class="next">Next ></span><span class="last">Last >></span>';
            echo '</p>';
        } 
    else 
        {
            $nextpage = $pageno+1;
            echo "<span class='next'><a href='{$_SERVER['PHP_SELF']}?pageno=$nextpage'>Next ></a></span>";
            echo "<span class='last'><a href='{$_SERVER['PHP_SELF']}?pageno=$lastpage'>Last >></a></span>";
            echo '</p>';
        } 

OUTPUT:

     <table class="admin-display">
    <thead>
        <tr>
            <th>Name:</th>
            <th>Age:
                <form class="sorting-form" method="GET" action="">
                <input type="hidden" name="sortby" value="age" />
                <input type="hidden" name="groupid" value="<?php echo $genderid; ?>" />
                <input type="submit" value="&#8657;" class="sort-submit" />
                </form>
            </th>
            <th>Colour:
                <form class="sorting-form" method="GET" action="">
                <input type="hidden" name="sortby" value="colour" />
                <input type="hidden" name="groupid" value="<?php echo $genderid; ?>" />
                <input type="submit" value="&#8657;" class="sort-submit" />
                </form>
            </th>
            <th>Gender:</th>
            <th>Owner:</th>
            <th>Breeder:</th>
        </tr>
    </thead>    

<?php

    $initialquery = "SELECT ProfileID, ProfileOwnerID, ProfileBreederID, 
            ProfileGenderID, ProfileAdultColourID, ProfileColourModifierID, ProfileYearOfBirth, 
            ProfileYearOfDeath, ProfileLocalRegNumber, ProfileName,
            owner.ContactFirstName AS owner_fname, owner.ContactLastName AS owner_lname,
            breeder.ContactFirstName AS breeder_fname, breeder.ContactLastName AS breeder_lname,
            BreedGender, BreedColour, BreedColourModifier
            FROM profiles
                        LEFT JOIN contacts AS owner
                            ON ProfileOwnerID = owner.ContactID
                        LEFT JOIN contacts AS breeder
                            ON ProfileBreederID = breeder.ContactID
                LEFT JOIN prm_breedgender
                            ON ProfileGenderID = prm_breedgender.BreedGenderID
                        LEFT JOIN prm_breedcolour
                            ON ProfileAdultColourID = prm_breedcolour.BreedColourID
                        LEFT JOIN prm_breedcolourmodifier
                            ON ProfileColourModifierID = prm_breedcolourmodifier.BreedColourModifierID
                      WHERE ProfileName != 'Unknown'";

$finalquery = ($initialquery . ' ' . $genderquery . ' ' . $orderby . ' ' . $limit);

$result = mysql_query($finalquery) or trigger_error("SQL", E_USER_ERROR);
//process results

Данные по-прежнему выводятся правильно (изначально), и пользователь также может повторно успешно отправить запрос.Проблема возникает, когда - после упорядочивания по чему-то другому, чем по умолчанию - я нажимаю, чтобы перейти на следующую страницу.Как только страница изменится, порядок вернется к значению по умолчанию.Я не знаю, как сохранить это предложение ORDER BY после первоначальной повторной отправки.

Это все, что я получил до того, как начал взламывать код, и все стало волосатым.Мне нужно, чтобы кто-то указал на явную ошибку!Большое спасибо заранее :)

Ответы [ 3 ]

1 голос
/ 16 августа 2011

Похоже, что это должно быть обработано в вызове AJAX. Теоретически, вы можете сохранить свое начальное и увеличенное состояние на протяжении всего сеанса (используя переменную сеанса, скрытое поле формы или даже файл cookie), но я думаю, что вы будете более счастливы с одновременным захватом X записей и сохранением аргументов на стороне клиента, пока вы не хотите получить следующую страницу.

В соответствии с запросом автора, сохранение необходимых значений в сеансе будет выглядеть примерно так:

session_start();

$startat = intval($_REQUEST["startat"]);
$numrows= intval($_REQUEST["numrows"]);

if (isset($_SESSION['base_query'])){
    $query = $_SESSION['base_query'];
} else {
    $query = <query construction minus limit clause>;
    $_SESSION['base_query'] = $query;
}


$query .= " LIMIT $startat, $numrows";

<query db/ send results>

Это дает вам идею?

0 голосов
/ 16 августа 2011

Хорошо, я работал над тем, что у меня было еще несколько часов, и придумал решение. Он не использует AJAX или SESSIONS и, несомненно, ужасен, но подходит для уровня, на котором я нахожусь.

Во-первых, я изменил скрипт разбивки на страницы, чтобы придать другое значение параметру смещения:

$title = intval($_GET['groupid']);
$genderid = intval($_GET['groupid']);
$sortby = $_GET['sortby'];

##################################

    $initialcount = "SELECT count(*) FROM profiles WHERE ProfileName != 'Unknown'";

    if($genderid > 0) {
        $where = "AND ProfileGenderID = $genderid";
        $finalcount = ($initialcount . ' ' . $where);
        }
    else { $finalcount = $initialcount;  }

    $result = mysql_query($finalcount) or trigger_error("SQL", E_USER_ERROR);
    $r = mysql_fetch_row($result);
    $numrows = $r[0];

$rowsperpage = 15;
$totalpages = ceil($numrows / $rowsperpage);

if (isset($_GET['currentpage']) && is_numeric($_GET['currentpage'])) {
   $currentpage = (int) $_GET['currentpage'];
} 
    else { $currentpage = 1; }

if ($currentpage > $totalpages) {
   $currentpage = $totalpages;
} 

    if ($currentpage < 1) 
       {  $currentpage = 1; } 

$offset = ($currentpage - 1) * $rowsperpage;

Я также изменил нумерацию ссылок на страницы: а) удерживать значение текущей сортировки по полу (в целом) и последующей сортировки (возраст, цвет); и б) для отображения пронумерованных ссылок на страницы в Google:

// range of num links to show
$range = 3;

##############

        $range = 3;

        echo '<p class="pagination">';
        if ($currentpage == 1) 
        {
            echo '<span class="first"><< First</span><span class="previous">< Previous</span>';
        } 
        if ($currentpage > 1) {
            echo "<span class='first'><a href='{$_SERVER['PHP_SELF']}?sortby=$sortby&groupid=$genderid&currentpage=1'><< First</a></span>";
            $prevpage = $currentpage - 1;
            echo "<span class='previous'><a href='{$_SERVER['PHP_SELF']}?sortby=$sortby&groupid=$genderid&currentpage=$prevpage'>< Previous</a></span>";
        }

        echo '<span class="pagination-nav">';

        for ($x = ($currentpage - $range); $x < (($currentpage + $range) + 1); $x++) {
            if (($x > 0) && ($x <= $totalpages)) {
                    if ($x == $currentpage) {
                        echo " [<b>$x</b>] ";
                    } 
                    else {
                        echo " <a href='{$_SERVER['PHP_SELF']}?sortby=$sortby&groupid=$genderid&currentpage=$x'>$x</a> ";
                    }
            }
        }

        echo '</span>';

        if ($currentpage != $totalpages) {
               $nextpage = $currentpage + 1;
               echo "<span class='next'><a href='{$_SERVER['PHP_SELF']}?sortby=$sortby&groupid=$genderid&currentpage=$nextpage'>Next ></a></span>";
               echo "<span class='last'><a href='{$_SERVER['PHP_SELF']}?sortby=$sortby&groupid=$genderid&currentpage=$totalpages'>Last >></a></span>";
        }
        if ($currentpage == $totalpages) 
            {
                echo '<span class="next">Next ></span><span class="last">Last >></span>';
            } 

        echo '</p>';

И наконец построил запрос:

$baseQuery = "SELECT ProfileID, ProfileOwnerID, ProfileBreederID, 
                ProfileGenderID, ProfileAdultColourID, ProfileColourModifierID, ProfileYearOfBirth, 
                ProfileYearOfDeath, ProfileLocalRegNumber, ProfileName,
                owner.ContactFirstName AS owner_fname, owner.ContactLastName AS owner_lname,
                breeder.ContactFirstName AS breeder_fname, breeder.ContactLastName AS breeder_lname,
                BreedGender, BreedColour, BreedColourModifier
                FROM profiles
                            LEFT JOIN contacts AS owner
                                ON ProfileOwnerID = owner.ContactID
                            LEFT JOIN contacts AS breeder
                                ON ProfileBreederID = breeder.ContactID
                    LEFT JOIN prm_breedgender
                                ON ProfileGenderID = prm_breedgender.BreedGenderID
                            LEFT JOIN prm_breedcolour
                                ON ProfileAdultColourID = prm_breedcolour.BreedColourID
                            LEFT JOIN prm_breedcolourmodifier
                                ON ProfileColourModifierID = prm_breedcolourmodifier.BreedColourModifierID
                          WHERE ProfileName != 'Unknown'";


    if($sortby == 'age') {
        $orderby = "ORDER BY ProfileYearOfBirth ASC";
        }
    elseif ($sortby == 'colour') {
        $orderby = "ORDER BY ProfileAdultColourID ASC";
        }

    $limitClause = "LIMIT $offset, $rowsperpage";

    $finalQuery = ($baseQuery . ' ' . $where . ' ' . $orderby . ' ' . $limitClause);

$result = mysql_query($finalQuery) or trigger_error("SQL", E_USER_ERROR);

Эффективен как молоток, бьющий по чизкейку :) Большое спасибо авторам выше, особенно за то, что познакомили меня с INTVAL (и постепенно, CASTING).

0 голосов
/ 16 августа 2011

может быть, вы можете обойти эту проблему, используя http://www.datatables.net/?Это клиентское решение ajax, которое заказывает результаты для вас.И вы избегаете перезагрузки всей страницы каждый раз.

...