143 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
		
		
			
		
	
	
			143 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
|  | <template> | ||
|  |     <div class="guestbook"> | ||
|  |         <div v-show="loading" class="alert alert-info">Loading guestbook.</div> | ||
|  |         <div class="guestbook-controls"> | ||
|  |             <label v-show="canEdit" class="control-label">Unapproved only: | ||
|  |                 <input type="checkbox" v-model="unapprovedOnly"/> | ||
|  |             </label> | ||
|  |             <nav> | ||
|  |                 <ul class="pager"> | ||
|  |                     <li class="previous" v-show="page > 1"> | ||
|  |                         <a @click="previousPage"> | ||
|  |                             <span aria-hidden="true">←</span>Previous Page | ||
|  |                         </a> | ||
|  |                     </li> | ||
|  |                     <li class="next" v-show="hasNextPage"> | ||
|  |                         <a @click="nextPage"> | ||
|  |                             Next Page<span aria-hidden="true">→</span> | ||
|  |                         </a> | ||
|  |                     </li> | ||
|  |                 </ul> | ||
|  |             </nav> | ||
|  |         </div> | ||
|  |         <template v-if="!loading"> | ||
|  |             <div class="alert alert-info" v-show="posts.length === 0">No guestbook posts.</div> | ||
|  |             <guestbook-post :post="post" :can-edit="canEdit" v-for="post in posts" :key="post.id" @reload="getPage"></guestbook-post> | ||
|  |             <div v-if="authenticated" class="form-horizontal"> | ||
|  |                 <bbcode-editor v-model="newPost.message" :maxlength="5000" classes="form-control"></bbcode-editor> | ||
|  |                 <label class="control-label" | ||
|  |                     for="guestbookPostPrivate">Private(only visible to owner): </label> | ||
|  |                 <input type="checkbox" | ||
|  |                     class="form-control" | ||
|  |                     id="guestbookPostPrivate" | ||
|  |                     v-model="newPost.privatePost"/> | ||
|  |                 <label class="control-label" | ||
|  |                     for="guestbook-post-character">Character: </label> | ||
|  |                 <character-select id="guestbook-post-character" v-model="newPost.character"></character-select> | ||
|  |                 <button @click="makePost" class="btn btn-success" :disabled="newPost.posting">Post</button> | ||
|  |             </div> | ||
|  |         </template> | ||
|  |         <div class="guestbook-controls"> | ||
|  |             <nav> | ||
|  |                 <ul class="pager"> | ||
|  |                     <li class="previous" v-show="page > 1"> | ||
|  |                         <a @click="previousPage"> | ||
|  |                             <span aria-hidden="true">←</span>Previous Page | ||
|  |                         </a> | ||
|  |                     </li> | ||
|  |                     <li class="next" v-show="hasNextPage"> | ||
|  |                         <a @click="nextPage"> | ||
|  |                             Next Page<span aria-hidden="true">→</span> | ||
|  |                         </a> | ||
|  |                     </li> | ||
|  |                 </ul> | ||
|  |             </nav> | ||
|  |         </div> | ||
|  |     </div> | ||
|  | </template> | ||
|  | 
 | ||
|  | <script lang="ts"> | ||
|  |     import Vue from 'vue'; | ||
|  |     import Component from 'vue-class-component'; | ||
|  |     import {Prop, Watch} from 'vue-property-decorator'; | ||
|  |     import * as Utils from '../utils'; | ||
|  |     import {methods, Store} from './data_store'; | ||
|  |     import {Character, GuestbookPost} from './interfaces'; | ||
|  | 
 | ||
|  |     import GuestbookPostView from './guestbook_post.vue'; | ||
|  | 
 | ||
|  |     @Component({ | ||
|  |         components: { | ||
|  |             'guestbook-post': GuestbookPostView | ||
|  |         } | ||
|  |     }) | ||
|  |     export default class GuestbookView extends Vue { | ||
|  |         @Prop({required: true}) | ||
|  |         private readonly character: Character; | ||
|  | 
 | ||
|  |         loading = true; | ||
|  |         error = ''; | ||
|  |         authenticated = Store.authenticated; | ||
|  | 
 | ||
|  |         posts: GuestbookPost[] = []; | ||
|  | 
 | ||
|  |         private unapprovedOnly = false; | ||
|  |         private page = 1; | ||
|  |         hasNextPage = false; | ||
|  |         canEdit = false; | ||
|  |         private newPost = { | ||
|  |             posting: false, | ||
|  |             privatePost: false, | ||
|  |             character: Utils.Settings.defaultCharacter, | ||
|  |             message: '' | ||
|  |         }; | ||
|  | 
 | ||
|  |         async nextPage(): Promise<void> { | ||
|  |             this.page += 1; | ||
|  |             return this.getPage(); | ||
|  |         } | ||
|  | 
 | ||
|  |         async previousPage(): Promise<void> { | ||
|  |             this.page -= 1; | ||
|  |             return this.getPage(); | ||
|  |         } | ||
|  | 
 | ||
|  |         @Watch('unapprovedOnly') | ||
|  |         async getPage(): Promise<void> { | ||
|  |             try { | ||
|  |                 this.loading = true; | ||
|  |                 const guestbookState = await methods.guestbookPageGet(this.character.character.id, this.page, this.unapprovedOnly); | ||
|  |                 this.posts = guestbookState.posts; | ||
|  |                 this.hasNextPage = guestbookState.nextPage; | ||
|  |                 this.canEdit = guestbookState.canEdit; | ||
|  |             } catch(e) { | ||
|  |                 this.posts = []; | ||
|  |                 this.hasNextPage = false; | ||
|  |                 this.canEdit = false; | ||
|  |                 if(Utils.isJSONError(e)) | ||
|  |                     this.error = <string>e.response.data.error; | ||
|  |                 Utils.ajaxError(e, 'Unable to load guestbook posts.'); | ||
|  |             } finally { | ||
|  |                 this.loading = false; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         async makePost(): Promise<void> { | ||
|  |             try { | ||
|  |                 this.newPost.posting = true; | ||
|  |                 await methods.guestbookPostPost(this.character.character.id, this.newPost.character, this.newPost.message, | ||
|  |                     this.newPost.privatePost); | ||
|  |                 this.page = 1; | ||
|  |                 await this.getPage(); | ||
|  |             } catch(e) { | ||
|  |                 Utils.ajaxError(e, 'Unable to post new guestbook post.'); | ||
|  |             } finally { | ||
|  |                 this.newPost.posting = false; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         async show(): Promise<void> { | ||
|  |             return this.getPage(); | ||
|  |         } | ||
|  |     } | ||
|  | </script> |