
Removes the following awards / fields on User: - flairlock - progressivestack - bird - longpost (pizzashill) - marseyawarded - rehab - deflector - mute - unmutable - eye (All-Seeing Eye) - alt (Alt-Seeing Eye) Primarily motivated by starting to remove some un-Mottelike cruft from core commenting/posting routes. Cleared out other inapplicable awards while in the process.
643 lines
33 KiB
HTML
643 lines
33 KiB
HTML
{% extends "default.html" %}
|
|
|
|
|
|
{% if p.should_hide_score %}
|
|
{% set ups="" %}
|
|
{% set downs="" %}
|
|
{% set score="" %}
|
|
{% else %}
|
|
{% set ups=p.upvotes %}
|
|
{% set downs=p.downvotes %}
|
|
{% set score=p.score %}
|
|
{% endif %}
|
|
|
|
{% if v %}
|
|
{% set voted=p.voted if p.voted else 0 %}
|
|
{% else %}
|
|
{% set voted=-2 %}
|
|
{% endif %}
|
|
|
|
{% block title %}
|
|
|
|
<style>
|
|
body > .container {
|
|
padding-left: 20px !important;
|
|
padding-right: 20px !important;
|
|
}
|
|
</style>
|
|
|
|
{% if success %}
|
|
<script>
|
|
history.pushState(null, null, '{{p.permalink}}');
|
|
localStorage.setItem("post_title", "")
|
|
localStorage.setItem("post_text", "")
|
|
localStorage.setItem("post_url", "")
|
|
localStorage.setItem("sub", "")
|
|
</script>
|
|
{% endif %}
|
|
|
|
<meta charset="utf-8">
|
|
<meta property="og:type" content="article">
|
|
|
|
{% if comment_info and not comment_info.is_banned and not comment_info.deleted_utc %}
|
|
<title>{{'@'+comment_info.author_name}} comments on "{{p.plaintitle(v)}} - {{SITE_TITLE}}"</title>
|
|
|
|
|
|
<meta property="og:article:author" content="{{'@'+comment_info.author_name}}">
|
|
<meta property="article:published_time" content="{{comment_info.created_datetime}}">
|
|
{% if comment_info.edited_utc %}<meta property="article:modified_time" content="{{comment_info.edited_string}}">{% endif %}
|
|
<meta property="og:description" name="description" content="{{comment_info.plainbody(v)}}">
|
|
<meta property="og:author" name="author" content="{{'@'+comment_info.author_name}}">
|
|
<meta property="og:title" content="{{'@'+comment_info.author_name}} comments on {{p.plaintitle(v)}} - {{SITE_TITLE}}">
|
|
<meta property="og:image" content="{% if p.is_image %}{{p.realurl(v)}}{% elif p.has_thumb%}{{p.thumb_url}}{% else %}{{SITE_FULL}}{{ ('images/'~SITE_ID~'/site_preview.webp') | asset }}{% endif %}">
|
|
{% if p.is_video %}
|
|
<meta property="og:video" content="{{p.realurl(v)}}">
|
|
{% endif %}
|
|
<meta property="og:url" content="{{comment_info.permalink}}">
|
|
<meta property="og:site_name" content="{{request.host}}">
|
|
|
|
<meta name="twitter:card" content="summary">
|
|
<meta name="twitter:site" content="{{SITE_FULL}}">
|
|
<meta name="twitter:title" content="{{'@'+comment_info.author_name}} comments on {{p.plaintitle(v)}} - {{SITE_TITLE}}">
|
|
<meta name="twitter:creator" content="{{'@'+comment_info.author_name}}">
|
|
<meta name="twitter:description" content="{{comment_info.plainbody(v)}}">
|
|
<meta name="twitter:image" content="{% if p.is_image %}{{p.realurl(v)}}{% elif p.has_thumb%}{{p.thumb_url}}{% else %}{{SITE_FULL}}{{ ('images/'~SITE_ID~'/site_preview.webp') | asset }}{% endif %}">
|
|
<meta name="twitter:url" content="{{p.permalink}}">
|
|
|
|
{% else %}
|
|
<title>{{p.plaintitle(v)}} - {{SITE_TITLE}}</title>
|
|
|
|
|
|
{% if p.author %}<meta property="og:article:author" content="{{'@'+p.author_name}}">{% endif %}
|
|
<meta property="article:published_time" content="{{p.created_datetime}}">
|
|
{% if p.edited_utc %}<meta property="article:modified_time" content="{{p.edited_string}}">{% endif %}
|
|
<meta property="og:description" name="description" content="{{p.plainbody(v)}}">
|
|
{% if p.author %}<meta property="og:author" name="author" content="{{'@'+p.author_name}}">{% endif %}
|
|
<meta property="og:title" content="{{p.plaintitle(v)}} - {{SITE_TITLE}}">
|
|
<meta property="og:image" content="{% if p.is_image %}{{p.realurl(v)}}{% elif p.has_thumb%}{{p.thumb_url}}{% else %}{{SITE_FULL}}{{ ('images/'~SITE_ID~'/site_preview.webp') | asset }}{% endif %}">
|
|
{% if p.url and p.is_video %}
|
|
<meta property="og:video" content="{{p.realurl(v)}}">
|
|
{% endif %}
|
|
<meta property="og:url" content="{{p.permalink}}">
|
|
<meta property="og:site_name" content="{{request.host}}">
|
|
|
|
<meta name="twitter:card" content="summary_large_image">
|
|
<meta name="twitter:site" content="{{SITE_FULL}}">
|
|
<meta name="twitter:title" content="{{p.plaintitle(v)}} - {{SITE_TITLE}}">
|
|
{% if p.author %}<meta name="twitter:creator" content="{{'@'+p.author_name}}">{% endif %}
|
|
<meta name="twitter:description" content="{{p.plainbody(v)}}">
|
|
<meta name="twitter:image" content="{% if p.is_image %}{{p.realurl(v)}}{% elif p.has_thumb %}{{p.thumb_url}}{% else %}{{SITE_FULL}}{{ ('images/'~SITE_ID~'/site_preview.webp') | asset }}{% endif %}">
|
|
<meta name="twitter:url" content="{{p.permalink}}">
|
|
|
|
{% endif %}
|
|
{% endblock %}
|
|
|
|
{% block pagetype %}thread{% endblock %}
|
|
|
|
{% block actionsModal %}
|
|
|
|
{% if v %}
|
|
<div class="modal fade d-md-none" id="actionsModal" tabindex="-1" role="dialog" aria-labelledby="actionsModalTitle" aria-hidden="true">
|
|
<div class="modal-dialog modal-dialog-centered" role="document">
|
|
<div class="modal-content">
|
|
<div class="modal-header p-3">
|
|
<h5 class="col modal-title text-center h6">More options</h5>
|
|
<button class="close position-absolute py-3" style="right: 1rem"data-bs-dismiss="modal" aria-label="Close">
|
|
<span aria-hidden="true"><i class="fas fa-times-circle text-gray-500"></i></span>
|
|
</button>
|
|
</div>
|
|
<div class="modal-body">
|
|
<ul class="list-group post-actions">
|
|
{% if v.id==p.author_id %}
|
|
<button class="nobackground btn btn-link btn-block btn-lg text-left text-muted" data-bs-dismiss="modal" onclick="togglePostEdit('{{p.id}}')"><i class="far fa-edit text-center text-muted mr-3"></i>Edit</button>
|
|
{% elif p.body %}
|
|
<button class="nobackground btn btn-link btn-block btn-lg text-left text-muted" data-bs-dismiss="modal" onclick="expandMarkdown(this,'{{p.id}}')"><i class="fas text-expand-icon-{{p.id}} fa-expand-alt text-center mr-3"></i><span>View source</span></button>
|
|
{% endif %}
|
|
|
|
{% include "post_actions_mobile.html" %}
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if v and v.admin_level > 1 %}
|
|
{% include "post_admin_actions_mobile.html" %}
|
|
{% endif %}
|
|
|
|
{% endblock %}
|
|
|
|
{% block content %}
|
|
|
|
{% set cc='COUNTRY CLUB' %}
|
|
|
|
<div class="row mb-3" style="background-color:var(--gray-600)">
|
|
|
|
<div id="post-root" class="col-12">
|
|
|
|
<div class="card border-0 mt-3{% if p.is_banned %} banned{% endif %}{% if p.stickied %} stickied{% endif %}{% if voted==1 %} upvoted{% elif voted==-1 %} downvoted{% endif %}">
|
|
<div id="post-{{p.id}}" class="{% if p.award_count('glowie') %}glow{% endif %} {% if p.deleted_utc %}deleted {% endif %}d-flex flex-row-reverse flex-nowrap justify-content-end">
|
|
|
|
<div id="post-content" class="{% if p.deleted_utc %}deleted {% endif %}card-block w-100 my-md-auto">
|
|
|
|
<div class="post-meta text-left mb-2">
|
|
{% if p.sub %}
|
|
<a href='/h/{{p.sub}}'>/h/{{p.sub}}</a>
|
|
{% endif %}
|
|
|
|
{% if p.sub and p.author.exiled_from(p.sub) %}
|
|
<a role="button"><i class="fas fa-campfire text-danger" data-bs-toggle="tooltip" data-bs-placement="bottom" title="User has been exiled from /h/{{p.sub}}"></i></a>
|
|
{% endif %}
|
|
|
|
{% if p.bannedfor %}
|
|
<a role="button"><i class="fas fa-hammer-crash text-danger" data-bs-toggle="tooltip" data-bs-placement="bottom" title="User was banned for this post{% if p.author.banned_by %} by @{{p.author.banned_by.username}}{% endif %}"></i></a>
|
|
{% endif %}
|
|
|
|
{% for a in p.awards|reverse %}
|
|
<i class="{{a.class_list}} px-1" data-bs-toggle="tooltip" data-bs-placement="bottom" title="{{a.title}} Award given by @{{a.user.username}}"></i>
|
|
{% endfor %}
|
|
|
|
{% if v and v.admin_level > 1 and p.author.shadowbanned %}<i class="fas fa-user-times text-admin" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Shadowbanned by @{{p.author.shadowbanned}}"></i>{% endif %}
|
|
|
|
{% if p.stickied %}
|
|
<i id='pinned-{{p.id}}' class="fas fa-thumbtack fa-rotate--45 text-admin" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Pinned by @{{p.stickied}}" {% if p.stickied_utc %}onmouseover="pinned_timestamp('pinned-{{p.id}}')" data-timestamp={{p.stickied_utc}} {% endif %}></i>
|
|
{% endif %}
|
|
|
|
{% if p.is_pinned %}<i class="fas fa-thumbtack fa-rotate--45 fa-fw text-admin" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Pinned to profile"></i>{% endif %}
|
|
{% if p.distinguish_level %} <i class="fas fa-broom text-admin" data-bs-toggle="tooltip" data-bs-placement="bottom" title="{{SITE_TITLE}} Admin, speaking officially"></i>{% endif %}
|
|
{% if p.is_bot %} <i class="fas fa-robot text-info" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Bot"></i>{% endif %}
|
|
{% if p.over_18 %}<span class="badge badge-danger text-small-extra mr-1">+18</span>{% endif %}
|
|
{% if p.private %}<span class="badge border-warning border-1 text-small-extra">Draft</span>{% endif %}
|
|
{% if v and p.filter_state == 'reported' and v.can_manage_reports() %}
|
|
<a class="btn btn-primary" id="submission-report-button" role="button" style="padding:1px 5px; font-size:10px"onclick="document.getElementById('flaggers').classList.toggle('d-none')">{{p.active_flags(v)}} Reports</a>
|
|
{% endif %}
|
|
|
|
{% if not p.author %}
|
|
{{p.print()}}
|
|
{% endif %}
|
|
|
|
{% if p.ghost %}
|
|
👻
|
|
{% else %}
|
|
|
|
{% if p.author.verified %}<i class="fas fa-badge-check align-middle ml-1 {% if p.author.verified=='Glowiefied' %}glow{% endif %}" style="color:{% if p.author.verifiedcolor %}#{{p.author.verifiedcolor}}{% else %}var(--primary){% endif %}" data-bs-toggle="tooltip" data-bs-placement="bottom" title="{{p.author.verified}}"></i>
|
|
{% endif %}
|
|
|
|
<a href="/@{{p.author_name}}" class="user-name" onclick='popclick({{p.author.json_popover(v) | tojson}}); return false' data-bs-placement="bottom" data-bs-toggle="popover" data-bs-trigger="click" data-content-id="popover" role="button" tabindex="0" style="font-weight: bold"><img loading="lazy" src="{{p.author.profile_url}}" class="profile-pic-20 mr-2"><span {% if p.author.patron and not p.distinguish_level %}class="patron" style="background-color:#{{p.author.namecolor}};"{% elif p.distinguish_level %}class="mod"{% endif %}>{{p.author_name}}</span></a>{% if p.author.customtitle %} <bdi style="color: #{{p.author.titlecolor}}"> {{p.author.customtitle | safe}}</bdi>{% endif %}
|
|
{% if v and v.admin_level > 1 %}
|
|
<span
|
|
class="usernote-link"
|
|
data-micromodal-trigger="modal-1"
|
|
onclick='fillnote( {{p.author.json_popover(v) | tojson}}, {{p.id}}, null )'>_U_</span>
|
|
{% endif %}
|
|
|
|
{% endif %}
|
|
<span data-bs-toggle="tooltip" data-bs-placement="bottom" id="timestamp" onmouseover="timestamp('timestamp','{{p.created_utc}}')"> {{p.age_string}}</span>
|
|
({% if p.is_image %}image post{% elif p.is_video %}video post{% elif p.domain %}<a href="/search/posts/?q=domain%3A{{p.domain}}&sort=new&t=all" class="post-meta-domain" {% if not v or v.newtabexternal %}target="_blank"{% endif %}>{{p.domain|truncate(35, True)}}</a>{% else %}text post{% endif %})
|
|
|
|
{% if p.edited_utc %}
|
|
Edited <span data-bs-toggle="tooltip" data-bs-placement="bottom" onmouseover="timestamp('edited_timestamp','{{p.edited_utc}}')" id="edited_timestamp">{{p.edited_string}}</span>
|
|
{% endif %}
|
|
{{p.views}} thread views
|
|
</div>
|
|
{% if v and p.filter_state == 'reported' and v.can_manage_reports() %}
|
|
<div id="flaggers" class="flaggers d-none">
|
|
<strong><i class="far fa-fw fa-flag"></i> Reports:</strong>
|
|
<a class="btn btn-primary" style="margin:1px 5px" onclick="filter_new_status({{p.id}}, 'normal')">Approve</a>
|
|
<a class="btn btn-secondary" style="margin:1px 5px" onclick="filter_new_status({{p.id}}, 'ignored')">Approve and Ignore</a>
|
|
<a class="btn btn-danger" style="margin:1px 5px" onclick="filter_new_status({{p.id}}, 'removed')">Remove</a>
|
|
<pre></pre>
|
|
<ul style="padding-left:20px; margin-bottom: 0;word-wrap:break-word">
|
|
{% for f in p.flags(v) %}
|
|
<li><a style="font-weight:bold" href="{{f.user.url}}">{{f.user.username}}</a>{% if f.reason %}: {{f.realreason(v) | safe}}{% endif %}</li>
|
|
{% endfor %}
|
|
</ul>
|
|
</div>
|
|
{% endif %}
|
|
{% if p.realurl(v) %}
|
|
<h1 id="post-title" class="card-title post-title text-left mb-md-3"><a {% if not v or v.newtabexternal %}target="_blank"{% endif %} rel="nofollow noopener noreferrer" href="{{p.realurl(v)}}">
|
|
{% if p.club %}<span class="patron font-weight-bolder mr-1" style="background-color:red; font-size:12px; line-height:2;">{{CC}}</span>{% endif %}
|
|
{% if p.flair %}<span class="patron font-weight-bolder mr-1" style="background-color:var(--primary); font-size:12px; line-height:2;">{{p.flair | safe}}</span>{% endif %}
|
|
{{p.realtitle(v) | safe}}
|
|
</a></h1>
|
|
{% else %}
|
|
<h1 id="post-title" class="card-title post-title text-left mb-md-3">
|
|
{% if p.club %}<span class="patron font-weight-bolder mr-1" style="background-color:red; font-size:12px; line-height:2;">{{CC}}</span>{% endif %}
|
|
{% if p.flair %}<span class="patron font-weight-bolder mr-1" style="background-color:var(--primary); font-size:12px; line-height:2;">{{p.flair | safe}}</span>{% endif %}
|
|
{{p.realtitle(v) | safe}}
|
|
</h1>
|
|
{% endif %}
|
|
|
|
<div id="post-body" class="post-body mt-3">
|
|
{% if p.realurl(v) %}
|
|
{% if not p.embed_url and not p.is_image and not p.is_video %}
|
|
<a rel="nofollow noopener noreferrer" href="{{p.realurl(v)}}" {% if not v or v.newtabexternal %}target="_blank"{% endif %}>
|
|
<div class="d-flex d-md-none justify-content-between align-items-center border rounded p-2{% if p.realbody(v) %} mb-3{% endif %}">
|
|
<span>{{p.domain|truncate(30, True)}}</span>
|
|
<i class="fas fa-external-link-alt text-small"></i>
|
|
</div>
|
|
</a>
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
|
|
{% if p.embed_url %}
|
|
{% if p.domain == "twitter.com" %}
|
|
{{p.embed_url | safe}}
|
|
{% if v and v.theme.split("_")[0] in ["light", "coffee", "4chan"] %}
|
|
<script src="{{ 'js/twitterlight.js' | asset }}"></script>
|
|
{% else %}
|
|
<script src="{{ 'js/twitter.js' | asset }}"></script>
|
|
{% endif %}
|
|
{% elif p.domain in ['youtu.be','youtube.com'] and p.embed_url and p.embed_url.startswith('<lite-youtube') %}
|
|
{{p.embed_url | safe}}
|
|
{% endif %}
|
|
{% endif %}
|
|
|
|
|
|
<div id="post-text">
|
|
{% if p.is_image %}
|
|
<div class="row no-gutters">
|
|
<div class="col">
|
|
<a {% if not v or v.newtabexternal %}target="_blank"{% endif %} rel="nofollow noopener noreferrer" href="{{p.realurl(v)}}">
|
|
<img src="{{p.realurl(v)}}" class="img-fluid" style="max-height:500px" alt="Unable to load image">
|
|
</a>
|
|
</div>
|
|
</div>
|
|
<pre></pre>
|
|
{% elif p.is_video %}
|
|
<div class="row no-gutters">
|
|
<div class="col">
|
|
<video controls preload="none" class="vid">
|
|
<source src="{{p.realurl(v)}}" type="video/mp4">
|
|
</video>
|
|
</div>
|
|
</div>
|
|
<pre></pre>
|
|
{% endif %}
|
|
{{p.realbody(v) | safe}}
|
|
|
|
{% if p.is_banned and p.ban_reason %}
|
|
<div class="text-removed mb-0">removed by @{{p.ban_reason}}</div>
|
|
{% endif %}
|
|
|
|
</div>
|
|
</div>
|
|
|
|
{% if v and (v.id==p.author_id or v.admin_level > 2) and not v.is_suspended %}
|
|
<div id="edit-post-body-{{p.id}}" class="d-none comment-write collapsed child">
|
|
<form id="post-edit-form-{{p.id}}" action="/edit_post/{{p.id}}" method="post" enctype="multipart/form-data">
|
|
<div class="d-flex flex-column">
|
|
<input type="hidden" name="formkey" value="{{v.formkey}}">
|
|
<input type="hidden" name="current_page" value="{{request.path}}">
|
|
<input type="text" autocomplete="off" maxlength="500" name="title" class="comment-box form-control rounded" required placeholder="title" value="{{p.title}}"\>
|
|
<textarea autocomplete="off" name="body" maxlength="20000" oninput="markdown('post-edit-box-{{p.id}}', 'post-edit-{{p.id}}');charLimit('post-edit-box-{{p.id}}','charcount-post-edit')" id="post-edit-box-{{p.id}}" form="post-edit-form-{{p.id}}" class="comment-box form-control rounded" aria-label="With textarea" placeholder="Add text to your post..." rows="10" data-id="{{p.id}}">{{p.body}}</textarea>
|
|
|
|
<div class="text-small font-weight-bold mt-1" id="charcount-post-edit" style="right: 1rem; bottom: 0.5rem; z-index: 3;"></div>
|
|
|
|
<div class="comment-format">
|
|
<small class="format btn btn-secondary"><i class="fas fa-bold" aria-hidden="true" onclick="makeBold('post-edit-box-{{p.id}}')" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Bold"></i></small>
|
|
<a class="format btn btn-secondary" role="button" onclick="makeItalics('post-edit-box-{{p.id}}')" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Italicize"><i class="fas fa-italic" aria-hidden="true"></i></a>
|
|
<a class="format btn btn-secondary" role="button" onclick="makeQuote('post-edit-box-{{p.id}}')" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Quote"><i class="fas fa-quote-right" aria-hidden="true"></i></a>
|
|
<a class="format btn btn-secondary" role="button" onclick="commentForm('post-edit-box-{{p.id}}');getGif()" aria-hidden="true" data-bs-toggle="modal" data-bs-target="#gifModal" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Add GIF"><span class="font-weight-bolder text-uppercase">GIF</span></a>
|
|
<label class="format btn btn-secondary m-0 ml-1 {% if v %}d-inline-block{% else %}d-none{% endif %}" for="file-upload-edit-{{p.id}}">
|
|
<div id="filename-show-edit-{{p.id}}"><i class="far fa-image"></i></div>
|
|
<input autocomplete="off" id="file-upload-edit-{{p.id}}" type="file" multiple="multiple" name="file" accept="image/*, video/*" {% if request.headers.get('cf-ipcountry')=="T1" %}disabled{% endif %} onchange="changename('filename-show-edit-{{p.id}}','file-upload-edit-{{p.id}}')" hidden>
|
|
</label>
|
|
|
|
<small class="format d-none"><i class="fas fa-link" aria-hidden="true"></i></small>
|
|
</div>
|
|
</div>
|
|
<button form="post-edit-form-{{p.id}}" class="btn btn-primary ml-2 fl-r">Save Edit</button>
|
|
<a role="button" onclick="togglePostEdit('{{p.id}}')" class="btn btn-link text-muted ml-auto cancel-form fl-r">Cancel</a>
|
|
</form>
|
|
<div id="post-edit-{{p.id}}" class="preview mb-3 mt-5"></div>
|
|
<div class="form-text text-small p-0 m-0"><a href="/formatting" {% if v and v.newtab and not g.webview %}target="_blank"{% endif %}>Formatting help</a></div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if p.embed_url and "http" not in p.embed_url and "<" not in p.embed_url %}
|
|
<div id="crosspost-embed">
|
|
<div class="row no-gutters">
|
|
<div id="frontpage" class="col-12 pt-0">
|
|
<div class="posts" id="posts">
|
|
{{p.embed_url | post_embed(v) | safe}}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
<div class="d-none d-md-flex justify-content-between align-items-center mt-2">
|
|
<div class="post-actions mt-2">
|
|
<ul class="list-inline text-right d-flex">
|
|
|
|
<li class="list-inline-item">
|
|
<a {% if v and v.newtab and not g.webview %}target="_blank"{% endif %} href="{{p.permalink}}">
|
|
<i class="fas fa-comment-dots"></i>{{p.comment_count}}
|
|
<span class="text-info d-none {{p.id}}-new-comments"></span>
|
|
</a>
|
|
</li>
|
|
|
|
{% if v and (v.id==p.author_id or v.admin_level > 1 and v.admin_level > 2) %}
|
|
<a class="list-inline-item" role="button" onclick="togglePostEdit('{{p.id}}')"><i class="fas fa-edit"></i>Edit</a>
|
|
{% endif %}
|
|
|
|
{% if v and v.id != p.author_id and p.body %}
|
|
<a class="list-inline-item" role="button" onclick="expandMarkdown(this,'{{p.id}}')"><i class="fas text-expand-icon-{{p.id}} fa-expand-alt"></i><span>View source</span></a>
|
|
{% endif %}
|
|
|
|
{% include 'post_actions.html' %}
|
|
</ul>
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{% if v %}
|
|
|
|
<div id="voting" class="voting d-none d-md-block mb-auto">
|
|
<div tabindex="0" role="button" onclick="vote('post', '{{p.id}}', '1')" class="post-{{p.id}}-up mx-auto arrow-up upvote-button post-{{p.id}}-up {% if voted==1 %}active{% endif %}"></div>
|
|
|
|
<span class="post-score-{{p.id}} score post-score-{{p.id}} {% if voted==1 %}score-up{% elif voted==-1%}score-down{% endif %}{% if p.controversial %} controversial{% endif %}" data-bs-toggle="tooltip" data-bs-placement="right" title="+{{ups}} | -{{downs}}">{{score}}</span>
|
|
<div {% if environ.get('DISABLE_DOWNVOTES') == '1' %}style="display:None!important"{% endif %} tabindex="0" role="button" onclick="vote('post', '{{p.id}}', '-1')" class="post-{{p.id}}-down text-muted mx-auto arrow-down downvote-button post-{{p.id}}-down {% if voted==-1 %}active{% endif %}"></div>
|
|
</div>
|
|
|
|
{% else %}
|
|
|
|
<div id="voting" class="voting d-none d-md-block mb-auto">
|
|
<div tabindex="0" role="button" onclick="vote('post', '{{p.id}}', '1')" class="post-{{p.id}}-up arrow-up mx-auto" onclick="location.href='/login?redirect={{request.path | urlencode}}';">
|
|
</div>
|
|
<span class="post-{{p.id}}-score-none score text-muted{% if p.controversial %} controversial{% endif %}"{% if not p.is_banned %} data-bs-toggle="tooltip" data-bs-placement="right" title="+{{ups}} | -{{downs}}"{% endif %}>{{score}}</span>
|
|
<div {% if environ.get('DISABLE_DOWNVOTES') == '1' %}style="display:None!important"{% endif %} tabindex="0" role="button" onclick="vote('post', '{{p.id}}', '-1')" class="post-{{p.id}}-down arrow-down mx-auto" onclick="location.href='/login?redirect={{request.path | urlencode}}';"></div>
|
|
</div>
|
|
|
|
{% endif %}
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{% if not p.is_image and not p.is_video %}
|
|
<div class="row no-gutters d-md-none">
|
|
<div class="col">
|
|
<a {% if not v or v.newtabexternal %}target="_blank"{% endif %} rel="nofollow noopener noreferrer" href="{{p.realurl(v)}}"></a>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
</div>
|
|
|
|
<div class="row mb-3 d-md-none">
|
|
<div class="col-12">
|
|
|
|
<div class="post-actions">
|
|
<ul class="list-inline text-right d-flex">
|
|
|
|
<li class="list-inline-item mr-auto">
|
|
<a href="{{p.permalink}}">
|
|
<i class="fas fa-comment-dots"></i>{{p.comment_count}}
|
|
<span class="text-info d-none {{p.id}}-new-comments"></span>
|
|
</a>
|
|
|
|
{% if v and v.admin_level > 1 %}
|
|
<a class="ml-2" role="button" data-bs-toggle="modal" data-bs-target="#adminModal-{{p.id}}">
|
|
<i class="fas fa-broom"></i>
|
|
</a>
|
|
{% endif %}
|
|
</li>
|
|
|
|
{% if v %}
|
|
<li class="list-inline-item">
|
|
<a role="button" data-bs-toggle="modal" data-bs-target="#actionsModal">
|
|
<i class="fas fa-ellipsis-h"></i>
|
|
</a>
|
|
</li>
|
|
{% endif %}
|
|
<li id="voting-{{p.id}}-mobile" class="voting list-inline-item d-md-none{% if voted==1 %} upvoted{% elif voted==-1 %} downvoted{% endif %}">
|
|
{% if v %}
|
|
<span tabindex="0" role="button" onclick="vote('post-mobile', '{{p.id}}', '1')" class="post-mobile-{{p.id}}-up mx-0 pr-1 arrow-up upvote-button post-{{p.id}}-up {% if voted==1 %}active{% endif %}">
|
|
</span>
|
|
{% else %}
|
|
<span tabindex="0" class="arrow-{{p.id}}-mobile-up mx-0 pr-1 arrow-mobile-up" onclick="location.href='/login?redirect={{request.path | urlencode}}';">
|
|
<i class="fas fa-arrow-alt-up mx-0" aria-hidden="true"></i>
|
|
</span>
|
|
{% endif %}
|
|
|
|
<span class="post-mobile-score-{{p.id}} score post-score-{{p.id}} {% if voted==1 %}score-up{% elif voted==-1%}score-down{% endif %}{% if p.controversial %} controversial{% endif %}" data-bs-toggle="tooltip" data-bs-placement="top" title="+{{ups}} | -{{downs}}">{{score}}</span>
|
|
|
|
{% if v %}
|
|
<span {% if environ.get('DISABLE_DOWNVOTES') == '1' %}style="display:None!important"{% endif %} tabindex="0" role="button" onclick="vote('post-mobile', '{{p.id}}', '-1')" class="post-mobile-{{p.id}}-down mx-0 pl-1 my-0 arrow-down downvote-button post-{{p.id}}-down {% if voted==-1 %}active{% endif %}"></span>
|
|
{% else %}
|
|
<span {% if environ.get('DISABLE_DOWNVOTES') == '1' %}style="display:None!important"{% endif %} tabindex="0" class="arrow-{{p.id}}-mobile-down arrow-mobile-down mx-0 pl-1 my-0" onclick="location.href='/login?redirect={{request.path | urlencode}}';">
|
|
<i class="fas fa-arrow-alt-down mx-0" aria-hidden="true"></i>
|
|
</span>
|
|
{% endif %}
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{% if v and v.id != p.author_id and p.body %}
|
|
<textarea autocomplete="off" class="d-none card border my-2 p-3 comment-box form-control rounded" id="markdown-{{p.id}}" readonly>{{p.body.strip()}}</textarea>
|
|
{% endif %}
|
|
|
|
<div class="row border-md-0 comment-section pb-3">
|
|
<div class="col border-top">
|
|
<div class="comments-count py-3">
|
|
<div class="dropdown dropdown-actions">
|
|
<button class="btn btn-secondary dropdown-toggle" id="dropdownMenuButton" data-bs-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
|
|
{% if sort=="top" %}<i class="fas fa-arrow-alt-circle-up mr-1"></i>{% endif %}
|
|
{% if sort=="bottom" %}<i class="fas fa-arrow-alt-circle-down mr-1"></i>{% endif %}
|
|
{% if sort=="new" %}<i class="fas fa-sparkles mr-1"></i>{% endif %}
|
|
{% if sort=="old" %}<i class="fas fa-book mr-1"></i>{% endif %}
|
|
{% if sort=="controversial" %}<i class="fas fa-bullhorn mr-1"></i>{% endif %}
|
|
{{sort | capitalize}}
|
|
</button>
|
|
<div class="dropdown-menu" aria-labelledby="dropdownMenuButton" x-placement="bottom-start" style="position: absolute; will-change: transform; top: 0px; left: 0px; transform: translate3d(0px, 31px, 0px);">
|
|
{% if sort != "top" %}<a class="dropdown-item" href="?sort=top"><i class="fas fa-arrow-alt-circle-up mr-2"></i>Top</a>{% endif %}
|
|
{% if sort != "bottom" %}<a class="dropdown-item" href="?sort=bottom"><i class="fas fa-arrow-alt-circle-down mr-2"></i>Bottom</a>{% endif %}
|
|
{% if sort != "new" %}<a class="dropdown-item" href="?sort=new"><i class="fas fa-sparkles mr-2"></i>New</a>{% endif %}
|
|
{% if sort != "old" %}<a class="dropdown-item" href="?sort=old"><i class="fas fa-book mr-2"></i>Old</a>{% endif %}
|
|
{% if sort != "controversial" %}<a class="dropdown-item" href="?sort=controversial"><i class="fas fa-bullhorn mr-2"></i>Controversial</a>{% endif %}
|
|
</div>
|
|
{% if comment_info and p.comment_count >= 2%}
|
|
<pre></pre>
|
|
<div class="total"><a href="{{p.permalink}}">View entire discussion</a></div>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
|
|
{% if v %}
|
|
<div id="comment-form-space-{{p.fullname}}" class="comment-write mb-3">
|
|
<form id="reply-to-{{p.fullname}}" action="/comment" method="post">
|
|
<input type="hidden" name="formkey" value="{{v.formkey}}">
|
|
<input type="hidden" name="parent_fullname" value="t2_{{p.id}}">
|
|
<input autocomplete="off" id="reply-form-submission-{{p.fullname}}" type="hidden" name="submission" value="{{p.id}}">
|
|
<textarea required autocomplete="off" minlength="1" maxlength="10000" oninput="markdown('reply-form-body-{{p.fullname}}', 'form-preview-{{p.id}}');charLimit('reply-form-body-{{p.fullname}}','charcount-reply')" id="reply-form-body-{{p.fullname}}" data-fullname="{{p.fullname}}" class="comment-box form-control rounded" id="comment-form" name="body" form="reply-to-{{p.fullname}}" aria-label="With textarea" placeholder="Add your comment..." rows="3"></textarea>
|
|
|
|
<div class="text-small font-weight-bold mt-1" id="charcount-reply" style="right: 1rem; bottom: 0.5rem; z-index: 3;"></div>
|
|
|
|
<div class="comment-format">
|
|
<a class="btn btn-secondary format d-inline-block m-0" role="button" onclick="makeBold('reply-form-body-{{p.fullname}}')" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Bold"><i class="fas fa-bold"></i></a>
|
|
|
|
<a class="btn btn-secondary format d-inline-block m-0" role="button" onclick="makeItalics('reply-form-body-{{p.fullname}}')" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Italicize"><i class="fas fa-italic"></i></a>
|
|
|
|
<a class="btn btn-secondary format d-inline-block m-0" role="button" onclick="makeQuote('reply-form-body-{{p.fullname}}')" data-bs-toggle="tooltip" data-bs-placement="bottom" title="Quote"><i class="fas fa-quote-right"></i></a>
|
|
|
|
<label class="format btn btn-secondary m-0 ml-1 {% if v %}d-inline-block{% else %}d-none{% endif %}" for="file-upload-reply-{{p.fullname}}">
|
|
<div id="filename-show-reply-{{p.fullname}}"><i class="far fa-image"></i></div>
|
|
<input autocomplete="off" id="file-upload-reply-{{p.fullname}}" type="file" multiple="multiple" name="file" accept="image/*, video/*" {% if request.headers.get('cf-ipcountry')=="T1" %}disabled{% endif %} onchange="changename('filename-show-reply-{{p.fullname}}','file-upload-reply-{{p.fullname}}')" hidden>
|
|
</label>
|
|
</div>
|
|
<a id="save-reply-to-{{p.fullname}}" role="button" form="reply-to-{{p.fullname}}" class="btn btn-primary text-whitebtn ml-auto fl-r" onclick="post_comment('{{p.fullname}}', '{{p.id}}')">Comment</a>
|
|
</form>
|
|
<div id="form-preview-{{p.id}}" class="text-small mb-3 mt-5"><p class="preview-msg">Comment preview</p></div>
|
|
<div class="form-text text-small p-0 m-0"><a href="/formatting" {% if v and v.newtab and not g.webview %}target="_blank"{% endif %}>Formatting help</a></div>
|
|
</div>
|
|
{% else %}
|
|
<div class="comment-write mb-3">
|
|
<textarea autocomplete="off" maxlength="10000" class="comment-box form-control rounded" name="body" aria-label="With textarea" placeholder="Add your comment..." rows="3" onclick="location.href='/login?redirect={{request.path | urlencode}}';"></textarea>
|
|
</div>
|
|
|
|
<div class="card border-0 mt-4">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Jump in the discussion.</h5>
|
|
<p class="card-text">No email address required.</p>
|
|
<div>
|
|
<a href="/signup" class="btn btn-primary">Sign up</a>
|
|
<a href="/login?redirect={{request.path | urlencode}}" class="btn btn-link text-muted">Sign in</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
{% if p.replies %}
|
|
<div class="comment-section" id="replies-of-{{p.id}}">
|
|
{% with comments=p.replies %}
|
|
{% include "comments.html" %}
|
|
{% endwith %}
|
|
</div>
|
|
|
|
{% if offset %}
|
|
<script>
|
|
function viewmore(pid,sort,offset,ids) {
|
|
btn = document.getElementById("viewbtn");
|
|
btn.disabled = true;
|
|
btn.innerHTML = "Requesting...";
|
|
var form = new FormData();
|
|
const xhr = new XMLHttpRequest();
|
|
xhr.open("get", `/viewmore/${pid}/${sort}/${offset}?ids=${ids}`);
|
|
xhr.setRequestHeader('xhr', 'xhr');
|
|
xhr.onload=function(){
|
|
if (xhr.status==200) {
|
|
const e = document.getElementById(`viewmore-${offset}`);
|
|
const html = xhr.response.replace(/data-src/g, "src").replace(/data-cfsrc/g, "src").replace(/style="display:none;visibility:hidden;"/g, "");
|
|
const fragment = document.createRange().createContextualFragment(html);
|
|
bs_trigger(fragment);
|
|
e.replaceWith(fragment);
|
|
|
|
comments = JSON.parse(localStorage.getItem("old-comment-counts")) || {}
|
|
lastCount = comments['{{p.id}}']
|
|
if (lastCount)
|
|
{
|
|
{% for c in p.comments %}
|
|
{% if not (v and v.id==c.author_id) and not c.voted %}
|
|
if ({{c.created_utc*1000}} > lastCount.t)
|
|
try {document.getElementById("comment-{{c.id}}-only").classList.add('unread')}
|
|
catch(e) {}
|
|
{% endif %}
|
|
{% endfor %}
|
|
}
|
|
|
|
collapsedCommentStorageApply();
|
|
}
|
|
btn.disabled = false;
|
|
}
|
|
xhr.send(form)
|
|
}
|
|
</script>
|
|
{% endif %}
|
|
|
|
{% elif not p.replies and p.deleted_utc == 0 %}
|
|
<div class="comment-section text-center py-7" id="replies-of-{{p.id}}">
|
|
<div class="row no-gutters" id="placeholder-comment">
|
|
<div class="col">
|
|
<span class="fa-stack fa-2x text-muted mb-4">
|
|
<i class="fas fa-square text-gray-500 opacity-25 fa-stack-2x"></i>
|
|
<i class="fas text-gray-500 fa-ghost fa-stack-1x text-lg"></i>
|
|
</span>
|
|
<h2 class="h5">Be the first to comment!</h2>
|
|
{% if v %}
|
|
<p class="text-muted">This comment section is a ghost town...</p>
|
|
{% else %}
|
|
<p class="text-muted">This comment section is a ghost town. <a href="/login?redirect={{request.path | urlencode}}">Sign in</a></p>
|
|
{% endif %}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{% if v and v.id==p.author_id %}
|
|
{% include "delete_post_modal.html" %}
|
|
{% endif %}
|
|
|
|
{% if v %}
|
|
{% include "report_post_modal.html" %}
|
|
{% endif %}
|
|
|
|
{% if v and (v.id == p.author_id or v.admin_level > 1 and v.admin_level > 2) %}
|
|
<script src="{{ 'js/togglePostEdit.js' | asset }}"></script>
|
|
{% endif %}
|
|
|
|
{% if not v or v.highlightcomments %}
|
|
<script src="{{ 'js/new_comments_count.js' | asset }}"></script>
|
|
{% endif %}
|
|
|
|
<script src="{{ 'js/clipboard.js' | asset }}"></script>
|
|
|
|
{% if not p.replies %}
|
|
{% include "comments.html" %}
|
|
{% endif %}
|
|
|
|
{% if p.award_count("shit") %}
|
|
<script src="{{ 'js/critters.js' | asset }}"></script>
|
|
<script src="{{ 'js/bugs.js' | asset }}"></script>
|
|
{% endif %}
|
|
|
|
|
|
{% if p.award_count("fireflies") %}
|
|
<script src="{{ 'js/critters.js' | asset }}"></script>
|
|
<script src="{{ 'js/fireflies.js' | asset }}"></script>
|
|
{% endif %}
|
|
|
|
|
|
<script>
|
|
(() => {
|
|
{% if not v or v.highlightcomments %}
|
|
showNewCommentCounts('{{p.id}}', {{p.comment_count}})
|
|
{% endif %}
|
|
|
|
{% if "?context" not in request.full_path %}
|
|
localStorage.setItem("old-comment-counts", localStorage.getItem("comment-counts"))
|
|
|
|
const comments = JSON.parse(localStorage.getItem("comment-counts")) || {}
|
|
const newTotal = {{p.comment_count}} || ((comments['{{p.id}}'] || {c: 0}).c + 1)
|
|
comments['{{p.id}}'] = {c: newTotal, t: Date.now()}
|
|
localStorage.setItem("comment-counts", JSON.stringify(comments))
|
|
{% endif %}
|
|
})()
|
|
</script>
|
|
{% endblock %}
|