Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: notification box #5442

Merged
merged 74 commits into from
Apr 9, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
74 commits
Select commit Hold shift + click to select a range
dea873f
feat: notification box
floyd-li Mar 30, 2023
c8b61e6
feat: update graphql file
floyd-li Mar 31, 2023
be4d385
feat: remove suffix
floyd-li Mar 31, 2023
cfadd6b
feat: watch filter change
floyd-li Mar 31, 2023
7e6e8b7
Merge branch 'main' into feat/notification-box
floyd-li Apr 3, 2023
bfc652a
feat: add limit for notification
floyd-li Apr 3, 2023
5156529
refactor: only select one collection
floyd-li Apr 3, 2023
169d1ec
feat: ui update
floyd-li Apr 3, 2023
96badb8
feat(wip): external link redirect
floyd-li Mar 25, 2023
8a39b72
Update pages/redirect.vue
vikiival Mar 29, 2023
1f78861
feat: redirect modal
floyd-li Mar 31, 2023
15f5e9b
chore: remove unused config
floyd-li Mar 31, 2023
ffe0202
chore: remove unused page
floyd-li Mar 31, 2023
18b8fb1
fix: use little hack to solve i18n not work
floyd-li Mar 31, 2023
f3b9b7e
fix: work break
floyd-li Mar 31, 2023
34579ce
feat: add for more markdown
floyd-li Apr 1, 2023
7610774
Apply suggestions from code review
vikiival Apr 2, 2023
5818931
Update components/massmint/OnBoarding.vue
vikiival Apr 2, 2023
e0251d3
fix: update i18n types
floyd-li Apr 3, 2023
a0f2200
refactor: fix codeclimate issues
floyd-li Apr 3, 2023
5813a53
refactor: fix codeclimate issue
floyd-li Apr 3, 2023
97a359d
feat: add fuzzy matching
floyd-li Apr 3, 2023
c9949a0
fix: Scrolling in profile always kicks me back to top when loading
Jarsen136 Apr 2, 2023
b306267
✨ explore items scrollTopButton
roiLeo Apr 3, 2023
c5cf3bb
added new deployments to bug report
yangwao Apr 3, 2023
e8e26ef
Update bug.yml - canary
yangwao Apr 3, 2023
8472f2f
fix: collection on mobile is broken
Jarsen136 Apr 2, 2023
fe8db27
fix: remove columns
Jarsen136 Apr 3, 2023
b8428e7
#5499 Playing with stepsize on chart
yangwao Apr 3, 2023
b153e37
Merge branch 'main' into feat/notification-box
floyd-li Apr 3, 2023
fa0acaf
fix: upgrade @sentry/browser from 7.30.0 to 7.42.0
snyk-bot Apr 3, 2023
4db2780
chore: update lockfile
preschian Apr 3, 2023
7b6409c
fix: Infinity scroll is slow
Jarsen136 Mar 30, 2023
8d20f08
fix: remvoe vue-lazyload
Jarsen136 Mar 31, 2023
ed74386
fix: remove deps
Jarsen136 Mar 31, 2023
cd09eec
fix: update pnpm lock
Jarsen136 Apr 1, 2023
b383bcf
fix: infinite scroll on higher screen
Jarsen136 Apr 1, 2023
5dda90c
fix: prefetch more page
Jarsen136 Apr 1, 2023
eb18221
fix: infinite scroll with incorrect item width
Jarsen136 Apr 1, 2023
6110fba
fix: remove lazy attr
Jarsen136 Apr 2, 2023
a0f0432
fix: remove default value
Jarsen136 Apr 3, 2023
a4c21a2
Update libs/ui/src/components/MediaItem/type/ImageMedia.vue
vikiival Apr 2, 2023
6a6cdc9
fix: dark mode fallback iamge
Jarsen136 Apr 3, 2023
c89d502
[skip netlify] Update Node.js to ^v16.20.0
renovate[bot] Apr 3, 2023
ad69b07
Merge branch 'main' into feat/notification-box
floyd-li Apr 4, 2023
a845f80
feat(wip): update event type
floyd-li Apr 4, 2023
d48d4ab
fix: deepsource/codeclimate issue
floyd-li Apr 4, 2023
ad0cf22
refactor: adjust locale key order
floyd-li Apr 4, 2023
2505063
Merge branch 'main' into feat/notification-box
vikiival Apr 4, 2023
5d1c8f7
Merge branch 'main' into feat/notification-box
vikiival Apr 5, 2023
e85cd37
Merge branch 'main' into feat/notification-box
vikiival Apr 5, 2023
7ce3b63
Merge branch 'main' into feat/notification-box
vikiival Apr 6, 2023
21109eb
feat: fix type
floyd-li Apr 6, 2023
987c061
refactor: notification graphql query
floyd-li Apr 6, 2023
3d7816e
feat: add url watch
floyd-li Apr 6, 2023
a8f701e
fix(deepsource): remove v-html
floyd-li Apr 6, 2023
1c1399f
Merge commit '6b2be2e4c4da60f4278f3b9ee5f02e62a6c0dcbc'
floyd-li Apr 6, 2023
4445d4a
Merge branch 'main' into feat/notification-box
floyd-li Apr 6, 2023
75d9be5
fix(ui): update color
floyd-li Apr 7, 2023
c1ae90b
fix: update when change network
floyd-li Apr 7, 2023
b6fc81c
refactor: use fragments
floyd-li Apr 7, 2023
aa60c13
feat(ui): update notification item
floyd-li Apr 7, 2023
855d966
fix: query not correct
floyd-li Apr 7, 2023
1fe073f
fix: remove unexpected line
floyd-li Apr 7, 2023
7bd6a45
fix: rounded style
floyd-li Apr 7, 2023
9461ab2
feat: item image size
floyd-li Apr 7, 2023
fb45be2
fix: offer accepted event
floyd-li Apr 7, 2023
2496c87
fix: type waring
floyd-li Apr 7, 2023
e94b764
feat: add tooltip
floyd-li Apr 7, 2023
4131996
fix: money field error
floyd-li Apr 7, 2023
0657a4c
feat: filter expired offer
floyd-li Apr 7, 2023
4170ac4
fix: review suggestions
floyd-li Apr 7, 2023
c4a8b3a
fix: code review suggestions
floyd-li Apr 8, 2023
24098fb
Merge branch 'main' into feat/notification-box
floyd-li Apr 8, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions components/Navbar.vue
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,10 @@
id="NavChainSelect"
class="navbar-chain custom-navbar-item"
data-cy="chain-select" />
<NotificationBoxButton
v-if="account"
:show-label="isMobile"
@closeBurgerMenu="closeBurgerMenu" />
<template v-if="isMobile">
<MobileLanguageOption v-if="!account" />
<MobileExpandableSection
Expand Down Expand Up @@ -150,6 +154,7 @@ import ChainSelectDropdown from '~/components/navbar/ChainSelectDropdown.vue'
import StatsDropdown from '~/components/navbar/StatsDropdown.vue'
import MobileNavbarProfile from '~/components/navbar/MobileNavbarProfile.vue'
import ConnectWalletButton from '~/components/shared/ConnectWalletButton.vue'
import NotificationBoxButton from '~/components/navbar/NotificationBoxButton.vue'
import { ConnectWalletModalConfig } from '@/components/common/ConnectWallet/useConnectWallet'
import { BModalConfig } from 'buefy/types/components'
import type Vue from 'vue'
Expand Down
291 changes: 291 additions & 0 deletions components/common/NotificationBox/NotificationBoxModal.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,291 @@
<template>
<div class="notification-modal-container is-flex is-flex-direction-column">
<header class="modal-card-head mb-4">
<span class="modal-card-title is-size-6 has-text-weight-bold">
{{ $t('notification.notifications') }}
</span>
<a class="is-flex is-align-items-center" @click="emit('close')">
<NeoIcon icon="close" />
</a>
</header>
<div class="modal-card-body">
<div class="notification-filter pb-5">
<div
class="is-flex is-justify-content-space-between is-align-items-center pb-4">
<span> {{ $t('notification.filters') }} </span>
<NeoButton
v-if="!showFilter"
no-shadow
class="button-rounded"
@click.native="showFilter = !showFilter">
{{ $t('notification.add') }}
<NeoIcon icon="plus" size="small" />
</NeoButton>
<NeoButton
v-else
no-shadow
class="button-rounded"
@click.native="onClickDone">
{{ $t('notification.done') }}
<NeoIcon icon="check" size="small" />
</NeoButton>
</div>
<div v-if="showFilter" class="filter-option">
<div
v-if="collections.length > 0"
class="is-flex is-flex-direction-column pb-4">
<span class="is-size-7 has-text-grey mb-2">
{{ $t('notification.byCollection') }}
</span>
<div class="is-flex filter-list">
<div
v-for="(item, index) in collections"
:key="`${item}-${index}`"
class="filter-item px-3 py-1 no-wrap"
:class="{
activated: isInFilter('collection', item),
}"
@click="toggleFilter('collection', item)">
{{ item.name }}
</div>
</div>
</div>
<div class="is-flex is-flex-direction-column pb-4">
<span class="is-size-7 has-text-grey mb-2">{{
$t('notification.byEvent')
}}</span>
<div class="is-flex filter-list">
<div
v-for="(item, index) in eventTypes"
:key="`${item}-${index}`"
class="filter-item px-3 py-1 no-wrap"
:class="{
activated: isInFilter('event', item),
}"
@click="toggleFilter('event', item)">
{{ item.name }}
</div>
</div>
</div>
</div>
<div v-if="!showFilter && !isFilterEmpty" class="filter-option">
<div class="is-flex filter-display-list pb-4">
<div
v-for="(item, index) in filters.collection"
:key="`${item}-${index}`"
class="filter-item no-wrap mr-1 px-3 py-1">
{{ item.name }}
<NeoIcon
icon="xmark"
@click.native="removeFilter('collection', item)" />
</div>
<div
v-for="(item, index) in filters.event"
:key="`${item}-${index}`"
class="filter-item no-wrap px-3 py-1">
{{ item.name }}
<NeoIcon
icon="xmark"
@click.native="removeFilter('event', item)" />
</div>
</div>
</div>
</div>
<div v-if="displayedEvents.length === 0" class="empty-tip">
<p v-html="$t('notification.emptyTip')"></p>
</div>
<div v-else class="is-flex is-flex-direction-column">
<NotificationItem
v-for="(event, index) in displayedEvents"
:key="`${event.id}-${index}`"
:event="event" />
</div>
</div>
</div>
</template>

