updated front page layout

This commit is contained in:
2023-05-03 20:23:35 +10:00
parent 21fa5d24af
commit 0072b28965
6 changed files with 628 additions and 382 deletions

View File

@@ -0,0 +1,79 @@
<template>
<router-link
:to="{ name: 'article', params: { slug: props.article.slug } }"
class="article-card">
<div
class="thumbnail"
:style="{
backgroundImage: `url(${mediaGetVariantUrl(
props.article.hero,
'medium'
)})`,
}"></div>
<div class="info">
{{ props.article.user.display_name }} -
{{ computedDate(props.article.publish_at) }}
</div>
<h3 class="title">{{ props.article.title }}</h3>
<p class="content">
{{ excerpt(props.article.content) }}
</p>
</router-link>
</template>
<script setup lang="ts">
import { Article } from "../helpers/api.types";
import { SMDate } from "../helpers/datetime";
import { mediaGetVariantUrl } from "../helpers/media";
import { excerpt } from "../helpers/string";
const props = defineProps({
article: {
type: Object as () => Article,
required: true,
},
});
const computedDate = (date) => {
return new SMDate(date, { format: "yMd" }).format("d MMMM yyyy");
};
</script>
<style lang="scss">
a.article-card {
text-decoration: none;
color: var(--card-color-text);
margin-bottom: 48px;
&:hover {
filter: none;
.thumbnail {
filter: brightness(115%);
}
}
.thumbnail {
aspect-ratio: 16 / 9;
border-radius: 7px;
background-position: center;
background-size: cover;
background-color: var(--card-color);
box-shadow: var(--base-shadow);
margin-bottom: 24px;
}
.info {
font-size: 80%;
}
.title {
margin: 16px 0;
word-break: break-word;
}
.content {
font-size: 90%;
}
}
</style>

View File

