我有一个正在使用的“相册”组件。我正在尝试创建某种东西,以便在给定时间只能切换一张专辑。因此,如果在单击另一个专辑时已经打开了一个专辑,它将关闭。
将组件递归安装在“父”应用中。
<album v-for="(album, index) in albums"
:key="index"
@expand="setAlbumContainerHeight"
@action="removeFromCollection"
:albumDetails="album"
page="collection">
</album>
组件本身。
<template>
<div v-on:click="toggleExpand" ref="album" class="album">
<img class="album-artwork" alt="album-artwork" :src="albumDetails.cover_url">
<div ref="expandContent" class="expanded-content">
<div class="left-block">
<div class="title">{{ albumDetails.title }}</div>
<div class="artist">{{ albumDetails.artist }}</div>
<!-- Search page specific -->
<div v-if="page === 'search'" class="info">{{ albumDetails.year }}<br> <span v-for="genre in albumDetails.genre"> {{ genre }}</span><br> {{ albumDetails.label }}</div>
<!-- Collection page specific -->
<div v-if="page === 'collection'" class="info">{{ albumDetails.year }}<br> <span v-for="genre in albumDetails.genres"> {{ genre.genre }}</span><br> {{ albumDetails.label }}</div>
<!-- Search page specific -->
<button v-if="page === 'search'" v-on:click.stop="addToCollection" class="add-collection">Add to collection</button>
<!-- Collection page specific -->
<button v-if="page === 'collection'" v-on:click.stop="removeFromCollection" class="add-collection">Remove from collection</button>
</div>
<div v-if="page === 'search'" class="right-block">
<div class="a-side">
<p v-for="track in albumDetails.track_list_one" class="track">{{ track }}</p>
</div>
<div class="b-side">
<p v-for="track in albumDetails.track_list_two" class="track">{{ track }}</p>
</div>
</div>
<div v-if="page === 'collection'" class="right-block">
<div class="a-side">
<p v-for="track in trackListOne" class="track">{{ track.track }}</p>
</div>
<div class="b-side">
<p v-for="track in trackListTwo" class="track">{{ track.track }}</p>
</div>
</div>
<img class="faded-album-artwork" alt="faded-album-artwork" :src="albumDetails.cover_url">
</div>
</div>
</template>
<script>
module.exports = {
name: 'Album',
props: ['albumDetails', 'page'],
data: function () {
return {
expanded: false,
expandedContentHeight: 0,
trackListOne: [],
trackListTwo: []
}
},
mounted() {
if (this.albumDetails.tracks) {
this.getTrackListOne(this.albumDetails.tracks);
this.getTrackListTwo(this.albumDetails.tracks);
}
},
methods: {
toggleExpand() {
if (!this.expanded) {
this.expanded = true;
this.$refs.expandContent.style.display = 'flex';
this.expandedContentHeight = this.$refs.expandContent.clientHeight;
let height = this.$refs.album.clientHeight + this.expandedContentHeight;
this.$emit('expand', height, event);
} else {
this.expanded = false;
this.$refs.expandContent.style.display = 'none';
let height = 'initial';
this.$emit('expand', height, event);
}
},
addToCollection() {
this.$emit('action', this.albumDetails.cat_no);
},
removeFromCollection() {
this.$emit('action', this.albumDetails.cat_no);
},
getTrackListOne(tracks) {
let halfWayThough = Math.floor(tracks.length / 2);
this.trackListOne = tracks.slice(0, halfWayThough);
},
getTrackListTwo(tracks) {
let halfWayThough = Math.floor(tracks.length / 2);
this.trackListTwo = tracks.slice(halfWayThough, tracks.length);
},
},
}
</script>
<style scoped lang="scss">
@import './styles/album';
</style>
组件本身使用简单的“扩展”数据属性存储其状态。当前,这意味着该组件在用户打开和关闭每张专辑时可以很好地工作,但是,当他们一次打开几张专辑时会出现问题。由于绝对定位,我不得不手动设置容器的高度。目的是一次只能打开一个。我在思考如何存储和跟踪当前打开的专辑的方式时遇到了麻烦-我也不确定父母如何管理每个专辑的舞台/在哪里打开和关闭它们。即使我知道父母中打开了哪张专辑,如何触发单个子组件方法?
我已尝试尽可能清楚,请让我知道是否还有其他可以澄清的地方。
答案 0 :(得分:1)
就我所知,您不是在递归地引用专辑组件,所以我不太确定您的意思是递归。
我会采用这种方法:
expanded
是组件的属性。
单击该组件会发出this.$emit('expand', albumID)
。
在您的父母中,您监听@expand
事件,并将expandedAlbumID
分配给事件有效负载。
剩下的最后一块难题就是传递您的expanded
属性,例如:expanded="expandedAlbumID == album.id"