2022-12-01 18:48:31 +00:00
|
|
|
{% extends 'layout.html' %}
|
2023-04-06 18:42:04 +00:00
|
|
|
{% block page_index %}
|
2023-04-07 09:18:03 +00:00
|
|
|
{% if return_page %}?page={{ return_page }}{% endif %}{% endblock %}
|
2023-03-15 16:19:42 +00:00
|
|
|
{% block head %}
|
2023-04-21 17:34:25 +00:00
|
|
|
<meta property="og:image" content="{{ url_for('media_api.media', path='uploads/' + image.filename) }}"/>
|
2023-04-22 09:02:05 +00:00
|
|
|
<meta name="twitter:image" content="{{ url_for('media_api.media', path='uploads/' + image.filename) }}">
|
2023-04-22 11:46:20 +00:00
|
|
|
<meta name="theme-color" content="rgb{{ image.colours.0 }}"/>
|
2023-04-22 09:02:05 +00:00
|
|
|
<meta name="twitter:card" content="summary_large_image">
|
2023-03-15 16:19:42 +00:00
|
|
|
|
2023-03-27 07:15:18 +00:00
|
|
|
<script type="text/javascript">
|
2023-04-07 14:51:37 +00:00
|
|
|
function fullscreen() {
|
2023-04-07 10:57:25 +00:00
|
|
|
let info = document.querySelector('.info-container');
|
|
|
|
let wrapper = document.querySelector('.image-grid');
|
|
|
|
|
|
|
|
if (info.classList.contains('collapsed')) {
|
|
|
|
info.classList.remove('collapsed');
|
|
|
|
wrapper.classList.remove('collapsed');
|
|
|
|
} else {
|
|
|
|
info.classList.add('collapsed');
|
|
|
|
wrapper.classList.add('collapsed');
|
|
|
|
}
|
|
|
|
}
|
2023-03-26 01:04:13 +00:00
|
|
|
|
2023-04-12 15:16:43 +00:00
|
|
|
{% if current_user.id == image.author.id %}
|
2023-03-26 01:04:13 +00:00
|
|
|
function imageDelete() {
|
2023-04-04 19:36:24 +00:00
|
|
|
cancelBtn = document.createElement('button');
|
|
|
|
cancelBtn.classList.add('btn-block');
|
2023-04-19 17:16:40 +00:00
|
|
|
cancelBtn.classList.add('transparent');
|
2023-04-04 19:36:24 +00:00
|
|
|
cancelBtn.innerHTML = 'nuuuuuuuu';
|
|
|
|
cancelBtn.onclick = popupDissmiss;
|
|
|
|
|
|
|
|
deleteBtn = document.createElement('button');
|
|
|
|
deleteBtn.classList.add('btn-block');
|
|
|
|
deleteBtn.classList.add('critical');
|
|
|
|
deleteBtn.innerHTML = 'Dewww eeeet!';
|
|
|
|
deleteBtn.onclick = deleteConfirm;
|
|
|
|
|
2023-04-03 03:43:20 +00:00
|
|
|
popUpShow('DESTRUCTION!!!!!!',
|
|
|
|
'Do you want to delete this image along with all of its data??? ' +
|
|
|
|
'This action is irreversible!',
|
|
|
|
null,
|
|
|
|
[cancelBtn, deleteBtn]);
|
2023-03-26 01:04:13 +00:00
|
|
|
}
|
|
|
|
function deleteConfirm() {
|
|
|
|
popupDissmiss();
|
|
|
|
|
2023-04-21 17:34:25 +00:00
|
|
|
fetch('{{ url_for('media_api.delete_image', image_id=image['id']) }}', {
|
2023-04-03 03:43:20 +00:00
|
|
|
method: 'POST',
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json'
|
2023-03-26 01:04:13 +00:00
|
|
|
},
|
2023-04-03 03:43:20 +00:00
|
|
|
body: JSON.stringify({
|
|
|
|
action: 'delete'
|
|
|
|
})
|
|
|
|
}).then(function(response) {
|
|
|
|
if (response.ok) {
|
2023-03-26 01:04:13 +00:00
|
|
|
window.location.href = '/';
|
2023-04-03 03:43:20 +00:00
|
|
|
} else {
|
2023-04-07 14:51:37 +00:00
|
|
|
addNotification(`Image *clings*`, 2);
|
2023-03-26 01:04:13 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function imageEdit() {
|
|
|
|
addNotification("Not an option, oops!", 3);
|
|
|
|
}
|
|
|
|
{% endif %}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style>
|
|
|
|
.background span {
|
2023-04-06 16:22:56 +00:00
|
|
|
background-image: linear-gradient(to top, rgba({{ image.colours.0.0 }}, {{ image.colours.0.1 }}, {{ image.colours.0.2 }}, 0.8),
|
|
|
|
rgba({{ image.colours.1.0 }}, {{ image.colours.1.1 }}, {{ image.colours.1.2 }}, 0.2));
|
2023-03-26 01:04:13 +00:00
|
|
|
}
|
|
|
|
</style>
|
|
|
|
{% endblock %}
|
2023-03-08 13:36:35 +00:00
|
|
|
{% block content %}
|
|
|
|
<div class="background">
|
2023-04-22 11:46:20 +00:00
|
|
|
<picture>
|
|
|
|
<source srcset="{{ url_for('media_api.media', path='uploads/' + image.filename) }}?r=prev&e=webp">
|
|
|
|
<source srcset="{{ url_for('media_api.media', path='uploads/' + image.filename) }}?r=prev&e=png">
|
|
|
|
<img
|
|
|
|
src="{{ url_for('media_api.media', path='uploads/' + image.filename) }}?r=prev"
|
|
|
|
alt="{{ image.alt }}"
|
|
|
|
onload="imgFade(this)"
|
|
|
|
style="opacity:0;"
|
|
|
|
/>
|
|
|
|
</picture>
|
2023-03-26 01:04:13 +00:00
|
|
|
<span></span>
|
2023-01-25 15:13:56 +00:00
|
|
|
</div>
|
2023-03-08 13:36:35 +00:00
|
|
|
<div class="image-grid">
|
2023-04-07 14:51:37 +00:00
|
|
|
<div class="image-block">
|
|
|
|
<div class="image-container">
|
2023-04-22 11:46:20 +00:00
|
|
|
<picture>
|
|
|
|
<source srcset="{{ url_for('media_api.media', path='uploads/' + image.filename) }}?r=prev&e=webp">
|
|
|
|
<source srcset="{{ url_for('media_api.media', path='uploads/' + image.filename) }}?r=prev&e=png">
|
|
|
|
<img
|
|
|
|
src="{{ url_for('media_api.media', path='uploads/' + image.filename) }}?r=prev"
|
|
|
|
alt="{{ image.alt }}"
|
|
|
|
onload="imgFade(this)"
|
|
|
|
style="opacity:0;"
|
|
|
|
onerror="this.src='{{ url_for('static', filename='error.png')}}'"
|
|
|
|
{% if "File" in image.exif %}
|
|
|
|
width="{{ image.exif.File.Width.raw }}"
|
|
|
|
height="{{ image.exif.File.Height.raw }}"
|
|
|
|
{% endif %}
|
|
|
|
/>
|
|
|
|
</picture>
|
2023-01-13 18:29:07 +00:00
|
|
|
</div>
|
2023-04-07 14:51:37 +00:00
|
|
|
<div class="pill-row">
|
2023-04-20 16:15:51 +00:00
|
|
|
{% if next_url %}<div><a class="pill-item" href="{{ next_url }}"><i class="ph ph-arrow-left"></i></a></div>{% endif %}
|
2023-03-08 13:36:35 +00:00
|
|
|
<div>
|
2023-04-20 16:15:51 +00:00
|
|
|
<button class="pill-item" onclick="fullscreen()" id="fullscreenImage"><i class="ph ph-info"></i></button>
|
2023-04-22 13:48:06 +00:00
|
|
|
<button class="pill-item" onclick="copyToClipboard(window.location.href)"><i class="ph ph-export"></i></button>
|
2023-04-21 17:34:25 +00:00
|
|
|
<a class="pill-item" href="{{ url_for('media_api.media', path='uploads/' + image.filename) }}" download onclick="addNotification('Download started!', 4)"><i class="ph ph-file-arrow-down"></i></a>
|
2023-03-09 12:22:25 +00:00
|
|
|
</div>
|
2023-04-12 15:16:43 +00:00
|
|
|
{% if current_user.id == image.author.id %}
|
2023-04-07 14:51:37 +00:00
|
|
|
<div>
|
2023-04-20 16:15:51 +00:00
|
|
|
<button class="pill-item pill__critical" onclick="imageDelete()"><i class="ph ph-trash"></i></button>
|
|
|
|
<button class="pill-item pill__critical" onclick="imageEdit()"><i class="ph ph-pencil-simple"></i></button>
|
2023-04-07 14:51:37 +00:00
|
|
|
</div>
|
|
|
|
{% endif %}
|
2023-04-20 16:15:51 +00:00
|
|
|
{% if prev_url %}<div><a class="pill-item" href="{{ prev_url }}"><i class="ph ph-arrow-right"></i></a></div>{% endif %}
|
2023-04-07 14:51:37 +00:00
|
|
|
</div>
|
2023-03-08 13:36:35 +00:00
|
|
|
</div>
|
2023-04-07 14:51:37 +00:00
|
|
|
<div class="info-container">
|
2023-03-08 13:36:35 +00:00
|
|
|
<div class="info-tab">
|
|
|
|
<div class="info-header">
|
2023-04-20 16:15:51 +00:00
|
|
|
<i class="ph ph-info"></i>
|
2023-03-08 13:36:35 +00:00
|
|
|
<h2>Info</h2>
|
2023-04-20 16:15:51 +00:00
|
|
|
<button class="collapse-indicator"><i class="ph ph-caret-down"></i></button>
|
2023-03-08 13:36:35 +00:00
|
|
|
</div>
|
|
|
|
<div class="info-table">
|
|
|
|
<table>
|
|
|
|
<tr>
|
|
|
|
<td>Author</td>
|
2023-04-22 09:39:27 +00:00
|
|
|
<td>
|
|
|
|
<img
|
|
|
|
src="{{ url_for('media_api.media', path='pfp/' + image.author.picture) }}"
|
|
|
|
alt="Profile Picture"
|
|
|
|
class="pfp"
|
|
|
|
onload="imgFade(this)"
|
|
|
|
style="opacity: 0;"
|
|
|
|
/>
|
|
|
|
<a href="{{ url_for('profile.profile', id=image.author.id) }}" class="link">{{ image.author.username }}</a>
|
|
|
|
</td>
|
2023-03-08 13:36:35 +00:00
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>Upload date</td>
|
2023-03-09 23:31:58 +00:00
|
|
|
<td><span class="time">{{ image.created_at }}</span></td>
|
2023-03-08 13:36:35 +00:00
|
|
|
</tr>
|
2023-04-09 18:00:10 +00:00
|
|
|
{% if image.description %}
|
|
|
|
<tr>
|
|
|
|
<td>Description</td>
|
|
|
|
<td>{{ image.description }}</td>
|
|
|
|
</tr>
|
|
|
|
{% endif %}
|
2023-03-08 13:36:35 +00:00
|
|
|
</table>
|
2023-04-01 16:16:07 +00:00
|
|
|
<div class="img-colours">
|
2023-04-05 16:35:59 +00:00
|
|
|
{% for col in image.colours %}
|
2023-04-22 14:11:25 +00:00
|
|
|
<button style="background-color: rgb{{ col }}" onclick="copyToClipboard('rgb{{ col }}')">
|
|
|
|
<i class="ph-fill ph-paint-bucket" style="{{ col|colour_contrast }}"></i>
|
|
|
|
</button>
|
2023-04-01 16:16:07 +00:00
|
|
|
{% endfor %}
|
|
|
|
</div>
|
|
|
|
{% if image.groups %}
|
2023-03-12 15:52:23 +00:00
|
|
|
<div class="img-groups">
|
|
|
|
{% for group in image.groups %}
|
2023-04-01 16:16:07 +00:00
|
|
|
<a href="{{ url_for('group.group', group_id=group.id) }}" class="tag-icon">
|
2023-04-19 22:58:31 +00:00
|
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" fill="currentColor" viewBox="0 0 256 256"><path d="M223.68,66.15,135.68,18a15.88,15.88,0,0,0-15.36,0l-88,48.17a16,16,0,0,0-8.32,14v95.64a16,16,0,0,0,8.32,14l88,48.17a15.88,15.88,0,0,0,15.36,0l88-48.17a16,16,0,0,0,8.32-14V80.18A16,16,0,0,0,223.68,66.15ZM128,32l80.34,44-29.77,16.3-80.35-44ZM128,120,47.66,76l33.9-18.56,80.34,44ZM40,90l80,43.78v85.79L40,175.82Zm176,85.78h0l-80,43.79V133.82l32-17.51V152a8,8,0,0,0,16,0V107.55L216,90v85.77Z"></path></svg>
|
2023-03-12 15:52:23 +00:00
|
|
|
{{ group['name'] }}
|
|
|
|
</a>
|
|
|
|
{% endfor %}
|
|
|
|
</div>
|
|
|
|
{% endif %}
|
2023-03-08 13:36:35 +00:00
|
|
|
</div>
|
2023-04-22 09:02:05 +00:00
|
|
|
</div>
|
2023-04-05 16:35:59 +00:00
|
|
|
{% for tag in image.exif %}
|
2023-03-08 13:36:35 +00:00
|
|
|
<div class="info-tab">
|
|
|
|
<div class="info-header">
|
|
|
|
{% if tag == 'Photographer' %}
|
2023-04-20 16:15:51 +00:00
|
|
|
<i class="ph ph-person"></i><h2>Photographer</h2>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% elif tag == 'Camera' %}
|
2023-04-20 16:15:51 +00:00
|
|
|
<i class="ph ph-camera"></i><h2>Camera</h2>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% elif tag == 'Software' %}
|
2023-04-20 16:15:51 +00:00
|
|
|
<i class="ph ph-desktop-tower"></i><h2>Software</h2>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% elif tag == 'File' %}
|
2023-04-20 16:15:51 +00:00
|
|
|
<i class="ph ph-file-image"></i><h2>File</h2>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% else %}
|
2023-04-20 16:15:51 +00:00
|
|
|
<i class="ph ph-file-image"></i><h2>{{ tag }}</h2>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% endif %}
|
2023-04-20 16:15:51 +00:00
|
|
|
<button class="collapse-indicator"><i class="ph ph-caret-down"></i></button>
|
2023-01-31 22:08:37 +00:00
|
|
|
</div>
|
2023-03-08 13:36:35 +00:00
|
|
|
<div class="info-table">
|
|
|
|
<table>
|
2023-04-05 16:35:59 +00:00
|
|
|
{% for subtag in image.exif[tag] %}
|
2023-03-08 13:36:35 +00:00
|
|
|
<tr>
|
2023-03-09 23:31:58 +00:00
|
|
|
<td>{{ subtag }}</td>
|
2023-04-05 16:35:59 +00:00
|
|
|
{% if image.exif[tag][subtag]['formatted'] %}
|
|
|
|
{% if image.exif[tag][subtag]['type'] == 'date' %}
|
|
|
|
<td><span class="time">{{ image.exif[tag][subtag]['formatted'] }}</span></td>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% else %}
|
2023-04-05 16:35:59 +00:00
|
|
|
<td>{{ image.exif[tag][subtag]['formatted'] }}</td>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% endif %}
|
2023-04-05 16:35:59 +00:00
|
|
|
{% elif image.exif[tag][subtag]['raw'] %}
|
|
|
|
<td>{{ image.exif[tag][subtag]['raw'] }}</td>
|
2023-01-31 17:32:22 +00:00
|
|
|
{% else %}
|
2023-03-08 13:36:35 +00:00
|
|
|
<td class="empty-table">Oops, an error</td>
|
2023-01-31 17:32:22 +00:00
|
|
|
{% endif %}
|
2023-03-08 13:36:35 +00:00
|
|
|
</tr>
|
|
|
|
{% endfor %}
|
|
|
|
</table>
|
|
|
|
</div>
|
2023-01-13 18:29:07 +00:00
|
|
|
</div>
|
2023-03-08 13:36:35 +00:00
|
|
|
{% endfor %}
|
|
|
|
</div>
|
2022-12-01 18:48:31 +00:00
|
|
|
</div>
|
2023-01-25 15:13:56 +00:00
|
|
|
{% endblock %}
|
|
|
|
|
|
|
|
{% block script %}
|
2023-03-27 07:15:18 +00:00
|
|
|
<script type="text/javascript">
|
2023-03-26 01:04:13 +00:00
|
|
|
let infoTab = document.querySelectorAll('.info-tab');
|
2023-04-07 14:51:37 +00:00
|
|
|
|
2023-03-26 01:04:13 +00:00
|
|
|
for (let i = 0; i < infoTab.length; i++) {
|
2023-04-22 09:39:27 +00:00
|
|
|
const tab = infoTab[i];
|
|
|
|
|
|
|
|
tab.querySelector('.collapse-indicator').addEventListener('click', function() {
|
|
|
|
tab.classList.toggle('collapsed');
|
|
|
|
});
|
|
|
|
|
|
|
|
/*
|
|
|
|
const tabHeader = tab.querySelector('.info-header');
|
|
|
|
const tabContent = tab.querySelector('.info-table');
|
|
|
|
|
|
|
|
const tabHeight = tabHeader.offsetHeight + tabContent.offsetHeight;
|
|
|
|
|
|
|
|
tab.style.height = tabHeight + 'px';
|
|
|
|
|
|
|
|
tab.querySelector('.collapse-indicator').addEventListener('click', function() {
|
|
|
|
if (tab.classList.contains('collapsed')) {
|
|
|
|
tab.style.height = tabHeight + 'px';
|
|
|
|
tab.classList.remove('collapsed');
|
|
|
|
} else {
|
|
|
|
tab.style.height = tabHeader.offsetHeight + 'px';
|
|
|
|
tab.classList.add('collapsed');
|
|
|
|
}
|
2023-01-31 23:44:44 +00:00
|
|
|
});
|
2023-04-22 09:39:27 +00:00
|
|
|
*/
|
2023-01-31 23:44:44 +00:00
|
|
|
}
|
2023-01-11 10:53:01 +00:00
|
|
|
</script>
|
2022-12-01 18:48:31 +00:00
|
|
|
{% endblock %}
|