1 {% set query=package.name %}
3 {% extends "base.html" %}
10 <meta name="og:title" content="{{ package.title }}"/>
11 <meta name="og:description" content="{{ package.short_desc }}"/>
12 <meta name="description" content="{{ package.short_desc }}"/>
13 <meta name="og:url" content="{{ package.getDetailsURL(absolute=True) }}"/>
14 {% if package.getMainScreenshotURL() %}
15 <meta name="og:image" content="{{ package.getMainScreenshotURL(absolute=True) }}"/>
20 {% if not package.license.is_foss and not package.media_license.is_foss and package.type != package.type.TXP %}
21 {% set package_warning="Non-free code and media" %}
22 {% elif not package.license.is_foss and package.type != package.type.TXP %}
23 {% set package_warning="Non-free code" %}
24 {% elif not package.media_license.is_foss %}
25 {% set package_warning="Non-free media" %}
27 {% set release = package.getDownloadRelease() %}
29 <header class="jumbotron pb-3"
30 style="background: linear-gradient(rgba(0, 0, 0, 0.4), rgba(0, 0, 0, 0.7)), url('{{ package.getMainScreenshotURL() }}');
31 background-size: cover;
32 background-repeat: no-repeat;
33 background-position: center;">
34 <div class="container">
35 <h1 class="display-3">
40 {{ package.short_desc }}
44 {% if package_warning %}
45 <a class="badge badge-danger" href="/help/non_free/">
46 <i class="fas fa-exclamation-circle" style="margin-right: 0.3em;"></i>
50 {% for warning in package.content_warnings %}
51 <a class="badge badge-warning" rel="nofollow" href="/help/content_flags/"
52 title="{{ warning.description }}">
53 <i class="fas fa-exclamation-circle" style="margin-right: 0.3em;"></i>
57 {% for t in package.tags %}
58 <a class="badge badge-primary" rel="nofollow"
59 title="{{ t.description or '' }}"
60 href="{{ url_for('packages.list_all', tag=t.name) }}">
66 <div class="info-row row" style="margin-top: 2rem;">
67 <div class="btn-group-horizontal col">
68 <a class="btn" href="{{ url_for('users.profile', username=package.author.username) }}">
69 <i class="fas fa-user"></i>
71 {{ package.author.display_name }}
74 <a class="btn" rel="nofollow" href="{{ package.getDownloadURL() }}">
75 <i class="fas fa-download"></i>
76 <span class="count">{{ package.downloads }}</span>
78 <a class="btn" href="{{ url_for('threads.list_all', pid=package.id) }}">
79 <i class="fas fa-comment-alt"></i>
80 <span class="count">{{ threads | length }}</span>
82 {% if package.website %}
83 <a class="btn" href="{{ package.website }}">
84 <i class="fas fa-globe-europe"></i>
85 <span class="count">{{ _("Website") }}</span>
89 <a class="btn" href="{{ package.repo }}">
90 <i class="fas fa-code"></i>
91 <span class="count">{{ _("Source") }}</span>
94 {% if package.forums %}
95 <a class="btn" href="https://forum.minetest.net/viewtopic.php?t={{ package.forums }}">
96 <i class="fas fa-comments"></i>
97 <span class="count">{{ _("Forums") }}</span>
100 {% if package.issueTracker %}
101 <a class="btn" href="{{ package.issueTracker }}">
102 <i class="fas fa-bug"></i>
103 <span class="count">{{ _("Issue Tracker") }}</span>
107 {% if release and (release.min_rel or release.max_rel) %}
108 <div class="btn col-md-auto">
109 <img src="https://www.minetest.net/media/icon.svg" style="max-height: 1.2em;">
111 {% if release.min_rel and release.max_rel %}
112 {{ _("%(min)s - %(max)s", min=release.min_rel.name, max=release.max_rel.name) }}
113 {% elif release.min_rel %}
114 {{ _("%(min)s and above", min=release.min_rel.name) }}
115 {% elif release.max_rel %}
116 {{ _("%(max)s and below", max=release.max_rel.name) }}
121 <div class="btn-group-horizontal col-md-auto">
123 <a class="btn btn-download btn_green" rel="nofollow"
124 href="{{ package.getDownloadURL() }}">
129 {{ _("No downloads available") }}
137 <main class="container mt-4">
138 {% if not package.approved %}
139 <div class="alert alert-warning">
140 <span class="icon_message"></span>
141 {% if package.releases.count() == 0 %}
142 <h4 class="alert-heading">Release Required</h4>
143 {% if package.checkPerm(current_user, "MAKE_RELEASE") %}
144 <p>You need to create a release before this package can be approved.</p>
146 A release is a single downloadable version of your {{ package.type.value | lower }}.
147 You need to create releases even if you use a rolling release development cycle,
148 as Minetest needs them to check for updates.
150 <a class="btn" href="{{ package.getCreateReleaseURL() }}">Create Release</a>
152 A release is required before this package can be approved.
155 {% elif (package.type == package.type.GAME or package.type == package.type.TXP) and package.screenshots.count() == 0 %}
156 You need to add at least one screenshot.
158 {% elif topic_error_lvl == "danger" %}
159 Please fix the below topic issue(s).
161 {% elif "Other" in package.license.name or "Other" in package.media_license.name %}
162 Please wait for the license to be added to CDB.
165 {% if package.screenshots.count() == 0 %}
166 <b>You should add at least one screenshot, but this isn't required.</b><br />
169 {% if not package.getDownloadRelease() %}
170 Please wait for the release to be approved.
171 {% elif package.checkPerm(current_user, "APPROVE_NEW") %}
172 <form class="float-right" method="post" action="{{ package.getApproveURL() }}">
173 <input type="hidden" name="csrf_token" value="{{ csrf_token() }}" />
174 <input class="btn btn-sm btn-warning" type="submit" value="Approve" />
176 You can now approve this package if you're ready.
178 Please wait for the package to be approved.
181 <div style="clear: both;"></div>
185 <div class="alert alert-{{ topic_error_lvl }}">
186 <span class="icon_message"></span>
187 {{ topic_error | safe }}
188 <div style="clear: both;"></div>
192 {% if similar_topics %}
193 <div class="alert alert-warning">
194 Please make sure that this package has the right to
195 the name '{{ package.name }}'.
197 <a href="/policy_and_guidance/">Inclusion Policy</a>
202 {% if not review_thread and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
203 <div class="alert alert-info">
204 <a class="float-right btn btn-sm btn-info" href="{{ url_for('threads.new', pid=package.id, title='Package approval comments') }}">Open Thread</a>
206 Privately ask a question or give feedback
207 <div style="clear:both;"></div>
212 <aside class="float-right ml-4" style="width: 18rem;">
213 <div class="card mb-4">
214 <div class="card-header">
216 <div class="btn-group float-right">
217 {% if package.checkPerm(current_user, "EDIT_PACKAGE") %}
218 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getEditURL() }}"><i class="fas fa-edit"></i></a>
220 {# {% if current_user.is_authenticated %}
221 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getCreateEditRequestURL() }}">Suggest Changes</a>
223 {% if package.checkPerm(current_user, "DELETE_PACKAGE") or package.checkPerm(current_user, "UNAPPROVE_PACKAGE") %}
224 <a class="btn btn-danger btn-sm ml-1" href="{{ package.getRemoveURL() }}"><i class="fas fa-trash"></i></a>
229 {% if package_warning %}
230 <div class="card-body">
231 <div class="alert alert-danger">
232 <a href="/help/non_free/" class="float-right">Info</a>
233 <b>Warning:</b> {{ package_warning }}
237 <table class="table">
240 <td>{{ package.type.value }}</td>
244 <td>{{ package.name }}</td>
246 {% if package.provides %}
249 <td>{% for meta in package.provides %}
250 <a class="badge badge-primary"
251 href="{{ url_for('metapackages.view', name=meta.name) }}">{{ meta.name }}</a>
258 {% if package.license == package.media_license %}
259 {{ package.license.name }}
260 {% elif package.type == package.type.TXP %}
261 {{ package.media_license.name }}
263 {{ package.license.name }} for code,<br />
264 {{ package.media_license.name }} for media.
270 <td>{{ package.created_at | datetime }}</td>
275 {% if package.checkPerm(current_user, "EDIT_MAINTAINERS") %}
276 <a class="btn btn-primary btn-sm ml-1 float-right" href="{{ package.getEditMaintainersURL() }}"><i class="fas fa-edit"></i></a>
279 {% for user in package.maintainers %}
280 <a class="badge badge-primary"
281 href="{{ url_for('users.profile', username=user.username) }}">
282 {{ user.display_name }}
286 {% if current_user in package.maintainers and current_user != package.author %}
287 <form class="mt-2" method="post" action="{{ package.getRemoveSelfMaintainerURL() }}">
288 <input type="hidden" name="csrf_token" value="{{ csrf_token() }}" />
289 <input class="btn btn-sm btn-link p-0" type="submit" value="{{ _("Remove myself") }}" />
297 {% if package.author.donate_url %}
298 <div class="alert alert-secondary">
299 Like {{ package.author.display_name }}'s work?
300 <a href="{{ package.author.donate_url }}" rel="nofollow">Donate now!</a>
304 {% if package.type == package.type.MOD %}
305 <div class="card my-4">
306 <div class="card-header">Dependencies</div>
307 <div class="card-body">
308 <div class="card-subtitle mb-2 text-muted">{{ _("Required") }}</div>
309 {% for dep in package.getSortedHardDependencies() %}
310 {%- if dep.package %}
311 <div </div class="badge badge-primary"
312 href="{{ dep.package.getDetailsURL() }}">
313 {{ dep.package.title }} by {{ dep.package.author.display_name }}
314 {% elif dep.meta_package %}
315 <a class="badge badge-primary"
316 href="{{ url_for('metapackages.view', name=dep.meta_package.name) }}">
317 {{ dep.meta_package.name }}
319 {{ "Excepted package or meta_package in dep!" | throw }}
322 <i>No required dependencies</i>
325 {% set optional_deps=package.getSortedOptionalDependencies() %}
326 {% if optional_deps %}
327 <div class="card-subtitle my-2 text-muted">{{ _("Optional") }}</div>
328 {% for dep in optional_deps %}
329 {%- if dep.package %}
330 <a class="badge badge-secondary"
331 href="{{ dep.package.getDetailsURL() }}">
332 {{ dep.package.title }} by {{ dep.package.author.display_name }}
333 {% elif dep.meta_package %}
334 <a class="badge badge-secondary"
335 href="{{ url_for('metapackages.view', name=dep.meta_package.name) }}">
336 {{ dep.meta_package.name }}
338 {{ "Excepted package or meta_package in dep!" | throw }}
346 <div class="card my-4">
347 <div class="card-header">
349 {% if package.checkPerm(current_user, "MAKE_RELEASE") %}
350 <div class="btn-group float-right">
351 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getBulkReleaseURL() }}">
352 <i class="fas fa-wrench"></i>
355 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getCreateReleaseURL() }}"><i class="fas fa-plus"></i></a>
359 <ul class="list-group list-group-flush">
360 {% for rel in releases %}
361 {% if rel.approved or package.checkPerm(current_user, "MAKE_RELEASE") or package.checkPerm(current_user, "APPROVE_RELEASE") %}
362 <li class="list-group-item">
364 {% if package.checkPerm(current_user, "MAKE_RELEASE") or package.checkPerm(current_user, "APPROVE_RELEASE") %}
365 <a class="btn btn-sm btn-primary float-right" href="{{ rel.getEditURL() }}">Edit
366 {% if not rel.task_id and not rel.approved and package.checkPerm(current_user, "APPROVE_RELEASE") %}
372 {% if not rel.approved %}<i>{% endif %}
374 <a href="{{ rel.getDownloadURL() }}" rel="nofollow">{{ rel.title }}</a>
376 <span style="color:#ddd;">
377 {% if rel.min_rel and rel.max_rel %}
378 [MT {{ rel.min_rel.name }}-{{ rel.max_rel.name }}]
379 {% elif rel.min_rel %}
380 [MT {{ rel.min_rel.name }}+]
381 {% elif rel.max_rel %}
382 [MT ≤{{ rel.max_rel.name }}]
388 <small style="color:#999;">
389 {% if rel.commit_hash %}
390 [{{ rel.commit_hash | truncate(5, end='') }}]
393 created {{ rel.releaseDate | date }}.
395 {% if (package.checkPerm(current_user, "MAKE_RELEASE") or package.checkPerm(current_user, "APPROVE_RELEASE")) and rel.task_id %}
396 <a href="{{ url_for('tasks.check', id=rel.task_id, r=package.getDetailsURL()) }}">Importing...</a>
397 {% elif not rel.approved %}
398 Waiting for approval.
401 {% if not rel.approved %}</i>{% endif %}
406 <li class="list-group-item">No releases available.</li>
411 <div class="card my-4">
412 <div class="card-header">
413 {% if package.approved and package.checkPerm(current_user, "CREATE_THREAD") %}
414 <div class="btn-group float-right">
415 <a class="btn btn-primary btn-sm mx-1" href="{{ url_for('threads.new', pid=package.id) }}"><i class="fas fa-plus"></i></a>
420 <ul class="list-group list-group-flush">
421 {% from "macros/threads.html" import render_threadlist %}
422 {{ render_threadlist(threads, compact=True) }}
426 {% if package.approved and package.checkPerm(current_user, "CREATE_THREAD") and current_user != package.author and not current_user.rank.atLeast(current_user.rank.EDITOR) %}
427 <a class="float-right"
428 href="{{ url_for('threads.new', pid=package.id) }}">
429 Report a problem with this listing
434 {% if not package.approved and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
435 {% if review_thread %}
436 <h2>{% if review_thread.private %}🔒{% endif %} {{ review_thread.title }}</h2>
437 {% if review_thread.private %}
439 This thread is only visible to the package owner and users of
440 Editor rank or above.
444 {% from "macros/threads.html" import render_thread %}
445 {{ render_thread(review_thread, current_user) }}
449 <ul class="screenshot_list mb-4">
450 {% for ss in package.screenshots %}
451 {% if ss.approved or package.checkPerm(current_user, "ADD_SCREENSHOTS") %}
453 <a href="{% if package.checkPerm(current_user, 'ADD_SCREENSHOTS') %}{{ ss.getEditURL() }}{% else %}{{ ss.url }}{% endif %}">
454 <img src="{{ ss.getThumbnailURL() }}" alt="{{ ss.title }}" />
459 {% if package.checkPerm(current_user, "EDIT_PACKAGE") %}
461 <a href="{{ package.getNewScreenshotURL() }}">
462 <div class="fas fa-plus screenshot-add"></div>
468 {{ package.desc | markdown }}
470 <div style="clear: both;"></div>
472 <h3>Ratings and Reviews</h3>
474 {% from "macros/reviews.html" import render_reviews, render_review_form, render_review_preview %}
475 {% if current_user.is_authenticated %}
478 <a class="btn btn-primary" href="{{ package.getReviewURL() }}">
479 {{ _("Edit Review") }}
482 {% elif current_user in package.maintainers %}
484 {{ _("You can't review your own package.") }}
487 {{ render_review_preview(package) }}
490 {{ render_review_preview(package) }}
492 {{ render_reviews(package.reviews, current_user) }}
495 {% if current_user.is_authenticated or requests %}
496 <h3>Edit Requests</h3>
499 {% for r in requests %}
501 <a href="{{ r.getURL() }}">{{ r.title }}</a>
503 <a href="{{ url_for('users.profile', username=r.author.username) }}">{{ r.author.display_name }}</a>
506 <li>No edit requests have been made.</li>
512 {% if alternatives %}
515 {% from "macros/packagegridtile.html" import render_pkggrid %}
516 {{ render_pkggrid(alternatives) }}
519 {% if similar_topics %}
520 <h3>Similar Forum Topics</h3>
522 {% for t in similar_topics %}
525 <a href="https://forum.minetest.net/viewtopic.php?t={{ t.topic_id }}">
526 {{ t.title }} by {{ t.author.display_name }}
528 {% if t.wip %}[WIP]{% endif %}