2025-03-28 13:53:35 +01:00
|
|
|
<app>
|
|
|
|
|
<nav>
|
|
|
|
|
<a href="#"><i class="fa-solid fa-house"></i></a>
|
2025-03-28 16:27:07 +01:00
|
|
|
<a if={authUser} href="#/favoris/">Favoris</a>
|
|
|
|
|
<a if={!authUser} href="#/login/"><i class="fa-solid fa-user"></i></a>
|
|
|
|
|
<a if={authUser} href="#" onclick={() => Logout()}><i class="fa-solid fa-right-from-bracket"></i></a>
|
2025-03-28 13:53:35 +01:00
|
|
|
</nav>
|
|
|
|
|
|
|
|
|
|
<router base={base}>
|
|
|
|
|
<route path="(#)?">
|
|
|
|
|
<form onsubmit="{search}">
|
|
|
|
|
<select name="type">
|
|
|
|
|
<option value="release">Release</option>
|
|
|
|
|
<option value="master">Master</option>
|
|
|
|
|
<option value="artist">Artist</option>
|
|
|
|
|
</select>
|
|
|
|
|
<input type="text" name="query" placeholder="Rechercher..." />
|
|
|
|
|
<button type="submit">Rechercher</button>
|
|
|
|
|
</form>
|
|
|
|
|
|
2025-03-28 16:27:07 +01:00
|
|
|
<!-- Pagination -->
|
|
|
|
|
<div class="pagination" if={searchs.pagination && searchs.pagination.pages > 1}>
|
|
|
|
|
<button
|
|
|
|
|
onclick={() => this.changePage(searchs.pagination.page - 1)}
|
|
|
|
|
disabled={searchs.pagination.page === 1}>
|
|
|
|
|
◀ Précédent
|
|
|
|
|
</button>
|
|
|
|
|
|
|
|
|
|
<span>Page {searchs.pagination.page} / {searchs.pagination.pages}</span>
|
|
|
|
|
|
|
|
|
|
<button
|
|
|
|
|
onclick={() => this.changePage(searchs.pagination.page + 1)}
|
|
|
|
|
disabled={searchs.pagination.page === searchs.pagination.pages}>
|
|
|
|
|
Suivant ▶
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
|
2025-03-28 13:53:35 +01:00
|
|
|
<div if="{searchs.items.length > 0}">
|
2025-03-29 16:42:22 +01:00
|
|
|
<h3>Résultats ({searchs.pagination.items} résultats trouvés)</h3>
|
2025-03-28 13:53:35 +01:00
|
|
|
<div class="results-grid">
|
|
|
|
|
<div class="card" each="{item in searchs.items}">
|
2025-03-29 16:38:00 +01:00
|
|
|
<!-- Afficher l'étoile avec la musique immédiatement -->
|
2025-04-02 13:18:06 +02:00
|
|
|
<a if={authUser && this.searchs.type !== "artist"} onclick={() => item.isFavorite ? this.removeFavoris(item.id, item.type) : this.addFavoris(item.id, item.type)}>
|
2025-03-29 16:42:22 +01:00
|
|
|
<button><i class={item.isFavorite ? "fa-solid fa-star" : "fa-regular fa-star"}></i></button>
|
2025-03-29 16:38:00 +01:00
|
|
|
</a>
|
2025-03-31 13:15:47 +02:00
|
|
|
<a href={ "#/release-details/" + item.type + "/" + item.id}>
|
2025-03-28 13:53:35 +01:00
|
|
|
{item.title}
|
2025-03-29 16:38:00 +01:00
|
|
|
<img src="{item.cover_image}" alt="cover" />
|
2025-03-28 13:53:35 +01:00
|
|
|
<div if="{item.type !=='artist' } ">
|
|
|
|
|
<p>{item.year}</p>
|
2025-03-29 16:38:00 +01:00
|
|
|
<footer>{item.community.want}<i class="fa-solid fa-check"></i> {item.community.have}<i class="fa-regular fa-heart"></i></footer>
|
2025-03-28 13:53:35 +01:00
|
|
|
</div>
|
|
|
|
|
</a>
|
2025-03-31 13:15:47 +02:00
|
|
|
<a if={this.searchs.type === "release"} href={"#/release-details/master/" + item.master_id}><button>master</button></a>
|
2025-04-03 14:38:06 +02:00
|
|
|
<a if={this.searchs.type === "master"} href="#"><button onclick={() => this.MainReleaseDetails(item)}>release</button></a>
|
2025-03-28 13:53:35 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</route>
|
|
|
|
|
|
2025-03-31 13:15:47 +02:00
|
|
|
<route path="#/release-details/:type/:id">
|
|
|
|
|
<release-details Type={route.params.type} Id={route.params.id}></release-details>
|
2025-03-28 13:53:35 +01:00
|
|
|
</route>
|
|
|
|
|
<route path="#/favoris">
|
|
|
|
|
<favorites></favorites>
|
|
|
|
|
</route>
|
|
|
|
|
<route path="#/login">
|
|
|
|
|
<login></login>
|
|
|
|
|
</route>
|
|
|
|
|
<route path="#/register">
|
|
|
|
|
<register></register>
|
|
|
|
|
</route>
|
|
|
|
|
</router>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<script>
|
|
|
|
|
export default {
|
2025-03-28 16:28:26 +01:00
|
|
|
base: 'https://dwarves.iut-fbleau.fr/~felix-vi/SAE_riot/',
|
2025-03-28 13:53:35 +01:00
|
|
|
route: 'search',
|
|
|
|
|
authUser: null,
|
|
|
|
|
searchs: {
|
|
|
|
|
items: [],
|
|
|
|
|
pagination: [],
|
2025-03-31 13:15:47 +02:00
|
|
|
type: "release",
|
|
|
|
|
query: ""
|
2025-03-28 13:53:35 +01:00
|
|
|
},
|
|
|
|
|
|
2025-03-28 16:27:07 +01:00
|
|
|
async search(e, page = 1) {
|
|
|
|
|
if (e) e.preventDefault();
|
|
|
|
|
|
|
|
|
|
const type = e?.target?.type?.value || this.searchs.type;
|
|
|
|
|
const query = e?.target?.query?.value || this.searchs.query;
|
|
|
|
|
|
|
|
|
|
const result = await window.discogsearch(query, type, page);
|
|
|
|
|
|
2025-04-02 13:18:06 +02:00
|
|
|
const favoritePromises = result.results.map(item => window.isFavorite(item.id, item.type));
|
2025-03-29 16:38:00 +01:00
|
|
|
const favoritesResults = await Promise.all(favoritePromises);
|
|
|
|
|
|
|
|
|
|
const enrichedItems = result.results.map((item, index) => ({
|
|
|
|
|
...item,
|
|
|
|
|
isFavorite: favoritesResults[index]
|
2025-03-29 16:08:04 +01:00
|
|
|
}));
|
2025-03-28 16:27:07 +01:00
|
|
|
|
2025-03-29 16:38:00 +01:00
|
|
|
this.searchs.items = enrichedItems;
|
2025-03-28 16:27:07 +01:00
|
|
|
this.searchs.pagination = result.pagination;
|
|
|
|
|
this.searchs.query = query;
|
|
|
|
|
this.searchs.type = type;
|
2025-04-03 14:38:06 +02:00
|
|
|
console.log()
|
2025-03-28 16:27:07 +01:00
|
|
|
this.update();
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
changePage(page) {
|
|
|
|
|
this.search(null, page);
|
2025-03-29 15:53:23 +01:00
|
|
|
this.update();
|
2025-03-28 13:53:35 +01:00
|
|
|
},
|
|
|
|
|
|
2025-03-28 16:27:07 +01:00
|
|
|
async onMounted() {
|
|
|
|
|
observeAuthState(async user => {
|
|
|
|
|
this.authUser = user;
|
|
|
|
|
this.update();
|
2025-03-28 13:53:35 +01:00
|
|
|
});
|
2025-03-31 13:15:47 +02:00
|
|
|
this.search(null, 1);
|
|
|
|
|
this.update();
|
2025-03-28 13:53:35 +01:00
|
|
|
},
|
|
|
|
|
|
2025-04-02 13:18:06 +02:00
|
|
|
addFavoris(id, type) {
|
|
|
|
|
window.favorite(id, type);
|
2025-03-29 15:53:23 +01:00
|
|
|
this.searchs.items = this.searchs.items.map(item =>
|
|
|
|
|
item.id === id ? { ...item, isFavorite: true } : item
|
|
|
|
|
);
|
|
|
|
|
this.update();
|
|
|
|
|
},
|
|
|
|
|
|
2025-04-02 13:18:06 +02:00
|
|
|
async removeFavoris(id, type) {
|
|
|
|
|
await window.removeFavorite(id, type);
|
2025-03-29 15:53:23 +01:00
|
|
|
this.searchs.items = this.searchs.items.map(item =>
|
|
|
|
|
item.id === id ? { ...item, isFavorite: false } : item
|
|
|
|
|
);
|
|
|
|
|
this.update();
|
2025-03-28 16:27:07 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
|
|
Logout(){
|
|
|
|
|
window.logout()
|
2025-03-29 15:53:23 +01:00
|
|
|
this.update();
|
2025-04-03 14:38:06 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
|
|
async MainReleaseDetails(item) {
|
|
|
|
|
const res = await fetch(item.resource_url);
|
|
|
|
|
const json = await res.json();
|
|
|
|
|
|
|
|
|
|
const mainUrl = json.main_release_url;
|
|
|
|
|
|
|
|
|
|
const res2 = await fetch(mainUrl);
|
|
|
|
|
const releaseData = await res2.json();
|
|
|
|
|
|
|
|
|
|
window.location.hash = `#/release-details/release/${releaseData.id}`;
|
2025-03-28 13:53:35 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style>
|
|
|
|
|
#view {
|
|
|
|
|
padding: 1rem;
|
|
|
|
|
max-width: 1200px;
|
|
|
|
|
margin: auto;
|
|
|
|
|
}
|
2025-03-31 13:15:47 +02:00
|
|
|
|
2025-03-28 13:53:35 +01:00
|
|
|
form {
|
|
|
|
|
display: flex;
|
|
|
|
|
gap: 10px;
|
|
|
|
|
margin-bottom: 20px;
|
2025-03-31 13:15:47 +02:00
|
|
|
flex-wrap: wrap;
|
2025-03-28 13:53:35 +01:00
|
|
|
}
|
2025-03-28 16:27:07 +01:00
|
|
|
|
2025-03-31 13:15:47 +02:00
|
|
|
form select,
|
|
|
|
|
form input {
|
|
|
|
|
padding: 0.6rem;
|
|
|
|
|
border: 1px solid #ccc;
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
font-size: 1rem;
|
2025-03-28 13:53:35 +01:00
|
|
|
}
|
2025-03-31 13:15:47 +02:00
|
|
|
|
|
|
|
|
form button {
|
2025-03-28 13:53:35 +01:00
|
|
|
background-color: #1976d2;
|
|
|
|
|
color: white;
|
2025-03-31 13:15:47 +02:00
|
|
|
padding: 0.6rem 1.2rem;
|
|
|
|
|
border: none;
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
transition: background-color 0.3s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
form button:hover {
|
|
|
|
|
background-color: #125aa3;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.results-grid {
|
|
|
|
|
display: grid;
|
|
|
|
|
grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
|
|
|
|
|
gap: 1rem;
|
|
|
|
|
padding: 1rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card {
|
|
|
|
|
background: white;
|
|
|
|
|
border-radius: 10px;
|
|
|
|
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.07);
|
|
|
|
|
padding: 1rem;
|
|
|
|
|
text-align: center;
|
|
|
|
|
transition: transform 0.2s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card:hover {
|
|
|
|
|
transform: translateY(-4px);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card img {
|
|
|
|
|
width: 100%;
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
margin-bottom: 0.5rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card h4 {
|
|
|
|
|
font-size: 1.1rem;
|
|
|
|
|
margin: 0.5rem 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
footer {
|
2025-03-28 13:53:35 +01:00
|
|
|
display: flex;
|
2025-03-31 13:15:47 +02:00
|
|
|
justify-content: space-around;
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
color: #666;
|
|
|
|
|
margin-top: 0.5rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card a i.fa-star {
|
|
|
|
|
color: #fbc02d;
|
|
|
|
|
font-size: 1.3rem;
|
|
|
|
|
transition: transform 0.2s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.card a i.fa-star:hover {
|
|
|
|
|
transform: scale(1.2);
|
2025-03-28 13:53:35 +01:00
|
|
|
}
|
2025-03-28 16:27:07 +01:00
|
|
|
|
|
|
|
|
.pagination {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 1rem;
|
|
|
|
|
margin-bottom: 2rem;
|
|
|
|
|
}
|
2025-03-31 13:15:47 +02:00
|
|
|
|
|
|
|
|
.pagination button {
|
|
|
|
|
background-color: #1976d2;
|
|
|
|
|
color: white;
|
|
|
|
|
padding: 0.5rem 1rem;
|
|
|
|
|
border: none;
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
transition: background-color 0.3s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.pagination button:hover {
|
|
|
|
|
background-color: #125aa3;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nav {
|
|
|
|
|
background-color: #1976d2;
|
|
|
|
|
padding: 1rem;
|
|
|
|
|
color: white;
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
align-items: center;
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 13:53:35 +01:00
|
|
|
nav a {
|
|
|
|
|
margin: 0 10px;
|
|
|
|
|
color: white;
|
|
|
|
|
text-decoration: none;
|
|
|
|
|
}
|
2025-03-31 13:15:47 +02:00
|
|
|
|
2025-03-28 13:53:35 +01:00
|
|
|
</style>
|
|
|
|
|
</app>
|