wip
This commit is contained in:
		| @@ -1,27 +0,0 @@ | ||||
| <mk-user-page> | ||||
| 	<mk-ui ref="ui"> | ||||
| 		<mk-user ref="user" user={ parent.user } page={ parent.opts.page }/> | ||||
| 	</mk-ui> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		import Progress from '../../../common/scripts/loading'; | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			Progress.start(); | ||||
|  | ||||
| 			this.$refs.ui.refs.user.on('user-fetched', user => { | ||||
| 				Progress.set(0.5); | ||||
| 				document.title = user.name + ' | Misskey'; | ||||
| 			}); | ||||
|  | ||||
| 			this.$refs.ui.refs.user.on('loaded', () => { | ||||
| 				Progress.done(); | ||||
| 			}); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user-page> | ||||
| @@ -1,852 +0,0 @@ | ||||
| <mk-user> | ||||
| 	<div class="user" v-if="!fetching"> | ||||
| 		<header> | ||||
| 			<mk-user-header user={ user }/> | ||||
| 		</header> | ||||
| 		<mk-user-home v-if="page == 'home'" user={ user }/> | ||||
| 		<mk-user-graphs v-if="page == 'graphs'" user={ user }/> | ||||
| 	</div> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
|  | ||||
| 			> .user | ||||
| 				> header | ||||
| 					> mk-user-header | ||||
| 						overflow hidden | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		this.mixin('api'); | ||||
|  | ||||
| 		this.username = this.opts.user; | ||||
| 		this.page = this.opts.page ? this.opts.page : 'home'; | ||||
| 		this.fetching = true; | ||||
| 		this.user = null; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			this.api('users/show', { | ||||
| 				username: this.username | ||||
| 			}).then(user => { | ||||
| 				this.update({ | ||||
| 					fetching: false, | ||||
| 					user: user | ||||
| 				}); | ||||
| 				this.$emit('loaded'); | ||||
| 			}); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user> | ||||
|  | ||||
| <mk-user-header data-is-dark-background={ user.banner_url != null }> | ||||
| 	<div class="banner-container" style={ user.banner_url ? 'background-image: url(' + user.banner_url + '?thumbnail&size=2048)' : '' }> | ||||
| 		<div class="banner" ref="banner" style={ user.banner_url ? 'background-image: url(' + user.banner_url + '?thumbnail&size=2048)' : '' } @click="onUpdateBanner"></div> | ||||
| 	</div> | ||||
| 	<div class="fade"></div> | ||||
| 	<div class="container"> | ||||
| 		<img class="avatar" src={ user.avatar_url + '?thumbnail&size=150' } alt="avatar"/> | ||||
| 		<div class="title"> | ||||
| 			<p class="name" href={ '/' + user.username }>{ user.name }</p> | ||||
| 			<p class="username">@{ user.username }</p> | ||||
| 			<p class="location" v-if="user.profile.location">%fa:map-marker%{ user.profile.location }</p> | ||||
| 		</div> | ||||
| 		<footer> | ||||
| 			<a href={ '/' + user.username } data-active={ parent.page == 'home' }>%fa:home%概要</a> | ||||
| 			<a href={ '/' + user.username + '/media' } data-active={ parent.page == 'media' }>%fa:image%メディア</a> | ||||
| 			<a href={ '/' + user.username + '/graphs' } data-active={ parent.page == 'graphs' }>%fa:chart-bar%グラフ</a> | ||||
| 		</footer> | ||||
| 	</div> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			$banner-height = 320px | ||||
| 			$footer-height = 58px | ||||
|  | ||||
| 			display block | ||||
| 			background #f7f7f7 | ||||
| 			box-shadow 0 1px 1px rgba(0, 0, 0, 0.075) | ||||
|  | ||||
| 			&[data-is-dark-background] | ||||
| 				> .banner-container | ||||
| 					> .banner | ||||
| 						background-color #383838 | ||||
|  | ||||
| 				> .fade | ||||
| 					background linear-gradient(transparent, rgba(0, 0, 0, 0.7)) | ||||
|  | ||||
| 				> .container | ||||
| 					> .title | ||||
| 						color #fff | ||||
|  | ||||
| 						> .name | ||||
| 							text-shadow 0 0 8px #000 | ||||
|  | ||||
| 			> .banner-container | ||||
| 				height $banner-height | ||||
| 				overflow hidden | ||||
| 				background-size cover | ||||
| 				background-position center | ||||
|  | ||||
| 				> .banner | ||||
| 					height 100% | ||||
| 					background-color #f5f5f5 | ||||
| 					background-size cover | ||||
| 					background-position center | ||||
|  | ||||
| 			> .fade | ||||
| 				$fade-hight = 78px | ||||
|  | ||||
| 				position absolute | ||||
| 				top ($banner-height - $fade-hight) | ||||
| 				left 0 | ||||
| 				width 100% | ||||
| 				height $fade-hight | ||||
|  | ||||
| 			> .container | ||||
| 				max-width 1200px | ||||
| 				margin 0 auto | ||||
|  | ||||
| 				> .avatar | ||||
| 					display block | ||||
| 					position absolute | ||||
| 					bottom 16px | ||||
| 					left 16px | ||||
| 					z-index 2 | ||||
| 					width 160px | ||||
| 					height 160px | ||||
| 					margin 0 | ||||
| 					border solid 3px #fff | ||||
| 					border-radius 8px | ||||
| 					box-shadow 1px 1px 3px rgba(0, 0, 0, 0.2) | ||||
|  | ||||
| 				> .title | ||||
| 					position absolute | ||||
| 					bottom $footer-height | ||||
| 					left 0 | ||||
| 					width 100% | ||||
| 					padding 0 0 8px 195px | ||||
| 					color #656565 | ||||
| 					font-family '游ゴシック', 'YuGothic', 'ヒラギノ角ゴ ProN W3', 'Hiragino Kaku Gothic ProN', 'Meiryo', 'メイリオ', sans-serif | ||||
|  | ||||
| 					> .name | ||||
| 						display block | ||||
| 						margin 0 | ||||
| 						line-height 40px | ||||
| 						font-weight bold | ||||
| 						font-size 2em | ||||
|  | ||||
| 					> .username | ||||
| 					> .location | ||||
| 						display inline-block | ||||
| 						margin 0 16px 0 0 | ||||
| 						line-height 20px | ||||
| 						opacity 0.8 | ||||
|  | ||||
| 						> i | ||||
| 							margin-right 4px | ||||
|  | ||||
| 				> footer | ||||
| 					z-index 1 | ||||
| 					height $footer-height | ||||
| 					padding-left 195px | ||||
|  | ||||
| 					> a | ||||
| 						display inline-block | ||||
| 						margin 0 | ||||
| 						padding 0 16px | ||||
| 						height $footer-height | ||||
| 						line-height $footer-height | ||||
| 						color #555 | ||||
|  | ||||
| 						&[data-active] | ||||
| 							border-bottom solid 4px $theme-color | ||||
|  | ||||
| 						> i | ||||
| 							margin-right 6px | ||||
|  | ||||
| 					> button | ||||
| 						display block | ||||
| 						position absolute | ||||
| 						top 0 | ||||
| 						right 0 | ||||
| 						margin 8px | ||||
| 						padding 0 | ||||
| 						width $footer-height - 16px | ||||
| 						line-height $footer-height - 16px - 2px | ||||
| 						font-size 1.2em | ||||
| 						color #777 | ||||
| 						border solid 1px #eee | ||||
| 						border-radius 4px | ||||
|  | ||||
| 						&:hover | ||||
| 							color #555 | ||||
| 							border solid 1px #ddd | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		import updateBanner from '../scripts/update-banner'; | ||||
|  | ||||
| 		this.mixin('i'); | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			window.addEventListener('load', this.scroll); | ||||
| 			window.addEventListener('scroll', this.scroll); | ||||
| 			window.addEventListener('resize', this.scroll); | ||||
| 		}); | ||||
|  | ||||
| 		this.on('unmount', () => { | ||||
| 			window.removeEventListener('load', this.scroll); | ||||
| 			window.removeEventListener('scroll', this.scroll); | ||||
| 			window.removeEventListener('resize', this.scroll); | ||||
| 		}); | ||||
|  | ||||
| 		this.scroll = () => { | ||||
| 			const top = window.scrollY; | ||||
|  | ||||
| 			const z = 1.25; // 奥行き(小さいほど奥) | ||||
| 			const pos = -(top / z); | ||||
| 			this.$refs.banner.style.backgroundPosition = `center calc(50% - ${pos}px)`; | ||||
|  | ||||
| 			const blur = top / 32 | ||||
| 			if (blur <= 10) this.$refs.banner.style.filter = `blur(${blur}px)`; | ||||
| 		}; | ||||
|  | ||||
| 		this.onUpdateBanner = () => { | ||||
| 			if (!this.SIGNIN || this.I.id != this.user.id) return; | ||||
|  | ||||
| 			updateBanner(this.I, i => { | ||||
| 				this.user.banner_url = i.banner_url; | ||||
| 				this.update(); | ||||
| 			}); | ||||
| 		}; | ||||
| 	</script> | ||||
| </mk-user-header> | ||||
|  | ||||
| <mk-user-profile> | ||||
| 	<div class="friend-form" v-if="SIGNIN && I.id != user.id"> | ||||
| 		<mk-big-follow-button user={ user }/> | ||||
| 		<p class="followed" v-if="user.is_followed">%i18n:desktop.tags.mk-user.follows-you%</p> | ||||
| 		<p v-if="user.is_muted">%i18n:desktop.tags.mk-user.muted% <a @click="unmute">%i18n:desktop.tags.mk-user.unmute%</a></p> | ||||
| 		<p v-if="!user.is_muted"><a @click="mute">%i18n:desktop.tags.mk-user.mute%</a></p> | ||||
| 	</div> | ||||
| 	<div class="description" v-if="user.description">{ user.description }</div> | ||||
| 	<div class="birthday" v-if="user.profile.birthday"> | ||||
| 		<p>%fa:birthday-cake%{ user.profile.birthday.replace('-', '年').replace('-', '月') + '日' } ({ age(user.profile.birthday) }歳)</p> | ||||
| 	</div> | ||||
| 	<div class="twitter" v-if="user.twitter"> | ||||
| 		<p>%fa:B twitter%<a href={ 'https://twitter.com/' + user.twitter.screen_name } target="_blank">@{ user.twitter.screen_name }</a></p> | ||||
| 	</div> | ||||
| 	<div class="status"> | ||||
| 	  <p class="posts-count">%fa:angle-right%<a>{ user.posts_count }</a><b>ポスト</b></p> | ||||
| 		<p class="following">%fa:angle-right%<a @click="showFollowing">{ user.following_count }</a>人を<b>フォロー</b></p> | ||||
| 		<p class="followers">%fa:angle-right%<a @click="showFollowers">{ user.followers_count }</a>人の<b>フォロワー</b></p> | ||||
| 	</div> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
| 			background #fff | ||||
| 			border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 			border-radius 6px | ||||
|  | ||||
| 			> *:first-child | ||||
| 				border-top none !important | ||||
|  | ||||
| 			> .friend-form | ||||
| 				padding 16px | ||||
| 				border-top solid 1px #eee | ||||
|  | ||||
| 				> mk-big-follow-button | ||||
| 					width 100% | ||||
|  | ||||
| 				> .followed | ||||
| 					margin 12px 0 0 0 | ||||
| 					padding 0 | ||||
| 					text-align center | ||||
| 					line-height 24px | ||||
| 					font-size 0.8em | ||||
| 					color #71afc7 | ||||
| 					background #eefaff | ||||
| 					border-radius 4px | ||||
|  | ||||
| 			> .description | ||||
| 				padding 16px | ||||
| 				color #555 | ||||
| 				border-top solid 1px #eee | ||||
|  | ||||
| 			> .birthday | ||||
| 				padding 16px | ||||
| 				color #555 | ||||
| 				border-top solid 1px #eee | ||||
|  | ||||
| 				> p | ||||
| 					margin 0 | ||||
|  | ||||
| 					> i | ||||
| 						margin-right 8px | ||||
|  | ||||
| 			> .twitter | ||||
| 				padding 16px | ||||
| 				color #555 | ||||
| 				border-top solid 1px #eee | ||||
|  | ||||
| 				> p | ||||
| 					margin 0 | ||||
|  | ||||
| 					> i | ||||
| 						margin-right 8px | ||||
|  | ||||
| 			> .status | ||||
| 				padding 16px | ||||
| 				color #555 | ||||
| 				border-top solid 1px #eee | ||||
|  | ||||
| 				> p | ||||
| 					margin 8px 0 | ||||
|  | ||||
| 					> i | ||||
| 						margin-left 8px | ||||
| 						margin-right 8px | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		this.age = require('s-age'); | ||||
|  | ||||
| 		this.mixin('i'); | ||||
| 		this.mixin('api'); | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
|  | ||||
| 		this.showFollowing = () => { | ||||
| 			riot.mount(document.body.appendChild(document.createElement('mk-user-following-window')), { | ||||
| 				user: this.user | ||||
| 			}); | ||||
| 		}; | ||||
|  | ||||
| 		this.showFollowers = () => { | ||||
| 			riot.mount(document.body.appendChild(document.createElement('mk-user-followers-window')), { | ||||
| 				user: this.user | ||||
| 			}); | ||||
| 		}; | ||||
|  | ||||
| 		this.mute = () => { | ||||
| 			this.api('mute/create', { | ||||
| 				user_id: this.user.id | ||||
| 			}).then(() => { | ||||
| 				this.user.is_muted = true; | ||||
| 				this.update(); | ||||
| 			}, e => { | ||||
| 				alert('error'); | ||||
| 			}); | ||||
| 		}; | ||||
|  | ||||
| 		this.unmute = () => { | ||||
| 			this.api('mute/delete', { | ||||
| 				user_id: this.user.id | ||||
| 			}).then(() => { | ||||
| 				this.user.is_muted = false; | ||||
| 				this.update(); | ||||
| 			}, e => { | ||||
| 				alert('error'); | ||||
| 			}); | ||||
| 		}; | ||||
| 	</script> | ||||
| </mk-user-profile> | ||||
|  | ||||
| <mk-user-photos> | ||||
| 	<p class="title">%fa:camera%%i18n:desktop.tags.mk-user.photos.title%</p> | ||||
| 	<p class="initializing" v-if="initializing">%fa:spinner .pulse .fw%%i18n:desktop.tags.mk-user.photos.loading%<mk-ellipsis/></p> | ||||
| 	<div class="stream" v-if="!initializing && images.length > 0"> | ||||
| 		<template each={ image in images }> | ||||
| 			<div class="img" style={ 'background-image: url(' + image.url + '?thumbnail&size=256)' }></div> | ||||
| 		</template> | ||||
| 	</div> | ||||
| 	<p class="empty" v-if="!initializing && images.length == 0">%i18n:desktop.tags.mk-user.photos.no-photos%</p> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
| 			background #fff | ||||
| 			border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 			border-radius 6px | ||||
|  | ||||
| 			> .title | ||||
| 				z-index 1 | ||||
| 				margin 0 | ||||
| 				padding 0 16px | ||||
| 				line-height 42px | ||||
| 				font-size 0.9em | ||||
| 				font-weight bold | ||||
| 				color #888 | ||||
| 				box-shadow 0 1px rgba(0, 0, 0, 0.07) | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 			> .stream | ||||
| 				display -webkit-flex | ||||
| 				display -moz-flex | ||||
| 				display -ms-flex | ||||
| 				display flex | ||||
| 				justify-content center | ||||
| 				flex-wrap wrap | ||||
| 				padding 8px | ||||
|  | ||||
| 				> .img | ||||
| 					flex 1 1 33% | ||||
| 					width 33% | ||||
| 					height 80px | ||||
| 					background-position center center | ||||
| 					background-size cover | ||||
| 					background-clip content-box | ||||
| 					border solid 2px transparent | ||||
|  | ||||
| 			> .initializing | ||||
| 			> .empty | ||||
| 				margin 0 | ||||
| 				padding 16px | ||||
| 				text-align center | ||||
| 				color #aaa | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		import isPromise from '../../common/scripts/is-promise'; | ||||
|  | ||||
| 		this.mixin('api'); | ||||
|  | ||||
| 		this.images = []; | ||||
| 		this.initializing = true; | ||||
| 		this.user = null; | ||||
| 		this.userPromise = isPromise(this.opts.user) | ||||
| 			? this.opts.user | ||||
| 			: Promise.resolve(this.opts.user); | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			this.userPromise.then(user => { | ||||
| 				this.update({ | ||||
| 					user: user | ||||
| 				}); | ||||
|  | ||||
| 				this.api('users/posts', { | ||||
| 					user_id: this.user.id, | ||||
| 					with_media: true, | ||||
| 					limit: 9 | ||||
| 				}).then(posts => { | ||||
| 					this.initializing = false; | ||||
| 					posts.forEach(post => { | ||||
| 						post.media.forEach(media => { | ||||
| 							if (this.images.length < 9) this.images.push(media); | ||||
| 						}); | ||||
| 					}); | ||||
| 					this.update(); | ||||
| 				}); | ||||
| 			}); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user-photos> | ||||
|  | ||||
| <mk-user-frequently-replied-users> | ||||
| 	<p class="title">%fa:users%%i18n:desktop.tags.mk-user.frequently-replied-users.title%</p> | ||||
| 	<p class="initializing" v-if="initializing">%fa:spinner .pulse .fw%%i18n:desktop.tags.mk-user.frequently-replied-users.loading%<mk-ellipsis/></p> | ||||
| 	<div class="user" v-if="!initializing && users.length != 0" each={ _user in users }> | ||||
| 		<a class="avatar-anchor" href={ '/' + _user.username }> | ||||
| 			<img class="avatar" src={ _user.avatar_url + '?thumbnail&size=42' } alt="" data-user-preview={ _user.id }/> | ||||
| 		</a> | ||||
| 		<div class="body"> | ||||
| 			<a class="name" href={ '/' + _user.username } data-user-preview={ _user.id }>{ _user.name }</a> | ||||
| 			<p class="username">@{ _user.username }</p> | ||||
| 		</div> | ||||
| 		<mk-follow-button user={ _user }/> | ||||
| 	</div> | ||||
| 	<p class="empty" v-if="!initializing && users.length == 0">%i18n:desktop.tags.mk-user.frequently-replied-users.no-users%</p> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
| 			background #fff | ||||
| 			border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 			border-radius 6px | ||||
|  | ||||
| 			> .title | ||||
| 				z-index 1 | ||||
| 				margin 0 | ||||
| 				padding 0 16px | ||||
| 				line-height 42px | ||||
| 				font-size 0.9em | ||||
| 				font-weight bold | ||||
| 				color #888 | ||||
| 				box-shadow 0 1px rgba(0, 0, 0, 0.07) | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 			> .initializing | ||||
| 			> .empty | ||||
| 				margin 0 | ||||
| 				padding 16px | ||||
| 				text-align center | ||||
| 				color #aaa | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 			> .user | ||||
| 				padding 16px | ||||
| 				border-bottom solid 1px #eee | ||||
|  | ||||
| 				&:last-child | ||||
| 					border-bottom none | ||||
|  | ||||
| 				&:after | ||||
| 					content "" | ||||
| 					display block | ||||
| 					clear both | ||||
|  | ||||
| 				> .avatar-anchor | ||||
| 					display block | ||||
| 					float left | ||||
| 					margin 0 12px 0 0 | ||||
|  | ||||
| 					> .avatar | ||||
| 						display block | ||||
| 						width 42px | ||||
| 						height 42px | ||||
| 						margin 0 | ||||
| 						border-radius 8px | ||||
| 						vertical-align bottom | ||||
|  | ||||
| 				> .body | ||||
| 					float left | ||||
| 					width calc(100% - 54px) | ||||
|  | ||||
| 					> .name | ||||
| 						margin 0 | ||||
| 						font-size 16px | ||||
| 						line-height 24px | ||||
| 						color #555 | ||||
|  | ||||
| 					> .username | ||||
| 						display block | ||||
| 						margin 0 | ||||
| 						font-size 15px | ||||
| 						line-height 16px | ||||
| 						color #ccc | ||||
|  | ||||
| 				> mk-follow-button | ||||
| 					position absolute | ||||
| 					top 16px | ||||
| 					right 16px | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		this.mixin('api'); | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
| 		this.initializing = true; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			this.api('users/get_frequently_replied_users', { | ||||
| 				user_id: this.user.id, | ||||
| 				limit: 4 | ||||
| 			}).then(docs => { | ||||
| 				this.update({ | ||||
| 					users: docs.map(doc => doc.user), | ||||
| 					initializing: false | ||||
| 				}); | ||||
| 			}); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user-frequently-replied-users> | ||||
|  | ||||
| <mk-user-followers-you-know> | ||||
| 	<p class="title">%fa:users%%i18n:desktop.tags.mk-user.followers-you-know.title%</p> | ||||
| 	<p class="initializing" v-if="initializing">%fa:spinner .pulse .fw%%i18n:desktop.tags.mk-user.followers-you-know.loading%<mk-ellipsis/></p> | ||||
| 	<div v-if="!initializing && users.length > 0"> | ||||
| 	<template each={ user in users }> | ||||
| 		<a href={ '/' + user.username }><img src={ user.avatar_url + '?thumbnail&size=64' } alt={ user.name }/></a> | ||||
| 	</template> | ||||
| 	</div> | ||||
| 	<p class="empty" v-if="!initializing && users.length == 0">%i18n:desktop.tags.mk-user.followers-you-know.no-users%</p> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
| 			background #fff | ||||
| 			border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 			border-radius 6px | ||||
|  | ||||
| 			> .title | ||||
| 				z-index 1 | ||||
| 				margin 0 | ||||
| 				padding 0 16px | ||||
| 				line-height 42px | ||||
| 				font-size 0.9em | ||||
| 				font-weight bold | ||||
| 				color #888 | ||||
| 				box-shadow 0 1px rgba(0, 0, 0, 0.07) | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 			> div | ||||
| 				padding 8px | ||||
|  | ||||
| 				> a | ||||
| 					display inline-block | ||||
| 					margin 4px | ||||
|  | ||||
| 					> img | ||||
| 						width 48px | ||||
| 						height 48px | ||||
| 						vertical-align bottom | ||||
| 						border-radius 100% | ||||
|  | ||||
| 			> .initializing | ||||
| 			> .empty | ||||
| 				margin 0 | ||||
| 				padding 16px | ||||
| 				text-align center | ||||
| 				color #aaa | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		this.mixin('api'); | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
| 		this.initializing = true; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			this.api('users/followers', { | ||||
| 				user_id: this.user.id, | ||||
| 				iknow: true, | ||||
| 				limit: 16 | ||||
| 			}).then(x => { | ||||
| 				this.update({ | ||||
| 					users: x.users, | ||||
| 					initializing: false | ||||
| 				}); | ||||
| 			}); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user-followers-you-know> | ||||
|  | ||||
| <mk-user-home> | ||||
| 	<div> | ||||
| 		<div ref="left"> | ||||
| 			<mk-user-profile user={ user }/> | ||||
| 			<mk-user-photos user={ user }/> | ||||
| 			<mk-user-followers-you-know v-if="SIGNIN && I.id !== user.id" user={ user }/> | ||||
| 			<p>%i18n:desktop.tags.mk-user.last-used-at%: <b><mk-time time={ user.last_used_at }/></b></p> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<main> | ||||
| 		<mk-post-detail v-if="user.pinned_post" post={ user.pinned_post } compact={ true }/> | ||||
| 		<mk-user-timeline ref="tl" user={ user }/> | ||||
| 	</main> | ||||
| 	<div> | ||||
| 		<div ref="right"> | ||||
| 			<mk-calendar-widget warp={ warp } start={ new Date(user.created_at) }/> | ||||
| 			<mk-activity-widget user={ user }/> | ||||
| 			<mk-user-frequently-replied-users user={ user }/> | ||||
| 			<div class="nav"><mk-nav-links/></div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display flex | ||||
| 			justify-content center | ||||
| 			margin 0 auto | ||||
| 			max-width 1200px | ||||
|  | ||||
| 			> main | ||||
| 			> div > div | ||||
| 				> *:not(:last-child) | ||||
| 					margin-bottom 16px | ||||
|  | ||||
| 			> main | ||||
| 				padding 16px | ||||
| 				width calc(100% - 275px * 2) | ||||
|  | ||||
| 				> mk-user-timeline | ||||
| 					border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 					border-radius 6px | ||||
|  | ||||
| 			> div | ||||
| 				width 275px | ||||
| 				margin 0 | ||||
|  | ||||
| 				&:first-child > div | ||||
| 					padding 16px 0 16px 16px | ||||
|  | ||||
| 					> p | ||||
| 						display block | ||||
| 						margin 0 | ||||
| 						padding 0 12px | ||||
| 						text-align center | ||||
| 						font-size 0.8em | ||||
| 						color #aaa | ||||
|  | ||||
| 				&:last-child > div | ||||
| 					padding 16px 16px 16px 0 | ||||
|  | ||||
| 					> .nav | ||||
| 						padding 16px | ||||
| 						font-size 12px | ||||
| 						color #aaa | ||||
| 						background #fff | ||||
| 						border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 						border-radius 6px | ||||
|  | ||||
| 						a | ||||
| 							color #999 | ||||
|  | ||||
| 						i | ||||
| 							color #ccc | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		import ScrollFollower from '../scripts/scroll-follower'; | ||||
|  | ||||
| 		this.mixin('i'); | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			this.$refs.tl.on('loaded', () => { | ||||
| 				this.$emit('loaded'); | ||||
| 			}); | ||||
|  | ||||
| 			this.scrollFollowerLeft = new ScrollFollower(this.$refs.left, this.parent.root.getBoundingClientRect().top); | ||||
| 			this.scrollFollowerRight = new ScrollFollower(this.$refs.right, this.parent.root.getBoundingClientRect().top); | ||||
| 		}); | ||||
|  | ||||
| 		this.on('unmount', () => { | ||||
| 			this.scrollFollowerLeft.dispose(); | ||||
| 			this.scrollFollowerRight.dispose(); | ||||
| 		}); | ||||
|  | ||||
| 		this.warp = date => { | ||||
| 			this.$refs.tl.warp(date); | ||||
| 		}; | ||||
| 	</script> | ||||
| </mk-user-home> | ||||
|  | ||||
| <mk-user-graphs> | ||||
| 	<section> | ||||
| 		<div> | ||||
| 			<h1>%fa:pencil-alt%投稿</h1> | ||||
| 			<mk-user-graphs-activity-chart user={ opts.user }/> | ||||
| 		</div> | ||||
| 	</section> | ||||
| 	<section> | ||||
| 		<div> | ||||
| 			<h1>フォロー/フォロワー</h1> | ||||
| 			<mk-user-friends-graph user={ opts.user }/> | ||||
| 		</div> | ||||
| 	</section> | ||||
| 	<section> | ||||
| 		<div> | ||||
| 			<h1>いいね</h1> | ||||
| 			<mk-user-likes-graph user={ opts.user }/> | ||||
| 		</div> | ||||
| 	</section> | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
|  | ||||
| 			> section | ||||
| 				margin 16px 0 | ||||
| 				color #666 | ||||
| 				border-bottom solid 1px rgba(0, 0, 0, 0.1) | ||||
|  | ||||
| 				> div | ||||
| 					max-width 1200px | ||||
| 					margin 0 auto | ||||
| 					padding 0 16px | ||||
|  | ||||
| 					> h1 | ||||
| 						margin 0 0 16px 0 | ||||
| 						padding 0 | ||||
| 						font-size 1.3em | ||||
|  | ||||
| 						> i | ||||
| 							margin-right 8px | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		this.on('mount', () => { | ||||
| 			this.$emit('loaded'); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user-graphs> | ||||
|  | ||||
| <mk-user-graphs-activity-chart> | ||||
| 	<svg v-if="data" ref="canvas" viewBox="0 0 365 1" preserveAspectRatio="none"> | ||||
| 		<g each={ d, i in data.reverse() }> | ||||
| 			<rect width="0.8" riot-height={ d.postsH } | ||||
| 				riot-x={ i + 0.1 } riot-y={ 1 - d.postsH - d.repliesH - d.repostsH } | ||||
| 				fill="#41ddde"/> | ||||
| 			<rect width="0.8" riot-height={ d.repliesH } | ||||
| 				riot-x={ i + 0.1 } riot-y={ 1 - d.repliesH - d.repostsH } | ||||
| 				fill="#f7796c"/> | ||||
| 			<rect width="0.8" riot-height={ d.repostsH } | ||||
| 				riot-x={ i + 0.1 } riot-y={ 1 - d.repostsH } | ||||
| 				fill="#a1de41"/> | ||||
| 			</g> | ||||
| 	</svg> | ||||
| 	<p>直近1年間分の統計です。一番右が現在で、一番左が1年前です。青は通常の投稿、赤は返信、緑はRepostをそれぞれ表しています。</p> | ||||
| 	<p> | ||||
| 		<span>だいたい*1日に<b>{ averageOfAllTypePostsEachDays }回</b>投稿(返信、Repost含む)しています。</span><br> | ||||
| 		<span>だいたい*1日に<b>{ averageOfPostsEachDays }回</b>投稿(通常の)しています。</span><br> | ||||
| 		<span>だいたい*1日に<b>{ averageOfRepliesEachDays }回</b>返信しています。</span><br> | ||||
| 		<span>だいたい*1日に<b>{ averageOfRepostsEachDays }回</b>Repostしています。</span><br> | ||||
| 	</p> | ||||
| 	<p>* 中央値</p> | ||||
|  | ||||
| 	<style lang="stylus" scoped> | ||||
| 		:scope | ||||
| 			display block | ||||
|  | ||||
| 			> svg | ||||
| 				display block | ||||
| 				width 100% | ||||
| 				height 180px | ||||
|  | ||||
| 				> rect | ||||
| 					transform-origin center | ||||
|  | ||||
| 	</style> | ||||
| 	<script lang="typescript"> | ||||
| 		import getMedian from '../../common/scripts/get-median'; | ||||
|  | ||||
| 		this.mixin('api'); | ||||
|  | ||||
| 		this.user = this.opts.user; | ||||
|  | ||||
| 		this.on('mount', () => { | ||||
| 			this.api('aggregation/users/activity', { | ||||
| 				user_id: this.user.id, | ||||
| 				limit: 365 | ||||
| 			}).then(data => { | ||||
| 				data.forEach(d => d.total = d.posts + d.replies + d.reposts); | ||||
| 				this.peak = Math.max.apply(null, data.map(d => d.total)); | ||||
| 				data.forEach(d => { | ||||
| 					d.postsH = d.posts / this.peak; | ||||
| 					d.repliesH = d.replies / this.peak; | ||||
| 					d.repostsH = d.reposts / this.peak; | ||||
| 				}); | ||||
|  | ||||
| 				this.update({ | ||||
| 					data, | ||||
| 					averageOfAllTypePostsEachDays: getMedian(data.map(d => d.total)), | ||||
| 					averageOfPostsEachDays: getMedian(data.map(d => d.posts)), | ||||
| 					averageOfRepliesEachDays: getMedian(data.map(d => d.replies)), | ||||
| 					averageOfRepostsEachDays: getMedian(data.map(d => d.reposts)) | ||||
| 				}); | ||||
| 			}); | ||||
| 		}); | ||||
| 	</script> | ||||
| </mk-user-graphs-activity-chart> | ||||
| @@ -0,0 +1,79 @@ | ||||
| <template> | ||||
| <div class="mk-user-followers-you-know"> | ||||
| 	<p class="title">%fa:users%%i18n:desktop.tags.mk-user.followers-you-know.title%</p> | ||||
| 	<p class="initializing" v-if="initializing">%fa:spinner .pulse .fw%%i18n:desktop.tags.mk-user.followers-you-know.loading%<mk-ellipsis/></p> | ||||
| 	<div v-if="!initializing && users.length > 0"> | ||||
| 	<template each={ user in users }> | ||||
| 		<a href={ '/' + user.username }><img src={ user.avatar_url + '?thumbnail&size=64' } alt={ user.name }/></a> | ||||
| 	</template> | ||||
| 	</div> | ||||
| 	<p class="empty" v-if="!initializing && users.length == 0">%i18n:desktop.tags.mk-user.followers-you-know.no-users%</p> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| export default Vue.extend({ | ||||
| 	props: ['user'], | ||||
| 	data() { | ||||
| 		return { | ||||
| 			users: [], | ||||
| 			fetching: true | ||||
| 		}; | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.$root.$data.os.api('users/followers', { | ||||
| 			user_id: this.user.id, | ||||
| 			iknow: true, | ||||
| 			limit: 16 | ||||
| 		}).then(x => { | ||||
| 			this.fetching = false; | ||||
| 			this.users = x.users; | ||||
| 		}); | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mk-user-followers-you-know | ||||
| 	background #fff | ||||
| 	border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 	border-radius 6px | ||||
|  | ||||
| 	> .title | ||||
| 		z-index 1 | ||||
| 		margin 0 | ||||
| 		padding 0 16px | ||||
| 		line-height 42px | ||||
| 		font-size 0.9em | ||||
| 		font-weight bold | ||||
| 		color #888 | ||||
| 		box-shadow 0 1px rgba(0, 0, 0, 0.07) | ||||
|  | ||||
| 		> i | ||||
| 			margin-right 4px | ||||
|  | ||||
| 	> div | ||||
| 		padding 8px | ||||
|  | ||||
| 		> a | ||||
| 			display inline-block | ||||
| 			margin 4px | ||||
|  | ||||
| 			> img | ||||
| 				width 48px | ||||
| 				height 48px | ||||
| 				vertical-align bottom | ||||
| 				border-radius 100% | ||||
|  | ||||
| 	> .initializing | ||||
| 	> .empty | ||||
| 		margin 0 | ||||
| 		padding 16px | ||||
| 		text-align center | ||||
| 		color #aaa | ||||
|  | ||||
| 		> i | ||||
| 			margin-right 4px | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										117
									
								
								src/web/app/desktop/views/pages/user/user-friends.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								src/web/app/desktop/views/pages/user/user-friends.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | ||||
| <template> | ||||
| <div class="mk-user-friends"> | ||||
| 	<p class="title">%fa:users%%i18n:desktop.tags.mk-user.frequently-replied-users.title%</p> | ||||
| 	<p class="initializing" v-if="fetching">%fa:spinner .pulse .fw%%i18n:desktop.tags.mk-user.frequently-replied-users.loading%<mk-ellipsis/></p> | ||||
| 	<div class="user" v-if="!fetching && users.length != 0" each={ _user in users }> | ||||
| 		<a class="avatar-anchor" href={ '/' + _user.username }> | ||||
| 			<img class="avatar" src={ _user.avatar_url + '?thumbnail&size=42' } alt="" data-user-preview={ _user.id }/> | ||||
| 		</a> | ||||
| 		<div class="body"> | ||||
| 			<a class="name" href={ '/' + _user.username } data-user-preview={ _user.id }>{ _user.name }</a> | ||||
| 			<p class="username">@{ _user.username }</p> | ||||
| 		</div> | ||||
| 		<mk-follow-button user={ _user }/> | ||||
| 	</div> | ||||
| 	<p class="empty" v-if="!fetching && users.length == 0">%i18n:desktop.tags.mk-user.frequently-replied-users.no-users%</p> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| export default Vue.extend({ | ||||
| 	props: ['user'], | ||||
| 	data() { | ||||
| 		return { | ||||
| 			users: [], | ||||
| 			fetching: true | ||||
| 		}; | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.$root.$data.os.api('users/get_frequently_replied_users', { | ||||
| 			user_id: this.user.id, | ||||
| 			limit: 4 | ||||
| 		}).then(docs => { | ||||
| 			this.fetching = false; | ||||
| 			this.users = docs.map(doc => doc.user); | ||||
| 		}); | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mk-user-friends | ||||
| 	background #fff | ||||
| 	border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 	border-radius 6px | ||||
|  | ||||
| 	> .title | ||||
| 		z-index 1 | ||||
| 		margin 0 | ||||
| 		padding 0 16px | ||||
| 		line-height 42px | ||||
| 		font-size 0.9em | ||||
| 		font-weight bold | ||||
| 		color #888 | ||||
| 		box-shadow 0 1px rgba(0, 0, 0, 0.07) | ||||
|  | ||||
| 		> i | ||||
| 			margin-right 4px | ||||
|  | ||||
| 	> .initializing | ||||
| 	> .empty | ||||
| 		margin 0 | ||||
| 		padding 16px | ||||
| 		text-align center | ||||
| 		color #aaa | ||||
|  | ||||
| 		> i | ||||
| 			margin-right 4px | ||||
|  | ||||
| 	> .user | ||||
| 		padding 16px | ||||
| 		border-bottom solid 1px #eee | ||||
|  | ||||
| 		&:last-child | ||||
| 			border-bottom none | ||||
|  | ||||
| 		&:after | ||||
| 			content "" | ||||
| 			display block | ||||
| 			clear both | ||||
|  | ||||
| 		> .avatar-anchor | ||||
| 			display block | ||||
| 			float left | ||||
| 			margin 0 12px 0 0 | ||||
|  | ||||
| 			> .avatar | ||||
| 				display block | ||||
| 				width 42px | ||||
| 				height 42px | ||||
| 				margin 0 | ||||
| 				border-radius 8px | ||||
| 				vertical-align bottom | ||||
|  | ||||
| 		> .body | ||||
| 			float left | ||||
| 			width calc(100% - 54px) | ||||
|  | ||||
| 			> .name | ||||
| 				margin 0 | ||||
| 				font-size 16px | ||||
| 				line-height 24px | ||||
| 				color #555 | ||||
|  | ||||
| 			> .username | ||||
| 				display block | ||||
| 				margin 0 | ||||
| 				font-size 15px | ||||
| 				line-height 16px | ||||
| 				color #ccc | ||||
|  | ||||
| 		> mk-follow-button | ||||
| 			position absolute | ||||
| 			top 16px | ||||
| 			right 16px | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										189
									
								
								src/web/app/desktop/views/pages/user/user-header.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								src/web/app/desktop/views/pages/user/user-header.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,189 @@ | ||||
| <template> | ||||
| <div class="mk-user-header" :data-is-dark-background="user.banner_url != null"> | ||||
| 	<div class="banner-container" :style="user.banner_url ? `background-image: url(${user.banner_url}?thumbnail&size=2048)` : ''"> | ||||
| 		<div class="banner" ref="banner" :style="user.banner_url ? `background-image: url(${user.banner_url}?thumbnail&size=2048)` : ''" @click="onBannerClick"></div> | ||||
| 	</div> | ||||
| 	<div class="fade"></div> | ||||
| 	<div class="container"> | ||||
| 		<img class="avatar" :src="`${user.avatar_url}?thumbnail&size=150`" alt="avatar"/> | ||||
| 		<div class="title"> | ||||
| 			<p class="name">{{ user.name }}</p> | ||||
| 			<p class="username">@{{ user.username }}</p> | ||||
| 			<p class="location" v-if="user.profile.location">%fa:map-marker%{{ user.profile.location }}</p> | ||||
| 		</div> | ||||
| 		<footer> | ||||
| 			<a :href="`/${user.username}`" :data-active="$parent.page == 'home'">%fa:home%概要</a> | ||||
| 			<a :href="`/${user.username}/media`" :data-active="$parent.page == 'media'">%fa:image%メディア</a> | ||||
| 			<a :href="`/${user.username}/graphs`" :data-active="$parent.page == 'graphs'">%fa:chart-bar%グラフ</a> | ||||
| 		</footer> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import updateBanner from '../../../scripts/update-banner'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	props: ['user'], | ||||
| 	mounted() { | ||||
| 		window.addEventListener('load', this.onScroll); | ||||
| 		window.addEventListener('scroll', this.onScroll); | ||||
| 		window.addEventListener('resize', this.onScroll); | ||||
| 	}, | ||||
| 	beforeDestroy() { | ||||
| 		window.removeEventListener('load', this.onScroll); | ||||
| 		window.removeEventListener('scroll', this.onScroll); | ||||
| 		window.removeEventListener('resize', this.onScroll); | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		onScroll() { | ||||
| 			const banner = this.$refs.banner as any; | ||||
|  | ||||
| 			const top = window.scrollY; | ||||
|  | ||||
| 			const z = 1.25; // 奥行き(小さいほど奥) | ||||
| 			const pos = -(top / z); | ||||
| 			banner.style.backgroundPosition = `center calc(50% - ${pos}px)`; | ||||
|  | ||||
| 			const blur = top / 32 | ||||
| 			if (blur <= 10) banner.style.filter = `blur(${blur}px)`; | ||||
| 		}, | ||||
|  | ||||
| 		onBannerClick() { | ||||
| 			if (!this.$root.$data.os.isSignedIn || this.$root.$data.os.i.id != this.user.id) return; | ||||
|  | ||||
| 			updateBanner(this.$root.$data.os.i, i => { | ||||
| 				this.user.banner_url = i.banner_url; | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mk-user-header | ||||
| 	$banner-height = 320px | ||||
| 	$footer-height = 58px | ||||
|  | ||||
| 	overflow hidden | ||||
| 	background #f7f7f7 | ||||
| 	box-shadow 0 1px 1px rgba(0, 0, 0, 0.075) | ||||
|  | ||||
| 	&[data-is-dark-background] | ||||
| 		> .banner-container | ||||
| 			> .banner | ||||
| 				background-color #383838 | ||||
|  | ||||
| 		> .fade | ||||
| 			background linear-gradient(transparent, rgba(0, 0, 0, 0.7)) | ||||
|  | ||||
| 		> .container | ||||
| 			> .title | ||||
| 				color #fff | ||||
|  | ||||
| 				> .name | ||||
| 					text-shadow 0 0 8px #000 | ||||
|  | ||||
| 	> .banner-container | ||||
| 		height $banner-height | ||||
| 		overflow hidden | ||||
| 		background-size cover | ||||
| 		background-position center | ||||
|  | ||||
| 		> .banner | ||||
| 			height 100% | ||||
| 			background-color #f5f5f5 | ||||
| 			background-size cover | ||||
| 			background-position center | ||||
|  | ||||
| 	> .fade | ||||
| 		$fade-hight = 78px | ||||
|  | ||||
| 		position absolute | ||||
| 		top ($banner-height - $fade-hight) | ||||
| 		left 0 | ||||
| 		width 100% | ||||
| 		height $fade-hight | ||||
|  | ||||
| 	> .container | ||||
| 		max-width 1200px | ||||
| 		margin 0 auto | ||||
|  | ||||
| 		> .avatar | ||||
| 			display block | ||||
| 			position absolute | ||||
| 			bottom 16px | ||||
| 			left 16px | ||||
| 			z-index 2 | ||||
| 			width 160px | ||||
| 			height 160px | ||||
| 			margin 0 | ||||
| 			border solid 3px #fff | ||||
| 			border-radius 8px | ||||
| 			box-shadow 1px 1px 3px rgba(0, 0, 0, 0.2) | ||||
|  | ||||
| 		> .title | ||||
| 			position absolute | ||||
| 			bottom $footer-height | ||||
| 			left 0 | ||||
| 			width 100% | ||||
| 			padding 0 0 8px 195px | ||||
| 			color #656565 | ||||
| 			font-family '游ゴシック', 'YuGothic', 'ヒラギノ角ゴ ProN W3', 'Hiragino Kaku Gothic ProN', 'Meiryo', 'メイリオ', sans-serif | ||||
|  | ||||
| 			> .name | ||||
| 				display block | ||||
| 				margin 0 | ||||
| 				line-height 40px | ||||
| 				font-weight bold | ||||
| 				font-size 2em | ||||
|  | ||||
| 			> .username | ||||
| 			> .location | ||||
| 				display inline-block | ||||
| 				margin 0 16px 0 0 | ||||
| 				line-height 20px | ||||
| 				opacity 0.8 | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 4px | ||||
|  | ||||
| 		> footer | ||||
| 			z-index 1 | ||||
| 			height $footer-height | ||||
| 			padding-left 195px | ||||
|  | ||||
| 			> a | ||||
| 				display inline-block | ||||
| 				margin 0 | ||||
| 				padding 0 16px | ||||
| 				height $footer-height | ||||
| 				line-height $footer-height | ||||
| 				color #555 | ||||
|  | ||||
| 				&[data-active] | ||||
| 					border-bottom solid 4px $theme-color | ||||
|  | ||||
| 				> i | ||||
| 					margin-right 6px | ||||
|  | ||||
| 			> button | ||||
| 				display block | ||||
| 				position absolute | ||||
| 				top 0 | ||||
| 				right 0 | ||||
| 				margin 8px | ||||
| 				padding 0 | ||||
| 				width $footer-height - 16px | ||||
| 				line-height $footer-height - 16px - 2px | ||||
| 				font-size 1.2em | ||||
| 				color #777 | ||||
| 				border solid 1px #eee | ||||
| 				border-radius 4px | ||||
|  | ||||
| 				&:hover | ||||
| 					color #555 | ||||
| 					border solid 1px #ddd | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										90
									
								
								src/web/app/desktop/views/pages/user/user-home.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								src/web/app/desktop/views/pages/user/user-home.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | ||||
| <template> | ||||
| <div class="mk-user-home"> | ||||
| 	<div> | ||||
| 		<div ref="left"> | ||||
| 			<mk-user-profile :user="user"/> | ||||
| 			<mk-user-photos :user="user"/> | ||||
| 			<mk-user-followers-you-know v-if="$root.$data.os.isSignedIn && $root.$data.os.i.id != user.id" :user="user"/> | ||||
| 			<p>%i18n:desktop.tags.mk-user.last-used-at%: <b><mk-time :time="user.last_used_at"/></b></p> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<main> | ||||
| 		<mk-post-detail v-if="user.pinned_post" :post="user.pinned_post" compact/> | ||||
| 		<mk-user-timeline ref="tl" :user="user"/> | ||||
| 	</main> | ||||
| 	<div> | ||||
| 		<div ref="right"> | ||||
| 			<mk-calendar-widget @warp="warp" :start="new Date(user.created_at)"/> | ||||
| 			<mk-activity-widget :user="user"/> | ||||
| 			<mk-user-friends :user="user"/> | ||||
| 			<div class="nav"><mk-nav-links/></div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| export default Vue.extend({ | ||||
| 	props: ['user'], | ||||
| 	methods: { | ||||
| 		warp(date) { | ||||
| 			(this.$refs.tl as any).warp(date); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mk-user-home | ||||
| 	display flex | ||||
| 	justify-content center | ||||
| 	margin 0 auto | ||||
| 	max-width 1200px | ||||
|  | ||||
| 	> main | ||||
| 	> div > div | ||||
| 		> *:not(:last-child) | ||||
| 			margin-bottom 16px | ||||
|  | ||||
| 	> main | ||||
| 		padding 16px | ||||
| 		width calc(100% - 275px * 2) | ||||
|  | ||||
| 		> mk-user-timeline | ||||
| 			border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 			border-radius 6px | ||||
|  | ||||
| 	> div | ||||
| 		width 275px | ||||
| 		margin 0 | ||||
|  | ||||
| 		&:first-child > div | ||||
| 			padding 16px 0 16px 16px | ||||
|  | ||||
| 			> p | ||||
| 				display block | ||||
| 				margin 0 | ||||
| 				padding 0 12px | ||||
| 				text-align center | ||||
| 				font-size 0.8em | ||||
| 				color #aaa | ||||
|  | ||||
| 		&:last-child > div | ||||
| 			padding 16px 16px 16px 0 | ||||
|  | ||||
| 			> .nav | ||||
| 				padding 16px | ||||
| 				font-size 12px | ||||
| 				color #aaa | ||||
| 				background #fff | ||||
| 				border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 				border-radius 6px | ||||
|  | ||||
| 				a | ||||
| 					color #999 | ||||
|  | ||||
| 				i | ||||
| 					color #ccc | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										89
									
								
								src/web/app/desktop/views/pages/user/user-photos.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								src/web/app/desktop/views/pages/user/user-photos.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | ||||
| <template> | ||||
| <div class="mk-user-photos"> | ||||
| 	<p class="title">%fa:camera%%i18n:desktop.tags.mk-user.photos.title%</p> | ||||
| 	<p class="initializing" v-if="fetching">%fa:spinner .pulse .fw%%i18n:desktop.tags.mk-user.photos.loading%<mk-ellipsis/></p> | ||||
| 	<div class="stream" v-if="!fetching && images.length > 0"> | ||||
| 		<div v-for="image in images" :key="image.id" | ||||
| 			class="img" | ||||
| 			:style="`background-image: url(${image.url}?thumbnail&size=256)`" | ||||
| 		></div> | ||||
| 	</div> | ||||
| 	<p class="empty" v-if="!fetching && images.length == 0">%i18n:desktop.tags.mk-user.photos.no-photos%</p> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| export default Vue.extend({ | ||||
| 	props: ['user'], | ||||
| 	data() { | ||||
| 		return { | ||||
| 			images: [], | ||||
| 			fetching: true | ||||
| 		}; | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		this.$root.$data.os.api('users/posts', { | ||||
| 			user_id: this.user.id, | ||||
| 			with_media: true, | ||||
| 			limit: 9 | ||||
| 		}).then(posts => { | ||||
| 			this.fetching = false; | ||||
| 			posts.forEach(post => { | ||||
| 				post.media.forEach(media => { | ||||
| 					if (this.images.length < 9) this.images.push(media); | ||||
| 				}); | ||||
| 			}); | ||||
| 		}); | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mk-user-photos | ||||
| 	background #fff | ||||
| 	border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 	border-radius 6px | ||||
|  | ||||
| 	> .title | ||||
| 		z-index 1 | ||||
| 		margin 0 | ||||
| 		padding 0 16px | ||||
| 		line-height 42px | ||||
| 		font-size 0.9em | ||||
| 		font-weight bold | ||||
| 		color #888 | ||||
| 		box-shadow 0 1px rgba(0, 0, 0, 0.07) | ||||
|  | ||||
| 		> i | ||||
| 			margin-right 4px | ||||
|  | ||||
| 	> .stream | ||||
| 		display -webkit-flex | ||||
| 		display -moz-flex | ||||
| 		display -ms-flex | ||||
| 		display flex | ||||
| 		justify-content center | ||||
| 		flex-wrap wrap | ||||
| 		padding 8px | ||||
|  | ||||
| 		> .img | ||||
| 			flex 1 1 33% | ||||
| 			width 33% | ||||
| 			height 80px | ||||
| 			background-position center center | ||||
| 			background-size cover | ||||
| 			background-clip content-box | ||||
| 			border solid 2px transparent | ||||
|  | ||||
| 	> .initializing | ||||
| 	> .empty | ||||
| 		margin 0 | ||||
| 		padding 16px | ||||
| 		text-align center | ||||
| 		color #aaa | ||||
|  | ||||
| 		> i | ||||
| 			margin-right 4px | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										142
									
								
								src/web/app/desktop/views/pages/user/user-profile.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								src/web/app/desktop/views/pages/user/user-profile.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,142 @@ | ||||
| <template> | ||||
| <div class="mk-user-profile"> | ||||
| 	<div class="friend-form" v-if="$root.$data.os.isSignedIn && $root.$data.os.i.id != user.id"> | ||||
| 		<mk-follow-button :user="user" size="big"/> | ||||
| 		<p class="followed" v-if="user.is_followed">%i18n:desktop.tags.mk-user.follows-you%</p> | ||||
| 		<p v-if="user.is_muted">%i18n:desktop.tags.mk-user.muted% <a @click="unmute">%i18n:desktop.tags.mk-user.unmute%</a></p> | ||||
| 		<p v-if="!user.is_muted"><a @click="mute">%i18n:desktop.tags.mk-user.mute%</a></p> | ||||
| 	</div> | ||||
| 	<div class="description" v-if="user.description">{{ user.description }}</div> | ||||
| 	<div class="birthday" v-if="user.profile.birthday"> | ||||
| 		<p>%fa:birthday-cake%{{ user.profile.birthday.replace('-', '年').replace('-', '月') + '日' }} ({{ age }}歳)</p> | ||||
| 	</div> | ||||
| 	<div class="twitter" v-if="user.twitter"> | ||||
| 		<p>%fa:B twitter%<a :href="`https://twitter.com/${user.twitter.screen_name}`" target="_blank">@{{ user.twitter.screen_name }}</a></p> | ||||
| 	</div> | ||||
| 	<div class="status"> | ||||
| 	  <p class="posts-count">%fa:angle-right%<a>{{ user.posts_count }}</a><b>投稿</b></p> | ||||
| 		<p class="following">%fa:angle-right%<a @click="showFollowing">{{ user.following_count }}</a>人を<b>フォロー</b></p> | ||||
| 		<p class="followers">%fa:angle-right%<a @click="showFollowers">{{ user.followers_count }}</a>人の<b>フォロワー</b></p> | ||||
| 	</div> | ||||
| </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| const age = require('s-age'); | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	props: ['user'], | ||||
| 	computed: { | ||||
| 		age(): number { | ||||
| 			return age(this.user.profile.birthday); | ||||
| 		} | ||||
| 	}, | ||||
| 	methods: { | ||||
| 		showFollowing() { | ||||
| 			document.body.appendChild(new MkUserFollowingWindow({ | ||||
| 				parent: this, | ||||
| 				propsData: { | ||||
| 					user: this.user | ||||
| 				} | ||||
| 			}).$mount().$el); | ||||
| 		}, | ||||
|  | ||||
| 		showFollowers() { | ||||
| 			document.body.appendChild(new MkUserFollowersWindow({ | ||||
| 				parent: this, | ||||
| 				propsData: { | ||||
| 					user: this.user | ||||
| 				} | ||||
| 			}).$mount().$el); | ||||
| 		}, | ||||
|  | ||||
| 		mute() { | ||||
| 			this.$root.$data.os.api('mute/create', { | ||||
| 				user_id: this.user.id | ||||
| 			}).then(() => { | ||||
| 				this.user.is_muted = true; | ||||
| 			}, e => { | ||||
| 				alert('error'); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		unmute() { | ||||
| 			this.$root.$data.os.api('mute/delete', { | ||||
| 				user_id: this.user.id | ||||
| 			}).then(() => { | ||||
| 				this.user.is_muted = false; | ||||
| 			}, e => { | ||||
| 				alert('error'); | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .mk-user-profile | ||||
| 	background #fff | ||||
| 	border solid 1px rgba(0, 0, 0, 0.075) | ||||
| 	border-radius 6px | ||||
|  | ||||
| 	> *:first-child | ||||
| 		border-top none !important | ||||
|  | ||||
| 	> .friend-form | ||||
| 		padding 16px | ||||
| 		border-top solid 1px #eee | ||||
|  | ||||
| 		> mk-big-follow-button | ||||
| 			width 100% | ||||
|  | ||||
| 		> .followed | ||||
| 			margin 12px 0 0 0 | ||||
| 			padding 0 | ||||
| 			text-align center | ||||
| 			line-height 24px | ||||
| 			font-size 0.8em | ||||
| 			color #71afc7 | ||||
| 			background #eefaff | ||||
| 			border-radius 4px | ||||
|  | ||||
| 	> .description | ||||
| 		padding 16px | ||||
| 		color #555 | ||||
| 		border-top solid 1px #eee | ||||
|  | ||||
| 	> .birthday | ||||
| 		padding 16px | ||||
| 		color #555 | ||||
| 		border-top solid 1px #eee | ||||
|  | ||||
| 		> p | ||||
| 			margin 0 | ||||
|  | ||||
| 			> i | ||||
| 				margin-right 8px | ||||
|  | ||||
| 	> .twitter | ||||
| 		padding 16px | ||||
| 		color #555 | ||||
| 		border-top solid 1px #eee | ||||
|  | ||||
| 		> p | ||||
| 			margin 0 | ||||
|  | ||||
| 			> i | ||||
| 				margin-right 8px | ||||
|  | ||||
| 	> .status | ||||
| 		padding 16px | ||||
| 		color #555 | ||||
| 		border-top solid 1px #eee | ||||
|  | ||||
| 		> p | ||||
| 			margin 8px 0 | ||||
|  | ||||
| 			> i | ||||
| 				margin-left 8px | ||||
| 				margin-right 8px | ||||
|  | ||||
| </style> | ||||
							
								
								
									
										43
									
								
								src/web/app/desktop/views/pages/user/user.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								src/web/app/desktop/views/pages/user/user.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| <template> | ||||
| <mk-ui> | ||||
| 	<div class="user" v-if="!fetching"> | ||||
| 		<mk-user-header :user="user"/> | ||||
| 		<mk-user-home v-if="page == 'home'" :user="user"/> | ||||
| 		<mk-user-graphs v-if="page == 'graphs'" :user="user"/> | ||||
| 	</div> | ||||
| </mk-ui> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import Vue from 'vue'; | ||||
| import Progress from '../../../common/scripts/loading'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	props: { | ||||
| 		username: { | ||||
| 			type: String | ||||
| 		}, | ||||
| 		page: { | ||||
| 			default: 'home' | ||||
| 		} | ||||
| 	}, | ||||
| 	data() { | ||||
| 		return { | ||||
| 			fetching: true, | ||||
| 			user: null | ||||
| 		}; | ||||
| 	}, | ||||
| 	mounted() { | ||||
| 		Progress.start(); | ||||
| 		this.$root.$data.os.api('users/show', { | ||||
| 			username: this.username | ||||
| 		}).then(user => { | ||||
| 			this.fetching = false; | ||||
| 			this.user = user; | ||||
| 			Progress.done(); | ||||
| 			document.title = user.name + ' | Misskey'; | ||||
| 		}); | ||||
| 	} | ||||
| }); | ||||
| </script> | ||||
|  | ||||
		Reference in New Issue
	
	Block a user
	 syuilo
					syuilo