README.md

# PaginationEx

A flexible and robust pagination library for Elixir/Phoenix applications that integrates seamlessly with Ecto.

## Features

- **Simple Integration**: Easily paginate Ecto queries with minimal configuration
- **Flexible Rendering**: HTML helpers with Tailwind CSS styling by default
- **Customizable**: Support for custom templates and styling
- **Internationalization**: Built-in i18n support via Gettext
- **Performance Optimized**: Smart counting strategies for different query types

## Installation

The package is not yet available on Hex.pm. To install directly from GitHub, add to your dependencies:

```elixir
def deps do
  [
    {:pagination_ex, github: "idopterlabs/pagination_ex"}
  ]
end
```

## Configuration

Configure PaginationEx in your application's config file:

```elixir
# In config/config.exs or environment-specific config files
config :pagination_ex,
  repo: MyApp.Repo,                        # Required
  per_page: 25,                            # Optional, defaults to 30
  per_group: 500,                          # Optional, defaults to 1000
  gettext_module: MyApp.Gettext,           # Optional for internationalization
  router_helpers: MyAppWeb.Router.Helpers  # Optional for route generation
```

## Usage

### Basic Pagination

```elixir
# In your controller
def index(conn, params) do
  pagination = 
    MyApp.Posts
    |> PaginationEx.new(params)
  
  render(conn, "index.html", pagination: pagination)
end
```

### In Templates

```elixir
# In your template
<%= for post <- @pagination.entries do %>
  <div class="post">
    <h2><%= post.title %></h2>
    <p><%= post.content %></p>
  </div>
<% end %>

<div class="pagination">
  <%= PaginationEx.paginate(@conn, :post_path) %>
</div>
```

### Batch Processing with Groups

For processing large datasets in batches:

```elixir
# This fetches all items in batches of the configured size
all_items = PaginationEx.in_groups(MyApp.Items)

# Process all items
Enum.each(all_items, fn item ->
  # Process each item
end)
```

### Custom Templates

Create a custom template module:

```elixir
defmodule MyApp.CustomPaginationTemplate do
  use PhoenixHTMLHelpers
  import PaginationEx.HTML, only: [translate: 1, build_url: 3]
  
  def render_pagination(conn, path, pagination) do
    # Your custom rendering logic here
  end
end
```

Then use it in your templates:

```elixir
<%= PaginationEx.paginate(@conn, :post_path, template_module: MyApp.CustomPaginationTemplate) %>
```

## API Documentation

### PaginationEx.Core

The `Core` module handles the pagination logic and query execution.

Key functions:
- `new/3`: Creates a new pagination struct from an Ecto query
- `in_groups/2`: Retrieves all records in batches of specified size

### PaginationEx.HTML

The `HTML` module renders pagination controls in templates.

Key functions:
- `paginate/3`: Renders complete pagination controls
- `page_links/3`: Generates numbered page links
- `previous_path/3` and `next_path/4`: Creates previous/next navigation links

### PaginationEx.Config

The `Config` module handles configuration retrieval.

## Pagination Structure

The pagination result is a struct with the following fields:

```elixir
%PaginationEx.Core{
  entries: [%Post{}, %Post{}, ...],  # The current page's items
  total_entries: 59,                 # Total number of items
  page_number: 2,                    # Current page number
  per_page: 10,                      # Items per page
  pages: 6,                          # Total number of pages
  query: #Ecto.Query<...>            # The original query
}
```

## Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

1. Fork the repository
2. Create your feature branch (`git checkout -b feature/amazing-feature`)
3. Commit your changes (`git commit -m 'Add some amazing feature'`)
4. Push to the branch (`git push origin feature/amazing-feature`)
5. Open a Pull Request
6. Wait for the review

And voilà! Your ice cream is ready ✨

## Authors

- **Rômulo Silva (Tomate)** - _Alchemist & Developer_ [Github](https://github.com/rohlacanna)
- **Paulo Igor (Pigor)** - _Alchemist & Developer_ [Github](https://github.com/pigor)
- **Mateus Linhares (Mateus)** - _Alchemist & Developer_ [Github](https://github.com/mateuslinhares)

## License

This project is licensed under the Apache License 2.0 - see the LICENSE file for details.