Add global + page-scoped market listings with infinite scroll
All checks were successful
Build and Deploy / build-and-deploy (push) Successful in 1m4s
All checks were successful
Build and Deploy / build-and-deploy (push) Successful in 1m4s
- New all_markets blueprint at / with paginated grid and HTMX infinite scroll - New page_markets blueprint at /<slug>/ for page-scoped market listing - list_marketplaces service method (via shared submodule update) - Updated slug preprocessor to handle both /<slug>/ and /<page_slug>/<market_slug>/ - Removed inline markets_listing() route (replaced by all_markets blueprint) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -1,2 +1,4 @@
|
||||
from .market.routes import register as register_market_bp
|
||||
from .product.routes import register as register_product
|
||||
from .all_markets.routes import register as register_all_markets
|
||||
from .page_markets.routes import register as register_page_markets
|
||||
|
||||
0
bp/all_markets/__init__.py
Normal file
0
bp/all_markets/__init__.py
Normal file
74
bp/all_markets/routes.py
Normal file
74
bp/all_markets/routes.py
Normal file
@@ -0,0 +1,74 @@
|
||||
"""
|
||||
All-markets blueprint — shows markets across ALL pages.
|
||||
|
||||
Mounted at / (root of market app). No slug context.
|
||||
|
||||
Routes:
|
||||
GET / — full page with first page of markets
|
||||
GET /all-markets — HTMX fragment for infinite scroll
|
||||
"""
|
||||
from __future__ import annotations
|
||||
|
||||
from quart import Blueprint, g, request, render_template, make_response
|
||||
|
||||
from shared.browser.app.utils.htmx import is_htmx_request
|
||||
from shared.services.registry import services
|
||||
|
||||
|
||||
def register() -> Blueprint:
|
||||
bp = Blueprint("all_markets", __name__)
|
||||
|
||||
async def _load_markets(page, per_page=20):
|
||||
"""Load all markets + page info for container badges."""
|
||||
markets, has_more = await services.market.list_marketplaces(
|
||||
g.s, page=page, per_page=per_page,
|
||||
)
|
||||
|
||||
# Batch-load page info for container_ids
|
||||
page_info = {}
|
||||
if markets:
|
||||
post_ids = list({
|
||||
m.container_id for m in markets
|
||||
if m.container_type == "page"
|
||||
})
|
||||
if post_ids:
|
||||
posts = await services.blog.get_posts_by_ids(g.s, post_ids)
|
||||
for p in posts:
|
||||
page_info[p.id] = {"title": p.title, "slug": p.slug}
|
||||
|
||||
return markets, has_more, page_info
|
||||
|
||||
@bp.get("/")
|
||||
async def index():
|
||||
page = int(request.args.get("page", 1))
|
||||
markets, has_more, page_info = await _load_markets(page)
|
||||
|
||||
ctx = dict(
|
||||
markets=markets,
|
||||
has_more=has_more,
|
||||
page_info=page_info,
|
||||
page=page,
|
||||
)
|
||||
|
||||
if is_htmx_request():
|
||||
html = await render_template("_types/all_markets/_main_panel.html", **ctx)
|
||||
else:
|
||||
html = await render_template("_types/all_markets/index.html", **ctx)
|
||||
|
||||
return await make_response(html, 200)
|
||||
|
||||
@bp.get("/all-markets")
|
||||
async def markets_fragment():
|
||||
page = int(request.args.get("page", 1))
|
||||
markets, has_more, page_info = await _load_markets(page)
|
||||
|
||||
html = await render_template(
|
||||
"_types/all_markets/_cards.html",
|
||||
markets=markets,
|
||||
has_more=has_more,
|
||||
page_info=page_info,
|
||||
page=page,
|
||||
)
|
||||
return await make_response(html, 200)
|
||||
|
||||
return bp
|
||||
0
bp/page_markets/__init__.py
Normal file
0
bp/page_markets/__init__.py
Normal file
65
bp/page_markets/routes.py
Normal file
65
bp/page_markets/routes.py
Normal file
@@ -0,0 +1,65 @@
|
||||
"""
|
||||
Page-markets blueprint — shows markets for a single page.
|
||||
|
||||
Mounted at /<slug> (page-scoped). Requires g.post_data from hydrate_post.
|
||||
|
||||
Routes:
|
||||
GET /<slug>/ — full page scoped to this page
|
||||
GET /<slug>/page-markets — HTMX fragment for infinite scroll
|
||||
"""
|
||||
from __future__ import annotations
|
||||
|
||||
from quart import Blueprint, g, request, render_template, make_response
|
||||
|
||||
from shared.browser.app.utils.htmx import is_htmx_request
|
||||
from shared.services.registry import services
|
||||
|
||||
|
||||
def register() -> Blueprint:
|
||||
bp = Blueprint("page_markets", __name__)
|
||||
|
||||
async def _load_markets(post_id, page, per_page=20):
|
||||
"""Load markets for this page's container."""
|
||||
markets, has_more = await services.market.list_marketplaces(
|
||||
g.s, "page", post_id, page=page, per_page=per_page,
|
||||
)
|
||||
return markets, has_more
|
||||
|
||||
@bp.get("/")
|
||||
async def index():
|
||||
post = g.post_data["post"]
|
||||
page = int(request.args.get("page", 1))
|
||||
|
||||
markets, has_more = await _load_markets(post["id"], page)
|
||||
|
||||
ctx = dict(
|
||||
markets=markets,
|
||||
has_more=has_more,
|
||||
page_info={},
|
||||
page=page,
|
||||
)
|
||||
|
||||
if is_htmx_request():
|
||||
html = await render_template("_types/page_markets/_main_panel.html", **ctx)
|
||||
else:
|
||||
html = await render_template("_types/page_markets/index.html", **ctx)
|
||||
|
||||
return await make_response(html, 200)
|
||||
|
||||
@bp.get("/page-markets")
|
||||
async def markets_fragment():
|
||||
post = g.post_data["post"]
|
||||
page = int(request.args.get("page", 1))
|
||||
|
||||
markets, has_more = await _load_markets(post["id"], page)
|
||||
|
||||
html = await render_template(
|
||||
"_types/page_markets/_cards.html",
|
||||
markets=markets,
|
||||
has_more=has_more,
|
||||
page_info={},
|
||||
page=page,
|
||||
)
|
||||
return await make_response(html, 200)
|
||||
|
||||
return bp
|
||||
Reference in New Issue
Block a user