@@ -0,0 +1,309 @@
<template>
<router-link
class="event-card"
:to="{ name: 'event', params: { id: props.event.id } }">
<div
class="thumbnail"
:style="{
backgroundImage: `url('${mediaGetVariantUrl(
props.event.hero,
'medium'
)}')`,
}">
<div :class="['banner', computedBanner(props.event)['type']]">
{{ computedBanner(props.event)["banner"] }}
</div>
<div class="date">
<div class="day">
{{ formatDateDay(props.event.start_at) }}
</div>
<div class="month">
{{ formatDateMonth(props.event.start_at) }}
</div>
</div>
</div>
<div class="content">
<h3 class="title">{{ props.event.title }}</h3>
<SMRow class="date" no-responsive>
<ion-icon name="calendar-outline" class="icon" />
<div class="text">{{ computedDate(props.event) }}</div>
</SMRow>
<SMRow class="location" no-responsive>
<ion-icon name="location-outline" class="icon" />
<div class="text">
{{ computedLocation(props.event) }}
</div>
</SMRow>
<SMRow class="ages" no-responsive>
<ion-icon name="body-outline" class="icon" />
<div class="text">
{{ computedAges(props.event.ages) }}
</div>
</SMRow>
<SMRow class="price" no-responsive>
<div class="icon">$</div>
<div class="text">
{{ computedPrice(props.event.price) }}
</div>
</SMRow>
</div>
</router-link>
</template>
<script setup lang="ts">
import { Event } from "../helpers/api.types";
import { SMDate } from "../helpers/datetime";
import { mediaGetVariantUrl } from "../helpers/media";
const props = defineProps({
event: {
type: Object as () => Event,
required: true,
},
});
/**
* Return a human readable Date string.
*
* @param {Event} event The event to convert.
* @returns The converted string.
*/
const computedDate = (event: Event) => {
let str = "";
if (event.start_at.length > 0) {
if (
event.end_at.length > 0 &&
event.start_at.substring(0, event.start_at.indexOf(" ")) !=
event.end_at.substring(0, event.end_at.indexOf(" "))
) {
str = new SMDate(event.start_at, { format: "yMd" }).format(
"dd/MM/yyyy"
);
if (event.end_at.length > 0) {
str =
str +
" - " +
new SMDate(event.end_at, { format: "yMd" }).format(
"dd/MM/yyyy"
);
}
} else {
str = new SMDate(event.start_at, { format: "yMd" }).format(
"dd/MM/yyyy @ h:mm aa"
);
}
}
return str;
};
/**
* Return a the event starting month day number.
*
* @param {string} date The date to format.
* @returns The converted string.
*/
const formatDateDay = (date: string) => {
return new SMDate(date, { format: "yMd" }).format("dd");
};
/**
* Return a the event starting month name.
*
* @param {string} date The date to format.
* @returns The converted string.
*/
const formatDateMonth = (date: string) => {
return new SMDate(date, { format: "yMd" }).format("MMM");
};
/**
* Return a human readable Location string.
*
* @param {Event} event The event to convert.
* @returns The converted string.
*/
const computedLocation = (event: Event): string => {
if (event.location == "online") {
return "Online";
}
return event.address;
};
/**
* Return a human readable Ages string.
*
* @param {string} ages The string to convert.
* @returns The converted string.
*/
const computedAges = (ages: string): string => {
const trimmed = ages.trim();
const regex = /^(\d+)(\s*\+?\s*|\s*-\s*\d+\s*)?$/;
if (trimmed.length === 0) {
return "All ages";
}
if (regex.test(trimmed)) {
return `Ages ${trimmed}`;
}
return ages;
};
/**
* Return a human readable Price string.
*
* @param {string} price The string to convert.
* @returns The converted string.
*/
const computedPrice = (price: string): string => {
const trimmed = parseInt(price.trim());
if (isNaN(trimmed) || trimmed == 0) {
return "Free";
}
return trimmed.toString();
};
type EventBanner = {
banner: string;
type: string;
};
const computedBanner = (event: Event): EventBanner => {
const parsedEndAt = new SMDate(event.end_at, {
format: "yyyy-MM-dd HH:mm:ss",
utc: true,
});
if (
(parsedEndAt.isBefore(new SMDate("now")) &&
(event.status == "open" || event.status == "soon")) ||
event.status == "closed"
) {
return {
banner: "closed",
type: "expired",
};
} else if (event.status == "open") {
return {
banner: "open",
type: "success",
};
} else if (event.status == "cancelled") {
return {
banner: "cancelled",
type: "danger",
};
}
return {
banner: "Open Soon",
type: "warning",
};
};
</script>
<style lang="scss">
a.event-card {
background-color: var(--base-color-light);
box-shadow: 0 5px 10px -3px rgba(0, 0, 0, 0.25);
border-radius: 8px;
text-decoration: none;
color: var(--base-color-text);
position: relative;
overflow: hidden;
.thumbnail {
width: 100%;
aspect-ratio: 16 / 9;
background-position: center;
background-repeat: no-repeat;
background-size: cover;
border-radius: 8px 8px 0 0;
.banner {
position: absolute;
background-color: var(--banner-green-color);
font-size: 70%;
font-weight: 700;
color: var(--banner-green-color-text);
padding: 6px 18px;
text-align: center;
top: 10px;
right: 10px;
text-transform: uppercase;
&.expired {
background-color: var(--banner-purple-color);
color: var(--banner-purple-color-text);
}
&.danger {
background-color: var(--banner-red-color);
color: var(--banner-red-color-text);
}
&.warning {
background-color: var(--banner-yellow-color);
color: var(--banner-yellow-color-text);
}
}
.date {
position: absolute;
top: 10px;
left: 10px;
background-color: var(--base-color);
box-shadow: var(--base-shadow);
padding: 8px 12px;
text-align: center;
border-radius: 2px;
.day {
font-weight: 700;
padding: 1px;
}
.month {
font-size: 65%;
text-transform: uppercase;
}
}
}
.content {
padding: 16px;
}
.title {
margin: 0 0 16px 0;
font-size: 100%;
word-break: break-all;
}
.row {
display: flex;
margin-bottom: 8px;
font-size: 80%;
.icon {
width: 20px;
text-align: center;
margin-right: 8px;
}
}
&:hover {
cursor: pointer;
filter: none;
.image {
filter: brightness(115%);
}
}
}
</style>

View File

