| @@ -10,6 +10,7 @@ unreleased | |||||||
| * ログイン時に二段階認証が分かりにくいのを改善 | * ログイン時に二段階認証が分かりにくいのを改善 | ||||||
| * 投稿のツールチップを出すのは時間の上だけに変更 | * 投稿のツールチップを出すのは時間の上だけに変更 | ||||||
| * ハッシュタグ判定の強化 | * ハッシュタグ判定の強化 | ||||||
|  | * ストーク機能の廃止 | ||||||
| * クライアントのAPIリクエストをストリーム経由で行うオプションを廃止 | * クライアントのAPIリクエストをストリーム経由で行うオプションを廃止 | ||||||
| * 一部箇所でカスタム絵文字が適用されていないのを修正 | * 一部箇所でカスタム絵文字が適用されていないのを修正 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1402,9 +1402,6 @@ desktop/views/pages/user/user.photos.vue: | |||||||
|  |  | ||||||
| desktop/views/pages/user/user.profile.vue: | desktop/views/pages/user/user.profile.vue: | ||||||
|   follows-you: "フォローされています" |   follows-you: "フォローされています" | ||||||
|   stalk: "ストークする" |  | ||||||
|   stalking: "ストーキングしています" |  | ||||||
|   unstalk: "ストーク解除" |  | ||||||
|   menu: "メニュー" |   menu: "メニュー" | ||||||
|  |  | ||||||
| desktop/views/pages/user/user.header.vue: | desktop/views/pages/user/user.header.vue: | ||||||
|   | |||||||
| @@ -3,10 +3,6 @@ | |||||||
| 	<div class="friend-form" v-if="$store.state.i.id != user.id"> | 	<div class="friend-form" v-if="$store.state.i.id != user.id"> | ||||||
| 		<mk-follow-button :user="user" block/> | 		<mk-follow-button :user="user" block/> | ||||||
| 		<p class="followed" v-if="user.isFollowed">{{ $t('follows-you') }}</p> | 		<p class="followed" v-if="user.isFollowed">{{ $t('follows-you') }}</p> | ||||||
| 		<p class="stalk" v-if="user.isFollowing"> |  | ||||||
| 			<span v-if="user.isStalking">{{ $t('stalking') }} <a @click="unstalk"><fa icon="meh"/> {{ $t('unstalk') }}</a></span> |  | ||||||
| 			<span v-if="!user.isStalking"><a @click="stalk"><fa icon="user-secret"/> {{ $t('stalk') }}</a></span> |  | ||||||
| 		</p> |  | ||||||
| 	</div> | 	</div> | ||||||
| 	<div class="action-form"> | 	<div class="action-form"> | ||||||
| 		<ui-button @click="menu" ref="menu">{{ $t('menu') }}</ui-button> | 		<ui-button @click="menu" ref="menu">{{ $t('menu') }}</ui-button> | ||||||
| @@ -24,26 +20,6 @@ export default Vue.extend({ | |||||||
| 	props: ['user'], | 	props: ['user'], | ||||||
|  |  | ||||||
| 	methods: { | 	methods: { | ||||||
| 		stalk() { |  | ||||||
| 			this.$root.api('following/stalk', { |  | ||||||
| 				userId: this.user.id |  | ||||||
| 			}).then(() => { |  | ||||||
| 				this.user.isStalking = true; |  | ||||||
| 			}, () => { |  | ||||||
| 				alert('error'); |  | ||||||
| 			}); |  | ||||||
| 		}, |  | ||||||
|  |  | ||||||
| 		unstalk() { |  | ||||||
| 			this.$root.api('following/unstalk', { |  | ||||||
| 				userId: this.user.id |  | ||||||
| 			}).then(() => { |  | ||||||
| 				this.user.isStalking = false; |  | ||||||
| 			}, () => { |  | ||||||
| 				alert('error'); |  | ||||||
| 			}); |  | ||||||
| 		}, |  | ||||||
|  |  | ||||||
| 		menu() { | 		menu() { | ||||||
| 			this.$root.new(XUserMenu, { | 			this.$root.new(XUserMenu, { | ||||||
| 				source: this.$refs.menu.$el, | 				source: this.$refs.menu.$el, | ||||||
| @@ -78,9 +54,6 @@ export default Vue.extend({ | |||||||
| 			background #eefaff | 			background #eefaff | ||||||
| 			border-radius 4px | 			border-radius 4px | ||||||
|  |  | ||||||
| 		> .stalk |  | ||||||
| 			margin 12px 0 0 0 |  | ||||||
|  |  | ||||||
| 	> .action-form | 	> .action-form | ||||||
| 		padding 16px | 		padding 16px | ||||||
| 		text-align center | 		text-align center | ||||||
|   | |||||||
| @@ -1,8 +1,3 @@ | |||||||
| # フォロー | # フォロー | ||||||
| ユーザーをフォローすると、タイムラインにそのユーザーの投稿が表示されるようになります。ただし、他のユーザーに対する返信は含まれません。 | ユーザーをフォローすると、タイムラインにそのユーザーの投稿が表示されるようになります。ただし、他のユーザーに対する返信は含まれません。 | ||||||
| ユーザーをフォローするには、ユーザーページの「フォロー」ボタンをクリックします。フォローを解除するには、もう一度クリックします。 | ユーザーをフォローするには、ユーザーページの「フォロー」ボタンをクリックします。フォローを解除するには、もう一度クリックします。 | ||||||
|  |  | ||||||
| ## ストーキング |  | ||||||
| ユーザーをフォローしている状態では、さらに「ストーキング」モードをオンにすることができます。ストーキングを行うと、タイムラインにそのユーザーの全ての投稿が表示されるようになります。つまり、他のユーザーに対する返信も含まれることになります。 |  | ||||||
| ストーキングするには、ユーザーページの「ストークする」をクリックします。ストーキングをやめるには、もう一度クリックします。 |  | ||||||
| ストーキングしていることは相手に通知されません。 |  | ||||||
|   | |||||||
| @@ -12,7 +12,6 @@ export type IFollowing = { | |||||||
| 	createdAt: Date; | 	createdAt: Date; | ||||||
| 	followeeId: mongo.ObjectID; | 	followeeId: mongo.ObjectID; | ||||||
| 	followerId: mongo.ObjectID; | 	followerId: mongo.ObjectID; | ||||||
| 	stalk: boolean; |  | ||||||
|  |  | ||||||
| 	// 非正規化 | 	// 非正規化 | ||||||
| 	_followee: { | 	_followee: { | ||||||
|   | |||||||
| @@ -219,7 +219,6 @@ export async function getRelation(me: mongo.ObjectId, target: mongo.ObjectId) { | |||||||
| 	return { | 	return { | ||||||
| 		id: target, | 		id: target, | ||||||
| 		isFollowing: following1 !== null, | 		isFollowing: following1 !== null, | ||||||
| 		isStalking: following1 ? following1.stalk : false, |  | ||||||
| 		hasPendingFollowRequestFromYou: followReq1 !== null, | 		hasPendingFollowRequestFromYou: followReq1 !== null, | ||||||
| 		hasPendingFollowRequestToYou: followReq2 !== null, | 		hasPendingFollowRequestToYou: followReq2 !== null, | ||||||
| 		isFollowed: following2 !== null, | 		isFollowed: following2 !== null, | ||||||
| @@ -352,7 +351,6 @@ export const pack = ( | |||||||
|  |  | ||||||
| 		_user.isFollowing = relation.isFollowing; | 		_user.isFollowing = relation.isFollowing; | ||||||
| 		_user.isFollowed = relation.isFollowed; | 		_user.isFollowed = relation.isFollowed; | ||||||
| 		_user.isStalking = relation.isStalking; |  | ||||||
| 		_user.hasPendingFollowRequestFromYou = relation.hasPendingFollowRequestFromYou; | 		_user.hasPendingFollowRequestFromYou = relation.hasPendingFollowRequestFromYou; | ||||||
| 		_user.hasPendingFollowRequestToYou = relation.hasPendingFollowRequestToYou; | 		_user.hasPendingFollowRequestToYou = relation.hasPendingFollowRequestToYou; | ||||||
| 		_user.isBlocking = relation.isBlocking; | 		_user.isBlocking = relation.isBlocking; | ||||||
|   | |||||||
| @@ -36,14 +36,12 @@ export const getFriends = async (me: mongodb.ObjectID, includeMe = true, remoteO | |||||||
|  |  | ||||||
| 	// ID list of other users who the I follows | 	// ID list of other users who the I follows | ||||||
| 	const myfollowings = followings.map(following => ({ | 	const myfollowings = followings.map(following => ({ | ||||||
| 		id: following.followeeId, | 		id: following.followeeId | ||||||
| 		stalk: following.stalk |  | ||||||
| 	})); | 	})); | ||||||
|  |  | ||||||
| 	if (includeMe) { | 	if (includeMe) { | ||||||
| 		myfollowings.push({ | 		myfollowings.push({ | ||||||
| 			id: me, | 			id: me | ||||||
| 			stalk: true |  | ||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,50 +0,0 @@ | |||||||
| import $ from 'cafy'; import ID, { transform } from '../../../../misc/cafy-id'; |  | ||||||
| import Following from '../../../../models/following'; |  | ||||||
| import define from '../../define'; |  | ||||||
|  |  | ||||||
| export const meta = { |  | ||||||
| 	desc: { |  | ||||||
| 		'ja-JP': '指定したユーザーをストーキングします。', |  | ||||||
| 		'en-US': 'Stalk a user.' |  | ||||||
| 	}, |  | ||||||
|  |  | ||||||
| 	requireCredential: true, |  | ||||||
|  |  | ||||||
| 	kind: 'following-write', |  | ||||||
|  |  | ||||||
| 	params: { |  | ||||||
| 		userId: { |  | ||||||
| 			validator: $.type(ID), |  | ||||||
| 			transform: transform, |  | ||||||
| 			desc: { |  | ||||||
| 				'ja-JP': '対象のユーザーのID', |  | ||||||
| 				'en-US': 'Target user ID' |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default define(meta, (ps, user) => new Promise(async (res, rej) => { |  | ||||||
| 	const follower = user; |  | ||||||
|  |  | ||||||
| 	// Fetch following |  | ||||||
| 	const following = await Following.findOne({ |  | ||||||
| 		followerId: follower._id, |  | ||||||
| 		followeeId: ps.userId |  | ||||||
| 	}); |  | ||||||
|  |  | ||||||
| 	if (following === null) { |  | ||||||
| 		return rej('following not found'); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Stalk |  | ||||||
| 	await Following.update({ _id: following._id }, { |  | ||||||
| 		$set: { |  | ||||||
| 			stalk: true |  | ||||||
| 		} |  | ||||||
| 	}); |  | ||||||
|  |  | ||||||
| 	res(); |  | ||||||
|  |  | ||||||
| 	// TODO: イベント |  | ||||||
| })); |  | ||||||
| @@ -1,50 +0,0 @@ | |||||||
| import $ from 'cafy'; import ID, { transform } from '../../../../misc/cafy-id'; |  | ||||||
| import Following from '../../../../models/following'; |  | ||||||
| import define from '../../define'; |  | ||||||
|  |  | ||||||
| export const meta = { |  | ||||||
| 	desc: { |  | ||||||
| 		'ja-JP': '指定したユーザーのストーキングをやめます。', |  | ||||||
| 		'en-US': 'Unstalk a user.' |  | ||||||
| 	}, |  | ||||||
|  |  | ||||||
| 	requireCredential: true, |  | ||||||
|  |  | ||||||
| 	kind: 'following-write', |  | ||||||
|  |  | ||||||
| 	params: { |  | ||||||
| 		userId: { |  | ||||||
| 			validator: $.type(ID), |  | ||||||
| 			transform: transform, |  | ||||||
| 			desc: { |  | ||||||
| 				'ja-JP': '対象のユーザーのID', |  | ||||||
| 				'en-US': 'Target user ID' |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export default define(meta, (ps, user) => new Promise(async (res, rej) => { |  | ||||||
| 	const follower = user; |  | ||||||
|  |  | ||||||
| 	// Fetch following |  | ||||||
| 	const following = await Following.findOne({ |  | ||||||
| 		followerId: follower._id, |  | ||||||
| 		followeeId: ps.userId |  | ||||||
| 	}); |  | ||||||
|  |  | ||||||
| 	if (following === null) { |  | ||||||
| 		return rej('following not found'); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Stalk |  | ||||||
| 	await Following.update({ _id: following._id }, { |  | ||||||
| 		$set: { |  | ||||||
| 			stalk: false |  | ||||||
| 		} |  | ||||||
| 	}); |  | ||||||
|  |  | ||||||
| 	res(); |  | ||||||
|  |  | ||||||
| 	// TODO: イベント |  | ||||||
| })); |  | ||||||
| @@ -119,12 +119,10 @@ export default define(meta, (ps, user) => new Promise(async (res, rej) => { | |||||||
| 		_id: -1 | 		_id: -1 | ||||||
| 	}; | 	}; | ||||||
|  |  | ||||||
| 	const followQuery = followings.map(f => f.stalk ? { | 	const followQuery = followings.map(f => ({ | ||||||
| 		userId: f.id |  | ||||||
| 	} : { |  | ||||||
| 		userId: f.id, | 		userId: f.id, | ||||||
|  |  | ||||||
| 		// ストーキングしてないならリプライは含めない(ただし投稿者自身の投稿へのリプライ、自分の投稿へのリプライ、自分のリプライは含める) | 		// リプライは含めない(ただし投稿者自身の投稿へのリプライ、自分の投稿へのリプライ、自分のリプライは含める) | ||||||
| 		$or: [{ | 		$or: [{ | ||||||
| 			// リプライでない | 			// リプライでない | ||||||
| 			replyId: null | 			replyId: null | ||||||
| @@ -140,7 +138,7 @@ export default define(meta, (ps, user) => new Promise(async (res, rej) => { | |||||||
| 			// 自分(フォロワー)が送信したリプライ | 			// 自分(フォロワー)が送信したリプライ | ||||||
| 			userId: user._id | 			userId: user._id | ||||||
| 		}] | 		}] | ||||||
| 	}); | 	})); | ||||||
|  |  | ||||||
| 	const visibleQuery = user == null ? [{ | 	const visibleQuery = user == null ? [{ | ||||||
| 		visibility: { $in: [ 'public', 'home' ] } | 		visibility: { $in: [ 'public', 'home' ] } | ||||||
|   | |||||||
| @@ -117,9 +117,7 @@ export default define(meta, (ps, user) => new Promise(async (res, rej) => { | |||||||
| 		_id: -1 | 		_id: -1 | ||||||
| 	}; | 	}; | ||||||
|  |  | ||||||
| 	const followQuery = followings.map(f => f.stalk ? { | 	const followQuery = followings.map(f => ({ | ||||||
| 		userId: f.id |  | ||||||
| 	} : { |  | ||||||
| 		userId: f.id, | 		userId: f.id, | ||||||
|  |  | ||||||
| 		// ストーキングしてないならリプライは含めない(ただし投稿者自身の投稿へのリプライ、自分の投稿へのリプライ、自分のリプライは含める) | 		// ストーキングしてないならリプライは含めない(ただし投稿者自身の投稿へのリプライ、自分の投稿へのリプライ、自分のリプライは含める) | ||||||
| @@ -138,7 +136,7 @@ export default define(meta, (ps, user) => new Promise(async (res, rej) => { | |||||||
| 			// 自分(フォロワー)が送信したリプライ | 			// 自分(フォロワー)が送信したリプライ | ||||||
| 			userId: user._id | 			userId: user._id | ||||||
| 		}] | 		}] | ||||||
| 	}); | 	})); | ||||||
|  |  | ||||||
| 	const visibleQuery = user == null ? [{ | 	const visibleQuery = user == null ? [{ | ||||||
| 		visibility: { $in: [ 'public', 'home' ] } | 		visibility: { $in: [ 'public', 'home' ] } | ||||||
|   | |||||||
| @@ -540,12 +540,9 @@ async function publishToFollowers(note: INote, user: IUser, noteActivity: any) { | |||||||
| 		const follower = following._follower; | 		const follower = following._follower; | ||||||
|  |  | ||||||
| 		if (isLocalUser(follower)) { | 		if (isLocalUser(follower)) { | ||||||
| 			// ストーキングしていない場合 |  | ||||||
| 			if (!following.stalk) { |  | ||||||
| 			// この投稿が返信ならスキップ | 			// この投稿が返信ならスキップ | ||||||
| 			if (note.replyId && !note._reply.userId.equals(following.followerId) && !note._reply.userId.equals(note.userId)) | 			if (note.replyId && !note._reply.userId.equals(following.followerId) && !note._reply.userId.equals(note.userId)) | ||||||
| 				continue; | 				continue; | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			// Publish event to followers stream | 			// Publish event to followers stream | ||||||
| 			publishHomeTimelineStream(following.followerId, detailPackedNote); | 			publishHomeTimelineStream(following.followerId, detailPackedNote); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 syuilo
					syuilo