Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Mono-manifest #256

Closed
wants to merge 60 commits into from
Closed
Show file tree
Hide file tree
Changes from 30 commits
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
0599159
🚧 Initial support for mono-manifest themes (no stars yet)
theRealPadster Jun 25, 2022
8fb8e5f
Merge branch 'main' into feat/mono-manifest
theRealPadster Jun 25, 2022
711695d
Fix some linting
theRealPadster Jun 25, 2022
f2595ae
🚧 Initial mono-manifest support for custom apps
theRealPadster Jun 25, 2022
1f1f8a2
Fix some type errors
theRealPadster Jun 25, 2022
e39df53
Naming
theRealPadster Jun 25, 2022
256254f
Fix type errors
theRealPadster Jun 25, 2022
74ed0d7
More type fixes
theRealPadster Jun 25, 2022
b72291c
Merge branch 'main' into feat/mono-manifest
theRealPadster Jun 25, 2022
ea7b22a
Change to use ES2021
CharlieS1103 Jun 25, 2022
73498af
Begin implementing github topic fetch option
CharlieS1103 Jun 25, 2022
316ba74
Resolve linting errors
CharlieS1103 Jun 25, 2022
41827f7
Finish implementing optionality for themes
CharlieS1103 Jun 25, 2022
f5c90e8
Change card type
CharlieS1103 Jun 25, 2022
96d65df
Add logic to check if enabled before calling loadPageRecursive()
CharlieS1103 Jun 25, 2022
d8565c6
Fill in documentation links
CharlieS1103 Jun 25, 2022
f0a6acd
Remove unnecessary conditional
CharlieS1103 Jun 25, 2022
86e5c7a
Remove unnecessary imports
CharlieS1103 Jun 25, 2022
be885d4
Patch multimanifest support
CharlieS1103 Jun 25, 2022
451e2f2
🔥 ♻️ Remove some duplicated code and do some cleanup
theRealPadster Jun 26, 2022
f0573d1
♻️ Use `buildThemeCardData` for topic and mono-manifest flow
theRealPadster Jun 26, 2022
7cb545a
♻️ Code deduplication for `buildAppCardData`
theRealPadster Jun 26, 2022
7cc8bae
♻️ Consolidate getMonoManifest functions
theRealPadster Jun 26, 2022
cf09e54
Add `buildExtensionCardData`
theRealPadster Jun 26, 2022
c186d61
🚧 Add extension mono-manifest handling (not implemented)
theRealPadster Jun 26, 2022
aa04101
Normalize the marketplace controls colors
CharlieS1103 Jun 28, 2022
f0ce736
refactor(FetchRemotes): use lookups
kyrie25 Jun 28, 2022
798e6d4
feat(Settings): add annotation
kyrie25 Jun 28, 2022
c225f80
feat(Grid): allow switching fetch method for Apps
kyrie25 Jun 28, 2022
1538241
refactor(Grid): merge duplicate functions
kyrie25 Jun 28, 2022
337bb25
Make order of extension/theme/app functions consistent with tabs
theRealPadster Jun 28, 2022
531bc66
♻️ 🚧 Refactor `fetchMonoManifest` temporarily to be more efficient
theRealPadster Jun 28, 2022
3f5fb6f
Use same spice-text var for dropdown hover
theRealPadster Jun 29, 2022
a284f93
Add arrow-parens rule, Rearrange extensions/themes/apps to line up wi…
theRealPadster Jun 29, 2022
4de06a4
Forgot the rule
theRealPadster Jun 29, 2022
0c9e261
Make really long function params more readable, arrow-parens
theRealPadster Jun 29, 2022
3869c57
Add classes for some inline css
theRealPadster Jun 29, 2022
ed14b2a
Missed an arrow-parens
theRealPadster Jun 29, 2022
d0f4dc1
♻️ Move icons to single file
theRealPadster Jun 29, 2022
e34d368
feat: use synchronous fetch method & assert type
kyrie25 Jun 29, 2022
a9ead9a
chore: remove unnecessary type assertion
kyrie25 Jun 29, 2022
6791dd9
fix: wait until item is fetched completely
kyrie25 Jun 29, 2022
97d882d
🔀 Merge branch 'main' into feat/mono-manifest
theRealPadster Jun 30, 2022
80bb41b
Merge branch 'main' into feat/mono-manifest
theRealPadster Jun 30, 2022
6ab8049
Merge branch 'main' into feat/mono-manifest
theRealPadster Jun 30, 2022
ba6659f
For some reason it didn't put this in
theRealPadster Jun 30, 2022
cf6eb53
🔧 Add EOF ESLint rule
theRealPadster Jun 30, 2022
a70a362
Revert "For some reason it didn't put this in"
theRealPadster Jun 30, 2022
0fed49c
Merge branch 'main' into feat/mono-manifest
kyrie25 Jul 2, 2022
fe45603
Merge branch 'main' into feat/mono-manifest
kyrie25 Jul 4, 2022
4631df1
Merge branch 'main' into feat/mono-manifest
kyrie25 Jul 11, 2022
7188a54
refactor(utils)
kyrie25 Jul 11, 2022
15a07d6
Merge branch 'main' into feat/mono-manifest
kyrie25 Jul 14, 2022
494c322
Merge branch 'main' into feat/mono-manifest
kyrie25 Aug 4, 2022
4bddf46
chore: remove duplicate identifier
kyrie25 Aug 4, 2022
71035c2
Merge branch 'main' into feat/mono-manifest
kyrie25 Aug 4, 2022
f2d5bda
🔀 Merge branch 'main' into feat/mono-manifest
theRealPadster Nov 21, 2022
bffac96
Fix some types/merge stuff
theRealPadster Nov 21, 2022
5c2e628
Linting
theRealPadster Nov 21, 2022
e403d60
🚧 Start fetch-stars script
theRealPadster Dec 3, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 3 additions & 3 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ All types of contributions are encouraged and valued. See the [Table of Contents

## I Have a Question

> If you want to ask a question, we assume that you have read the available [Documentation]().
> If you want to ask a question, we assume that you have read the available [Documentation](https://github.com/spicetify/spicetify-marketplace/wiki).

Before you ask a question, it is best to search for existing [Issues](https://github.com/spicetify/spicetify-marketplaceissues) that might help you. In case you have found a suitable issue and still need clarification, you can write your question in this issue. It is also advisable to search the internet for answers first.

Expand Down Expand Up @@ -68,7 +68,7 @@ Depending on how large the project is, you may want to outsource the questioning
A good bug report shouldn't leave others needing to chase you up for more information. Therefore, we ask you to investigate carefully, collect information and describe the issue in detail in your report. Please complete the following steps in advance to help us fix any potential bug as fast as possible.

- Make sure that you are using the latest version.
- Determine if your bug is really a bug and not an error on your side e.g. using incompatible environment components/versions (Make sure that you have read the [documentation](). If you are looking for support, you might want to check [this section](#i-have-a-question)).
- Determine if your bug is really a bug and not an error on your side e.g. using incompatible environment components/versions (Make sure that you have read the [documentation](https://github.com/spicetify/spicetify-marketplace/wiki). If you are looking for support, you might want to check [this section](#i-have-a-question)).
- To see if other users have experienced (and potentially already solved) the same issue you are having, check if there is not already a bug report existing for your bug or error in the [bug tracker](https://github.com/spicetify/spicetify-marketplaceissues?q=label%3Abug).
- Also make sure to search the internet (including Stack Overflow) to see if users outside of the GitHub community have discussed the issue.
- Collect information about the bug:
Expand Down Expand Up @@ -108,7 +108,7 @@ This section guides you through submitting an enhancement suggestion for Spiceti
#### Before Submitting an Enhancement

- Make sure that you are using the latest version.
- Read the [documentation]() carefully and find out if the functionality is already covered, maybe by an individual configuration.
- Read the [documentation](https://github.com/spicetify/spicetify-marketplace/wiki) carefully and find out if the functionality is already covered, maybe by an individual configuration.
- Perform a [search](https://github.com/spicetify/spicetify-marketplaceissues) to see if the enhancement has already been suggested. If it has, add a comment to the existing issue instead of opening a new one.
- Find out whether your idea fits with the scope and aims of the project. It's up to you to make a strong case to convince the project's developers of the merits of this feature. Keep in mind that we want features that will be useful to the majority of our users and not just a small subset. If you're just targeting a minority of users, consider writing an add-on/plugin library.

Expand Down
22 changes: 22 additions & 0 deletions resources/manifests/apps.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
export default [
{
"name": "Name That Tune!!1",
"description": "Heardle.app made for Spicetify",

// TODO: Get the image working (once there is a repo?)
"preview": "https://github.com/theRealPadster/name-that-tune/raw/main/docs/preview.png",
// TODO: Get the readme working (once there is a repo?)
"readme": "README.md",

"authors": [
{
"name": "theRealPadster",
"url": "https://github.com/theRealPadster",
},
],
"tags": [
"game",
"beta",
],
},
];
5 changes: 5 additions & 0 deletions resources/manifests/extensions.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
export default [
{
// TODO: add something here to test
},
];
1 change: 1 addition & 0 deletions src/app.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ class App extends React.Component<null, {count: number, CONFIG: Config}> {
// of stargazers, and the subscribers_count isn't returned in the main API call we make
// https://github.community/t/bug-watchers-count-is-the-duplicate-of-stargazers-count/140865/4
followers: JSON.parse(getLocalStorageDataFromKey("marketplace:followers", false)),
githubTopics: JSON.parse(getLocalStorageDataFromKey("marketplace:githubTopics", false)),
},
tabs,
activeTab: getLocalStorageDataFromKey(LOCALSTORAGE_KEYS.activeTab, tabs[0]),
Expand Down
160 changes: 62 additions & 98 deletions src/components/Grid.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,10 @@ import React from "react";
import semver from "semver";
import { Option } from "react-dropdown";

import { CardItem, CardType, Config, SchemeIni, Snippet, TabItemConfig } from "../types/marketplace-types";
import { CardItem, CardType, Config, RepoTopic, RepoType, SchemeIni, Snippet, TabItemConfig } from "../types/marketplace-types";
import { getLocalStorageDataFromKey, generateSchemesOptions, injectColourScheme } from "../logic/Utils";
import { LOCALSTORAGE_KEYS, ITEMS_PER_REQUEST, MARKETPLACE_VERSION, LATEST_RELEASE } from "../constants";
import { openModal } from "../logic/LaunchModals";
import {
getTaggedRepos,
fetchExtensionManifest, fetchThemeManifest, fetchAppManifest,
fetchCssSnippets, getBlacklist,
} from "../logic/FetchRemotes";
import LoadMoreIcon from "./Icons/LoadMoreIcon";
import LoadingIcon from "./Icons/LoadingIcon";
import SettingsIcon from "./Icons/SettingsIcon";
Expand All @@ -21,6 +16,14 @@ import Card from "./Card/Card";
import Button from "./Button";
import DownloadIcon from "./Icons/DownloadIcon";
import Changelog from "./Modals/Changelog";
import {
fetchBlacklist, fetchCssSnippets, fetchMonoManifest,
buildThemeCardData, buildAppCardData, buildExtensionCardData,
} from "../logic/FetchRemotes";
import {
getTaggedRepos,
fetchExtensionManifest, fetchThemeManifest, fetchAppManifest,
} from "../logic/FetchTopicRemotes";

export default class Grid extends React.Component<
{
Expand Down Expand Up @@ -178,43 +181,7 @@ export default class Grid extends React.Component<
// TODO: maybe we should rename `loadPage()`, since it's slightly confusing when we have github pages as well
async loadPage(queue: never[], query?: string) {
switch (this.CONFIG.activeTab) {
case "Extensions": {
const pageOfRepos = await getTaggedRepos("spicetify-extensions", this.requestPage, this.BLACKLIST, query);
for (const repo of pageOfRepos.items) {
const extensions = await fetchExtensionManifest(
repo.contents_url,
repo.default_branch,
repo.stargazers_count,
this.CONFIG.visual.hideInstalled,
);

// I believe this stops the requests when switching tabs?
if (this.requestQueue.length > 1 && queue !== this.requestQueue[0]) {
// Stop this queue from continuing to fetch and append to cards list
return -1;
}

if (extensions && extensions.length) {
// console.log(`${repo.name} has ${extensions.length} extensions:`, extensions);
extensions.forEach((extension) => {
Object.assign(extension, { lastUpdated: repo.pushed_at });
this.appendCard(extension, "extension");
});
}
}

// First result is null or -1 so it coerces to 1
const currentPage = this.requestPage > -1 && this.requestPage ? this.requestPage : 1;
// Sets the amount of items that have thus been fetched
const soFarResults = ITEMS_PER_REQUEST * (currentPage - 1) + pageOfRepos.page_count;
const remainingResults = pageOfRepos.total_count - soFarResults;

// If still have more results, return next page number to fetch
console.log(`Parsed ${soFarResults}/${pageOfRepos.total_count} extensions`);
if (remainingResults > 0) return currentPage + 1;
else console.log("No more extension results");
break;
} case "Installed": {
case "Installed": {
const installedStuff = {
theme: getLocalStorageDataFromKey(LOCALSTORAGE_KEYS.installedThemes, []),
extension: getLocalStorageDataFromKey(LOCALSTORAGE_KEYS.installedExtensions, []),
Expand All @@ -240,71 +207,68 @@ export default class Grid extends React.Component<

// Don't need to return a page number because
// installed extension do them all in one go, since it's local
} case "Themes": {
const pageOfRepos = await getTaggedRepos("spicetify-themes", this.requestPage, this.BLACKLIST, query);
for (const repo of pageOfRepos.items) {

const themes = await fetchThemeManifest(
repo.contents_url,
repo.default_branch,
repo.stargazers_count,
);
// I believe this stops the requests when switching tabs?
if (this.requestQueue.length > 1 && queue !== this.requestQueue[0]) {
// Stop this queue from continuing to fetch and append to cards list
return -1;
}

if (themes && themes.length) {
themes.forEach((theme) => {
Object.assign(theme, { lastUpdated: repo.pushed_at });
this.appendCard(theme, "theme");
});
}
} case "Themes":
case "Extensions":
case "Apps": {
const type = this.CONFIG.activeTab.slice(0, -1).toLowerCase() as Omit<CardType, "snippet">;
let allRepos;
if (this.CONFIG.visual.githubTopics) {
const topicResponse = await getTaggedRepos(`spicetify-${type}s` as RepoTopic, this.requestPage, this.BLACKLIST, query);
allRepos = topicResponse.items;
} else {
allRepos = await fetchMonoManifest(type as RepoType);
}

// First request is null, so coerces to 1
const currentPage = this.requestPage > -1 && this.requestPage ? this.requestPage : 1;
// -1 because the page number is 1-indexed
const soFarResults = ITEMS_PER_REQUEST * (currentPage - 1) + pageOfRepos.page_count;
const remainingResults = pageOfRepos.total_count - soFarResults;
for (const repo of allRepos) {
let cardData;
if (this.CONFIG.visual.githubTopics) {
switch (type) {
case "theme":
cardData = await fetchThemeManifest(repo.contents_url, repo.default_branch, repo.stargazers_count);
break;
case "extension":
cardData = await fetchExtensionManifest(repo.contents_url, repo.default_branch, repo.stargazers_count);
break;
case "app":
cardData = await fetchAppManifest(repo.contents_url, repo.default_branch, repo.stargazers_count);
break;
}
} else {
switch (type) {
case "theme":
cardData = buildThemeCardData(repo);
break;
case "extension":
cardData = buildExtensionCardData(repo);
break;
case "app":
cardData = buildAppCardData(repo);
break;
default:
throw new Error(`Unknown type: ${type}`);
}
}

console.log(`Parsed ${soFarResults}/${pageOfRepos.total_count} themes`);
if (remainingResults > 0) return currentPage + 1;
else console.log("No more theme results");
break;
} case "Apps": {
const pageOfRepos = await getTaggedRepos("spicetify-apps", this.requestPage, this.BLACKLIST, query);
for (const repo of pageOfRepos.items) {

const apps = await fetchAppManifest(
repo.contents_url,
repo.default_branch,
repo.stargazers_count,
);
// TODO: do we need this queue stuff any more?
// I believe this stops the requests when switching tabs?
if (this.requestQueue.length > 1 && queue !== this.requestQueue[0]) {
// Stop this queue from continuing to fetch and append to cards list
return -1;
}

if (apps && apps.length) {
apps.forEach((app) => {
Object.assign(app, { lastUpdated: repo.pushed_at });
this.appendCard(app, "app");
});
if (cardData) {
console.log(cardData);
if (this.CONFIG.visual.githubTopics) {
for (const item of cardData as CardItem[]) {
Object.assign(item, { lastUpdated: repo.pushed_at });
this.appendCard(item, type as CardType);
}
} else {
this.appendCard(cardData as CardItem, type as CardType);
}
}
}

// First request is null, so coerces to 1
const currentPage = this.requestPage > -1 && this.requestPage ? this.requestPage : 1;
// -1 because the page number is 1-indexed
const soFarResults = ITEMS_PER_REQUEST * (currentPage - 1) + pageOfRepos.page_count;
const remainingResults = pageOfRepos.total_count - soFarResults;

console.log(`Parsed ${soFarResults}/${pageOfRepos.total_count} apps`);
if (remainingResults > 0) return currentPage + 1;
else console.log("No more app results");
console.log(`Parsed ${this.CONFIG.activeTab.toLowerCase()}`);
break;
} case "Snippets": {
const snippets = await fetchCssSnippets();
Expand Down Expand Up @@ -441,7 +405,7 @@ export default class Grid extends React.Component<
}

// Load blacklist
this.BLACKLIST = await getBlacklist();
this.BLACKLIST = await fetchBlacklist();
this.newRequest(ITEMS_PER_REQUEST);
}

Expand Down
1 change: 1 addition & 0 deletions src/components/Modals/Settings/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ const SettingsModal = ({ CONFIG, updateAppConfig } : Props) => {
<ConfigRow name='Theme developer tools' storageKey='themeDevTools' modalConfig={modalConfig} updateConfig={updateConfig} />
<ConfigRow name='Hide installed when browsing' storageKey='hideInstalled' modalConfig={modalConfig} updateConfig={updateConfig} />
<ConfigRow name='Shift colors every minute' storageKey='colorShift' modalConfig={modalConfig} updateConfig={updateConfig} />
<ConfigRow name='Use Github topics to fetch cards (legacy)' storageKey='githubTopics' modalConfig={modalConfig} updateConfig={updateConfig} />
<h2>Tabs</h2>
<div className="tabs-container">
{modalConfig.tabs.map(({ name }, index) => {
Expand Down
3 changes: 3 additions & 0 deletions src/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ export const LOCALSTORAGE_KEYS = {
// Theme installed store the localsorage key of the theme (e.g. marketplace:installed:NYRI4/Comfy-spicetify/user.css)
"themeInstalled": "marketplace:theme-installed",
"colorShift": "marketplace:colorShift",
"githubTopics": "marketplace:githubTopics",
};

// Initalize topbar tabs
Expand All @@ -34,6 +35,8 @@ export const CUSTOM_APP_PATH = "/marketplace";
// Used in Card.tsx
export const MAX_TAGS = 4;

export const THEMES_URL = "https://raw.githubusercontent.com/spicetify/spicetify-themes/generated-manifest/manifest.json";

export const SNIPPETS_PAGE_URL = "https://github.com/spicetify/spicetify-marketplace/blob/main/resources/snippets.ts";

export const BLACKLIST_URL = "https://raw.githubusercontent.com/spicetify/spicetify-marketplace/main/resources/blacklist.json";
Expand Down
39 changes: 16 additions & 23 deletions src/extensions/extension.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,12 @@ import {
sleep,
} from "../logic/Utils";
import {
getBlacklist,
fetchBlacklist,
} from "../logic/FetchRemotes";
import {
fetchThemeManifest,
fetchExtensionManifest,
} from "../logic/FetchRemotes";
} from "../logic/FetchTopicRemotes";

(async () => {
while (!(Spicetify?.LocalStorage && Spicetify?.showNotification)) {
Expand Down Expand Up @@ -171,11 +173,11 @@ async function queryRepos(type: RepoType, pageNum = 1) {
*/
async function loadPageRecursive(type: RepoType, pageNum: number) {
const pageOfRepos = await queryRepos(type, pageNum);
// TODO: Once we migrate to the mono-manifest repo, implement the option for enabling topics search here
appendInformationToLocalStorage(pageOfRepos, type);

// Sets the amount of items that have thus been fetched
const soFarResults = ITEMS_PER_REQUEST * (pageNum - 1) + pageOfRepos.page_count;
console.log({ pageOfRepos });
const remainingResults = pageOfRepos.total_count - soFarResults;

// If still have more results, recursively fetch next page
Expand All @@ -187,36 +189,26 @@ async function loadPageRecursive(type: RepoType, pageNum: number) {
(async function initializePreload() {
console.log("Preloading extensions and themes...");
window.sessionStorage.clear();
const BLACKLIST = await getBlacklist();
const BLACKLIST = await fetchBlacklist();
window.sessionStorage.setItem("marketplace:blacklist", JSON.stringify(BLACKLIST));

// TODO: does this work?
// The recursion isn't super clean...

// Begin by getting the themes and extensions from github
// const [extensionReposArray, themeReposArray] = await Promise.all([
await Promise.all([
loadPageRecursive("extension", 1),
loadPageRecursive("theme", 1),
]);

// let extensionsNextPage = 1;
// let themesNextPage = 1;
// do {
// extensionReposArray = await loadPage("extension", extensionsNextPage);
// appendInformationToLocalStorage(extensionReposArray, "extension");
// } while (extensionsNextPage);

// do {
// themeReposArray = await loadPage("theme", themesNextPage);
// appendInformationToLocalStorage(themeReposArray, "theme");
// } while (themesNextPage);
// TODO: re-enable this once everything works with mono-manifest...
if (LOCALSTORAGE_KEYS.githubTopics) {
await Promise.all([
loadPageRecursive("extension", 1),
loadPageRecursive("theme", 1),
]);
}

})();

async function appendInformationToLocalStorage(array, type: RepoType) {
// This system should make it so themes and extensions are stored concurrently
for (const repo of array.items) {
// console.log(repo);
console.log(repo);
const data = (type === "theme")
? await fetchThemeManifest(repo.contents_url, repo.default_branch, repo.stargazers_count)
: await fetchExtensionManifest(repo.contents_url, repo.default_branch, repo.stargazers_count);
Expand All @@ -225,4 +217,5 @@ async function appendInformationToLocalStorage(array, type: RepoType) {
await sleep(5000);
}
}

}
Loading