wip
This commit is contained in:
		| @@ -1,20 +0,0 @@ | |||||||
| <mk-search-page> |  | ||||||
| 	<mk-ui ref="ui"> |  | ||||||
| 		<mk-search ref="search" query={ parent.opts.query }/> |  | ||||||
| 	</mk-ui> |  | ||||||
| 	<style lang="stylus" scoped> |  | ||||||
| 		:scope |  | ||||||
| 			display block |  | ||||||
| 	</style> |  | ||||||
| 	<script lang="typescript"> |  | ||||||
| 		import Progress from '../../../common/scripts/loading'; |  | ||||||
|  |  | ||||||
| 		this.on('mount', () => { |  | ||||||
| 			Progress.start(); |  | ||||||
|  |  | ||||||
| 			this.$refs.ui.refs.search.on('loaded', () => { |  | ||||||
| 				Progress.done(); |  | ||||||
| 			}); |  | ||||||
| 		}); |  | ||||||
| 	</script> |  | ||||||
| </mk-search-page> |  | ||||||
| @@ -1,96 +0,0 @@ | |||||||
| <mk-search-posts> |  | ||||||
| 	<div class="loading" v-if="isLoading"> |  | ||||||
| 		<mk-ellipsis-icon/> |  | ||||||
| 	</div> |  | ||||||
| 	<p class="empty" v-if="isEmpty">%fa:search%「{ query }」に関する投稿は見つかりませんでした。</p> |  | ||||||
| 	<mk-timeline ref="timeline"> |  | ||||||
| 		<yield to="footer"> |  | ||||||
| 			<template v-if="!parent.moreLoading">%fa:moon%</template> |  | ||||||
| 			<template v-if="parent.moreLoading">%fa:spinner .pulse .fw%</template> |  | ||||||
| 		</yield/> |  | ||||||
| 	</mk-timeline> |  | ||||||
| 	<style lang="stylus" scoped> |  | ||||||
| 		:scope |  | ||||||
| 			display block |  | ||||||
| 			background #fff |  | ||||||
|  |  | ||||||
| 			> .loading |  | ||||||
| 				padding 64px 0 |  | ||||||
|  |  | ||||||
| 			> .empty |  | ||||||
| 				display block |  | ||||||
| 				margin 0 auto |  | ||||||
| 				padding 32px |  | ||||||
| 				max-width 400px |  | ||||||
| 				text-align center |  | ||||||
| 				color #999 |  | ||||||
|  |  | ||||||
| 				> [data-fa] |  | ||||||
| 					display block |  | ||||||
| 					margin-bottom 16px |  | ||||||
| 					font-size 3em |  | ||||||
| 					color #ccc |  | ||||||
|  |  | ||||||
| 	</style> |  | ||||||
| 	<script lang="typescript"> |  | ||||||
| 		import parse from '../../common/scripts/parse-search-query'; |  | ||||||
|  |  | ||||||
| 		this.mixin('api'); |  | ||||||
|  |  | ||||||
| 		this.query = this.opts.query; |  | ||||||
| 		this.isLoading = true; |  | ||||||
| 		this.isEmpty = false; |  | ||||||
| 		this.moreLoading = false; |  | ||||||
| 		this.limit = 30; |  | ||||||
| 		this.offset = 0; |  | ||||||
|  |  | ||||||
| 		this.on('mount', () => { |  | ||||||
| 			document.addEventListener('keydown', this.onDocumentKeydown); |  | ||||||
| 			window.addEventListener('scroll', this.onScroll); |  | ||||||
|  |  | ||||||
| 			this.$root.$data.os.api('posts/search', parse(this.query)).then(posts => { |  | ||||||
| 				this.update({ |  | ||||||
| 					isLoading: false, |  | ||||||
| 					isEmpty: posts.length == 0 |  | ||||||
| 				}); |  | ||||||
| 				this.$refs.timeline.setPosts(posts); |  | ||||||
| 				this.$emit('loaded'); |  | ||||||
| 			}); |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		this.on('unmount', () => { |  | ||||||
| 			document.removeEventListener('keydown', this.onDocumentKeydown); |  | ||||||
| 			window.removeEventListener('scroll', this.onScroll); |  | ||||||
| 		}); |  | ||||||
|  |  | ||||||
| 		this.onDocumentKeydown = e => { |  | ||||||
| 			if (e.target.tagName != 'INPUT' && e.target.tagName != 'TEXTAREA') { |  | ||||||
| 				if (e.which == 84) { // t |  | ||||||
| 					this.$refs.timeline.focus(); |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		}; |  | ||||||
|  |  | ||||||
| 		this.more = () => { |  | ||||||
| 			if (this.moreLoading || this.isLoading || this.timeline.posts.length == 0) return; |  | ||||||
| 			this.offset += this.limit; |  | ||||||
| 			this.update({ |  | ||||||
| 				moreLoading: true |  | ||||||
| 			}); |  | ||||||
| 			return this.$root.$data.os.api('posts/search', Object.assign({}, parse(this.query), { |  | ||||||
| 				limit: this.limit, |  | ||||||
| 				offset: this.offset |  | ||||||
| 			})).then(posts => { |  | ||||||
| 				this.update({ |  | ||||||
| 					moreLoading: false |  | ||||||
| 				}); |  | ||||||
| 				this.$refs.timeline.prependPosts(posts); |  | ||||||
| 			}); |  | ||||||
| 		}; |  | ||||||
|  |  | ||||||
| 		this.onScroll = () => { |  | ||||||
| 			const current = window.scrollY + window.innerHeight; |  | ||||||
| 			if (current > document.body.offsetHeight - 16) this.more(); |  | ||||||
| 		}; |  | ||||||
| 	</script> |  | ||||||
| </mk-search-posts> |  | ||||||
| @@ -1,34 +0,0 @@ | |||||||
| <mk-search> |  | ||||||
| 	<header> |  | ||||||
| 		<h1>{ query }</h1> |  | ||||||
| 	</header> |  | ||||||
| 	<mk-search-posts ref="posts" query={ query }/> |  | ||||||
| 	<style lang="stylus" scoped> |  | ||||||
| 		:scope |  | ||||||
| 			display block |  | ||||||
| 			padding-bottom 32px |  | ||||||
|  |  | ||||||
| 			> header |  | ||||||
| 				width 100% |  | ||||||
| 				max-width 600px |  | ||||||
| 				margin 0 auto |  | ||||||
| 				color #555 |  | ||||||
|  |  | ||||||
| 			> mk-search-posts |  | ||||||
| 				max-width 600px |  | ||||||
| 				margin 0 auto |  | ||||||
| 				border solid 1px rgba(0, 0, 0, 0.075) |  | ||||||
| 				border-radius 6px |  | ||||||
| 				overflow hidden |  | ||||||
|  |  | ||||||
| 	</style> |  | ||||||
| 	<script lang="typescript"> |  | ||||||
| 		this.query = this.opts.query; |  | ||||||
|  |  | ||||||
| 		this.on('mount', () => { |  | ||||||
| 			this.$refs.posts.on('loaded', () => { |  | ||||||
| 				this.$emit('loaded'); |  | ||||||
| 			}); |  | ||||||
| 		}); |  | ||||||
| 	</script> |  | ||||||
| </mk-search> |  | ||||||
							
								
								
									
										115
									
								
								src/web/app/desktop/views/pages/search.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								src/web/app/desktop/views/pages/search.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,115 @@ | |||||||
|  | <template> | ||||||
|  | <mk-ui> | ||||||
|  | 	<header :class="$style.header"> | ||||||
|  | 		<h1>{{ query }}</h1> | ||||||
|  | 	</header> | ||||||
|  | 	<div :class="$style.loading" v-if="fetching"> | ||||||
|  | 		<mk-ellipsis-icon/> | ||||||
|  | 	</div> | ||||||
|  | 	<p :class="$style.empty" v-if="empty">%fa:search%「{{ query }}」に関する投稿は見つかりませんでした。</p> | ||||||
|  | 	<mk-posts ref="timeline" :class="$style.posts"> | ||||||
|  | 		<div slot="footer"> | ||||||
|  | 			<template v-if="!moreFetching">%fa:search%</template> | ||||||
|  | 			<template v-if="moreFetching">%fa:spinner .pulse .fw%</template> | ||||||
|  | 		</div> | ||||||
|  | 	</mk-posts> | ||||||
|  | </mk-ui> | ||||||
|  | </template> | ||||||
|  |  | ||||||
|  | <script lang="ts"> | ||||||
|  | import Vue from 'vue'; | ||||||
|  | import Progress from '../../../common/scripts/loading'; | ||||||
|  | import parse from '../../../common/scripts/parse-search-query'; | ||||||
|  |  | ||||||
|  | const limit = 30; | ||||||
|  |  | ||||||
|  | export default Vue.extend({ | ||||||
|  | 	props: ['query'], | ||||||
|  | 	data() { | ||||||
|  | 		return { | ||||||
|  | 			fetching: true, | ||||||
|  | 			moreFetching: false, | ||||||
|  | 			offset: 0, | ||||||
|  | 			posts: [] | ||||||
|  | 		}; | ||||||
|  | 	}, | ||||||
|  | 	computed: { | ||||||
|  | 		empty(): boolean { | ||||||
|  | 			return this.posts.length == 0; | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
|  | 	mounted() { | ||||||
|  | 		Progress.start(); | ||||||
|  |  | ||||||
|  | 		document.addEventListener('keydown', this.onDocumentKeydown); | ||||||
|  | 		window.addEventListener('scroll', this.onScroll); | ||||||
|  |  | ||||||
|  | 		this.$root.$data.os.api('posts/search', parse(this.query)).then(posts => { | ||||||
|  | 			this.fetching = false; | ||||||
|  | 			this.posts = posts; | ||||||
|  | 		}); | ||||||
|  | 	}, | ||||||
|  | 	beforeDestroy() { | ||||||
|  | 		document.removeEventListener('keydown', this.onDocumentKeydown); | ||||||
|  | 		window.removeEventListener('scroll', this.onScroll); | ||||||
|  | 	}, | ||||||
|  | 	methods: { | ||||||
|  | 		onDocumentKeydown(e) { | ||||||
|  | 			if (e.target.tagName != 'INPUT' && e.target.tagName != 'TEXTAREA') { | ||||||
|  | 				if (e.which == 84) { // t | ||||||
|  | 					(this.$refs.timeline as any).focus(); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  | 		more() { | ||||||
|  | 			if (this.moreFetching || this.fetching || this.posts.length == 0) return; | ||||||
|  | 			this.offset += limit; | ||||||
|  | 			this.moreFetching = true; | ||||||
|  | 			return this.$root.$data.os.api('posts/search', Object.assign({}, parse(this.query), { | ||||||
|  | 				limit: limit, | ||||||
|  | 				offset: this.offset | ||||||
|  | 			})).then(posts => { | ||||||
|  | 				this.moreFetching = false; | ||||||
|  | 				this.posts = this.posts.concat(posts); | ||||||
|  | 			}); | ||||||
|  | 		}, | ||||||
|  | 		onScroll() { | ||||||
|  | 			const current = window.scrollY + window.innerHeight; | ||||||
|  | 			if (current > document.body.offsetHeight - 16) this.more(); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style lang="stylus" module> | ||||||
|  | .header | ||||||
|  | 	width 100% | ||||||
|  | 	max-width 600px | ||||||
|  | 	margin 0 auto | ||||||
|  | 	color #555 | ||||||
|  |  | ||||||
|  | .posts | ||||||
|  | 	max-width 600px | ||||||
|  | 	margin 0 auto | ||||||
|  | 	border solid 1px rgba(0, 0, 0, 0.075) | ||||||
|  | 	border-radius 6px | ||||||
|  | 	overflow hidden | ||||||
|  |  | ||||||
|  | .loading | ||||||
|  | 	padding 64px 0 | ||||||
|  |  | ||||||
|  | .empty | ||||||
|  | 	display block | ||||||
|  | 	margin 0 auto | ||||||
|  | 	padding 32px | ||||||
|  | 	max-width 400px | ||||||
|  | 	text-align center | ||||||
|  | 	color #999 | ||||||
|  |  | ||||||
|  | 	> [data-fa] | ||||||
|  | 		display block | ||||||
|  | 		margin-bottom 16px | ||||||
|  | 		font-size 3em | ||||||
|  | 		color #ccc | ||||||
|  |  | ||||||
|  | </style> | ||||||
		Reference in New Issue
	
	Block a user
	 こぴなたみぽ
					こぴなたみぽ