Compare commits
6 Commits
v0.50.1
...
scale-work
Author | SHA1 | Date | |
---|---|---|---|
|
85917a40b0 | ||
|
d3e26652d6 | ||
|
276652b44e | ||
|
746f804568 | ||
|
00319c01d4 | ||
|
b509273759 |
101
.clang-tidy
@@ -1,101 +0,0 @@
|
|||||||
WarningsAsErrors: '*'
|
|
||||||
HeaderFilterRegex: '.*\.hpp'
|
|
||||||
FormatStyle: file
|
|
||||||
Checks: >
|
|
||||||
-*,
|
|
||||||
bugprone-*,
|
|
||||||
-bugprone-easily-swappable-parameters,
|
|
||||||
-bugprone-forward-declararion-namespace,
|
|
||||||
-bugprone-forward-declararion-namespace,
|
|
||||||
-bugprone-macro-parentheses,
|
|
||||||
-bugprone-narrowing-conversions,
|
|
||||||
-bugprone-branch-clone,
|
|
||||||
-bugprone-assignment-in-if-condition,
|
|
||||||
concurrency-*,
|
|
||||||
-concurrency-mt-unsafe,
|
|
||||||
cppcoreguidelines-*,
|
|
||||||
-cppcoreguidelines-owning-memory,
|
|
||||||
-cppcoreguidelines-avoid-magic-numbers,
|
|
||||||
-cppcoreguidelines-pro-bounds-constant-array-index,
|
|
||||||
-cppcoreguidelines-avoid-const-or-ref-data-members,
|
|
||||||
-cppcoreguidelines-non-private-member-variables-in-classes,
|
|
||||||
-cppcoreguidelines-avoid-goto,
|
|
||||||
-cppcoreguidelines-pro-bounds-array-to-pointer-decay,
|
|
||||||
-cppcoreguidelines-avoid-do-while,
|
|
||||||
-cppcoreguidelines-avoid-non-const-global-variables,
|
|
||||||
-cppcoreguidelines-special-member-functions,
|
|
||||||
-cppcoreguidelines-explicit-virtual-functions,
|
|
||||||
-cppcoreguidelines-avoid-c-arrays,
|
|
||||||
-cppcoreguidelines-pro-bounds-pointer-arithmetic,
|
|
||||||
-cppcoreguidelines-narrowing-conversions,
|
|
||||||
-cppcoreguidelines-pro-type-union-access,
|
|
||||||
-cppcoreguidelines-pro-type-member-init,
|
|
||||||
-cppcoreguidelines-macro-usage,
|
|
||||||
-cppcoreguidelines-macro-to-enum,
|
|
||||||
-cppcoreguidelines-init-variables,
|
|
||||||
-cppcoreguidelines-pro-type-cstyle-cast,
|
|
||||||
-cppcoreguidelines-pro-type-vararg,
|
|
||||||
-cppcoreguidelines-pro-type-reinterpret-cast,
|
|
||||||
google-global-names-in-headers,
|
|
||||||
-google-readability-casting,
|
|
||||||
google-runtime-operator,
|
|
||||||
misc-*,
|
|
||||||
-misc-unused-parameters,
|
|
||||||
-misc-no-recursion,
|
|
||||||
-misc-non-private-member-variables-in-classes,
|
|
||||||
-misc-include-cleaner,
|
|
||||||
-misc-use-anonymous-namespace,
|
|
||||||
-misc-const-correctness,
|
|
||||||
modernize-*,
|
|
||||||
-modernize-return-braced-init-list,
|
|
||||||
-modernize-use-trailing-return-type,
|
|
||||||
-modernize-use-using,
|
|
||||||
-modernize-use-override,
|
|
||||||
-modernize-avoid-c-arrays,
|
|
||||||
-modernize-macro-to-enum,
|
|
||||||
-modernize-loop-convert,
|
|
||||||
-modernize-use-nodiscard,
|
|
||||||
-modernize-pass-by-value,
|
|
||||||
-modernize-use-auto,
|
|
||||||
performance-*,
|
|
||||||
-performance-avoid-endl,
|
|
||||||
-performance-unnecessary-value-param,
|
|
||||||
portability-std-allocator-const,
|
|
||||||
readability-*,
|
|
||||||
-readability-function-cognitive-complexity,
|
|
||||||
-readability-function-size,
|
|
||||||
-readability-identifier-length,
|
|
||||||
-readability-magic-numbers,
|
|
||||||
-readability-uppercase-literal-suffix,
|
|
||||||
-readability-braces-around-statements,
|
|
||||||
-readability-redundant-access-specifiers,
|
|
||||||
-readability-else-after-return,
|
|
||||||
-readability-container-data-pointer,
|
|
||||||
-readability-implicit-bool-conversion,
|
|
||||||
-readability-avoid-nested-conditional-operator,
|
|
||||||
-readability-redundant-member-init,
|
|
||||||
-readability-redundant-string-init,
|
|
||||||
-readability-avoid-const-params-in-decls,
|
|
||||||
-readability-named-parameter,
|
|
||||||
-readability-convert-member-functions-to-static,
|
|
||||||
-readability-qualified-auto,
|
|
||||||
-readability-make-member-function-const,
|
|
||||||
-readability-isolate-declaration,
|
|
||||||
-readability-inconsistent-declaration-parameter-name,
|
|
||||||
-clang-diagnostic-error,
|
|
||||||
|
|
||||||
CheckOptions:
|
|
||||||
performance-for-range-copy.WarnOnAllAutoCopies: true
|
|
||||||
performance-inefficient-string-concatenation.StrictMode: true
|
|
||||||
readability-braces-around-statements.ShortStatementLines: 0
|
|
||||||
readability-identifier-naming.ClassCase: CamelCase
|
|
||||||
readability-identifier-naming.ClassIgnoredRegexp: I.*
|
|
||||||
readability-identifier-naming.ClassPrefix: C # We can't use regex here?!?!?!?
|
|
||||||
readability-identifier-naming.EnumCase: CamelCase
|
|
||||||
readability-identifier-naming.EnumPrefix: e
|
|
||||||
readability-identifier-naming.EnumConstantCase: UPPER_CASE
|
|
||||||
readability-identifier-naming.FunctionCase: camelBack
|
|
||||||
readability-identifier-naming.NamespaceCase: CamelCase
|
|
||||||
readability-identifier-naming.NamespacePrefix: N
|
|
||||||
readability-identifier-naming.StructPrefix: S
|
|
||||||
readability-identifier-naming.StructCase: CamelCase
|
|
77
.github/ISSUE_TEMPLATE/bug.yml
vendored
@@ -1,15 +1,74 @@
|
|||||||
name: Do not open issues, go to discussions please!
|
name: Bug Report
|
||||||
description: Do not open an issue
|
description: Something is not working right
|
||||||
|
labels: ["bug"]
|
||||||
body:
|
body:
|
||||||
- type: checkboxes
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
label: Please close this issue.
|
value: |
|
||||||
description: Users cannot open issues. I want my issue to be closed.
|
## Before opening a new issue, please take a moment to search through the current open and closed issues to check if it already exists.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: type
|
||||||
|
attributes:
|
||||||
|
label: Regression?
|
||||||
|
description: |
|
||||||
|
Regression means that something used to work but no longer does.
|
||||||
|
**BEFORE CONTINUING**, please check if this bug is a regression or not, and if it is, we need you to bisect with the help of the wiki: https://wiki.hyprland.org/Crashes-and-Bugs/#bisecting-an-issue
|
||||||
|
multiple: true
|
||||||
options:
|
options:
|
||||||
- label: Yes, I want this issue to be closed.
|
- "Yes"
|
||||||
required: true
|
- "No"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
- type: textarea
|
- type: textarea
|
||||||
id: body
|
id: ver
|
||||||
attributes:
|
attributes:
|
||||||
label: Issue body
|
label: System Info and Version
|
||||||
|
description: |
|
||||||
|
Paste the output of `hyprctl systeminfo -c` here (If you are on a
|
||||||
|
version that shows you help menu, omit the `-c` and attach config files
|
||||||
|
to the issue). If you have configs outside of the main config shown
|
||||||
|
here, please attach.
|
||||||
|
value: "<details>
|
||||||
|
<summary>System/Version info</summary>
|
||||||
|
|
||||||
|
|
||||||
|
```sh
|
||||||
|
|
||||||
|
<Paste the output of the command here>
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
</details>"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: desc
|
||||||
|
attributes:
|
||||||
|
label: Description
|
||||||
|
description: "What went wrong?"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: repro
|
||||||
|
attributes:
|
||||||
|
label: How to reproduce
|
||||||
|
description: "How can someone else reproduce the issue?"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: logs
|
||||||
|
attributes:
|
||||||
|
label: Crash reports, logs, images, videos
|
||||||
|
description: |
|
||||||
|
Anything that can help. Please always ATTACH and not paste them.
|
||||||
|
Logs can be found in $XDG_RUNTIME_DIR/hypr
|
||||||
|
Crash reports are stored in ~/.cache/hyprland or $XDG_CACHE_HOME/hyprland
|
||||||
|
|
||||||
|
19
.github/ISSUE_TEMPLATE/feature.yml
vendored
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
name: Feature Request
|
||||||
|
description: I'd like to request additional functionality
|
||||||
|
labels: ["enhancement"]
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
Before opening a new issue, take a moment to search through the current open ones.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: desc
|
||||||
|
attributes:
|
||||||
|
label: Description
|
||||||
|
description: "Describe your idea"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
12
.github/actions/setup_base/action.yml
vendored
@@ -20,7 +20,6 @@ runs:
|
|||||||
clang \
|
clang \
|
||||||
cmake \
|
cmake \
|
||||||
git \
|
git \
|
||||||
glaze \
|
|
||||||
glm \
|
glm \
|
||||||
glslang \
|
glslang \
|
||||||
go \
|
go \
|
||||||
@@ -34,10 +33,7 @@ runs:
|
|||||||
libfontenc \
|
libfontenc \
|
||||||
libglvnd \
|
libglvnd \
|
||||||
libinput \
|
libinput \
|
||||||
libjxl \
|
|
||||||
libliftoff \
|
libliftoff \
|
||||||
libspng \
|
|
||||||
libwebp \
|
|
||||||
libxcursor \
|
libxcursor \
|
||||||
libxcvt \
|
libxcvt \
|
||||||
libxfont2 \
|
libxfont2 \
|
||||||
@@ -62,8 +58,7 @@ runs:
|
|||||||
xcb-util \
|
xcb-util \
|
||||||
xcb-util-image \
|
xcb-util-image \
|
||||||
libzip \
|
libzip \
|
||||||
librsvg \
|
librsvg
|
||||||
re2
|
|
||||||
|
|
||||||
- name: Get hyprwayland-scanner-git
|
- name: Get hyprwayland-scanner-git
|
||||||
shell: bash
|
shell: bash
|
||||||
@@ -74,11 +69,6 @@ runs:
|
|||||||
cmake --build ./build --config Release --target all -j`nproc 2>/dev/null || getconf NPROCESSORS_CONF`
|
cmake --build ./build --config Release --target all -j`nproc 2>/dev/null || getconf NPROCESSORS_CONF`
|
||||||
cmake --install build
|
cmake --install build
|
||||||
|
|
||||||
- name: Get hyprgraphics-git
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
git clone https://github.com/hyprwm/hyprgraphics && cd hyprgraphics && cmake -DCMAKE_BUILD_TYPE:STRING=Release -DCMAKE_INSTALL_PREFIX:PATH=/usr -B build && cmake --build build --target hyprgraphics && cmake --install build
|
|
||||||
|
|
||||||
- name: Get hyprutils-git
|
- name: Get hyprutils-git
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
|
6
.github/pull_request_template.md
vendored
@@ -1,9 +1,3 @@
|
|||||||
<!--
|
|
||||||
BEFORE you submit your PR, please check out the PR guidelines
|
|
||||||
on our wiki: https://wiki.hyprland.org/Contributing-and-Debugging/PR-Guidelines/
|
|
||||||
-->
|
|
||||||
|
|
||||||
|
|
||||||
#### Describe your PR, what does it fix/add?
|
#### Describe your PR, what does it fix/add?
|
||||||
|
|
||||||
|
|
||||||
|
5
.github/workflows/ci.yaml
vendored
@@ -21,7 +21,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Build Hyprland
|
- name: Build Hyprland
|
||||||
run: |
|
run: |
|
||||||
CFLAGS=-Werror CXXFLAGS=-Werror make all
|
make all
|
||||||
|
|
||||||
- name: Compress and package artifacts
|
- name: Compress and package artifacts
|
||||||
run: |
|
run: |
|
||||||
@@ -39,7 +39,7 @@ jobs:
|
|||||||
tar -cvf Hyprland.tar.xz hyprland
|
tar -cvf Hyprland.tar.xz hyprland
|
||||||
|
|
||||||
- name: Release
|
- name: Release
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: Build archive
|
name: Build archive
|
||||||
path: Hyprland.tar.xz
|
path: Hyprland.tar.xz
|
||||||
@@ -107,7 +107,6 @@ jobs:
|
|||||||
run: make release
|
run: make release
|
||||||
|
|
||||||
clang-format:
|
clang-format:
|
||||||
permissions: read-all
|
|
||||||
if: github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork
|
if: github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork
|
||||||
name: "Code Style (Arch)"
|
name: "Code Style (Arch)"
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
48
.github/workflows/clang-format.yml
vendored
@@ -1,48 +0,0 @@
|
|||||||
name: clang-format
|
|
||||||
on: pull_request_target
|
|
||||||
jobs:
|
|
||||||
clang-format:
|
|
||||||
permissions: write-all
|
|
||||||
if: github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork
|
|
||||||
name: "Code Style (Arch)"
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
container:
|
|
||||||
image: archlinux
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository actions
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
sparse-checkout: .github/actions
|
|
||||||
|
|
||||||
- name: Setup base
|
|
||||||
uses: ./.github/actions/setup_base
|
|
||||||
|
|
||||||
- name: Configure
|
|
||||||
run: meson setup build -Ddefault_library=static
|
|
||||||
|
|
||||||
- name: clang-format check
|
|
||||||
run: ninja -C build clang-format-check
|
|
||||||
|
|
||||||
- name: clang-format apply
|
|
||||||
if: ${{ failure() && github.event_name == 'pull_request' }}
|
|
||||||
run: ninja -C build clang-format
|
|
||||||
|
|
||||||
- name: Create patch
|
|
||||||
if: ${{ failure() && github.event_name == 'pull_request' }}
|
|
||||||
run: |
|
|
||||||
echo 'Please fix the formatting issues by running [`clang-format`](https://wiki.hyprland.org/Contributing-and-Debugging/PR-Guidelines/#code-style), or directly apply this patch:' > clang-format.patch
|
|
||||||
echo '<details>' >> clang-format.patch
|
|
||||||
echo '<summary>clang-format.patch</summary>' >> clang-format.patch
|
|
||||||
echo >> clang-format.patch
|
|
||||||
echo '```diff' >> clang-format.patch
|
|
||||||
git diff >> clang-format.patch
|
|
||||||
echo '```' >> clang-format.patch
|
|
||||||
echo >> clang-format.patch
|
|
||||||
echo '</details>' >> clang-format.patch
|
|
||||||
|
|
||||||
- name: Comment patch
|
|
||||||
if: ${{ failure() && github.event_name == 'pull_request' }}
|
|
||||||
uses: mshick/add-pr-comment@v2
|
|
||||||
with:
|
|
||||||
message-path: |
|
|
||||||
clang-format.patch
|
|
101
.github/workflows/close-issues.yml
vendored
@@ -1,101 +0,0 @@
|
|||||||
name: Close Unauthorized Issues
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
issues:
|
|
||||||
types: [opened]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
close-unauthorized-issues:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
issues: write
|
|
||||||
steps:
|
|
||||||
# XXX: This *could* be done in Bash by abusing GitHub's own tool to interact with its API
|
|
||||||
# but that's too much of a hack, and we'll be adding a layer of abstraction. github-script
|
|
||||||
# is a workflow that eases interaction with GitHub API in the workflow run context.
|
|
||||||
- name: "Close 'unauthorized' issues"
|
|
||||||
uses: actions/github-script@v7
|
|
||||||
with:
|
|
||||||
github-token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
script: |
|
|
||||||
const ALLOWED_USERS = ['vaxerski', 'fufexan', 'NotAShelf'];
|
|
||||||
const CLOSING_COMMENT = 'Users are no longer allowed to open issues themselves, please open a discussion instead.\n\nPlease see the [wiki](https://wiki.hyprland.org/Contributing-and-Debugging/Issue-Guidelines/) on why this is the case.\n\nWe are volunteers, and we need your cooperation to make the best software we can. Thank you for understanding! ❤️\n\n[Open a discussion here](https://github.com/hyprwm/Hyprland/discussions)';
|
|
||||||
|
|
||||||
async function closeUnauthorizedIssue(issueNumber, userName) {
|
|
||||||
if (ALLOWED_USERS.includes(userName)) {
|
|
||||||
console.log(`Issue #${issueNumber} - Created by authorized user ${userName}`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`Issue #${issueNumber} - Unauthorized, closing`);
|
|
||||||
|
|
||||||
await github.rest.issues.update({
|
|
||||||
owner: context.repo.owner,
|
|
||||||
repo: context.repo.repo,
|
|
||||||
issue_number: issueNumber,
|
|
||||||
state: 'closed',
|
|
||||||
state_reason: 'not_planned'
|
|
||||||
});
|
|
||||||
|
|
||||||
await github.rest.issues.createComment({
|
|
||||||
owner: context.repo.owner,
|
|
||||||
repo: context.repo.repo,
|
|
||||||
issue_number: issueNumber,
|
|
||||||
body: CLOSING_COMMENT
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (context.eventName === 'issues' && context.payload.action === 'opened') {
|
|
||||||
// Direct access to the issue that triggered the workflow
|
|
||||||
const issue = context.payload.issue;
|
|
||||||
|
|
||||||
// Skip if this is a PR
|
|
||||||
if (issue.pull_request) {
|
|
||||||
console.log(`Issue #${issue.number} - Skipping, this is a pull request`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Process the single issue that triggered the workflow
|
|
||||||
await closeUnauthorizedIssue(issue.number, issue.user.login);
|
|
||||||
} else {
|
|
||||||
// For manual runs, we need to handle pagination
|
|
||||||
async function* fetchAllOpenIssues() {
|
|
||||||
let page = 1;
|
|
||||||
let hasNextPage = true;
|
|
||||||
|
|
||||||
while (hasNextPage) {
|
|
||||||
const response = await github.rest.issues.listForRepo({
|
|
||||||
owner: context.repo.owner,
|
|
||||||
repo: context.repo.repo,
|
|
||||||
state: 'open',
|
|
||||||
per_page: 100,
|
|
||||||
page: page
|
|
||||||
});
|
|
||||||
|
|
||||||
if (response.data.length === 0) {
|
|
||||||
hasNextPage = false;
|
|
||||||
} else {
|
|
||||||
for (const issue of response.data) {
|
|
||||||
yield issue;
|
|
||||||
}
|
|
||||||
page++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Process issues one by one
|
|
||||||
for await (const issue of fetchAllOpenIssues()) {
|
|
||||||
try {
|
|
||||||
// Skip pull requests
|
|
||||||
if (issue.pull_request) {
|
|
||||||
console.log(`Issue #${issue.number} - Skipping, this is a pull request`);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
await closeUnauthorizedIssue(issue.number, issue.user.login);
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Error processing issue #${issue.number}: ${error.message}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
4
.github/workflows/man-update.yaml
vendored
@@ -17,14 +17,14 @@ jobs:
|
|||||||
run: sudo apt install pandoc
|
run: sudo apt install pandoc
|
||||||
|
|
||||||
- name: Clone repository
|
- name: Clone repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.PAT }}
|
token: ${{ secrets.PAT }}
|
||||||
|
|
||||||
- name: Build man pages
|
- name: Build man pages
|
||||||
run: make man
|
run: make man
|
||||||
|
|
||||||
- uses: stefanzweifel/git-auto-commit-action@v5
|
- uses: stefanzweifel/git-auto-commit-action@v4
|
||||||
name: Commit
|
name: Commit
|
||||||
with:
|
with:
|
||||||
commit_message: "[gha] build man pages"
|
commit_message: "[gha] build man pages"
|
||||||
|
30
.github/workflows/nix-build.yml
vendored
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
secrets:
|
||||||
|
CACHIX_AUTH_TOKEN:
|
||||||
|
required: false
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
package:
|
||||||
|
- hyprland
|
||||||
|
- xdg-desktop-portal-hyprland
|
||||||
|
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Clone repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
ref: ${{ github.ref }}
|
||||||
|
submodules: recursive
|
||||||
|
|
||||||
|
- uses: cachix/install-nix-action@v27
|
||||||
|
- uses: DeterminateSystems/magic-nix-cache-action@main
|
||||||
|
- uses: cachix/cachix-action@v15
|
||||||
|
with:
|
||||||
|
name: hyprland
|
||||||
|
authToken: '${{ secrets.CACHIX_AUTH_TOKEN }}'
|
||||||
|
|
||||||
|
- run: nix build '.?submodules=1#${{ matrix.package }}' -L --extra-substituters "https://hyprland.cachix.org"
|
25
.github/workflows/nix-ci.yml
vendored
@@ -4,27 +4,12 @@ on: [push, pull_request, workflow_dispatch]
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
update-inputs:
|
update-inputs:
|
||||||
if: (github.event_name == 'push' || github.event_name == 'workflow_dispatch')
|
if: github.event_name == 'push' || github.event_name == 'workflow_dispatch'
|
||||||
uses: ./.github/workflows/nix-update-inputs.yml
|
uses: ./.github/workflows/nix-update-inputs.yml
|
||||||
secrets: inherit
|
secrets: inherit
|
||||||
|
|
||||||
hyprland:
|
build:
|
||||||
if: (github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork)
|
if: (github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork) && !contains(needs.*.result, 'failure')
|
||||||
uses: ./.github/workflows/nix.yml
|
needs: update-inputs
|
||||||
secrets: inherit
|
uses: ./.github/workflows/nix-build.yml
|
||||||
with:
|
|
||||||
command: nix build 'github:hyprwm/Hyprland?ref=${{ github.ref }}' -L --extra-substituters "https://hyprland.cachix.org"
|
|
||||||
|
|
||||||
xdph:
|
|
||||||
if: (github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork)
|
|
||||||
needs: hyprland
|
|
||||||
uses: ./.github/workflows/nix.yml
|
|
||||||
secrets: inherit
|
|
||||||
with:
|
|
||||||
command: nix build 'github:hyprwm/Hyprland?ref=${{ github.ref }}#xdg-desktop-portal-hyprland' -L --extra-substituters "https://hyprland.cachix.org"
|
|
||||||
|
|
||||||
test:
|
|
||||||
if: (github.event_name != 'pull_request' || github.event.pull_request.head.repo.fork)
|
|
||||||
needs: hyprland
|
|
||||||
uses: ./.github/workflows/nix-test.yml
|
|
||||||
secrets: inherit
|
secrets: inherit
|
||||||
|
59
.github/workflows/nix-test.yml
vendored
@@ -1,59 +0,0 @@
|
|||||||
name: Nix (Test)
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_call:
|
|
||||||
secrets:
|
|
||||||
CACHIX_AUTH_TOKEN:
|
|
||||||
required: false
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Install Nix
|
|
||||||
uses: nixbuild/nix-quick-install-action@v31
|
|
||||||
with:
|
|
||||||
nix_conf: |
|
|
||||||
keep-env-derivations = true
|
|
||||||
keep-outputs = true
|
|
||||||
|
|
||||||
- name: Restore and save Nix store
|
|
||||||
uses: nix-community/cache-nix-action@v6
|
|
||||||
with:
|
|
||||||
# restore and save a cache using this key
|
|
||||||
primary-key: nix-${{ runner.os }}
|
|
||||||
# if there's no cache hit, restore a cache by this prefix
|
|
||||||
restore-prefixes-first-match: nix-${{ runner.os }}
|
|
||||||
# collect garbage until the Nix store size (in bytes) is at most this number
|
|
||||||
# before trying to save a new cache
|
|
||||||
# 1G = 1073741824
|
|
||||||
gc-max-store-size-linux: 5G
|
|
||||||
# do purge caches
|
|
||||||
purge: true
|
|
||||||
# purge all versions of the cache
|
|
||||||
purge-prefixes: nix-${{ runner.os }}
|
|
||||||
# created more than this number of seconds ago
|
|
||||||
purge-created: 0
|
|
||||||
# or, last accessed more than this number of seconds ago
|
|
||||||
# relative to the start of the `Post Restore and save Nix store` phase
|
|
||||||
purge-last-accessed: 0
|
|
||||||
# except any version with the key that is the same as the `primary-key`
|
|
||||||
purge-primary-key: never
|
|
||||||
|
|
||||||
- uses: cachix/cachix-action@v15
|
|
||||||
with:
|
|
||||||
name: hyprland
|
|
||||||
authToken: "${{ secrets.CACHIX_AUTH_TOKEN }}"
|
|
||||||
|
|
||||||
- name: Run test VM
|
|
||||||
run: nix build 'github:hyprwm/Hyprland?ref=${{ github.ref }}#checks.x86_64-linux.tests' -L --extra-substituters "https://hyprland.cachix.org"
|
|
||||||
|
|
||||||
- name: Check exit status
|
|
||||||
run: grep 0 result/exit_status
|
|
||||||
|
|
||||||
- name: Upload artifacts
|
|
||||||
if: always()
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: logs
|
|
||||||
path: result
|
|
38
.github/workflows/nix-update-inputs.yml
vendored
@@ -1,4 +1,4 @@
|
|||||||
name: Nix (Update Inputs)
|
name: Nix
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_call:
|
workflow_call:
|
||||||
@@ -8,49 +8,19 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
update:
|
update:
|
||||||
if: github.repository == 'hyprwm/Hyprland'
|
|
||||||
name: inputs
|
name: inputs
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Clone repository
|
- name: Clone repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.PAT }}
|
token: ${{ secrets.PAT }}
|
||||||
|
|
||||||
- name: Install Nix
|
- uses: DeterminateSystems/nix-installer-action@main
|
||||||
uses: nixbuild/nix-quick-install-action@v31
|
|
||||||
with:
|
|
||||||
nix_conf: |
|
|
||||||
keep-env-derivations = true
|
|
||||||
keep-outputs = true
|
|
||||||
|
|
||||||
- name: Restore and save Nix store
|
|
||||||
uses: nix-community/cache-nix-action@v6
|
|
||||||
with:
|
|
||||||
# restore and save a cache using this key
|
|
||||||
primary-key: nix-${{ runner.os }}-${{ hashFiles('**/*.nix', '**/flake.lock') }}
|
|
||||||
# if there's no cache hit, restore a cache by this prefix
|
|
||||||
restore-prefixes-first-match: nix-${{ runner.os }}-
|
|
||||||
# collect garbage until the Nix store size (in bytes) is at most this number
|
|
||||||
# before trying to save a new cache
|
|
||||||
# 1G = 1073741824
|
|
||||||
gc-max-store-size-linux: 1G
|
|
||||||
# do purge caches
|
|
||||||
purge: true
|
|
||||||
# purge all versions of the cache
|
|
||||||
purge-prefixes: nix-${{ runner.os }}-
|
|
||||||
# created more than this number of seconds ago
|
|
||||||
purge-created: 0
|
|
||||||
# or, last accessed more than this number of seconds ago
|
|
||||||
# relative to the start of the `Post Restore and save Nix store` phase
|
|
||||||
purge-last-accessed: 0
|
|
||||||
# except any version with the key that is the same as the `primary-key`
|
|
||||||
purge-primary-key: never
|
|
||||||
|
|
||||||
- name: Update inputs
|
- name: Update inputs
|
||||||
run: nix/update-inputs.sh
|
run: nix/update-inputs.sh
|
||||||
|
|
||||||
- name: Commit
|
- name: Commit
|
||||||
uses: stefanzweifel/git-auto-commit-action@v5
|
uses: stefanzweifel/git-auto-commit-action@v4
|
||||||
with:
|
with:
|
||||||
commit_message: "[gha] Nix: update inputs"
|
commit_message: "[gha] Nix: update inputs"
|
||||||
|
53
.github/workflows/nix.yml
vendored
@@ -1,53 +0,0 @@
|
|||||||
name: Build
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_call:
|
|
||||||
inputs:
|
|
||||||
command:
|
|
||||||
required: true
|
|
||||||
type: string
|
|
||||||
description: Command to run
|
|
||||||
secrets:
|
|
||||||
CACHIX_AUTH_TOKEN:
|
|
||||||
required: false
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Install Nix
|
|
||||||
uses: nixbuild/nix-quick-install-action@v31
|
|
||||||
with:
|
|
||||||
nix_conf: |
|
|
||||||
keep-env-derivations = true
|
|
||||||
keep-outputs = true
|
|
||||||
|
|
||||||
- name: Restore and save Nix store
|
|
||||||
uses: nix-community/cache-nix-action@v6
|
|
||||||
with:
|
|
||||||
# restore and save a cache using this key
|
|
||||||
primary-key: nix-${{ runner.os }}
|
|
||||||
# if there's no cache hit, restore a cache by this prefix
|
|
||||||
restore-prefixes-first-match: nix-${{ runner.os }}
|
|
||||||
# collect garbage until the Nix store size (in bytes) is at most this number
|
|
||||||
# before trying to save a new cache
|
|
||||||
# 1G = 1073741824
|
|
||||||
gc-max-store-size-linux: 5G
|
|
||||||
# do purge caches
|
|
||||||
purge: true
|
|
||||||
# purge all versions of the cache
|
|
||||||
purge-prefixes: nix-${{ runner.os }}
|
|
||||||
# created more than this number of seconds ago
|
|
||||||
purge-created: 0
|
|
||||||
# or, last accessed more than this number of seconds ago
|
|
||||||
# relative to the start of the `Post Restore and save Nix store` phase
|
|
||||||
purge-last-accessed: 0
|
|
||||||
# except any version with the key that is the same as the `primary-key`
|
|
||||||
purge-primary-key: never
|
|
||||||
|
|
||||||
- uses: cachix/cachix-action@v15
|
|
||||||
with:
|
|
||||||
name: hyprland
|
|
||||||
authToken: "${{ secrets.CACHIX_AUTH_TOKEN }}"
|
|
||||||
|
|
||||||
- run: ${{ inputs.command }}
|
|
3
.github/workflows/release.yaml
vendored
@@ -11,7 +11,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout Hyprland
|
- name: Checkout Hyprland
|
||||||
id: checkout
|
id: checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: recursive
|
submodules: recursive
|
||||||
|
|
||||||
@@ -20,6 +20,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
git fetch --unshallow || echo "failed unshallowing"
|
git fetch --unshallow || echo "failed unshallowing"
|
||||||
bash -c scripts/generateVersion.sh
|
bash -c scripts/generateVersion.sh
|
||||||
|
mv scripts/generateVersion.sh scripts/generateVersion.sh.bak
|
||||||
|
|
||||||
- name: Create tarball with submodules
|
- name: Create tarball with submodules
|
||||||
id: tar
|
id: tar
|
||||||
|
2
.github/workflows/security-checks.yml
vendored
@@ -13,7 +13,7 @@ jobs:
|
|||||||
security-events: write
|
security-events: write
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
- name: Scan with Flawfinder
|
- name: Scan with Flawfinder
|
||||||
uses: david-a-wheeler/flawfinder@8e4a779ad59dbfaee5da586aa9210853b701959c
|
uses: david-a-wheeler/flawfinder@8e4a779ad59dbfaee5da586aa9210853b701959c
|
||||||
|
28
.github/workflows/stale.yml
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
# This workflow warns and then closes issues and PRs that have had no activity for a specified amount of time.
|
||||||
|
#
|
||||||
|
# You can adjust the behavior by modifying this file.
|
||||||
|
# For more information, see:
|
||||||
|
# https://github.com/actions/stale
|
||||||
|
name: Mark stale issues and pull requests
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '7 */4 * * *'
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
stale:
|
||||||
|
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/stale@v5
|
||||||
|
with:
|
||||||
|
repo-token: ${{ secrets.STALEBOT_PAT }}
|
||||||
|
stale-issue-label: 'stale'
|
||||||
|
stale-pr-label: 'stale'
|
||||||
|
operations-per-run: 40
|
||||||
|
days-before-close: -1
|
6
.gitignore
vendored
@@ -14,7 +14,6 @@ _deps
|
|||||||
|
|
||||||
build/
|
build/
|
||||||
result*
|
result*
|
||||||
/.pre-commit-config.yaml
|
|
||||||
/.vscode/
|
/.vscode/
|
||||||
/.idea/
|
/.idea/
|
||||||
.envrc
|
.envrc
|
||||||
@@ -28,8 +27,6 @@ protocols/*.c*
|
|||||||
protocols/*.h*
|
protocols/*.h*
|
||||||
.ccls-cache
|
.ccls-cache
|
||||||
*.so
|
*.so
|
||||||
src/render/shaders/*.inc
|
|
||||||
src/render/shaders/Shaders.hpp
|
|
||||||
|
|
||||||
hyprctl/hyprctl
|
hyprctl/hyprctl
|
||||||
|
|
||||||
@@ -42,6 +39,3 @@ PKGBUILD
|
|||||||
src/version.h
|
src/version.h
|
||||||
hyprpm/Makefile
|
hyprpm/Makefile
|
||||||
hyprctl/Makefile
|
hyprctl/Makefile
|
||||||
|
|
||||||
**/.#*.*
|
|
||||||
**/#*.*#
|
|
||||||
|
196
CMakeLists.txt
@@ -1,4 +1,4 @@
|
|||||||
cmake_minimum_required(VERSION 3.30)
|
cmake_minimum_required(VERSION 3.27)
|
||||||
|
|
||||||
# Get version
|
# Get version
|
||||||
file(READ "${CMAKE_SOURCE_DIR}/VERSION" VER_RAW)
|
file(READ "${CMAKE_SOURCE_DIR}/VERSION" VER_RAW)
|
||||||
@@ -9,14 +9,12 @@ project(
|
|||||||
DESCRIPTION "A Modern C++ Wayland Compositor"
|
DESCRIPTION "A Modern C++ Wayland Compositor"
|
||||||
VERSION ${VER})
|
VERSION ${VER})
|
||||||
|
|
||||||
include(CTest)
|
|
||||||
include(CheckIncludeFile)
|
include(CheckIncludeFile)
|
||||||
include(GNUInstallDirs)
|
include(GNUInstallDirs)
|
||||||
|
|
||||||
set(HYPRLAND_VERSION ${VER})
|
set(HYPRLAND_VERSION ${VER})
|
||||||
set(PREFIX ${CMAKE_INSTALL_PREFIX})
|
set(PREFIX ${CMAKE_INSTALL_PREFIX})
|
||||||
set(INCLUDEDIR ${CMAKE_INSTALL_INCLUDEDIR})
|
set(INCLUDEDIR ${CMAKE_INSTALL_INCLUDEDIR})
|
||||||
set(BINDIR ${CMAKE_INSTALL_BINDIR})
|
|
||||||
configure_file(hyprland.pc.in hyprland.pc @ONLY)
|
configure_file(hyprland.pc.in hyprland.pc @ONLY)
|
||||||
|
|
||||||
set(CMAKE_MESSAGE_LOG_LEVEL "STATUS")
|
set(CMAKE_MESSAGE_LOG_LEVEL "STATUS")
|
||||||
@@ -26,22 +24,9 @@ message(STATUS "Gathering git info")
|
|||||||
# Get git info hash and branch
|
# Get git info hash and branch
|
||||||
execute_process(COMMAND ./scripts/generateVersion.sh
|
execute_process(COMMAND ./scripts/generateVersion.sh
|
||||||
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
|
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
|
||||||
# Make shader files includable
|
|
||||||
execute_process(COMMAND ./scripts/generateShaderIncludes.sh
|
|
||||||
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
|
|
||||||
|
|
||||||
find_package(PkgConfig REQUIRED)
|
# udis
|
||||||
|
add_subdirectory("subprojects/udis86")
|
||||||
# Try to find canihavesomecoffee's udis86 using pkgconfig vmd/udis86 does not
|
|
||||||
# provide a .pc file and won't be detected this way
|
|
||||||
pkg_check_modules(udis_dep IMPORTED_TARGET udis86>=1.7.2)
|
|
||||||
|
|
||||||
# Fallback to subproject
|
|
||||||
if(NOT udis_dep_FOUND)
|
|
||||||
add_subdirectory("subprojects/udis86")
|
|
||||||
include_directories("subprojects/udis86")
|
|
||||||
message(STATUS "udis86 dependency not found, falling back to subproject")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(CMAKE_BUILD_TYPE)
|
if(CMAKE_BUILD_TYPE)
|
||||||
string(TOLOWER ${CMAKE_BUILD_TYPE} BUILDTYPE_LOWER)
|
string(TOLOWER ${CMAKE_BUILD_TYPE} BUILDTYPE_LOWER)
|
||||||
@@ -62,6 +47,8 @@ else()
|
|||||||
set(BUILDTYPE_LOWER "release")
|
set(BUILDTYPE_LOWER "release")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
find_package(PkgConfig REQUIRED)
|
||||||
|
|
||||||
pkg_get_variable(WAYLAND_PROTOCOLS_DIR wayland-protocols pkgdatadir)
|
pkg_get_variable(WAYLAND_PROTOCOLS_DIR wayland-protocols pkgdatadir)
|
||||||
message(STATUS "Found wayland-protocols at ${WAYLAND_PROTOCOLS_DIR}")
|
message(STATUS "Found wayland-protocols at ${WAYLAND_PROTOCOLS_DIR}")
|
||||||
pkg_get_variable(WAYLAND_SCANNER_PKGDATA_DIR wayland-scanner pkgdatadir)
|
pkg_get_variable(WAYLAND_SCANNER_PKGDATA_DIR wayland-scanner pkgdatadir)
|
||||||
@@ -76,12 +63,8 @@ else()
|
|||||||
message(STATUS "Configuring Hyprland in Release with CMake")
|
message(STATUS "Configuring Hyprland in Release with CMake")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
add_compile_definitions(HYPRLAND_VERSION="${HYPRLAND_VERSION}")
|
include_directories(. "src/" "subprojects/udis86/" "protocols/")
|
||||||
|
|
||||||
include_directories(. "src/" "protocols/")
|
|
||||||
|
|
||||||
set(CMAKE_CXX_STANDARD 26)
|
set(CMAKE_CXX_STANDARD 26)
|
||||||
set(CXX_STANDARD_REQUIRED ON)
|
|
||||||
add_compile_options(
|
add_compile_options(
|
||||||
-Wall
|
-Wall
|
||||||
-Wextra
|
-Wextra
|
||||||
@@ -90,8 +73,6 @@ add_compile_options(
|
|||||||
-Wno-missing-field-initializers
|
-Wno-missing-field-initializers
|
||||||
-Wno-narrowing
|
-Wno-narrowing
|
||||||
-Wno-pointer-arith
|
-Wno-pointer-arith
|
||||||
-Wno-clobbered
|
|
||||||
-Wpedantic
|
|
||||||
-fmacro-prefix-map=${CMAKE_SOURCE_DIR}/=)
|
-fmacro-prefix-map=${CMAKE_SOURCE_DIR}/=)
|
||||||
|
|
||||||
set(CMAKE_EXECUTABLE_ENABLE_EXPORTS TRUE)
|
set(CMAKE_EXECUTABLE_ENABLE_EXPORTS TRUE)
|
||||||
@@ -101,47 +82,43 @@ message(STATUS "Checking deps...")
|
|||||||
|
|
||||||
find_package(Threads REQUIRED)
|
find_package(Threads REQUIRED)
|
||||||
|
|
||||||
set(GLES_VERSION "GLES3")
|
if(LEGACY_RENDERER)
|
||||||
|
set(GLES_VERSION "GLES2")
|
||||||
|
else()
|
||||||
|
set(GLES_VERSION "GLES3")
|
||||||
|
endif()
|
||||||
find_package(OpenGL REQUIRED COMPONENTS ${GLES_VERSION})
|
find_package(OpenGL REQUIRED COMPONENTS ${GLES_VERSION})
|
||||||
|
|
||||||
pkg_check_modules(aquamarine_dep REQUIRED IMPORTED_TARGET aquamarine>=0.9.0)
|
pkg_check_modules(hyprctl_deps REQUIRED IMPORTED_TARGET hyprutils>=0.2.1)
|
||||||
pkg_check_modules(hyprlang_dep REQUIRED IMPORTED_TARGET hyprlang>=0.3.2)
|
|
||||||
pkg_check_modules(hyprcursor_dep REQUIRED IMPORTED_TARGET hyprcursor>=0.1.7)
|
|
||||||
pkg_check_modules(hyprutils_dep REQUIRED IMPORTED_TARGET hyprutils>=0.8.1)
|
|
||||||
pkg_check_modules(hyprgraphics_dep REQUIRED IMPORTED_TARGET hyprgraphics>=0.1.3)
|
|
||||||
|
|
||||||
string(REPLACE "." ";" AQ_VERSION_LIST ${aquamarine_dep_VERSION})
|
|
||||||
list(GET AQ_VERSION_LIST 0 AQ_VERSION_MAJOR)
|
|
||||||
list(GET AQ_VERSION_LIST 1 AQ_VERSION_MINOR)
|
|
||||||
list(GET AQ_VERSION_LIST 2 AQ_VERSION_PATCH)
|
|
||||||
|
|
||||||
add_compile_definitions(AQUAMARINE_VERSION="${aquamarine_dep_VERSION}")
|
|
||||||
add_compile_definitions(AQUAMARINE_VERSION_MAJOR=${AQ_VERSION_MAJOR})
|
|
||||||
add_compile_definitions(AQUAMARINE_VERSION_MINOR=${AQ_VERSION_MINOR})
|
|
||||||
add_compile_definitions(AQUAMARINE_VERSION_PATCH=${AQ_VERSION_PATCH})
|
|
||||||
add_compile_definitions(HYPRLANG_VERSION="${hyprlang_dep_VERSION}")
|
|
||||||
add_compile_definitions(HYPRUTILS_VERSION="${hyprutils_dep_VERSION}")
|
|
||||||
add_compile_definitions(HYPRCURSOR_VERSION="${hyprcursor_dep_VERSION}")
|
|
||||||
add_compile_definitions(HYPRGRAPHICS_VERSION="${hyprgraphics_dep_VERSION}")
|
|
||||||
|
|
||||||
pkg_check_modules(
|
pkg_check_modules(
|
||||||
deps
|
deps
|
||||||
REQUIRED
|
REQUIRED
|
||||||
IMPORTED_TARGET
|
IMPORTED_TARGET
|
||||||
|
aquamarine
|
||||||
xkbcommon
|
xkbcommon
|
||||||
uuid
|
uuid
|
||||||
wayland-server>=1.22.90
|
wayland-server
|
||||||
wayland-protocols>=1.43
|
wayland-client
|
||||||
|
wayland-cursor
|
||||||
|
wayland-protocols
|
||||||
cairo
|
cairo
|
||||||
pango
|
pango
|
||||||
pangocairo
|
pangocairo
|
||||||
pixman-1
|
pixman-1
|
||||||
xcursor
|
xcursor
|
||||||
libdrm
|
libdrm
|
||||||
libinput>=1.28
|
libinput
|
||||||
|
hwdata
|
||||||
|
libseat
|
||||||
|
libdisplay-info
|
||||||
|
libliftoff
|
||||||
|
libudev
|
||||||
gbm
|
gbm
|
||||||
gio-2.0
|
gio-2.0
|
||||||
re2)
|
hyprlang>=0.3.2
|
||||||
|
hyprcursor>=0.1.7
|
||||||
|
hyprutils>=0.2.1)
|
||||||
|
|
||||||
find_package(hyprwayland-scanner 0.3.10 REQUIRED)
|
find_package(hyprwayland-scanner 0.3.10 REQUIRED)
|
||||||
|
|
||||||
@@ -155,7 +132,7 @@ endif()
|
|||||||
|
|
||||||
add_executable(Hyprland ${SRCFILES} ${TRACY_CPP_FILES})
|
add_executable(Hyprland ${SRCFILES} ${TRACY_CPP_FILES})
|
||||||
|
|
||||||
set(USE_GPROF OFF)
|
set(USE_GPROF ON)
|
||||||
|
|
||||||
if(CMAKE_BUILD_TYPE MATCHES Debug OR CMAKE_BUILD_TYPE MATCHES DEBUG)
|
if(CMAKE_BUILD_TYPE MATCHES Debug OR CMAKE_BUILD_TYPE MATCHES DEBUG)
|
||||||
message(STATUS "Setting debug flags")
|
message(STATUS "Setting debug flags")
|
||||||
@@ -208,10 +185,9 @@ if(NOT HAS_TIMERFD AND epoll_FOUND)
|
|||||||
target_link_libraries(Hyprland PkgConfig::epoll)
|
target_link_libraries(Hyprland PkgConfig::epoll)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_include_file("sys/inotify.h" HAS_INOTIFY)
|
if(LEGACY_RENDERER)
|
||||||
pkg_check_modules(inotify IMPORTED_TARGET libinotify)
|
message(STATUS "Using the legacy GLES2 renderer!")
|
||||||
if(NOT HAS_INOTIFY AND inotify_FOUND)
|
add_compile_definitions(LEGACY_RENDERER)
|
||||||
target_link_libraries(Hyprland PkgConfig::inotify)
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(NO_XWAYLAND)
|
if(NO_XWAYLAND)
|
||||||
@@ -224,11 +200,14 @@ else()
|
|||||||
REQUIRED
|
REQUIRED
|
||||||
IMPORTED_TARGET
|
IMPORTED_TARGET
|
||||||
xcb
|
xcb
|
||||||
|
xwayland
|
||||||
|
xcb-util
|
||||||
xcb-render
|
xcb-render
|
||||||
xcb-xfixes
|
xcb-xfixes
|
||||||
xcb-icccm
|
xcb-icccm
|
||||||
xcb-composite
|
xcb-composite
|
||||||
xcb-res
|
xcb-res
|
||||||
|
xcb-ewmh
|
||||||
xcb-errors)
|
xcb-errors)
|
||||||
target_link_libraries(Hyprland PkgConfig::xdeps)
|
target_link_libraries(Hyprland PkgConfig::xdeps)
|
||||||
endif()
|
endif()
|
||||||
@@ -238,15 +217,6 @@ if(NO_SYSTEMD)
|
|||||||
else()
|
else()
|
||||||
message(STATUS "SYSTEMD support is requested (NO_SYSTEMD not defined)...")
|
message(STATUS "SYSTEMD support is requested (NO_SYSTEMD not defined)...")
|
||||||
add_compile_definitions(USES_SYSTEMD)
|
add_compile_definitions(USES_SYSTEMD)
|
||||||
|
|
||||||
# session file -uwsm
|
|
||||||
if(NO_UWSM)
|
|
||||||
message(STATUS "UWSM support is disabled...")
|
|
||||||
else()
|
|
||||||
message(STATUS "UWSM support is enabled (NO_UWSM not defined)...")
|
|
||||||
install(FILES ${CMAKE_SOURCE_DIR}/systemd/hyprland-uwsm.desktop
|
|
||||||
DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/wayland-sessions)
|
|
||||||
endif()
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(CPACK_PROJECT_NAME ${PROJECT_NAME})
|
set(CPACK_PROJECT_NAME ${PROJECT_NAME})
|
||||||
@@ -260,27 +230,14 @@ target_precompile_headers(Hyprland PRIVATE
|
|||||||
|
|
||||||
message(STATUS "Setting link libraries")
|
message(STATUS "Setting link libraries")
|
||||||
|
|
||||||
target_link_libraries(
|
target_link_libraries(Hyprland rt PkgConfig::deps)
|
||||||
Hyprland
|
|
||||||
rt
|
|
||||||
PkgConfig::aquamarine_dep
|
|
||||||
PkgConfig::hyprlang_dep
|
|
||||||
PkgConfig::hyprutils_dep
|
|
||||||
PkgConfig::hyprcursor_dep
|
|
||||||
PkgConfig::hyprgraphics_dep
|
|
||||||
PkgConfig::deps)
|
|
||||||
if(udis_dep_FOUND)
|
|
||||||
target_link_libraries(Hyprland PkgConfig::udis_dep)
|
|
||||||
else()
|
|
||||||
target_link_libraries(Hyprland libudis86)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# used by `make installheaders`, to ensure the headers are generated
|
# used by `make installheaders`, to ensure the headers are generated
|
||||||
add_custom_target(generate-protocol-headers)
|
add_custom_target(generate-protocol-headers)
|
||||||
|
|
||||||
function(protocolnew protoPath protoName external)
|
function(protocolnew protoPath protoName external)
|
||||||
if(external)
|
if(external)
|
||||||
set(path ${protoPath})
|
set(path ${CMAKE_SOURCE_DIR}/${protoPath})
|
||||||
else()
|
else()
|
||||||
set(path ${WAYLAND_PROTOCOLS_DIR}/${protoPath})
|
set(path ${WAYLAND_PROTOCOLS_DIR}/${protoPath})
|
||||||
endif()
|
endif()
|
||||||
@@ -308,22 +265,14 @@ function(protocolWayland)
|
|||||||
PRIVATE ${CMAKE_SOURCE_DIR}/protocols/wayland.hpp)
|
PRIVATE ${CMAKE_SOURCE_DIR}/protocols/wayland.hpp)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
target_link_libraries(Hyprland OpenGL::EGL OpenGL::GL Threads::Threads)
|
target_link_libraries(Hyprland OpenGL::EGL OpenGL::GL Threads::Threads
|
||||||
|
libudis86 uuid)
|
||||||
|
|
||||||
pkg_check_modules(hyprland_protocols_dep hyprland-protocols>=0.6.4)
|
protocolnew("subprojects/hyprland-protocols/protocols"
|
||||||
if(hyprland_protocols_dep_FOUND)
|
"hyprland-global-shortcuts-v1" true)
|
||||||
pkg_get_variable(HYPRLAND_PROTOCOLS hyprland-protocols pkgdatadir)
|
|
||||||
message(STATUS "hyprland-protocols dependency set to ${HYPRLAND_PROTOCOLS}")
|
|
||||||
else()
|
|
||||||
set(HYPRLAND_PROTOCOLS "subprojects/hyprland-protocols")
|
|
||||||
message(STATUS "hyprland-protocols subproject set to ${HYPRLAND_PROTOCOLS}")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-global-shortcuts-v1"
|
|
||||||
true)
|
|
||||||
protocolnew("unstable/text-input" "text-input-unstable-v1" false)
|
protocolnew("unstable/text-input" "text-input-unstable-v1" false)
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-toplevel-export-v1"
|
protocolnew("subprojects/hyprland-protocols/protocols"
|
||||||
true)
|
"hyprland-toplevel-export-v1" true)
|
||||||
protocolnew("protocols" "wlr-screencopy-unstable-v1" true)
|
protocolnew("protocols" "wlr-screencopy-unstable-v1" true)
|
||||||
protocolnew("protocols" "wlr-gamma-control-unstable-v1" true)
|
protocolnew("protocols" "wlr-gamma-control-unstable-v1" true)
|
||||||
protocolnew("protocols" "wlr-foreign-toplevel-management-unstable-v1" true)
|
protocolnew("protocols" "wlr-foreign-toplevel-management-unstable-v1" true)
|
||||||
@@ -334,17 +283,10 @@ protocolnew("protocols" "input-method-unstable-v2" true)
|
|||||||
protocolnew("protocols" "wlr-output-management-unstable-v1" true)
|
protocolnew("protocols" "wlr-output-management-unstable-v1" true)
|
||||||
protocolnew("protocols" "kde-server-decoration" true)
|
protocolnew("protocols" "kde-server-decoration" true)
|
||||||
protocolnew("protocols" "wlr-data-control-unstable-v1" true)
|
protocolnew("protocols" "wlr-data-control-unstable-v1" true)
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-focus-grab-v1" true)
|
protocolnew("subprojects/hyprland-protocols/protocols" "hyprland-focus-grab-v1"
|
||||||
protocolnew("protocols" "wlr-layer-shell-unstable-v1" true)
|
|
||||||
protocolnew("protocols" "xx-color-management-v4" true)
|
|
||||||
protocolnew("protocols" "frog-color-management-v1" true)
|
|
||||||
protocolnew("protocols" "wayland-drm" true)
|
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-ctm-control-v1" true)
|
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-surface-v1" true)
|
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-lock-notify-v1" true)
|
|
||||||
protocolnew("${HYPRLAND_PROTOCOLS}/protocols" "hyprland-toplevel-mapping-v1"
|
|
||||||
true)
|
true)
|
||||||
|
protocolnew("protocols" "wlr-layer-shell-unstable-v1" true)
|
||||||
|
protocolnew("protocols" "wayland-drm" true)
|
||||||
protocolnew("staging/tearing-control" "tearing-control-v1" false)
|
protocolnew("staging/tearing-control" "tearing-control-v1" false)
|
||||||
protocolnew("staging/fractional-scale" "fractional-scale-v1" false)
|
protocolnew("staging/fractional-scale" "fractional-scale-v1" false)
|
||||||
protocolnew("unstable/xdg-output" "xdg-output-unstable-v1" false)
|
protocolnew("unstable/xdg-output" "xdg-output-unstable-v1" false)
|
||||||
@@ -373,32 +315,12 @@ protocolnew("stable/viewporter" "viewporter" false)
|
|||||||
protocolnew("stable/linux-dmabuf" "linux-dmabuf-v1" false)
|
protocolnew("stable/linux-dmabuf" "linux-dmabuf-v1" false)
|
||||||
protocolnew("staging/drm-lease" "drm-lease-v1" false)
|
protocolnew("staging/drm-lease" "drm-lease-v1" false)
|
||||||
protocolnew("staging/linux-drm-syncobj" "linux-drm-syncobj-v1" false)
|
protocolnew("staging/linux-drm-syncobj" "linux-drm-syncobj-v1" false)
|
||||||
protocolnew("staging/xdg-dialog" "xdg-dialog-v1" false)
|
|
||||||
protocolnew("staging/single-pixel-buffer" "single-pixel-buffer-v1" false)
|
|
||||||
protocolnew("staging/security-context" "security-context-v1" false)
|
|
||||||
protocolnew("staging/content-type" "content-type-v1" false)
|
|
||||||
protocolnew("staging/color-management" "color-management-v1" false)
|
|
||||||
protocolnew("staging/xdg-toplevel-tag" "xdg-toplevel-tag-v1" false)
|
|
||||||
protocolnew("staging/xdg-system-bell" "xdg-system-bell-v1" false)
|
|
||||||
protocolnew("staging/ext-workspace" "ext-workspace-v1" false)
|
|
||||||
|
|
||||||
protocolwayland()
|
protocolwayland()
|
||||||
|
|
||||||
# tools
|
# tools
|
||||||
add_subdirectory(hyprctl)
|
add_subdirectory(hyprctl)
|
||||||
|
add_subdirectory(hyprpm)
|
||||||
if(NO_HYPRPM)
|
|
||||||
message(STATUS "hyprpm is disabled")
|
|
||||||
else()
|
|
||||||
add_subdirectory(hyprpm)
|
|
||||||
message(STATUS "hyprpm is enabled (NO_HYPRPM not defined)")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NO_TESTS)
|
|
||||||
message(STATUS "building tests is disabled")
|
|
||||||
else()
|
|
||||||
message(STATUS "building tests is enabled (NO_TESTS not defined)")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# binary and symlink
|
# binary and symlink
|
||||||
install(TARGETS Hyprland)
|
install(TARGETS Hyprland)
|
||||||
@@ -409,18 +331,17 @@ install(
|
|||||||
${CMAKE_INSTALL_FULL_BINDIR}/Hyprland \
|
${CMAKE_INSTALL_FULL_BINDIR}/Hyprland \
|
||||||
\"\$ENV{DESTDIR}${CMAKE_INSTALL_FULL_BINDIR}/hyprland\" \
|
\"\$ENV{DESTDIR}${CMAKE_INSTALL_FULL_BINDIR}/hyprland\" \
|
||||||
)")
|
)")
|
||||||
|
|
||||||
# session file
|
# session file
|
||||||
install(FILES ${CMAKE_SOURCE_DIR}/example/hyprland.desktop
|
install(FILES ${CMAKE_SOURCE_DIR}/example/hyprland.desktop
|
||||||
DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/wayland-sessions)
|
DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/wayland-sessions)
|
||||||
|
|
||||||
# allow Hyprland to find assets
|
# allow Hyprland to find wallpapers
|
||||||
add_compile_definitions(DATAROOTDIR="${CMAKE_INSTALL_FULL_DATAROOTDIR}")
|
add_compile_definitions(DATAROOTDIR="${CMAKE_INSTALL_FULL_DATAROOTDIR}")
|
||||||
|
|
||||||
# installable assets
|
# wallpapers
|
||||||
file(GLOB_RECURSE INSTALLABLE_ASSETS "assets/install/*")
|
file(GLOB_RECURSE WALLPAPERS "assets/wall*")
|
||||||
list(FILTER INSTALLABLE_ASSETS EXCLUDE REGEX "meson.build")
|
install(FILES ${WALLPAPERS} DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/hypr)
|
||||||
install(FILES ${INSTALLABLE_ASSETS}
|
|
||||||
DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/hypr)
|
|
||||||
|
|
||||||
# default config
|
# default config
|
||||||
install(FILES ${CMAKE_SOURCE_DIR}/example/hyprland.conf
|
install(FILES ${CMAKE_SOURCE_DIR}/example/hyprland.conf
|
||||||
@@ -452,19 +373,4 @@ install(
|
|||||||
DIRECTORY ${HEADERS_SRC}
|
DIRECTORY ${HEADERS_SRC}
|
||||||
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/hyprland
|
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/hyprland
|
||||||
FILES_MATCHING
|
FILES_MATCHING
|
||||||
PATTERN "*.h"
|
PATTERN "*.h*")
|
||||||
PATTERN "*.hpp"
|
|
||||||
PATTERN "*.inc")
|
|
||||||
|
|
||||||
if(TESTS)
|
|
||||||
enable_testing()
|
|
||||||
add_custom_target(tests)
|
|
||||||
|
|
||||||
add_subdirectory(hyprtester)
|
|
||||||
add_test(
|
|
||||||
NAME "Main Test"
|
|
||||||
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}/hyprtester
|
|
||||||
COMMAND hyprtester)
|
|
||||||
|
|
||||||
add_dependencies(tests hyprtester)
|
|
||||||
endif()
|
|
||||||
|
27
Makefile
@@ -1,19 +1,28 @@
|
|||||||
PREFIX = /usr/local
|
PREFIX = /usr/local
|
||||||
|
|
||||||
stub:
|
legacyrenderer:
|
||||||
@echo "Do not run $(MAKE) directly without any arguments. Please refer to the wiki on how to compile Hyprland."
|
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Release -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -DLEGACY_RENDERER:BOOL=true -S . -B ./build -G Ninja
|
||||||
|
cmake --build ./build --config Release --target all
|
||||||
|
chmod -R 777 ./build
|
||||||
|
|
||||||
|
legacyrendererdebug:
|
||||||
|
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Debug -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -DLEGACY_RENDERER:BOOL=true -S . -B ./build -G Ninja
|
||||||
|
cmake --build ./build --config Release --target all
|
||||||
|
chmod -R 777 ./build
|
||||||
|
|
||||||
release:
|
release:
|
||||||
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Release -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -S . -B ./build
|
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Release -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -S . -B ./build -G Ninja
|
||||||
cmake --build ./build --config Release --target all -j`nproc 2>/dev/null || getconf NPROCESSORS_CONF`
|
cmake --build ./build --config Release --target all
|
||||||
|
chmod -R 777 ./build
|
||||||
|
|
||||||
debug:
|
debug:
|
||||||
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Debug -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -S . -B ./build
|
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Debug -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -S . -B ./build -G Ninja
|
||||||
cmake --build ./build --config Debug --target all -j`nproc 2>/dev/null || getconf NPROCESSORS_CONF`
|
cmake --build ./build --config Debug --target all
|
||||||
|
chmod -R 777 ./build
|
||||||
|
|
||||||
nopch:
|
nopch:
|
||||||
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Release -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -DCMAKE_DISABLE_PRECOMPILE_HEADERS=ON -S . -B ./build
|
cmake --no-warn-unused-cli -DCMAKE_BUILD_TYPE:STRING=Release -DCMAKE_INSTALL_PREFIX:STRING=${PREFIX} -DCMAKE_DISABLE_PRECOMPILE_HEADERS=ON -S . -B ./build -G Ninja
|
||||||
cmake --build ./build --config Release --target all -j`nproc 2>/dev/null || getconf NPROCESSORS_CONF`
|
cmake --build ./build --config Release --target all
|
||||||
|
|
||||||
clear:
|
clear:
|
||||||
rm -rf build
|
rm -rf build
|
||||||
@@ -44,7 +53,7 @@ installheaders:
|
|||||||
|
|
||||||
cmake --build ./build --config Release --target generate-protocol-headers
|
cmake --build ./build --config Release --target generate-protocol-headers
|
||||||
|
|
||||||
find src -type f \( -name '*.hpp' -o -name '*.h' -o -name '*.inc' \) -print0 | cpio --quiet -0dump ${PREFIX}/include/hyprland
|
find src -name '*.h*' -print0 | cpio --quiet -0dump ${PREFIX}/include/hyprland
|
||||||
cp ./protocols/*.h* ${PREFIX}/include/hyprland/protocols
|
cp ./protocols/*.h* ${PREFIX}/include/hyprland/protocols
|
||||||
cp ./build/hyprland.pc ${PREFIX}/share/pkgconfig
|
cp ./build/hyprland.pc ${PREFIX}/share/pkgconfig
|
||||||
if [ -d /usr/share/pkgconfig ]; then cp ./build/hyprland.pc /usr/share/pkgconfig 2>/dev/null || true; fi
|
if [ -d /usr/share/pkgconfig ]; then cp ./build/hyprland.pc /usr/share/pkgconfig 2>/dev/null || true; fi
|
||||||
|
14
README.md
@@ -100,7 +100,7 @@ easy IPC, much more QoL stuff than other compositors and more...
|
|||||||
|
|
||||||
<!----------------------------------------------------------------------------->
|
<!----------------------------------------------------------------------------->
|
||||||
|
|
||||||
[Configure]: https://wiki.hypr.land/Configuring/
|
[Configure]: https://wiki.hyprland.org/Configuring/Configuring-Hyprland/
|
||||||
[Stars]: https://starchart.cc/hyprwm/Hyprland
|
[Stars]: https://starchart.cc/hyprwm/Hyprland
|
||||||
[Hypr]: https://github.com/hyprwm/Hypr
|
[Hypr]: https://github.com/hyprwm/Hypr
|
||||||
|
|
||||||
@@ -108,9 +108,9 @@ easy IPC, much more QoL stuff than other compositors and more...
|
|||||||
[Issues]: https://github.com/hyprwm/Hyprland/issues
|
[Issues]: https://github.com/hyprwm/Hyprland/issues
|
||||||
[Todo]: https://github.com/hyprwm/Hyprland/projects?type=beta
|
[Todo]: https://github.com/hyprwm/Hyprland/projects?type=beta
|
||||||
|
|
||||||
[Contribute]: https://wiki.hypr.land/Contributing-and-Debugging/
|
[Contribute]: https://wiki.hyprland.org/Contributing-and-Debugging/
|
||||||
[Install]: https://wiki.hypr.land/Getting-Started/Installation/
|
[Install]: https://wiki.hyprland.org/Getting-Started/Installation/
|
||||||
[Quick Start]: https://wiki.hypr.land/Getting-Started/Master-Tutorial/
|
[Quick Start]: https://wiki.hyprland.org/Getting-Started/Master-Tutorial/
|
||||||
[License]: LICENSE
|
[License]: LICENSE
|
||||||
|
|
||||||
|
|
||||||
@@ -125,9 +125,9 @@ easy IPC, much more QoL stuff than other compositors and more...
|
|||||||
|
|
||||||
<!----------------------------------{ Images }--------------------------------->
|
<!----------------------------------{ Images }--------------------------------->
|
||||||
|
|
||||||
[Preview A]: https://i.ibb.co/XxFY75Mk/greerggergerhtrytghjnyhjn.png
|
[Preview A]: https://i.ibb.co/C1yTb0r/falf.png
|
||||||
[Preview B]: https://i.ibb.co/C1yTb0r/falf.png
|
[Preview B]: https://linfindel.github.io/cdn/hyprland-preview-b.png
|
||||||
[Preview C]: https://i.ibb.co/2Yc4q835/hyprland-preview-b.png
|
[Preview C]: https://i.ibb.co/B3GJg28/20221126-20h53m26s-grim.png
|
||||||
|
|
||||||
|
|
||||||
<!----------------------------------{ Badges }--------------------------------->
|
<!----------------------------------{ Badges }--------------------------------->
|
||||||
|
32
SECURITY.md
@@ -1,32 +0,0 @@
|
|||||||
# Hyprland Development Security Policy
|
|
||||||
|
|
||||||
If you have a bug that affects the security of your system, you may
|
|
||||||
want to privately disclose it instead of making it immediately public.
|
|
||||||
|
|
||||||
## Supported versions
|
|
||||||
|
|
||||||
_Only_ the most recent release on Github is supported. There are no LTS releases.
|
|
||||||
|
|
||||||
## What is not a security issue
|
|
||||||
|
|
||||||
Some examples of issues that should not be reported as security issues:
|
|
||||||
|
|
||||||
- An app can execute a command when ran outside of a sandbox
|
|
||||||
- An app can write / read hyprland sockets when ran outside of a sandbox
|
|
||||||
- Crashes
|
|
||||||
- Things that are protected via permissions when the permission system is disabled
|
|
||||||
|
|
||||||
## What is a security issue
|
|
||||||
|
|
||||||
Some examples of issues that should be reported as security issues:
|
|
||||||
|
|
||||||
- Sandboxed application executing arbitrary code via Hyprland
|
|
||||||
- Application being able to modify Hyprland's code on the fly
|
|
||||||
- Application being able to keylog / track user's activity beyond what the wayland protocols allow
|
|
||||||
|
|
||||||
## How to report security issues
|
|
||||||
|
|
||||||
Please report your security issues via either of these channels:
|
|
||||||
- Mail: `vaxry [at] vaxry [dot] net`
|
|
||||||
- Matrix: `@vaxry:matrix.vaxry.net`
|
|
||||||
- Discord: `@vaxry`
|
|
Before Width: | Height: | Size: 6.6 KiB After Width: | Height: | Size: 506 KiB |
Before Width: | Height: | Size: 110 KiB |
Before Width: | Height: | Size: 48 KiB |
@@ -1,10 +0,0 @@
|
|||||||
globber = run_command('sh', '-c', 'find . -type f -not -name "*.build"', check: true)
|
|
||||||
files = globber.stdout().strip().split('\n')
|
|
||||||
|
|
||||||
foreach file : files
|
|
||||||
install_data(
|
|
||||||
file,
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'hypr'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
)
|
|
||||||
endforeach
|
|
@@ -1,7 +1,7 @@
|
|||||||
install_data(
|
wallpapers = ['0', '1', '2']
|
||||||
'hyprland-portals.conf',
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'xdg-desktop-portal'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
)
|
|
||||||
|
|
||||||
subdir('install')
|
foreach type : wallpapers
|
||||||
|
install_data(f'wall@type@.png', install_dir: join_paths(get_option('datadir'), 'hypr'), install_tag: 'runtime')
|
||||||
|
endforeach
|
||||||
|
|
||||||
|
install_data('hyprland-portals.conf', install_dir: join_paths(get_option('datadir'), 'xdg-desktop-portal'), install_tag: 'runtime')
|
||||||
|
Before Width: | Height: | Size: 14 MiB After Width: | Height: | Size: 14 MiB |
Before Width: | Height: | Size: 5.9 MiB After Width: | Height: | Size: 5.9 MiB |
Before Width: | Height: | Size: 27 MiB After Width: | Height: | Size: 27 MiB |
@@ -1,19 +1,5 @@
|
|||||||
.\" Automatically generated by Pandoc 3.1.3
|
.\" Automatically generated by Pandoc 2.9.2.1
|
||||||
.\"
|
.\"
|
||||||
.\" Define V font for inline verbatim, using C font in formats
|
|
||||||
.\" that render this, and otherwise B font.
|
|
||||||
.ie "\f[CB]x\f[]"x" \{\
|
|
||||||
. ftr V B
|
|
||||||
. ftr VI BI
|
|
||||||
. ftr VB B
|
|
||||||
. ftr VBI BI
|
|
||||||
.\}
|
|
||||||
.el \{\
|
|
||||||
. ftr V CR
|
|
||||||
. ftr VI CI
|
|
||||||
. ftr VB CB
|
|
||||||
. ftr VBI CBI
|
|
||||||
.\}
|
|
||||||
.TH "Hyprland" "1" "" "" "Hyprland User Manual"
|
.TH "Hyprland" "1" "" "" "Hyprland User Manual"
|
||||||
.hy
|
.hy
|
||||||
.SH NAME
|
.SH NAME
|
||||||
@@ -24,8 +10,8 @@ Hyprland - Dynamic tiling Wayland compositor
|
|||||||
\f[B]Hyprland\f[R] [\f[I]arg [...]\f[R]].
|
\f[B]Hyprland\f[R] [\f[I]arg [...]\f[R]].
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
.PP
|
.PP
|
||||||
\f[B]Hyprland\f[R] is an independent, highly customizable, dynamic
|
\f[B]Hyprland\f[R] is a dynamic tiling Wayland compositor based on
|
||||||
tiling Wayland compositor that doesn\[aq]t sacrifice on its looks.
|
wlroots that doesn\[aq]t sacrifice on its looks.
|
||||||
.PP
|
.PP
|
||||||
You can launch Hyprland by either going into a TTY and executing
|
You can launch Hyprland by either going into a TTY and executing
|
||||||
\f[B]Hyprland\f[R], or with a login manager.
|
\f[B]Hyprland\f[R], or with a login manager.
|
||||||
|
@@ -14,8 +14,8 @@ SYNOPSIS
|
|||||||
DESCRIPTION
|
DESCRIPTION
|
||||||
===========
|
===========
|
||||||
|
|
||||||
**Hyprland** is an independent, highly customizable,
|
**Hyprland** is a dynamic tiling Wayland compositor based on
|
||||||
dynamic tiling Wayland compositor that doesn't sacrifice on its looks.
|
wlroots that doesn't sacrifice on its looks.
|
||||||
|
|
||||||
You can launch Hyprland by either going into a TTY and
|
You can launch Hyprland by either going into a TTY and
|
||||||
executing **Hyprland**, or with a login manager.
|
executing **Hyprland**, or with a login manager.
|
||||||
|
@@ -2,15 +2,15 @@
|
|||||||
|
|
||||||
First of all, please remember to:
|
First of all, please remember to:
|
||||||
- Check that your issue is not a duplicate
|
- Check that your issue is not a duplicate
|
||||||
- Read the [FAQ](https://wiki.hypr.land/FAQ/)
|
- Read the [FAQ](https://wiki.hyprland.org/FAQ/)
|
||||||
- Read the [Configuring Page](https://wiki.hypr.land/Configuring/)
|
- Read the [Configuring Page](https://wiki.hyprland.org/Configuring/Configuring-Hyprland)
|
||||||
|
|
||||||
<br/>
|
<br/>
|
||||||
|
|
||||||
# Reporting suggestions
|
# Reporting suggestions
|
||||||
Suggestions are welcome.
|
Suggestions are welcome.
|
||||||
|
|
||||||
Many features can be implemented using bash scripts and Hyprland sockets, read up on those [Here](https://wiki.hypr.land/IPC). Please do not suggest features that can be implemented as such.
|
Many features can be implemented using bash scripts and Hyprland sockets, read up on those [Here](https://wiki.hyprland.org/IPC). Please do not suggest features that can be implemented as such.
|
||||||
|
|
||||||
<br/>
|
<br/>
|
||||||
|
|
||||||
@@ -70,7 +70,7 @@ A debug coredump provides more information for debugging and may speed up the pr
|
|||||||
|
|
||||||
Make sure you're on latest git. Run `git pull --recurse-submodules` to sync everything.
|
Make sure you're on latest git. Run `git pull --recurse-submodules` to sync everything.
|
||||||
|
|
||||||
1. [Compile Hyprland with debug mode](http://wiki.hypr.land/Contributing-and-Debugging/#build-in-debug-mode)
|
1. [Compile Hyprland with debug mode](http://wiki.hyprland.org/Contributing-and-Debugging/#build-in-debug-mode)
|
||||||
> Note: The config file used will be `hyprlandd.conf` instead of `hyprland.conf`
|
> Note: The config file used will be `hyprlandd.conf` instead of `hyprland.conf`
|
||||||
|
|
||||||
2. `cd ~`
|
2. `cd ~`
|
||||||
|
@@ -1,19 +1,5 @@
|
|||||||
.\" Automatically generated by Pandoc 3.1.3
|
.\" Automatically generated by Pandoc 2.9.2.1
|
||||||
.\"
|
.\"
|
||||||
.\" Define V font for inline verbatim, using C font in formats
|
|
||||||
.\" that render this, and otherwise B font.
|
|
||||||
.ie "\f[CB]x\f[]"x" \{\
|
|
||||||
. ftr V B
|
|
||||||
. ftr VI BI
|
|
||||||
. ftr VB B
|
|
||||||
. ftr VBI BI
|
|
||||||
.\}
|
|
||||||
.el \{\
|
|
||||||
. ftr V CR
|
|
||||||
. ftr VI CI
|
|
||||||
. ftr VB CB
|
|
||||||
. ftr VBI CBI
|
|
||||||
.\}
|
|
||||||
.TH "hyprctl" "1" "" "" "hyprctl User Manual"
|
.TH "hyprctl" "1" "" "" "hyprctl User Manual"
|
||||||
.hy
|
.hy
|
||||||
.SH NAME
|
.SH NAME
|
||||||
|
@@ -1,2 +1,2 @@
|
|||||||
install_man('Hyprland.1')
|
install_man ('Hyprland.1')
|
||||||
install_man('hyprctl.1')
|
install_man ('hyprctl.1')
|
||||||
|
14
example/hyprland-session.service
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Hyprland - Tiling compositor with the looks
|
||||||
|
Documentation=man:Hyprland(1)
|
||||||
|
BindsTo=graphical-session.target
|
||||||
|
Before=graphical-session.target
|
||||||
|
Wants=graphical-session-pre.target
|
||||||
|
After=graphical-session-pre.target
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=notify
|
||||||
|
ExecStart=/usr/bin/Hyprland
|
||||||
|
ExecStop=/usr/bin/hyprctl dispatch exit
|
||||||
|
Restart=on-failure
|
||||||
|
Slice=session.slice
|
5
example/hyprland-systemd.desktop
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
[Desktop Entry]
|
||||||
|
Name=Hyprland
|
||||||
|
Comment=An intelligent dynamic tiling Wayland compositor
|
||||||
|
Exec=systemctl --user start --wait hyprland-session
|
||||||
|
Type=Application
|
@@ -1,6 +1,6 @@
|
|||||||
# This is an example Hyprland config file.
|
# This is an example Hyprland config file.
|
||||||
# Refer to the wiki for more information.
|
# Refer to the wiki for more information.
|
||||||
# https://wiki.hypr.land/Configuring/
|
# https://wiki.hyprland.org/Configuring/Configuring-Hyprland/
|
||||||
|
|
||||||
# Please note not all available settings / options are set here.
|
# Please note not all available settings / options are set here.
|
||||||
# For a full list, see the wiki
|
# For a full list, see the wiki
|
||||||
@@ -14,7 +14,7 @@
|
|||||||
### MONITORS ###
|
### MONITORS ###
|
||||||
################
|
################
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Monitors/
|
# See https://wiki.hyprland.org/Configuring/Monitors/
|
||||||
monitor=,preferred,auto,auto
|
monitor=,preferred,auto,auto
|
||||||
|
|
||||||
|
|
||||||
@@ -22,7 +22,7 @@ monitor=,preferred,auto,auto
|
|||||||
### MY PROGRAMS ###
|
### MY PROGRAMS ###
|
||||||
###################
|
###################
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Keywords/
|
# See https://wiki.hyprland.org/Configuring/Keywords/
|
||||||
|
|
||||||
# Set programs that you use
|
# Set programs that you use
|
||||||
$terminal = kitty
|
$terminal = kitty
|
||||||
@@ -46,134 +46,90 @@ $menu = wofi --show drun
|
|||||||
### ENVIRONMENT VARIABLES ###
|
### ENVIRONMENT VARIABLES ###
|
||||||
#############################
|
#############################
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Environment-variables/
|
# See https://wiki.hyprland.org/Configuring/Environment-variables/
|
||||||
|
|
||||||
env = XCURSOR_SIZE,24
|
env = XCURSOR_SIZE,24
|
||||||
env = HYPRCURSOR_SIZE,24
|
env = HYPRCURSOR_SIZE,24
|
||||||
|
|
||||||
|
|
||||||
###################
|
|
||||||
### PERMISSIONS ###
|
|
||||||
###################
|
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Permissions/
|
|
||||||
# Please note permission changes here require a Hyprland restart and are not applied on-the-fly
|
|
||||||
# for security reasons
|
|
||||||
|
|
||||||
# ecosystem {
|
|
||||||
# enforce_permissions = 1
|
|
||||||
# }
|
|
||||||
|
|
||||||
# permission = /usr/(bin|local/bin)/grim, screencopy, allow
|
|
||||||
# permission = /usr/(lib|libexec|lib64)/xdg-desktop-portal-hyprland, screencopy, allow
|
|
||||||
# permission = /usr/(bin|local/bin)/hyprpm, plugin, allow
|
|
||||||
|
|
||||||
|
|
||||||
#####################
|
#####################
|
||||||
### LOOK AND FEEL ###
|
### LOOK AND FEEL ###
|
||||||
#####################
|
#####################
|
||||||
|
|
||||||
# Refer to https://wiki.hypr.land/Configuring/Variables/
|
# Refer to https://wiki.hyprland.org/Configuring/Variables/
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#general
|
# https://wiki.hyprland.org/Configuring/Variables/#general
|
||||||
general {
|
general {
|
||||||
gaps_in = 5
|
gaps_in = 5
|
||||||
gaps_out = 20
|
gaps_out = 20
|
||||||
|
|
||||||
border_size = 2
|
border_size = 2
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#variable-types for info about colors
|
# https://wiki.hyprland.org/Configuring/Variables/#variable-types for info about colors
|
||||||
col.active_border = rgba(33ccffee) rgba(00ff99ee) 45deg
|
col.active_border = rgba(33ccffee) rgba(00ff99ee) 45deg
|
||||||
col.inactive_border = rgba(595959aa)
|
col.inactive_border = rgba(595959aa)
|
||||||
|
|
||||||
# Set to true enable resizing windows by clicking and dragging on borders and gaps
|
# Set to true enable resizing windows by clicking and dragging on borders and gaps
|
||||||
resize_on_border = false
|
resize_on_border = false
|
||||||
|
|
||||||
# Please see https://wiki.hypr.land/Configuring/Tearing/ before you turn this on
|
# Please see https://wiki.hyprland.org/Configuring/Tearing/ before you turn this on
|
||||||
allow_tearing = false
|
allow_tearing = false
|
||||||
|
|
||||||
layout = dwindle
|
layout = dwindle
|
||||||
}
|
}
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#decoration
|
# https://wiki.hyprland.org/Configuring/Variables/#decoration
|
||||||
decoration {
|
decoration {
|
||||||
rounding = 10
|
rounding = 10
|
||||||
rounding_power = 2
|
|
||||||
|
|
||||||
# Change transparency of focused and unfocused windows
|
# Change transparency of focused and unfocused windows
|
||||||
active_opacity = 1.0
|
active_opacity = 1.0
|
||||||
inactive_opacity = 1.0
|
inactive_opacity = 1.0
|
||||||
|
|
||||||
shadow {
|
drop_shadow = true
|
||||||
enabled = true
|
shadow_range = 4
|
||||||
range = 4
|
shadow_render_power = 3
|
||||||
render_power = 3
|
col.shadow = rgba(1a1a1aee)
|
||||||
color = rgba(1a1a1aee)
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#blur
|
# https://wiki.hyprland.org/Configuring/Variables/#blur
|
||||||
blur {
|
blur {
|
||||||
enabled = true
|
enabled = true
|
||||||
size = 3
|
size = 3
|
||||||
passes = 1
|
passes = 1
|
||||||
|
|
||||||
vibrancy = 0.1696
|
vibrancy = 0.1696
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#animations
|
# https://wiki.hyprland.org/Configuring/Variables/#animations
|
||||||
animations {
|
animations {
|
||||||
enabled = yes, please :)
|
enabled = true
|
||||||
|
|
||||||
# Default animations, see https://wiki.hypr.land/Configuring/Animations/ for more
|
# Default animations, see https://wiki.hyprland.org/Configuring/Animations/ for more
|
||||||
|
|
||||||
bezier = easeOutQuint,0.23,1,0.32,1
|
bezier = myBezier, 0.05, 0.9, 0.1, 1.05
|
||||||
bezier = easeInOutCubic,0.65,0.05,0.36,1
|
|
||||||
bezier = linear,0,0,1,1
|
|
||||||
bezier = almostLinear,0.5,0.5,0.75,1.0
|
|
||||||
bezier = quick,0.15,0,0.1,1
|
|
||||||
|
|
||||||
animation = global, 1, 10, default
|
animation = windows, 1, 7, myBezier
|
||||||
animation = border, 1, 5.39, easeOutQuint
|
animation = windowsOut, 1, 7, default, popin 80%
|
||||||
animation = windows, 1, 4.79, easeOutQuint
|
animation = border, 1, 10, default
|
||||||
animation = windowsIn, 1, 4.1, easeOutQuint, popin 87%
|
animation = borderangle, 1, 8, default
|
||||||
animation = windowsOut, 1, 1.49, linear, popin 87%
|
animation = fade, 1, 7, default
|
||||||
animation = fadeIn, 1, 1.73, almostLinear
|
animation = workspaces, 1, 6, default
|
||||||
animation = fadeOut, 1, 1.46, almostLinear
|
|
||||||
animation = fade, 1, 3.03, quick
|
|
||||||
animation = layers, 1, 3.81, easeOutQuint
|
|
||||||
animation = layersIn, 1, 4, easeOutQuint, fade
|
|
||||||
animation = layersOut, 1, 1.5, linear, fade
|
|
||||||
animation = fadeLayersIn, 1, 1.79, almostLinear
|
|
||||||
animation = fadeLayersOut, 1, 1.39, almostLinear
|
|
||||||
animation = workspaces, 1, 1.94, almostLinear, fade
|
|
||||||
animation = workspacesIn, 1, 1.21, almostLinear, fade
|
|
||||||
animation = workspacesOut, 1, 1.94, almostLinear, fade
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# Ref https://wiki.hypr.land/Configuring/Workspace-Rules/
|
# See https://wiki.hyprland.org/Configuring/Dwindle-Layout/ for more
|
||||||
# "Smart gaps" / "No gaps when only"
|
|
||||||
# uncomment all if you wish to use that.
|
|
||||||
# workspace = w[tv1], gapsout:0, gapsin:0
|
|
||||||
# workspace = f[1], gapsout:0, gapsin:0
|
|
||||||
# windowrule = bordersize 0, floating:0, onworkspace:w[tv1]
|
|
||||||
# windowrule = rounding 0, floating:0, onworkspace:w[tv1]
|
|
||||||
# windowrule = bordersize 0, floating:0, onworkspace:f[1]
|
|
||||||
# windowrule = rounding 0, floating:0, onworkspace:f[1]
|
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Dwindle-Layout/ for more
|
|
||||||
dwindle {
|
dwindle {
|
||||||
pseudotile = true # Master switch for pseudotiling. Enabling is bound to mainMod + P in the keybinds section below
|
pseudotile = true # Master switch for pseudotiling. Enabling is bound to mainMod + P in the keybinds section below
|
||||||
preserve_split = true # You probably want this
|
preserve_split = true # You probably want this
|
||||||
}
|
}
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Master-Layout/ for more
|
# See https://wiki.hyprland.org/Configuring/Master-Layout/ for more
|
||||||
master {
|
master {
|
||||||
new_status = master
|
new_status = master
|
||||||
}
|
}
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#misc
|
# https://wiki.hyprland.org/Configuring/Variables/#misc
|
||||||
misc {
|
misc {
|
||||||
force_default_wallpaper = -1 # Set to 0 or 1 to disable the anime mascot wallpapers
|
force_default_wallpaper = -1 # Set to 0 or 1 to disable the anime mascot wallpapers
|
||||||
disable_hyprland_logo = false # If true disables the random hyprland logo / anime girl background. :(
|
disable_hyprland_logo = false # If true disables the random hyprland logo / anime girl background. :(
|
||||||
}
|
}
|
||||||
@@ -183,7 +139,7 @@ misc {
|
|||||||
### INPUT ###
|
### INPUT ###
|
||||||
#############
|
#############
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#input
|
# https://wiki.hyprland.org/Configuring/Variables/#input
|
||||||
input {
|
input {
|
||||||
kb_layout = us
|
kb_layout = us
|
||||||
kb_variant =
|
kb_variant =
|
||||||
@@ -200,13 +156,13 @@ input {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# https://wiki.hypr.land/Configuring/Variables/#gestures
|
# https://wiki.hyprland.org/Configuring/Variables/#gestures
|
||||||
gestures {
|
gestures {
|
||||||
workspace_swipe = false
|
workspace_swipe = false
|
||||||
}
|
}
|
||||||
|
|
||||||
# Example per-device config
|
# Example per-device config
|
||||||
# See https://wiki.hypr.land/Configuring/Keywords/#per-device-input-configs for more
|
# See https://wiki.hyprland.org/Configuring/Keywords/#per-device-input-configs for more
|
||||||
device {
|
device {
|
||||||
name = epic-mouse-v1
|
name = epic-mouse-v1
|
||||||
sensitivity = -0.5
|
sensitivity = -0.5
|
||||||
@@ -217,10 +173,10 @@ device {
|
|||||||
### KEYBINDINGS ###
|
### KEYBINDINGS ###
|
||||||
###################
|
###################
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Keywords/
|
# See https://wiki.hyprland.org/Configuring/Keywords/
|
||||||
$mainMod = SUPER # Sets "Windows" key as main modifier
|
$mainMod = SUPER # Sets "Windows" key as main modifier
|
||||||
|
|
||||||
# Example binds, see https://wiki.hypr.land/Configuring/Binds/ for more
|
# Example binds, see https://wiki.hyprland.org/Configuring/Binds/ for more
|
||||||
bind = $mainMod, Q, exec, $terminal
|
bind = $mainMod, Q, exec, $terminal
|
||||||
bind = $mainMod, C, killactive,
|
bind = $mainMod, C, killactive,
|
||||||
bind = $mainMod, M, exit,
|
bind = $mainMod, M, exit,
|
||||||
@@ -272,32 +228,18 @@ bind = $mainMod, mouse_up, workspace, e-1
|
|||||||
bindm = $mainMod, mouse:272, movewindow
|
bindm = $mainMod, mouse:272, movewindow
|
||||||
bindm = $mainMod, mouse:273, resizewindow
|
bindm = $mainMod, mouse:273, resizewindow
|
||||||
|
|
||||||
# Laptop multimedia keys for volume and LCD brightness
|
|
||||||
bindel = ,XF86AudioRaiseVolume, exec, wpctl set-volume -l 1 @DEFAULT_AUDIO_SINK@ 5%+
|
|
||||||
bindel = ,XF86AudioLowerVolume, exec, wpctl set-volume @DEFAULT_AUDIO_SINK@ 5%-
|
|
||||||
bindel = ,XF86AudioMute, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ toggle
|
|
||||||
bindel = ,XF86AudioMicMute, exec, wpctl set-mute @DEFAULT_AUDIO_SOURCE@ toggle
|
|
||||||
bindel = ,XF86MonBrightnessUp, exec, brightnessctl -e4 -n2 set 5%+
|
|
||||||
bindel = ,XF86MonBrightnessDown, exec, brightnessctl -e4 -n2 set 5%-
|
|
||||||
|
|
||||||
# Requires playerctl
|
|
||||||
bindl = , XF86AudioNext, exec, playerctl next
|
|
||||||
bindl = , XF86AudioPause, exec, playerctl play-pause
|
|
||||||
bindl = , XF86AudioPlay, exec, playerctl play-pause
|
|
||||||
bindl = , XF86AudioPrev, exec, playerctl previous
|
|
||||||
|
|
||||||
##############################
|
##############################
|
||||||
### WINDOWS AND WORKSPACES ###
|
### WINDOWS AND WORKSPACES ###
|
||||||
##############################
|
##############################
|
||||||
|
|
||||||
# See https://wiki.hypr.land/Configuring/Window-Rules/ for more
|
# See https://wiki.hyprland.org/Configuring/Window-Rules/ for more
|
||||||
# See https://wiki.hypr.land/Configuring/Workspace-Rules/ for workspace rules
|
# See https://wiki.hyprland.org/Configuring/Workspace-Rules/ for workspace rules
|
||||||
|
|
||||||
# Example windowrule
|
# Example windowrule v1
|
||||||
# windowrule = float,class:^(kitty)$,title:^(kitty)$
|
# windowrule = float, ^(kitty)$
|
||||||
|
|
||||||
# Ignore maximize requests from apps. You'll probably like this.
|
# Example windowrule v2
|
||||||
windowrule = suppressevent maximize, class:.*
|
# windowrulev2 = float,class:^(kitty)$,title:^(kitty)$
|
||||||
|
|
||||||
# Fix some dragging issues with XWayland
|
windowrulev2 = suppressevent maximize, class:.* # You'll probably like this.
|
||||||
windowrule = nofocus,class:^$,title:^$,xwayland:1,floating:1,fullscreen:0,pinned:0
|
|
||||||
|
@@ -2,6 +2,4 @@
|
|||||||
Name=Hyprland
|
Name=Hyprland
|
||||||
Comment=An intelligent dynamic tiling Wayland compositor
|
Comment=An intelligent dynamic tiling Wayland compositor
|
||||||
Exec=Hyprland
|
Exec=Hyprland
|
||||||
Type=Application
|
Type=Application
|
||||||
DesktopNames=Hyprland
|
|
||||||
Keywords=tiling;wayland;compositor;
|
|
@@ -22,6 +22,5 @@
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
||||||
]
|
]
|
||||||
}
|
}
|
@@ -1,10 +1,2 @@
|
|||||||
install_data(
|
install_data('hyprland.conf', install_dir: join_paths(get_option('datadir'), 'hypr'), install_tag: 'runtime')
|
||||||
'hyprland.conf',
|
install_data('hyprland.desktop', install_dir: join_paths(get_option('datadir'), 'wayland-sessions'), install_tag: 'runtime')
|
||||||
install_dir: join_paths(get_option('datadir'), 'hypr'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
)
|
|
||||||
install_data(
|
|
||||||
'hyprland.desktop',
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'wayland-sessions'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
)
|
|
||||||
|
210
flake.lock
generated
@@ -16,11 +16,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1752936381,
|
"lastModified": 1724273991,
|
||||||
"narHash": "sha256-b191B12GRfvOT3odGpx5IFyGRPZbBrvCLADZfFHoJFg=",
|
"narHash": "sha256-+aUSOXKGpS5CRm1oTitgNAr05ThQNbKIXalZHl3nC6Y=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "aquamarine",
|
"repo": "aquamarine",
|
||||||
"rev": "141a991678b34e768f09b3a670c61a4c1d5d7110",
|
"rev": "9a3161ad4c78dc420d1cbb3aae638222608c7de4",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -29,43 +29,6 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"flake-compat": {
|
|
||||||
"flake": false,
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1696426674,
|
|
||||||
"narHash": "sha256-kvjfFW7WAETZlt09AgDn1MrtKzP7t90Vf7vypd3OL1U=",
|
|
||||||
"owner": "edolstra",
|
|
||||||
"repo": "flake-compat",
|
|
||||||
"rev": "0f9255e01c2351cc7d116c072cb317785dd33b33",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "edolstra",
|
|
||||||
"repo": "flake-compat",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"gitignore": {
|
|
||||||
"inputs": {
|
|
||||||
"nixpkgs": [
|
|
||||||
"pre-commit-hooks",
|
|
||||||
"nixpkgs"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1709087332,
|
|
||||||
"narHash": "sha256-HG2cCnktfHsKV0s4XW83gU3F57gaTljL9KNSuG6bnQs=",
|
|
||||||
"owner": "hercules-ci",
|
|
||||||
"repo": "gitignore.nix",
|
|
||||||
"rev": "637db329424fd7e46cf4185293b9cc8c88c95394",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "hercules-ci",
|
|
||||||
"repo": "gitignore.nix",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"hyprcursor": {
|
"hyprcursor": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"hyprlang": [
|
"hyprlang": [
|
||||||
@@ -79,11 +42,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1749155331,
|
"lastModified": 1722623071,
|
||||||
"narHash": "sha256-XR9fsI0zwLiFWfqi/pdS/VD+YNorKb3XIykgTg4l1nA=",
|
"narHash": "sha256-sLADpVgebpCBFXkA1FlCXtvEPu1tdEsTfqK1hfeHySE=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "hyprcursor",
|
"repo": "hyprcursor",
|
||||||
"rev": "45fcc10b4c282746d93ec406a740c43b48b4ef80",
|
"rev": "912d56025f03d41b1ad29510c423757b4379eb1c",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -92,47 +55,23 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"hyprgraphics": {
|
|
||||||
"inputs": {
|
|
||||||
"hyprutils": [
|
|
||||||
"hyprutils"
|
|
||||||
],
|
|
||||||
"nixpkgs": [
|
|
||||||
"nixpkgs"
|
|
||||||
],
|
|
||||||
"systems": [
|
|
||||||
"systems"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1752149140,
|
|
||||||
"narHash": "sha256-gbh1HL98Fdqu0jJIWN4OJQN7Kkth7+rbkFpSZLm/62A=",
|
|
||||||
"owner": "hyprwm",
|
|
||||||
"repo": "hyprgraphics",
|
|
||||||
"rev": "340494a38b5ec453dfc542c6226481f736cc8a9a",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "hyprwm",
|
|
||||||
"repo": "hyprgraphics",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"hyprland-protocols": {
|
"hyprland-protocols": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"nixpkgs": [
|
"nixpkgs": [
|
||||||
|
"xdph",
|
||||||
"nixpkgs"
|
"nixpkgs"
|
||||||
],
|
],
|
||||||
"systems": [
|
"systems": [
|
||||||
|
"xdph",
|
||||||
"systems"
|
"systems"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1749046714,
|
"lastModified": 1721326555,
|
||||||
"narHash": "sha256-kymV5FMnddYGI+UjwIw8ceDjdeg7ToDVjbHCvUlhn14=",
|
"narHash": "sha256-zCu4R0CSHEactW9JqYki26gy8h9f6rHmSwj4XJmlHgg=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "hyprland-protocols",
|
"repo": "hyprland-protocols",
|
||||||
"rev": "613878cb6f459c5e323aaafe1e6f388ac8a36330",
|
"rev": "5a11232266bf1a1f5952d5b179c3f4b2facaaa84",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -141,67 +80,6 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"hyprland-qt-support": {
|
|
||||||
"inputs": {
|
|
||||||
"hyprlang": [
|
|
||||||
"hyprland-qtutils",
|
|
||||||
"hyprlang"
|
|
||||||
],
|
|
||||||
"nixpkgs": [
|
|
||||||
"hyprland-qtutils",
|
|
||||||
"nixpkgs"
|
|
||||||
],
|
|
||||||
"systems": [
|
|
||||||
"hyprland-qtutils",
|
|
||||||
"systems"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1749154592,
|
|
||||||
"narHash": "sha256-DO7z5CeT/ddSGDEnK9mAXm1qlGL47L3VAHLlLXoCjhE=",
|
|
||||||
"owner": "hyprwm",
|
|
||||||
"repo": "hyprland-qt-support",
|
|
||||||
"rev": "4c8053c3c888138a30c3a6c45c2e45f5484f2074",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "hyprwm",
|
|
||||||
"repo": "hyprland-qt-support",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"hyprland-qtutils": {
|
|
||||||
"inputs": {
|
|
||||||
"hyprland-qt-support": "hyprland-qt-support",
|
|
||||||
"hyprlang": [
|
|
||||||
"hyprlang"
|
|
||||||
],
|
|
||||||
"hyprutils": [
|
|
||||||
"hyprland-qtutils",
|
|
||||||
"hyprlang",
|
|
||||||
"hyprutils"
|
|
||||||
],
|
|
||||||
"nixpkgs": [
|
|
||||||
"nixpkgs"
|
|
||||||
],
|
|
||||||
"systems": [
|
|
||||||
"systems"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1750371812,
|
|
||||||
"narHash": "sha256-D868K1dVEACw17elVxRgXC6hOxY+54wIEjURztDWLk8=",
|
|
||||||
"owner": "hyprwm",
|
|
||||||
"repo": "hyprland-qtutils",
|
|
||||||
"rev": "b13c7481e37856f322177010bdf75fccacd1adc8",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "hyprwm",
|
|
||||||
"repo": "hyprland-qtutils",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"hyprlang": {
|
"hyprlang": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"hyprutils": [
|
"hyprutils": [
|
||||||
@@ -215,11 +93,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1750371198,
|
"lastModified": 1724174162,
|
||||||
"narHash": "sha256-/iuJ1paQOBoSLqHflRNNGyroqfF/yvPNurxzcCT0cAE=",
|
"narHash": "sha256-fOOBLwil6M9QWMCiSULwjMQzrXhHXUnEqmjHX5ZHeVI=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "hyprlang",
|
"repo": "hyprlang",
|
||||||
"rev": "cee01452bca58d6cadb3224e21e370de8bc20f0b",
|
"rev": "16e5c9465f04477d8a3dd48a0a26bf437986336c",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -238,11 +116,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1752252310,
|
"lastModified": 1722869141,
|
||||||
"narHash": "sha256-06i1pIh6wb+sDeDmWlzuPwIdaFMxLlj1J9I5B9XqSeo=",
|
"narHash": "sha256-0KU4qhyMp441qfwbirNg3+wbm489KnEjXOz2I/RbeFs=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "hyprutils",
|
"repo": "hyprutils",
|
||||||
"rev": "bcabcbada90ed2aacb435dc09b91001819a6dc82",
|
"rev": "0252fd13e78e60fb0da512a212e56007515a49f7",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -261,11 +139,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1751897909,
|
"lastModified": 1721324119,
|
||||||
"narHash": "sha256-FnhBENxihITZldThvbO7883PdXC/2dzW4eiNvtoV5Ao=",
|
"narHash": "sha256-SOOqIT27/X792+vsLSeFdrNTF+OSRp5qXv6Te+fb2Qg=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "hyprwayland-scanner",
|
"repo": "hyprwayland-scanner",
|
||||||
"rev": "fcca0c61f988a9d092cbb33e906775014c61579d",
|
"rev": "a048a6cb015340bd82f97c1f40a4b595ca85cc30",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -276,11 +154,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1752687322,
|
"lastModified": 1724224976,
|
||||||
"narHash": "sha256-RKwfXA4OZROjBTQAl9WOZQFm7L8Bo93FQwSJpAiSRvo=",
|
"narHash": "sha256-Z/ELQhrSd7bMzTO8r7NZgi9g5emh+aRKoCdaAv5fiO0=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "6e987485eb2c77e5dcc5af4e3c70843711ef9251",
|
"rev": "c374d94f1536013ca8e92341b540eba4c22f9c62",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -290,40 +168,14 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"pre-commit-hooks": {
|
|
||||||
"inputs": {
|
|
||||||
"flake-compat": "flake-compat",
|
|
||||||
"gitignore": "gitignore",
|
|
||||||
"nixpkgs": [
|
|
||||||
"nixpkgs"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1750779888,
|
|
||||||
"narHash": "sha256-wibppH3g/E2lxU43ZQHC5yA/7kIKLGxVEnsnVK1BtRg=",
|
|
||||||
"owner": "cachix",
|
|
||||||
"repo": "git-hooks.nix",
|
|
||||||
"rev": "16ec914f6fb6f599ce988427d9d94efddf25fe6d",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "cachix",
|
|
||||||
"repo": "git-hooks.nix",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"root": {
|
"root": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"aquamarine": "aquamarine",
|
"aquamarine": "aquamarine",
|
||||||
"hyprcursor": "hyprcursor",
|
"hyprcursor": "hyprcursor",
|
||||||
"hyprgraphics": "hyprgraphics",
|
|
||||||
"hyprland-protocols": "hyprland-protocols",
|
|
||||||
"hyprland-qtutils": "hyprland-qtutils",
|
|
||||||
"hyprlang": "hyprlang",
|
"hyprlang": "hyprlang",
|
||||||
"hyprutils": "hyprutils",
|
"hyprutils": "hyprutils",
|
||||||
"hyprwayland-scanner": "hyprwayland-scanner",
|
"hyprwayland-scanner": "hyprwayland-scanner",
|
||||||
"nixpkgs": "nixpkgs",
|
"nixpkgs": "nixpkgs",
|
||||||
"pre-commit-hooks": "pre-commit-hooks",
|
|
||||||
"systems": "systems",
|
"systems": "systems",
|
||||||
"xdph": "xdph"
|
"xdph": "xdph"
|
||||||
}
|
}
|
||||||
@@ -345,18 +197,10 @@
|
|||||||
},
|
},
|
||||||
"xdph": {
|
"xdph": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"hyprland-protocols": [
|
"hyprland-protocols": "hyprland-protocols",
|
||||||
"hyprland-protocols"
|
|
||||||
],
|
|
||||||
"hyprlang": [
|
"hyprlang": [
|
||||||
"hyprlang"
|
"hyprlang"
|
||||||
],
|
],
|
||||||
"hyprutils": [
|
|
||||||
"hyprutils"
|
|
||||||
],
|
|
||||||
"hyprwayland-scanner": [
|
|
||||||
"hyprwayland-scanner"
|
|
||||||
],
|
|
||||||
"nixpkgs": [
|
"nixpkgs": [
|
||||||
"nixpkgs"
|
"nixpkgs"
|
||||||
],
|
],
|
||||||
@@ -365,11 +209,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1751300244,
|
"lastModified": 1724073926,
|
||||||
"narHash": "sha256-PFuv1TZVYvQhha0ac53E3YgdtmLShrN0t4T6xqHl0jE=",
|
"narHash": "sha256-nWlUL43jOFHf+KW6Hqrx+W/r1XdXuDyb0wC/SrHsOu4=",
|
||||||
"owner": "hyprwm",
|
"owner": "hyprwm",
|
||||||
"repo": "xdg-desktop-portal-hyprland",
|
"repo": "xdg-desktop-portal-hyprland",
|
||||||
"rev": "6115f3fdcb2c1a57b4a80a69f3c797e47607b90a",
|
"rev": "a08ecbbf33598924e93542f737fc6169a26b481e",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
90
flake.nix
@@ -22,26 +22,6 @@
|
|||||||
inputs.hyprlang.follows = "hyprlang";
|
inputs.hyprlang.follows = "hyprlang";
|
||||||
};
|
};
|
||||||
|
|
||||||
hyprgraphics = {
|
|
||||||
url = "github:hyprwm/hyprgraphics";
|
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
|
||||||
inputs.systems.follows = "systems";
|
|
||||||
inputs.hyprutils.follows = "hyprutils";
|
|
||||||
};
|
|
||||||
|
|
||||||
hyprland-protocols = {
|
|
||||||
url = "github:hyprwm/hyprland-protocols";
|
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
|
||||||
inputs.systems.follows = "systems";
|
|
||||||
};
|
|
||||||
|
|
||||||
hyprland-qtutils = {
|
|
||||||
url = "github:hyprwm/hyprland-qtutils";
|
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
|
||||||
inputs.systems.follows = "systems";
|
|
||||||
inputs.hyprlang.follows = "hyprlang";
|
|
||||||
};
|
|
||||||
|
|
||||||
hyprlang = {
|
hyprlang = {
|
||||||
url = "github:hyprwm/hyprlang";
|
url = "github:hyprwm/hyprlang";
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
@@ -65,15 +45,7 @@
|
|||||||
url = "github:hyprwm/xdg-desktop-portal-hyprland";
|
url = "github:hyprwm/xdg-desktop-portal-hyprland";
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
inputs.systems.follows = "systems";
|
inputs.systems.follows = "systems";
|
||||||
inputs.hyprland-protocols.follows = "hyprland-protocols";
|
|
||||||
inputs.hyprlang.follows = "hyprlang";
|
inputs.hyprlang.follows = "hyprlang";
|
||||||
inputs.hyprutils.follows = "hyprutils";
|
|
||||||
inputs.hyprwayland-scanner.follows = "hyprwayland-scanner";
|
|
||||||
};
|
|
||||||
|
|
||||||
pre-commit-hooks = {
|
|
||||||
url = "github:cachix/git-hooks.nix";
|
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -93,30 +65,6 @@
|
|||||||
hyprland-extras
|
hyprland-extras
|
||||||
];
|
];
|
||||||
});
|
});
|
||||||
pkgsCrossFor = eachSystem (system: crossSystem:
|
|
||||||
import nixpkgs {
|
|
||||||
localSystem = system;
|
|
||||||
inherit crossSystem;
|
|
||||||
overlays = with self.overlays; [
|
|
||||||
hyprland-packages
|
|
||||||
hyprland-extras
|
|
||||||
];
|
|
||||||
});
|
|
||||||
pkgsDebugFor = eachSystem (system:
|
|
||||||
import nixpkgs {
|
|
||||||
localSystem = system;
|
|
||||||
overlays = with self.overlays; [
|
|
||||||
hyprland-debug
|
|
||||||
];
|
|
||||||
});
|
|
||||||
pkgsDebugCrossFor = eachSystem (system: crossSystem:
|
|
||||||
import nixpkgs {
|
|
||||||
localSystem = system;
|
|
||||||
inherit crossSystem;
|
|
||||||
overlays = with self.overlays; [
|
|
||||||
hyprland-debug
|
|
||||||
];
|
|
||||||
});
|
|
||||||
in {
|
in {
|
||||||
overlays = import ./nix/overlays.nix {inherit self lib inputs;};
|
overlays = import ./nix/overlays.nix {inherit self lib inputs;};
|
||||||
|
|
||||||
@@ -126,59 +74,43 @@
|
|||||||
self.packages.${system})
|
self.packages.${system})
|
||||||
// {
|
// {
|
||||||
inherit (self.packages.${system}) xdg-desktop-portal-hyprland;
|
inherit (self.packages.${system}) xdg-desktop-portal-hyprland;
|
||||||
pre-commit-check = inputs.pre-commit-hooks.lib.${system}.run {
|
});
|
||||||
src = ./.;
|
|
||||||
hooks = {
|
|
||||||
hyprland-treewide-formatter = {
|
|
||||||
enable = true;
|
|
||||||
entry = "${self.formatter.${system}}/bin/hyprland-treewide-formatter";
|
|
||||||
pass_filenames = false;
|
|
||||||
excludes = ["subprojects"];
|
|
||||||
always_run = true;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
// (import ./nix/tests inputs pkgsFor.${system}));
|
|
||||||
|
|
||||||
packages = eachSystem (system: {
|
packages = eachSystem (system: {
|
||||||
default = self.packages.${system}.hyprland;
|
default = self.packages.${system}.hyprland;
|
||||||
inherit
|
inherit
|
||||||
(pkgsFor.${system})
|
(pkgsFor.${system})
|
||||||
# hyprland-packages
|
# hyprland-packages
|
||||||
|
|
||||||
hyprland
|
hyprland
|
||||||
|
hyprland-debug
|
||||||
|
hyprland-legacy-renderer
|
||||||
hyprland-unwrapped
|
hyprland-unwrapped
|
||||||
hyprtester
|
|
||||||
# hyprland-extras
|
# hyprland-extras
|
||||||
|
|
||||||
xdg-desktop-portal-hyprland
|
xdg-desktop-portal-hyprland
|
||||||
;
|
;
|
||||||
inherit (pkgsDebugFor.${system}) hyprland-debug;
|
|
||||||
hyprland-cross = (pkgsCrossFor.${system} "aarch64-linux").hyprland;
|
|
||||||
hyprland-debug-cross = (pkgsDebugCrossFor.${system} "aarch64-linux").hyprland-debug;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
devShells = eachSystem (system: {
|
devShells = eachSystem (system: {
|
||||||
default =
|
default =
|
||||||
pkgsFor.${system}.mkShell.override {
|
pkgsFor.${system}.mkShell.override {
|
||||||
inherit (self.packages.${system}.default) stdenv;
|
stdenv = pkgsFor.${system}.gcc14Stdenv;
|
||||||
} {
|
} {
|
||||||
name = "hyprland-shell";
|
name = "hyprland-shell";
|
||||||
|
nativeBuildInputs = with pkgsFor.${system}; [
|
||||||
|
expat
|
||||||
|
libxml2
|
||||||
|
];
|
||||||
hardeningDisable = ["fortify"];
|
hardeningDisable = ["fortify"];
|
||||||
inputsFrom = [pkgsFor.${system}.hyprland];
|
inputsFrom = [pkgsFor.${system}.hyprland];
|
||||||
packages = [pkgsFor.${system}.clang-tools];
|
packages = [pkgsFor.${system}.clang-tools];
|
||||||
inherit (self.checks.${system}.pre-commit-check) shellHook;
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
formatter = eachSystem (system: pkgsFor.${system}.callPackage ./nix/formatter.nix {});
|
formatter = eachSystem (system: nixpkgs.legacyPackages.${system}.alejandra);
|
||||||
|
|
||||||
nixosModules.default = import ./nix/module.nix inputs;
|
nixosModules.default = import ./nix/module.nix inputs;
|
||||||
homeManagerModules.default = import ./nix/hm-module.nix self;
|
homeManagerModules.default = import ./nix/hm-module.nix self;
|
||||||
|
|
||||||
# Hydra build jobs
|
|
||||||
# Recent versions of Hydra can aggregate jobsets from 'hydraJobs' intead of a release.nix
|
|
||||||
# or similar. Remember to filter large or incompatible attributes here. More eval jobs can
|
|
||||||
# be added by merging, e.g., self.packages // self.devShells.
|
|
||||||
hydraJobs = self.packages;
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@@ -5,7 +5,7 @@ project(
|
|||||||
DESCRIPTION "Control utility for Hyprland"
|
DESCRIPTION "Control utility for Hyprland"
|
||||||
)
|
)
|
||||||
|
|
||||||
pkg_check_modules(hyprctl_deps REQUIRED IMPORTED_TARGET hyprutils>=0.2.4 re2)
|
pkg_check_modules(deps REQUIRED IMPORTED_TARGET hyprutils>=0.1.1)
|
||||||
|
|
||||||
add_executable(hyprctl "main.cpp")
|
add_executable(hyprctl "main.cpp")
|
||||||
|
|
||||||
|
@@ -1,7 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <string_view>
|
|
||||||
|
|
||||||
const std::string_view USAGE = R"#(usage: hyprctl [flags] <command> [args...|--help]
|
const std::string_view USAGE = R"#(usage: hyprctl [flags] <command> [args...|--help]
|
||||||
|
|
||||||
commands:
|
commands:
|
||||||
@@ -22,7 +20,6 @@ commands:
|
|||||||
getoption <option> → Gets the config option status (values)
|
getoption <option> → Gets the config option status (values)
|
||||||
globalshortcuts → Lists all global shortcuts
|
globalshortcuts → Lists all global shortcuts
|
||||||
hyprpaper ... → Issue a hyprpaper request
|
hyprpaper ... → Issue a hyprpaper request
|
||||||
hyprsunset ... → Issue a hyprsunset request
|
|
||||||
instances → Lists all running instances of Hyprland with
|
instances → Lists all running instances of Hyprland with
|
||||||
their info
|
their info
|
||||||
keyword <name> <value> → Issue a keyword to call a config keyword
|
keyword <name> <value> → Issue a keyword to call a config keyword
|
||||||
@@ -82,16 +79,6 @@ requests:
|
|||||||
flags:
|
flags:
|
||||||
See 'hyprctl --help')#";
|
See 'hyprctl --help')#";
|
||||||
|
|
||||||
const std::string_view HYPRSUNSET_HELP = R"#(usage: hyprctl [flags] hyprsunset <request>
|
|
||||||
|
|
||||||
requests:
|
|
||||||
temperature <temp> → Enable blue-light filter
|
|
||||||
identity → Disable blue-light filter
|
|
||||||
gamma <gamma> → Enable gamma filter
|
|
||||||
|
|
||||||
flags:
|
|
||||||
See 'hyprctl --help')#";
|
|
||||||
|
|
||||||
const std::string_view NOTIFY_HELP = R"#(usage: hyprctl [flags] notify <icon> <time_ms> <color> <message...>
|
const std::string_view NOTIFY_HELP = R"#(usage: hyprctl [flags] notify <icon> <time_ms> <color> <message...>
|
||||||
|
|
||||||
icon:
|
icon:
|
||||||
@@ -146,7 +133,7 @@ regex:
|
|||||||
Regular expression by which a window will be searched
|
Regular expression by which a window will be searched
|
||||||
|
|
||||||
property:
|
property:
|
||||||
See https://wiki.hypr.land/Configuring/Using-hyprctl/#setprop for list
|
See https://wiki.hyprland.org/Configuring/Using-hyprctl/#setprop for list
|
||||||
of properties
|
of properties
|
||||||
|
|
||||||
value:
|
value:
|
||||||
|
@@ -23,7 +23,7 @@ _hyprctl () {
|
|||||||
local words cword
|
local words cword
|
||||||
_get_comp_words_by_ref -n "$COMP_WORDBREAKS" words cword
|
_get_comp_words_by_ref -n "$COMP_WORDBREAKS" words cword
|
||||||
|
|
||||||
declare -a literals=(resizeactive 2 changegroupactive -r moveintogroup forceallowsinput 4 ::= systeminfo all layouts setprop animationstyle switchxkblayout create denywindowfromgroup headless activebordercolor exec setcursor wayland focusurgentorlast workspacerules movecurrentworkspacetomonitor movetoworkspacesilent hyprpaper alpha inactivebordercolor movegroupwindow movecursortocorner movewindowpixel prev movewindow globalshortcuts clients dimaround setignoregrouplock splash execr monitors 0 forcenoborder -q animations 1 nomaxsize splitratio moveactive pass swapnext devices layers rounding lockactivegroup 5 moveworkspacetomonitor -f -i --quiet forcenodim pin 0 1 forceopaque forcenoshadow setfloating minsize alphaoverride sendshortcut workspaces cyclenext alterzorder togglegroup lockgroups bordersize dpms focuscurrentorlast -1 --batch notify remove instances 1 3 moveoutofgroup killactive 2 movetoworkspace movecursor configerrors closewindow swapwindow tagwindow forcerendererreload centerwindow auto focuswindow seterror nofocus alphafullscreen binds version -h togglespecialworkspace fullscreen windowdancecompat 0 keyword toggleopaque 3 --instance togglefloating renameworkspace alphafullscreenoverride activeworkspace x11 kill forceopaqueoverriden output global dispatch reload forcenoblur -j event --help disable -1 activewindow keepaspectratio dismissnotify focusmonitor movefocus plugin exit workspace fullscreenstate getoption alphainactiveoverride alphainactive decorations settiled config-only descriptions resizewindowpixel fakefullscreen rollinglog swapactiveworkspaces submap next movewindoworgroup cursorpos forcenoanims focusworkspaceoncurrentmonitor maxsize sendkeystate)
|
declare -a literals=(resizeactive 2 changegroupactive -r moveintogroup forceallowsinput 4 ::= systeminfo all layouts setprop animationstyle switchxkblayout create denywindowfromgroup headless activebordercolor exec setcursor wayland focusurgentorlast workspacerules movecurrentworkspacetomonitor movetoworkspacesilent hyprpaper alpha inactivebordercolor movegroupwindow movecursortocorner movewindowpixel prev movewindow globalshortcuts clients dimaround setignoregrouplock splash execr monitors 0 forcenoborder -q animations 1 nomaxsize splitratio moveactive pass swapnext devices layers rounding lockactivegroup 5 moveworkspacetomonitor -f -i --quiet forcenodim pin 0 1 forceopaque forcenoshadow setfloating minsize alphaoverride sendshortcut workspaces cyclenext alterzorder togglegroup lockgroups bordersize dpms focuscurrentorlast -1 --batch notify remove instances 1 3 moveoutofgroup killactive 2 movetoworkspace movecursor configerrors closewindow swapwindow tagwindow forcerendererreload centerwindow auto focuswindow seterror nofocus alphafullscreen binds version -h togglespecialworkspace fullscreen windowdancecompat 0 keyword toggleopaque 3 --instance togglefloating renameworkspace alphafullscreenoverride activeworkspace x11 kill forceopaqueoverriden output global dispatch reload forcenoblur -j event --help disable -1 activewindow keepaspectratio dismissnotify focusmonitor movefocus plugin exit workspace fullscreenstate getoption alphainactiveoverride alphainactive decorations settiled config-only descriptions resizewindowpixel fakefullscreen rollinglog swapactiveworkspaces submap next movewindoworgroup cursorpos forcenoanims focusworkspaceoncurrentmonitor maxsize)
|
||||||
declare -A literal_transitions
|
declare -A literal_transitions
|
||||||
literal_transitions[0]="([120]=14 [43]=2 [125]=21 [81]=2 [3]=21 [51]=2 [50]=2 [128]=2 [89]=2 [58]=21 [8]=2 [10]=2 [11]=3 [130]=4 [13]=5 [97]=6 [101]=2 [102]=21 [133]=7 [100]=2 [137]=2 [22]=2 [19]=2 [140]=8 [25]=2 [143]=2 [107]=9 [146]=10 [69]=2 [33]=2 [34]=2 [78]=21 [114]=2 [37]=2 [151]=2 [116]=2 [121]=13 [123]=21 [39]=11 [42]=21 [79]=15 [118]=12)"
|
literal_transitions[0]="([120]=14 [43]=2 [125]=21 [81]=2 [3]=21 [51]=2 [50]=2 [128]=2 [89]=2 [58]=21 [8]=2 [10]=2 [11]=3 [130]=4 [13]=5 [97]=6 [101]=2 [102]=21 [133]=7 [100]=2 [137]=2 [22]=2 [19]=2 [140]=8 [25]=2 [143]=2 [107]=9 [146]=10 [69]=2 [33]=2 [34]=2 [78]=21 [114]=2 [37]=2 [151]=2 [116]=2 [121]=13 [123]=21 [39]=11 [42]=21 [79]=15 [118]=12)"
|
||||||
literal_transitions[1]="([81]=2 [51]=2 [50]=2 [128]=2 [8]=2 [89]=2 [10]=2 [11]=3 [130]=4 [13]=5 [97]=6 [101]=2 [133]=7 [100]=2 [22]=2 [19]=2 [137]=2 [140]=8 [25]=2 [143]=2 [107]=9 [146]=10 [69]=2 [33]=2 [34]=2 [114]=2 [37]=2 [151]=2 [116]=2 [39]=11 [118]=12 [121]=13 [120]=14 [79]=15 [43]=2)"
|
literal_transitions[1]="([81]=2 [51]=2 [50]=2 [128]=2 [8]=2 [89]=2 [10]=2 [11]=3 [130]=4 [13]=5 [97]=6 [101]=2 [133]=7 [100]=2 [22]=2 [19]=2 [137]=2 [140]=8 [25]=2 [143]=2 [107]=9 [146]=10 [69]=2 [33]=2 [34]=2 [114]=2 [37]=2 [151]=2 [116]=2 [39]=11 [118]=12 [121]=13 [120]=14 [79]=15 [43]=2)"
|
||||||
|
@@ -29,7 +29,7 @@ function _hyprctl
|
|||||||
set COMP_CWORD (count $COMP_WORDS)
|
set COMP_CWORD (count $COMP_WORDS)
|
||||||
end
|
end
|
||||||
|
|
||||||
set literals "resizeactive" "2" "changegroupactive" "-r" "moveintogroup" "forceallowsinput" "4" "::=" "systeminfo" "all" "layouts" "setprop" "animationstyle" "switchxkblayout" "create" "denywindowfromgroup" "headless" "activebordercolor" "exec" "setcursor" "wayland" "focusurgentorlast" "workspacerules" "movecurrentworkspacetomonitor" "movetoworkspacesilent" "hyprpaper" "alpha" "inactivebordercolor" "movegroupwindow" "movecursortocorner" "movewindowpixel" "prev" "movewindow" "globalshortcuts" "clients" "dimaround" "setignoregrouplock" "splash" "execr" "monitors" "0" "forcenoborder" "-q" "animations" "1" "nomaxsize" "splitratio" "moveactive" "pass" "swapnext" "devices" "layers" "rounding" "lockactivegroup" "5" "moveworkspacetomonitor" "-f" "-i" "--quiet" "forcenodim" "pin" "0" "1" "forceopaque" "forcenoshadow" "setfloating" "minsize" "alphaoverride" "sendshortcut" "workspaces" "cyclenext" "alterzorder" "togglegroup" "lockgroups" "bordersize" "dpms" "focuscurrentorlast" "-1" "--batch" "notify" "remove" "instances" "1" "3" "moveoutofgroup" "killactive" "2" "movetoworkspace" "movecursor" "configerrors" "closewindow" "swapwindow" "tagwindow" "forcerendererreload" "centerwindow" "auto" "focuswindow" "seterror" "nofocus" "alphafullscreen" "binds" "version" "-h" "togglespecialworkspace" "fullscreen" "windowdancecompat" "0" "keyword" "toggleopaque" "3" "--instance" "togglefloating" "renameworkspace" "alphafullscreenoverride" "activeworkspace" "x11" "kill" "forceopaqueoverriden" "output" "global" "dispatch" "reload" "forcenoblur" "-j" "event" "--help" "disable" "-1" "activewindow" "keepaspectratio" "dismissnotify" "focusmonitor" "movefocus" "plugin" "exit" "workspace" "fullscreenstate" "getoption" "alphainactiveoverride" "alphainactive" "decorations" "settiled" "config-only" "descriptions" "resizewindowpixel" "fakefullscreen" "rollinglog" "swapactiveworkspaces" "submap" "next" "movewindoworgroup" "cursorpos" "forcenoanims" "focusworkspaceoncurrentmonitor" "maxsize" "sendkeystate"
|
set literals "resizeactive" "2" "changegroupactive" "-r" "moveintogroup" "forceallowsinput" "4" "::=" "systeminfo" "all" "layouts" "setprop" "animationstyle" "switchxkblayout" "create" "denywindowfromgroup" "headless" "activebordercolor" "exec" "setcursor" "wayland" "focusurgentorlast" "workspacerules" "movecurrentworkspacetomonitor" "movetoworkspacesilent" "hyprpaper" "alpha" "inactivebordercolor" "movegroupwindow" "movecursortocorner" "movewindowpixel" "prev" "movewindow" "globalshortcuts" "clients" "dimaround" "setignoregrouplock" "splash" "execr" "monitors" "0" "forcenoborder" "-q" "animations" "1" "nomaxsize" "splitratio" "moveactive" "pass" "swapnext" "devices" "layers" "rounding" "lockactivegroup" "5" "moveworkspacetomonitor" "-f" "-i" "--quiet" "forcenodim" "pin" "0" "1" "forceopaque" "forcenoshadow" "setfloating" "minsize" "alphaoverride" "sendshortcut" "workspaces" "cyclenext" "alterzorder" "togglegroup" "lockgroups" "bordersize" "dpms" "focuscurrentorlast" "-1" "--batch" "notify" "remove" "instances" "1" "3" "moveoutofgroup" "killactive" "2" "movetoworkspace" "movecursor" "configerrors" "closewindow" "swapwindow" "tagwindow" "forcerendererreload" "centerwindow" "auto" "focuswindow" "seterror" "nofocus" "alphafullscreen" "binds" "version" "-h" "togglespecialworkspace" "fullscreen" "windowdancecompat" "0" "keyword" "toggleopaque" "3" "--instance" "togglefloating" "renameworkspace" "alphafullscreenoverride" "activeworkspace" "x11" "kill" "forceopaqueoverriden" "output" "global" "dispatch" "reload" "forcenoblur" "-j" "event" "--help" "disable" "-1" "activewindow" "keepaspectratio" "dismissnotify" "focusmonitor" "movefocus" "plugin" "exit" "workspace" "fullscreenstate" "getoption" "alphainactiveoverride" "alphainactive" "decorations" "settiled" "config-only" "descriptions" "resizewindowpixel" "fakefullscreen" "rollinglog" "swapactiveworkspaces" "submap" "next" "movewindoworgroup" "cursorpos" "forcenoanims" "focusworkspaceoncurrentmonitor" "maxsize"
|
||||||
|
|
||||||
set descriptions
|
set descriptions
|
||||||
set descriptions[1] "Resize the active window"
|
set descriptions[1] "Resize the active window"
|
||||||
|
@@ -106,7 +106,6 @@ hyprctl [<OPTIONS>]... <ARGUMENTS>
|
|||||||
| (execr) "Execute a raw shell command"
|
| (execr) "Execute a raw shell command"
|
||||||
| (pass) "Pass the key to a specified window"
|
| (pass) "Pass the key to a specified window"
|
||||||
| (sendshortcut) "On shortcut X sends shortcut Y to a specified window"
|
| (sendshortcut) "On shortcut X sends shortcut Y to a specified window"
|
||||||
| (sendkeystate) "Send a key with specific state (down/repeat/up) to a specified window (window must keep focus for events to continue)"
|
|
||||||
| (killactive) "Close the active window"
|
| (killactive) "Close the active window"
|
||||||
| (closewindow) "Close a specified window"
|
| (closewindow) "Close a specified window"
|
||||||
| (workspace) "Change the workspace"
|
| (workspace) "Change the workspace"
|
||||||
|
@@ -17,7 +17,7 @@ _hyprctl_cmd_0 () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_hyprctl () {
|
_hyprctl () {
|
||||||
local -a literals=("resizeactive" "2" "changegroupactive" "-r" "moveintogroup" "forceallowsinput" "4" "::=" "systeminfo" "all" "layouts" "setprop" "animationstyle" "switchxkblayout" "create" "denywindowfromgroup" "headless" "activebordercolor" "exec" "setcursor" "wayland" "focusurgentorlast" "workspacerules" "movecurrentworkspacetomonitor" "movetoworkspacesilent" "hyprpaper" "alpha" "inactivebordercolor" "movegroupwindow" "movecursortocorner" "movewindowpixel" "prev" "movewindow" "globalshortcuts" "clients" "dimaround" "setignoregrouplock" "splash" "execr" "monitors" "0" "forcenoborder" "-q" "animations" "1" "nomaxsize" "splitratio" "moveactive" "pass" "swapnext" "devices" "layers" "rounding" "lockactivegroup" "5" "moveworkspacetomonitor" "-f" "-i" "--quiet" "forcenodim" "pin" "0" "1" "forceopaque" "forcenoshadow" "setfloating" "minsize" "alphaoverride" "sendshortcut" "workspaces" "cyclenext" "alterzorder" "togglegroup" "lockgroups" "bordersize" "dpms" "focuscurrentorlast" "-1" "--batch" "notify" "remove" "instances" "1" "3" "moveoutofgroup" "killactive" "2" "movetoworkspace" "movecursor" "configerrors" "closewindow" "swapwindow" "tagwindow" "forcerendererreload" "centerwindow" "auto" "focuswindow" "seterror" "nofocus" "alphafullscreen" "binds" "version" "-h" "togglespecialworkspace" "fullscreen" "windowdancecompat" "0" "keyword" "toggleopaque" "3" "--instance" "togglefloating" "renameworkspace" "alphafullscreenoverride" "activeworkspace" "x11" "kill" "forceopaqueoverriden" "output" "global" "dispatch" "reload" "forcenoblur" "-j" "event" "--help" "disable" "-1" "activewindow" "keepaspectratio" "dismissnotify" "focusmonitor" "movefocus" "plugin" "exit" "workspace" "fullscreenstate" "getoption" "alphainactiveoverride" "alphainactive" "decorations" "settiled" "config-only" "descriptions" "resizewindowpixel" "fakefullscreen" "rollinglog" "swapactiveworkspaces" "submap" "next" "movewindoworgroup" "cursorpos" "forcenoanims" "focusworkspaceoncurrentmonitor" "maxsize" "sendkeystate")
|
local -a literals=("resizeactive" "2" "changegroupactive" "-r" "moveintogroup" "forceallowsinput" "4" "::=" "systeminfo" "all" "layouts" "setprop" "animationstyle" "switchxkblayout" "create" "denywindowfromgroup" "headless" "activebordercolor" "exec" "setcursor" "wayland" "focusurgentorlast" "workspacerules" "movecurrentworkspacetomonitor" "movetoworkspacesilent" "hyprpaper" "alpha" "inactivebordercolor" "movegroupwindow" "movecursortocorner" "movewindowpixel" "prev" "movewindow" "globalshortcuts" "clients" "dimaround" "setignoregrouplock" "splash" "execr" "monitors" "0" "forcenoborder" "-q" "animations" "1" "nomaxsize" "splitratio" "moveactive" "pass" "swapnext" "devices" "layers" "rounding" "lockactivegroup" "5" "moveworkspacetomonitor" "-f" "-i" "--quiet" "forcenodim" "pin" "0" "1" "forceopaque" "forcenoshadow" "setfloating" "minsize" "alphaoverride" "sendshortcut" "workspaces" "cyclenext" "alterzorder" "togglegroup" "lockgroups" "bordersize" "dpms" "focuscurrentorlast" "-1" "--batch" "notify" "remove" "instances" "1" "3" "moveoutofgroup" "killactive" "2" "movetoworkspace" "movecursor" "configerrors" "closewindow" "swapwindow" "tagwindow" "forcerendererreload" "centerwindow" "auto" "focuswindow" "seterror" "nofocus" "alphafullscreen" "binds" "version" "-h" "togglespecialworkspace" "fullscreen" "windowdancecompat" "0" "keyword" "toggleopaque" "3" "--instance" "togglefloating" "renameworkspace" "alphafullscreenoverride" "activeworkspace" "x11" "kill" "forceopaqueoverriden" "output" "global" "dispatch" "reload" "forcenoblur" "-j" "event" "--help" "disable" "-1" "activewindow" "keepaspectratio" "dismissnotify" "focusmonitor" "movefocus" "plugin" "exit" "workspace" "fullscreenstate" "getoption" "alphainactiveoverride" "alphainactive" "decorations" "settiled" "config-only" "descriptions" "resizewindowpixel" "fakefullscreen" "rollinglog" "swapactiveworkspaces" "submap" "next" "movewindoworgroup" "cursorpos" "forcenoanims" "focusworkspaceoncurrentmonitor" "maxsize")
|
||||||
|
|
||||||
local -A descriptions
|
local -A descriptions
|
||||||
descriptions[1]="Resize the active window"
|
descriptions[1]="Resize the active window"
|
||||||
|
253
hyprctl/main.cpp
@@ -1,5 +1,3 @@
|
|||||||
#include <re2/re2.h>
|
|
||||||
|
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
#include <netdb.h>
|
#include <netdb.h>
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
@@ -12,24 +10,29 @@
|
|||||||
#include <sys/un.h>
|
#include <sys/un.h>
|
||||||
#include <pwd.h>
|
#include <pwd.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <ranges>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <csignal>
|
#include <csignal>
|
||||||
#include <ranges>
|
#include <format>
|
||||||
#include <optional>
|
|
||||||
#include <charconv>
|
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <print>
|
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <deque>
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
#include <cstdarg>
|
#include <cstdarg>
|
||||||
|
#include <regex>
|
||||||
|
#include <sys/socket.h>
|
||||||
#include <hyprutils/string/String.hpp>
|
#include <hyprutils/string/String.hpp>
|
||||||
|
#include <cstring>
|
||||||
using namespace Hyprutils::String;
|
using namespace Hyprutils::String;
|
||||||
|
|
||||||
#include "Strings.hpp"
|
#include "Strings.hpp"
|
||||||
|
|
||||||
|
#define PAD
|
||||||
|
|
||||||
std::string instanceSignature;
|
std::string instanceSignature;
|
||||||
bool quiet = false;
|
bool quiet = false;
|
||||||
|
|
||||||
@@ -38,100 +41,63 @@ struct SInstanceData {
|
|||||||
uint64_t time;
|
uint64_t time;
|
||||||
uint64_t pid;
|
uint64_t pid;
|
||||||
std::string wlSocket;
|
std::string wlSocket;
|
||||||
|
bool valid = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
void log(const std::string_view str) {
|
void log(std::string str) {
|
||||||
if (quiet)
|
if (quiet)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
std::println("{}", str);
|
std::cout << str << "\n";
|
||||||
}
|
|
||||||
|
|
||||||
static int getUID() {
|
|
||||||
const auto UID = getuid();
|
|
||||||
const auto PWUID = getpwuid(UID);
|
|
||||||
return PWUID ? PWUID->pw_uid : UID;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string getRuntimeDir() {
|
std::string getRuntimeDir() {
|
||||||
const auto XDG = getenv("XDG_RUNTIME_DIR");
|
const auto XDG = getenv("XDG_RUNTIME_DIR");
|
||||||
|
|
||||||
if (!XDG) {
|
if (!XDG) {
|
||||||
const std::string USERID = std::to_string(getUID());
|
const std::string USERID = std::to_string(getpwuid(getuid())->pw_uid);
|
||||||
return "/run/user/" + USERID + "/hypr";
|
return "/run/user/" + USERID + "/hypr";
|
||||||
}
|
}
|
||||||
|
|
||||||
return std::string{XDG} + "/hypr";
|
return std::string{XDG} + "/hypr";
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<uint64_t> toUInt64(const std::string_view str) {
|
|
||||||
uint64_t value = 0;
|
|
||||||
const auto [ptr, ec] = std::from_chars(str.data(), str.data() + str.size(), value);
|
|
||||||
if (ec != std::errc() || ptr != str.data() + str.size())
|
|
||||||
return std::nullopt;
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::optional<SInstanceData> parseInstance(const std::filesystem::directory_entry& entry) {
|
|
||||||
if (!entry.is_directory())
|
|
||||||
return std::nullopt;
|
|
||||||
|
|
||||||
const auto lockPath = entry.path() / "hyprland.lock";
|
|
||||||
std::ifstream ifs(lockPath);
|
|
||||||
if (!ifs.is_open())
|
|
||||||
return std::nullopt;
|
|
||||||
|
|
||||||
SInstanceData data;
|
|
||||||
data.id = entry.path().filename().string();
|
|
||||||
|
|
||||||
const auto first = std::string_view{data.id}.find_first_of('_');
|
|
||||||
const auto last = std::string_view{data.id}.find_last_of('_');
|
|
||||||
if (first == std::string_view::npos || last == std::string_view::npos || last <= first)
|
|
||||||
return std::nullopt;
|
|
||||||
|
|
||||||
auto time = toUInt64(std::string_view{data.id}.substr(first + 1, last - first - 1));
|
|
||||||
if (!time)
|
|
||||||
return std::nullopt;
|
|
||||||
data.time = *time;
|
|
||||||
|
|
||||||
std::string line;
|
|
||||||
if (!std::getline(ifs, line))
|
|
||||||
return std::nullopt;
|
|
||||||
|
|
||||||
auto pid = toUInt64(std::string_view{line});
|
|
||||||
if (!pid)
|
|
||||||
return std::nullopt;
|
|
||||||
data.pid = *pid;
|
|
||||||
|
|
||||||
if (!std::getline(ifs, data.wlSocket))
|
|
||||||
return std::nullopt;
|
|
||||||
|
|
||||||
if (std::getline(ifs, line) && !line.empty())
|
|
||||||
return std::nullopt; // more lines than expected
|
|
||||||
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<SInstanceData> instances() {
|
std::vector<SInstanceData> instances() {
|
||||||
std::vector<SInstanceData> result;
|
std::vector<SInstanceData> result;
|
||||||
|
|
||||||
std::error_code ec;
|
for (const auto& el : std::filesystem::directory_iterator(getRuntimeDir())) {
|
||||||
const auto runtimeDir = getRuntimeDir();
|
if (!el.is_directory() || !std::filesystem::exists(el.path().string() + "/hyprland.lock"))
|
||||||
if (!std::filesystem::exists(runtimeDir, ec) || ec)
|
continue;
|
||||||
return result;
|
|
||||||
|
|
||||||
std::filesystem::directory_iterator it(runtimeDir, std::filesystem::directory_options::skip_permission_denied, ec);
|
// read lock
|
||||||
if (ec)
|
SInstanceData* data = &result.emplace_back();
|
||||||
return result;
|
data->id = el.path().filename().string();
|
||||||
|
|
||||||
for (const auto& el : it) {
|
try {
|
||||||
if (auto instance = parseInstance(el))
|
data->time = std::stoull(data->id.substr(data->id.find_first_of('_') + 1, data->id.find_last_of('_') - (data->id.find_first_of('_') + 1)));
|
||||||
result.emplace_back(std::move(*instance));
|
} catch (std::exception& e) { continue; }
|
||||||
|
|
||||||
|
// read file
|
||||||
|
std::ifstream ifs(el.path().string() + "/hyprland.lock");
|
||||||
|
|
||||||
|
int i = 0;
|
||||||
|
for (std::string line; std::getline(ifs, line); ++i) {
|
||||||
|
if (i == 0) {
|
||||||
|
try {
|
||||||
|
data->pid = std::stoull(line);
|
||||||
|
} catch (std::exception& e) { continue; }
|
||||||
|
} else if (i == 1) {
|
||||||
|
data->wlSocket = line;
|
||||||
|
} else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ifs.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::erase_if(result, [](const auto& el) { return kill(el.pid, 0) != 0 && errno == ESRCH; });
|
std::erase_if(result, [&](const auto& el) { return kill(el.pid, 0) != 0 && errno == ESRCH; });
|
||||||
|
|
||||||
std::ranges::sort(result, {}, &SInstanceData::time);
|
std::sort(result.begin(), result.end(), [&](const auto& a, const auto& b) { return a.time < b.time; });
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -139,7 +105,7 @@ std::vector<SInstanceData> instances() {
|
|||||||
static volatile bool sigintReceived = false;
|
static volatile bool sigintReceived = false;
|
||||||
void intHandler(int sig) {
|
void intHandler(int sig) {
|
||||||
sigintReceived = true;
|
sigintReceived = true;
|
||||||
std::println("[hyprctl] SIGINT received, closing connection");
|
std::cout << "[hyprctl] SIGINT received, closing connection" << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
int rollingRead(const int socket) {
|
int rollingRead(const int socket) {
|
||||||
@@ -149,12 +115,12 @@ int rollingRead(const int socket) {
|
|||||||
constexpr size_t BUFFER_SIZE = 8192;
|
constexpr size_t BUFFER_SIZE = 8192;
|
||||||
std::array<char, BUFFER_SIZE> buffer = {0};
|
std::array<char, BUFFER_SIZE> buffer = {0};
|
||||||
long sizeWritten = 0;
|
long sizeWritten = 0;
|
||||||
std::println("[hyprctl] reading from socket following up log:");
|
std::cout << "[hyprctl] reading from socket following up log:" << std::endl;
|
||||||
while (!sigintReceived) {
|
while (!sigintReceived) {
|
||||||
sizeWritten = read(socket, buffer.data(), BUFFER_SIZE);
|
sizeWritten = read(socket, buffer.data(), BUFFER_SIZE);
|
||||||
if (sizeWritten < 0 && errno != EAGAIN) {
|
if (sizeWritten < 0 && errno != EAGAIN) {
|
||||||
if (errno != EINTR)
|
if (errno != EINTR)
|
||||||
std::println("Couldn't read (5): {}: {}", strerror(errno), errno);
|
std::cout << "Couldn't read (5) " << strerror(errno) << ":" << errno << std::endl;
|
||||||
close(socket);
|
close(socket);
|
||||||
return 5;
|
return 5;
|
||||||
}
|
}
|
||||||
@@ -163,7 +129,7 @@ int rollingRead(const int socket) {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
if (sizeWritten > 0) {
|
if (sizeWritten > 0) {
|
||||||
std::println("{}", std::string(buffer.data(), sizeWritten));
|
std::cout << std::string(buffer.data(), sizeWritten);
|
||||||
buffer.fill('\0');
|
buffer.fill('\0');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -173,19 +139,11 @@ int rollingRead(const int socket) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int request(std::string_view arg, int minArgs = 0, bool needRoll = false) {
|
int request(std::string arg, int minArgs = 0, bool needRoll = false) {
|
||||||
const auto SERVERSOCKET = socket(AF_UNIX, SOCK_STREAM, 0);
|
const auto SERVERSOCKET = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||||
|
|
||||||
if (SERVERSOCKET < 0) {
|
auto t = timeval{.tv_sec = 5, .tv_usec = 0};
|
||||||
log("Couldn't open a socket (1)");
|
setsockopt(SERVERSOCKET, SOL_SOCKET, SO_RCVTIMEO, &t, sizeof(struct timeval));
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto t = timeval{.tv_sec = 5, .tv_usec = 0};
|
|
||||||
if (setsockopt(SERVERSOCKET, SOL_SOCKET, SO_RCVTIMEO, &t, sizeof(struct timeval)) < 0) {
|
|
||||||
log("Couldn't set socket timeout (2)");
|
|
||||||
return 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto ARGS = std::count(arg.begin(), arg.end(), ' ');
|
const auto ARGS = std::count(arg.begin(), arg.end(), ' ');
|
||||||
|
|
||||||
@@ -194,53 +152,59 @@ int request(std::string_view arg, int minArgs = 0, bool needRoll = false) {
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (instanceSignature.empty()) {
|
if (SERVERSOCKET < 0) {
|
||||||
log("HYPRLAND_INSTANCE_SIGNATURE was not set! (Is Hyprland running?) (3)");
|
log("Couldn't open a socket (1)");
|
||||||
return 3;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
sockaddr_un serverAddress = {0};
|
if (instanceSignature.empty()) {
|
||||||
serverAddress.sun_family = AF_UNIX;
|
log("HYPRLAND_INSTANCE_SIGNATURE was not set! (Is Hyprland running?)");
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string USERID = std::to_string(getpwuid(getuid())->pw_uid);
|
||||||
|
|
||||||
|
sockaddr_un serverAddress = {0};
|
||||||
|
serverAddress.sun_family = AF_UNIX;
|
||||||
|
|
||||||
std::string socketPath = getRuntimeDir() + "/" + instanceSignature + "/.socket.sock";
|
std::string socketPath = getRuntimeDir() + "/" + instanceSignature + "/.socket.sock";
|
||||||
|
|
||||||
strncpy(serverAddress.sun_path, socketPath.c_str(), sizeof(serverAddress.sun_path) - 1);
|
strncpy(serverAddress.sun_path, socketPath.c_str(), sizeof(serverAddress.sun_path) - 1);
|
||||||
|
|
||||||
if (connect(SERVERSOCKET, (sockaddr*)&serverAddress, SUN_LEN(&serverAddress)) < 0) {
|
if (connect(SERVERSOCKET, (sockaddr*)&serverAddress, SUN_LEN(&serverAddress)) < 0) {
|
||||||
log("Couldn't connect to " + socketPath + ". (4)");
|
log("Couldn't connect to " + socketPath + ". (3)");
|
||||||
return 4;
|
return 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto sizeWritten = write(SERVERSOCKET, arg.data(), arg.size());
|
auto sizeWritten = write(SERVERSOCKET, arg.c_str(), arg.length());
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
if (sizeWritten < 0) {
|
||||||
log("Couldn't write (5)");
|
log("Couldn't write (4)");
|
||||||
return 5;
|
return 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (needRoll)
|
if (needRoll)
|
||||||
return rollingRead(SERVERSOCKET);
|
return rollingRead(SERVERSOCKET);
|
||||||
|
|
||||||
std::string reply = "";
|
std::string reply = "";
|
||||||
constexpr size_t BUFFER_SIZE = 8192;
|
char buffer[8192] = {0};
|
||||||
char buffer[BUFFER_SIZE] = {0};
|
|
||||||
|
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, BUFFER_SIZE);
|
sizeWritten = read(SERVERSOCKET, buffer, 8192);
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
if (sizeWritten < 0) {
|
||||||
if (errno == EWOULDBLOCK)
|
if (errno == EWOULDBLOCK)
|
||||||
log("Hyprland IPC didn't respond in time\n");
|
log("Hyprland IPC didn't respond in time\n");
|
||||||
log("Couldn't read (6)");
|
log("Couldn't read (5)");
|
||||||
return 6;
|
return 5;
|
||||||
}
|
}
|
||||||
|
|
||||||
reply += std::string(buffer, sizeWritten);
|
reply += std::string(buffer, sizeWritten);
|
||||||
|
|
||||||
while (sizeWritten == BUFFER_SIZE) {
|
while (sizeWritten == 8192) {
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, BUFFER_SIZE);
|
sizeWritten = read(SERVERSOCKET, buffer, 8192);
|
||||||
if (sizeWritten < 0) {
|
if (sizeWritten < 0) {
|
||||||
log("Couldn't read (6)");
|
log("Couldn't read (5)");
|
||||||
return 6;
|
return 5;
|
||||||
}
|
}
|
||||||
reply += std::string(buffer, sizeWritten);
|
reply += std::string(buffer, sizeWritten);
|
||||||
}
|
}
|
||||||
@@ -252,7 +216,7 @@ int request(std::string_view arg, int minArgs = 0, bool needRoll = false) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int requestIPC(std::string_view filename, std::string_view arg) {
|
int requestHyprpaper(std::string arg) {
|
||||||
const auto SERVERSOCKET = socket(AF_UNIX, SOCK_STREAM, 0);
|
const auto SERVERSOCKET = socket(AF_UNIX, SOCK_STREAM, 0);
|
||||||
|
|
||||||
if (SERVERSOCKET < 0) {
|
if (SERVERSOCKET < 0) {
|
||||||
@@ -268,7 +232,9 @@ int requestIPC(std::string_view filename, std::string_view arg) {
|
|||||||
sockaddr_un serverAddress = {0};
|
sockaddr_un serverAddress = {0};
|
||||||
serverAddress.sun_family = AF_UNIX;
|
serverAddress.sun_family = AF_UNIX;
|
||||||
|
|
||||||
std::string socketPath = getRuntimeDir() + "/" + instanceSignature + "/" + filename;
|
const std::string USERID = std::to_string(getpwuid(getuid())->pw_uid);
|
||||||
|
|
||||||
|
std::string socketPath = getRuntimeDir() + "/" + instanceSignature + "/.hyprpaper.sock";
|
||||||
|
|
||||||
strncpy(serverAddress.sun_path, socketPath.c_str(), sizeof(serverAddress.sun_path) - 1);
|
strncpy(serverAddress.sun_path, socketPath.c_str(), sizeof(serverAddress.sun_path) - 1);
|
||||||
|
|
||||||
@@ -280,16 +246,16 @@ int requestIPC(std::string_view filename, std::string_view arg) {
|
|||||||
arg = arg.substr(arg.find_first_of('/') + 1); // strip flags
|
arg = arg.substr(arg.find_first_of('/') + 1); // strip flags
|
||||||
arg = arg.substr(arg.find_first_of(' ') + 1); // strip "hyprpaper"
|
arg = arg.substr(arg.find_first_of(' ') + 1); // strip "hyprpaper"
|
||||||
|
|
||||||
auto sizeWritten = write(SERVERSOCKET, arg.data(), arg.size());
|
auto sizeWritten = write(SERVERSOCKET, arg.c_str(), arg.length());
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
if (sizeWritten < 0) {
|
||||||
log("Couldn't write (4)");
|
log("Couldn't write (4)");
|
||||||
return 4;
|
return 4;
|
||||||
}
|
}
|
||||||
constexpr size_t BUFFER_SIZE = 8192;
|
|
||||||
char buffer[BUFFER_SIZE] = {0};
|
|
||||||
|
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, BUFFER_SIZE);
|
char buffer[8192] = {0};
|
||||||
|
|
||||||
|
sizeWritten = read(SERVERSOCKET, buffer, 8192);
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
if (sizeWritten < 0) {
|
||||||
log("Couldn't read (5)");
|
log("Couldn't read (5)");
|
||||||
@@ -303,20 +269,11 @@ int requestIPC(std::string_view filename, std::string_view arg) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int requestHyprpaper(std::string_view arg) {
|
void batchRequest(std::string arg, bool json) {
|
||||||
return requestIPC(".hyprpaper.sock", arg);
|
std::string commands = arg.substr(arg.find_first_of(" ") + 1);
|
||||||
}
|
|
||||||
|
|
||||||
int requestHyprsunset(std::string_view arg) {
|
|
||||||
return requestIPC(".hyprsunset.sock", arg);
|
|
||||||
}
|
|
||||||
|
|
||||||
void batchRequest(std::string_view arg, bool json) {
|
|
||||||
std::string commands(arg.substr(arg.find_first_of(' ') + 1));
|
|
||||||
|
|
||||||
if (json) {
|
if (json) {
|
||||||
RE2::GlobalReplace(&commands, ";\\s*", ";j/");
|
commands = "j/" + std::regex_replace(commands, std::regex(";\\s*"), ";j/");
|
||||||
commands.insert(0, "j/");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string rq = "[[BATCH]]" + commands;
|
std::string rq = "[[BATCH]]" + commands;
|
||||||
@@ -330,12 +287,12 @@ void instancesRequest(bool json) {
|
|||||||
std::vector<SInstanceData> inst = instances();
|
std::vector<SInstanceData> inst = instances();
|
||||||
|
|
||||||
if (!json) {
|
if (!json) {
|
||||||
for (auto const& el : inst) {
|
for (auto& el : inst) {
|
||||||
result += std::format("instance {}:\n\ttime: {}\n\tpid: {}\n\twl socket: {}\n\n", el.id, el.time, el.pid, el.wlSocket);
|
result += std::format("instance {}:\n\ttime: {}\n\tpid: {}\n\twl socket: {}\n\n", el.id, el.time, el.pid, el.wlSocket);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
result += '[';
|
result += '[';
|
||||||
for (auto const& el : inst) {
|
for (auto& el : inst) {
|
||||||
result += std::format(R"#(
|
result += std::format(R"#(
|
||||||
{{
|
{{
|
||||||
"instance": "{}",
|
"instance": "{}",
|
||||||
@@ -353,11 +310,11 @@ void instancesRequest(bool json) {
|
|||||||
log(result + "\n");
|
log(result + "\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<std::string> splitArgs(int argc, char** argv) {
|
std::deque<std::string> splitArgs(int argc, char** argv) {
|
||||||
std::vector<std::string> result;
|
std::deque<std::string> result;
|
||||||
|
|
||||||
for (auto i = 1 /* skip the executable */; i < argc; ++i)
|
for (auto i = 1 /* skip the executable */; i < argc; ++i)
|
||||||
result.emplace_back(argv[i]);
|
result.push_back(std::string(argv[i]));
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -366,7 +323,7 @@ int main(int argc, char** argv) {
|
|||||||
bool parseArgs = true;
|
bool parseArgs = true;
|
||||||
|
|
||||||
if (argc < 2) {
|
if (argc < 2) {
|
||||||
std::println("{}", USAGE);
|
std::cout << USAGE << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -383,7 +340,7 @@ int main(int argc, char** argv) {
|
|||||||
parseArgs = false;
|
parseArgs = false;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (parseArgs && (ARGS[i][0] == '-') && !(isNumber(ARGS[i], true) || isNumber(ARGS[i].substr(0, ARGS[i].length() - 1), true)) /* For stuff like -2 or -2, */) {
|
if (parseArgs && (ARGS[i][0] == '-') && !isNumber(ARGS[i], true) /* For stuff like -2 */) {
|
||||||
// parse
|
// parse
|
||||||
if (ARGS[i] == "-j" && !fullArgs.contains("j")) {
|
if (ARGS[i] == "-j" && !fullArgs.contains("j")) {
|
||||||
fullArgs += "j";
|
fullArgs += "j";
|
||||||
@@ -403,7 +360,7 @@ int main(int argc, char** argv) {
|
|||||||
++i;
|
++i;
|
||||||
|
|
||||||
if (i >= ARGS.size()) {
|
if (i >= ARGS.size()) {
|
||||||
std::println("{}", USAGE);
|
std::cout << USAGE << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -414,26 +371,24 @@ int main(int argc, char** argv) {
|
|||||||
const std::string& cmd = ARGS[0];
|
const std::string& cmd = ARGS[0];
|
||||||
|
|
||||||
if (cmd == "hyprpaper") {
|
if (cmd == "hyprpaper") {
|
||||||
std::println("{}", HYPRPAPER_HELP);
|
std::cout << HYPRPAPER_HELP << std::endl;
|
||||||
} else if (cmd == "hyprsunset") {
|
|
||||||
std::println("{}", HYPRSUNSET_HELP);
|
|
||||||
} else if (cmd == "notify") {
|
} else if (cmd == "notify") {
|
||||||
std::println("{}", NOTIFY_HELP);
|
std::cout << NOTIFY_HELP << std::endl;
|
||||||
} else if (cmd == "output") {
|
} else if (cmd == "output") {
|
||||||
std::println("{}", OUTPUT_HELP);
|
std::cout << OUTPUT_HELP << std::endl;
|
||||||
} else if (cmd == "plugin") {
|
} else if (cmd == "plugin") {
|
||||||
std::println("{}", PLUGIN_HELP);
|
std::cout << PLUGIN_HELP << std::endl;
|
||||||
} else if (cmd == "setprop") {
|
} else if (cmd == "setprop") {
|
||||||
std::println("{}", SETPROP_HELP);
|
std::cout << SETPROP_HELP << std::endl;
|
||||||
} else if (cmd == "switchxkblayout") {
|
} else if (cmd == "switchxkblayout") {
|
||||||
std::println("{}", SWITCHXKBLAYOUT_HELP);
|
std::cout << SWITCHXKBLAYOUT_HELP << std::endl;
|
||||||
} else {
|
} else {
|
||||||
std::println("{}", USAGE);
|
std::cout << USAGE << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
} else {
|
} else {
|
||||||
std::println("{}", USAGE);
|
std::cout << USAGE << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -444,7 +399,7 @@ int main(int argc, char** argv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (fullRequest.empty()) {
|
if (fullRequest.empty()) {
|
||||||
std::println("{}", USAGE);
|
std::cout << USAGE << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -498,8 +453,6 @@ int main(int argc, char** argv) {
|
|||||||
batchRequest(fullRequest, json);
|
batchRequest(fullRequest, json);
|
||||||
else if (fullRequest.contains("/hyprpaper"))
|
else if (fullRequest.contains("/hyprpaper"))
|
||||||
exitStatus = requestHyprpaper(fullRequest);
|
exitStatus = requestHyprpaper(fullRequest);
|
||||||
else if (fullRequest.contains("/hyprsunset"))
|
|
||||||
exitStatus = requestHyprsunset(fullRequest);
|
|
||||||
else if (fullRequest.contains("/switchxkblayout"))
|
else if (fullRequest.contains("/switchxkblayout"))
|
||||||
exitStatus = request(fullRequest, 2);
|
exitStatus = request(fullRequest, 2);
|
||||||
else if (fullRequest.contains("/seterror"))
|
else if (fullRequest.contains("/seterror"))
|
||||||
@@ -523,7 +476,7 @@ int main(int argc, char** argv) {
|
|||||||
else if (fullRequest.contains("/decorations"))
|
else if (fullRequest.contains("/decorations"))
|
||||||
exitStatus = request(fullRequest, 1);
|
exitStatus = request(fullRequest, 1);
|
||||||
else if (fullRequest.contains("/--help"))
|
else if (fullRequest.contains("/--help"))
|
||||||
std::println("{}", USAGE);
|
std::cout << USAGE << std::endl;
|
||||||
else if (fullRequest.contains("/rollinglog") && needRoll)
|
else if (fullRequest.contains("/rollinglog") && needRoll)
|
||||||
exitStatus = request(fullRequest, 0, true);
|
exitStatus = request(fullRequest, 0, true);
|
||||||
else {
|
else {
|
||||||
|
@@ -1,27 +1,10 @@
|
|||||||
executable(
|
executable('hyprctl', 'main.cpp',
|
||||||
'hyprctl',
|
|
||||||
'main.cpp',
|
|
||||||
dependencies: [
|
dependencies: [
|
||||||
dependency('hyprutils', version: '>= 0.1.1'),
|
dependency('hyprutils', version: '>= 0.1.1'),
|
||||||
dependency('re2', required: true)
|
|
||||||
],
|
],
|
||||||
install: true,
|
install: true
|
||||||
)
|
)
|
||||||
|
|
||||||
install_data(
|
install_data('hyprctl.bash', install_dir: join_paths(get_option('datadir'), 'bash-completion/completions'), install_tag: 'runtime', rename: 'hyprctl')
|
||||||
'hyprctl.bash',
|
install_data('hyprctl.fish', install_dir: join_paths(get_option('datadir'), 'fish/vendor_completions.d'), install_tag: 'runtime')
|
||||||
install_dir: join_paths(get_option('datadir'), 'bash-completion/completions'),
|
install_data('hyprctl.zsh', install_dir: join_paths(get_option('datadir'), 'zsh/site-functions'), install_tag: 'runtime', rename: '_hyprctl')
|
||||||
install_tag: 'runtime',
|
|
||||||
rename: 'hyprctl',
|
|
||||||
)
|
|
||||||
install_data(
|
|
||||||
'hyprctl.fish',
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'fish/vendor_completions.d'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
)
|
|
||||||
install_data(
|
|
||||||
'hyprctl.zsh',
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'zsh/site-functions'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
rename: '_hyprctl',
|
|
||||||
)
|
|
||||||
|
@@ -9,25 +9,11 @@ file(GLOB_RECURSE SRCFILES CONFIGURE_DEPENDS "src/*.cpp")
|
|||||||
|
|
||||||
set(CMAKE_CXX_STANDARD 23)
|
set(CMAKE_CXX_STANDARD 23)
|
||||||
|
|
||||||
pkg_check_modules(hyprpm_deps REQUIRED IMPORTED_TARGET tomlplusplus hyprutils>=0.7.0)
|
pkg_check_modules(deps REQUIRED IMPORTED_TARGET tomlplusplus hyprutils>=0.1.1)
|
||||||
|
|
||||||
find_package(glaze QUIET)
|
|
||||||
if (NOT glaze_FOUND)
|
|
||||||
set(GLAZE_VERSION v5.1.1)
|
|
||||||
message(STATUS "glaze dependency not found, retrieving ${GLAZE_VERSION} with FetchContent")
|
|
||||||
include(FetchContent)
|
|
||||||
FetchContent_Declare(
|
|
||||||
glaze
|
|
||||||
GIT_REPOSITORY https://github.com/stephenberry/glaze.git
|
|
||||||
GIT_TAG ${GLAZE_VERSION}
|
|
||||||
GIT_SHALLOW TRUE
|
|
||||||
)
|
|
||||||
FetchContent_MakeAvailable(glaze)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
add_executable(hyprpm ${SRCFILES})
|
add_executable(hyprpm ${SRCFILES})
|
||||||
|
|
||||||
target_link_libraries(hyprpm PUBLIC PkgConfig::hyprpm_deps glaze::glaze)
|
target_link_libraries(hyprpm PUBLIC PkgConfig::deps)
|
||||||
|
|
||||||
# binary
|
# binary
|
||||||
install(TARGETS hyprpm)
|
install(TARGETS hyprpm)
|
||||||
|
@@ -14,7 +14,7 @@ hyprpm [<FLAGS>]... <ARGUMENT>
|
|||||||
| (list) "List all installed plugins"
|
| (list) "List all installed plugins"
|
||||||
| (enable <PLUGINS>) "Load a plugin"
|
| (enable <PLUGINS>) "Load a plugin"
|
||||||
| (disable <PLUGINS>) "Unload a plugin"
|
| (disable <PLUGINS>) "Unload a plugin"
|
||||||
| (reload) "Reload plugins to match the enabled/disabled state. Use -f to force reload."
|
| (reload) "Reload all plugins"
|
||||||
;
|
;
|
||||||
|
|
||||||
<PLUGINS> ::= {{{ hyprpm list | awk '/Plugin/{print $4}' }}};
|
<PLUGINS> ::= {{{ hyprpm list | awk '/Plugin/{print $4}' }}};
|
||||||
|
@@ -1,91 +1,45 @@
|
|||||||
#include "DataState.hpp"
|
#include "DataState.hpp"
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <toml++/toml.hpp>
|
#include <toml++/toml.hpp>
|
||||||
#include <print>
|
#include <iostream>
|
||||||
#include <sstream>
|
#include <filesystem>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include "PluginManager.hpp"
|
#include "PluginManager.hpp"
|
||||||
#include "../helpers/Die.hpp"
|
|
||||||
#include "../helpers/Sys.hpp"
|
|
||||||
#include "../helpers/StringUtils.hpp"
|
|
||||||
|
|
||||||
static std::string getTempRoot() {
|
std::string DataState::getDataStatePath() {
|
||||||
static auto ENV = getenv("XDG_RUNTIME_DIR");
|
const auto HOME = getenv("HOME");
|
||||||
if (!ENV) {
|
if (!HOME) {
|
||||||
std::cerr << "\nERROR: XDG_RUNTIME_DIR not set!\n";
|
std::cerr << "DataState: no $HOME\n";
|
||||||
exit(1);
|
throw std::runtime_error("no $HOME");
|
||||||
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto STR = ENV + std::string{"/hyprpm/"};
|
const auto XDG_DATA_HOME = getenv("XDG_DATA_HOME");
|
||||||
|
|
||||||
return STR;
|
if (XDG_DATA_HOME)
|
||||||
}
|
return std::string{XDG_DATA_HOME} + "/hyprpm";
|
||||||
|
return std::string{HOME} + "/.local/share/hyprpm";
|
||||||
// write the state to a file
|
|
||||||
static bool writeState(const std::string& str, const std::string& to) {
|
|
||||||
// create temp file in a safe temp root
|
|
||||||
std::ofstream of(getTempRoot() + ".temp-state", std::ios::trunc);
|
|
||||||
if (!of.good())
|
|
||||||
return false;
|
|
||||||
|
|
||||||
of << str;
|
|
||||||
of.close();
|
|
||||||
|
|
||||||
return NSys::root::install(getTempRoot() + ".temp-state", to, "644");
|
|
||||||
}
|
|
||||||
|
|
||||||
std::filesystem::path DataState::getDataStatePath() {
|
|
||||||
return std::filesystem::path("/var/cache/hyprpm/" + g_pPluginManager->m_szUsername);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string DataState::getHeadersPath() {
|
std::string DataState::getHeadersPath() {
|
||||||
return getDataStatePath() / "headersRoot";
|
return getDataStatePath() + "/headersRoot";
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<std::filesystem::path> DataState::getPluginStates() {
|
|
||||||
ensureStateStoreExists();
|
|
||||||
|
|
||||||
std::vector<std::filesystem::path> states;
|
|
||||||
for (const auto& entry : std::filesystem::directory_iterator(getDataStatePath())) {
|
|
||||||
if (!entry.is_directory() || entry.path().stem() == "headersRoot")
|
|
||||||
continue;
|
|
||||||
|
|
||||||
const auto stateFile = entry.path() / "state.toml";
|
|
||||||
if (!std::filesystem::exists(stateFile))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
states.emplace_back(stateFile);
|
|
||||||
}
|
|
||||||
return states;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DataState::ensureStateStoreExists() {
|
void DataState::ensureStateStoreExists() {
|
||||||
std::error_code ec;
|
const auto PATH = getDataStatePath();
|
||||||
if (!std::filesystem::exists(getHeadersPath(), ec) || ec) {
|
|
||||||
std::println("{}", infoString("The hyprpm state store doesn't exist. Creating now..."));
|
if (!std::filesystem::exists(PATH))
|
||||||
if (!std::filesystem::exists("/var/cache/hyprpm/", ec) || ec) {
|
std::filesystem::create_directories(PATH);
|
||||||
if (!NSys::root::createDirectory("/var/cache/hyprpm", "755"))
|
|
||||||
Debug::die("ensureStateStoreExists: Failed to run a superuser cmd");
|
if (!std::filesystem::exists(getHeadersPath()))
|
||||||
}
|
std::filesystem::create_directories(getHeadersPath());
|
||||||
if (!std::filesystem::exists(getDataStatePath(), ec) || ec) {
|
|
||||||
if (!NSys::root::createDirectory(getDataStatePath().string(), "755"))
|
|
||||||
Debug::die("ensureStateStoreExists: Failed to run a superuser cmd");
|
|
||||||
}
|
|
||||||
if (!NSys::root::createDirectory(getHeadersPath(), "755"))
|
|
||||||
Debug::die("ensureStateStoreExists: Failed to run a superuser cmd");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DataState::addNewPluginRepo(const SPluginRepository& repo) {
|
void DataState::addNewPluginRepo(const SPluginRepository& repo) {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
const auto PATH = getDataStatePath() / repo.name;
|
const auto PATH = getDataStatePath() + "/" + repo.name;
|
||||||
|
|
||||||
std::error_code ec;
|
std::filesystem::create_directories(PATH);
|
||||||
if (!std::filesystem::exists(PATH, ec) || ec) {
|
|
||||||
if (!NSys::root::createDirectory(PATH.string(), "755"))
|
|
||||||
Debug::die("addNewPluginRepo: failed to create cache dir");
|
|
||||||
}
|
|
||||||
// clang-format off
|
// clang-format off
|
||||||
auto DATA = toml::table{
|
auto DATA = toml::table{
|
||||||
{"repository", toml::table{
|
{"repository", toml::table{
|
||||||
@@ -95,37 +49,40 @@ void DataState::addNewPluginRepo(const SPluginRepository& repo) {
|
|||||||
{"rev", repo.rev}
|
{"rev", repo.rev}
|
||||||
}}
|
}}
|
||||||
};
|
};
|
||||||
for (auto const& p : repo.plugins) {
|
for (auto& p : repo.plugins) {
|
||||||
const auto filename = p.name + ".so";
|
// copy .so to the good place
|
||||||
|
if (std::filesystem::exists(p.filename))
|
||||||
// copy .so to the good place and chmod 755
|
std::filesystem::copy_file(p.filename, PATH + "/" + p.name + ".so");
|
||||||
if (std::filesystem::exists(p.filename)) {
|
|
||||||
if (!NSys::root::install(p.filename, (PATH / filename).string(), "0755"))
|
|
||||||
Debug::die("addNewPluginRepo: failed to install so file");
|
|
||||||
}
|
|
||||||
|
|
||||||
DATA.emplace(p.name, toml::table{
|
DATA.emplace(p.name, toml::table{
|
||||||
{"filename", filename},
|
{"filename", p.name + ".so"},
|
||||||
{"enabled", p.enabled},
|
{"enabled", p.enabled},
|
||||||
{"failed", p.failed}
|
{"failed", p.failed}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// clang-format on
|
// clang-format on
|
||||||
|
|
||||||
std::stringstream ss;
|
std::ofstream ofs(PATH + "/state.toml", std::ios::trunc);
|
||||||
ss << DATA;
|
ofs << DATA;
|
||||||
|
ofs.close();
|
||||||
if (!writeState(ss.str(), (PATH / "state.toml").string()))
|
|
||||||
Debug::die("{}", failureString("Failed to write plugin state"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DataState::pluginRepoExists(const std::string& urlOrName) {
|
bool DataState::pluginRepoExists(const std::string& urlOrName) {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
for (const auto& stateFile : getPluginStates()) {
|
const auto PATH = getDataStatePath();
|
||||||
const auto STATE = toml::parse_file(stateFile.c_str());
|
|
||||||
const auto NAME = STATE["repository"]["name"].value_or("");
|
for (const auto& entry : std::filesystem::directory_iterator(PATH)) {
|
||||||
const auto URL = STATE["repository"]["url"].value_or("");
|
if (!entry.is_directory() || entry.path().stem() == "headersRoot")
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(entry.path().string() + "/state.toml"))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto STATE = toml::parse_file(entry.path().string() + "/state.toml");
|
||||||
|
|
||||||
|
const auto NAME = STATE["repository"]["name"].value_or("");
|
||||||
|
const auto URL = STATE["repository"]["url"].value_or("");
|
||||||
|
|
||||||
if (URL == urlOrName || NAME == urlOrName)
|
if (URL == urlOrName || NAME == urlOrName)
|
||||||
return true;
|
return true;
|
||||||
@@ -137,29 +94,31 @@ bool DataState::pluginRepoExists(const std::string& urlOrName) {
|
|||||||
void DataState::removePluginRepo(const std::string& urlOrName) {
|
void DataState::removePluginRepo(const std::string& urlOrName) {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
for (const auto& stateFile : getPluginStates()) {
|
const auto PATH = getDataStatePath();
|
||||||
const auto STATE = toml::parse_file(stateFile.c_str());
|
|
||||||
const auto NAME = STATE["repository"]["name"].value_or("");
|
for (const auto& entry : std::filesystem::directory_iterator(PATH)) {
|
||||||
const auto URL = STATE["repository"]["url"].value_or("");
|
if (!entry.is_directory() || entry.path().stem() == "headersRoot")
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(entry.path().string() + "/state.toml"))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto STATE = toml::parse_file(entry.path().string() + "/state.toml");
|
||||||
|
|
||||||
|
const auto NAME = STATE["repository"]["name"].value_or("");
|
||||||
|
const auto URL = STATE["repository"]["url"].value_or("");
|
||||||
|
|
||||||
if (URL == urlOrName || NAME == urlOrName) {
|
if (URL == urlOrName || NAME == urlOrName) {
|
||||||
|
|
||||||
// unload the plugins!!
|
// unload the plugins!!
|
||||||
for (const auto& file : std::filesystem::directory_iterator(stateFile.parent_path())) {
|
for (const auto& file : std::filesystem::directory_iterator(entry.path())) {
|
||||||
if (!file.path().string().ends_with(".so"))
|
if (!file.path().string().ends_with(".so"))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
g_pPluginManager->loadUnloadPlugin(std::filesystem::absolute(file.path()), false);
|
g_pPluginManager->loadUnloadPlugin(std::filesystem::absolute(file.path()), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto PATH = stateFile.parent_path().string();
|
std::filesystem::remove_all(entry.path());
|
||||||
|
|
||||||
if (!PATH.starts_with("/var/cache/hyprpm") || PATH.contains('\''))
|
|
||||||
return; // WTF?
|
|
||||||
|
|
||||||
// scary!
|
|
||||||
if (!NSys::root::removeRecursive(PATH))
|
|
||||||
Debug::die("removePluginRepo: failed to remove dir");
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -168,13 +127,9 @@ void DataState::removePluginRepo(const std::string& urlOrName) {
|
|||||||
void DataState::updateGlobalState(const SGlobalState& state) {
|
void DataState::updateGlobalState(const SGlobalState& state) {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
const auto PATH = getDataStatePath();
|
const auto PATH = getDataStatePath();
|
||||||
|
|
||||||
std::error_code ec;
|
std::filesystem::create_directories(PATH);
|
||||||
if (!std::filesystem::exists(PATH, ec) || ec) {
|
|
||||||
if (!NSys::root::createDirectory(PATH.string(), "755"))
|
|
||||||
Debug::die("updateGlobalState: failed to create dir");
|
|
||||||
}
|
|
||||||
// clang-format off
|
// clang-format off
|
||||||
auto DATA = toml::table{
|
auto DATA = toml::table{
|
||||||
{"state", toml::table{
|
{"state", toml::table{
|
||||||
@@ -184,23 +139,20 @@ void DataState::updateGlobalState(const SGlobalState& state) {
|
|||||||
};
|
};
|
||||||
// clang-format on
|
// clang-format on
|
||||||
|
|
||||||
std::stringstream ss;
|
std::ofstream ofs(PATH + "/state.toml", std::ios::trunc);
|
||||||
ss << DATA;
|
ofs << DATA;
|
||||||
|
ofs.close();
|
||||||
if (!writeState(ss.str(), (PATH / "state.toml").string()))
|
|
||||||
Debug::die("{}", failureString("Failed to write plugin state"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
SGlobalState DataState::getGlobalState() {
|
SGlobalState DataState::getGlobalState() {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
const auto stateFile = getDataStatePath() / "state.toml";
|
const auto PATH = getDataStatePath();
|
||||||
|
|
||||||
std::error_code ec;
|
if (!std::filesystem::exists(PATH + "/state.toml"))
|
||||||
if (!std::filesystem::exists(stateFile, ec) || ec)
|
|
||||||
return SGlobalState{};
|
return SGlobalState{};
|
||||||
|
|
||||||
auto DATA = toml::parse_file(stateFile.c_str());
|
auto DATA = toml::parse_file(PATH + "/state.toml");
|
||||||
|
|
||||||
SGlobalState state;
|
SGlobalState state;
|
||||||
state.headersHashCompiled = DATA["state"]["hash"].value_or("");
|
state.headersHashCompiled = DATA["state"]["hash"].value_or("");
|
||||||
@@ -212,9 +164,18 @@ SGlobalState DataState::getGlobalState() {
|
|||||||
std::vector<SPluginRepository> DataState::getAllRepositories() {
|
std::vector<SPluginRepository> DataState::getAllRepositories() {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
|
const auto PATH = getDataStatePath();
|
||||||
|
|
||||||
std::vector<SPluginRepository> repos;
|
std::vector<SPluginRepository> repos;
|
||||||
for (const auto& stateFile : getPluginStates()) {
|
|
||||||
const auto STATE = toml::parse_file(stateFile.c_str());
|
for (const auto& entry : std::filesystem::directory_iterator(PATH)) {
|
||||||
|
if (!entry.is_directory() || entry.path().stem() == "headersRoot")
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(entry.path().string() + "/state.toml"))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto STATE = toml::parse_file(entry.path().string() + "/state.toml");
|
||||||
|
|
||||||
const auto NAME = STATE["repository"]["name"].value_or("");
|
const auto NAME = STATE["repository"]["name"].value_or("");
|
||||||
const auto URL = STATE["repository"]["url"].value_or("");
|
const auto URL = STATE["repository"]["url"].value_or("");
|
||||||
@@ -247,8 +208,17 @@ std::vector<SPluginRepository> DataState::getAllRepositories() {
|
|||||||
bool DataState::setPluginEnabled(const std::string& name, bool enabled) {
|
bool DataState::setPluginEnabled(const std::string& name, bool enabled) {
|
||||||
ensureStateStoreExists();
|
ensureStateStoreExists();
|
||||||
|
|
||||||
for (const auto& stateFile : getPluginStates()) {
|
const auto PATH = getDataStatePath();
|
||||||
const auto STATE = toml::parse_file(stateFile.c_str());
|
|
||||||
|
for (const auto& entry : std::filesystem::directory_iterator(PATH)) {
|
||||||
|
if (!entry.is_directory() || entry.path().stem() == "headersRoot")
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(entry.path().string() + "/state.toml"))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto STATE = toml::parse_file(entry.path().string() + "/state.toml");
|
||||||
|
|
||||||
for (const auto& [key, val] : STATE) {
|
for (const auto& [key, val] : STATE) {
|
||||||
if (key == "repository")
|
if (key == "repository")
|
||||||
continue;
|
continue;
|
||||||
@@ -261,33 +231,15 @@ bool DataState::setPluginEnabled(const std::string& name, bool enabled) {
|
|||||||
if (FAILED)
|
if (FAILED)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
auto modifiedState = STATE;
|
(*STATE[key].as_table()).insert_or_assign("enabled", enabled);
|
||||||
(*modifiedState[key].as_table()).insert_or_assign("enabled", enabled);
|
|
||||||
|
|
||||||
std::stringstream ss;
|
std::ofstream state(entry.path().string() + "/state.toml", std::ios::trunc);
|
||||||
ss << modifiedState;
|
state << STATE;
|
||||||
|
state.close();
|
||||||
if (!writeState(ss.str(), stateFile.string()))
|
|
||||||
Debug::die("{}", failureString("Failed to write plugin state"));
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DataState::purgeAllCache() {
|
|
||||||
std::error_code ec;
|
|
||||||
if (!std::filesystem::exists(getDataStatePath()) && !ec) {
|
|
||||||
std::println("{}", infoString("Nothing to do"));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto PATH = getDataStatePath().string();
|
|
||||||
if (PATH.contains('\''))
|
|
||||||
return;
|
|
||||||
// scary!
|
|
||||||
if (!NSys::root::removeRecursive(PATH))
|
|
||||||
Debug::die("Failed to run a superuser cmd");
|
|
||||||
}
|
|
@@ -1,5 +1,4 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <filesystem>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "Plugin.hpp"
|
#include "Plugin.hpp"
|
||||||
@@ -10,16 +9,14 @@ struct SGlobalState {
|
|||||||
};
|
};
|
||||||
|
|
||||||
namespace DataState {
|
namespace DataState {
|
||||||
std::filesystem::path getDataStatePath();
|
std::string getDataStatePath();
|
||||||
std::string getHeadersPath();
|
std::string getHeadersPath();
|
||||||
std::vector<std::filesystem::path> getPluginStates();
|
void ensureStateStoreExists();
|
||||||
void ensureStateStoreExists();
|
void addNewPluginRepo(const SPluginRepository& repo);
|
||||||
void addNewPluginRepo(const SPluginRepository& repo);
|
void removePluginRepo(const std::string& urlOrName);
|
||||||
void removePluginRepo(const std::string& urlOrName);
|
bool pluginRepoExists(const std::string& urlOrName);
|
||||||
bool pluginRepoExists(const std::string& urlOrName);
|
void updateGlobalState(const SGlobalState& state);
|
||||||
void updateGlobalState(const SGlobalState& state);
|
SGlobalState getGlobalState();
|
||||||
void purgeAllCache();
|
bool setPluginEnabled(const std::string& name, bool enabled);
|
||||||
SGlobalState getGlobalState();
|
std::vector<SPluginRepository> getAllRepositories();
|
||||||
bool setPluginEnabled(const std::string& name, bool enabled);
|
|
||||||
std::vector<SPluginRepository> getAllRepositories();
|
|
||||||
};
|
};
|
@@ -1,86 +0,0 @@
|
|||||||
#include "HyprlandSocket.hpp"
|
|
||||||
#include <pwd.h>
|
|
||||||
#include <sys/socket.h>
|
|
||||||
#include "../helpers/StringUtils.hpp"
|
|
||||||
#include <print>
|
|
||||||
#include <sys/un.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <cstring>
|
|
||||||
|
|
||||||
static int getUID() {
|
|
||||||
const auto UID = getuid();
|
|
||||||
const auto PWUID = getpwuid(UID);
|
|
||||||
return PWUID ? PWUID->pw_uid : UID;
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::string getRuntimeDir() {
|
|
||||||
const auto XDG = getenv("XDG_RUNTIME_DIR");
|
|
||||||
|
|
||||||
if (!XDG) {
|
|
||||||
const std::string USERID = std::to_string(getUID());
|
|
||||||
return "/run/user/" + USERID + "/hypr";
|
|
||||||
}
|
|
||||||
|
|
||||||
return std::string{XDG} + "/hypr";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string NHyprlandSocket::send(const std::string& cmd) {
|
|
||||||
const auto SERVERSOCKET = socket(AF_UNIX, SOCK_STREAM, 0);
|
|
||||||
|
|
||||||
if (SERVERSOCKET < 0) {
|
|
||||||
std::println("{}", failureString("Couldn't open a socket (1)"));
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto HIS = getenv("HYPRLAND_INSTANCE_SIGNATURE");
|
|
||||||
|
|
||||||
if (!HIS) {
|
|
||||||
std::println("{}", failureString("HYPRLAND_INSTANCE_SIGNATURE was not set! (Is Hyprland running?) (3)"));
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
sockaddr_un serverAddress = {0};
|
|
||||||
serverAddress.sun_family = AF_UNIX;
|
|
||||||
|
|
||||||
std::string socketPath = getRuntimeDir() + "/" + HIS + "/.socket.sock";
|
|
||||||
|
|
||||||
strncpy(serverAddress.sun_path, socketPath.c_str(), sizeof(serverAddress.sun_path) - 1);
|
|
||||||
|
|
||||||
if (connect(SERVERSOCKET, (sockaddr*)&serverAddress, SUN_LEN(&serverAddress)) < 0) {
|
|
||||||
std::println("{}", failureString("Couldn't connect to " + socketPath + ". (4)"));
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
auto sizeWritten = write(SERVERSOCKET, cmd.c_str(), cmd.length());
|
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
|
||||||
std::println("{}", failureString("Couldn't write (5)"));
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string reply = "";
|
|
||||||
constexpr size_t BUFFER_SIZE = 8192;
|
|
||||||
char buffer[BUFFER_SIZE] = {0};
|
|
||||||
|
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, BUFFER_SIZE);
|
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
|
||||||
std::println("{}", failureString("Couldn't read (6)"));
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
reply += std::string(buffer, sizeWritten);
|
|
||||||
|
|
||||||
while (sizeWritten == BUFFER_SIZE) {
|
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, BUFFER_SIZE);
|
|
||||||
if (sizeWritten < 0) {
|
|
||||||
std::println("{}", failureString("Couldn't read (7)"));
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
reply += std::string(buffer, sizeWritten);
|
|
||||||
}
|
|
||||||
|
|
||||||
close(SERVERSOCKET);
|
|
||||||
|
|
||||||
return reply;
|
|
||||||
}
|
|
@@ -1,7 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace NHyprlandSocket {
|
|
||||||
std::string send(const std::string& cmd);
|
|
||||||
};
|
|
@@ -1,33 +1,21 @@
|
|||||||
#include "Manifest.hpp"
|
#include "Manifest.hpp"
|
||||||
#include <toml++/toml.hpp>
|
#include <toml++/toml.hpp>
|
||||||
#include <algorithm>
|
#include <iostream>
|
||||||
|
|
||||||
// Alphanumerics and -_ allowed for plugin names. No magic names.
|
|
||||||
// [A-Za-z0-9\-_]*
|
|
||||||
static bool validManifestName(const std::string_view& n) {
|
|
||||||
return std::ranges::all_of(n, [](const char& c) { return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || c == '-' || c == '_' || c == '=' || (c >= '0' && c <= '9'); });
|
|
||||||
}
|
|
||||||
|
|
||||||
CManifest::CManifest(const eManifestType type, const std::string& path) {
|
CManifest::CManifest(const eManifestType type, const std::string& path) {
|
||||||
auto manifest = toml::parse_file(path);
|
auto manifest = toml::parse_file(path);
|
||||||
|
|
||||||
if (type == MANIFEST_HYPRLOAD) {
|
if (type == MANIFEST_HYPRLOAD) {
|
||||||
for (auto const& [key, val] : manifest) {
|
for (auto& [key, val] : manifest) {
|
||||||
if (key.str().ends_with(".build"))
|
if (key.str().ends_with(".build"))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
CManifest::SManifestPlugin plugin;
|
CManifest::SManifestPlugin plugin;
|
||||||
|
|
||||||
if (!validManifestName(key.str())) {
|
|
||||||
m_good = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
plugin.name = key;
|
plugin.name = key;
|
||||||
m_plugins.push_back(plugin);
|
m_vPlugins.push_back(plugin);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto& plugin : m_plugins) {
|
for (auto& plugin : m_vPlugins) {
|
||||||
plugin.description = manifest[plugin.name]["description"].value_or("?");
|
plugin.description = manifest[plugin.name]["description"].value_or("?");
|
||||||
plugin.version = manifest[plugin.name]["version"].value_or("?");
|
plugin.version = manifest[plugin.name]["version"].value_or("?");
|
||||||
plugin.output = manifest[plugin.name]["build"]["output"].value_or("?");
|
plugin.output = manifest[plugin.name]["build"]["output"].value_or("?");
|
||||||
@@ -49,21 +37,21 @@ CManifest::CManifest(const eManifestType type, const std::string& path) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (plugin.output.empty() || plugin.buildSteps.empty()) {
|
if (plugin.output.empty() || plugin.buildSteps.empty()) {
|
||||||
m_good = false;
|
m_bGood = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (type == MANIFEST_HYPRPM) {
|
} else if (type == MANIFEST_HYPRPM) {
|
||||||
m_repository.name = manifest["repository"]["name"].value_or("");
|
m_sRepository.name = manifest["repository"]["name"].value_or("");
|
||||||
auto authors = manifest["repository"]["authors"].as_array();
|
auto authors = manifest["repository"]["authors"].as_array();
|
||||||
if (authors) {
|
if (authors) {
|
||||||
for (auto&& a : *authors) {
|
for (auto&& a : *authors) {
|
||||||
m_repository.authors.push_back(a.as_string()->value_or("?"));
|
m_sRepository.authors.push_back(a.as_string()->value_or("?"));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
auto author = manifest["repository"]["author"].value_or("");
|
auto author = manifest["repository"]["author"].value_or("");
|
||||||
if (!std::string{author}.empty())
|
if (!std::string{author}.empty())
|
||||||
m_repository.authors.push_back(author);
|
m_sRepository.authors.push_back(author);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto pins = manifest["repository"]["commit_pins"].as_array();
|
auto pins = manifest["repository"]["commit_pins"].as_array();
|
||||||
@@ -71,26 +59,20 @@ CManifest::CManifest(const eManifestType type, const std::string& path) {
|
|||||||
for (auto&& pin : *pins) {
|
for (auto&& pin : *pins) {
|
||||||
auto pinArr = pin.as_array();
|
auto pinArr = pin.as_array();
|
||||||
if (pinArr && pinArr->get(1))
|
if (pinArr && pinArr->get(1))
|
||||||
m_repository.commitPins.push_back(std::make_pair<>(pinArr->get(0)->as_string()->get(), pinArr->get(1)->as_string()->get()));
|
m_sRepository.commitPins.push_back(std::make_pair<>(pinArr->get(0)->as_string()->get(), pinArr->get(1)->as_string()->get()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto const& [key, val] : manifest) {
|
for (auto& [key, val] : manifest) {
|
||||||
if (key.str() == "repository")
|
if (key.str() == "repository")
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
CManifest::SManifestPlugin plugin;
|
CManifest::SManifestPlugin plugin;
|
||||||
|
|
||||||
if (!validManifestName(key.str())) {
|
|
||||||
m_good = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
plugin.name = key;
|
plugin.name = key;
|
||||||
m_plugins.push_back(plugin);
|
m_vPlugins.push_back(plugin);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto& plugin : m_plugins) {
|
for (auto& plugin : m_vPlugins) {
|
||||||
plugin.description = manifest[plugin.name]["description"].value_or("?");
|
plugin.description = manifest[plugin.name]["description"].value_or("?");
|
||||||
plugin.output = manifest[plugin.name]["output"].value_or("?");
|
plugin.output = manifest[plugin.name]["output"].value_or("?");
|
||||||
plugin.since = manifest[plugin.name]["since_hyprland"].value_or(0);
|
plugin.since = manifest[plugin.name]["since_hyprland"].value_or(0);
|
||||||
@@ -112,12 +94,12 @@ CManifest::CManifest(const eManifestType type, const std::string& path) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (plugin.output.empty() || plugin.buildSteps.empty()) {
|
if (plugin.output.empty() || plugin.buildSteps.empty()) {
|
||||||
m_good = false;
|
m_bGood = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// ???
|
// ???
|
||||||
m_good = false;
|
m_bGood = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -27,8 +27,8 @@ class CManifest {
|
|||||||
std::string name;
|
std::string name;
|
||||||
std::vector<std::string> authors;
|
std::vector<std::string> authors;
|
||||||
std::vector<std::pair<std::string, std::string>> commitPins;
|
std::vector<std::pair<std::string, std::string>> commitPins;
|
||||||
} m_repository;
|
} m_sRepository;
|
||||||
|
|
||||||
std::vector<SManifestPlugin> m_plugins;
|
std::vector<SManifestPlugin> m_vPlugins;
|
||||||
bool m_good = true;
|
bool m_bGood = true;
|
||||||
};
|
};
|
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <utility>
|
|
||||||
|
|
||||||
enum eHeadersErrors {
|
enum eHeadersErrors {
|
||||||
HEADERS_OK = 0,
|
HEADERS_OK = 0,
|
||||||
@@ -27,8 +26,7 @@ enum ePluginLoadStateReturn {
|
|||||||
LOADSTATE_OK = 0,
|
LOADSTATE_OK = 0,
|
||||||
LOADSTATE_FAIL,
|
LOADSTATE_FAIL,
|
||||||
LOADSTATE_PARTIAL_FAIL,
|
LOADSTATE_PARTIAL_FAIL,
|
||||||
LOADSTATE_HEADERS_OUTDATED,
|
LOADSTATE_HEADERS_OUTDATED
|
||||||
LOADSTATE_HYPRLAND_UPDATED
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct SHyprlandVersion {
|
struct SHyprlandVersion {
|
||||||
@@ -40,8 +38,6 @@ struct SHyprlandVersion {
|
|||||||
|
|
||||||
class CPluginManager {
|
class CPluginManager {
|
||||||
public:
|
public:
|
||||||
CPluginManager();
|
|
||||||
|
|
||||||
bool addNewPluginRepo(const std::string& url, const std::string& rev);
|
bool addNewPluginRepo(const std::string& url, const std::string& rev);
|
||||||
bool removePluginRepo(const std::string& urlOrName);
|
bool removePluginRepo(const std::string& urlOrName);
|
||||||
|
|
||||||
@@ -53,10 +49,10 @@ class CPluginManager {
|
|||||||
|
|
||||||
bool enablePlugin(const std::string& name);
|
bool enablePlugin(const std::string& name);
|
||||||
bool disablePlugin(const std::string& name);
|
bool disablePlugin(const std::string& name);
|
||||||
ePluginLoadStateReturn ensurePluginsLoadState(bool forceReload = false);
|
ePluginLoadStateReturn ensurePluginsLoadState();
|
||||||
|
|
||||||
bool loadUnloadPlugin(const std::string& path, bool load);
|
bool loadUnloadPlugin(const std::string& path, bool load);
|
||||||
SHyprlandVersion getHyprlandVersion(bool running = true);
|
SHyprlandVersion getHyprlandVersion();
|
||||||
|
|
||||||
void notify(const eNotifyIcons icon, uint32_t color, int durationMs, const std::string& message);
|
void notify(const eNotifyIcons icon, uint32_t color, int durationMs, const std::string& message);
|
||||||
|
|
||||||
@@ -64,7 +60,6 @@ class CPluginManager {
|
|||||||
|
|
||||||
bool m_bVerbose = false;
|
bool m_bVerbose = false;
|
||||||
bool m_bNoShallow = false;
|
bool m_bNoShallow = false;
|
||||||
std::string m_szCustomHlUrl, m_szUsername;
|
|
||||||
|
|
||||||
// will delete recursively if exists!!
|
// will delete recursively if exists!!
|
||||||
bool createSafeDirectory(const std::string& path);
|
bool createSafeDirectory(const std::string& path);
|
||||||
@@ -73,7 +68,7 @@ class CPluginManager {
|
|||||||
std::string headerError(const eHeadersErrors err);
|
std::string headerError(const eHeadersErrors err);
|
||||||
std::string headerErrorShort(const eHeadersErrors err);
|
std::string headerErrorShort(const eHeadersErrors err);
|
||||||
|
|
||||||
std::string m_szWorkingPluginDirectory;
|
std::string m_szWorkingPluginDirectory = "";
|
||||||
};
|
};
|
||||||
|
|
||||||
inline std::unique_ptr<CPluginManager> g_pPluginManager;
|
inline std::unique_ptr<CPluginManager> g_pPluginManager;
|
||||||
|
@@ -1,15 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <format>
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
// NOLINTNEXTLINE
|
|
||||||
namespace Debug {
|
|
||||||
template <typename... Args>
|
|
||||||
void die(std::format_string<Args...> fmt, Args&&... args) {
|
|
||||||
const std::string logMsg = std::vformat(fmt.get(), std::make_format_args(args...));
|
|
||||||
|
|
||||||
std::cout << "\n[ERR] " << logMsg << "\n";
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
};
|
|
@@ -1,32 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <format>
|
|
||||||
#include <string>
|
|
||||||
#include "Colors.hpp"
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string statusString(const std::string_view emoji, const std::string_view color, const std::string_view fmt, Args&&... args) {
|
|
||||||
std::string ret = std::format("{}{}{} ", color, emoji, Colors::RESET);
|
|
||||||
ret += std::vformat(fmt, std::make_format_args(args...));
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string successString(const std::string_view fmt, Args&&... args) {
|
|
||||||
return statusString("✔", Colors::GREEN, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string failureString(const std::string_view fmt, Args&&... args) {
|
|
||||||
return statusString("✖", Colors::RED, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string verboseString(const std::string_view fmt, Args&&... args) {
|
|
||||||
return statusString("[v]", Colors::BLUE, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string infoString(const std::string_view fmt, Args&&... args) {
|
|
||||||
return statusString("→", Colors::RESET, fmt, args...);
|
|
||||||
}
|
|
@@ -1,168 +0,0 @@
|
|||||||
#include "Sys.hpp"
|
|
||||||
#include "Die.hpp"
|
|
||||||
#include "StringUtils.hpp"
|
|
||||||
|
|
||||||
#include <pwd.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <sstream>
|
|
||||||
#include <print>
|
|
||||||
#include <filesystem>
|
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/string/VarList.hpp>
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::String;
|
|
||||||
|
|
||||||
inline constexpr std::array<std::string_view, 3> SUPERUSER_BINARIES = {
|
|
||||||
"sudo",
|
|
||||||
"doas",
|
|
||||||
"run0",
|
|
||||||
};
|
|
||||||
|
|
||||||
static std::string validSubinsAsStr() {
|
|
||||||
std::ostringstream oss;
|
|
||||||
auto it = SUPERUSER_BINARIES.begin();
|
|
||||||
if (it != SUPERUSER_BINARIES.end()) {
|
|
||||||
oss << *it++;
|
|
||||||
for (; it != SUPERUSER_BINARIES.end(); ++it)
|
|
||||||
oss << ", " << *it;
|
|
||||||
}
|
|
||||||
|
|
||||||
return oss.str();
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool executableExistsInPath(const std::string& exe) {
|
|
||||||
const char* PATHENV = std::getenv("PATH");
|
|
||||||
if (!PATHENV)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
CVarList paths(PATHENV, 0, ':', true);
|
|
||||||
std::error_code ec;
|
|
||||||
|
|
||||||
for (const auto& PATH : paths) {
|
|
||||||
std::filesystem::path candidate = std::filesystem::path(PATH) / exe;
|
|
||||||
if (!std::filesystem::exists(candidate, ec) || ec)
|
|
||||||
continue;
|
|
||||||
if (!std::filesystem::is_regular_file(candidate, ec) || ec)
|
|
||||||
continue;
|
|
||||||
auto perms = std::filesystem::status(candidate, ec).permissions();
|
|
||||||
if (ec)
|
|
||||||
continue;
|
|
||||||
if ((perms & std::filesystem::perms::others_exec) != std::filesystem::perms::none)
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::string subin() {
|
|
||||||
static std::string bin;
|
|
||||||
static bool once = true;
|
|
||||||
if (!once)
|
|
||||||
return bin;
|
|
||||||
|
|
||||||
for (const auto& BIN : SUPERUSER_BINARIES) {
|
|
||||||
if (!executableExistsInPath(std::string{BIN}))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
bin = BIN;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
once = false;
|
|
||||||
|
|
||||||
if (bin.empty())
|
|
||||||
Debug::die("{}", failureString("No valid superuser binary present. Supported: {}", validSubinsAsStr()));
|
|
||||||
|
|
||||||
return bin;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool verifyStringValid(const std::string& s) {
|
|
||||||
return std::ranges::none_of(s, [](const char& c) { return c == '`' || c == '$' || c == '(' || c == ')' || c == '\'' || c == '"'; });
|
|
||||||
}
|
|
||||||
|
|
||||||
int NSys::getUID() {
|
|
||||||
const auto UID = getuid();
|
|
||||||
const auto PWUID = getpwuid(UID);
|
|
||||||
return PWUID ? PWUID->pw_uid : UID;
|
|
||||||
}
|
|
||||||
|
|
||||||
int NSys::getEUID() {
|
|
||||||
const auto UID = geteuid();
|
|
||||||
const auto PWUID = getpwuid(UID);
|
|
||||||
return PWUID ? PWUID->pw_uid : UID;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool NSys::isSuperuser() {
|
|
||||||
return getuid() != geteuid() || geteuid() == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void NSys::root::cacheSudo() {
|
|
||||||
// "caches" the sudo so that the prompt later doesn't pop up in a weird spot
|
|
||||||
// sudo will not ask us again for a moment
|
|
||||||
CProcess proc(subin(), {"echo", "hyprland"});
|
|
||||||
proc.runSync();
|
|
||||||
}
|
|
||||||
|
|
||||||
void NSys::root::dropSudo() {
|
|
||||||
if (subin() != "sudo") {
|
|
||||||
std::println("{}", infoString("Don't know how to drop timestamp for '{}', ignoring.", subin()));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
CProcess proc(subin(), {"-k"});
|
|
||||||
proc.runSync();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool NSys::root::createDirectory(const std::string& path, const std::string& mode) {
|
|
||||||
if (!verifyStringValid(path))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!std::ranges::all_of(mode, [](const char& c) { return c >= '0' && c <= '9'; }))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
CProcess proc(subin(), {"mkdir", "-p", "-m", mode, path});
|
|
||||||
|
|
||||||
return proc.runSync() && proc.exitCode() == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool NSys::root::removeRecursive(const std::string& path) {
|
|
||||||
if (!verifyStringValid(path))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
std::error_code ec;
|
|
||||||
const std::string PATH_ABSOLUTE = std::filesystem::canonical(path, ec);
|
|
||||||
|
|
||||||
if (ec)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!PATH_ABSOLUTE.starts_with("/var/cache/hyprpm"))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
CProcess proc(subin(), {"rm", "-fr", PATH_ABSOLUTE});
|
|
||||||
|
|
||||||
return proc.runSync() && proc.exitCode() == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool NSys::root::install(const std::string& what, const std::string& where, const std::string& mode) {
|
|
||||||
if (!verifyStringValid(what) || !verifyStringValid(where))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!std::ranges::all_of(mode, [](const char& c) { return c >= '0' && c <= '9'; }))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
CProcess proc(subin(), {"install", "-m" + mode, "-o", "0", "-g", "0", what, where});
|
|
||||||
|
|
||||||
return proc.runSync() && proc.exitCode() == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string NSys::root::runAsSuperuserUnsafe(const std::string& cmd) {
|
|
||||||
CProcess proc(subin(), {"/bin/sh", "-c", cmd});
|
|
||||||
|
|
||||||
if (!proc.runSync())
|
|
||||||
return "";
|
|
||||||
|
|
||||||
return proc.stdOut();
|
|
||||||
}
|
|
@@ -1,23 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace NSys {
|
|
||||||
bool isSuperuser();
|
|
||||||
int getUID();
|
|
||||||
int getEUID();
|
|
||||||
|
|
||||||
// NOLINTNEXTLINE
|
|
||||||
namespace root {
|
|
||||||
void cacheSudo();
|
|
||||||
void dropSudo();
|
|
||||||
|
|
||||||
//
|
|
||||||
[[nodiscard("Discarding could lead to vulnerabilities and bugs")]] bool createDirectory(const std::string& path, const std::string& mode);
|
|
||||||
[[nodiscard("Discarding could lead to vulnerabilities and bugs")]] bool removeRecursive(const std::string& path);
|
|
||||||
[[nodiscard("Discarding could lead to vulnerabilities and bugs")]] bool install(const std::string& what, const std::string& where, const std::string& mode);
|
|
||||||
|
|
||||||
// Do not use this unless absolutely necessary!
|
|
||||||
std::string runAsSuperuserUnsafe(const std::string& cmd);
|
|
||||||
};
|
|
||||||
};
|
|
@@ -1,139 +1,105 @@
|
|||||||
|
#include "progress/CProgressBar.hpp"
|
||||||
#include "helpers/Colors.hpp"
|
#include "helpers/Colors.hpp"
|
||||||
#include "helpers/StringUtils.hpp"
|
|
||||||
#include "core/PluginManager.hpp"
|
#include "core/PluginManager.hpp"
|
||||||
#include "core/DataState.hpp"
|
#include "core/DataState.hpp"
|
||||||
#include "helpers/Sys.hpp"
|
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <print>
|
#include <chrono>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
#include <hyprutils/utils/ScopeGuard.hpp>
|
const std::string HELP = R"#(┏ hyprpm, a Hyprland Plugin Manager
|
||||||
using namespace Hyprutils::Utils;
|
|
||||||
|
|
||||||
constexpr std::string_view HELP = R"#(┏ hyprpm, a Hyprland Plugin Manager
|
|
||||||
┃
|
┃
|
||||||
┣ add [url] [git rev] → Install a new plugin repository from git. Git revision
|
┣ add [url] [git rev] → Install a new plugin repository from git. Git revision
|
||||||
┃ is optional, when set, commit locks are ignored.
|
┃ is optional, when set, commit locks are ignored.
|
||||||
┣ remove [url/name] → Remove an installed plugin repository.
|
┣ remove [url/name] → Remove an installed plugin repository
|
||||||
┣ enable [name] → Enable a plugin.
|
┣ enable [name] → Enable a plugin
|
||||||
┣ disable [name] → Disable a plugin.
|
┣ disable [name] → Disable a plugin
|
||||||
┣ update → Check and update all plugins if needed.
|
┣ update → Check and update all plugins if needed
|
||||||
┣ reload → Reload hyprpm state. Ensure all enabled plugins are loaded.
|
┣ reload → Reload hyprpm state. Ensure all enabled plugins are loaded.
|
||||||
┣ list → List all installed plugins.
|
┣ list → List all installed plugins
|
||||||
┣ purge-cache → Remove the entire hyprpm cache, built plugins, hyprpm settings and headers.
|
|
||||||
┃
|
┃
|
||||||
┣ Flags:
|
┣ Flags:
|
||||||
┃
|
┃
|
||||||
┣ --notify | -n → Send a hyprland notification for important events (including both successes and fail events).
|
┣ --notify | -n → Send a hyprland notification for important events (e.g. load fail)
|
||||||
┣ --notify-fail | -nn → Send a hyprland notification for fail events only.
|
┣ --help | -h → Show this menu
|
||||||
┣ --help | -h → Show this menu.
|
┣ --verbose | -v → Enable too much logging
|
||||||
┣ --verbose | -v → Enable too much logging.
|
┣ --force | -f → Force an operation ignoring checks (e.g. update -f)
|
||||||
┣ --force | -f → Force an operation ignoring checks (e.g. update -f).
|
┣ --no-shallow | -s → Disable shallow cloning of Hyprland sources
|
||||||
┣ --no-shallow | -s → Disable shallow cloning of Hyprland sources.
|
|
||||||
┣ --hl-url | → Pass a custom hyprland source url.
|
|
||||||
┗
|
┗
|
||||||
)#";
|
)#";
|
||||||
|
|
||||||
int main(int argc, char** argv, char** envp) {
|
int main(int argc, char** argv, char** envp) {
|
||||||
std::vector<std::string> ARGS{argc};
|
std::vector<std::string> ARGS{argc};
|
||||||
for (int i = 0; i < argc; ++i) {
|
for (int i = 0; i < argc; ++i) {
|
||||||
ARGS[i] = std::string{argv[i]};
|
ARGS[i] = std::string{argv[i]};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ARGS.size() < 2) {
|
if (ARGS.size() < 2) {
|
||||||
std::println(stderr, "{}", HELP);
|
std::cout << HELP;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<std::string> command;
|
std::vector<std::string> command;
|
||||||
bool notify = false, notifyFail = false, verbose = false, force = false, noShallow = false;
|
bool notify = false, verbose = false, force = false, noShallow = false;
|
||||||
std::string customHlUrl;
|
|
||||||
|
|
||||||
for (int i = 1; i < argc; ++i) {
|
for (int i = 1; i < argc; ++i) {
|
||||||
if (ARGS[i].starts_with("-")) {
|
if (ARGS[i].starts_with("-")) {
|
||||||
if (ARGS[i] == "--help" || ARGS[i] == "-h") {
|
if (ARGS[i] == "--help" || ARGS[i] == "-h") {
|
||||||
std::println("{}", HELP);
|
std::cout << HELP;
|
||||||
return 0;
|
return 0;
|
||||||
} else if (ARGS[i] == "--notify" || ARGS[i] == "-n") {
|
} else if (ARGS[i] == "--notify" || ARGS[i] == "-n") {
|
||||||
notify = true;
|
notify = true;
|
||||||
} else if (ARGS[i] == "--notify-fail" || ARGS[i] == "-nn") {
|
|
||||||
notifyFail = notify = true;
|
|
||||||
} else if (ARGS[i] == "--verbose" || ARGS[i] == "-v") {
|
} else if (ARGS[i] == "--verbose" || ARGS[i] == "-v") {
|
||||||
verbose = true;
|
verbose = true;
|
||||||
} else if (ARGS[i] == "--no-shallow" || ARGS[i] == "-s") {
|
} else if (ARGS[i] == "--no-shallow" || ARGS[i] == "-s") {
|
||||||
noShallow = true;
|
noShallow = true;
|
||||||
} else if (ARGS[i] == "--hl-url") {
|
|
||||||
if (i + 1 >= argc) {
|
|
||||||
std::println(stderr, "Missing argument for --hl-url");
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
customHlUrl = ARGS[i + 1];
|
|
||||||
i++;
|
|
||||||
} else if (ARGS[i] == "--force" || ARGS[i] == "-f") {
|
} else if (ARGS[i] == "--force" || ARGS[i] == "-f") {
|
||||||
force = true;
|
force = true;
|
||||||
std::println("{}", statusString("!", Colors::RED, "Using --force, I hope you know what you are doing."));
|
std::cout << Colors::RED << "!" << Colors::RESET << " Using --force, I hope you know what you are doing.\n";
|
||||||
} else {
|
} else {
|
||||||
std::println(stderr, "Unrecognized option {}", ARGS[i]);
|
std::cerr << "Unrecognized option " << ARGS[i] << "\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
} else
|
} else {
|
||||||
command.push_back(ARGS[i]);
|
command.push_back(ARGS[i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (command.empty()) {
|
if (command.empty()) {
|
||||||
std::println(stderr, "{}", HELP);
|
std::cout << HELP;
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_pPluginManager = std::make_unique<CPluginManager>();
|
g_pPluginManager = std::make_unique<CPluginManager>();
|
||||||
g_pPluginManager->m_bVerbose = verbose;
|
g_pPluginManager->m_bVerbose = verbose;
|
||||||
g_pPluginManager->m_bNoShallow = noShallow;
|
g_pPluginManager->m_bNoShallow = noShallow;
|
||||||
g_pPluginManager->m_szCustomHlUrl = customHlUrl;
|
|
||||||
|
|
||||||
if (command[0] == "add") {
|
if (command[0] == "add") {
|
||||||
if (command.size() < 2) {
|
if (command.size() < 2) {
|
||||||
std::println(stderr, "{}", failureString("Not enough args for add."));
|
std::cerr << Colors::RED << "✖" << Colors::RESET << " Not enough args for add.\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string rev = "";
|
std::string rev = "";
|
||||||
if (command.size() >= 3)
|
if (command.size() >= 3) {
|
||||||
rev = command[2];
|
rev = command[2];
|
||||||
|
|
||||||
const auto HLVER = g_pPluginManager->getHyprlandVersion();
|
|
||||||
auto GLOBALSTATE = DataState::getGlobalState();
|
|
||||||
|
|
||||||
if (GLOBALSTATE.headersHashCompiled != HLVER.hash) {
|
|
||||||
std::println(stderr, "{}", failureString("Headers outdated, please run hyprpm update."));
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
NSys::root::cacheSudo();
|
|
||||||
CScopeGuard x([] { NSys::root::dropSudo(); });
|
|
||||||
|
|
||||||
g_pPluginManager->updateHeaders(false);
|
|
||||||
|
|
||||||
return g_pPluginManager->addNewPluginRepo(command[1], rev) ? 0 : 1;
|
return g_pPluginManager->addNewPluginRepo(command[1], rev) ? 0 : 1;
|
||||||
} else if (command[0] == "remove") {
|
} else if (command[0] == "remove") {
|
||||||
if (command.size() < 2) {
|
if (ARGS.size() < 2) {
|
||||||
std::println(stderr, "{}", failureString("Not enough args for remove."));
|
std::cerr << Colors::RED << "✖" << Colors::RESET << " Not enough args for remove.\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
NSys::root::cacheSudo();
|
|
||||||
CScopeGuard x([] { NSys::root::dropSudo(); });
|
|
||||||
|
|
||||||
return g_pPluginManager->removePluginRepo(command[1]) ? 0 : 1;
|
return g_pPluginManager->removePluginRepo(command[1]) ? 0 : 1;
|
||||||
} else if (command[0] == "update") {
|
} else if (command[0] == "update") {
|
||||||
NSys::root::cacheSudo();
|
bool headersValid = g_pPluginManager->headersValid() == HEADERS_OK;
|
||||||
CScopeGuard x([] { NSys::root::dropSudo(); });
|
bool headers = g_pPluginManager->updateHeaders(force);
|
||||||
|
|
||||||
bool headersValid = g_pPluginManager->headersValid() == HEADERS_OK;
|
|
||||||
bool headers = g_pPluginManager->updateHeaders(force);
|
|
||||||
|
|
||||||
if (headers) {
|
if (headers) {
|
||||||
const auto HLVER = g_pPluginManager->getHyprlandVersion(false);
|
const auto HLVER = g_pPluginManager->getHyprlandVersion();
|
||||||
auto GLOBALSTATE = DataState::getGlobalState();
|
auto GLOBALSTATE = DataState::getGlobalState();
|
||||||
const auto COMPILEDOUTDATED = HLVER.hash != GLOBALSTATE.headersHashCompiled;
|
const auto COMPILEDOUTDATED = HLVER.hash != GLOBALSTATE.headersHashCompiled;
|
||||||
|
|
||||||
@@ -144,81 +110,59 @@ int main(int argc, char** argv, char** envp) {
|
|||||||
|
|
||||||
auto ret2 = g_pPluginManager->ensurePluginsLoadState();
|
auto ret2 = g_pPluginManager->ensurePluginsLoadState();
|
||||||
|
|
||||||
if (ret2 == LOADSTATE_HYPRLAND_UPDATED)
|
|
||||||
g_pPluginManager->notify(ICON_INFO, 0, 10000, "[hyprpm] Updated plugins, but Hyprland was updated. Please restart Hyprland.");
|
|
||||||
|
|
||||||
if (ret2 != LOADSTATE_OK)
|
if (ret2 != LOADSTATE_OK)
|
||||||
return 1;
|
return 1;
|
||||||
} else if (notify)
|
} else if (notify)
|
||||||
g_pPluginManager->notify(ICON_ERROR, 0, 10000, "[hyprpm] Couldn't update headers");
|
g_pPluginManager->notify(ICON_ERROR, 0, 10000, "[hyprpm] Couldn't update headers");
|
||||||
} else if (command[0] == "enable") {
|
} else if (command[0] == "enable") {
|
||||||
if (command.size() < 2) {
|
if (ARGS.size() < 2) {
|
||||||
std::println(stderr, "{}", failureString("Not enough args for enable."));
|
std::cerr << Colors::RED << "✖" << Colors::RESET << " Not enough args for enable.\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!g_pPluginManager->enablePlugin(command[1])) {
|
if (!g_pPluginManager->enablePlugin(command[1])) {
|
||||||
std::println(stderr, "{}", failureString("Couldn't enable plugin (missing?)"));
|
std::cerr << Colors::RED << "✖" << Colors::RESET << " Couldn't enable plugin (missing?)\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
NSys::root::cacheSudo();
|
auto ret = g_pPluginManager->ensurePluginsLoadState();
|
||||||
CScopeGuard x([] { NSys::root::dropSudo(); });
|
|
||||||
|
|
||||||
auto ret = g_pPluginManager->ensurePluginsLoadState();
|
|
||||||
|
|
||||||
if (ret == LOADSTATE_HYPRLAND_UPDATED)
|
|
||||||
g_pPluginManager->notify(ICON_INFO, 0, 10000, "[hyprpm] Enabled plugin, but Hyprland was updated. Please restart Hyprland.");
|
|
||||||
|
|
||||||
if (ret != LOADSTATE_OK)
|
if (ret != LOADSTATE_OK)
|
||||||
return 1;
|
return 1;
|
||||||
} else if (command[0] == "disable") {
|
} else if (command[0] == "disable") {
|
||||||
if (command.size() < 2) {
|
if (command.size() < 2) {
|
||||||
std::println(stderr, "{}", failureString("Not enough args for disable."));
|
std::cerr << Colors::RED << "✖" << Colors::RESET << " Not enough args for disable.\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!g_pPluginManager->disablePlugin(command[1])) {
|
if (!g_pPluginManager->disablePlugin(command[1])) {
|
||||||
std::println(stderr, "{}", failureString("Couldn't disable plugin (missing?)"));
|
std::cerr << Colors::RED << "✖" << Colors::RESET << " Couldn't disable plugin (missing?)\n";
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
NSys::root::cacheSudo();
|
auto ret = g_pPluginManager->ensurePluginsLoadState();
|
||||||
CScopeGuard x([] { NSys::root::dropSudo(); });
|
|
||||||
|
|
||||||
auto ret = g_pPluginManager->ensurePluginsLoadState();
|
|
||||||
|
|
||||||
if (ret != LOADSTATE_OK)
|
if (ret != LOADSTATE_OK)
|
||||||
return 1;
|
return 1;
|
||||||
} else if (command[0] == "reload") {
|
} else if (command[0] == "reload") {
|
||||||
auto ret = g_pPluginManager->ensurePluginsLoadState(force);
|
auto ret = g_pPluginManager->ensurePluginsLoadState();
|
||||||
|
|
||||||
if (ret != LOADSTATE_OK) {
|
if (ret != LOADSTATE_OK && notify) {
|
||||||
if (notify) {
|
switch (ret) {
|
||||||
switch (ret) {
|
case LOADSTATE_FAIL:
|
||||||
case LOADSTATE_FAIL:
|
case LOADSTATE_PARTIAL_FAIL: g_pPluginManager->notify(ICON_ERROR, 0, 10000, "[hyprpm] Failed to load plugins"); break;
|
||||||
case LOADSTATE_PARTIAL_FAIL: g_pPluginManager->notify(ICON_ERROR, 0, 10000, "[hyprpm] Failed to load plugins"); break;
|
case LOADSTATE_HEADERS_OUTDATED:
|
||||||
case LOADSTATE_HEADERS_OUTDATED:
|
g_pPluginManager->notify(ICON_ERROR, 0, 10000, "[hyprpm] Failed to load plugins: Outdated headers. Please run hyprpm update manually.");
|
||||||
g_pPluginManager->notify(ICON_ERROR, 0, 10000, "[hyprpm] Failed to load plugins: Outdated headers. Please run hyprpm update manually.");
|
break;
|
||||||
break;
|
default: break;
|
||||||
default: break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
} else if (notify) {
|
||||||
return 1;
|
|
||||||
} else if (notify && !notifyFail) {
|
|
||||||
g_pPluginManager->notify(ICON_OK, 0, 4000, "[hyprpm] Loaded plugins");
|
g_pPluginManager->notify(ICON_OK, 0, 4000, "[hyprpm] Loaded plugins");
|
||||||
}
|
}
|
||||||
} else if (command[0] == "purge-cache") {
|
|
||||||
NSys::root::cacheSudo();
|
|
||||||
CScopeGuard x([] { NSys::root::dropSudo(); });
|
|
||||||
DataState::purgeAllCache();
|
|
||||||
} else if (command[0] == "list") {
|
} else if (command[0] == "list") {
|
||||||
g_pPluginManager->listAllPlugins();
|
g_pPluginManager->listAllPlugins();
|
||||||
} else {
|
} else {
|
||||||
std::println(stderr, "{}", HELP);
|
std::cout << HELP;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
@@ -1,32 +1,15 @@
|
|||||||
globber = run_command('sh', '-c', 'find . -name "*.cpp" | sort', check: true)
|
globber = run_command('sh', '-c', 'find . -name "*.cpp" | sort', check: true)
|
||||||
src = globber.stdout().strip().split('\n')
|
src = globber.stdout().strip().split('\n')
|
||||||
|
|
||||||
executable(
|
executable('hyprpm', src,
|
||||||
'hyprpm',
|
|
||||||
src,
|
|
||||||
dependencies: [
|
dependencies: [
|
||||||
dependency('hyprutils', version: '>= 0.1.1'),
|
dependency('hyprutils', version: '>= 0.1.1'),
|
||||||
dependency('threads'),
|
dependency('threads'),
|
||||||
dependency('tomlplusplus'),
|
dependency('tomlplusplus')
|
||||||
dependency('glaze', method: 'cmake'),
|
|
||||||
],
|
],
|
||||||
install: true,
|
install : true
|
||||||
)
|
)
|
||||||
|
|
||||||
install_data(
|
install_data('../hyprpm.bash', install_dir: join_paths(get_option('datadir'), 'bash-completion/completions'), install_tag: 'runtime', rename: 'hyprpm')
|
||||||
'../hyprpm.bash',
|
install_data('../hyprpm.fish', install_dir: join_paths(get_option('datadir'), 'fish/vendor_completions.d'), install_tag: 'runtime')
|
||||||
install_dir: join_paths(get_option('datadir'), 'bash-completion/completions'),
|
install_data('../hyprpm.zsh', install_dir: join_paths(get_option('datadir'), 'zsh/site-functions'), install_tag: 'runtime', rename: '_hyprpm')
|
||||||
install_tag: 'runtime',
|
|
||||||
rename: 'hyprpm',
|
|
||||||
)
|
|
||||||
install_data(
|
|
||||||
'../hyprpm.fish',
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'fish/vendor_completions.d'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
)
|
|
||||||
install_data(
|
|
||||||
'../hyprpm.zsh',
|
|
||||||
install_dir: join_paths(get_option('datadir'), 'zsh/site-functions'),
|
|
||||||
install_tag: 'runtime',
|
|
||||||
rename: '_hyprpm',
|
|
||||||
)
|
|
||||||
|
@@ -1,78 +1,80 @@
|
|||||||
#include "CProgressBar.hpp"
|
#include "CProgressBar.hpp"
|
||||||
|
|
||||||
#include <sys/ioctl.h>
|
#include <iostream>
|
||||||
#include <unistd.h>
|
#include <algorithm>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <format>
|
#include <format>
|
||||||
#include <print>
|
|
||||||
#include <cstdio>
|
|
||||||
|
|
||||||
#include <algorithm>
|
#include <sys/ioctl.h>
|
||||||
#include <sstream>
|
#include <stdio.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
#include "../helpers/Colors.hpp"
|
#include "../helpers/Colors.hpp"
|
||||||
|
|
||||||
static winsize getTerminalSize() {
|
|
||||||
winsize w{};
|
|
||||||
ioctl(STDOUT_FILENO, TIOCGWINSZ, &w);
|
|
||||||
return w;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void clearCurrentLine() {
|
|
||||||
std::print("\r\33[2K"); // ansi escape sequence to clear entire line
|
|
||||||
}
|
|
||||||
|
|
||||||
void CProgressBar::printMessageAbove(const std::string& msg) {
|
void CProgressBar::printMessageAbove(const std::string& msg) {
|
||||||
clearCurrentLine();
|
struct winsize w;
|
||||||
std::print("\r{}\n", msg);
|
ioctl(STDOUT_FILENO, TIOCGWINSZ, &w);
|
||||||
|
|
||||||
print(); // reprint bar underneath
|
std::string spaces;
|
||||||
|
for (size_t i = 0; i < w.ws_col; ++i) {
|
||||||
|
spaces += ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "\r" << spaces << "\r" << msg << "\n";
|
||||||
|
print();
|
||||||
}
|
}
|
||||||
|
|
||||||
void CProgressBar::print() {
|
void CProgressBar::print() {
|
||||||
const auto w = getTerminalSize();
|
struct winsize w;
|
||||||
|
ioctl(STDOUT_FILENO, TIOCGWINSZ, &w);
|
||||||
|
|
||||||
if (m_bFirstPrint) {
|
if (m_bFirstPrint)
|
||||||
std::print("\n");
|
std::cout << "\n";
|
||||||
m_bFirstPrint = false;
|
m_bFirstPrint = false;
|
||||||
|
|
||||||
|
std::string spaces;
|
||||||
|
for (size_t i = 0; i < w.ws_col; ++i) {
|
||||||
|
spaces += ' ';
|
||||||
}
|
}
|
||||||
|
|
||||||
clearCurrentLine();
|
std::cout << "\r" << spaces << "\r";
|
||||||
|
|
||||||
float percentDone = 0.0f;
|
std::string message = "";
|
||||||
if (m_fPercentage >= 0.0f)
|
|
||||||
|
float percentDone = 0;
|
||||||
|
if (m_fPercentage >= 0)
|
||||||
percentDone = m_fPercentage;
|
percentDone = m_fPercentage;
|
||||||
else {
|
else
|
||||||
// check for divide-by-zero
|
percentDone = (float)m_iSteps / (float)m_iMaxSteps;
|
||||||
percentDone = m_iMaxSteps > 0 ? static_cast<float>(m_iSteps) / m_iMaxSteps : 0.0f;
|
|
||||||
|
const auto BARWIDTH = std::clamp(w.ws_col - static_cast<unsigned long>(m_szCurrentMessage.length()) - 2, 0UL, 50UL);
|
||||||
|
|
||||||
|
// draw bar
|
||||||
|
message += std::string{" "} + Colors::GREEN;
|
||||||
|
size_t i = 0;
|
||||||
|
for (; i < std::floor(percentDone * BARWIDTH); ++i) {
|
||||||
|
message += "━";
|
||||||
}
|
}
|
||||||
// clamp to ensure no overflows (sanity check)
|
|
||||||
percentDone = std::clamp(percentDone, 0.0f, 1.0f);
|
|
||||||
|
|
||||||
const size_t BARWIDTH = std::clamp<size_t>(w.ws_col - m_szCurrentMessage.length() - 2, 0, 50);
|
|
||||||
|
|
||||||
std::ostringstream oss;
|
|
||||||
oss << ' ' << Colors::GREEN;
|
|
||||||
|
|
||||||
size_t filled = static_cast<size_t>(std::floor(percentDone * BARWIDTH));
|
|
||||||
size_t i = 0;
|
|
||||||
|
|
||||||
for (; i < filled; ++i)
|
|
||||||
oss << "━";
|
|
||||||
|
|
||||||
if (i < BARWIDTH) {
|
if (i < BARWIDTH) {
|
||||||
oss << "╍" << Colors::RESET;
|
i++;
|
||||||
++i;
|
|
||||||
for (; i < BARWIDTH; ++i)
|
message += std::string{"╍"} + Colors::RESET;
|
||||||
oss << "━";
|
|
||||||
|
for (; i < BARWIDTH; ++i) {
|
||||||
|
message += "━";
|
||||||
|
}
|
||||||
} else
|
} else
|
||||||
oss << Colors::RESET;
|
message += Colors::RESET;
|
||||||
|
|
||||||
if (m_fPercentage >= 0.0f)
|
// draw progress
|
||||||
oss << " " << std::format("{}%", static_cast<int>(percentDone * 100.0)) << ' ';
|
if (m_fPercentage >= 0)
|
||||||
|
message += " " + std::format("{}%", static_cast<int>(percentDone * 100.0)) + " ";
|
||||||
else
|
else
|
||||||
oss << " " << std::format("{} / {}", m_iSteps, m_iMaxSteps) << ' ';
|
message += " " + std::format("{} / {}", m_iSteps, m_iMaxSteps) + " ";
|
||||||
|
|
||||||
|
// draw message
|
||||||
|
std::cout << message + " " + m_szCurrentMessage;
|
||||||
|
|
||||||
std::print("{} {}", oss.str(), m_szCurrentMessage);
|
|
||||||
std::fflush(stdout);
|
std::fflush(stdout);
|
||||||
}
|
}
|
@@ -14,4 +14,4 @@ class CProgressBar {
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
bool m_bFirstPrint = true;
|
bool m_bFirstPrint = true;
|
||||||
};
|
};
|
@@ -1,30 +0,0 @@
|
|||||||
cmake_minimum_required(VERSION 3.19)
|
|
||||||
|
|
||||||
project(hyprtester DESCRIPTION "Hyprland test suite")
|
|
||||||
|
|
||||||
include(GNUInstallDirs)
|
|
||||||
|
|
||||||
set(CMAKE_CXX_STANDARD 26)
|
|
||||||
|
|
||||||
find_package(PkgConfig REQUIRED)
|
|
||||||
|
|
||||||
pkg_check_modules(hyprtester_deps REQUIRED IMPORTED_TARGET hyprutils>=0.5.0)
|
|
||||||
|
|
||||||
file(GLOB_RECURSE SRCFILES CONFIGURE_DEPENDS "src/*.cpp")
|
|
||||||
|
|
||||||
add_executable(hyprtester ${SRCFILES})
|
|
||||||
add_custom_command(
|
|
||||||
TARGET hyprtester
|
|
||||||
POST_BUILD
|
|
||||||
COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/plugin/build.sh
|
|
||||||
WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/plugin)
|
|
||||||
|
|
||||||
target_link_libraries(hyprtester PUBLIC PkgConfig::hyprtester_deps)
|
|
||||||
|
|
||||||
install(TARGETS hyprtester)
|
|
||||||
|
|
||||||
install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/test.conf
|
|
||||||
DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/hypr)
|
|
||||||
|
|
||||||
install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/plugin/hyprtestplugin.so
|
|
||||||
DESTINATION ${CMAKE_INSTALL_PREFIX}/lib)
|
|
@@ -1,16 +0,0 @@
|
|||||||
CXXFLAGS = -shared -fPIC --no-gnu-unique -g -std=c++2b -Wno-c++11-narrowing
|
|
||||||
INCLUDES = `pkg-config --cflags pixman-1 libdrm pangocairo libinput libudev wayland-server xkbcommon`
|
|
||||||
LIBS = `pkg-config --libs pangocairo`
|
|
||||||
|
|
||||||
SRC = src/main.cpp
|
|
||||||
TARGET = hyprtestplugin.so
|
|
||||||
|
|
||||||
all: $(TARGET)
|
|
||||||
|
|
||||||
$(TARGET): $(SRC)
|
|
||||||
$(CXX) $(CXXFLAGS) -I../.. -I../../protocols $(INCLUDES) $^ $> -o $@ $(LIBS) -O2
|
|
||||||
|
|
||||||
clean:
|
|
||||||
rm -f ./$(TARGET)
|
|
||||||
|
|
||||||
.PHONY: all clean
|
|
@@ -1,4 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
make clean
|
|
||||||
make all
|
|
@@ -1,5 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <src/plugins/PluginAPI.hpp>
|
|
||||||
|
|
||||||
inline HANDLE PHANDLE = nullptr;
|
|
@@ -1,62 +0,0 @@
|
|||||||
#include <unistd.h>
|
|
||||||
#include <src/includes.hpp>
|
|
||||||
#include <sstream>
|
|
||||||
#include <any>
|
|
||||||
|
|
||||||
#define private public
|
|
||||||
#include <src/config/ConfigManager.hpp>
|
|
||||||
#include <src/config/ConfigDescriptions.hpp>
|
|
||||||
#include <src/layout/IHyprLayout.hpp>
|
|
||||||
#include <src/managers/LayoutManager.hpp>
|
|
||||||
#include <src/Compositor.hpp>
|
|
||||||
#undef private
|
|
||||||
|
|
||||||
#include "globals.hpp"
|
|
||||||
|
|
||||||
// Do NOT change this function.
|
|
||||||
APICALL EXPORT std::string PLUGIN_API_VERSION() {
|
|
||||||
return HYPRLAND_API_VERSION;
|
|
||||||
}
|
|
||||||
|
|
||||||
static SDispatchResult test(std::string in) {
|
|
||||||
bool success = true;
|
|
||||||
std::string errors = "";
|
|
||||||
|
|
||||||
if (g_pConfigManager->m_configValueNumber != CONFIG_OPTIONS.size() + 1 /* autogenerated is special */) {
|
|
||||||
errors += "config value number mismatches descriptions size\n";
|
|
||||||
success = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return SDispatchResult{
|
|
||||||
.success = success,
|
|
||||||
.error = errors,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Trigger a snap move event for the active window
|
|
||||||
static SDispatchResult snapMove(std::string in) {
|
|
||||||
const auto PLASTWINDOW = g_pCompositor->m_lastWindow.lock();
|
|
||||||
if (!PLASTWINDOW->m_isFloating)
|
|
||||||
return {.success = false, .error = "Window must be floating"};
|
|
||||||
|
|
||||||
Vector2D pos = PLASTWINDOW->m_realPosition->goal();
|
|
||||||
Vector2D size = PLASTWINDOW->m_realSize->goal();
|
|
||||||
|
|
||||||
g_pLayoutManager->getCurrentLayout()->performSnap(pos, size, PLASTWINDOW, MBIND_MOVE, -1, size);
|
|
||||||
*PLASTWINDOW->m_realPosition = pos.round();
|
|
||||||
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
APICALL EXPORT PLUGIN_DESCRIPTION_INFO PLUGIN_INIT(HANDLE handle) {
|
|
||||||
PHANDLE = handle;
|
|
||||||
|
|
||||||
HyprlandAPI::addDispatcherV2(PHANDLE, "plugin:test:test", ::test);
|
|
||||||
HyprlandAPI::addDispatcherV2(PHANDLE, "plugin:test:snapmove", ::snapMove);
|
|
||||||
|
|
||||||
return {"hyprtestplugin", "hyprtestplugin", "Vaxry", "1.0"};
|
|
||||||
}
|
|
||||||
|
|
||||||
APICALL EXPORT void PLUGIN_EXIT() {
|
|
||||||
;
|
|
||||||
}
|
|
@@ -1,17 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
#include <string>
|
|
||||||
#include <format>
|
|
||||||
#include <print>
|
|
||||||
|
|
||||||
namespace NLog {
|
|
||||||
template <typename... Args>
|
|
||||||
//NOLINTNEXTLINE
|
|
||||||
void log(std::format_string<Args...> fmt, Args&&... args) {
|
|
||||||
std::string logMsg = "";
|
|
||||||
|
|
||||||
logMsg += std::vformat(fmt.get(), std::make_format_args(args...));
|
|
||||||
|
|
||||||
std::println("{}", logMsg);
|
|
||||||
std::fflush(stdout);
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,136 +0,0 @@
|
|||||||
#include "hyprctlCompat.hpp"
|
|
||||||
#include "shared.hpp"
|
|
||||||
|
|
||||||
#include <pwd.h>
|
|
||||||
#include <sys/socket.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/un.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
|
|
||||||
#include <filesystem>
|
|
||||||
#include <fstream>
|
|
||||||
#include <algorithm>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include <print>
|
|
||||||
|
|
||||||
static int getUID() {
|
|
||||||
const auto UID = getuid();
|
|
||||||
const auto PWUID = getpwuid(UID);
|
|
||||||
return PWUID ? PWUID->pw_uid : UID;
|
|
||||||
}
|
|
||||||
|
|
||||||
static std::string getRuntimeDir() {
|
|
||||||
const auto XDG = getenv("XDG_RUNTIME_DIR");
|
|
||||||
|
|
||||||
if (!XDG) {
|
|
||||||
const std::string USERID = std::to_string(getUID());
|
|
||||||
return "/run/user/" + USERID + "/hypr";
|
|
||||||
}
|
|
||||||
|
|
||||||
return std::string{XDG} + "/hypr";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<SInstanceData> instances() {
|
|
||||||
std::vector<SInstanceData> result;
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (!std::filesystem::exists(getRuntimeDir()))
|
|
||||||
return {};
|
|
||||||
} catch (std::exception& e) { return {}; }
|
|
||||||
|
|
||||||
for (const auto& el : std::filesystem::directory_iterator(getRuntimeDir())) {
|
|
||||||
if (!el.is_directory() || !std::filesystem::exists(el.path().string() + "/hyprland.lock"))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
// read lock
|
|
||||||
SInstanceData* data = &result.emplace_back();
|
|
||||||
data->id = el.path().filename().string();
|
|
||||||
|
|
||||||
try {
|
|
||||||
data->time = std::stoull(data->id.substr(data->id.find_first_of('_') + 1, data->id.find_last_of('_') - (data->id.find_first_of('_') + 1)));
|
|
||||||
} catch (std::exception& e) { continue; }
|
|
||||||
|
|
||||||
// read file
|
|
||||||
std::ifstream ifs(el.path().string() + "/hyprland.lock");
|
|
||||||
|
|
||||||
int i = 0;
|
|
||||||
for (std::string line; std::getline(ifs, line); ++i) {
|
|
||||||
if (i == 0) {
|
|
||||||
try {
|
|
||||||
data->pid = std::stoull(line);
|
|
||||||
} catch (std::exception& e) { continue; }
|
|
||||||
} else if (i == 1) {
|
|
||||||
data->wlSocket = line;
|
|
||||||
} else
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
ifs.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::erase_if(result, [&](const auto& el) { return kill(el.pid, 0) != 0 && errno == ESRCH; });
|
|
||||||
|
|
||||||
std::sort(result.begin(), result.end(), [&](const auto& a, const auto& b) { return a.time < b.time; });
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getFromSocket(const std::string& cmd) {
|
|
||||||
const auto SERVERSOCKET = socket(AF_UNIX, SOCK_STREAM, 0);
|
|
||||||
|
|
||||||
auto t = timeval{.tv_sec = 5, .tv_usec = 0};
|
|
||||||
setsockopt(SERVERSOCKET, SOL_SOCKET, SO_RCVTIMEO, &t, sizeof(struct timeval));
|
|
||||||
|
|
||||||
if (SERVERSOCKET < 0) {
|
|
||||||
std::println("socket: Couldn't open a socket (1)");
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
sockaddr_un serverAddress = {0};
|
|
||||||
serverAddress.sun_family = AF_UNIX;
|
|
||||||
|
|
||||||
std::string socketPath = getRuntimeDir() + "/" + HIS + "/.socket.sock";
|
|
||||||
|
|
||||||
strncpy(serverAddress.sun_path, socketPath.c_str(), sizeof(serverAddress.sun_path) - 1);
|
|
||||||
|
|
||||||
if (connect(SERVERSOCKET, (sockaddr*)&serverAddress, SUN_LEN(&serverAddress)) < 0) {
|
|
||||||
std::println("Couldn't connect to {}. (3)", socketPath);
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
auto sizeWritten = write(SERVERSOCKET, cmd.c_str(), cmd.length());
|
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
|
||||||
std::println("Couldn't write (4)");
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string reply = "";
|
|
||||||
char buffer[8192] = {0};
|
|
||||||
|
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, 8192);
|
|
||||||
|
|
||||||
if (sizeWritten < 0) {
|
|
||||||
if (errno == EWOULDBLOCK)
|
|
||||||
std::println("Hyprland IPC didn't respond in time");
|
|
||||||
std::println("Couldn't read (5)");
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
reply += std::string(buffer, sizeWritten);
|
|
||||||
|
|
||||||
while (sizeWritten == 8192) {
|
|
||||||
sizeWritten = read(SERVERSOCKET, buffer, 8192);
|
|
||||||
if (sizeWritten < 0) {
|
|
||||||
std::println("Couldn't read (5)");
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
reply += std::string(buffer, sizeWritten);
|
|
||||||
}
|
|
||||||
|
|
||||||
close(SERVERSOCKET);
|
|
||||||
|
|
||||||
return reply;
|
|
||||||
}
|
|
@@ -1,16 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <vector>
|
|
||||||
#include <string>
|
|
||||||
#include <cstdint>
|
|
||||||
|
|
||||||
struct SInstanceData {
|
|
||||||
std::string id;
|
|
||||||
uint64_t time;
|
|
||||||
uint64_t pid;
|
|
||||||
std::string wlSocket;
|
|
||||||
bool valid = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
std::vector<SInstanceData> instances();
|
|
||||||
std::string getFromSocket(const std::string& cmd);
|
|
@@ -1,240 +0,0 @@
|
|||||||
|
|
||||||
// This is a tester for Hyprland. It will launch the built binary in ./build/Hyprland
|
|
||||||
// in headless mode and test various things.
|
|
||||||
// for now it's quite basic and limited, but will be expanded in the future.
|
|
||||||
|
|
||||||
// NOTE: This tester has to be ran from its directory!!
|
|
||||||
|
|
||||||
// Some TODO:
|
|
||||||
// - Add a plugin built alongside so that we can do more detailed tests (e.g. simulating keystrokes)
|
|
||||||
// - test coverage
|
|
||||||
// - maybe figure out a way to do some visual tests too?
|
|
||||||
|
|
||||||
// Required runtime deps for checks:
|
|
||||||
// - kitty
|
|
||||||
// - xeyes
|
|
||||||
|
|
||||||
#include "shared.hpp"
|
|
||||||
#include "hyprctlCompat.hpp"
|
|
||||||
#include "tests/main/tests.hpp"
|
|
||||||
#include "tests/plugin/plugin.hpp"
|
|
||||||
|
|
||||||
#include <filesystem>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include <chrono>
|
|
||||||
#include <thread>
|
|
||||||
#include <print>
|
|
||||||
|
|
||||||
#include "Log.hpp"
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::Memory;
|
|
||||||
|
|
||||||
#define SP CSharedPointer
|
|
||||||
|
|
||||||
static int ret = 0;
|
|
||||||
static SP<CProcess> hyprlandProc;
|
|
||||||
static const std::string cwd = std::filesystem::current_path().string();
|
|
||||||
|
|
||||||
//
|
|
||||||
static bool launchHyprland(std::string configPath, std::string binaryPath) {
|
|
||||||
if (binaryPath == "") {
|
|
||||||
std::error_code ec;
|
|
||||||
if (!std::filesystem::exists(cwd + "/../build/Hyprland", ec) || ec) {
|
|
||||||
NLog::log("{}No Hyprland binary", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
binaryPath = cwd + "/../build/Hyprland";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (configPath == "") {
|
|
||||||
std::error_code ec;
|
|
||||||
if (!std::filesystem::exists(cwd + "/test.conf", ec) || ec) {
|
|
||||||
NLog::log("{}No test config", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
configPath = cwd + "/test.conf";
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Launching Hyprland", Colors::YELLOW);
|
|
||||||
hyprlandProc = makeShared<CProcess>(binaryPath, std::vector<std::string>{"--config", configPath});
|
|
||||||
hyprlandProc->addEnv("HYPRLAND_HEADLESS_ONLY", "1");
|
|
||||||
|
|
||||||
NLog::log("{}Launched async process", Colors::YELLOW);
|
|
||||||
|
|
||||||
return hyprlandProc->runAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool hyprlandAlive() {
|
|
||||||
NLog::log("{}hyprlandAlive", Colors::YELLOW);
|
|
||||||
kill(hyprlandProc->pid(), 0);
|
|
||||||
return errno != ESRCH;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void help() {
|
|
||||||
NLog::log("usage: hyprtester [arg [...]].\n");
|
|
||||||
NLog::log(R"(Arguments:
|
|
||||||
--help -h - Show this message again
|
|
||||||
--config FILE -c FILE - Specify config file to use
|
|
||||||
--binary FILE -b FILE - Specify Hyprland binary to use
|
|
||||||
--plugin FILE -p FILE - Specify the location of the test plugin)");
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int argc, char** argv, char** envp) {
|
|
||||||
|
|
||||||
std::string configPath = "";
|
|
||||||
std::string binaryPath = "";
|
|
||||||
std::string pluginPath = std::filesystem::current_path().string();
|
|
||||||
|
|
||||||
std::vector<std::string> args{argv + 1, argv + argc};
|
|
||||||
|
|
||||||
for (auto it = args.begin(); it != args.end(); it++) {
|
|
||||||
if (*it == "--config" || *it == "-c") {
|
|
||||||
if (std::next(it) == args.end()) {
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
configPath = *std::next(it);
|
|
||||||
|
|
||||||
try {
|
|
||||||
configPath = std::filesystem::canonical(configPath);
|
|
||||||
|
|
||||||
if (!std::filesystem::is_regular_file(configPath)) {
|
|
||||||
throw std::exception();
|
|
||||||
}
|
|
||||||
} catch (...) {
|
|
||||||
std::println(stderr, "[ ERROR ] Config file '{}' doesn't exist!", configPath);
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
it++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
} else if (*it == "--binary" || *it == "-b") {
|
|
||||||
if (std::next(it) == args.end()) {
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
binaryPath = *std::next(it);
|
|
||||||
|
|
||||||
try {
|
|
||||||
binaryPath = std::filesystem::canonical(binaryPath);
|
|
||||||
|
|
||||||
if (!std::filesystem::is_regular_file(binaryPath)) {
|
|
||||||
throw std::exception();
|
|
||||||
}
|
|
||||||
} catch (...) {
|
|
||||||
std::println(stderr, "[ ERROR ] Binary '{}' doesn't exist!", binaryPath);
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
it++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
} else if (*it == "--plugin" || *it == "-p") {
|
|
||||||
if (std::next(it) == args.end()) {
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
pluginPath = *std::next(it);
|
|
||||||
|
|
||||||
try {
|
|
||||||
pluginPath = std::filesystem::canonical(pluginPath);
|
|
||||||
|
|
||||||
if (!std::filesystem::is_regular_file(pluginPath)) {
|
|
||||||
throw std::exception();
|
|
||||||
}
|
|
||||||
} catch (...) {
|
|
||||||
std::println(stderr, "[ ERROR ] plugin '{}' doesn't exist!", pluginPath);
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
it++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
} else if (*it == "--help" || *it == "-h") {
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
} else {
|
|
||||||
std::println(stderr, "[ ERROR ] Unknown option '{}' !", *it);
|
|
||||||
help();
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}launching hl", Colors::YELLOW);
|
|
||||||
if (!launchHyprland(configPath, binaryPath)) {
|
|
||||||
NLog::log("{}well it failed", Colors::RED);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// hyprland has launched, let's check if it's alive after 10s
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(10000));
|
|
||||||
NLog::log("{}slept for 10s", Colors::YELLOW);
|
|
||||||
if (!hyprlandAlive()) {
|
|
||||||
NLog::log("{}Hyprland failed to launch", Colors::RED);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// wonderful, we are in. Let's get the instance signature.
|
|
||||||
NLog::log("{}trying to get INSTANCES", Colors::YELLOW);
|
|
||||||
const auto INSTANCES = instances();
|
|
||||||
if (INSTANCES.empty()) {
|
|
||||||
NLog::log("{}Hyprland failed to launch (2)", Colors::RED);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
HIS = INSTANCES.back().id;
|
|
||||||
WLDISPLAY = INSTANCES.back().wlSocket;
|
|
||||||
|
|
||||||
NLog::log("{}trying to get create headless output", Colors::YELLOW);
|
|
||||||
getFromSocket("/output create headless");
|
|
||||||
|
|
||||||
NLog::log("{}trying to load plugin", Colors::YELLOW);
|
|
||||||
if (const auto R = getFromSocket(std::format("/plugin load {}", pluginPath)); R != "ok") {
|
|
||||||
NLog::log("{}Failed to load the test plugin: {}", Colors::RED, R);
|
|
||||||
getFromSocket("/dispatch exit 1");
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Loaded plugin", Colors::YELLOW);
|
|
||||||
|
|
||||||
for (const auto& fn : testFns) {
|
|
||||||
EXPECT(fn(), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}running plugin test", Colors::YELLOW);
|
|
||||||
EXPECT(testPlugin(), true);
|
|
||||||
|
|
||||||
// kill hyprland
|
|
||||||
NLog::log("{}dispatching exit", Colors::YELLOW);
|
|
||||||
getFromSocket("/dispatch exit");
|
|
||||||
|
|
||||||
NLog::log("\n{}Summary:\n\tPASSED: {}{}{}/{}\n\tFAILED: {}{}{}/{}\n{}", Colors::RESET, Colors::GREEN, TESTS_PASSED, Colors::RESET, TESTS_PASSED + TESTS_FAILED, Colors::RED,
|
|
||||||
TESTS_FAILED, Colors::RESET, TESTS_PASSED + TESTS_FAILED, (TESTS_FAILED > 0 ? std::string{Colors::RED} + "\nSome tests failed.\n" : ""));
|
|
||||||
|
|
||||||
kill(hyprlandProc->pid(), SIGKILL);
|
|
||||||
|
|
||||||
hyprlandProc.reset();
|
|
||||||
|
|
||||||
return ret || TESTS_FAILED;
|
|
||||||
}
|
|
@@ -1,90 +0,0 @@
|
|||||||
// Stolen from hyprutils
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
inline std::string HIS = "";
|
|
||||||
inline std::string WLDISPLAY = "";
|
|
||||||
inline int TESTS_PASSED = 0;
|
|
||||||
inline int TESTS_FAILED = 0;
|
|
||||||
|
|
||||||
namespace Colors {
|
|
||||||
constexpr const char* RED = "\x1b[31m";
|
|
||||||
constexpr const char* GREEN = "\x1b[32m";
|
|
||||||
constexpr const char* YELLOW = "\x1b[33m";
|
|
||||||
constexpr const char* BLUE = "\x1b[34m";
|
|
||||||
constexpr const char* MAGENTA = "\x1b[35m";
|
|
||||||
constexpr const char* CYAN = "\x1b[36m";
|
|
||||||
constexpr const char* RESET = "\x1b[0m";
|
|
||||||
};
|
|
||||||
|
|
||||||
#define EXPECT(expr, val) \
|
|
||||||
if (const auto RESULT = expr; RESULT != (val)) { \
|
|
||||||
NLog::log("{}Failed: {}{}, expected {}, got {}. Source: {}@{}.", Colors::RED, Colors::RESET, #expr, val, RESULT, __FILE__, __LINE__); \
|
|
||||||
ret = 1; \
|
|
||||||
TESTS_FAILED++; \
|
|
||||||
} else { \
|
|
||||||
NLog::log("{}Passed: {}{}. Got {}", Colors::GREEN, Colors::RESET, #expr, val); \
|
|
||||||
TESTS_PASSED++; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define EXPECT_VECTOR2D(expr, val) \
|
|
||||||
do { \
|
|
||||||
const auto& RESULT = expr; \
|
|
||||||
const auto& EXPECTED = val; \
|
|
||||||
if (!(std::abs(RESULT.x - EXPECTED.x) < 1e-6 && std::abs(RESULT.y - EXPECTED.y) < 1e-6)) { \
|
|
||||||
NLog::log("{}Failed: {}{}, expected [{}, {}], got [{}, {}]. Source: {}@{}.", Colors::RED, Colors::RESET, #expr, EXPECTED.x, EXPECTED.y, RESULT.x, RESULT.y, __FILE__, \
|
|
||||||
__LINE__); \
|
|
||||||
ret = 1; \
|
|
||||||
TESTS_FAILED++; \
|
|
||||||
} else { \
|
|
||||||
NLog::log("{}Passed: {}{}. Got [{}, {}].", Colors::GREEN, Colors::RESET, #expr, RESULT.x, RESULT.y); \
|
|
||||||
TESTS_PASSED++; \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define EXPECT_CONTAINS(haystack, needle) \
|
|
||||||
if (const auto EXPECTED = needle; !std::string{haystack}.contains(EXPECTED)) { \
|
|
||||||
NLog::log("{}Failed: {}{} should contain {} but doesn't. Source: {}@{}. Haystack is:\n{}", Colors::RED, Colors::RESET, #haystack, EXPECTED, __FILE__, __LINE__, \
|
|
||||||
std::string{haystack}); \
|
|
||||||
ret = 1; \
|
|
||||||
TESTS_FAILED++; \
|
|
||||||
} else { \
|
|
||||||
NLog::log("{}Passed: {}{} contains {}.", Colors::GREEN, Colors::RESET, #haystack, EXPECTED); \
|
|
||||||
TESTS_PASSED++; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define EXPECT_NOT_CONTAINS(haystack, needle) \
|
|
||||||
if (std::string{haystack}.contains(needle)) { \
|
|
||||||
NLog::log("{}Failed: {}{} shouldn't contain {} but does. Source: {}@{}. Haystack is:\n{}", Colors::RED, Colors::RESET, #haystack, #needle, __FILE__, __LINE__, \
|
|
||||||
std::string{haystack}); \
|
|
||||||
ret = 1; \
|
|
||||||
TESTS_FAILED++; \
|
|
||||||
} else { \
|
|
||||||
NLog::log("{}Passed: {}{} doesn't contain {}.", Colors::GREEN, Colors::RESET, #haystack, #needle); \
|
|
||||||
TESTS_PASSED++; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define EXPECT_STARTS_WITH(str, what) \
|
|
||||||
if (!std::string{str}.starts_with(what)) { \
|
|
||||||
NLog::log("{}Failed: {}{} should start with {} but doesn't. Source: {}@{}. String is:\n{}", Colors::RED, Colors::RESET, #str, #what, __FILE__, __LINE__, \
|
|
||||||
std::string{str}); \
|
|
||||||
ret = 1; \
|
|
||||||
TESTS_FAILED++; \
|
|
||||||
} else { \
|
|
||||||
NLog::log("{}Passed: {}{} starts with {}.", Colors::GREEN, Colors::RESET, #str, #what); \
|
|
||||||
TESTS_PASSED++; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define EXPECT_COUNT_STRING(str, what, no) \
|
|
||||||
if (Tests::countOccurrences(str, what) != no) { \
|
|
||||||
NLog::log("{}Failed: {}{} should contain {} {} times, but doesn't. Source: {}@{}. String is:\n{}", Colors::RED, Colors::RESET, #str, #what, no, __FILE__, __LINE__, \
|
|
||||||
std::string{str}); \
|
|
||||||
ret = 1; \
|
|
||||||
TESTS_FAILED++; \
|
|
||||||
} else { \
|
|
||||||
NLog::log("{}Passed: {}{} contains {} {} times.", Colors::GREEN, Colors::RESET, #str, #what, no); \
|
|
||||||
TESTS_PASSED++; \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define OK(x) EXPECT(x, "ok")
|
|
@@ -1,179 +0,0 @@
|
|||||||
#include "tests.hpp"
|
|
||||||
#include "../../shared.hpp"
|
|
||||||
#include "../../hyprctlCompat.hpp"
|
|
||||||
#include <print>
|
|
||||||
#include <thread>
|
|
||||||
#include <chrono>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include "../shared.hpp"
|
|
||||||
|
|
||||||
static int ret = 0;
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::Memory;
|
|
||||||
|
|
||||||
#define UP CUniquePointer
|
|
||||||
#define SP CSharedPointer
|
|
||||||
|
|
||||||
static bool test() {
|
|
||||||
NLog::log("{}Testing groups", Colors::GREEN);
|
|
||||||
|
|
||||||
// test on workspace "window"
|
|
||||||
NLog::log("{}Dispatching workspace `groups`", Colors::YELLOW);
|
|
||||||
getFromSocket("/dispatch workspace name:groups");
|
|
||||||
|
|
||||||
NLog::log("{}Spawning kittyProcA", Colors::YELLOW);
|
|
||||||
auto kittyProcA = Tests::spawnKitty();
|
|
||||||
if (!kittyProcA) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 1 window", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 1);
|
|
||||||
|
|
||||||
// check kitty properties. One kitty should take the entire screen, minus the gaps.
|
|
||||||
NLog::log("{}Check kitty dimensions", Colors::YELLOW);
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
EXPECT_COUNT_STRING(str, "at: 22,22", 1);
|
|
||||||
EXPECT_COUNT_STRING(str, "size: 1876,1036", 1);
|
|
||||||
EXPECT_COUNT_STRING(str, "fullscreen: 0", 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// group the kitty
|
|
||||||
NLog::log("{}Enable group and groupbar", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch togglegroup"));
|
|
||||||
OK(getFromSocket("/keyword group:groupbar:enabled 1"));
|
|
||||||
|
|
||||||
// check the height of the window now
|
|
||||||
NLog::log("{}Recheck kitty dimensions", Colors::YELLOW);
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
EXPECT_CONTAINS(str, "at: 22,43");
|
|
||||||
EXPECT_CONTAINS(str, "size: 1876,1015");
|
|
||||||
}
|
|
||||||
|
|
||||||
// disable the groupbar for ease of testing for now
|
|
||||||
NLog::log("{}Disable groupbar", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("r/keyword group:groupbar:enabled 0"));
|
|
||||||
|
|
||||||
// kill all
|
|
||||||
NLog::log("{}Kill windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Spawn kitty again", Colors::YELLOW);
|
|
||||||
kittyProcA = Tests::spawnKitty();
|
|
||||||
if (!kittyProcA) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Group kitty", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch togglegroup"));
|
|
||||||
|
|
||||||
// check the height of the window now
|
|
||||||
NLog::log("{}Check kitty dimensions 2", Colors::YELLOW);
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
EXPECT_CONTAINS(str, "at: 22,22");
|
|
||||||
EXPECT_CONTAINS(str, "size: 1876,1036");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Spawn kittyProcB", Colors::YELLOW);
|
|
||||||
auto kittyProcB = Tests::spawnKitty();
|
|
||||||
if (!kittyProcB) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 2 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 2);
|
|
||||||
|
|
||||||
size_t lastActiveKittyIdx = 0;
|
|
||||||
|
|
||||||
NLog::log("{}Get last active kitty id", Colors::YELLOW);
|
|
||||||
try {
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
lastActiveKittyIdx = std::stoull(str.substr(7, str.find(" -> ") - 7), nullptr, 16);
|
|
||||||
} catch (...) {
|
|
||||||
NLog::log("{}Fail at getting prop", Colors::RED);
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// test cycling through
|
|
||||||
|
|
||||||
NLog::log("{}Test cycling through grouped windows", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch changegroupactive f"));
|
|
||||||
|
|
||||||
try {
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT(lastActiveKittyIdx != std::stoull(str.substr(7, str.find(" -> ") - 7), nullptr, 16), true);
|
|
||||||
} catch (...) {
|
|
||||||
NLog::log("{}Fail at getting prop", Colors::RED);
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
getFromSocket("/dispatch changegroupactive f");
|
|
||||||
|
|
||||||
try {
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT(lastActiveKittyIdx, std::stoull(str.substr(7, str.find(" -> ") - 7), nullptr, 16));
|
|
||||||
} catch (...) {
|
|
||||||
NLog::log("{}Fail at getting prop", Colors::RED);
|
|
||||||
ret = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Disable autogrouping", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword group:auto_group false"));
|
|
||||||
|
|
||||||
NLog::log("{}Spawn kittyProcC", Colors::YELLOW);
|
|
||||||
auto kittyProcC = Tests::spawnKitty();
|
|
||||||
if (!kittyProcC) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 3 windows 2", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 3);
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
EXPECT_COUNT_STRING(str, "at: 22,22", 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch movefocus l"));
|
|
||||||
OK(getFromSocket("/dispatch changegroupactive 1"));
|
|
||||||
OK(getFromSocket("/keyword group:auto_group true"));
|
|
||||||
OK(getFromSocket("/keyword group:insert_after_current false"));
|
|
||||||
|
|
||||||
NLog::log("{}Spawn kittyProcD", Colors::YELLOW);
|
|
||||||
auto kittyProcD = Tests::spawnKitty();
|
|
||||||
if (!kittyProcD) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 4 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 4);
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch changegroupactive 3"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, std::format("pid: {}", kittyProcD->pid()));
|
|
||||||
}
|
|
||||||
|
|
||||||
// kill all
|
|
||||||
NLog::log("{}Kill windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 0 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 0);
|
|
||||||
|
|
||||||
return !ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
REGISTER_TEST_FN(test)
|
|
@@ -1,144 +0,0 @@
|
|||||||
#include "tests.hpp"
|
|
||||||
#include "../../shared.hpp"
|
|
||||||
#include "../../hyprctlCompat.hpp"
|
|
||||||
#include <print>
|
|
||||||
#include <thread>
|
|
||||||
#include <chrono>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include "../shared.hpp"
|
|
||||||
|
|
||||||
static int ret = 0;
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::Memory;
|
|
||||||
|
|
||||||
#define UP CUniquePointer
|
|
||||||
#define SP CSharedPointer
|
|
||||||
|
|
||||||
static bool test() {
|
|
||||||
NLog::log("{}Testing config: misc:", Colors::GREEN);
|
|
||||||
|
|
||||||
NLog::log("{}Testing close_special_on_empty", Colors::YELLOW);
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:close_special_on_empty false"));
|
|
||||||
OK(getFromSocket("/dispatch workspace special:test"));
|
|
||||||
|
|
||||||
Tests::spawnKitty();
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/monitors");
|
|
||||||
EXPECT_CONTAINS(str, "special workspace: -");
|
|
||||||
}
|
|
||||||
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/monitors");
|
|
||||||
EXPECT_CONTAINS(str, "special workspace: -");
|
|
||||||
}
|
|
||||||
|
|
||||||
Tests::spawnKitty();
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:close_special_on_empty true"));
|
|
||||||
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/monitors");
|
|
||||||
EXPECT_NOT_CONTAINS(str, "special workspace: -");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Testing new_window_takes_over_fullscreen", Colors::YELLOW);
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:new_window_takes_over_fullscreen 0"));
|
|
||||||
|
|
||||||
Tests::spawnKitty("kitty_A");
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch fullscreen 0"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 2");
|
|
||||||
EXPECT_CONTAINS(str, "kitty_A");
|
|
||||||
}
|
|
||||||
|
|
||||||
Tests::spawnKitty("kitty_B");
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 2");
|
|
||||||
EXPECT_CONTAINS(str, "kitty_A");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:new_window_takes_over_fullscreen 1"));
|
|
||||||
|
|
||||||
Tests::spawnKitty("kitty_C");
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 2");
|
|
||||||
EXPECT_CONTAINS(str, "kitty_C");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:new_window_takes_over_fullscreen 2"));
|
|
||||||
|
|
||||||
Tests::spawnKitty("kitty_D");
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 0");
|
|
||||||
EXPECT_CONTAINS(str, "kitty_D");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:new_window_takes_over_fullscreen 0"));
|
|
||||||
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Testing exit_window_retains_fullscreen", Colors::YELLOW);
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword misc:exit_window_retains_fullscreen false"));
|
|
||||||
|
|
||||||
Tests::spawnKitty("kitty_A");
|
|
||||||
Tests::spawnKitty("kitty_B");
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch fullscreen 0"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 2");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch killwindow activewindow"));
|
|
||||||
Tests::waitUntilWindowsN(1);
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 0");
|
|
||||||
}
|
|
||||||
|
|
||||||
Tests::spawnKitty("kitty_B");
|
|
||||||
OK(getFromSocket("/dispatch fullscreen 0"));
|
|
||||||
OK(getFromSocket("/keyword misc:exit_window_retains_fullscreen true"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch killwindow activewindow"));
|
|
||||||
Tests::waitUntilWindowsN(1);
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "fullscreen: 2");
|
|
||||||
}
|
|
||||||
|
|
||||||
// kill all
|
|
||||||
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 0 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 0);
|
|
||||||
|
|
||||||
return !ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
REGISTER_TEST_FN(test);
|
|
@@ -1,175 +0,0 @@
|
|||||||
#include <hyprutils/math/Vector2D.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
|
|
||||||
#include "../../shared.hpp"
|
|
||||||
#include "../../hyprctlCompat.hpp"
|
|
||||||
#include "../shared.hpp"
|
|
||||||
#include "tests.hpp"
|
|
||||||
|
|
||||||
using Hyprutils::Math::Vector2D;
|
|
||||||
|
|
||||||
static int ret = 0;
|
|
||||||
|
|
||||||
static bool spawnFloatingKitty() {
|
|
||||||
if (!Tests::spawnKitty()) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
OK(getFromSocket("/dispatch setfloating active"));
|
|
||||||
OK(getFromSocket("/dispatch resizeactive exact 100 100"));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void expectSocket(const std::string& CMD) {
|
|
||||||
if (const auto RESULT = getFromSocket(CMD); RESULT != "ok") {
|
|
||||||
NLog::log("{}Failed: {}getFromSocket({}), expected ok, got {}. Source: {}@{}.", Colors::RED, Colors::RESET, CMD, RESULT, __FILE__, __LINE__);
|
|
||||||
ret = 1;
|
|
||||||
TESTS_FAILED++;
|
|
||||||
} else {
|
|
||||||
NLog::log("{}Passed: {}getFromSocket({}). Got ok", Colors::GREEN, Colors::RESET, CMD);
|
|
||||||
TESTS_PASSED++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void expectSnapMove(const Vector2D FROM, const Vector2D* TO) {
|
|
||||||
const Vector2D& A = FROM;
|
|
||||||
const Vector2D& B = TO ? *TO : FROM;
|
|
||||||
if (TO)
|
|
||||||
NLog::log("{}Expecting snap to ({},{}) when window is moved to ({},{})", Colors::YELLOW, B.x, B.y, A.x, A.y);
|
|
||||||
else
|
|
||||||
NLog::log("{}Expecting no snap when window is moved to ({},{})", Colors::YELLOW, A.x, A.y);
|
|
||||||
|
|
||||||
expectSocket(std::format("/dispatch moveactive exact {} {}", A.x, A.y));
|
|
||||||
expectSocket("/dispatch plugin:test:snapmove");
|
|
||||||
EXPECT_CONTAINS(getFromSocket("/activewindow"), std::format("at: {},{}", B.x, B.y));
|
|
||||||
}
|
|
||||||
|
|
||||||
static void testWindowSnap(const bool RESPECTGAPS) {
|
|
||||||
const double BORDERSIZE = 2;
|
|
||||||
const double WINDOWSIZE = 100;
|
|
||||||
|
|
||||||
const double OTHER = 500;
|
|
||||||
const double WINDOWGAP = 8;
|
|
||||||
const double GAPSIN = 5;
|
|
||||||
const double GAP = (RESPECTGAPS ? 2 * GAPSIN : 0) + (2 * BORDERSIZE);
|
|
||||||
const double END = GAP + WINDOWSIZE;
|
|
||||||
|
|
||||||
double x;
|
|
||||||
Vector2D predict;
|
|
||||||
|
|
||||||
x = WINDOWGAP + END;
|
|
||||||
expectSnapMove({OTHER + x, OTHER}, nullptr);
|
|
||||||
expectSnapMove({OTHER - x, OTHER}, nullptr);
|
|
||||||
expectSnapMove({OTHER, OTHER + x}, nullptr);
|
|
||||||
expectSnapMove({OTHER, OTHER - x}, nullptr);
|
|
||||||
x -= 1;
|
|
||||||
expectSnapMove({OTHER + x, OTHER}, &(predict = {OTHER + END, OTHER}));
|
|
||||||
expectSnapMove({OTHER - x, OTHER}, &(predict = {OTHER - END, OTHER}));
|
|
||||||
expectSnapMove({OTHER, OTHER + x}, &(predict = {OTHER, OTHER + END}));
|
|
||||||
expectSnapMove({OTHER, OTHER - x}, &(predict = {OTHER, OTHER - END}));
|
|
||||||
}
|
|
||||||
|
|
||||||
static void testMonitorSnap(const bool RESPECTGAPS, const bool OVERLAP) {
|
|
||||||
const double BORDERSIZE = 2;
|
|
||||||
const double WINDOWSIZE = 100;
|
|
||||||
|
|
||||||
const double MONITORGAP = 10;
|
|
||||||
const double GAPSOUT = 20;
|
|
||||||
const double RESP = (RESPECTGAPS ? GAPSOUT : 0);
|
|
||||||
const double GAP = RESP + (OVERLAP ? 0 : BORDERSIZE);
|
|
||||||
const double END = GAP + WINDOWSIZE;
|
|
||||||
|
|
||||||
double x;
|
|
||||||
Vector2D predict;
|
|
||||||
|
|
||||||
x = MONITORGAP + GAP;
|
|
||||||
expectSnapMove({x, x}, nullptr);
|
|
||||||
x -= 1;
|
|
||||||
expectSnapMove({x, x}, &(predict = {GAP, GAP}));
|
|
||||||
|
|
||||||
x = MONITORGAP + END;
|
|
||||||
expectSnapMove({1920 - x, 1080 - x}, nullptr);
|
|
||||||
x -= 1;
|
|
||||||
expectSnapMove({1920 - x, 1080 - x}, &(predict = {1920 - END, 1080 - END}));
|
|
||||||
|
|
||||||
// test reserved area
|
|
||||||
const double RESERVED = 200;
|
|
||||||
const double RGAP = RESERVED + RESP + BORDERSIZE;
|
|
||||||
const double REND = RGAP + WINDOWSIZE;
|
|
||||||
|
|
||||||
x = MONITORGAP + RGAP;
|
|
||||||
expectSnapMove({x, x}, nullptr);
|
|
||||||
x -= 1;
|
|
||||||
expectSnapMove({x, x}, &(predict = {RGAP, RGAP}));
|
|
||||||
|
|
||||||
x = MONITORGAP + REND;
|
|
||||||
expectSnapMove({1920 - x, 1080 - x}, nullptr);
|
|
||||||
x -= 1;
|
|
||||||
expectSnapMove({1920 - x, 1080 - x}, &(predict = {1920 - REND, 1080 - REND}));
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool test() {
|
|
||||||
NLog::log("{}Testing snap", Colors::GREEN);
|
|
||||||
|
|
||||||
// move to monitor HEADLESS-2
|
|
||||||
NLog::log("{}Moving to monitor HEADLESS-2", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch focusmonitor HEADLESS-2"));
|
|
||||||
NLog::log("{}Adding reserved monitor area to HEADLESS-2", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword monitor HEADLESS-2,addreserved,200,200,200,200"));
|
|
||||||
|
|
||||||
// test on workspace "snap"
|
|
||||||
NLog::log("{}Dispatching workspace `snap`", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace name:snap"));
|
|
||||||
|
|
||||||
// spawn a kitty terminal and move to (500,500)
|
|
||||||
NLog::log("{}Spawning kittyProcA", Colors::YELLOW);
|
|
||||||
if (!spawnFloatingKitty())
|
|
||||||
return false;
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 1 window", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 1);
|
|
||||||
|
|
||||||
NLog::log("{}Move the kitty window to (500,500)", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch moveactive exact 500 500"));
|
|
||||||
|
|
||||||
// spawn a second kitty terminal
|
|
||||||
NLog::log("{}Spawning kittyProcB", Colors::YELLOW);
|
|
||||||
if (!spawnFloatingKitty())
|
|
||||||
return false;
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 2 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 2);
|
|
||||||
|
|
||||||
NLog::log("");
|
|
||||||
testWindowSnap(false);
|
|
||||||
testMonitorSnap(false, false);
|
|
||||||
|
|
||||||
NLog::log("\n{}Turning on respect_gaps", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword general:snap:respect_gaps true"));
|
|
||||||
testWindowSnap(true);
|
|
||||||
testMonitorSnap(true, false);
|
|
||||||
|
|
||||||
NLog::log("\n{}Turning on border_overlap", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword general:snap:respect_gaps false"));
|
|
||||||
OK(getFromSocket("/keyword general:snap:border_overlap true"));
|
|
||||||
testMonitorSnap(false, true);
|
|
||||||
|
|
||||||
NLog::log("\n{}Turning on both border_overlap and respect_gaps", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword general:snap:respect_gaps true"));
|
|
||||||
testMonitorSnap(true, true);
|
|
||||||
|
|
||||||
// kill all
|
|
||||||
NLog::log("\n{}Killing all windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 0 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 0);
|
|
||||||
|
|
||||||
NLog::log("{}Reloading the config", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/reload"));
|
|
||||||
|
|
||||||
return !ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
REGISTER_TEST_FN(test)
|
|
@@ -1,12 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <vector>
|
|
||||||
#include <functional>
|
|
||||||
|
|
||||||
inline std::vector<std::function<bool()>> testFns;
|
|
||||||
|
|
||||||
#define REGISTER_TEST_FN(fn) \
|
|
||||||
static auto _register_fn = [] { \
|
|
||||||
testFns.emplace_back(fn); \
|
|
||||||
return 1; \
|
|
||||||
}();
|
|
@@ -1,98 +0,0 @@
|
|||||||
#include "tests.hpp"
|
|
||||||
#include "../../shared.hpp"
|
|
||||||
#include "../../hyprctlCompat.hpp"
|
|
||||||
#include <print>
|
|
||||||
#include <thread>
|
|
||||||
#include <chrono>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include "../shared.hpp"
|
|
||||||
|
|
||||||
static int ret = 0;
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::Memory;
|
|
||||||
|
|
||||||
#define UP CUniquePointer
|
|
||||||
#define SP CSharedPointer
|
|
||||||
|
|
||||||
static bool test() {
|
|
||||||
NLog::log("{}Testing windows", Colors::GREEN);
|
|
||||||
|
|
||||||
// test on workspace "window"
|
|
||||||
NLog::log("{}Switching to workspace `window`", Colors::YELLOW);
|
|
||||||
getFromSocket("/dispatch workspace name:window");
|
|
||||||
|
|
||||||
NLog::log("{}Spawning kittyProcA", Colors::YELLOW);
|
|
||||||
auto kittyProcA = Tests::spawnKitty();
|
|
||||||
|
|
||||||
if (!kittyProcA) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 1 window", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 1);
|
|
||||||
|
|
||||||
// check kitty properties. One kitty should take the entire screen, as this is smart gaps
|
|
||||||
NLog::log("{}Expecting kitty to take up the whole screen", Colors::YELLOW);
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
EXPECT(str.contains("at: 0,0"), true);
|
|
||||||
EXPECT(str.contains("size: 1920,1080"), true);
|
|
||||||
EXPECT(str.contains("fullscreen: 0"), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Spawning kittyProcB", Colors::YELLOW);
|
|
||||||
auto kittyProcB = Tests::spawnKitty();
|
|
||||||
if (!kittyProcB) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 2 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 2);
|
|
||||||
|
|
||||||
// open xeyes
|
|
||||||
NLog::log("{}Spawning xeyes", Colors::YELLOW);
|
|
||||||
getFromSocket("/dispatch exec xeyes");
|
|
||||||
|
|
||||||
NLog::log("{}Keep checking if xeyes spawned", Colors::YELLOW);
|
|
||||||
int counter = 0;
|
|
||||||
while (Tests::windowCount() != 3) {
|
|
||||||
counter++;
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(100));
|
|
||||||
|
|
||||||
if (counter > 50) {
|
|
||||||
EXPECT(Tests::windowCount(), 3);
|
|
||||||
return !ret;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 3 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 3);
|
|
||||||
|
|
||||||
NLog::log("{}Checking props of xeyes", Colors::YELLOW);
|
|
||||||
// check some window props of xeyes, try to tile them
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
EXPECT_CONTAINS(str, "floating: 1");
|
|
||||||
getFromSocket("/dispatch settiled class:XEyes");
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
|
||||||
str = getFromSocket("/clients");
|
|
||||||
EXPECT_NOT_CONTAINS(str, "floating: 1");
|
|
||||||
}
|
|
||||||
|
|
||||||
// kill all
|
|
||||||
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 0 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 0);
|
|
||||||
|
|
||||||
return !ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
REGISTER_TEST_FN(test)
|
|
@@ -1,351 +0,0 @@
|
|||||||
#include "tests.hpp"
|
|
||||||
#include "../../shared.hpp"
|
|
||||||
#include "../../hyprctlCompat.hpp"
|
|
||||||
#include <print>
|
|
||||||
#include <thread>
|
|
||||||
#include <chrono>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include "../shared.hpp"
|
|
||||||
|
|
||||||
static int ret = 0;
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::Memory;
|
|
||||||
|
|
||||||
#define UP CUniquePointer
|
|
||||||
#define SP CSharedPointer
|
|
||||||
|
|
||||||
static bool test() {
|
|
||||||
NLog::log("{}Testing workspaces", Colors::GREEN);
|
|
||||||
|
|
||||||
// test on workspace "window"
|
|
||||||
NLog::log("{}Switching to workspace 1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
NLog::log("{}Spawning kittyProc on ws 1", Colors::YELLOW);
|
|
||||||
auto kittyProcA = Tests::spawnKitty();
|
|
||||||
|
|
||||||
if (!kittyProcA) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace 3", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 3"));
|
|
||||||
|
|
||||||
NLog::log("{}Spawning kittyProc on ws 3", Colors::YELLOW);
|
|
||||||
auto kittyProcB = Tests::spawnKitty();
|
|
||||||
|
|
||||||
if (!kittyProcB) {
|
|
||||||
NLog::log("{}Error: kitty did not spawn", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace 1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace +1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace +1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 2 (2)");
|
|
||||||
}
|
|
||||||
|
|
||||||
// check if the other workspaces are alive
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/workspaces");
|
|
||||||
EXPECT_CONTAINS(str, "workspace ID 3 (3)");
|
|
||||||
EXPECT_CONTAINS(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace 1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace m+1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace m+1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace -1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace -1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 2 (2)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace 1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace r+1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace r+1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 2 (2)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace r+1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace r+1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace r~1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace r~1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace empty", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace empty"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 2 (2)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace previous", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace previous"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace name:TEST_WORKSPACE_NULL", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace name:TEST_WORKSPACE_NULL"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID -1337 (TEST_WORKSPACE_NULL)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace 1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
// add a new monitor
|
|
||||||
NLog::log("{}Adding a new monitor", Colors::YELLOW);
|
|
||||||
EXPECT(getFromSocket("/output create headless"), "ok")
|
|
||||||
|
|
||||||
// should take workspace 2
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/monitors");
|
|
||||||
EXPECT_CONTAINS(str, "active workspace: 2 (2)");
|
|
||||||
EXPECT_CONTAINS(str, "active workspace: 1 (1)");
|
|
||||||
EXPECT_CONTAINS(str, "HEADLESS-3");
|
|
||||||
}
|
|
||||||
|
|
||||||
// focus the first monitor
|
|
||||||
OK(getFromSocket("/dispatch focusmonitor 0"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace r+1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace r+1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace r~2", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
OK(getFromSocket("/dispatch workspace r~2"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace m+1", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch workspace m+1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Switching to workspace 1", Colors::YELLOW);
|
|
||||||
// no OK: this will throw an error as it should
|
|
||||||
getFromSocket("/dispatch workspace 1");
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Testing back_and_forth", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword binds:workspace_back_and_forth true"));
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword binds:workspace_back_and_forth false"));
|
|
||||||
|
|
||||||
NLog::log("{}Testing hide_special_on_workspace_change", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword binds:hide_special_on_workspace_change true"));
|
|
||||||
OK(getFromSocket("/dispatch workspace special:HELLO"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/monitors");
|
|
||||||
EXPECT_CONTAINS(str, "special workspace: -");
|
|
||||||
EXPECT_CONTAINS(str, "special:HELLO");
|
|
||||||
}
|
|
||||||
|
|
||||||
// no OK: will err (it shouldnt prolly but oh well)
|
|
||||||
getFromSocket("/dispatch workspace 3");
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/monitors");
|
|
||||||
EXPECT_COUNT_STRING(str, "special workspace: 0 ()", 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword binds:hide_special_on_workspace_change false"));
|
|
||||||
|
|
||||||
NLog::log("{}Testing allow_workspace_cycles", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword binds:allow_workspace_cycles true"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
OK(getFromSocket("/dispatch workspace 3"));
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch workspace previous"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch workspace previous"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 1 (1)");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch workspace previous"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activeworkspace");
|
|
||||||
EXPECT_STARTS_WITH(str, "workspace ID 3 (3)");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword binds:allow_workspace_cycles false"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch workspace 1"));
|
|
||||||
|
|
||||||
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
// spawn 3 kitties
|
|
||||||
NLog::log("{}Testing focus_preferred_method", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/keyword dwindle:force_split 2"));
|
|
||||||
Tests::spawnKitty("kitty_A");
|
|
||||||
Tests::spawnKitty("kitty_B");
|
|
||||||
Tests::spawnKitty("kitty_C");
|
|
||||||
OK(getFromSocket("/keyword dwindle:force_split 0"));
|
|
||||||
|
|
||||||
// focus kitty 2: will be top right (dwindle)
|
|
||||||
OK(getFromSocket("/dispatch focuswindow class:kitty_B"));
|
|
||||||
|
|
||||||
// resize it to be a bit taller
|
|
||||||
OK(getFromSocket("/dispatch resizeactive +20 +20"));
|
|
||||||
|
|
||||||
// now we test focus methods.
|
|
||||||
OK(getFromSocket("/keyword binds:focus_preferred_method 0"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch focuswindow class:kitty_C"));
|
|
||||||
OK(getFromSocket("/dispatch focuswindow class:kitty_A"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch movefocus r"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_C");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch focuswindow class:kitty_A"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword binds:focus_preferred_method 1"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch movefocus r"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_B");
|
|
||||||
}
|
|
||||||
|
|
||||||
NLog::log("{}Testing movefocus_cycles_fullscreen", Colors::YELLOW);
|
|
||||||
OK(getFromSocket("/dispatch focuswindow class:kitty_A"));
|
|
||||||
OK(getFromSocket("/dispatch focusmonitor HEADLESS-3"));
|
|
||||||
Tests::spawnKitty("kitty_D");
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_D");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch focusmonitor l"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_A");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword binds:movefocus_cycles_fullscreen false"));
|
|
||||||
OK(getFromSocket("/dispatch fullscreen 0"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch movefocus r"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_D");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch focusmonitor l"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_A");
|
|
||||||
}
|
|
||||||
|
|
||||||
OK(getFromSocket("/keyword binds:movefocus_cycles_fullscreen true"));
|
|
||||||
|
|
||||||
OK(getFromSocket("/dispatch movefocus r"));
|
|
||||||
|
|
||||||
{
|
|
||||||
auto str = getFromSocket("/activewindow");
|
|
||||||
EXPECT_CONTAINS(str, "class: kitty_B");
|
|
||||||
}
|
|
||||||
|
|
||||||
// destroy the headless output
|
|
||||||
OK(getFromSocket("/output remove HEADLESS-3"));
|
|
||||||
|
|
||||||
// kill all
|
|
||||||
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
||||||
Tests::killAllWindows();
|
|
||||||
|
|
||||||
NLog::log("{}Expecting 0 windows", Colors::YELLOW);
|
|
||||||
EXPECT(Tests::windowCount(), 0);
|
|
||||||
|
|
||||||
return !ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
REGISTER_TEST_FN(test)
|
|
@@ -1,21 +0,0 @@
|
|||||||
#include "plugin.hpp"
|
|
||||||
#include "../../shared.hpp"
|
|
||||||
#include "../../hyprctlCompat.hpp"
|
|
||||||
#include <print>
|
|
||||||
#include <thread>
|
|
||||||
#include <chrono>
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include "../shared.hpp"
|
|
||||||
|
|
||||||
bool testPlugin() {
|
|
||||||
const auto RESPONSE = getFromSocket("/dispatch plugin:test:test");
|
|
||||||
|
|
||||||
if (RESPONSE != "ok") {
|
|
||||||
NLog::log("{}Plugin tests failed, plugin returned:\n{}{}", Colors::RED, Colors::RESET, RESPONSE);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
@@ -1,3 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
bool testPlugin();
|
|
@@ -1,92 +0,0 @@
|
|||||||
#include "shared.hpp"
|
|
||||||
#include <csignal>
|
|
||||||
#include <cerrno>
|
|
||||||
#include <thread>
|
|
||||||
#include <print>
|
|
||||||
#include "../shared.hpp"
|
|
||||||
#include "../hyprctlCompat.hpp"
|
|
||||||
|
|
||||||
using namespace Hyprutils::OS;
|
|
||||||
using namespace Hyprutils::Memory;
|
|
||||||
|
|
||||||
CUniquePointer<CProcess> Tests::spawnKitty(const std::string& class_) {
|
|
||||||
const auto COUNT_BEFORE = windowCount();
|
|
||||||
|
|
||||||
CUniquePointer<CProcess> kitty = makeUnique<CProcess>("kitty", class_.empty() ? std::vector<std::string>{} : std::vector<std::string>{"--class", class_});
|
|
||||||
kitty->addEnv("WAYLAND_DISPLAY", WLDISPLAY);
|
|
||||||
kitty->runAsync();
|
|
||||||
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(10));
|
|
||||||
|
|
||||||
// wait while kitty spawns
|
|
||||||
int counter = 0;
|
|
||||||
while (processAlive(kitty->pid()) && windowCount() == COUNT_BEFORE) {
|
|
||||||
counter++;
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(100));
|
|
||||||
|
|
||||||
if (counter > 50)
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!processAlive(kitty->pid()))
|
|
||||||
return nullptr;
|
|
||||||
|
|
||||||
return kitty;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Tests::processAlive(pid_t pid) {
|
|
||||||
errno = 0;
|
|
||||||
int ret = kill(pid, 0);
|
|
||||||
return ret != -1 || errno != ESRCH;
|
|
||||||
}
|
|
||||||
|
|
||||||
int Tests::windowCount() {
|
|
||||||
return countOccurrences(getFromSocket("/clients"), "focusHistoryID: ");
|
|
||||||
}
|
|
||||||
|
|
||||||
int Tests::countOccurrences(const std::string& in, const std::string& what) {
|
|
||||||
int cnt = 0;
|
|
||||||
auto pos = in.find(what);
|
|
||||||
while (pos != std::string::npos) {
|
|
||||||
cnt++;
|
|
||||||
pos = in.find(what, pos + what.length() - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return cnt;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Tests::killAllWindows() {
|
|
||||||
auto str = getFromSocket("/clients");
|
|
||||||
auto pos = str.find("Window ");
|
|
||||||
while (pos != std::string::npos) {
|
|
||||||
auto pos2 = str.find(" -> ", pos);
|
|
||||||
getFromSocket("/dispatch killwindow address:0x" + str.substr(pos + 7, pos2 - pos - 7));
|
|
||||||
pos = str.find("Window ", pos + 5);
|
|
||||||
}
|
|
||||||
|
|
||||||
int counter = 0;
|
|
||||||
while (Tests::windowCount() != 0) {
|
|
||||||
counter++;
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(100));
|
|
||||||
|
|
||||||
if (counter > 50) {
|
|
||||||
std::println("{}Timed out waiting for windows to close", Colors::RED);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Tests::waitUntilWindowsN(int n) {
|
|
||||||
int counter = 0;
|
|
||||||
while (Tests::windowCount() != n) {
|
|
||||||
counter++;
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(100));
|
|
||||||
|
|
||||||
if (counter > 50) {
|
|
||||||
std::println("{}Timed out waiting for windows", Colors::RED);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,17 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <hyprutils/os/Process.hpp>
|
|
||||||
#include <hyprutils/memory/WeakPtr.hpp>
|
|
||||||
#include <sys/types.h>
|
|
||||||
|
|
||||||
#include "../Log.hpp"
|
|
||||||
|
|
||||||
//NOLINTNEXTLINE
|
|
||||||
namespace Tests {
|
|
||||||
Hyprutils::Memory::CUniquePointer<Hyprutils::OS::CProcess> spawnKitty(const std::string& class_ = "");
|
|
||||||
bool processAlive(pid_t pid);
|
|
||||||
int windowCount();
|
|
||||||
int countOccurrences(const std::string& in, const std::string& what);
|
|
||||||
bool killAllWindows();
|
|
||||||
void waitUntilWindowsN(int n);
|
|
||||||
};
|
|
@@ -1,313 +0,0 @@
|
|||||||
# This is an example Hyprland config file.
|
|
||||||
# Refer to the wiki for more information.
|
|
||||||
# https://wiki.hyprland.org/Configuring/
|
|
||||||
|
|
||||||
# Please note not all available settings / options are set here.
|
|
||||||
# For a full list, see the wiki
|
|
||||||
|
|
||||||
# You can split this configuration into multiple files
|
|
||||||
# Create your files separately and then link them to this file like this:
|
|
||||||
# source = ~/.config/hypr/myColors.conf
|
|
||||||
|
|
||||||
|
|
||||||
################
|
|
||||||
### MONITORS ###
|
|
||||||
################
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Monitors/
|
|
||||||
|
|
||||||
monitor=HEADLESS-1,1920x1080@60,auto-right,1
|
|
||||||
monitor=HEADLESS-2,1920x1080@60,auto-right,1
|
|
||||||
monitor=HEADLESS-3,1920x1080@60,auto-right,1
|
|
||||||
monitor=HEADLESS-4,1920x1080@60,auto-right,1
|
|
||||||
monitor=HEADLESS-5,1920x1080@60,auto-right,1
|
|
||||||
monitor=HEADLESS-6,1920x1080@60,auto-right,1
|
|
||||||
|
|
||||||
monitor=,disabled
|
|
||||||
|
|
||||||
|
|
||||||
###################
|
|
||||||
### MY PROGRAMS ###
|
|
||||||
###################
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Keywords/
|
|
||||||
|
|
||||||
# Set programs that you use
|
|
||||||
$terminal = kitty
|
|
||||||
$fileManager = dolphin
|
|
||||||
$menu = wofi --show drun
|
|
||||||
|
|
||||||
|
|
||||||
#################
|
|
||||||
### AUTOSTART ###
|
|
||||||
#################
|
|
||||||
|
|
||||||
# Autostart necessary processes (like notifications daemons, status bars, etc.)
|
|
||||||
# Or execute your favorite apps at launch like this:
|
|
||||||
|
|
||||||
# exec-once = $terminal
|
|
||||||
# exec-once = nm-applet &
|
|
||||||
# exec-once = waybar & hyprpaper & firefox
|
|
||||||
|
|
||||||
|
|
||||||
#############################
|
|
||||||
### ENVIRONMENT VARIABLES ###
|
|
||||||
#############################
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Environment-variables/
|
|
||||||
|
|
||||||
env = XCURSOR_SIZE,24
|
|
||||||
env = HYPRCURSOR_SIZE,24
|
|
||||||
|
|
||||||
|
|
||||||
#####################
|
|
||||||
### LOOK AND FEEL ###
|
|
||||||
#####################
|
|
||||||
|
|
||||||
# Refer to https://wiki.hyprland.org/Configuring/Variables/
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#general
|
|
||||||
general {
|
|
||||||
gaps_in = 5
|
|
||||||
gaps_out = 20
|
|
||||||
|
|
||||||
border_size = 2
|
|
||||||
|
|
||||||
snap {
|
|
||||||
enabled = true
|
|
||||||
window_gap = 8
|
|
||||||
monitor_gap = 10
|
|
||||||
respect_gaps = false
|
|
||||||
border_overlap = false
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#variable-types for info about colors
|
|
||||||
col.active_border = rgba(33ccffee) rgba(00ff99ee) 45deg
|
|
||||||
col.inactive_border = rgba(595959aa)
|
|
||||||
|
|
||||||
# Set to true enable resizing windows by clicking and dragging on borders and gaps
|
|
||||||
resize_on_border = false
|
|
||||||
|
|
||||||
# Please see https://wiki.hyprland.org/Configuring/Tearing/ before you turn this on
|
|
||||||
allow_tearing = false
|
|
||||||
|
|
||||||
layout = dwindle
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#decoration
|
|
||||||
decoration {
|
|
||||||
rounding = 10
|
|
||||||
rounding_power = 2
|
|
||||||
|
|
||||||
# Change transparency of focused and unfocused windows
|
|
||||||
active_opacity = 1.0
|
|
||||||
inactive_opacity = 1.0
|
|
||||||
|
|
||||||
shadow {
|
|
||||||
enabled = true
|
|
||||||
range = 4
|
|
||||||
render_power = 3
|
|
||||||
color = rgba(1a1a1aee)
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#blur
|
|
||||||
blur {
|
|
||||||
enabled = true
|
|
||||||
size = 3
|
|
||||||
passes = 1
|
|
||||||
|
|
||||||
vibrancy = 0.1696
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#animations
|
|
||||||
animations {
|
|
||||||
enabled = 0
|
|
||||||
|
|
||||||
# Default animations, see https://wiki.hyprland.org/Configuring/Animations/ for more
|
|
||||||
|
|
||||||
bezier = easeOutQuint,0.23,1,0.32,1
|
|
||||||
bezier = easeInOutCubic,0.65,0.05,0.36,1
|
|
||||||
bezier = linear,0,0,1,1
|
|
||||||
bezier = almostLinear,0.5,0.5,0.75,1.0
|
|
||||||
bezier = quick,0.15,0,0.1,1
|
|
||||||
|
|
||||||
animation = global, 1, 10, default
|
|
||||||
animation = border, 1, 5.39, easeOutQuint
|
|
||||||
animation = windows, 1, 4.79, easeOutQuint
|
|
||||||
animation = windowsIn, 1, 4.1, easeOutQuint, popin 87%
|
|
||||||
animation = windowsOut, 1, 1.49, linear, popin 87%
|
|
||||||
animation = fadeIn, 1, 1.73, almostLinear
|
|
||||||
animation = fadeOut, 1, 1.46, almostLinear
|
|
||||||
animation = fade, 1, 3.03, quick
|
|
||||||
animation = layers, 1, 3.81, easeOutQuint
|
|
||||||
animation = layersIn, 1, 4, easeOutQuint, fade
|
|
||||||
animation = layersOut, 1, 1.5, linear, fade
|
|
||||||
animation = fadeLayersIn, 1, 1.79, almostLinear
|
|
||||||
animation = fadeLayersOut, 1, 1.39, almostLinear
|
|
||||||
animation = workspaces, 1, 1.94, almostLinear, fade
|
|
||||||
animation = workspacesIn, 1, 1.21, almostLinear, fade
|
|
||||||
animation = workspacesOut, 1, 1.94, almostLinear, fade
|
|
||||||
}
|
|
||||||
|
|
||||||
# Ref https://wiki.hyprland.org/Configuring/Workspace-Rules/
|
|
||||||
# "Smart gaps" / "No gaps when only"
|
|
||||||
# uncomment all if you wish to use that.
|
|
||||||
# workspace = w[tv1], gapsout:0, gapsin:0
|
|
||||||
# workspace = f[1], gapsout:0, gapsin:0
|
|
||||||
# windowrulev2 = bordersize 0, floating:0, onworkspace:w[tv1]
|
|
||||||
# windowrulev2 = rounding 0, floating:0, onworkspace:w[tv1]
|
|
||||||
# windowrulev2 = bordersize 0, floating:0, onworkspace:f[1]
|
|
||||||
# windowrulev2 = rounding 0, floating:0, onworkspace:f[1]
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Dwindle-Layout/ for more
|
|
||||||
dwindle {
|
|
||||||
pseudotile = true # Master switch for pseudotiling. Enabling is bound to mainMod + P in the keybinds section below
|
|
||||||
preserve_split = true # You probably want this
|
|
||||||
}
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Master-Layout/ for more
|
|
||||||
master {
|
|
||||||
new_status = master
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#misc
|
|
||||||
misc {
|
|
||||||
force_default_wallpaper = -1 # Set to 0 or 1 to disable the anime mascot wallpapers
|
|
||||||
disable_hyprland_logo = false # If true disables the random hyprland logo / anime girl background. :(
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#############
|
|
||||||
### INPUT ###
|
|
||||||
#############
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#input
|
|
||||||
input {
|
|
||||||
kb_layout = us
|
|
||||||
kb_variant =
|
|
||||||
kb_model =
|
|
||||||
kb_options =
|
|
||||||
kb_rules =
|
|
||||||
|
|
||||||
follow_mouse = 1
|
|
||||||
|
|
||||||
sensitivity = 0 # -1.0 - 1.0, 0 means no modification.
|
|
||||||
|
|
||||||
touchpad {
|
|
||||||
natural_scroll = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
# https://wiki.hyprland.org/Configuring/Variables/#gestures
|
|
||||||
gestures {
|
|
||||||
workspace_swipe = false
|
|
||||||
}
|
|
||||||
|
|
||||||
# Example per-device config
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Keywords/#per-device-input-configs for more
|
|
||||||
device {
|
|
||||||
name = epic-mouse-v1
|
|
||||||
sensitivity = -0.5
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
###################
|
|
||||||
### KEYBINDINGS ###
|
|
||||||
###################
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Keywords/
|
|
||||||
$mainMod = SUPER # Sets "Windows" key as main modifier
|
|
||||||
|
|
||||||
# Example binds, see https://wiki.hyprland.org/Configuring/Binds/ for more
|
|
||||||
bind = $mainMod, Q, exec, $terminal
|
|
||||||
bind = $mainMod, C, killactive,
|
|
||||||
bind = $mainMod, M, exit,
|
|
||||||
bind = $mainMod, E, exec, $fileManager
|
|
||||||
bind = $mainMod, V, togglefloating,
|
|
||||||
bind = $mainMod, R, exec, $menu
|
|
||||||
bind = $mainMod, P, pseudo, # dwindle
|
|
||||||
bind = $mainMod, J, togglesplit, # dwindle
|
|
||||||
|
|
||||||
# Move focus with mainMod + arrow keys
|
|
||||||
bind = $mainMod, left, movefocus, l
|
|
||||||
bind = $mainMod, right, movefocus, r
|
|
||||||
bind = $mainMod, up, movefocus, u
|
|
||||||
bind = $mainMod, down, movefocus, d
|
|
||||||
|
|
||||||
# Switch workspaces with mainMod + [0-9]
|
|
||||||
bind = $mainMod, 1, workspace, 1
|
|
||||||
bind = $mainMod, 2, workspace, 2
|
|
||||||
bind = $mainMod, 3, workspace, 3
|
|
||||||
bind = $mainMod, 4, workspace, 4
|
|
||||||
bind = $mainMod, 5, workspace, 5
|
|
||||||
bind = $mainMod, 6, workspace, 6
|
|
||||||
bind = $mainMod, 7, workspace, 7
|
|
||||||
bind = $mainMod, 8, workspace, 8
|
|
||||||
bind = $mainMod, 9, workspace, 9
|
|
||||||
bind = $mainMod, 0, workspace, 10
|
|
||||||
|
|
||||||
# Move active window to a workspace with mainMod + SHIFT + [0-9]
|
|
||||||
bind = $mainMod SHIFT, 1, movetoworkspace, 1
|
|
||||||
bind = $mainMod SHIFT, 2, movetoworkspace, 2
|
|
||||||
bind = $mainMod SHIFT, 3, movetoworkspace, 3
|
|
||||||
bind = $mainMod SHIFT, 4, movetoworkspace, 4
|
|
||||||
bind = $mainMod SHIFT, 5, movetoworkspace, 5
|
|
||||||
bind = $mainMod SHIFT, 6, movetoworkspace, 6
|
|
||||||
bind = $mainMod SHIFT, 7, movetoworkspace, 7
|
|
||||||
bind = $mainMod SHIFT, 8, movetoworkspace, 8
|
|
||||||
bind = $mainMod SHIFT, 9, movetoworkspace, 9
|
|
||||||
bind = $mainMod SHIFT, 0, movetoworkspace, 10
|
|
||||||
|
|
||||||
# Example special workspace (scratchpad)
|
|
||||||
bind = $mainMod, S, togglespecialworkspace, magic
|
|
||||||
bind = $mainMod SHIFT, S, movetoworkspace, special:magic
|
|
||||||
|
|
||||||
# Scroll through existing workspaces with mainMod + scroll
|
|
||||||
bind = $mainMod, mouse_down, workspace, e+1
|
|
||||||
bind = $mainMod, mouse_up, workspace, e-1
|
|
||||||
|
|
||||||
# Move/resize windows with mainMod + LMB/RMB and dragging
|
|
||||||
bindm = $mainMod, mouse:272, movewindow
|
|
||||||
bindm = $mainMod, mouse:273, resizewindow
|
|
||||||
|
|
||||||
# Laptop multimedia keys for volume and LCD brightness
|
|
||||||
bindel = ,XF86AudioRaiseVolume, exec, wpctl set-volume -l 1 @DEFAULT_AUDIO_SINK@ 5%+
|
|
||||||
bindel = ,XF86AudioLowerVolume, exec, wpctl set-volume @DEFAULT_AUDIO_SINK@ 5%-
|
|
||||||
bindel = ,XF86AudioMute, exec, wpctl set-mute @DEFAULT_AUDIO_SINK@ toggle
|
|
||||||
bindel = ,XF86AudioMicMute, exec, wpctl set-mute @DEFAULT_AUDIO_SOURCE@ toggle
|
|
||||||
bindel = ,XF86MonBrightnessUp, exec, brightnessctl s 10%+
|
|
||||||
bindel = ,XF86MonBrightnessDown, exec, brightnessctl s 10%-
|
|
||||||
|
|
||||||
# Requires playerctl
|
|
||||||
bindl = , XF86AudioNext, exec, playerctl next
|
|
||||||
bindl = , XF86AudioPause, exec, playerctl play-pause
|
|
||||||
bindl = , XF86AudioPlay, exec, playerctl play-pause
|
|
||||||
bindl = , XF86AudioPrev, exec, playerctl previous
|
|
||||||
|
|
||||||
##############################
|
|
||||||
### WINDOWS AND WORKSPACES ###
|
|
||||||
##############################
|
|
||||||
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Window-Rules/ for more
|
|
||||||
# See https://wiki.hyprland.org/Configuring/Workspace-Rules/ for workspace rules
|
|
||||||
|
|
||||||
# Example windowrule v1
|
|
||||||
# windowrule = float, ^(kitty)$
|
|
||||||
|
|
||||||
# Example windowrule v2
|
|
||||||
# windowrulev2 = float,class:^(kitty)$,title:^(kitty)$
|
|
||||||
|
|
||||||
# Ignore maximize requests from apps. You'll probably like this.
|
|
||||||
windowrulev2 = suppressevent maximize, class:.*
|
|
||||||
|
|
||||||
# Fix some dragging issues with XWayland
|
|
||||||
windowrulev2 = nofocus,class:^$,title:^$,xwayland:1,floating:1,fullscreen:0,pinned:0
|
|
||||||
|
|
||||||
# Workspace "windows" is a smart gaps one
|
|
||||||
workspace = n[s:window] w[tv1], gapsout:0, gapsin:0
|
|
||||||
workspace = n[s:window] f[1], gapsout:0, gapsin:0
|
|
||||||
windowrulev2 = bordersize 0, floating:0, onworkspace:n[s:window] w[tv1]
|
|
||||||
windowrulev2 = rounding 0, floating:0, onworkspace:n[s:window] w[tv1]
|
|
||||||
windowrulev2 = bordersize 0, floating:0, onworkspace:n[s:window] f[1]
|
|
||||||
windowrulev2 = rounding 0, floating:0, onworkspace:n[s:window] f[1]
|
|
78
meson.build
@@ -1,17 +1,13 @@
|
|||||||
project(
|
project('Hyprland', 'cpp', 'c',
|
||||||
'Hyprland',
|
version : run_command('cat', join_paths(meson.source_root(), 'VERSION'), check: true).stdout().strip(),
|
||||||
'cpp',
|
default_options : [
|
||||||
'c',
|
|
||||||
version: run_command('cat', join_paths(meson.project_source_root(), 'VERSION'), check: true).stdout().strip(),
|
|
||||||
default_options: [
|
|
||||||
'warning_level=2',
|
'warning_level=2',
|
||||||
'default_library=static',
|
'default_library=static',
|
||||||
'optimization=3',
|
'optimization=3',
|
||||||
'buildtype=release',
|
'buildtype=release',
|
||||||
'debug=false',
|
'debug=false',
|
||||||
'cpp_std=c++26',
|
'cpp_std=c++26',
|
||||||
],
|
])
|
||||||
)
|
|
||||||
|
|
||||||
datarootdir = '-DDATAROOTDIR="' + get_option('prefix') / get_option('datadir') + '"'
|
datarootdir = '-DDATAROOTDIR="' + get_option('prefix') / get_option('datadir') + '"'
|
||||||
add_project_arguments(
|
add_project_arguments(
|
||||||
@@ -20,32 +16,16 @@ add_project_arguments(
|
|||||||
'-Wno-unused-value',
|
'-Wno-unused-value',
|
||||||
'-Wno-missing-field-initializers',
|
'-Wno-missing-field-initializers',
|
||||||
'-Wno-narrowing',
|
'-Wno-narrowing',
|
||||||
'-Wno-pointer-arith', datarootdir,
|
'-Wno-pointer-arith',
|
||||||
'-DHYPRLAND_VERSION="' + meson.project_version() + '"',
|
datarootdir,
|
||||||
],
|
],
|
||||||
language: 'cpp',
|
language: 'cpp')
|
||||||
)
|
|
||||||
|
|
||||||
cpp_compiler = meson.get_compiler('cpp')
|
cpp_compiler = meson.get_compiler('cpp')
|
||||||
if cpp_compiler.check_header('execinfo.h')
|
if cpp_compiler.check_header('execinfo.h')
|
||||||
add_project_arguments('-DHAS_EXECINFO', language: 'cpp')
|
add_project_arguments('-DHAS_EXECINFO', language: 'cpp')
|
||||||
endif
|
endif
|
||||||
|
|
||||||
aquamarine = dependency('aquamarine', version: '>=0.9.0')
|
|
||||||
hyprcursor = dependency('hyprcursor', version: '>=0.1.7')
|
|
||||||
hyprgraphics = dependency('hyprgraphics', version: '>= 0.1.3')
|
|
||||||
hyprlang = dependency('hyprlang', version: '>= 0.3.2')
|
|
||||||
hyprutils = dependency('hyprutils', version: '>= 0.8.1')
|
|
||||||
aquamarine_version_list = aquamarine.version().split('.')
|
|
||||||
add_project_arguments(['-DAQUAMARINE_VERSION="@0@"'.format(aquamarine.version())], language: 'cpp')
|
|
||||||
add_project_arguments(['-DAQUAMARINE_VERSION_MAJOR=@0@'.format(aquamarine_version_list.get(0))], language: 'cpp')
|
|
||||||
add_project_arguments(['-DAQUAMARINE_VERSION_MINOR=@0@'.format(aquamarine_version_list.get(1))], language: 'cpp')
|
|
||||||
add_project_arguments(['-DAQUAMARINE_VERSION_PATCH=@0@'.format(aquamarine_version_list.get(2))], language: 'cpp')
|
|
||||||
add_project_arguments(['-DHYPRCURSOR_VERSION="@0@"'.format(hyprcursor.version())], language: 'cpp')
|
|
||||||
add_project_arguments(['-DHYPRGRAPHICS_VERSION="@0@"'.format(hyprgraphics.version())], language: 'cpp')
|
|
||||||
add_project_arguments(['-DHYPRLANG_VERSION="@0@"'.format(hyprlang.version())], language: 'cpp')
|
|
||||||
add_project_arguments(['-DHYPRUTILS_VERSION="@0@"'.format(hyprutils.version())], language: 'cpp')
|
|
||||||
|
|
||||||
xcb_dep = dependency('xcb', required: get_option('xwayland'))
|
xcb_dep = dependency('xcb', required: get_option('xwayland'))
|
||||||
xcb_composite_dep = dependency('xcb-composite', required: get_option('xwayland'))
|
xcb_composite_dep = dependency('xcb-composite', required: get_option('xwayland'))
|
||||||
xcb_errors_dep = dependency('xcb-errors', required: get_option('xwayland'))
|
xcb_errors_dep = dependency('xcb-errors', required: get_option('xwayland'))
|
||||||
@@ -54,7 +34,11 @@ xcb_render_dep = dependency('xcb-render', required: get_option('xwayland'))
|
|||||||
xcb_res_dep = dependency('xcb-res', required: get_option('xwayland'))
|
xcb_res_dep = dependency('xcb-res', required: get_option('xwayland'))
|
||||||
xcb_xfixes_dep = dependency('xcb-xfixes', required: get_option('xwayland'))
|
xcb_xfixes_dep = dependency('xcb-xfixes', required: get_option('xwayland'))
|
||||||
|
|
||||||
gio_dep = dependency('gio-2.0', required: true)
|
gio_dep = dependency('gio-2.0', required:true)
|
||||||
|
|
||||||
|
cmake = import('cmake')
|
||||||
|
udis = cmake.subproject('udis86')
|
||||||
|
udis86 = udis.dependency('libudis86')
|
||||||
|
|
||||||
if not xcb_dep.found()
|
if not xcb_dep.found()
|
||||||
add_project_arguments('-DNO_XWAYLAND', language: 'cpp')
|
add_project_arguments('-DNO_XWAYLAND', language: 'cpp')
|
||||||
@@ -62,57 +46,35 @@ endif
|
|||||||
|
|
||||||
backtrace_dep = cpp_compiler.find_library('execinfo', required: false)
|
backtrace_dep = cpp_compiler.find_library('execinfo', required: false)
|
||||||
epoll_dep = dependency('epoll-shim', required: false) # timerfd on BSDs
|
epoll_dep = dependency('epoll-shim', required: false) # timerfd on BSDs
|
||||||
inotify_dep = dependency('libinotify', required: false) # inotify on BSDs
|
|
||||||
|
|
||||||
re2 = dependency('re2', required: true)
|
if get_option('systemd').enabled()
|
||||||
|
|
||||||
# Handle options
|
|
||||||
systemd_option = get_option('systemd')
|
|
||||||
systemd = dependency('systemd', required: systemd_option)
|
|
||||||
systemd_option.enable_auto_if(systemd.found())
|
|
||||||
|
|
||||||
if (systemd_option.enabled())
|
|
||||||
message('Enabling systemd integration')
|
|
||||||
add_project_arguments('-DUSES_SYSTEMD', language: 'cpp')
|
add_project_arguments('-DUSES_SYSTEMD', language: 'cpp')
|
||||||
subdir('systemd')
|
endif
|
||||||
|
|
||||||
|
if get_option('legacy_renderer').enabled()
|
||||||
|
add_project_arguments('-DLEGACY_RENDERER', language: 'cpp')
|
||||||
endif
|
endif
|
||||||
|
|
||||||
if get_option('buildtype') == 'debug'
|
if get_option('buildtype') == 'debug'
|
||||||
add_project_arguments('-DHYPRLAND_DEBUG', language: 'cpp')
|
add_project_arguments('-DHYPRLAND_DEBUG', language: 'cpp')
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# Generate hyprland version and populate version.h
|
version_h = run_command('sh', '-c', 'scripts/generateVersion.sh', check: true)
|
||||||
run_command('sh', '-c', 'scripts/generateVersion.sh', check: true)
|
|
||||||
# Make shader files includable
|
|
||||||
run_command('sh', '-c', 'scripts/generateShaderIncludes.sh', check: true)
|
|
||||||
|
|
||||||
# Install headers
|
globber = run_command('find', 'src', '-name', '*.h*', check: true)
|
||||||
globber = run_command('find', 'src', '-name', '*.h*', '-o', '-name', '*.inc', check: true)
|
|
||||||
headers = globber.stdout().strip().split('\n')
|
headers = globber.stdout().strip().split('\n')
|
||||||
foreach file : headers
|
foreach file : headers
|
||||||
install_headers(file, subdir: 'hyprland', preserve_path: true)
|
install_headers(file, subdir: 'hyprland', preserve_path: true)
|
||||||
endforeach
|
endforeach
|
||||||
|
|
||||||
tracy = dependency('tracy', static: true, required: get_option('tracy_enable'))
|
|
||||||
|
|
||||||
if get_option('tracy_enable') and get_option('buildtype') != 'debugoptimized'
|
|
||||||
warning('Profiling builds should set -- buildtype = debugoptimized')
|
|
||||||
endif
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
subdir('protocols')
|
subdir('protocols')
|
||||||
subdir('src')
|
subdir('src')
|
||||||
subdir('hyprctl')
|
subdir('hyprctl')
|
||||||
|
subdir('hyprpm/src')
|
||||||
subdir('assets')
|
subdir('assets')
|
||||||
subdir('example')
|
subdir('example')
|
||||||
subdir('docs')
|
subdir('docs')
|
||||||
|
|
||||||
if get_option('hyprpm').enabled()
|
|
||||||
subdir('hyprpm/src')
|
|
||||||
endif
|
|
||||||
|
|
||||||
# Generate hyprland.pc
|
|
||||||
pkg_install_dir = join_paths(get_option('datadir'), 'pkgconfig')
|
pkg_install_dir = join_paths(get_option('datadir'), 'pkgconfig')
|
||||||
|
|
||||||
import('pkgconfig').generate(
|
import('pkgconfig').generate(
|
||||||
|
@@ -1,5 +1,3 @@
|
|||||||
option('xwayland', type: 'feature', value: 'auto', description: 'Enable support for X11 applications')
|
option('xwayland', type: 'feature', value: 'auto', description: 'Enable support for X11 applications')
|
||||||
option('systemd', type: 'feature', value: 'auto', description: 'Enable systemd integration')
|
option('systemd', type: 'feature', value: 'auto', description: 'Enable systemd integration')
|
||||||
option('uwsm', type: 'feature', value: 'enabled', description: 'Enable uwsm integration (only if systemd is enabled)')
|
option('legacy_renderer', type: 'feature', value: 'disabled', description: 'Enable legacy renderer')
|
||||||
option('hyprpm', type: 'feature', value: 'enabled', description: 'Enable hyprpm')
|
|
||||||
option('tracy_enable', type: 'boolean', value: false , description: 'Enable profiling')
|
|
||||||
|
297
nix/default.nix
@@ -1,39 +1,43 @@
|
|||||||
{
|
{
|
||||||
lib,
|
lib,
|
||||||
stdenv,
|
stdenv,
|
||||||
stdenvAdapters,
|
|
||||||
pkg-config,
|
pkg-config,
|
||||||
pkgconf,
|
pkgconf,
|
||||||
makeWrapper,
|
makeWrapper,
|
||||||
cmake,
|
cmake,
|
||||||
meson,
|
|
||||||
ninja,
|
ninja,
|
||||||
aquamarine,
|
aquamarine,
|
||||||
binutils,
|
binutils,
|
||||||
cairo,
|
cairo,
|
||||||
epoll-shim,
|
expat,
|
||||||
|
fribidi,
|
||||||
git,
|
git,
|
||||||
glaze,
|
hwdata,
|
||||||
hyprcursor,
|
hyprcursor,
|
||||||
hyprgraphics,
|
|
||||||
hyprland-protocols,
|
|
||||||
hyprland-qtutils,
|
|
||||||
hyprlang,
|
hyprlang,
|
||||||
hyprutils,
|
hyprutils,
|
||||||
hyprwayland-scanner,
|
hyprwayland-scanner,
|
||||||
|
jq,
|
||||||
libGL,
|
libGL,
|
||||||
|
libdatrie,
|
||||||
|
libdisplay-info,
|
||||||
libdrm,
|
libdrm,
|
||||||
libexecinfo,
|
libexecinfo,
|
||||||
libinput,
|
libinput,
|
||||||
libxkbcommon,
|
libliftoff,
|
||||||
|
libselinux,
|
||||||
|
libsepol,
|
||||||
|
libthai,
|
||||||
libuuid,
|
libuuid,
|
||||||
libgbm,
|
libxkbcommon,
|
||||||
|
mesa,
|
||||||
pango,
|
pango,
|
||||||
pciutils,
|
pciutils,
|
||||||
re2,
|
pcre2,
|
||||||
|
python3,
|
||||||
|
seatd,
|
||||||
systemd,
|
systemd,
|
||||||
tomlplusplus,
|
tomlplusplus,
|
||||||
udis86-hyprland,
|
|
||||||
wayland,
|
wayland,
|
||||||
wayland-protocols,
|
wayland-protocols,
|
||||||
wayland-scanner,
|
wayland-scanner,
|
||||||
@@ -41,176 +45,145 @@
|
|||||||
xwayland,
|
xwayland,
|
||||||
debug ? false,
|
debug ? false,
|
||||||
enableXWayland ? true,
|
enableXWayland ? true,
|
||||||
|
legacyRenderer ? false,
|
||||||
withSystemd ? lib.meta.availableOn stdenv.hostPlatform systemd,
|
withSystemd ? lib.meta.availableOn stdenv.hostPlatform systemd,
|
||||||
wrapRuntimeDeps ? true,
|
wrapRuntimeDeps ? true,
|
||||||
version ? "git",
|
version ? "git",
|
||||||
commit,
|
commit,
|
||||||
revCount,
|
|
||||||
date,
|
date,
|
||||||
# deprecated flags
|
# deprecated flags
|
||||||
enableNvidiaPatches ? false,
|
enableNvidiaPatches ? false,
|
||||||
nvidiaPatches ? false,
|
nvidiaPatches ? false,
|
||||||
hidpiXWayland ? false,
|
hidpiXWayland ? false,
|
||||||
legacyRenderer ? false,
|
}:
|
||||||
}: let
|
assert lib.assertMsg (!nvidiaPatches) "The option `nvidiaPatches` has been removed.";
|
||||||
inherit (builtins) foldl' readFile;
|
assert lib.assertMsg (!enableNvidiaPatches) "The option `enableNvidiaPatches` has been removed.";
|
||||||
inherit (lib.asserts) assertMsg;
|
assert lib.assertMsg (!hidpiXWayland) "The option `hidpiXWayland` has been removed. Please refer https://wiki.hyprland.org/Configuring/XWayland";
|
||||||
inherit (lib.attrsets) mapAttrsToList;
|
stdenv.mkDerivation {
|
||||||
inherit (lib.lists) flatten concatLists optional optionals;
|
pname = "hyprland${lib.optionalString debug "-debug"}";
|
||||||
inherit (lib.strings) makeBinPath optionalString mesonBool mesonEnable trim;
|
inherit version;
|
||||||
fs = lib.fileset;
|
|
||||||
|
|
||||||
adapters = flatten [
|
src = lib.cleanSourceWith {
|
||||||
stdenvAdapters.useMoldLinker
|
filter = name: type: let
|
||||||
(lib.optional debug stdenvAdapters.keepDebugInfo)
|
baseName = baseNameOf (toString name);
|
||||||
];
|
in
|
||||||
|
! (lib.hasSuffix ".nix" baseName);
|
||||||
|
src = lib.cleanSource ../.;
|
||||||
|
};
|
||||||
|
|
||||||
customStdenv = foldl' (acc: adapter: adapter acc) stdenv adapters;
|
patches = [
|
||||||
in
|
# forces GCC to use -std=c++26
|
||||||
assert assertMsg (!nvidiaPatches) "The option `nvidiaPatches` has been removed.";
|
./stdcxx.patch
|
||||||
assert assertMsg (!enableNvidiaPatches) "The option `enableNvidiaPatches` has been removed.";
|
];
|
||||||
assert assertMsg (!hidpiXWayland) "The option `hidpiXWayland` has been removed. Please refer https://wiki.hypr.land/Configuring/XWayland";
|
|
||||||
assert assertMsg (!legacyRenderer) "The option `legacyRenderer` has been removed. Legacy renderer is no longer supported.";
|
|
||||||
customStdenv.mkDerivation (finalAttrs: {
|
|
||||||
pname = "hyprland${optionalString debug "-debug"}";
|
|
||||||
inherit version;
|
|
||||||
|
|
||||||
src = fs.toSource {
|
postPatch = ''
|
||||||
root = ../.;
|
# Fix hardcoded paths to /usr installation
|
||||||
fileset =
|
sed -i "s#/usr#$out#" src/render/OpenGL.cpp
|
||||||
fs.intersection
|
|
||||||
# allows non-flake builds to only include files tracked by git
|
|
||||||
(fs.gitTracked ../.)
|
|
||||||
(fs.unions [
|
|
||||||
../assets/hyprland-portals.conf
|
|
||||||
../assets/install
|
|
||||||
../hyprctl
|
|
||||||
../hyprland.pc.in
|
|
||||||
../LICENSE
|
|
||||||
../meson_options.txt
|
|
||||||
../protocols
|
|
||||||
../src
|
|
||||||
../systemd
|
|
||||||
../VERSION
|
|
||||||
(fs.fileFilter (file: file.hasExt "1") ../docs)
|
|
||||||
(fs.fileFilter (file: file.hasExt "conf" || file.hasExt "desktop") ../example)
|
|
||||||
(fs.fileFilter (file: file.hasExt "sh") ../scripts)
|
|
||||||
(fs.fileFilter (file: file.name == "meson.build") ../.)
|
|
||||||
]);
|
|
||||||
};
|
|
||||||
|
|
||||||
postPatch = ''
|
# Remove extra @PREFIX@ to fix pkg-config paths
|
||||||
# Fix hardcoded paths to /usr installation
|
sed -i "s#@PREFIX@/##g" hyprland.pc.in
|
||||||
sed -i "s#/usr#$out#" src/render/OpenGL.cpp
|
'';
|
||||||
|
|
||||||
# Remove extra @PREFIX@ to fix pkg-config paths
|
COMMITS = commit;
|
||||||
sed -i "s#@PREFIX@/##g" hyprland.pc.in
|
DATE = date;
|
||||||
'';
|
DIRTY = lib.optionalString (commit == "") "dirty";
|
||||||
|
HASH = commit;
|
||||||
|
|
||||||
COMMITS = revCount;
|
nativeBuildInputs = [
|
||||||
DATE = date;
|
hyprwayland-scanner
|
||||||
DIRTY = optionalString (commit == "") "dirty";
|
jq
|
||||||
HASH = commit;
|
makeWrapper
|
||||||
TAG = "v${trim (readFile "${finalAttrs.src}/VERSION")}";
|
cmake
|
||||||
|
ninja
|
||||||
|
pkg-config
|
||||||
|
python3 # for udis86
|
||||||
|
# re-add after https://github.com/NixOS/nixpkgs/pull/214906 hits nixos-unstable
|
||||||
|
# wayland-scanner
|
||||||
|
];
|
||||||
|
|
||||||
depsBuildBuild = [
|
outputs = [
|
||||||
pkg-config
|
"out"
|
||||||
];
|
"man"
|
||||||
|
"dev"
|
||||||
|
];
|
||||||
|
|
||||||
nativeBuildInputs = [
|
buildInputs = lib.concatLists [
|
||||||
hyprwayland-scanner
|
[
|
||||||
makeWrapper
|
aquamarine
|
||||||
meson
|
cairo
|
||||||
ninja
|
expat
|
||||||
cmake # needed for glaze
|
fribidi
|
||||||
pkg-config
|
git
|
||||||
];
|
hwdata
|
||||||
|
hyprcursor
|
||||||
|
hyprlang
|
||||||
|
hyprutils
|
||||||
|
libdatrie
|
||||||
|
libdisplay-info
|
||||||
|
libdrm
|
||||||
|
libGL
|
||||||
|
libinput
|
||||||
|
libliftoff
|
||||||
|
libselinux
|
||||||
|
libsepol
|
||||||
|
libthai
|
||||||
|
libuuid
|
||||||
|
libxkbcommon
|
||||||
|
mesa
|
||||||
|
pango
|
||||||
|
pciutils
|
||||||
|
pcre2
|
||||||
|
seatd
|
||||||
|
tomlplusplus
|
||||||
|
wayland
|
||||||
|
wayland-protocols
|
||||||
|
xorg.libXcursor
|
||||||
|
]
|
||||||
|
(lib.optionals stdenv.hostPlatform.isMusl [libexecinfo])
|
||||||
|
(lib.optionals enableXWayland [
|
||||||
|
xorg.libxcb
|
||||||
|
xorg.libXdmcp
|
||||||
|
xorg.xcbutil
|
||||||
|
xorg.xcbutilerrors
|
||||||
|
xorg.xcbutilrenderutil
|
||||||
|
xorg.xcbutilwm
|
||||||
|
xwayland
|
||||||
|
])
|
||||||
|
(lib.optionals withSystemd [systemd])
|
||||||
|
];
|
||||||
|
|
||||||
outputs = [
|
cmakeBuildType =
|
||||||
"out"
|
if debug
|
||||||
"man"
|
then "Debug"
|
||||||
"dev"
|
else "RelWithDebInfo";
|
||||||
];
|
|
||||||
|
|
||||||
buildInputs = concatLists [
|
# we want as much debug info as possible
|
||||||
[
|
dontStrip = debug;
|
||||||
aquamarine
|
|
||||||
cairo
|
cmakeFlags = [
|
||||||
git
|
(lib.cmakeBool "NO_XWAYLAND" (!enableXWayland))
|
||||||
glaze
|
(lib.cmakeBool "LEGACY_RENDERER" legacyRenderer)
|
||||||
hyprcursor
|
(lib.cmakeBool "NO_SYSTEMD" (!withSystemd))
|
||||||
hyprgraphics
|
];
|
||||||
hyprland-protocols
|
|
||||||
hyprlang
|
postInstall = ''
|
||||||
hyprutils
|
${lib.optionalString wrapRuntimeDeps ''
|
||||||
libdrm
|
wrapProgram $out/bin/Hyprland \
|
||||||
libGL
|
--suffix PATH : ${lib.makeBinPath [
|
||||||
libinput
|
binutils
|
||||||
libuuid
|
|
||||||
libxkbcommon
|
|
||||||
libgbm
|
|
||||||
pango
|
|
||||||
pciutils
|
pciutils
|
||||||
re2
|
pkgconf
|
||||||
tomlplusplus
|
]}
|
||||||
udis86-hyprland
|
''}
|
||||||
wayland
|
'';
|
||||||
wayland-protocols
|
|
||||||
wayland-scanner
|
|
||||||
xorg.libXcursor
|
|
||||||
]
|
|
||||||
(optionals customStdenv.hostPlatform.isBSD [epoll-shim])
|
|
||||||
(optionals customStdenv.hostPlatform.isMusl [libexecinfo])
|
|
||||||
(optionals enableXWayland [
|
|
||||||
xorg.libxcb
|
|
||||||
xorg.libXdmcp
|
|
||||||
xorg.xcbutilerrors
|
|
||||||
xorg.xcbutilrenderutil
|
|
||||||
xorg.xcbutilwm
|
|
||||||
xwayland
|
|
||||||
])
|
|
||||||
(optional withSystemd systemd)
|
|
||||||
];
|
|
||||||
|
|
||||||
strictDeps = true;
|
passthru.providedSessions = ["hyprland"];
|
||||||
|
|
||||||
mesonBuildType =
|
meta = {
|
||||||
if debug
|
homepage = "https://github.com/hyprwm/Hyprland";
|
||||||
then "debug"
|
description = "Dynamic tiling Wayland compositor that doesn't sacrifice on its looks";
|
||||||
else "release";
|
license = lib.licenses.bsd3;
|
||||||
|
platforms = lib.platforms.linux;
|
||||||
mesonFlags = flatten [
|
mainProgram = "Hyprland";
|
||||||
(mapAttrsToList mesonEnable {
|
};
|
||||||
"xwayland" = enableXWayland;
|
}
|
||||||
"systemd" = withSystemd;
|
|
||||||
"uwsm" = false;
|
|
||||||
"hyprpm" = false;
|
|
||||||
})
|
|
||||||
(mapAttrsToList mesonBool {
|
|
||||||
"b_pch" = false;
|
|
||||||
"tracy_enable" = false;
|
|
||||||
})
|
|
||||||
];
|
|
||||||
|
|
||||||
postInstall = ''
|
|
||||||
${optionalString wrapRuntimeDeps ''
|
|
||||||
wrapProgram $out/bin/Hyprland \
|
|
||||||
--suffix PATH : ${makeBinPath [
|
|
||||||
binutils
|
|
||||||
hyprland-qtutils
|
|
||||||
pciutils
|
|
||||||
pkgconf
|
|
||||||
]}
|
|
||||||
''}
|
|
||||||
'';
|
|
||||||
|
|
||||||
passthru.providedSessions = ["hyprland"];
|
|
||||||
|
|
||||||
meta = {
|
|
||||||
homepage = "https://github.com/hyprwm/Hyprland";
|
|
||||||
description = "Dynamic tiling Wayland compositor that doesn't sacrifice on its looks";
|
|
||||||
license = lib.licenses.bsd3;
|
|
||||||
platforms = lib.platforms.linux;
|
|
||||||
mainProgram = "Hyprland";
|
|
||||||
};
|
|
||||||
})
|
|
||||||
|
@@ -1,64 +0,0 @@
|
|||||||
{
|
|
||||||
writeShellApplication,
|
|
||||||
deadnix,
|
|
||||||
statix,
|
|
||||||
alejandra,
|
|
||||||
llvmPackages_19,
|
|
||||||
fd,
|
|
||||||
}:
|
|
||||||
writeShellApplication {
|
|
||||||
name = "hyprland-treewide-formatter";
|
|
||||||
runtimeInputs = [
|
|
||||||
deadnix
|
|
||||||
statix
|
|
||||||
alejandra
|
|
||||||
llvmPackages_19.clang-tools
|
|
||||||
fd
|
|
||||||
];
|
|
||||||
text = ''
|
|
||||||
# shellcheck disable=SC2148
|
|
||||||
|
|
||||||
# common excludes
|
|
||||||
excludes="subprojects"
|
|
||||||
|
|
||||||
nix_format() {
|
|
||||||
if [ "$*" = 0 ]; then
|
|
||||||
fd '.*\.nix' . -E "$excludes" -x statix fix -- {} \;
|
|
||||||
fd '.*\.nix' . -E "$excludes" -X deadnix -e -- {} \; -X alejandra {} \;
|
|
||||||
elif [ -d "$1" ]; then
|
|
||||||
fd '.*\.nix' "$1" -E "$excludes" -i -x statix fix -- {} \;
|
|
||||||
fd '.*\.nix' "$1" -E "$excludes" -i -X deadnix -e -- {} \; -X alejandra {} \;
|
|
||||||
else
|
|
||||||
statix fix -- "$1"
|
|
||||||
deadnix -e "$1"
|
|
||||||
alejandra "$1"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
cpp_format() {
|
|
||||||
if [ "$*" = 0 ] || [ "$1" = "." ]; then
|
|
||||||
fd '.*\.cpp' . -E "$excludes" | xargs clang-format --verbose -i
|
|
||||||
elif [ -d "$1" ]; then
|
|
||||||
fd '.*\.cpp' "$1" -E "$excludes" | xargs clang-format --verbose -i
|
|
||||||
else
|
|
||||||
clang-format --verbose -i "$1"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
for i in "$@"; do
|
|
||||||
case ''${i##*.} in
|
|
||||||
"nix")
|
|
||||||
nix_format "$i"
|
|
||||||
;;
|
|
||||||
"cpp")
|
|
||||||
cpp_format "$i"
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
nix_format "$i"
|
|
||||||
cpp_format "$i"
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
done
|
|
||||||
'';
|
|
||||||
}
|
|
@@ -1,60 +0,0 @@
|
|||||||
{
|
|
||||||
lib,
|
|
||||||
stdenv,
|
|
||||||
stdenvAdapters,
|
|
||||||
cmake,
|
|
||||||
pkg-config,
|
|
||||||
hyprland,
|
|
||||||
hyprwayland-scanner,
|
|
||||||
version ? "git",
|
|
||||||
}: let
|
|
||||||
inherit (lib.lists) flatten foldl';
|
|
||||||
inherit (lib.sources) cleanSourceWith cleanSource;
|
|
||||||
inherit (lib.strings) hasSuffix cmakeBool;
|
|
||||||
|
|
||||||
adapters = flatten [
|
|
||||||
stdenvAdapters.useMoldLinker
|
|
||||||
stdenvAdapters.keepDebugInfo
|
|
||||||
];
|
|
||||||
|
|
||||||
customStdenv = foldl' (acc: adapter: adapter acc) stdenv adapters;
|
|
||||||
in
|
|
||||||
customStdenv.mkDerivation (finalAttrs: {
|
|
||||||
pname = "hyprtester";
|
|
||||||
inherit version;
|
|
||||||
|
|
||||||
src = cleanSourceWith {
|
|
||||||
filter = name: _type: let
|
|
||||||
baseName = baseNameOf (toString name);
|
|
||||||
in
|
|
||||||
! (hasSuffix ".nix" baseName);
|
|
||||||
src = cleanSource ../.;
|
|
||||||
};
|
|
||||||
|
|
||||||
nativeBuildInputs = [
|
|
||||||
cmake
|
|
||||||
pkg-config
|
|
||||||
hyprwayland-scanner
|
|
||||||
];
|
|
||||||
|
|
||||||
buildInputs = hyprland.buildInputs;
|
|
||||||
|
|
||||||
preConfigure = ''
|
|
||||||
cmake -S . -B .
|
|
||||||
cmake --build . --target generate-protocol-headers -j`nproc 2>/dev/null || getconf NPROCESSORS_CONF`
|
|
||||||
|
|
||||||
cd hyprtester
|
|
||||||
'';
|
|
||||||
|
|
||||||
cmakeBuildType = "Debug";
|
|
||||||
|
|
||||||
cmakeFlags = [(cmakeBool "TESTS" true)];
|
|
||||||
|
|
||||||
meta = {
|
|
||||||
homepage = "https://github.com/hyprwm/Hyprland";
|
|
||||||
description = "Hyprland testing framework";
|
|
||||||
license = lib.licenses.bsd3;
|
|
||||||
platforms = hyprland.meta.platforms;
|
|
||||||
mainProgram = "hyprtester";
|
|
||||||
};
|
|
||||||
})
|
|
201
nix/lib.nix
@@ -1,201 +0,0 @@
|
|||||||
lib: let
|
|
||||||
inherit (lib)
|
|
||||||
attrNames
|
|
||||||
filterAttrs
|
|
||||||
foldl
|
|
||||||
generators
|
|
||||||
partition
|
|
||||||
;
|
|
||||||
|
|
||||||
inherit (lib.strings)
|
|
||||||
concatMapStrings
|
|
||||||
hasPrefix
|
|
||||||
;
|
|
||||||
|
|
||||||
/**
|
|
||||||
Convert a structured Nix attribute set into Hyprland's configuration format.
|
|
||||||
|
|
||||||
This function takes a nested attribute set and converts it into Hyprland-compatible
|
|
||||||
configuration syntax, supporting top, bottom, and regular command sections.
|
|
||||||
|
|
||||||
Commands are flattened using the `flattenAttrs` function, and attributes are formatted as
|
|
||||||
`key = value` pairs. Lists are expanded as duplicate keys to match Hyprland's expected format.
|
|
||||||
|
|
||||||
Configuration:
|
|
||||||
|
|
||||||
* `topCommandsPrefixes` - A list of prefixes to define **top** commands (default: `["$"]`).
|
|
||||||
* `bottomCommandsPrefixes` - A list of prefixes to define **bottom** commands (default: `[]`).
|
|
||||||
|
|
||||||
Attention:
|
|
||||||
|
|
||||||
- The function ensures top commands appear **first** and bottom commands **last**.
|
|
||||||
- The generated configuration is a **single string**, suitable for writing to a config file.
|
|
||||||
- Lists are converted into multiple entries, ensuring compatibility with Hyprland.
|
|
||||||
|
|
||||||
# Inputs
|
|
||||||
|
|
||||||
Structured function argument:
|
|
||||||
|
|
||||||
: topCommandsPrefixes (optional, default: `["$"]`)
|
|
||||||
: A list of prefixes that define **top** commands. Any key starting with one of these
|
|
||||||
prefixes will be placed at the beginning of the configuration.
|
|
||||||
: bottomCommandsPrefixes (optional, default: `[]`)
|
|
||||||
: A list of prefixes that define **bottom** commands. Any key starting with one of these
|
|
||||||
prefixes will be placed at the end of the configuration.
|
|
||||||
|
|
||||||
Value:
|
|
||||||
|
|
||||||
: The attribute set to be converted to Hyprland configuration format.
|
|
||||||
|
|
||||||
# Type
|
|
||||||
|
|
||||||
```
|
|
||||||
toHyprlang :: AttrSet -> AttrSet -> String
|
|
||||||
```
|
|
||||||
|
|
||||||
# Examples
|
|
||||||
:::{.example}
|
|
||||||
|
|
||||||
```nix
|
|
||||||
let
|
|
||||||
config = {
|
|
||||||
"$mod" = "SUPER";
|
|
||||||
monitor = {
|
|
||||||
"HDMI-A-1" = "1920x1080@60,0x0,1";
|
|
||||||
};
|
|
||||||
exec = [
|
|
||||||
"waybar"
|
|
||||||
"dunst"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
in lib.toHyprlang {} config
|
|
||||||
```
|
|
||||||
|
|
||||||
**Output:**
|
|
||||||
```nix
|
|
||||||
"$mod = SUPER"
|
|
||||||
"monitor:HDMI-A-1 = 1920x1080@60,0x0,1"
|
|
||||||
"exec = waybar"
|
|
||||||
"exec = dunst"
|
|
||||||
```
|
|
||||||
|
|
||||||
:::
|
|
||||||
*/
|
|
||||||
toHyprlang = {
|
|
||||||
topCommandsPrefixes ? ["$"],
|
|
||||||
bottomCommandsPrefixes ? [],
|
|
||||||
}: attrs: let
|
|
||||||
toHyprlang' = attrs: let
|
|
||||||
# Specially configured `toKeyValue` generator with support for duplicate keys
|
|
||||||
# and a legible key-value separator.
|
|
||||||
mkCommands = generators.toKeyValue {
|
|
||||||
mkKeyValue = generators.mkKeyValueDefault {} " = ";
|
|
||||||
listsAsDuplicateKeys = true;
|
|
||||||
indent = ""; # No indent, since we don't have nesting
|
|
||||||
};
|
|
||||||
|
|
||||||
# Flatten the attrset, combining keys in a "path" like `"a:b:c" = "x"`.
|
|
||||||
# Uses `flattenAttrs` with a colon separator.
|
|
||||||
commands = flattenAttrs (p: k: "${p}:${k}") attrs;
|
|
||||||
|
|
||||||
# General filtering function to check if a key starts with any prefix in a given list.
|
|
||||||
filterCommands = list: n:
|
|
||||||
foldl (acc: prefix: acc || hasPrefix prefix n) false list;
|
|
||||||
|
|
||||||
# Partition keys into top commands and the rest
|
|
||||||
result = partition (filterCommands topCommandsPrefixes) (attrNames commands);
|
|
||||||
topCommands = filterAttrs (n: _: builtins.elem n result.right) commands;
|
|
||||||
remainingCommands = removeAttrs commands result.right;
|
|
||||||
|
|
||||||
# Partition remaining commands into bottom commands and regular commands
|
|
||||||
result2 = partition (filterCommands bottomCommandsPrefixes) result.wrong;
|
|
||||||
bottomCommands = filterAttrs (n: _: builtins.elem n result2.right) remainingCommands;
|
|
||||||
regularCommands = removeAttrs remainingCommands result2.right;
|
|
||||||
in
|
|
||||||
# Concatenate strings from mapping `mkCommands` over top, regular, and bottom commands.
|
|
||||||
concatMapStrings mkCommands [
|
|
||||||
topCommands
|
|
||||||
regularCommands
|
|
||||||
bottomCommands
|
|
||||||
];
|
|
||||||
in
|
|
||||||
toHyprlang' attrs;
|
|
||||||
|
|
||||||
/**
|
|
||||||
Flatten a nested attribute set into a flat attribute set, using a custom key separator function.
|
|
||||||
|
|
||||||
This function recursively traverses a nested attribute set and produces a flat attribute set
|
|
||||||
where keys are joined using a user-defined function (`pred`). It allows transforming deeply
|
|
||||||
nested structures into a single-level attribute set while preserving key-value relationships.
|
|
||||||
|
|
||||||
Configuration:
|
|
||||||
|
|
||||||
* `pred` - A function `(string -> string -> string)` defining how keys should be concatenated.
|
|
||||||
|
|
||||||
# Inputs
|
|
||||||
|
|
||||||
Structured function argument:
|
|
||||||
|
|
||||||
: pred (required)
|
|
||||||
: A function that determines how parent and child keys should be combined into a single key.
|
|
||||||
It takes a `prefix` (parent key) and `key` (current key) and returns the joined key.
|
|
||||||
|
|
||||||
Value:
|
|
||||||
|
|
||||||
: The nested attribute set to be flattened.
|
|
||||||
|
|
||||||
# Type
|
|
||||||
|
|
||||||
```
|
|
||||||
flattenAttrs :: (String -> String -> String) -> AttrSet -> AttrSet
|
|
||||||
```
|
|
||||||
|
|
||||||
# Examples
|
|
||||||
:::{.example}
|
|
||||||
|
|
||||||
```nix
|
|
||||||
let
|
|
||||||
nested = {
|
|
||||||
a = "3";
|
|
||||||
b = { c = "4"; d = "5"; };
|
|
||||||
};
|
|
||||||
|
|
||||||
separator = (prefix: key: "${prefix}.${key}"); # Use dot notation
|
|
||||||
in lib.flattenAttrs separator nested
|
|
||||||
```
|
|
||||||
|
|
||||||
**Output:**
|
|
||||||
```nix
|
|
||||||
{
|
|
||||||
"a" = "3";
|
|
||||||
"b.c" = "4";
|
|
||||||
"b.d" = "5";
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
:::
|
|
||||||
|
|
||||||
*/
|
|
||||||
flattenAttrs = pred: attrs: let
|
|
||||||
flattenAttrs' = prefix: attrs:
|
|
||||||
builtins.foldl' (
|
|
||||||
acc: key: let
|
|
||||||
value = attrs.${key};
|
|
||||||
newKey =
|
|
||||||
if prefix == ""
|
|
||||||
then key
|
|
||||||
else pred prefix key;
|
|
||||||
in
|
|
||||||
acc
|
|
||||||
// (
|
|
||||||
if builtins.isAttrs value
|
|
||||||
then flattenAttrs' newKey value
|
|
||||||
else {"${newKey}" = value;}
|
|
||||||
)
|
|
||||||
) {} (builtins.attrNames attrs);
|
|
||||||
in
|
|
||||||
flattenAttrs' "" attrs;
|
|
||||||
in
|
|
||||||
{
|
|
||||||
inherit flattenAttrs toHyprlang;
|
|
||||||
}
|
|
148
nix/module.nix
@@ -5,149 +5,17 @@ inputs: {
|
|||||||
...
|
...
|
||||||
}: let
|
}: let
|
||||||
inherit (pkgs.stdenv.hostPlatform) system;
|
inherit (pkgs.stdenv.hostPlatform) system;
|
||||||
selflib = import ./lib.nix lib;
|
|
||||||
cfg = config.programs.hyprland;
|
cfg = config.programs.hyprland;
|
||||||
|
|
||||||
|
package = inputs.self.packages.${system}.hyprland;
|
||||||
|
portalPackage = inputs.self.packages.${system}.xdg-desktop-portal-hyprland.override {
|
||||||
|
hyprland = cfg.finalPackage;
|
||||||
|
};
|
||||||
in {
|
in {
|
||||||
options = {
|
config = {
|
||||||
programs.hyprland = {
|
programs.hyprland = {
|
||||||
plugins = lib.mkOption {
|
package = lib.mkDefault package;
|
||||||
type = with lib.types; listOf (either package path);
|
portalPackage = lib.mkDefault portalPackage;
|
||||||
default = [];
|
|
||||||
description = ''
|
|
||||||
List of Hyprland plugins to use. Can either be packages or
|
|
||||||
absolute plugin paths.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
settings = lib.mkOption {
|
|
||||||
type = with lib.types; let
|
|
||||||
valueType =
|
|
||||||
nullOr (oneOf [
|
|
||||||
bool
|
|
||||||
int
|
|
||||||
float
|
|
||||||
str
|
|
||||||
path
|
|
||||||
(attrsOf valueType)
|
|
||||||
(listOf valueType)
|
|
||||||
])
|
|
||||||
// {
|
|
||||||
description = "Hyprland configuration value";
|
|
||||||
};
|
|
||||||
in
|
|
||||||
valueType;
|
|
||||||
default = {};
|
|
||||||
description = ''
|
|
||||||
Hyprland configuration written in Nix. Entries with the same key
|
|
||||||
should be written as lists. Variables' and colors' names should be
|
|
||||||
quoted. See <https://wiki.hypr.land> for more examples.
|
|
||||||
|
|
||||||
Special categories (e.g `devices`) should be written as
|
|
||||||
`"devices[device-name]"`.
|
|
||||||
|
|
||||||
::: {.note}
|
|
||||||
Use the [](#programs.hyprland.plugins) option to
|
|
||||||
declare plugins.
|
|
||||||
:::
|
|
||||||
|
|
||||||
'';
|
|
||||||
example = lib.literalExpression ''
|
|
||||||
{
|
|
||||||
decoration = {
|
|
||||||
shadow_offset = "0 5";
|
|
||||||
"col.shadow" = "rgba(00000099)";
|
|
||||||
};
|
|
||||||
|
|
||||||
"$mod" = "SUPER";
|
|
||||||
|
|
||||||
bindm = [
|
|
||||||
# mouse movements
|
|
||||||
"$mod, mouse:272, movewindow"
|
|
||||||
"$mod, mouse:273, resizewindow"
|
|
||||||
"$mod ALT, mouse:272, resizewindow"
|
|
||||||
];
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
extraConfig = lib.mkOption {
|
|
||||||
type = lib.types.lines;
|
|
||||||
default = "";
|
|
||||||
example = ''
|
|
||||||
# window resize
|
|
||||||
bind = $mod, S, submap, resize
|
|
||||||
|
|
||||||
submap = resize
|
|
||||||
binde = , right, resizeactive, 10 0
|
|
||||||
binde = , left, resizeactive, -10 0
|
|
||||||
binde = , up, resizeactive, 0 -10
|
|
||||||
binde = , down, resizeactive, 0 10
|
|
||||||
bind = , escape, submap, reset
|
|
||||||
submap = reset
|
|
||||||
'';
|
|
||||||
description = ''
|
|
||||||
Extra configuration lines to add to `/etc/xdg/hypr/hyprland.conf`.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
topPrefixes = lib.mkOption {
|
|
||||||
type = with lib.types; listOf str;
|
|
||||||
default = ["$" "bezier"];
|
|
||||||
example = ["$" "bezier" "source"];
|
|
||||||
description = ''
|
|
||||||
List of prefix of attributes to put at the top of the config.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
bottomPrefixes = lib.mkOption {
|
|
||||||
type = with lib.types; listOf str;
|
|
||||||
default = [];
|
|
||||||
example = ["source"];
|
|
||||||
description = ''
|
|
||||||
List of prefix of attributes to put at the bottom of the config.
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
config = lib.mkMerge [
|
|
||||||
{
|
|
||||||
programs.hyprland = {
|
|
||||||
package = lib.mkDefault inputs.self.packages.${system}.hyprland;
|
|
||||||
portalPackage = lib.mkDefault inputs.self.packages.${system}.xdg-desktop-portal-hyprland;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
(lib.mkIf cfg.enable {
|
|
||||||
environment.etc."xdg/hypr/hyprland.conf" = let
|
|
||||||
shouldGenerate = cfg.extraConfig != "" || cfg.settings != {} || cfg.plugins != [];
|
|
||||||
|
|
||||||
pluginsToHyprlang = plugins:
|
|
||||||
selflib.toHyprlang {
|
|
||||||
topCommandsPrefixes = cfg.topPrefixes;
|
|
||||||
bottomCommandsPrefixes = cfg.bottomPrefixes;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
"exec-once" = let
|
|
||||||
mkEntry = entry:
|
|
||||||
if lib.types.package.check entry
|
|
||||||
then "${entry}/lib/lib${entry.pname}.so"
|
|
||||||
else entry;
|
|
||||||
hyprctl = lib.getExe' config.programs.hyprland.package "hyprctl";
|
|
||||||
in
|
|
||||||
map (p: "${hyprctl} plugin load ${mkEntry p}") cfg.plugins;
|
|
||||||
};
|
|
||||||
in
|
|
||||||
lib.mkIf shouldGenerate {
|
|
||||||
text =
|
|
||||||
lib.optionalString (cfg.plugins != [])
|
|
||||||
(pluginsToHyprlang cfg.plugins)
|
|
||||||
+ lib.optionalString (cfg.settings != {})
|
|
||||||
(selflib.toHyprlang {
|
|
||||||
topCommandsPrefixes = cfg.topPrefixes;
|
|
||||||
bottomCommandsPrefixes = cfg.bottomPrefixes;
|
|
||||||
}
|
|
||||||
cfg.settings)
|
|
||||||
+ lib.optionalString (cfg.extraConfig != "") cfg.extraConfig;
|
|
||||||
};
|
|
||||||
})
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,7 @@
|
|||||||
(builtins.substring 4 2 longDate)
|
(builtins.substring 4 2 longDate)
|
||||||
(builtins.substring 6 2 longDate)
|
(builtins.substring 6 2 longDate)
|
||||||
]);
|
]);
|
||||||
ver = lib.removeSuffix "\n" (builtins.readFile ../VERSION);
|
version = lib.removeSuffix "\n" (builtins.readFile ../VERSION);
|
||||||
in {
|
in {
|
||||||
# Contains what a user is most likely to care about:
|
# Contains what a user is most likely to care about:
|
||||||
# Hyprland itself, XDPH and the Share Picker.
|
# Hyprland itself, XDPH and the Share Picker.
|
||||||
@@ -22,39 +22,25 @@ in {
|
|||||||
# Dependencies
|
# Dependencies
|
||||||
inputs.aquamarine.overlays.default
|
inputs.aquamarine.overlays.default
|
||||||
inputs.hyprcursor.overlays.default
|
inputs.hyprcursor.overlays.default
|
||||||
inputs.hyprgraphics.overlays.default
|
|
||||||
inputs.hyprland-protocols.overlays.default
|
|
||||||
inputs.hyprland-qtutils.overlays.default
|
|
||||||
inputs.hyprlang.overlays.default
|
inputs.hyprlang.overlays.default
|
||||||
inputs.hyprutils.overlays.default
|
inputs.hyprutils.overlays.default
|
||||||
inputs.hyprwayland-scanner.overlays.default
|
inputs.hyprwayland-scanner.overlays.default
|
||||||
self.overlays.udis86
|
|
||||||
|
|
||||||
# Hyprland packages themselves
|
# Hyprland packages themselves
|
||||||
(final: _prev: let
|
(final: prev: let
|
||||||
date = mkDate (self.lastModifiedDate or "19700101");
|
date = mkDate (self.lastModifiedDate or "19700101");
|
||||||
version = "${ver}+date=${date}_${self.shortRev or "dirty"}";
|
|
||||||
in {
|
in {
|
||||||
hyprland = final.callPackage ./default.nix {
|
hyprland = final.callPackage ./default.nix {
|
||||||
stdenv = final.gcc15Stdenv;
|
stdenv = final.gcc14Stdenv;
|
||||||
|
version = "${version}+date=${date}_${self.shortRev or "dirty"}";
|
||||||
commit = self.rev or "";
|
commit = self.rev or "";
|
||||||
revCount = self.sourceInfo.revCount or "";
|
inherit date;
|
||||||
inherit date version;
|
|
||||||
};
|
};
|
||||||
hyprland-unwrapped = final.hyprland.override {wrapRuntimeDeps = false;};
|
hyprland-unwrapped = final.hyprland.override {wrapRuntimeDeps = false;};
|
||||||
|
hyprland-debug = final.hyprland.override {debug = true;};
|
||||||
hyprtester = final.callPackage ./hyprtester.nix {
|
hyprland-legacy-renderer = final.hyprland.override {legacyRenderer = true;};
|
||||||
inherit version;
|
|
||||||
};
|
|
||||||
|
|
||||||
# deprecated packages
|
# deprecated packages
|
||||||
hyprland-legacy-renderer =
|
|
||||||
builtins.trace ''
|
|
||||||
hyprland-legacy-renderer was removed. Please use the hyprland package.
|
|
||||||
Legacy renderer is no longer supported.
|
|
||||||
''
|
|
||||||
final.hyprland;
|
|
||||||
|
|
||||||
hyprland-nvidia =
|
hyprland-nvidia =
|
||||||
builtins.trace ''
|
builtins.trace ''
|
||||||
hyprland-nvidia was removed. Please use the hyprland package.
|
hyprland-nvidia was removed. Please use the hyprland package.
|
||||||
@@ -65,43 +51,15 @@ in {
|
|||||||
hyprland-hidpi =
|
hyprland-hidpi =
|
||||||
builtins.trace ''
|
builtins.trace ''
|
||||||
hyprland-hidpi was removed. Please use the hyprland package.
|
hyprland-hidpi was removed. Please use the hyprland package.
|
||||||
For more information, refer to https://wiki.hypr.land/Configuring/XWayland.
|
For more information, refer to https://wiki.hyprland.org/Configuring/XWayland.
|
||||||
''
|
''
|
||||||
final.hyprland;
|
final.hyprland;
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
|
|
||||||
# Debug
|
|
||||||
hyprland-debug = lib.composeManyExtensions [
|
|
||||||
# Dependencies
|
|
||||||
self.overlays.hyprland-packages
|
|
||||||
|
|
||||||
(final: prev: {
|
|
||||||
aquamarine = prev.aquamarine.override {debug = true;};
|
|
||||||
hyprutils = prev.hyprutils.override {debug = true;};
|
|
||||||
hyprland-debug = prev.hyprland.override {debug = true;};
|
|
||||||
})
|
|
||||||
];
|
|
||||||
|
|
||||||
# Packages for extra software recommended for usage with Hyprland,
|
# Packages for extra software recommended for usage with Hyprland,
|
||||||
# including forked or patched packages for compatibility.
|
# including forked or patched packages for compatibility.
|
||||||
hyprland-extras = lib.composeManyExtensions [
|
hyprland-extras = lib.composeManyExtensions [
|
||||||
inputs.xdph.overlays.default
|
inputs.xdph.overlays.xdg-desktop-portal-hyprland
|
||||||
];
|
];
|
||||||
|
|
||||||
# udis86 from nixpkgs is too old, and also does not provide a .pc file
|
|
||||||
# this version is the one used in the git submodule, and allows us to
|
|
||||||
# fetch the source without '?submodules=1'
|
|
||||||
udis86 = final: prev: {
|
|
||||||
udis86-hyprland = prev.udis86.overrideAttrs (_self: _super: {
|
|
||||||
src = final.fetchFromGitHub {
|
|
||||||
owner = "canihavesomecoffee";
|
|
||||||
repo = "udis86";
|
|
||||||
rev = "5336633af70f3917760a6d441ff02d93477b0c86";
|
|
||||||
hash = "sha256-HifdUQPGsKQKQprByeIznvRLONdOXeolOsU5nkwIv3g=";
|
|
||||||
};
|
|
||||||
|
|
||||||
patches = [];
|
|
||||||
});
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
12
nix/stdcxx.patch
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
diff --git a/CMakeLists.txt b/CMakeLists.txt
|
||||||
|
index cfbd431f..73e8e0c2 100644
|
||||||
|
--- a/CMakeLists.txt
|
||||||
|
+++ b/CMakeLists.txt
|
||||||
|
@@ -64,6 +64,7 @@ endif()
|
||||||
|
include_directories(. "src/" "subprojects/udis86/" "protocols/")
|
||||||
|
set(CMAKE_CXX_STANDARD 26)
|
||||||
|
add_compile_options(
|
||||||
|
+ -std=c++26
|
||||||
|
-Wall
|
||||||
|
-Wextra
|
||||||
|
-Wno-unused-parameter
|