diff --git a/config/filters/media.js b/config/filters/media.js index ff222f9b..87cc61a5 100644 --- a/config/filters/media.js +++ b/config/filters/media.js @@ -1,57 +1,4 @@ -import { DateTime } from 'luxon' -import { shuffleArray } from '../utilities/index.js' - export default { - featuredWatching: (watching, count) => { - const data = [...watching] - return shuffleArray(data).slice(0, count) - }, - normalizeMedia: (media, limit) => { - const mediaData = limit ? media.slice(0, limit) : media - return mediaData.map((item) => { - let normalized = { - title: item['title'], - image: item['image'], - url: item['url'], - type: item['type'] - } - - switch (item['type']) { - case 'artist': - normalized.alt = `${item['plays']} plays of ${item['title']}` - normalized.subtext = `${item['plays']} plays` - break - case 'album': - normalized.alt = `${item['title']} by ${item['artist']}` - normalized.subtext = `${item['artist']}` - break - case 'album-release': - normalized.alt = `${item['title']} by ${item['artist']['name']}` - normalized.subtext = `${item['artist']['name']} / ${item['release_date_formatted']}` - break - case 'movie': - normalized.alt = item['title'] - normalized.rating = item['rating'] - normalized.favorite = item['favorite'] - normalized.subtext = item.rating ? `${item['rating']} (${item['year']})` : `(${item['year']})` - break - case 'book': - normalized.title = `${item['title']} by ${item['author']}` - if (item['rating']) { - normalized.rating = item['rating'] - normalized.subtext = item['rating'] - } - break - case 'tv': - normalized.alt = item['formatted_episode'] - normalized.subtext = item['formatted_episode'] - break - } - - return normalized - }) - }, - calculatePlayPercentage: (plays, mostPlayed) => `${plays/mostPlayed * 100}%`, bookStatus: (books, status) => books.filter(book => book['status'] === status), bookFavorites: (books) => books.filter(book => book.favorite === true), bookYearLinks: (years) => years.sort((a, b) => b.value - a.value).map((year, index) => { @@ -71,7 +18,9 @@ export default { if (dataSlice.length === 0) return null if (dataSlice.length === 1) { const item = dataSlice[0] - if (type === 'genre' || type === 'artist') { + if (type === 'genre') { + return `${item['genre_name']}` + } else if (type === 'artist') { return `${item['name']}` } else if (type === 'book') { return `${item['title']}` @@ -79,7 +28,9 @@ export default { } const allButLast = dataSlice.slice(0, -1).map(item => { - if (type === 'genre' || type === 'artist') { + if (type === 'genre') { + return `${item['genre_name']}` + } else if (type === 'artist') { return `${item['name']}` } else if (type === 'book') { return `${item['title']}` @@ -89,7 +40,9 @@ export default { let last const lastItem = dataSlice[dataSlice.length - 1] - if (type === 'genre' || type === 'artist') { + if (type === 'genre') { + last = `${lastItem['genre_name']}` + } else if (type === 'artist') { last = `${lastItem['name']}` } else if (type === 'book') { last = `${lastItem['title']}` diff --git a/package-lock.json b/package-lock.json index 4c4c2381..aa0b65b0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "coryd.dev", - "version": "1.0.3", + "version": "1.0.4", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "coryd.dev", - "version": "1.0.3", + "version": "1.0.4", "license": "MIT", "dependencies": { "@cdransf/api-text": "^1.5.0", diff --git a/package.json b/package.json index f0bfd908..ac680863 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "coryd.dev", - "version": "1.0.3", + "version": "1.0.4", "description": "The source for my personal site. Built using 11ty (and other tools).", "type": "module", "scripts": { diff --git a/src/assets/styles/pages/watching.css b/src/assets/styles/pages/watching.css index 90644e8a..9f4c8091 100644 --- a/src/assets/styles/pages/watching.css +++ b/src/assets/styles/pages/watching.css @@ -3,69 +3,71 @@ a:active > .watching.hero::after { border-color: var(--accent-color-hover); } -.icon-link + .watching.grid { +.watching.hero { + position: relative; + display: flex; + aspect-ratio: var(--aspect-ratio-banner); + + & img { + aspect-ratio: var(--aspect-ratio-banner); + border-radius: var(--border-radius-slight); + height: auto; + width: 100%; + } + + & .meta-text { + color: white; + position: absolute; + left: var(--spacing-sm); + bottom: var(--spacing-sm); + z-index: 2; + display: flex; + flex-direction: column; + + & .header { + font-weight: var(--font-weight-bold); + } + + & .subheader { + font-size: var(--font-size-sm); + display: inline-flex; + gap: var(--spacing-xs); + } + + & .header, + & .subheader { + line-height: var(--line-height-md); + text-shadow: var(--text-shadow-default); + } + } + + &::after { + position: absolute; + z-index: 1; + content: ''; + top: 0; + left: 0; + box-shadow: var(--box-shadow-media); + width: 100%; + height: 100%; + border: var(--border-default); + border-radius: var(--border-radius-slight); + transition: border-color var(--transition-duration-default) var(--transition-ease-in-out); + } +} + +.icon-link + .poster.grid { margin-top: var(--spacing-base); } -.watching { +.poster { & img { border-radius: var(--border-radius-slight); width: 100%; height: auto; } - &.hero { - position: relative; - display: flex; - aspect-ratio: var(--aspect-ratio-banner); - - & img { - aspect-ratio: var(--aspect-ratio-banner); - border-radius: var(--border-radius-slight); - } - - & .meta-text { - color: white; - position: absolute; - left: var(--spacing-sm); - bottom: var(--spacing-sm); - z-index: 2; - display: flex; - flex-direction: column; - - & .header { - font-weight: var(--font-weight-bold); - } - - & .subheader { - font-size: var(--font-size-sm); - display: inline-flex; - gap: var(--spacing-xs); - } - - & .header, - & .subheader { - line-height: var(--line-height-md); - text-shadow: var(--text-shadow-default); - } - } - - &::after { - position: absolute; - z-index: 1; - content: ''; - top: 0; - left: 0; - box-shadow: var(--box-shadow-media); - width: 100%; - height: 100%; - border: var(--border-default); - border-radius: var(--border-radius-slight); - transition: border-color var(--transition-duration-default) var(--transition-ease-in-out); - } - } - - &.grid { + &.media-grid { display: grid; gap: var(--spacing-sm); grid-template-columns: repeat(2, minmax(0, 1fr)); diff --git a/src/data/music.js b/src/data/music.js index b4902222..ae9c94ed 100644 --- a/src/data/music.js +++ b/src/data/music.js @@ -13,7 +13,6 @@ const fetchDataFromView = async (viewName) => { const { data, error } = await supabase .from(viewName) .select('*') - .order('listened_at', { ascending: false }) .range(rangeStart, rangeStart + PAGE_SIZE - 1) if (error) { @@ -23,7 +22,7 @@ const fetchDataFromView = async (viewName) => { if (data.length === 0) break - rows = rows.concat(data) + rows = [...rows, ...data] if (data.length < PAGE_SIZE) break rangeStart += PAGE_SIZE @@ -32,113 +31,70 @@ const fetchDataFromView = async (viewName) => { return rows } -const aggregateData = (data, groupByField, groupByType) => { - const aggregation = {} - - data.forEach(item => { - const key = item[groupByField] - if (!aggregation[key]) { - let imageField = '' - - switch (groupByType) { - case 'artist': - imageField = item['artist_art'] - break - case 'album': - imageField = item['album_art'] - break - case 'track': - imageField = item['album_art'] - break - default: - imageField = '' - } - - aggregation[key] = { - title: item[groupByField], - plays: 0, - url: item['artist_url'], - image: imageField, - genre: item['artist_genres'], - type: groupByType - } - - if (groupByType === 'track' || groupByType === 'album') aggregation[key]['artist'] = item['artist_name'] - } - - aggregation[key].plays++ - }) - - return Object.values(aggregation).sort((a, b) => b['plays'] - a['plays']).map((item, index) => ({ ...item, rank: index + 1 })) -} - -const buildRecents = (data) => { - return data.map(listen => ({ - title: listen['track_name'], - artist: listen['artist_name'], - url: listen['artist_url'], - timestamp: listen['listened_at'], - image: listen['album_art'], - type: 'track' - })).sort((a, b) => new Date(b.timestamp) - new Date(a.timestamp)) -} - -const aggregateGenres = (data) => { - const genreAggregation = {} - - data.forEach(item => { - const genre = item['genre_name'] || '' - const genreUrl = item['genre_url'] || '' - - if (!genreAggregation[genre]) { - genreAggregation[genre] = { - name: genre, - url: genreUrl, - plays: 0, - type: 'genre' - } - } - - genreAggregation[genre]['plays']++ - }) - - return Object.values(genreAggregation).sort((a, b) => b['plays'] - a['plays']) -} - -export default async function () { +export default async function fetchMusicData() { try { - const [recentTracks, monthTracks, threeMonthTracks] = await Promise.all([ + const [ + recentTracks, + weekTracks, + weekArtists, + weekAlbums, + weekGenres, + monthTracks, + monthArtists, + monthAlbums, + monthGenres, + threeMonthTracks, + threeMonthArtists, + threeMonthAlbums, + threeMonthGenres, + ] = await Promise.all([ fetchDataFromView('recent_tracks'), + fetchDataFromView('week_tracks'), + fetchDataFromView('week_artists'), + fetchDataFromView('week_albums'), + fetchDataFromView('week_genres'), fetchDataFromView('month_tracks'), - fetchDataFromView('three_month_tracks') + fetchDataFromView('month_artists'), + fetchDataFromView('month_albums'), + fetchDataFromView('month_genres'), + fetchDataFromView('three_month_tracks'), + fetchDataFromView('three_month_artists'), + fetchDataFromView('three_month_albums'), + fetchDataFromView('three_month_genres'), ]) return { - recent: buildRecents(recentTracks), + recent: recentTracks, week: { - artists: aggregateData(recentTracks, 'artist_name', 'artist'), - albums: aggregateData(recentTracks, 'album_name', 'album'), - tracks: aggregateData(recentTracks, 'track_name', 'track'), - genres: aggregateGenres(recentTracks), - totalTracks: recentTracks.length.toLocaleString('en-US') + tracks: weekTracks, + artists: weekArtists, + albums: weekAlbums, + genres: weekGenres, + totalTracks: weekTracks + .reduce((acc, track) => acc + track.plays, 0) + .toLocaleString('en-US'), }, month: { - artists: aggregateData(monthTracks, 'artist_name', 'artist'), - albums: aggregateData(monthTracks, 'album_name', 'album'), - tracks: aggregateData(monthTracks, 'track_name', 'track'), - genres: aggregateGenres(monthTracks), - totalTracks: monthTracks.length.toLocaleString('en-US') + tracks: monthTracks, + artists: monthArtists, + albums: monthAlbums, + genres: monthGenres, + totalTracks: monthTracks + .reduce((acc, track) => acc + track.plays, 0) + .toLocaleString('en-US'), }, threeMonth: { - artists: aggregateData(threeMonthTracks, 'artist_name', 'artist'), - albums: aggregateData(threeMonthTracks, 'album_name', 'album'), - tracks: aggregateData(threeMonthTracks, 'track_name', 'track'), - genres: aggregateGenres(threeMonthTracks), - totalTracks: threeMonthTracks.length.toLocaleString('en-US') - } + tracks: threeMonthTracks, + artists: threeMonthArtists, + albums: threeMonthAlbums, + genres: threeMonthGenres, + totalTracks: threeMonthTracks + .reduce((acc, track) => acc + track.plays, 0) + .toLocaleString('en-US'), + }, } } catch (error) { - console.error('Error in fetching and processing music data:', error) + console.error('Error fetching and processing music data:', error) return {} } } \ No newline at end of file diff --git a/src/data/tv.js b/src/data/tv.js index 4cf0135c..06a9e4e1 100644 --- a/src/data/tv.js +++ b/src/data/tv.js @@ -40,6 +40,7 @@ export default async function () { image: show['episode']['image'], backdrop: show['episode']['backdrop'], last_watched_at: show['episode']['last_watched_at'], + grid: show['grid'], type: 'tv' })) diff --git a/src/includes/partials/media/grid.liquid b/src/includes/partials/media/grid.liquid index 8b36eeb6..4ce1e9c1 100644 --- a/src/includes/partials/media/grid.liquid +++ b/src/includes/partials/media/grid.liquid @@ -1,44 +1,51 @@ -{%- assign hidePagination = count or data.pages.size <= 1 -%} -{%- assign media = data.items | default: data | normalizeMedia: count -%} -
I finished {{ bookData.size }} books in {{ year.value }}.{%- if favoriteBooks %} Among my favorites were {{ favoriteBooks }}.{%- endif -%}
{% endif %}You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
You can also take a look at the concerts I've been to.
These are my favorite movies. There are many like them, but these are mine.
These are my favorite shows. There are many like them, but these are mine.
Here's all of the TV and movies I've been watching presented in what is (hopefully) an organized fashion.
@@ -18,27 +18,27 @@ schema: watching Recent movies -{% render "partials/media/grid.liquid", data:movies.recentlyWatched, shape: "vertical", count: 6 %} +{% render "partials/media/grid.liquid", data:movies.recentlyWatched, shape:"vertical", count: 6 %}These are the movies I've watched recently. There are many like them, but these are mine. (Or well, all the movies I've watched — they're ordered latest watched, descending, hence the recent part).
These are the shows I've watched recently. There are many like them, but these are mine. (Or well, all the movies I've watched — they're ordered latest watched, descending, hence the recent part).