@@ -5,33 +5,28 @@
<div class="hero-background" :style="heroStyles"></div> <div class="hero-background" :style="heroStyles"></div>
<SMContainer class="align-items-start"> <SMContainer class="align-items-start">
<div class="hero-content"> <div class="hero-content">
<h1>{{ heroTitle }}</h1> <h1>{{ props.title }}</h1>
<p>{{ heroExcerpt }}</p> <p>{{ props.excerpt }}</p>
<div class="hero-buttons"> <div class="hero-buttons">
<SMButton <SMButton
v-if="loaded" v-if="loaded"
type="primary" type="primary"
:to="{ :to="props.to"
name: 'article', :label="props.more" />
params: { slug: heroSlug },
}"
label="Read More" />
</div> </div>
</div> </div>
</SMContainer> </SMContainer>
<div class="hero-caption"> <div class="hero-caption">
<router-link <router-link v-if="loaded" :to="props.to">{{
v-if="loaded" props.imageTitle
:to="{ name: 'article', params: { slug: heroSlug } }" }}</router-link>
>{{ heroImageTitle }}</router-link
>
</div> </div>
</template> </template>
</section> </section>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { onBeforeUnmount, onMounted, ref, watch } from "vue"; import { computed, onBeforeUnmount, onMounted, ref, watch } from "vue";
import { api, getApiResultData } from "../helpers/api"; import { api, getApiResultData } from "../helpers/api";
import { ArticleCollection } from "../helpers/api.types"; import { ArticleCollection } from "../helpers/api.types";
import { mediaGetVariantUrl } from "../helpers/media"; import { mediaGetVariantUrl } from "../helpers/media";
@@ -39,12 +34,35 @@ import { excerpt } from "../helpers/string";
import SMButton from "./SMButton.vue"; import SMButton from "./SMButton.vue";
import SMLoading from "./SMLoading.vue"; import SMLoading from "./SMLoading.vue";
const props = defineProps({
title: {
type: String,
required: true,
},
excerpt: {
type: String,
required: true,
},
imageUrl: {
type: String,
required: true,
},
imageTitle: {
type: String,
required: true,
},
to: {
type: Object,
required: true,
},
more: {
type: String,
default: "Read More",
required: false,
},
});
const loaded = ref(false); const loaded = ref(false);
let heroTitle = ref("");
let heroExcerpt = ref("");
let heroImageUrl = ref("");
let heroImageTitle = "";
let heroSlug = ref("");
const translateY = ref(0); const translateY = ref(0);
const heroStyles = ref({ const heroStyles = ref({
backgroundImage: "none", backgroundImage: "none",
@@ -60,6 +78,18 @@ watch(translateY, () => {
heroStyles.value.transform = `translateY(${translateY.value}px)`; heroStyles.value.transform = `translateY(${translateY.value}px)`;
}); });
const updateImageUrl = (url: string) => {
heroStyles.value.backgroundImage = `linear-gradient(to right, rgba(0, 0, 0, 0.7),rgba(0, 0, 0, 0.2)),url('${url}')`;
loaded.value = true;
};
watch(
() => props.imageUrl,
() => {
updateImageUrl(props.imageUrl);
}
);
onMounted(() => { onMounted(() => {
window.addEventListener("scroll", handleScroll); window.addEventListener("scroll", handleScroll);
}); });
@@ -68,44 +98,9 @@ onBeforeUnmount(() => {
window.removeEventListener("scroll", handleScroll); window.removeEventListener("scroll", handleScroll);
}); });
const handleLoad = async () => { if (props.imageUrl && props.imageUrl !== "") {
try { updateImageUrl(props.imageUrl);
let articlesResult = await api.get({ }
url: "/articles",
params: {
limit: 3,
},
});
const articlesData =
getApiResultData<ArticleCollection>(articlesResult);
if (articlesData && articlesData.articles) {
const randomIndex = Math.floor(
Math.random() * articlesData.articles.length
);
heroTitle.value = articlesData.articles[randomIndex].title;
heroExcerpt.value = excerpt(
articlesData.articles[randomIndex].content,
200
);
heroImageUrl.value = mediaGetVariantUrl(
articlesData.articles[randomIndex].hero,
"large"
);
heroImageTitle = articlesData.articles[randomIndex].hero.title;
heroSlug.value = articlesData.articles[randomIndex].slug;
heroStyles.value.backgroundImage = `linear-gradient(to right, rgba(0, 0, 0, 0.7),rgba(0, 0, 0, 0.2)),url('${heroImageUrl.value}')`;
loaded.value = true;
}
} catch {
// empty
}
};
handleLoad();
</script> </script>
<style lang="scss"> <style lang="scss">

