diff --git a/src/pages/funding.astro b/src/pages/funding.astro
index f8d2759..7d75ce4 100644
--- a/src/pages/funding.astro
+++ b/src/pages/funding.astro
@@ -1,6 +1,210 @@
---
import Layout from "../layouts/Layout.astro";
import { Icon } from "astro-icon/components";
+
+interface DonationMethod {
+ name: string;
+ bgClass: string;
+ titleColorClass?: string;
+ recurring?: boolean;
+ oneTime?: boolean;
+ preferred?: boolean;
+ flexible?: boolean;
+ crypto?: boolean;
+ badgeText?: string;
+ badgeColorClass?: string;
+ linkText: string;
+ linkUrl: string;
+ oldLinkText?: string;
+ oldLinkUrl?: string;
+ extraBadgeText?: string;
+ extraBadgeColorClass?: string;
+ extraBadgeText2?: string;
+ extraBadgeColorClass2?: string;
+ icon: string;
+}
+
+interface Task {
+ status: string;
+ statusColorClass: string;
+ title: string;
+ description: string;
+ progress: number;
+ link: string;
+ extras?: string[];
+ icons?: { name: string; title: string }[];
+ infoText?: string;
+}
+
+interface Sponsor {
+ username: string;
+ avatar: string;
+}
+
+interface PeopleList {
+ github: Sponsor[];
+ other: string[];
+}
+
+const donationMethods: DonationMethod[] = [
+ {
+ name: "GitHub Sponsors",
+ bgClass: "bg-blue-900 text-white",
+ recurring: true,
+ linkText: "Sponsor via GitHub",
+ linkUrl: "https://github.com/sponsors/bottlesdevs",
+ icon: "material-symbols:favorite-outline",
+ },
+ {
+ name: "PayPal",
+ bgClass: "bg-blue-500 dark:bg-blue-700 text-white",
+ recurring: true,
+ oneTime: true,
+ preferred: true,
+ linkText: "Donate via PayPal",
+ linkUrl: "https://www.paypal.com/donate?hosted_button_id=HM2VKUH5STU4J",
+ oldLinkText: "Old PayPal Method",
+ oldLinkUrl: "https://paypal.me/MirkoBrombin",
+ icon: "material-symbols:credit-card-outline",
+ },
+ {
+ name: "LiberaPay",
+ bgClass: "bg-yellow-400 dark:bg-yellow-500 text-gray-900",
+ flexible: true,
+ linkText: "Donate via LiberaPay",
+ linkUrl: "https://liberapay.com/bottles",
+ icon: "material-symbols:volunteer-activism-outline",
+ },
+ {
+ name: "Patreon",
+ bgClass: "bg-red-500 text-white",
+ recurring: true,
+ linkText: "Support via Patreon",
+ linkUrl: "https://www.patreon.com/MirkoBrombin",
+ icon: "material-symbols:favorite-outline",
+ },
+ {
+ name: "Cryptocurrency",
+ bgClass: "bg-purple-500 dark:bg-purple-700 text-white",
+ crypto: true,
+ linkText: "View Addresses",
+ linkUrl: "#",
+ icon: "material-symbols:currency-bitcoin",
+ },
+];
+
+const tasks: Task[] = [
+ {
+ status: "DONE",
+ statusColorClass: "bg-green-500",
+ title: "Library mode",
+ description:
+ "Allow users to add any program from any bottle in a library view for easy access.",
+ progress: 100,
+ link: "https://usebottles.com/posts/2022-08-28-release-2022.8.28",
+ },
+ {
+ status: "ALPHA",
+ statusColorClass: "bg-yellow-500",
+ title: "Per-bottle sandbox",
+ description:
+ "Experimental feature for wrapping each bottle with a sandbox to limit permissions.",
+ progress: 80,
+ link: "https://github.com/bottlesdevs/Bottles/issues/1158",
+ icons: [
+ { name: "material-symbols:build-circle", title: "In progress" },
+ { name: "material-symbols:help-outline", title: "Needs funding" },
+ ],
+ infoText: "Estimated time: 1 week of work based on average hourly cost.",
+ },
+ {
+ status: "REJECTED",
+ statusColorClass: "bg-red-500",
+ title: "Layered bottles",
+ description:
+ "Our solution for multiple environments in one bottle. Planned for Bottles Next. Please refer to the Bottles Next task for more details.",
+ progress: 0,
+ link: "https://github.com/bottlesdevs/Bottles/issues/510",
+ infoText:
+ "This feature is planned for Bottles Next. Read the Bottles Next task for more information.",
+ },
+ {
+ status: "WIP",
+ statusColorClass: "bg-blue-500",
+ title: "Integration of UMU",
+ description:
+ "Integration of UMU to enhance the gaming compatibility in Bottles.",
+ progress: 10,
+ link: "https://usebottles.com/posts/2024-09-24-umu-next/",
+ icons: [
+ { name: "material-symbols:build-circle", title: "In progress" },
+ { name: "material-symbols:help-outline", title: "Needs funding" },
+ ],
+ infoText: "Estimated time: 4 weeks of work based on average hourly cost.",
+ },
+ {
+ status: "NEXT",
+ statusColorClass: "bg-purple-500",
+ title: "Bottles Next",
+ description: `A complete redesign of Bottles to overcome current limitations and provide a consistent user experience across Linux and macOS. We need funds, contributors, and companies to provide infrastructure support.
+
+Additionally, we are experimenting with two technologies developed by the same developer of Bottles to determine the best candidate for Bottles Next:
+
`,
+ progress: 10,
+ link: "https://usebottles.com/posts/2023-10-05-bottles-next-a-new-chapter/",
+ icons: [
+ { name: "material-symbols:build-circle", title: "In progress" },
+ { name: "material-symbols:help-outline", title: "Needs funding" },
+ ],
+ infoText:
+ "We are looking for funds, contributors, and companies to provide infrastructure support.",
+ },
+];
+
+const filterButtons = [
+ { status: "WIP", text: "WIP" },
+ { status: "DONE", text: "Done" },
+ { status: "ALPHA", text: "Alpha" },
+ { status: "NEXT", text: "Next" },
+ { status: "REJECTED", text: "Rejected" },
+];
+
+const legendIcons = [
+ {
+ name: "material-symbols:build-circle",
+ text: "Work in progress",
+ title: "In progress",
+ },
+ {
+ name: "material-symbols:help-outline",
+ text: "Needs funding",
+ title: "Needs funding",
+ },
+ {
+ name: "material-symbols:info-outline",
+ text: "More information",
+ title: "More information",
+ },
+];
+
+const people: PeopleList = {
+ github: [],
+ other: [
+ "Christopher (kit) Eubanks",
+ "Black_file",
+ "Chris",
+ "Robin Lee",
+ "Andrew Ego",
+ "Sonny Piers",
+ "Dan G",
+ "Caleb Woodbine",
+ "Robert Krisztian Sandor",
+ "+ All the anonymous donations",
+ ],
+};
---
@@ -42,87 +246,74 @@ import { Icon } from "astro-icon/components";
Here are the official channels from which you can donate to Bottles
Developers.
+
+ {
+ donationMethods.map((method) => (
+
+
+
{method.name}
-
-
-
-
GitHub Sponsors
-
Recurring
-
Sponsor via GitHub
-
-
-
-
PayPal
-
One-Time / Recurring
-
Preferred Method
-
Donate via PayPal
-
Old PayPal Method
-
-
-
-
LiberaPay
-
Flexible Subscription
-
Donate via LiberaPay
-
-
-
-
Cryptocurrency
-
Bitcoin, Ethereum, and more
-
View Addresses
-
+ {method.recurring && method.oneTime ? (
+
+ One-Time / Recurring
+
+ ) : method.recurring ? (
+
+ Recurring
+
+ ) : method.oneTime ? (
+
+ One-Time
+
+ ) : (
+ ""
+ )}
+
+ {method.preferred ? (
+
+ Preferred Method
+
+ ) : (
+ ""
+ )}
+ {method.flexible ? (
+
+ Flexible Subscription
+
+ ) : (
+ ""
+ )}
+ {method.crypto ? (
+
+ Bitcoin, Ethereum, and more
+
+ ) : (
+ ""
+ )}
+
+
+ {method.linkText}
+
+ {method.oldLinkText ? (
+
+ {method.oldLinkText}
+
+ ) : (
+ ""
+ )}
+
+ ))
+ }
-
-
Bitcoin (BTC)
- Network: BTC(SegWit)à
+ Network: BTC(SegWit)
-
-
Litecoin (LTC)
Network: LTC
@@ -179,9 +366,7 @@ import { Icon } from "astro-icon/components";
LWqpd2411CQTuWD15bcPP7bfAw3wmYHScm
-
-
Ripple (XRP)
Network: XRP
@@ -200,29 +385,12 @@ import { Icon } from "astro-icon/components";
Close
+ Close
+
-
-
-
Patreon
-
Recurring
-
Donate via Patreon
-
-
- WIP
-
-
- Done
-
-
- Alpha
-
-
- Next
-
-
- Rejected
-
+ {
+ filterButtons.map((btn) => (
+
+ {btn.text}
+
+ ))
+ }
Reset Filters
- Reset Filters
-
-
-
- Work in progress
-
-
-
- Needs funding
-
-
-
- More information
-
+ {
+ legendIcons.map((icon) => (
+
+
+
+ {icon.text}
+
+
+ ))
+ }
-
-
-
[DONE] (
+
-
Library mode
-
- Allow users to add any program from any bottle in a library view
- for easy access.
-
-
-
-
-
-
-
[ALPHA]
-
Per-bottle sandbox
-
-
-
-
-
- Experimental feature for wrapping each bottle with a sandbox to
- limit permissions.
-
-
-
-
-
-
-
-
-
-
-
-
-
[REJECTED]
-
Layered bottles
-
- Our solution for multiple environments in one bottle. Planned
- for Bottles Next. Please refer to the Bottles Next task for more
- details.
-
-
-
-
-
-
-
-
-
-
-
-
-
[WIP]
-
Integration of UMU
-
-
-
-
-
- Integration of UMU to enhance the gaming compatibility in
- Bottles.
-
-
-
-
-
-
-
-
-
-
-
-
-
[NEXT]
-
Bottles Next
-
-
-
-
-
- A complete redesign of Bottles to overcome current limitations
- and provide a consistent user experience across Linux and macOS.
- We need funds, contributors, and companies to provide
- infrastructure support.
-
- Additionally, we are experimenting with two technologies developed
- by the same developer of Bottles to determine the best candidate
- for Bottles Next:
-
-
-
-
-
-
-
+ class={`${task.statusColorClass} text-white px-2 py-1 rounded-full text-sm inline-block mb-4`}
+ >
+ [{task.status}]
+
+
{task.title}
+ {task.icons ? (
+
+ {task.icons.map((ic) => (
+
+ ))}
+
+ ) : (
+ ""
+ )}
+
+
+
+
+
+ {task.infoText ? (
+
+ ) : (
+ ""
+ )}
+
+
+
-
-
-
+ ))
+ }
@@ -593,17 +518,11 @@ import { Icon } from "astro-icon/components";
Thanks to them who made a donation or became our GitHub sponsor ❤️!
-
-
-
-
+
Other platforms
-
-
-
-
+
- Thanks from the Bottles Developers.
@@ -662,29 +581,24 @@ import { Icon } from "astro-icon/components";
username: string;
avatar: string;
}
-
interface PeopleList {
github: Sponsor[];
other: string[];
}
-
class Donations {
domSelector: string;
list: PeopleList;
-
constructor(domSelector: string, list: PeopleList) {
this.domSelector = domSelector;
this.list = list;
this.renderPeople();
}
-
async renderPeople(): Promise
{
const githubPeople = await this.getGitHubSponsors();
const otherPeople = this.list.other;
this.addSection("#github", githubPeople, true);
this.addSection("#other", otherPeople);
}
-
addSection(
sectionDOM: string,
personList: Sponsor[] | string[],
@@ -706,22 +620,16 @@ import { Icon } from "astro-icon/components";
.querySelector(sectionDOM)
?.insertAdjacentHTML("beforeend", sectionHtml);
}
-
getPersonMarkup(
person: string,
hasPic: boolean,
avatarUrl: string = ""
): string {
return `
- ${
- hasPic
- ? `
`
- : ""
- }
+ ${hasPic ? `
` : ""}
${person}
`;
}
-
async getGitHubSponsors(): Promise {
try {
const response = await fetch(
@@ -736,13 +644,11 @@ import { Icon } from "astro-icon/components";
avatar: sponsor.avatar,
}));
return allSponsors;
- } catch (error) {
- console.error("Error fetching GitHub sponsors:", error);
+ } catch {
return [];
}
}
}
-
const people: PeopleList = {
github: [],
other: [
@@ -758,85 +664,71 @@ import { Icon } from "astro-icon/components";
"+ All the anonymous donations",
],
};
-
document.addEventListener("DOMContentLoaded", () => {
new Donations("#people", people);
-
const urlParams = new URLSearchParams(window.location.search);
if (urlParams.has("thankyou")) {
document.getElementById("donationModal")?.classList.remove("hidden");
}
-
document.getElementById("closeModal")?.addEventListener("click", () => {
document.getElementById("donationModal")?.classList.add("hidden");
});
-
document
.getElementById("closeCryptoModal")
?.addEventListener("click", () => {
document.getElementById("cryptoModal")?.classList.add("hidden");
});
-
document
.getElementById("closeInfoModal")
?.addEventListener("click", () => {
document.getElementById("infoModal")?.classList.add("hidden");
});
-
const filterButtons = document.querySelectorAll(".filter-button");
- const searchInput = document.getElementById("searchInput");
+ const searchInput = document.getElementById(
+ "searchInput"
+ ) as HTMLInputElement;
const resetButton = document.getElementById("resetButton");
const taskCards = document.querySelectorAll(".task-card");
-
let currentStatusFilter = "";
let currentSearchQuery = "";
-
function updateTasksDisplay() {
taskCards.forEach((card) => {
- const cardStatus = card.getAttribute("data-status");
- // @ts-ignore
- const title = card.getAttribute("data-title").toLowerCase();
- // @ts-ignore
- const description = card.querySelector("p").textContent.toLowerCase();
+ const c = card as HTMLElement;
+ const cardStatus = c.getAttribute("data-status") || "";
+ const title = (c.getAttribute("data-title") || "").toLowerCase();
+ const description = (
+ c.querySelector("p")?.textContent || ""
+ ).toLowerCase();
let matchesStatus = true;
let matchesSearch = true;
-
if (currentStatusFilter !== "") {
matchesStatus = cardStatus === currentStatusFilter;
}
-
if (currentSearchQuery !== "") {
matchesSearch =
title.includes(currentSearchQuery) ||
description.includes(currentSearchQuery);
}
-
- if (matchesStatus && matchesSearch) {
- // @ts-ignore
- card.style.display = "";
- } else {
- // @ts-ignore
- card.style.display = "none";
- }
+ c.style.display = matchesStatus && matchesSearch ? "" : "none";
});
}
-
filterButtons.forEach((button) => {
button.addEventListener("click", () => {
- const status = button.getAttribute("data-status");
- // @ts-ignore
+ const b = button as HTMLElement;
+ const status = b.getAttribute("data-status") || "";
currentStatusFilter = status;
filterButtons.forEach((btn) => {
- btn.classList.remove("active", "bg-blue-500", "text-white");
- btn.classList.add(
+ const bb = btn as HTMLElement;
+ bb.classList.remove("active", "bg-blue-500", "text-white");
+ bb.classList.add(
"bg-gray-200",
"dark:bg-gray-700",
"text-black",
"dark:text-white"
);
});
- button.classList.add("active", "bg-blue-500", "text-white");
- button.classList.remove(
+ b.classList.add("active", "bg-blue-500", "text-white");
+ b.classList.remove(
"bg-gray-200",
"dark:bg-gray-700",
"text-black",
@@ -845,23 +737,18 @@ import { Icon } from "astro-icon/components";
updateTasksDisplay();
});
});
-
- // @ts-ignore
searchInput.addEventListener("input", () => {
- // @ts-ignore
currentSearchQuery = searchInput.value.toLowerCase();
updateTasksDisplay();
});
-
- // @ts-ignore
- resetButton.addEventListener("click", () => {
+ resetButton?.addEventListener("click", () => {
currentStatusFilter = "";
currentSearchQuery = "";
- // @ts-ignore
searchInput.value = "";
filterButtons.forEach((btn) => {
- btn.classList.remove("active", "bg-blue-500", "text-white");
- btn.classList.add(
+ const bb = btn as HTMLElement;
+ bb.classList.remove("active", "bg-blue-500", "text-white");
+ bb.classList.add(
"bg-gray-200",
"dark:bg-gray-700",
"text-black",