<script setup lang="ts">
import { Event, FilterOption } from './types'
import { NeoButton, NeoIcon } from '@kodadot1/brick'
import { Interaction } from '@kodadot1/minimark'
import NotificationItem from './NotificationItem.vue'
import { useNotification } from './useNotification'
const { $i18n, $store } = useNuxtApp()
type FilterType = 'collection' | 'event'
const eventTypes = ref<FilterOption[]>([
{
id: Interaction.BUY,
name: $i18n.t('filters.buy'),
},
{
// TODO
id: 'Offer',
name: $i18n.t('filters.offer'),
},
{
// TODO
id: 'Accepted Offer',
name: $i18n.t('filters.acceptedOffer'),
},
])
const { collections, events: allEvents } = useNotification(
$store.getters.getAuthAddress
)
const showFilter = ref(false)
const isFilterEmpty = computed(
() =>
filters.value.collection.length === 0 && filters.value.event.length === 0
)
const filters = ref<{
collection: FilterOption[]
event: FilterOption[]
}>({
collection: [],
event: [],
})
const isInFilter = (key: FilterType, target: FilterOption) => {
return filters.value[key].some((x) => x.id === target.id)
}
const toggleFilter = (key: FilterType, target: FilterOption) => {
if (isInFilter(key, target)) {
const index = filters.value[key].findIndex((x) => x.id === target.id)
filters.value[key].splice(index, 1)
} else {
filters.value[key].push(target)
}
}
const displayedEvents = ref<Event[]>([])
const doSearch = () => {
const { collection, event } = filters.value
displayedEvents.value = allEvents.value
if (collection.length > 0) {
displayedEvents.value = displayedEvents.value.filter((item) =>
collection.some((x) => x.id === item.nft.collection.id)
)
}
if (event.length > 0) {
displayedEvents.value = displayedEvents.value.filter((item) =>
event.some((x) => x.id === item.interaction)
)
}
}
const onClickDone = () => {
doSearch()
showFilter.value = false
}
const removeFilter = (key: FilterType, target: FilterOption) => {
toggleFilter(key, target)
doSearch()
}
watch(allEvents, doSearch, {
immediate: true,
})
const emit = defineEmits(['close'])
</script>