View File

@@ -24,28 +24,10 @@
:total="articlesTotal" :total="articlesTotal"
:per-page="articlesPerPage" /> :per-page="articlesPerPage" />
<div class="articles"> <div class="articles">
<router-link <SMArticleCard
:to="{ name: 'article', params: { slug: article.slug } }" v-for="(article, index) in articles"
class="article-card" :key="index"
v-for="(article, idx) in articles" :article="article" />
:key="idx">
<div
class="thumbnail"
:style="{
backgroundImage: `url(${mediaGetVariantUrl(
article.hero,
'medium'
)})`,
}"></div>
<div class="info">
{{ article.user.display_name }} -
{{ computedDate(article.publish_at) }}
</div>
<h3 class="title">{{ article.title }}</h3>
<p class="content">
{{ excerpt(article.content) }}
</p>
</router-link>
</div> </div>
</template> </template>
</SMContainer> </SMContainer>
@@ -57,13 +39,12 @@ import SMPagination from "../components/SMPagination.vue";
import { api } from "../helpers/api"; import { api } from "../helpers/api";
import { Article, ArticleCollection } from "../helpers/api.types"; import { Article, ArticleCollection } from "../helpers/api.types";
import { SMDate } from "../helpers/datetime"; import { SMDate } from "../helpers/datetime";
import { mediaGetVariantUrl } from "../helpers/media";
import SMMastHead from "../components/SMMastHead.vue"; import SMMastHead from "../components/SMMastHead.vue";
import SMInput from "../components/SMInput.vue"; import SMInput from "../components/SMInput.vue";
import SMButton from "../components/SMButton.vue"; import SMButton from "../components/SMButton.vue";
import { excerpt } from "../helpers/string";
import SMLoading from "../components/SMLoading.vue"; import SMLoading from "../components/SMLoading.vue";
import SMNoItems from "../components/SMNoItems.vue"; import SMNoItems from "../components/SMNoItems.vue";
import SMArticleCard from "../components/SMArticleCard.vue";
const message = ref(""); const message = ref("");
const pageLoading = ref(true); const pageLoading = ref(true);
@@ -127,10 +108,6 @@ const handleLoad = () => {
}); });
}; };
const computedDate = (date) => {
return new SMDate(date, { format: "yMd" }).format("d MMMM yyyy");
};
watch( watch(
() => articlesPage.value, () => articlesPage.value,
() => { () => {
@@ -147,43 +124,6 @@ handleLoad();
display: grid; display: grid;
grid-template-columns: 1fr; grid-template-columns: 1fr;
gap: 30px; gap: 30px;
.article-card {
text-decoration: none;
color: var(--card-color-text);
margin-bottom: 48px;
&:hover {
filter: none;
.thumbnail {
filter: brightness(115%);
}
}
.thumbnail {
aspect-ratio: 16 / 9;
border-radius: 7px;
background-position: center;
background-size: cover;
background-color: var(--card-color);
box-shadow: var(--base-shadow);
margin-bottom: 24px;
}
.info {
font-size: 80%;
}
.title {
margin: 16px 0;
word-break: break-word;
}
.content {
font-size: 90%;
}
}
} }
} }

View File

