-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlauncher.js
124 lines (113 loc) · 4.37 KB
/
launcher.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
import { ansi } from "../justjs/ansiStyle.js";
import { ProcessSync } from "../qjs-ext-lib/src/process.js";
import { getAppMenu } from "./applicationMenu.js";
import Fzf from "../justjs/fzf.js";
import { getWindowSize, handleFzfExec, setCommonFzfArgs } from "./utils.js";
/**
* @param {Array} list - The list of options to present to the user for selection.
*/
export default async function Launcher() {
const appMenu = getAppMenu();
const list = appMenu.Apps;
const listName = "Apps";
// Get the terminal window size (width and height) for formatting purposes
const [width, height] = getWindowSize();
// Retrieve the icon size from the global user arguments (this will influence the display format)
const iconSize = USER_ARGUMENTS.iconSize; // WxH
const [padding, iconPlacement] = (() => {
switch (USER_ARGUMENTS.preset) {
case "1":
return [
`${parseInt(iconSize / 2)},0,0,0`,
`${iconSize}x${iconSize}@${
Math.abs(parseInt(width / 2 - (iconSize / 2)))
}x1`,
];
case "2":
return [
`0,0,0,${parseInt(iconSize)}`,
`${iconSize}x${iconSize}@${1}x${
Math.abs(parseInt((height / 2) - (iconSize / 2)))
}`,
];
case "3":
default:
return [
`0,${parseInt(iconSize)},0,0`,
`${iconSize}x${iconSize}@${width - iconSize - 1}x${
Math.abs(parseInt(height / 2 - (iconSize / 2)))
}`,
];
}
})();
// Calculate the maximum name length among the options in the list to properly align the display
const maxNameLength = list.reduce(
(length, option) =>
option.name.length > length ? option.name.length : length,
0,
);
const fzfArgs = new Fzf().ansi().header("''").read0().delimiter("'#'")
.withNth(-1).info("right").padding(padding)
.infoCommand(
`'kitty icat --clear --transfer-mode=memory --unicode-placeholder --stdin=no --scale-up --place=${iconPlacement}` +
` "$(echo {} | head -n 1 | cut -d'#' -f1)" >>/dev/tty ${
USER_ARGUMENTS.printCategory
? `&& echo {} | head -n 4 | tail -n 1'`
: `'`
}`,
)
.preview('"echo {} | head -n 2 | tail -n 1 | column -c 1"').previewWindow(
"down,1,wrap,border-top",
)
.prompt(`"${listName}: "`).marker("''").pointer("''").highlightLine()
.bind(
"'enter:execute(`echo {} | head -n 3 | tail -n 1` > /dev/null 2>&1 &)+abort'",
)
.headerFirst().bind(
`"${USER_ARGUMENTS.modKey}-space:become(jiffy -m a -r)"`,
);
setCommonFzfArgs(fzfArgs);
// Format each option in the list with the app icon, category, keywords, name, and description
const styledOptions = list.map((option) => ({
displayName: `${option?.icon ?? ""}\n` // Display the app's icon (if any)
.concat( // Display the description, if available
option?.description ?? "",
"\n",
).concat( // Command to execute
"setsid ", // run the command as seperate process
option.terminal ? `${USER_ARGUMENTS.terminal} ` : "",
option.exec,
"\n",
)
.concat(option?.category ?? "", "\n") // Display the app's category
.concat( // Display the app's name and keywords, with proper formatting
"#\n",
ansi.style.green + option.name + ansi.style.reset +
" ".repeat(maxNameLength - option.name.length), // Align names by padding with spaces
option?.keywords
? ` : ${
width - maxNameLength - 10 < option.keywords.length
? ansi.style.gray +
option.keywords.substring(0, width - maxNameLength - 13)
.concat("...") +
ansi.style.reset // Truncate keywords line if it exceeds available space
: ansi.style.gray + option.keywords + ansi.style.reset
}`
: "",
),
...option, // Include all other properties of the option
}));
// Create a single string containing all the display names for use in the fzf input
const optionNames = styledOptions.map((option) =>
option.displayName.concat("\0") // Use null-terminated strings for fzf input
).join("");
// Create a new `ProcessSync` to run the `fzf` command synchronously with the formatted options
const launcher = new ProcessSync(
fzfArgs.toArray(),
{
input: optionNames,
useShell: true,
},
);
await handleFzfExec(launcher);
}