misskey/src/web/app/mobile/views/pages/home.vue

239 lines
5.9 KiB
Vue
Raw Normal View History

2018-02-15 17:50:19 +09:00
<template>
2018-02-24 02:46:09 +09:00
<mk-ui>
<span slot="header" @click="showTl = !showTl">
<template v-if="showTl">%fa:home%タイムライン</template>
<template v-else>%fa:home%ウィジェット</template>
<span style="margin-left:8px">
<template v-if="showTl">%fa:angle-down%</template>
<template v-else>%fa:angle-up%</template>
</span>
</span>
<template slot="func">
<button @click="fn" v-if="showTl">%fa:pencil-alt%</button>
<button @click="customizing = !customizing" v-else>%fa:cog%</button>
</template>
<main>
<div class="tl">
<mk-timeline @loaded="onLoaded" v-show="showTl"/>
</div>
<div class="widgets" v-if="!showTl">
<template v-if="customizing">
<header>
<select v-model="widgetAdderSelected">
<option value="profile">プロフィール</option>
<option value="calendar">カレンダー</option>
<option value="activity">アクティビティ</option>
<option value="rss">RSSリーダー</option>
<option value="photo-stream">フォトストリーム</option>
<option value="version">バージョン</option>
<option value="access-log">アクセスログ</option>
<option value="server">サーバー情報</option>
<option value="donation">寄付のお願い</option>
<option value="nav">ナビゲーション</option>
<option value="tips">ヒント</option>
</select>
<button @click="addWidget">追加</button>
2018-02-24 03:05:50 +09:00
<p><a @click="hint">カスタマイズのヒント</a></p>
2018-02-24 02:46:09 +09:00
</header>
<x-draggable
:list="widgets"
:options="{ handle: '.handle', animation: 150 }"
@sort="onWidgetSort"
>
<div v-for="widget in widgets" class="customize-container" :key="widget.id">
<header>
<span class="handle">%fa:bars%</span>{{ widget.name }}<button class="remove" @click="removeWidget(widget)">%fa:times%</button>
</header>
2018-02-24 03:03:26 +09:00
<div @click="widgetFunc(widget.id)">
2018-02-24 02:46:09 +09:00
<component :is="`mkw-${widget.name}`" :widget="widget" :ref="widget.id" :is-mobile="true"/>
</div>
</div>
</x-draggable>
</template>
<template v-else>
<component class="widget" v-for="widget in widgets" :is="`mkw-${widget.name}`" :key="widget.id" :widget="widget" :is-mobile="true" @chosen="warp"/>
</template>
</div>
</main>
2018-02-15 17:50:19 +09:00
</mk-ui>
</template>
<script lang="ts">
import Vue from 'vue';
2018-02-24 02:46:09 +09:00
import * as XDraggable from 'vuedraggable';
import * as uuid from 'uuid';
2018-02-15 17:50:19 +09:00
import Progress from '../../../common/scripts/loading';
import getPostSummary from '../../../../../common/get-post-summary';
export default Vue.extend({
2018-02-24 02:46:09 +09:00
components: {
XDraggable
},
2018-02-15 17:50:19 +09:00
data() {
return {
connection: null,
connectionId: null,
2018-02-24 02:46:09 +09:00
unreadCount: 0,
showTl: true,
widgets: [],
customizing: false,
widgetAdderSelected: null
2018-02-15 17:50:19 +09:00
};
},
2018-02-24 02:46:09 +09:00
created() {
if ((this as any).os.i.client_settings.mobile_home == null) {
2018-02-24 03:03:26 +09:00
Vue.set((this as any).os.i.client_settings, 'mobile_home', [{
2018-02-24 02:46:09 +09:00
name: 'calendar',
2018-02-24 03:03:26 +09:00
id: 'a', data: {}
2018-02-24 02:46:09 +09:00
}, {
name: 'activity',
2018-02-24 03:03:26 +09:00
id: 'b', data: {}
2018-02-24 02:46:09 +09:00
}, {
name: 'rss',
2018-02-24 03:03:26 +09:00
id: 'c', data: {}
2018-02-24 02:46:09 +09:00
}, {
name: 'photo-stream',
2018-02-24 03:03:26 +09:00
id: 'd', data: {}
2018-02-24 02:46:09 +09:00
}, {
name: 'donation',
2018-02-24 03:03:26 +09:00
id: 'e', data: {}
2018-02-24 02:46:09 +09:00
}, {
name: 'nav',
2018-02-24 03:03:26 +09:00
id: 'f', data: {}
2018-02-24 02:46:09 +09:00
}, {
name: 'version',
2018-02-24 03:03:26 +09:00
id: 'g', data: {}
2018-02-24 02:46:09 +09:00
}]);
2018-02-24 03:03:26 +09:00
this.widgets = (this as any).os.i.client_settings.mobile_home;
this.saveHome();
} else {
this.widgets = (this as any).os.i.client_settings.mobile_home;
2018-02-24 02:46:09 +09:00
}
2018-02-24 03:03:26 +09:00
this.$watch('os.i', i => {
this.widgets = (this as any).os.i.client_settings.mobile_home;
}, {
deep: true
});
2018-02-24 02:46:09 +09:00
},
2018-02-15 17:50:19 +09:00
mounted() {
document.title = 'Misskey';
document.documentElement.style.background = '#313a42';
2018-02-18 12:35:18 +09:00
this.connection = (this as any).os.stream.getConnection();
this.connectionId = (this as any).os.stream.use();
2018-02-15 17:50:19 +09:00
this.connection.on('post', this.onStreamPost);
document.addEventListener('visibilitychange', this.onVisibilitychange, false);
Progress.start();
},
beforeDestroy() {
this.connection.off('post', this.onStreamPost);
2018-02-18 12:35:18 +09:00
(this as any).os.stream.dispose(this.connectionId);
2018-02-15 17:50:19 +09:00
document.removeEventListener('visibilitychange', this.onVisibilitychange);
},
methods: {
fn() {
2018-02-22 03:11:24 +09:00
(this as any).apis.post();
2018-02-15 17:50:19 +09:00
},
2018-02-24 02:46:09 +09:00
onLoaded() {
2018-02-15 17:50:19 +09:00
Progress.done();
},
onStreamPost(post) {
2018-02-18 12:35:18 +09:00
if (document.hidden && post.user_id !== (this as any).os.i.id) {
2018-02-15 17:50:19 +09:00
this.unreadCount++;
document.title = `(${this.unreadCount}) ${getPostSummary(post)}`;
}
},
onVisibilitychange() {
if (!document.hidden) {
this.unreadCount = 0;
document.title = 'Misskey';
}
2018-02-24 02:46:09 +09:00
},
2018-02-24 03:05:50 +09:00
hint() {
alert('ウィジェットを追加/削除したり並べ替えたりできます。ウィジェットを移動するには「三」をドラッグします。ウィジェットを削除するには「x」をタップします。いくつかのウィジェットはタップすることで表示を変更できます。');
},
2018-02-24 03:03:26 +09:00
widgetFunc(id) {
const w = this.$refs[id][0];
if (w.func) w.func();
},
2018-02-24 02:46:09 +09:00
onWidgetSort() {
this.saveHome();
},
addWidget() {
const widget = {
name: this.widgetAdderSelected,
id: uuid(),
data: {}
};
this.widgets.unshift(widget);
this.saveHome();
},
removeWidget(widget) {
this.widgets = this.widgets.filter(w => w.id != widget.id);
this.saveHome();
},
saveHome() {
(this as any).api('i/update_mobile_home', {
home: this.widgets
});
},
warp() {
2018-02-15 17:50:19 +09:00
}
}
});
</script>
2018-02-24 02:46:09 +09:00
<style lang="stylus" scoped>
main
> .tl
> .mk-timeline
max-width 600px
margin 0 auto
padding 8px
@media (min-width 500px)
padding 16px
> .widgets
margin 0 auto
max-width 500px
> header
padding 8px
background #fff
.widget
margin 8px
.customize-container
margin 8px
background #fff
> header
line-height 32px
background #eee
> .handle
padding 0 8px
> .remove
position absolute
top 0
right 0
padding 0 8px
line-height 32px
> div
padding 8px
> *
pointer-events none
</style>