@@ -1,5 +1,11 @@
<template> <template>
<SMHero class="hero-offset" /> <SMHero
class="hero-offset"
:title="heroTitle"
:excerpt="heroExcerpt"
:image-url="heroImageUrl"
:image-title="heroImageTitle"
:to="heroTo" />
<SMContainer class="about align-items-center"> <SMContainer class="about align-items-center">
<template #inner> <template #inner>
@@ -23,11 +29,25 @@
</p> </p>
</template> </template>
</SMContainer> </SMContainer>
<SMContainer class="upcoming align-items-center">
<h2>Upcoming Workshops</h2>
<div class="events">
<SMEventCard
v-for="event in events"
:event="event"
:key="event.id" />
</div>
</SMContainer>
<SMContainer class="workshops align-items-center"> <SMContainer class="workshops align-items-center">
<template #inner> <template #inner>
<SMRow> <SMRow>
<SMColumn class="align-items-center flex-basis-55"> <SMColumn
<h2>Build skills while having a great time</h2> ><h2>Build skills while having a great time</h2></SMColumn
>
</SMRow>
<SMRow class="align-items-stretch">
<SMColumn
class="align-items-center justify-content-center flex-basis-55">
<p> <p>
Our online and in-person workshops are filled with Our online and in-person workshops are filled with
engaging and exciting activities that kids will love. engaging and exciting activities that kids will love.
@@ -45,6 +65,15 @@
</SMRow> </SMRow>
</template> </template>
</SMContainer> </SMContainer>
<SMContainer class="latest-articles align-items-center">
<h2>Latest Posts</h2>
<div class="articles">
<SMArticleCard
v-for="(article, index) in articles"
:key="index"
:article="article" />
</div>
</SMContainer>
<SMContainer full class="minecraft"> <SMContainer full class="minecraft">
<SMContainer> <SMContainer>
<h2>Play Minecraft with us</h2> <h2>Play Minecraft with us</h2>
@@ -110,8 +139,101 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref } from "vue";
import SMButton from "../components/SMButton.vue"; import SMButton from "../components/SMButton.vue";
import SMHero from "../components/SMHero.vue"; import SMHero from "../components/SMHero.vue";
import { api, getApiResultData } from "../helpers/api";
import { ArticleCollection, EventCollection } from "../helpers/api.types";
import { excerpt } from "../helpers/string";
import { mediaGetVariantUrl } from "../helpers/media";
import { SMDate } from "../helpers/datetime";
import SMEventCard from "../components/SMEventCard.vue";
import SMArticleCard from "../components/SMArticleCard.vue";
const articles = ref([]);
const events = ref([]);
const heroTitle = ref("");
const heroExcerpt = ref("");
const heroImageUrl = ref("");
const heroImageTitle = ref("");
const heroTo = ref({});
const computedDate = (date) => {
return new SMDate(date, { format: "yMd" }).format("d MMMM yyyy");
};
const handleLoad = async () => {
try {
await Promise.all([
api
.get({
url: "/articles",
params: {
limit: 5,
},
})
.then((articlesResult) => {
const articlesData =
getApiResultData<ArticleCollection>(articlesResult);
if (articlesData && articlesData.articles) {
const randomIndex = 1;
// Math.floor(
// Math.random() * articlesData.articles.length
// );
heroTitle.value =
articlesData.articles[randomIndex].title;
heroExcerpt.value = excerpt(
articlesData.articles[randomIndex].content,
200
);
heroImageUrl.value = mediaGetVariantUrl(
articlesData.articles[randomIndex].hero,
"large"
);
heroImageTitle.value =
articlesData.articles[randomIndex].hero.title;
heroTo.value = {
name: "article",
params: {
slug: articlesData.articles[randomIndex].slug,
},
};
articles.value = articlesData.articles.filter(
(article, index) => index !== randomIndex
);
}
}),
api
.get({
url: "/events",
params: {
limit: 4,
status: "open,soon",
sort: "start_at",
start_at:
">" +
new SMDate("now").format("yyyy-MM-dd hh:mm:ss"),
},
})
.then((eventsResult) => {
const eventsData =
getApiResultData<EventCollection>(eventsResult);
if (eventsData && eventsData.events) {
events.value = eventsData.events;
}
}),
]);
} catch {
// Handle error
}
};
handleLoad();
</script> </script>
<style lang="scss"> <style lang="scss">
@@ -139,13 +261,35 @@ import SMHero from "../components/SMHero.vue";
} }
} }
.upcoming,
.latest-articles {
h2 {
font-size: 250%;
margin-bottom: #{calc(var(--header-font-size-2))};
}
.events,
.articles {
display: grid;
grid-template-columns: 1fr;
gap: 30px;
width: 100%;
max-width: 1200px;
}
}
.workshops .container-inner { .workshops .container-inner {
margin: 64px 24px; margin: 64px 32px 32px;
padding: 0 90px 64px 90px;
background-color: var(--accent-3-color);
color: var(--accent-3-color-text);
border-radius: 24px;
max-width: 960px; max-width: 960px;
h2 { h2 {
font-size: 300%; font-size: 300%;
text-align: center; text-align: center;
color: var(--accent-3-color-text);
} }
p { p {
@@ -299,4 +443,35 @@ import SMHero from "../components/SMHero.vue";
} }
} }
} }
@media only screen and (min-width: 512px) {
.page-home {
.upcoming,
.latest-articles {
.events,
.articles {
grid-template-columns: 1fr 1fr;
}
}
}
}
@media only screen and (min-width: 832px) {
.page-home {
.upcoming,
.latest-articles {
.events,
.articles {
grid-template-columns: 1fr 1fr 1fr;
.event-card,
.article-card {
&:nth-child(4) {
display: none;
}
}
}
}
}
}
</style> </style>

