mirror of
https://github.com/sysadminsmedia/homebox.git
synced 2025-12-21 13:23:14 +01:00
Compare commits
13 Commits
v0.22.0-rc
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e4d8bb2ada | ||
|
|
3becf046e6 | ||
|
|
a21b3257d4 | ||
|
|
5f9ab577bb | ||
|
|
0a969bb64d | ||
|
|
2d1d3d927b | ||
|
|
540028a22e | ||
|
|
14b0d51894 | ||
|
|
4334f926c0 | ||
|
|
1088972ff0 | ||
|
|
55e247ac71 | ||
|
|
05a2700718 | ||
|
|
06c11cdcd5 |
@@ -243,6 +243,6 @@ jobs:
|
|||||||
uses: actions/attest-build-provenance@v1
|
uses: actions/attest-build-provenance@v1
|
||||||
if: (github.event_name == 'schedule' || startsWith(github.ref, 'refs/tags/'))
|
if: (github.event_name == 'schedule' || startsWith(github.ref, 'refs/tags/'))
|
||||||
with:
|
with:
|
||||||
subject-name: ${{ env.DOCKERHUB_REPO }}
|
subject-name: docker.io/${{ env.DOCKERHUB_REPO }}
|
||||||
subject-digest: ${{ steps.push-dockerhub.outputs.digest }}
|
subject-digest: ${{ steps.push-dockerhub.outputs.digest }}
|
||||||
push-to-registry: true
|
push-to-registry: true
|
||||||
|
|||||||
@@ -245,6 +245,6 @@ jobs:
|
|||||||
uses: actions/attest-build-provenance@v1
|
uses: actions/attest-build-provenance@v1
|
||||||
if: (github.event_name == 'schedule' || startsWith(github.ref, 'refs/tags/'))
|
if: (github.event_name == 'schedule' || startsWith(github.ref, 'refs/tags/'))
|
||||||
with:
|
with:
|
||||||
subject-name: ${{ env.DOCKERHUB_REPO }}
|
subject-name: docker.io/${{ env.DOCKERHUB_REPO }}
|
||||||
subject-digest: ${{ steps.push-dockerhub.outputs.digest }}
|
subject-digest: ${{ steps.push-dockerhub.outputs.digest }}
|
||||||
push-to-registry: true
|
push-to-registry: true
|
||||||
|
|||||||
2
.github/workflows/docker-publish.yaml
vendored
2
.github/workflows/docker-publish.yaml
vendored
@@ -236,6 +236,6 @@ jobs:
|
|||||||
uses: actions/attest-build-provenance@v1
|
uses: actions/attest-build-provenance@v1
|
||||||
if: (github.event_name == 'schedule' || startsWith(github.ref, 'refs/tags/'))
|
if: (github.event_name == 'schedule' || startsWith(github.ref, 'refs/tags/'))
|
||||||
with:
|
with:
|
||||||
subject-name: ${{ env.DOCKERHUB_REPO }}
|
subject-name: docker.io/${{ env.DOCKERHUB_REPO }}
|
||||||
subject-digest: ${{ steps.push-dockerhub.outputs.digest }}
|
subject-digest: ${{ steps.push-dockerhub.outputs.digest }}
|
||||||
push-to-registry: true
|
push-to-registry: true
|
||||||
|
|||||||
@@ -0,0 +1,2 @@
|
|||||||
|
-- +goose Up
|
||||||
|
ALTER TABLE users ALTER COLUMN password DROP NOT NULL;
|
||||||
@@ -22,7 +22,7 @@ aside: false
|
|||||||
| HBOX_WEB_IDLE_TIMEOUT | 30s | Idle timeout of HTTP server |
|
| HBOX_WEB_IDLE_TIMEOUT | 30s | Idle timeout of HTTP server |
|
||||||
| HBOX_STORAGE_CONN_STRING | file:///./ | path to the data directory, do not change this if you're using docker |
|
| HBOX_STORAGE_CONN_STRING | file:///./ | path to the data directory, do not change this if you're using docker |
|
||||||
| HBOX_STORAGE_PREFIX_PATH | .data | prefix path for the storage, if not set the storage will be used as is |
|
| HBOX_STORAGE_PREFIX_PATH | .data | prefix path for the storage, if not set the storage will be used as is |
|
||||||
| HBOX_LOG_LEVEL | `info` | log level to use, can be one of `trace`, `debug`, `info`, `warn`, `error`, `critical` |
|
| HBOX_LOG_LEVEL | `info` | log level to use, can be one of `trace`, `debug`, `info`, `warn`, `error`, `fatal`, `panic` |
|
||||||
| HBOX_LOG_FORMAT | `text` | log format to use, can be one of: `text`, `json` |
|
| HBOX_LOG_FORMAT | `text` | log format to use, can be one of: `text`, `json` |
|
||||||
| HBOX_MAILER_HOST | | email host to use, if not set no email provider will be used |
|
| HBOX_MAILER_HOST | | email host to use, if not set no email provider will be used |
|
||||||
| HBOX_MAILER_PORT | 587 | email port to use |
|
| HBOX_MAILER_PORT | 587 | email port to use |
|
||||||
@@ -71,6 +71,7 @@ aside: false
|
|||||||
| HBOX_THUMBNAIL_ENABLED | true | enable thumbnail generation for images, supports PNG, JPEG, AVIF, WEBP, GIF file types |
|
| HBOX_THUMBNAIL_ENABLED | true | enable thumbnail generation for images, supports PNG, JPEG, AVIF, WEBP, GIF file types |
|
||||||
| HBOX_THUMBNAIL_WIDTH | 500 | width for generated thumbnails in pixels |
|
| HBOX_THUMBNAIL_WIDTH | 500 | width for generated thumbnails in pixels |
|
||||||
| HBOX_THUMBNAIL_HEIGHT | 500 | height for generated thumbnails in pixels |
|
| HBOX_THUMBNAIL_HEIGHT | 500 | height for generated thumbnails in pixels |
|
||||||
|
| HBOX_BARCODE_TOKEN_BARCODESPIDER | | API token for BarcodeSpider.com service used for barcode product lookups. If not set, barcode product lookups will not be performed. |
|
||||||
|
|
||||||
### HBOX_WEB_HOST examples
|
### HBOX_WEB_HOST examples
|
||||||
|
|
||||||
@@ -180,7 +181,7 @@ HomeBox supports OpenID Connect (OIDC) authentication, allowing users to login u
|
|||||||
- `HBOX_OIDC_CLIENT_SECRET`: Client secret from your OIDC provider
|
- `HBOX_OIDC_CLIENT_SECRET`: Client secret from your OIDC provider
|
||||||
|
|
||||||
3. **Configure Redirect URI**: In your OIDC provider, set the redirect URI to:
|
3. **Configure Redirect URI**: In your OIDC provider, set the redirect URI to:
|
||||||
`https://your-homebox-domain.com/api/v1/users/login/oidc/callback`
|
`https://your-homebox-domain.example.com/api/v1/users/login/oidc/callback`
|
||||||
|
|
||||||
### Advanced OIDC Configuration
|
### Advanced OIDC Configuration
|
||||||
|
|
||||||
@@ -204,70 +205,79 @@ If you're deploying without docker you can use command line arguments to configu
|
|||||||
for more information.
|
for more information.
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
Usage: api [options] [arguments]
|
Options:
|
||||||
|
--barcode-token-barcodespider <string>
|
||||||
OPTIONS
|
--database-database <string>
|
||||||
--mode/$HBOX_MODE <string> (default: development)
|
--database-driver <string> (default: sqlite3)
|
||||||
--web-port/$HBOX_WEB_PORT <string> (default: 7745)
|
--database-host <string>
|
||||||
--web-host/$HBOX_WEB_HOST <string>
|
--database-password <string>
|
||||||
--web-max-upload-size/$HBOX_WEB_MAX_UPLOAD_SIZE <int> (default: 10)
|
--database-port <string>
|
||||||
--storage-conn-string/$HBOX_STORAGE_CONN_STRING <string> (default: file:///./)
|
--database-pub-sub-conn-string <string> (default: mem://{{ .Topic }})
|
||||||
--storage-prefix-path/$HBOX_STORAGE_PREFIX_PATH <string> (default: .data)
|
--database-sqlite-path <string> (default: ./.data/homebox.db?_pragma=busy_timeout=999&_pragma=journal_mode=WAL&_fk=1&_time_format=sqlite)
|
||||||
--log-level/$HBOX_LOG_LEVEL <string> (default: info)
|
--database-ssl-cert <string>
|
||||||
--log-format/$HBOX_LOG_FORMAT <string> (default: text)
|
--database-ssl-key <string>
|
||||||
--mailer-host/$HBOX_MAILER_HOST <string>
|
--database-ssl-mode <string> (default: require)
|
||||||
--mailer-port/$HBOX_MAILER_PORT <int>
|
--database-ssl-root-cert <string>
|
||||||
--mailer-username/$HBOX_MAILER_USERNAME <string>
|
--database-username <string>
|
||||||
--mailer-password/$HBOX_MAILER_PASSWORD <string>
|
--debug-enabled <bool> (default: false)
|
||||||
--mailer-from/$HBOX_MAILER_FROM <string>
|
--debug-port <string> (default: 4000)
|
||||||
--demo/$HBOX_DEMO <bool>
|
--demo <bool>
|
||||||
--debug-enabled/$HBOX_DEBUG_ENABLED <bool> (default: false)
|
-h, --help display this help message
|
||||||
--debug-port/$HBOX_DEBUG_PORT <string> (default: 4000)
|
--label-maker-additional-information <string>
|
||||||
--database-driver/$HBOX_DATABASE_DRIVER <string> (default: sqlite3)
|
--label-maker-bold-font-path <string>
|
||||||
--database-sqlite-path/$HBOX_DATABASE_SQLITE_PATH <string> (default: ./.data/homebox.db?_pragma=busy_timeout=999&_pragma=journal_mode=WAL&_fk=1&_time_format=sqlite)
|
--label-maker-dynamic-length <bool> (default: true)
|
||||||
--database-host/$HBOX_DATABASE_HOST <string>
|
--label-maker-font-size <float> (default: 32.0)
|
||||||
--database-port/$HBOX_DATABASE_PORT <string>
|
--label-maker-height <int> (default: 200)
|
||||||
--database-username/$HBOX_DATABASE_USERNAME <string>
|
--label-maker-label-service-timeout <int>
|
||||||
--database-password/$HBOX_DATABASE_PASSWORD <string>
|
--label-maker-label-service-url <string>
|
||||||
--database-database/$HBOX_DATABASE_DATABASE <string>
|
--label-maker-margin <int> (default: 32)
|
||||||
--database-ssl-mode/$HBOX_DATABASE_SSL_MODE <string> (default: prefer)
|
--label-maker-padding <int> (default: 32)
|
||||||
--options-allow-registration/$HBOX_OPTIONS_ALLOW_REGISTRATION <bool> (default: true)
|
--label-maker-print-command <string>
|
||||||
--options-auto-increment-asset-id/$HBOX_OPTIONS_AUTO_INCREMENT_ASSET_ID <bool> (default: true)
|
--label-maker-regular-font-path <string>
|
||||||
--options-currency-config/$HBOX_OPTIONS_CURRENCY_CONFIG <string>
|
--label-maker-width <int> (default: 526)
|
||||||
--options-github-release-check/$HBOX_OPTIONS_GITHUB_RELEASE_CHECK <bool> (default: true)
|
--log-format <string> (default: text)
|
||||||
--options-allow-analytics/$HBOX_OPTIONS_ALLOW_ANALYTICS <bool> (default: false)
|
--log-level <string> (default: info)
|
||||||
--options-allow-local-login/$HBOX_OPTIONS_ALLOW_LOCAL_LOGIN <bool> (default: true)
|
--mailer-from <string>
|
||||||
--options-trust-proxy/$HBOX_OPTIONS_TRUST_PROXY <bool> (default: false)
|
--mailer-host <string>
|
||||||
--options-hostname/$HBOX_OPTIONS_HOSTNAME <string>
|
--mailer-password <string>
|
||||||
--oidc-enabled/$HBOX_OIDC_ENABLED <bool> (default: false)
|
--mailer-port <int>
|
||||||
--oidc-issuer-url/$HBOX_OIDC_ISSUER_URL <string>
|
--mailer-username <string>
|
||||||
--oidc-client-id/$HBOX_OIDC_CLIENT_ID <string>
|
--mode <string> (default: development)
|
||||||
--oidc-client-secret/$HBOX_OIDC_CLIENT_SECRET <string>
|
--oidc-allowed-groups <string>
|
||||||
--oidc-scope/$HBOX_OIDC_SCOPE <string> (default: openid profile email)
|
--oidc-auto-redirect <bool> (default: false)
|
||||||
--oidc-allowed-groups/$HBOX_OIDC_ALLOWED_GROUPS <string>
|
--oidc-button-text <string> (default: Sign in with OIDC)
|
||||||
--oidc-auto-redirect/$HBOX_OIDC_AUTO_REDIRECT <bool> (default: false)
|
--oidc-client-id <string>
|
||||||
--oidc-verify-email/$HBOX_OIDC_VERIFY_EMAIL <bool> (default: false)
|
--oidc-client-secret <string>
|
||||||
--oidc-group-claim/$HBOX_OIDC_GROUP_CLAIM <string> (default: groups)
|
--oidc-email-claim <string> (default: email)
|
||||||
--oidc-email-claim/$HBOX_OIDC_EMAIL_CLAIM <string> (default: email)
|
--oidc-email-verified-claim <string> (default: email_verified)
|
||||||
--oidc-name-claim/$HBOX_OIDC_NAME_CLAIM <string> (default: name)
|
--oidc-enabled <bool> (default: false)
|
||||||
--oidc-email-verified-claim/$HBOX_OIDC_EMAIL_VERIFIED_CLAIM <string> (default: email_verified)
|
--oidc-group-claim <string> (default: groups)
|
||||||
--oidc-button-text/$HBOX_OIDC_BUTTON_TEXT <string> (default: Sign in with OIDC)
|
--oidc-issuer-url <string>
|
||||||
--oidc-state-expiry/$HBOX_OIDC_STATE_EXPIRY <duration> (default: 10m)
|
--oidc-name-claim <string> (default: name)
|
||||||
--oidc-request-timeout/$HBOX_OIDC_REQUEST_TIMEOUT <duration> (default: 30s)
|
--oidc-request-timeout <duration> (default: 30s)
|
||||||
--label-maker-width/$HBOX_LABEL_MAKER_WIDTH <int> (default: 526)
|
--oidc-scope <string> (default: openid profile email)
|
||||||
--label-maker-height/$HBOX_LABEL_MAKER_HEIGHT <int> (default: 200)
|
--oidc-state-expiry <duration> (default: 10m)
|
||||||
--label-maker-padding/$HBOX_LABEL_MAKER_PADDING <int> (default: 32)
|
--oidc-verify-email <bool> (default: false)
|
||||||
--label-maker-margin/$HBOX_LABEL_MAKER_MARGIN <int> (default: 32)
|
--options-allow-analytics <bool> (default: false)
|
||||||
--label-maker-font-size/$HBOX_LABEL_MAKER_FONT_SIZE <float> (default: 32.0)
|
--options-allow-local-login <bool> (default: true)
|
||||||
--label-maker-print-command/$HBOX_LABEL_MAKER_PRINT_COMMAND <string>
|
--options-allow-registration <bool> (default: true)
|
||||||
--label-maker-dynamic-length/$HBOX_LABEL_MAKER_DYNAMIC_LENGTH <bool> (default: true)
|
--options-auto-increment-asset-id <bool> (default: true)
|
||||||
--label-maker-additional-information/$HBOX_LABEL_MAKER_ADDITIONAL_INFORMATION <string>
|
--options-currency-config <string>
|
||||||
--label-maker-regular-font-path/$HBOX_LABEL_MAKER_REGULAR_FONT_PATH <string>
|
--options-github-release-check <bool> (default: true)
|
||||||
--label-maker-bold-font-path/$HBOX_LABEL_MAKER_BOLD_FONT_PATH <string>
|
--options-hostname <string>
|
||||||
--thumbnail-enabled/$HBOX_THUMBNAIL_ENABLED <bool> (default: true)
|
--options-trust-proxy <bool> (default: false)
|
||||||
--thumbnail-width/$HBOX_THUMBNAIL_WIDTH <int> (default: 500)
|
--storage-conn-string <string> (default: file:///./)
|
||||||
--thumbnail-height/$HBOX_THUMBNAIL_HEIGHT <int> (default: 500)
|
--storage-prefix-path <string> (default: .data)
|
||||||
--help/-h display this help message
|
--thumbnail-enabled <bool> (default: true)
|
||||||
|
--thumbnail-height <int> (default: 500)
|
||||||
|
--thumbnail-width <int> (default: 500)
|
||||||
|
-v, --version display version
|
||||||
|
--web-host <string>
|
||||||
|
--web-idle-timeout <duration> (default: 30s)
|
||||||
|
--web-max-upload-size <int> (default: 10)
|
||||||
|
--web-port <string> (default: 7745)
|
||||||
|
--web-read-timeout <duration> (default: 10s)
|
||||||
|
--web-write-timeout <duration> (default: 10s)
|
||||||
```
|
```
|
||||||
|
|
||||||
:::
|
:::
|
||||||
|
|||||||
@@ -30,19 +30,19 @@ the bucket name in the connection string.
|
|||||||
### S3-Compatible Storage
|
### S3-Compatible Storage
|
||||||
|
|
||||||
You can also use S3-compatible storage by setting the `HBOX_STORAGE_CONN_STRING` to
|
You can also use S3-compatible storage by setting the `HBOX_STORAGE_CONN_STRING` to
|
||||||
`s3://my-bucket?awssdk=v2&endpoint=http://my-s3-compatible-endpoint.tld&disableSSL=true&s3ForcePathStyle=true`.
|
`s3://my-bucket?awssdk=v2&endpoint=http://my-s3-compatible-endpoint.tld&disable_https=true&s3ForcePathStyle=true`.
|
||||||
|
|
||||||
This allows you to connect to S3-compatible services like MinIO, DigitalOcean Spaces, or any other service that supports
|
This allows you to connect to S3-compatible services like MinIO, DigitalOcean Spaces, or any other service that supports
|
||||||
the S3 API. Configure the `disableSSL`, `s3ForcePathStyle`, and `endpoint` parameters as needed for your specific
|
the S3 API. Configure the `disable_https`, `s3ForcePathStyle`, and `endpoint` parameters as needed for your specific
|
||||||
service.
|
service.
|
||||||
|
|
||||||
#### Tested S3-Compatible Storage
|
#### Tested S3-Compatible Storage
|
||||||
|
|
||||||
| Service | Working | Connection String |
|
| Service | Working | Connection String |
|
||||||
|---------------------|---------|--------------------------------------------------------------------------------------------------------------------------|
|
|---------------------|---------|--------------------------------------------------------------------------------------------------------------------------|
|
||||||
| MinIO | Yes | `s3://my-bucket?awssdk=v2&endpoint=http://minio:9000&disableSSL=true&s3ForcePathStyle=true` |
|
| MinIO | Yes | `s3://my-bucket?awssdk=v2&endpoint=http://minio:9000&disable_https=true&s3ForcePathStyle=true` |
|
||||||
| Cloudflare R2 | Yes | `s3://my-bucket?awssdk=v2&endpoint=https://<account-id>.r2.cloudflarestorage.com&disableSSL=false&s3ForcePathStyle=true` |
|
| Cloudflare R2 | Yes | `s3://my-bucket?awssdk=v2&endpoint=https://<account-id>.r2.cloudflarestorage.com&disable_https=false&s3ForcePathStyle=true` |
|
||||||
| Backblaze B2 | Yes | `s3://my-bucket?awssdk=v2&endpoint=https://s3.us-west-004.backblazeb2.com&disableSSL=false&s3ForcePathStyle=true` |
|
| Backblaze B2 | Yes | `s3://my-bucket?awssdk=v2&endpoint=https://s3.us-west-004.backblazeb2.com&disable_https=false&s3ForcePathStyle=true` |
|
||||||
|
|
||||||
::: info
|
::: info
|
||||||
If you know of any other S3-compatible storage that works with Homebox, please let us know or create a pull request to update the table.
|
If you know of any other S3-compatible storage that works with Homebox, please let us know or create a pull request to update the table.
|
||||||
@@ -57,7 +57,7 @@ Additionally, the parameters in the URL can be used to configure specific S3 set
|
|||||||
features.)
|
features.)
|
||||||
- `endpoint`: The custom endpoint for S3-compatible storage services.
|
- `endpoint`: The custom endpoint for S3-compatible storage services.
|
||||||
- `s3ForcePathStyle`: Whether to force path-style access (set to `true` or `false`).
|
- `s3ForcePathStyle`: Whether to force path-style access (set to `true` or `false`).
|
||||||
- `disableSSL`: Whether to disable SSL (set to `true` or `false`).
|
- `disable_https`: Whether to disable SSL (set to `true` or `false`).
|
||||||
- `sseType`: The server-side encryption type (e.g., `AES256` or `aws:kms` or `aws:kms:dsse`).
|
- `sseType`: The server-side encryption type (e.g., `AES256` or `aws:kms` or `aws:kms:dsse`).
|
||||||
- `kmskeyid`: The KMS key ID for server-side encryption.
|
- `kmskeyid`: The KMS key ID for server-side encryption.
|
||||||
- `fips`: Whether to use FIPS endpoints (set to `true` or `false`).
|
- `fips`: Whether to use FIPS endpoints (set to `true` or `false`).
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ services:
|
|||||||
environment:
|
environment:
|
||||||
- HBOX_LOG_LEVEL=info
|
- HBOX_LOG_LEVEL=info
|
||||||
- HBOX_LOG_FORMAT=text
|
- HBOX_LOG_FORMAT=text
|
||||||
- HBOX_WEB_MAX_FILE_UPLOAD=10
|
- HBOX_WEB_MAX_UPLOAD_SIZE=10
|
||||||
# Please consider allowing analytics to help us improve Homebox (basic computer information, no personal data)
|
# Please consider allowing analytics to help us improve Homebox (basic computer information, no personal data)
|
||||||
- HBOX_OPTIONS_ALLOW_ANALYTICS=false
|
- HBOX_OPTIONS_ALLOW_ANALYTICS=false
|
||||||
volumes:
|
volumes:
|
||||||
|
|||||||
@@ -81,17 +81,6 @@
|
|||||||
errorMessage.value = t("scanner.error");
|
errorMessage.value = t("scanner.error");
|
||||||
};
|
};
|
||||||
|
|
||||||
const checkPermissionsError = async () => {
|
|
||||||
if (navigator.permissions) {
|
|
||||||
const permissionStatus = await navigator.permissions.query({ name: "camera" as PermissionName });
|
|
||||||
if (permissionStatus.state === "denied") {
|
|
||||||
errorMessage.value = t("scanner.permission_denied");
|
|
||||||
console.error("Camera permission denied");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleButtonClick = () => {
|
const handleButtonClick = () => {
|
||||||
openDialog(DialogID.ProductImport, { params: { barcode: detectedBarcode.value } });
|
openDialog(DialogID.ProductImport, { params: { barcode: detectedBarcode.value } });
|
||||||
};
|
};
|
||||||
@@ -103,11 +92,19 @@
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (await checkPermissionsError()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
// Request camera permission first
|
||||||
|
try {
|
||||||
|
const stream = await navigator.mediaDevices.getUserMedia({ video: true });
|
||||||
|
stream.getTracks().forEach(track => track.stop());
|
||||||
|
} catch (err: unknown) {
|
||||||
|
if (err instanceof Error && err.name === "NotAllowedError") {
|
||||||
|
errorMessage.value = t("scanner.permission_denied");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
const devices = await codeReader.listVideoInputDevices();
|
const devices = await codeReader.listVideoInputDevices();
|
||||||
sources.value = devices;
|
sources.value = devices;
|
||||||
|
|
||||||
|
|||||||
@@ -47,7 +47,7 @@
|
|||||||
{{ btn.name.value }}
|
{{ btn.name.value }}
|
||||||
<Shortcut
|
<Shortcut
|
||||||
v-if="btn.shortcut"
|
v-if="btn.shortcut"
|
||||||
class="ml-auto hidden group-hover:inline"
|
class="invisible ml-auto group-hover:visible"
|
||||||
:keys="btn.shortcut.replace('Shift', '⇧').split('+')"
|
:keys="btn.shortcut.replace('Shift', '⇧').split('+')"
|
||||||
/>
|
/>
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
|
|||||||
Reference in New Issue
Block a user