Started refactoring, added prettier and checks and reformatted project, added cspell and checks and custom project words, beginning of robotic oceanographic surface sampler content
This commit is contained in:
145
src/components/Media/CustomCarousel/CustomCarousel.astro
Normal file
145
src/components/Media/CustomCarousel/CustomCarousel.astro
Normal file
@@ -0,0 +1,145 @@
|
||||
---
|
||||
import { Image } from "astro:assets";
|
||||
|
||||
import type { carouselGroup } from "@interfaces/image-carousel.ts";
|
||||
|
||||
const groupToShow: carouselGroup = Astro.props.carouselGroup;
|
||||
---
|
||||
|
||||
<custom-carousel
|
||||
class="relative flex w-full flex-col"
|
||||
data-custom-carousel={groupToShow.animation}
|
||||
data-custom-carousel-interval={groupToShow.interval}
|
||||
>
|
||||
<!-- Modal for fullscreen viewing -->
|
||||
<div
|
||||
tabindex="-1"
|
||||
aria-hidden="true"
|
||||
class="fixed inset-0 z-100 hidden w-full items-center justify-center"
|
||||
data-custom-carousel-modal
|
||||
>
|
||||
<div class="relative h-full w-full p-4">
|
||||
<!-- Modal content -->
|
||||
<div
|
||||
class="border-caperren-green relative h-full max-h-screen max-w-screen rounded-lg border-2 bg-black shadow-sm"
|
||||
>
|
||||
<!-- Modal header -->
|
||||
<div class="flex items-center justify-between rounded-t p-1">
|
||||
<button
|
||||
type="button"
|
||||
class="text-caperren-green ring-caperren-green-dark hover:ring-caperren-green-light hover:text-caperren-green-light z-100 ms-auto inline-flex size-8 items-center justify-center rounded-lg bg-transparent text-sm ring-2"
|
||||
data-custom-carousel-modal-hide
|
||||
>
|
||||
<svg
|
||||
class="h-3 w-3"
|
||||
aria-hidden="true"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
viewBox="0 0 14 14"
|
||||
>
|
||||
<path
|
||||
stroke="currentColor"
|
||||
stroke-linecap="round"
|
||||
stroke-linejoin="round"
|
||||
stroke-width="2"
|
||||
d="m1 1 6 6m0 0 6 6M7 7l6-6M7 7l-6 6"></path>
|
||||
</svg>
|
||||
<span class="sr-only">Close modal</span>
|
||||
</button>
|
||||
</div>
|
||||
<!-- Modal body -->
|
||||
<div
|
||||
class="flex h-full w-full items-center justify-center overflow-hidden"
|
||||
data-custom-carousel-modal-image
|
||||
>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Carousel wrapper -->
|
||||
<div class="relative h-56 w-full overflow-hidden md:h-120">
|
||||
{
|
||||
groupToShow.images.map((image, index) => (
|
||||
<div
|
||||
class="hidden duration-1500 ease-in-out"
|
||||
data-custom-carousel-item={index}
|
||||
>
|
||||
<Image
|
||||
src={image}
|
||||
class="absolute inset-0 m-auto h-full max-h-full w-auto max-w-full object-contain"
|
||||
alt="..."
|
||||
layout="constrained"
|
||||
loading="eager"
|
||||
/>
|
||||
</div>
|
||||
))
|
||||
}
|
||||
</div>
|
||||
|
||||
<!-- Slider indicators -->
|
||||
{
|
||||
groupToShow.images.length > 1 && (
|
||||
<div>
|
||||
<div class="absolute bottom-2 left-1/2 z-30 flex -translate-x-1/2 space-x-3 rounded-full">
|
||||
{groupToShow.images.map((_, index) => (
|
||||
<button
|
||||
type="button"
|
||||
class="hover:bg-caperren-green-light h-3 w-3 rounded-full bg-black"
|
||||
aria-current={index ? "false" : "true"}
|
||||
aria-label={index.toString()}
|
||||
data-custom-carousel-slide-to={index}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
<button
|
||||
type="button"
|
||||
class="group absolute start-0 top-0 z-30 flex h-full cursor-pointer items-center justify-center px-4 focus:outline-none"
|
||||
data-custom-carousel-prev
|
||||
>
|
||||
<span class="ring-caperren-green/25 inline-flex h-10 w-10 items-center justify-center rounded-full bg-black/25 ring-2 group-hover:bg-black/75">
|
||||
<svg
|
||||
class="text-caperren-green group-hover:text-caperren-green-light h-4 w-4"
|
||||
aria-hidden="true"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
viewBox="0 0 6 10"
|
||||
>
|
||||
<path
|
||||
stroke="currentColor"
|
||||
stroke-linecap="round"
|
||||
stroke-linejoin="round"
|
||||
stroke-width="2"
|
||||
d="M5 1 1 5l4 4"
|
||||
/>
|
||||
</svg>
|
||||
<span class="hidden">Previous</span>
|
||||
</span>
|
||||
</button>
|
||||
<button
|
||||
type="button"
|
||||
class="group absolute end-0 top-0 z-30 flex h-full cursor-pointer items-center justify-center px-4 focus:outline-none"
|
||||
data-custom-carousel-next
|
||||
>
|
||||
<span class="ring-caperren-green/25 inline-flex h-10 w-10 items-center justify-center rounded-full bg-black/25 ring-2 group-hover:bg-black/75">
|
||||
<svg
|
||||
class="text-caperren-green group-hover:text-caperren-green-light h-4 w-4"
|
||||
aria-hidden="true"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
viewBox="0 0 6 10"
|
||||
>
|
||||
<path
|
||||
stroke="currentColor"
|
||||
stroke-linecap="round"
|
||||
stroke-linejoin="round"
|
||||
stroke-width="2"
|
||||
d="m1 9 4-4-4-4"
|
||||
/>
|
||||
</svg>
|
||||
<span class="hidden">Next</span>
|
||||
</span>
|
||||
</button>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
</custom-carousel>
|
||||
|
||||
<script src="./custom-carousel.ts"></script>
|
||||
111
src/components/Media/CustomCarousel/custom-carousel.ts
Normal file
111
src/components/Media/CustomCarousel/custom-carousel.ts
Normal file
@@ -0,0 +1,111 @@
|
||||
import {
|
||||
Carousel,
|
||||
type CarouselItem,
|
||||
type CarouselInterface,
|
||||
type CarouselOptions,
|
||||
type IndicatorItem,
|
||||
Modal,
|
||||
type ModalInterface,
|
||||
} from "flowbite";
|
||||
|
||||
class CustomCarousel extends HTMLElement {
|
||||
_slide: boolean;
|
||||
_items: CarouselItem[];
|
||||
_options: CarouselOptions;
|
||||
_carousel: CarouselInterface;
|
||||
|
||||
_modalEl: HTMLElement;
|
||||
_modal: ModalInterface;
|
||||
|
||||
constructor() {
|
||||
super();
|
||||
|
||||
this._slide = this.getAttribute("data-custom-carousel") === "slide";
|
||||
this._items = this._getItems();
|
||||
this._options = this._getOptions();
|
||||
this._carousel = new Carousel(this, this._items, this._options);
|
||||
|
||||
if (this._slide && this._items.length > 0) this._carousel.cycle();
|
||||
|
||||
this._modalEl =
|
||||
(this.querySelector("[data-custom-carousel-modal]") as HTMLElement) ||
|
||||
undefined;
|
||||
this._modal = new Modal(this._modalEl);
|
||||
|
||||
window.addEventListener("load", this._attachHandlers);
|
||||
}
|
||||
|
||||
_getItems = (): CarouselItem[] => {
|
||||
let customItems =
|
||||
this.querySelectorAll("[data-custom-carousel-item]") || [];
|
||||
|
||||
return Array.from(customItems).map((item): CarouselItem => {
|
||||
return {
|
||||
el: item as HTMLElement,
|
||||
position: Number(item.getAttribute("data-custom-carousel-item")),
|
||||
};
|
||||
});
|
||||
};
|
||||
|
||||
_getOptions = (): CarouselOptions => {
|
||||
let customIndicators =
|
||||
this.querySelectorAll("[data-custom-carousel-slide-to]") || [];
|
||||
|
||||
return {
|
||||
defaultPosition: 0,
|
||||
interval: this.dataset.customCarouselInterval
|
||||
? Number(this.dataset.customCarouselInterval)
|
||||
: 8000,
|
||||
|
||||
indicators: {
|
||||
activeClasses: "border-2 border-caperren-green bg-black",
|
||||
inactiveClasses: "bg-caperren-green/40 hover:bg-caperren-green-light",
|
||||
items: Array.from(customIndicators).map(
|
||||
(item): IndicatorItem => {
|
||||
return {
|
||||
el: item as HTMLElement,
|
||||
position: Number(
|
||||
item.getAttribute("data-custom-carousel-slide-to"),
|
||||
),
|
||||
};
|
||||
},
|
||||
),
|
||||
},
|
||||
};
|
||||
};
|
||||
|
||||
_attachHandlers = (): void => {
|
||||
// Carousel controls
|
||||
this.querySelector("[data-custom-carousel-next]")?.addEventListener(
|
||||
"click",
|
||||
() => this._carousel.next(),
|
||||
);
|
||||
this.querySelector("[data-custom-carousel-prev]")?.addEventListener(
|
||||
"click",
|
||||
() => this._carousel.prev(),
|
||||
);
|
||||
|
||||
// Close fullscreen modal
|
||||
this._modalEl
|
||||
.querySelector("[data-custom-carousel-modal-hide]")
|
||||
?.addEventListener("click", () => this._modal.hide());
|
||||
|
||||
// Click to open fullscreen modal
|
||||
this._items.forEach((item) => {
|
||||
item.el.addEventListener("click", () => {
|
||||
const imgCloned = item.el.querySelector("img")?.cloneNode() as Node;
|
||||
const imageDiv =
|
||||
(this._modalEl.querySelector(
|
||||
"[data-custom-carousel-modal-image]",
|
||||
) as HTMLElement) || undefined;
|
||||
|
||||
imageDiv.innerHTML = "";
|
||||
imageDiv?.appendChild(imgCloned);
|
||||
|
||||
this._modal.show();
|
||||
});
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
customElements.define("custom-carousel", CustomCarousel);
|
||||
7
src/components/Media/PdfViewer.astro
Normal file
7
src/components/Media/PdfViewer.astro
Normal file
@@ -0,0 +1,7 @@
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
<iframe
|
||||
src={Astro.props.pdf}
|
||||
class={"w-9/10 md:w-3/5 h-7/10 md:h-4/5 " + Astro.props.class || ""}></iframe>
|
||||
11
src/components/Media/Video.astro
Normal file
11
src/components/Media/Video.astro
Normal file
@@ -0,0 +1,11 @@
|
||||
---
|
||||
import { type videoConfig } from "@interfaces/video.ts";
|
||||
|
||||
const config: videoConfig = Astro.props.videoConfig;
|
||||
console.log(config);
|
||||
---
|
||||
|
||||
<video class="h-auto w-full max-w-1/2" controls>
|
||||
<source src={config.videoPath} type={config.videoType ?? "video/mp4"} />
|
||||
Your browser does not support the video tag.
|
||||
</video>
|
||||
13
src/components/Media/YtVideo.astro
Normal file
13
src/components/Media/YtVideo.astro
Normal file
@@ -0,0 +1,13 @@
|
||||
---
|
||||
import type { videoConfig } from "@interfaces/video.ts";
|
||||
|
||||
const config: videoConfig = Astro.props.videoConfig;
|
||||
---
|
||||
|
||||
<iframe
|
||||
class="h-128 w-full max-w-1/2"
|
||||
src={config.videoPath}
|
||||
title={config.videoTitle ?? ""}
|
||||
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share"
|
||||
referrerpolicy="strict-origin-when-cross-origin"
|
||||
allowfullscreen></iframe>
|
||||
Reference in New Issue
Block a user