View File

@@ -1,5 +1,5 @@
<template> <template>
<SMPage :error="pageError"> <SMPage :page-error="pageError">
<SMMastHead title="Workshops" /> <SMMastHead title="Workshops" />
<SMContainer class="flex-grow-1"> <SMContainer class="flex-grow-1">
<SMToolbar class="align-items-start"> <SMToolbar class="align-items-start">
@@ -31,57 +31,10 @@
v-else-if="events.length == 0" v-else-if="events.length == 0"
text="No Workshops Found" /> text="No Workshops Found" />
<div v-else class="events"> <div v-else class="events">
<router-link <SMEventCard
class="event-card"
v-for="event in events" v-for="event in events"
:key="event.id" :event="event"
:to="{ name: 'event', params: { id: event.id } }"> :key="event.id" />
<div
class="thumbnail"
:style="{
backgroundImage: `url('${mediaGetVariantUrl(
event.hero,
'medium'
)}')`,
}">
<div :class="['banner', event['bannerType']]">
{{ event["banner"] }}
</div>
<div class="date">
<div class="day">
{{ formatDateDay(event.start_at) }}
</div>
<div class="month">
{{ formatDateMonth(event.start_at) }}
</div>
</div>
</div>
<div class="content">
<h3 class="title">{{ event.title }}</h3>
<SMRow class="date" no-responsive>
<ion-icon name="calendar-outline" class="icon" />
<div class="text">{{ computedDate(event) }}</div>
</SMRow>
<SMRow class="location" no-responsive>
<ion-icon name="location-outline" class="icon" />
<div class="text">
{{ computedLocation(event) }}
</div>
</SMRow>
<SMRow class="ages" no-responsive>
<ion-icon name="body-outline" class="icon" />
<div class="text">
{{ computedAges(event.ages) }}
</div>
</SMRow>
<SMRow class="price" no-responsive>
<div class="icon">$</div>
<div class="text">
{{ computedPrice(event.price) }}
</div>
</SMRow>
</div>
</router-link>
</div> </div>
</SMContainer> </SMContainer>
</SMPage> </SMPage>
@@ -96,11 +49,11 @@ import SMPage from "../components/SMPage.vue";
import { api } from "../helpers/api"; import { api } from "../helpers/api";
import { Event, EventCollection } from "../helpers/api.types"; import { Event, EventCollection } from "../helpers/api.types";
import { SMDate } from "../helpers/datetime"; import { SMDate } from "../helpers/datetime";
import { mediaGetVariantUrl } from "../helpers/media";
import SMMastHead from "../components/SMMastHead.vue"; import SMMastHead from "../components/SMMastHead.vue";
import SMContainer from "../components/SMContainer.vue"; import SMContainer from "../components/SMContainer.vue";
import SMNoItems from "../components/SMNoItems.vue"; import SMNoItems from "../components/SMNoItems.vue";
import SMLoading from "../components/SMLoading.vue"; import SMLoading from "../components/SMLoading.vue";
import SMEventCard from "../components/SMEventCard.vue";
const pageLoading = ref(true); const pageLoading = ref(true);
let events: Event[] = reactive([]); let events: Event[] = reactive([]);
@@ -261,112 +214,6 @@ const handleFilter = async () => {
handleLoad(); handleLoad();
}; };
/**
* Return a human readable Date string.
*
* @param {Event} event The event to convert.
* @returns The converted string.
*/
const computedDate = (event: Event) => {
let str = "";
if (event.start_at.length > 0) {
if (
event.end_at.length > 0 &&
event.start_at.substring(0, event.start_at.indexOf(" ")) !=
event.end_at.substring(0, event.end_at.indexOf(" "))
) {
str = new SMDate(event.start_at, { format: "yMd" }).format(
"dd/MM/yyyy"
);
if (event.end_at.length > 0) {
str =
str +
" - " +
new SMDate(event.end_at, { format: "yMd" }).format(
"dd/MM/yyyy"
);
}
} else {
str = new SMDate(event.start_at, { format: "yMd" }).format(
"dd/MM/yyyy @ h:mm aa"
);
}
}
return str;
};
/**
* Return a the event starting month day number.
*
* @param {string} date The date to format.
* @returns The converted string.
*/
const formatDateDay = (date: string) => {
return new SMDate(date, { format: "yMd" }).format("dd");
};
/**
* Return a the event starting month name.
*
* @param {string} date The date to format.
* @returns The converted string.
*/
const formatDateMonth = (date: string) => {
return new SMDate(date, { format: "yMd" }).format("MMM");
};
/**
* Return a human readable Location string.
*
* @param {Event} event The event to convert.
* @returns The converted string.
*/
const computedLocation = (event: Event): string => {
if (event.location == "online") {
return "Online";
}
return event.address;
};
/**
* Return a human readable Ages string.
*
* @param {string} ages The string to convert.
* @returns The converted string.
*/
const computedAges = (ages: string): string => {
const trimmed = ages.trim();
const regex = /^(\d+)(\s*\+?\s*|\s*-\s*\d+\s*)?$/;
if (trimmed.length === 0) {
return "All ages";
}
if (regex.test(trimmed)) {
return `Ages ${trimmed}`;
}
return ages;
};
/**
* Return a human readable Price string.
*
* @param {string} price The string to convert.
* @returns The converted string.
*/
const computedPrice = (price: string): string => {
const trimmed = parseInt(price.trim());
if (isNaN(trimmed) || trimmed == 0) {
return "Free";
}
return trimmed.toString();
};
watch( watch(
() => postsPage.value, () => postsPage.value,
() => { () => {
@@ -384,105 +231,6 @@ handleLoad();
grid-template-columns: 1fr; grid-template-columns: 1fr;
gap: 30px; gap: 30px;
width: 100%; width: 100%;
.event-card {
background-color: var(--base-color-light);
box-shadow: 0 5px 10px -3px rgba(0, 0, 0, 0.25);
border-radius: 8px;
text-decoration: none;
color: var(--base-color-text);
position: relative;
overflow: hidden;
.thumbnail {
width: 100%;
aspect-ratio: 16 / 9;
background-position: center;
background-repeat: no-repeat;
background-size: cover;
border-radius: 8px 8px 0 0;
.banner {
position: absolute;
background-color: var(--banner-green-color);
font-size: 70%;
font-weight: 700;
color: var(--banner-green-color-text);
padding: 6px 18px;
text-align: center;
top: 10px;
right: 10px;
text-transform: uppercase;
&.expired {
background-color: var(--banner-purple-color);
color: var(--banner-purple-color-text);
}
&.danger {
background-color: var(--banner-red-color);
color: var(--banner-red-color-text);
}
&.warning {
background-color: var(--banner-yellow-color);
color: var(--banner-yellow-color-text);
}
}
.date {
position: absolute;
top: 10px;
left: 10px;
background-color: var(--base-color);
box-shadow: var(--base-shadow);
padding: 8px 12px;
text-align: center;
border-radius: 2px;
.day {
font-weight: 700;
padding: 1px;
}
.month {
font-size: 65%;
text-transform: uppercase;
}
}
}
.content {
padding: 16px;
}
.title {
margin: 0 0 16px 0;
font-size: 100%;
word-break: break-all;
}
.row {
display: flex;
margin-bottom: 8px;
font-size: 80%;
.icon {
width: 20px;
text-align: center;
margin-right: 8px;
}
}
&:hover {
cursor: pointer;
filter: none;
.image {
filter: brightness(115%);
}
}
}
} }
} }