<style scoped lang="scss">
@import '@/styles/abstracts/variables';
.no-wrap {
floyd-li marked this conversation as resolved.
Show resolved Hide resolved
white-space: nowrap;
}
.notification-box-modal {
.notification-modal-container {
position: fixed;
top: 0;
right: 0;
height: 100%;
padding-top: 83px;
daiagi marked this conversation as resolved.
Show resolved Hide resolved
max-width: 360px;
width: 100%;
@include mobile {
padding-top: 58px;
max-width: 100vw;
}
@include ktheme() {
background: theme('background-color');
border-left: 1px solid theme('border-color');
floyd-li marked this conversation as resolved.
Show resolved Hide resolved
}
.modal-card-title {
@include ktheme() {
color: theme('text-color');
}
}
.modal-card-head {
background: unset;
padding: 1rem 2rem;
@include ktheme() {
border-bottom: 1px solid theme('border-color');
floyd-li marked this conversation as resolved.
Show resolved Hide resolved
}
}
.modal-card-body {
padding: 0rem 2rem 1rem 2rem;
display: block;
@include ktheme() {
background-color: theme('background-color');
}
}
.notification-filter {
flex-shrink: 0;
flex-grow: 0;
position: sticky;
top: 0;
@include ktheme() {
background-color: theme('background-color');
}
& > div:last-child {
@include ktheme() {
border-bottom: 1px solid theme('k-grey');
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

is it not border-color? can use the global class?

}
}
.button-rounded {
border-radius: 2rem;
}
.filter-item {
border-radius: 2rem;
cursor: pointer;
margin-right: 0.25rem;
floyd-li marked this conversation as resolved.
Show resolved Hide resolved
&:last-child {
margin-right: 0;
}
}
.filter-list {
overflow-x: auto;
.filter-item {
@include ktheme() {
border: 1px solid theme('k-shade');
}
&.activated {
@include ktheme() {
background-color: theme('k-shade');
}
}
}
}
.filter-display-list {
overflow-x: auto;
.filter-item {
@include ktheme() {
border: 1px solid theme('k-primary');
}
}
}
}
.empty-tip {
@include ktheme() {
color: theme('k-grey');
text-align: center;
}
}
}
}
</style>
<style lang="scss">
Comment on lines +256 to +257
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why do we have two style tag here?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

