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 href="{{ url_for('packages.list_all', tag=t.name) }}">{{ t.title }}</a>
63 <div class="info-row row" style="margin-top: 2rem;">
64 <div class="btn-group-horizontal col">
65 <a class="btn" href="{{ url_for('users.profile', username=package.author.username) }}">
66 <i class="fas fa-user"></i>
68 {{ package.author.display_name }}
71 <a class="btn" rel="nofollow" href="{{ package.getDownloadURL() }}">
72 <i class="fas fa-download"></i>
73 <span class="count">{{ package.downloads }}</span>
75 <a class="btn" href="{{ url_for('threads.list_all', pid=package.id) }}">
76 <i class="fas fa-comment-alt"></i>
77 <span class="count">{{ threads | length }}</span>
79 {% if package.website %}
80 <a class="btn" href="{{ package.website }}">
81 <i class="fas fa-globe-europe"></i>
82 <span class="count">{{ _("Website") }}</span>
86 <a class="btn" href="{{ package.repo }}">
87 <i class="fas fa-code"></i>
88 <span class="count">{{ _("Source") }}</span>
91 {% if package.forums %}
92 <a class="btn" href="https://forum.minetest.net/viewtopic.php?t={{ package.forums }}">
93 <i class="fas fa-comments"></i>
94 <span class="count">{{ _("Forums") }}</span>
97 {% if package.issueTracker %}
98 <a class="btn" href="{{ package.issueTracker }}">
99 <i class="fas fa-bug"></i>
100 <span class="count">{{ _("Issue Tracker") }}</span>
104 {% if release and (release.min_rel or release.max_rel) %}
105 <div class="btn col-md-auto">
106 <img src="https://www.minetest.net/media/icon.svg" style="max-height: 1.2em;">
108 {% if release.min_rel and release.max_rel %}
109 {{ _("%(min)s - %(max)s", min=release.min_rel.name, max=release.max_rel.name) }}
110 {% elif release.min_rel %}
111 {{ _("%(min)s and above", min=release.min_rel.name) }}
112 {% elif release.max_rel %}
113 {{ _("%(max)s and below", max=release.max_rel.name) }}
118 <div class="btn-group-horizontal col-md-auto">
120 <a class="btn btn-download btn_green" rel="nofollow"
121 href="{{ package.getDownloadURL() }}">
126 {{ _("No downloads available") }}
134 <main class="container mt-4">
135 {% if not package.approved %}
136 <div class="alert alert-warning">
137 <span class="icon_message"></span>
138 {% if package.releases.count() == 0 %}
139 <h4 class="alert-heading">Release Required</h4>
140 {% if package.checkPerm(current_user, "MAKE_RELEASE") %}
141 <p>You need to create a release before this package can be approved.</p>
143 A release is a single downloadable version of your {{ package.type.value | lower }}.
144 You need to create releases even if you use a rolling release development cycle,
145 as Minetest needs them to check for updates.
147 <a class="btn" href="{{ package.getCreateReleaseURL() }}">Create Release</a>
149 A release is required before this package can be approved.
152 {% elif (package.type == package.type.GAME or package.type == package.type.TXP) and package.screenshots.count() == 0 %}
153 You need to add at least one screenshot.
155 {% elif topic_error_lvl == "danger" %}
156 Please fix the below topic issue(s).
158 {% elif "Other" in package.license.name or "Other" in package.media_license.name %}
159 Please wait for the license to be added to CDB.
162 {% if package.screenshots.count() == 0 %}
163 <b>You should add at least one screenshot, but this isn't required.</b><br />
166 {% if not package.getDownloadRelease() %}
167 Please wait for the release to be approved.
168 {% elif package.checkPerm(current_user, "APPROVE_NEW") %}
169 <form class="float-right" method="post" action="{{ package.getApproveURL() }}">
170 <input type="hidden" name="csrf_token" value="{{ csrf_token() }}" />
171 <input class="btn btn-sm btn-warning" type="submit" value="Approve" />
173 You can now approve this package if you're ready.
175 Please wait for the package to be approved.
178 <div style="clear: both;"></div>
182 <div class="alert alert-{{ topic_error_lvl }}">
183 <span class="icon_message"></span>
184 {{ topic_error | safe }}
185 <div style="clear: both;"></div>
189 {% if similar_topics %}
190 <div class="alert alert-warning">
191 Please make sure that this package has the right to
192 the name '{{ package.name }}'.
194 <a href="/policy_and_guidance/">Inclusion Policy</a>
199 {% if not review_thread and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
200 <div class="alert alert-info">
201 <a class="float-right btn btn-sm btn-info" href="{{ url_for('threads.new', pid=package.id, title='Package approval comments') }}">Open Thread</a>
203 Privately ask a question or give feedback
204 <div style="clear:both;"></div>
209 <aside class="float-right ml-4" style="width: 18rem;">
210 <div class="card mb-4">
211 <div class="card-header">
213 <div class="btn-group float-right">
214 {% if package.checkPerm(current_user, "EDIT_PACKAGE") %}
215 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getEditURL() }}"><i class="fas fa-edit"></i></a>
217 {# {% if current_user.is_authenticated %}
218 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getCreateEditRequestURL() }}">Suggest Changes</a>
220 {% if package.checkPerm(current_user, "DELETE_PACKAGE") or package.checkPerm(current_user, "UNAPPROVE_PACKAGE") %}
221 <a class="btn btn-danger btn-sm ml-1" href="{{ package.getRemoveURL() }}"><i class="fas fa-trash"></i></a>
226 {% if package_warning %}
227 <div class="card-body">
228 <div class="alert alert-danger">
229 <a href="/help/non_free/" class="float-right">Info</a>
230 <b>Warning:</b> {{ package_warning }}
234 <table class="table">
237 <td>{{ package.type.value }}</td>
241 <td>{{ package.name }}</td>
243 {% if package.provides %}
246 <td>{% for meta in package.provides %}
247 <a class="badge badge-primary"
248 href="{{ url_for('metapackages.view', name=meta.name) }}">{{ meta.name }}</a>
255 {% if package.license == package.media_license %}
256 {{ package.license.name }}
257 {% elif package.type == package.type.TXP %}
258 {{ package.media_license.name }}
260 {{ package.license.name }} for code,<br />
261 {{ package.media_license.name }} for media.
267 <td>{{ package.created_at | datetime }}</td>
272 {% if package.checkPerm(current_user, "EDIT_MAINTAINERS") %}
273 <a class="btn btn-primary btn-sm ml-1 float-right" href="{{ package.getEditMaintainersURL() }}"><i class="fas fa-edit"></i></a>
276 {% for user in package.maintainers %}
277 <a class="badge badge-primary"
278 href="{{ url_for('users.profile', username=user.username) }}">
279 {{ user.display_name }}
283 {% if current_user in package.maintainers and current_user != package.author %}
284 <form class="mt-2" method="post" action="{{ package.getRemoveSelfMaintainerURL() }}">
285 <input type="hidden" name="csrf_token" value="{{ csrf_token() }}" />
286 <input class="btn btn-sm btn-link p-0" type="submit" value="{{ _("Remove myself") }}" />
294 {% if package.author.donate_url %}
295 <div class="alert alert-secondary">
296 Like {{ package.author.display_name }}'s work?
297 <a href="{{ package.author.donate_url }}" rel="nofollow">Donate now!</a>
301 {% if package.type == package.type.MOD %}
302 <div class="card my-4">
303 <div class="card-header">Dependencies</div>
304 <div class="card-body">
305 <div class="card-subtitle mb-2 text-muted">{{ _("Required") }}</div>
306 {% for dep in package.getSortedHardDependencies() %}
307 {%- if dep.package %}
308 <div </div class="badge badge-primary"
309 href="{{ dep.package.getDetailsURL() }}">
310 {{ dep.package.title }} by {{ dep.package.author.display_name }}
311 {% elif dep.meta_package %}
312 <a class="badge badge-primary"
313 href="{{ url_for('metapackages.view', name=dep.meta_package.name) }}">
314 {{ dep.meta_package.name }}
316 {{ "Excepted package or meta_package in dep!" | throw }}
319 <i>No required dependencies</i>
322 {% set optional_deps=package.getSortedOptionalDependencies() %}
323 {% if optional_deps %}
324 <div class="card-subtitle my-2 text-muted">{{ _("Optional") }}</div>
325 {% for dep in optional_deps %}
326 {%- if dep.package %}
327 <a class="badge badge-secondary"
328 href="{{ dep.package.getDetailsURL() }}">
329 {{ dep.package.title }} by {{ dep.package.author.display_name }}
330 {% elif dep.meta_package %}
331 <a class="badge badge-secondary"
332 href="{{ url_for('metapackages.view', name=dep.meta_package.name) }}">
333 {{ dep.meta_package.name }}
335 {{ "Excepted package or meta_package in dep!" | throw }}
343 <div class="card my-4">
344 <div class="card-header">
346 {% if package.checkPerm(current_user, "MAKE_RELEASE") %}
347 <div class="btn-group float-right">
348 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getBulkReleaseURL() }}">
349 <i class="fas fa-wrench"></i>
352 <a class="btn btn-primary btn-sm ml-1" href="{{ package.getCreateReleaseURL() }}"><i class="fas fa-plus"></i></a>
356 <ul class="list-group list-group-flush">
357 {% for rel in releases %}
358 {% if rel.approved or package.checkPerm(current_user, "MAKE_RELEASE") or package.checkPerm(current_user, "APPROVE_RELEASE") %}
359 <li class="list-group-item">
361 {% if package.checkPerm(current_user, "MAKE_RELEASE") or package.checkPerm(current_user, "APPROVE_RELEASE") %}
362 <a class="btn btn-sm btn-primary float-right" href="{{ rel.getEditURL() }}">Edit
363 {% if not rel.task_id and not rel.approved and package.checkPerm(current_user, "APPROVE_RELEASE") %}
369 {% if not rel.approved %}<i>{% endif %}
371 <a href="{{ rel.getDownloadURL() }}" rel="nofollow">{{ rel.title }}</a>
373 <span style="color:#ddd;">
374 {% if rel.min_rel and rel.max_rel %}
375 [MT {{ rel.min_rel.name }}-{{ rel.max_rel.name }}]
376 {% elif rel.min_rel %}
377 [MT {{ rel.min_rel.name }}+]
378 {% elif rel.max_rel %}
379 [MT ≤{{ rel.max_rel.name }}]
385 <small style="color:#999;">
386 {% if rel.commit_hash %}
387 [{{ rel.commit_hash | truncate(5, end='') }}]
390 created {{ rel.releaseDate | date }}.
392 {% if (package.checkPerm(current_user, "MAKE_RELEASE") or package.checkPerm(current_user, "APPROVE_RELEASE")) and rel.task_id %}
393 <a href="{{ url_for('tasks.check', id=rel.task_id, r=package.getDetailsURL()) }}">Importing...</a>
394 {% elif not rel.approved %}
395 Waiting for approval.
398 {% if not rel.approved %}</i>{% endif %}
403 <li class="list-group-item">No releases available.</li>
408 <div class="card my-4">
409 <div class="card-header">
410 {% if package.approved and package.checkPerm(current_user, "CREATE_THREAD") %}
411 <div class="btn-group float-right">
412 <a class="btn btn-primary btn-sm mx-1" href="{{ url_for('threads.new', pid=package.id) }}"><i class="fas fa-plus"></i></a>
417 <ul class="list-group list-group-flush">
418 {% from "macros/threads.html" import render_threadlist %}
419 {{ render_threadlist(threads, compact=True) }}
423 {% 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) %}
424 <a class="float-right"
425 href="{{ url_for('threads.new', pid=package.id) }}">
426 Report a problem with this listing
431 {% if not package.approved and (package.author == current_user or package.checkPerm(current_user, "APPROVE_NEW")) %}
432 {% if review_thread %}
433 <h2>{% if review_thread.private %}🔒{% endif %} {{ review_thread.title }}</h2>
434 {% if review_thread.private %}
436 This thread is only visible to the package owner and users of
437 Editor rank or above.
441 {% from "macros/threads.html" import render_thread %}
442 {{ render_thread(review_thread, current_user) }}
446 <ul class="screenshot_list mb-4">
447 {% for ss in package.screenshots %}
448 {% if ss.approved or package.checkPerm(current_user, "ADD_SCREENSHOTS") %}
450 <a href="{% if package.checkPerm(current_user, 'ADD_SCREENSHOTS') %}{{ ss.getEditURL() }}{% else %}{{ ss.url }}{% endif %}">
451 <img src="{{ ss.getThumbnailURL() }}" alt="{{ ss.title }}" />
456 {% if package.checkPerm(current_user, "EDIT_PACKAGE") %}
458 <a href="{{ package.getNewScreenshotURL() }}">
459 <div class="fas fa-plus screenshot-add"></div>
465 {{ package.desc | markdown }}
467 <div style="clear: both;"></div>
469 <h3>Ratings and Reviews</h3>
471 {% from "macros/reviews.html" import render_reviews, render_review_form, render_review_preview %}
472 {% if current_user.is_authenticated %}
475 <a class="btn btn-primary" href="{{ package.getReviewURL() }}">
476 {{ _("Edit Review") }}
479 {% elif current_user in package.maintainers %}
481 {{ _("You can't review your own package.") }}
484 {{ render_review_preview(package) }}
487 {{ render_review_preview(package) }}
489 {{ render_reviews(package.reviews, current_user) }}
492 {% if current_user.is_authenticated or requests %}
493 <h3>Edit Requests</h3>
496 {% for r in requests %}
498 <a href="{{ r.getURL() }}">{{ r.title }}</a>
500 <a href="{{ url_for('users.profile', username=r.author.username) }}">{{ r.author.display_name }}</a>
503 <li>No edit requests have been made.</li>
509 {% if alternatives %}
512 {% from "macros/packagegridtile.html" import render_pkggrid %}
513 {{ render_pkggrid(alternatives) }}
516 {% if similar_topics %}
517 <h3>Similar Forum Topics</h3>
519 {% for t in similar_topics %}
522 <a href="https://forum.minetest.net/viewtopic.php?t={{ t.topic_id }}">
523 {{ t.title }} by {{ t.author.display_name }}
525 {% if t.wip %}[WIP]{% endif %}