Skip to content

Commit 72c6d99

Browse files
feat: added --logo and --logo-alt (#851)
## PR Checklist - [x] Addresses an existing open issue: fixes #739 - [x] That issue was marked as [`status: accepting prs`](https://github.com/JoshuaKGoldberg/create-typescript-app/issues?q=is%3Aopen+is%3Aissue+label%3A%22status%3A+accepting+prs%22) - [x] Steps in [CONTRIBUTING.md](https://github.com/JoshuaKGoldberg/create-typescript-app/blob/main/.github/CONTRIBUTING.md) were taken ## Overview Adds in options to put an image in the README.md, and uses them in this repo with the "new" logo. (it's really [TypeStat](https://github.com/JoshuaKGoldberg/TypeStat)'s old logo) While I'm in the area, fills in test coverage for `readOptions.test.ts`. I also noticed it was taking >500ms to run sometimes. Turns out it wasn't mocking `readOptionDefaults`! 😱
1 parent ad148b0 commit 72c6d99

24 files changed

+333
-91
lines changed

README.md

+7-5
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,7 @@
2626
<img alt="TypeScript: Strict 💪" src="https://img.shields.io/badge/typescript-strict_💪-21bb42.svg" />
2727
</p>
2828

29-
Note that this template is early stage, opinionated, and not endorsed by the TypeScript team.
30-
It can be configured to set up a _lot_ of tooling out of the box.
31-
Each of the included tools exists for a good reason and provides real value.
32-
33-
If you don't want to use any particular tool, you can always remove it manually.
29+
<img align="right" alt="Project logo: the TypeScript blue square with rounded corners, but a plus sign instead of 'TS'" src="./create-typescript-app.png">
3430

3531
## Getting Started
3632

@@ -47,6 +43,12 @@ Use the corresponding docs page to get started:
4743
- [Initializing from the template](./docs/Initialization.md): creating a new repository with the [_Use this template_](https://github.com/JoshuaKGoldberg/create-typescript-app/generate) button on GitHub
4844
- [Migrating an existing repository](./docs/Migration.md): adding this template's tooling on top of an existing repository
4945

46+
Note that this template is early stage, opinionated, and not endorsed by the TypeScript team.
47+
It can be configured to set up a _lot_ of tooling out of the box.
48+
Each of the included tools exists for a good reason and provides real value.
49+
50+
If you don't want to use any particular tool, you can always remove it manually.
51+
5052
## Explainer
5153

5254
This template is available for anybody who wants to set up a Node application using TypeScript.

create-typescript-app.png

3.04 KB
Loading

docs/Options.md

+2
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,8 @@ The setup scripts also allow for optional overrides of the following inputs whos
5252
- `--author` _(`string`)_: Username on npm to publish packages under (by default, an existing npm author, or the currently logged in npm user, or `owner.toLowerCase()`)
5353
- `--email` _(`string`)_: Email address to be listed as the point of contact in docs and packages (e.g. `[email protected]`)
5454
- `--funding` _(`string`)_: GitHub organization or username to mention in `funding.yml` (by default, `owner`)
55+
- `--logo` _(`string`)_: Local image file in the repository to display near the top of the README.md as a logo
56+
- `--logoAlt` _(`string`)_: If `--logo` is provided or detected from an existing README.md, alt text that describes the image will be prompted for if not provided
5557

5658
For example, customizing the ownership and users associated with a new repository:
5759

+58-27
Original file line numberDiff line numberDiff line change
@@ -1,41 +1,72 @@
11
import { describe, expect, it } from "vitest";
22

3+
import { Options } from "../shared/types.js";
34
import { createRerunSuggestion } from "./createRerunSuggestion.js";
45

6+
const options = {
7+
author: "TestAuthor",
8+
base: "everything",
9+
createRepository: true,
10+
description: "Test description.",
11+
12+
excludeCompliance: true,
13+
excludeContributors: true,
14+
excludeLintJson: true,
15+
excludeLintKnip: true,
16+
excludeLintMd: false,
17+
excludeLintPackageJson: true,
18+
excludeLintPackages: false,
19+
excludeLintPerfectionist: true,
20+
excludeLintSpelling: false,
21+
excludeLintYml: false,
22+
excludeReleases: false,
23+
excludeRenovate: undefined,
24+
excludeTests: undefined,
25+
funding: undefined,
26+
logo: undefined,
27+
owner: "TestOwner",
28+
repository: "test-repository",
29+
skipGitHubApi: true,
30+
skipInstall: true,
31+
skipRemoval: true,
32+
skipRestore: undefined,
33+
skipUninstall: undefined,
34+
title: "Test Title",
35+
} satisfies Options;
36+
537
describe("createRerunSuggestion", () => {
638
it("includes key-value pairs with mixed truthy and falsy values", () => {
39+
const actual = createRerunSuggestion("initialize", options);
40+
41+
expect(actual).toMatchInlineSnapshot(
42+
'"npx create-typescript-app --mode initialize --author TestAuthor --base everything --create-repository true --description \\"Test description.\\" --email [email protected] --exclude-compliance true --exclude-contributors true --exclude-lint-json true --exclude-lint-knip true --exclude-lint-package-json true --exclude-lint-perfectionist true --owner TestOwner --repository test-repository --skip-github-api true --skip-install true --skip-removal true --title \\"Test Title\\""',
43+
);
44+
});
45+
46+
it("includes stringified logo when it exists", () => {
747
const actual = createRerunSuggestion("initialize", {
8-
author: "TestAuthor",
9-
base: "everything",
10-
createRepository: true,
11-
description: "Test description.",
12-
48+
...options,
49+
logo: {
50+
alt: "Test alt.",
51+
src: "test/src.png",
52+
},
53+
});
54+
55+
expect(actual).toMatchInlineSnapshot(
56+
'"npx create-typescript-app --mode initialize --author TestAuthor --base everything --create-repository true --description \\"Test description.\\" --email [email protected] --exclude-compliance true --exclude-contributors true --exclude-lint-json true --exclude-lint-knip true --exclude-lint-package-json true --exclude-lint-perfectionist true --logo test/src.png --owner TestOwner --repository test-repository --skip-github-api true --skip-install true --skip-removal true --title \\"Test Title\\" --logo-alt \\"Test alt.\\""',
57+
);
58+
});
59+
60+
it("includes exclusions when they exist", () => {
61+
const actual = createRerunSuggestion("initialize", {
62+
...options,
1363
excludeCompliance: true,
14-
excludeContributors: true,
15-
excludeLintJson: true,
16-
excludeLintKnip: true,
17-
excludeLintMd: false,
18-
excludeLintPackageJson: true,
19-
excludeLintPackages: false,
20-
excludeLintPerfectionist: true,
21-
excludeLintSpelling: false,
22-
excludeLintYml: false,
23-
excludeReleases: false,
24-
excludeRenovate: undefined,
25-
excludeTests: undefined,
26-
funding: undefined,
27-
owner: "TestOwner",
28-
repository: "test-repository",
29-
skipGitHubApi: true,
30-
skipInstall: true,
31-
skipRemoval: true,
32-
skipRestore: undefined,
33-
skipUninstall: undefined,
34-
title: "Test Title",
64+
excludeLintMd: true,
65+
excludeLintSpelling: true,
3566
});
3667

3768
expect(actual).toMatchInlineSnapshot(
38-
'"npx create-typescript-app --mode initialize --author TestAuthor --base everything --create-repository true --description \\"Test description.\\" --email [email protected] --exclude-compliance true --exclude-contributors true --exclude-lint-json true --exclude-lint-knip true --exclude-lint-package-json true --exclude-lint-perfectionist true --owner TestOwner --repository test-repository --skip-github-api true --skip-install true --skip-removal true --title \\"Test Title\\""',
69+
'"npx create-typescript-app --mode initialize --author TestAuthor --base everything --create-repository true --description \\"Test description.\\" --email [email protected] --exclude-compliance true --exclude-contributors true --exclude-lint-json true --exclude-lint-knip true --exclude-lint-md true --exclude-lint-package-json true --exclude-lint-perfectionist true --exclude-lint-spelling true --owner TestOwner --repository test-repository --skip-github-api true --skip-install true --skip-removal true --title \\"Test Title\\""',
3970
);
4071
});
4172
});

src/create/createRerunSuggestion.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,17 @@ export function createRerunSuggestion(
1212
mode: Mode,
1313
options: Partial<Options>,
1414
): string {
15-
const args = Object.entries(options)
15+
const optionsNormalized = {
16+
...options,
17+
...(options.logo
18+
? {
19+
logo: options.logo.src,
20+
logoAlt: options.logo.alt,
21+
}
22+
: { logo: undefined }),
23+
};
24+
25+
const args = Object.entries(optionsNormalized)
1626
.filter(([, value]) => !!value)
1727
.map(([key, value]) => {
1828
const valueStringified = `${value}`;

src/shared/options/args.ts

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ export const allArgOptions = {
1818
"exclude-renovate": { type: "boolean" },
1919
"exclude-tests": { type: "boolean" },
2020
funding: { type: "string" },
21+
logo: { type: "string" },
22+
"logo-alt": { type: "string" },
2123
owner: { type: "string" },
2224
repository: { type: "string" },
2325
"skip-github-api": { type: "boolean" },

src/shared/options/augmentOptionsWithExcludes.test.ts

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ const optionsBase = {
2323
excludeRenovate: undefined,
2424
excludeTests: undefined,
2525
funding: undefined,
26+
logo: undefined,
2627
owner: "",
2728
repository: "",
2829
skipGitHubApi: false,

src/shared/options/getPrefillOrPromptedOption.ts

+1-3
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ export async function getPrefillOrPromptedOption(
1111
return existingValue;
1212
}
1313

14-
const value = filterPromptCancel(
14+
return filterPromptCancel(
1515
await prompts.text({
1616
message,
1717
placeholder: await getPlaceholder?.(),
@@ -22,6 +22,4 @@ export async function getPrefillOrPromptedOption(
2222
},
2323
}),
2424
);
25-
26-
return value;
2725
}

src/shared/options/readGitAndNpmDefaults/readTitleFromReadme.test.ts

-33
This file was deleted.

src/shared/options/readGitAndNpmDefaults/readTitleFromReadme.ts

-8
This file was deleted.

src/shared/options/readGitAndNpmDefaults/index.ts src/shared/options/readOptionDefaults/index.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import { readPackageData } from "../../packages.js";
99
import { tryCatchAsync } from "../../tryCatchAsync.js";
1010
import { tryCatchLazyValueAsync } from "../../tryCatchLazyValueAsync.js";
1111
import { parsePackageAuthor } from "./parsePackageAuthor.js";
12-
import { readTitleFromReadme } from "./readTitleFromReadme.js";
12+
import { readDefaultsFromReadme } from "./readDefaultsFromReadme.js";
1313

1414
export function getGitAndNpmDefaults() {
1515
const gitDefaults = tryCatchLazyValueAsync(async () =>
@@ -46,6 +46,6 @@ export function getGitAndNpmDefaults() {
4646
(await gitDefaults())?.organization ?? (await packageAuthor()).author,
4747
repository: async () =>
4848
(await gitDefaults())?.name ?? (await packageData()).name,
49-
title: async () => await readTitleFromReadme(),
49+
...readDefaultsFromReadme(),
5050
};
5151
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
import { describe, expect, it, vi } from "vitest";
2+
3+
import { readDefaultsFromReadme } from "./readDefaultsFromReadme.js";
4+
5+
const mockReadFileSafe = vi.fn();
6+
7+
vi.mock("../../readFileSafe.js", () => ({
8+
get readFileSafe() {
9+
return mockReadFileSafe;
10+
},
11+
}));
12+
13+
describe("readDefaultsFromReadme", () => {
14+
describe("logo", () => {
15+
it("defaults to undefined when it cannot be found", async () => {
16+
mockReadFileSafe.mockResolvedValue("nothing.");
17+
18+
const logo = await readDefaultsFromReadme().logo();
19+
20+
expect(logo).toBeUndefined();
21+
});
22+
23+
it("parses when found in an unquoted string", async () => {
24+
mockReadFileSafe.mockResolvedValue("<img src=abc/def.jpg/>");
25+
26+
const logo = await readDefaultsFromReadme().logo();
27+
28+
expect(logo).toBe("abc/def.jpg");
29+
});
30+
31+
it("parses when found in a single quoted string", async () => {
32+
mockReadFileSafe.mockResolvedValue("<img src='abc/def.jpg'/>");
33+
34+
const logo = await readDefaultsFromReadme().logo();
35+
36+
expect(logo).toBe("abc/def.jpg");
37+
});
38+
39+
it("parses when found in a double quoted string", async () => {
40+
mockReadFileSafe.mockResolvedValue('<img src="abc/def.jpg"/>');
41+
42+
const logo = await readDefaultsFromReadme().logo();
43+
44+
expect(logo).toBe("abc/def.jpg");
45+
});
46+
});
47+
48+
describe("title", () => {
49+
it("defaults to undefined when it cannot be found", async () => {
50+
mockReadFileSafe.mockResolvedValue("nothing.");
51+
52+
const title = await readDefaultsFromReadme().title();
53+
54+
expect(title).toBeUndefined();
55+
});
56+
57+
it('reads title as markdown from "README.md" when it exists', async () => {
58+
mockReadFileSafe.mockResolvedValue("# My Awesome Package");
59+
60+
const title = await readDefaultsFromReadme().title();
61+
62+
expect(title).toBe("My Awesome Package");
63+
});
64+
65+
it('reads title as HTML from "README.md" when it exists', async () => {
66+
mockReadFileSafe.mockResolvedValue(
67+
'<h1 align="center">My Awesome Package</h1>',
68+
);
69+
70+
const title = await readDefaultsFromReadme().title();
71+
72+
expect(title).toBe("My Awesome Package");
73+
});
74+
75+
it("returns undefined when title does not exist", async () => {
76+
mockReadFileSafe.mockResolvedValue("");
77+
78+
const title = await readDefaultsFromReadme().title();
79+
80+
expect(title).toBeUndefined();
81+
});
82+
});
83+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
import lazyValue from "lazy-value";
2+
3+
import { readFileSafe } from "../../readFileSafe.js";
4+
5+
export function readDefaultsFromReadme() {
6+
const readme = lazyValue(async () => await readFileSafe("README.md", ""));
7+
8+
const imageTag = lazyValue(
9+
async () => (await readme()).match(/<img.+src.+\/>/)?.[0],
10+
);
11+
12+
return {
13+
logo: async () =>
14+
(await imageTag())
15+
?.match(/src\s*=(.+)?\/>/)?.[1]
16+
?.replaceAll(/^['"]|['"]$/g, ""),
17+
title: async () =>
18+
(await readme())
19+
.match(/^(?:# |<h1\s+align="center">)(.*?)(?:<\/h1>)?$/i)?.[1]
20+
?.trim()
21+
?.replace(/<[^>]+(?:>|$)/g, ""),
22+
};
23+
}

0 commit comments

Comments
 (0)