123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270 |
- <template>
- <div class="container mt-2 mt-md-5">
- <input type="file" id="pf-dz" name="media" class="d-none file-input" v-bind:accept="config.mimes">
- <div class="row">
- <div class="col-12 col-md-6 offset-md-3">
- <!-- LANDING -->
- <div v-if="page == 'landing'" class="card card-body bg-transparent border-0 shadow-none d-flex justify-content-center" style="height: 90vh;">
- <div class="text-center flex-fill mt-5 pt-5">
- <img src="/img/pixelfed-icon-grey.svg" width="60px" height="60px">
- <p class="font-weight-bold lead text-lighter mt-1">Stories</p>
- </div>
- <div class="flex-fill">
- <div class="card w-100 shadow-none">
- <div class="list-group">
- <a class="list-group-item text-center lead text-decoration-none text-dark" href="#" @click.prevent="upload()">Add Photo</a>
- <a v-if="stories.length" class="list-group-item text-center lead text-decoration-none text-dark" href="#" @click.prevent="edit()">Edit Story</a>
- </div>
- </div>
- </div>
- <div class="text-center flex-fill">
- <p class="text-lighter small text-uppercase">
- <a href="/" class="text-muted font-weight-bold">Home</a>
- <span class="px-2 text-lighter">|</span>
- <a href="/i/my/story" class="text-muted font-weight-bold">View My Story</a>
- <span class="px-2 text-lighter">|</span>
- <a href="/site/help" class="text-muted font-weight-bold">Help</a>
- </p>
- </div>
- </div>
- <!-- CROP -->
- <div v-if="page == 'crop'" class="card card-body bg-transparent border-0 shadow-none d-flex justify-content-center" style="height: 95vh;">
- <div class="text-center pt-5 mb-3 d-flex justify-content-between align-items-center">
- <div>
- <button class="btn btn-outline-lighter btn-sm py-0 px-md-3"><i class="pr-2 fas fa-chevron-left fa-sm"></i> Delete</button>
- </div>
- <div class="d-flex align-items-center">
- <img class="d-inline-block mr-2" src="/img/pixelfed-icon-grey.svg" width="30px" height="30px">
- <span class="font-weight-bold lead text-lighter">Stories</span>
- </div>
- <div>
- <button class="btn btn-outline-success btn-sm py-0 px-md-3">Crop <i class="pl-2 fas fa-chevron-right fa-sm"></i></button>
- </div>
- </div>
- <div class="flex-fill">
- <div class="card w-100 mt-3">
- <div class="card-body p-0">
- <vue-cropper
- ref="cropper"
- :relativeZoom="cropper.zoom"
- :aspectRatio="cropper.aspectRatio"
- :viewMode="cropper.viewMode"
- :zoomable="cropper.zoomable"
- :rotatable="true"
- :src="mediaUrl"
- >
- </vue-cropper>
- </div>
- </div>
- </div>
- <div class="text-center flex-fill">
- <p class="text-lighter small text-uppercase pt-2">
- <!-- <a href="#" class="text-muted font-weight-bold">Home</a>
- <span class="px-2 text-lighter">|</span>
- <a href="#" class="text-muted font-weight-bold">View My Story</a>
- <span class="px-2 text-lighter">|</span> -->
- <a href="/site/help" class="text-muted font-weight-bold mb-0">Help</a>
- </p>
- </div>
- </div>
- <!-- ERROR -->
- <div v-if="page == 'error'" class="card card-body bg-transparent border-0 shadow-none d-flex justify-content-center align-items-center" style="height: 90vh;">
- <p class="h3 mb-0">Oops!</p>
- <p class="text-muted lead">An error occurred, please try again later.</p>
- <p class="text-muted mb-0">
- <a class="btn btn-outline-secondary py-0 px-5 font-weight-bold" href="/">Go back</a>
- </p>
- </div>
- <div v-if="page == 'edit'" class="card card-body bg-transparent border-0 shadow-none d-flex justify-content-center" style="height: 90vh;">
- <div class="text-center flex-fill mt-5 pt-5">
- <img src="/img/pixelfed-icon-grey.svg" width="60px" height="60px">
- <p class="font-weight-bold lead text-lighter mt-1">Stories</p>
- </div>
- <div class="flex-fill py-5">
- <div class="card w-100 shadow-none" style="max-height: 500px; overflow-y: auto">
- <div class="list-group">
- <div v-for="(story, index) in stories" class="list-group-item text-center text-dark" href="#">
- <div class="media align-items-center">
- <img :src="story.src" class="img-fluid mr-3 cursor-pointer" width="70px" height="70px" @click="showLightbox(story)">
- <div class="media-body">
- <p class="mb-0">Expires</p>
- <p class="mb-0 text-muted small"><span>{{expiresTimestamp(story.expires_at)}}</span></p>
- </div>
- <div class="float-right">
- <button @click="deleteStory(story, index)" class="btn btn-danger btn-sm font-weight-bold text-uppercase">Delete</button>
- </div>
- </div>
- </div>
- </div>
- </div>
- </div>
- <div class="flex-fill text-center">
- <a class="btn btn-outline-secondary py-0 px-5 font-weight-bold" href="/i/stories/new">Go back</a>
- </div>
- </div>
- </div>
- </div>
- <b-modal
- id="lightbox"
- ref="lightboxModal"
- hide-header
- hide-footer
- centered
- size="lg"
- body-class="p-0"
- >
- <div v-if="lightboxMedia" class="w-100 h-100">
- <img :src="lightboxMedia.url" style="max-height: 100%; max-width: 100%">
- </div>
- </b-modal>
- </div>
- </template>
- <style type="text/css" scoped>
- </style>
- <script type="text/javascript">
- import VueTimeago from 'vue-timeago';
- import VueCropper from 'vue-cropperjs';
- import 'cropperjs/dist/cropper.css';
- export default {
- components: {
- VueCropper,
- VueTimeago
- },
- props: ['profile-id'],
- data() {
- return {
- config: window.App.config,
- mimes: [
- 'image/jpeg',
- 'image/png'
- ],
- page: 'landing',
- pages: [
- 'landing',
- 'crop',
- 'edit',
- 'confirm',
- 'error'
- ],
- uploading: false,
- uploadProgress: 100,
- cropper: {
- aspectRatio: 9/16,
- viewMode: 1,
- zoomable: true,
- zoom: null
- },
- mediaUrl: null,
- stories: [],
- lightboxMedia: false,
- };
- },
- mounted() {
- this.mediaWatcher();
- axios.get('/api/stories/v1/fetch/' + this.profileId)
- .then(res => this.stories = res.data);
- },
- methods: {
- upload() {
- let fi = $('.file-input[name="media"]');
- fi.trigger('click');
- },
- mediaWatcher() {
- let self = this;
- $(document).on('change', '#pf-dz', function(e) {
- self.triggerUpload();
- });
- },
- triggerUpload() {
- let self = this;
- self.uploading = true;
- let io = document.querySelector('#pf-dz');
- Array.prototype.forEach.call(io.files, function(io, i) {
- if(self.media && self.media.length + i >= self.config.uploader.album_limit) {
- swal('Error', 'You can only upload ' + self.config.uploader.album_limit + ' photos per album', 'error');
- self.uploading = false;
- self.page = 2;
- return;
- }
- let type = io.type;
- let validated = $.inArray(type, self.mimes);
- if(validated == -1) {
- swal('Invalid File Type', 'The file you are trying to add is not a valid mime type. Please upload a '+self.mimes+' only.', 'error');
- self.uploading = false;
- self.page = 'error';
- return;
- }
- let form = new FormData();
- form.append('file', io);
- let xhrConfig = {
- onUploadProgress: function(e) {
- let progress = Math.round( (e.loaded * 100) / e.total );
- self.uploadProgress = progress;
- }
- };
- axios.post('/api/stories/v1/add', form, xhrConfig)
- .then(function(e) {
- self.uploadProgress = 100;
- self.uploading = false;
- window.location.href = '/i/my/story';
- self.mediaUrl = e.data.media_url;
- }).catch(function(e) {
- self.uploading = false;
- io.value = null;
- let msg = e.response.data.message ? e.response.data.message : 'Something went wrong.'
- swal('Oops!', msg, 'warning');
- });
- io.value = null;
- self.uploadProgress = 0;
- });
- },
- expiresTimestamp(ts) {
- ts = new Date(ts * 1000);
- return ts.toDateString() + ' ' + ts.toLocaleTimeString();
- },
- edit() {
- this.page = 'edit';
- },
- showLightbox(story) {
- this.lightboxMedia = {
- url: story.src
- }
- this.$refs.lightboxModal.show();
- },
- deleteStory(story, index) {
- if(window.confirm('Are you sure you want to delete this Story?') != true) {
- return;
- }
- axios.delete('/api/stories/v1/delete/' + story.id)
- .then(res => {
- this.stories.splice(index, 1);
- if(this.stories.length == 0) {
- window.location.href = '/i/stories/new';
- }
- });
- }
- }
- }
- </script>
|