notification-box-modal container is created dynamically and will not take effect with scoped styels, i create this modal mainly refrence the wallet connect part. hmm, maybe this two modal need a refactor next

.modal-content {
border: none!important;
}
.modal-background {
background-color: rgba(0, 0, 0, 0.17)!important;
}

.notification-box-modal {
.modal-content {
border: none !important;
}
.modal-background {
background-color: rgba(0, 0, 0, 0.17) !important;
daiagi marked this conversation as resolved.
Show resolved Hide resolved
}
}
</style>
85 changes: 85 additions & 0 deletions components/common/NotificationBox/NotificationItem.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
<template>
<nuxt-link :to="`/${urlPrefix}/gallery/${event.nft.id}`" class="is-flex mb-5">
<img
v-if="avatar"
:src="avatar"
:alt="event.nft.name"
width="50"
height="50"
class="border image-size" />
<img
v-else
src="/placeholder.webp"
width="50"
height="50"
class="border image-size" />
roiLeo marked this conversation as resolved.
Show resolved Hide resolved
<div class="notify-content ml-5 is-flex is-flex-direction-column">
<div class="is-flex is-justify-content-space-between">
<div class="nft-name mr-4 has-text-weight-bold is-ellipsis">
{{ event.nft.name }}
</div>
<div class="nft-price is-ellipsis">
<Money :value="event.nft.price" />
</div>
</div>
<div class="is-flex is-justify-content-space-between">
<div>
<div class="height-50px is-flex is-align-items-center">
<div
class="event-type mr-4 border is-size-7 is-justify-content-center px-4 py-1 is-flex is-align-items-center"
:class="getInteractionColor(event.interaction)">
{{ event.interaction }}
</div>
</div>
</div>
<div class="is-ellipsis">
{{ formatToNow(new Date(event.timestamp)) }}
</div>
</div>
</div>
</nuxt-link>
</template>

<script setup lang="ts">
import { Event } from './types'
import { InteractionWithNFT } from '@/composables/collectionActivity/types'
import Money from '@/components/shared/format/ChainMoney.vue'
import { formatToNow } from '@/utils/format/time'
import {
getInteractionColor,
getNFTAvatar,
} from '@/components/collection/activity/events/eventRow/common'
const { urlPrefix } = usePrefix()
const props = defineProps<{
event: Event
}>()
const avatar = ref<string>()
onMounted(() => {
getAvatar()
})
const getAvatar = async () => {
if (props.event) {
avatar.value = await getNFTAvatar(
props.event as unknown as InteractionWithNFT
)
}
}
</script>

<style scoped lang="scss">
.notify-content {
flex: 1;
overflow: auto;
}
.nft-price {
overflow: auto;
}
.nft-name {
width: 0;
flex: 1;
}
.event-type {
line-height: 0.75rem;
border-radius: 2rem;
}
</style>
Loading