Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e5bbc5ae19 | |||
|
def2127ed6
|
|||
| 2b58cc8a13 | |||
|
e0967a8388
|
|||
|
8d94bc4006
|
|||
| 10014d5ee9 | |||
|
8333d249d9
|
|||
| 383d547a32 | |||
|
1d92dc911e
|
|||
| 24747230e1 | |||
|
867b8ccf26
|
|||
|
7c882bb26c
|
|||
| f2be6c151e | |||
|
b78becad5f
|
@@ -0,0 +1,42 @@
|
|||||||
|
name: Auto Label PRs
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
types: [opened, edited]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
label:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Apply labels based on PR title
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
PR_TITLE: ${{ github.event.pull_request.title }}
|
||||||
|
PR_NUMBER: ${{ github.event.pull_request.number }}
|
||||||
|
REPO_OWNER: ${{ github.repository_owner }}
|
||||||
|
REPO_NAME: ${{ github.event.repository.name }}
|
||||||
|
API_URL: ${{ github.api_url }}
|
||||||
|
run: |
|
||||||
|
shopt -s nocasematch
|
||||||
|
|
||||||
|
labels="["
|
||||||
|
|
||||||
|
if [[ "$PR_TITLE" =~ ^(ci|test)(/|:) ]]; then
|
||||||
|
labels="${labels}13," # 13 is Kind/Testing
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$PR_TITLE" =~ ^docs(/|:) ]]; then
|
||||||
|
labels="${labels}14," # 14 is Kind/Documentation
|
||||||
|
fi
|
||||||
|
|
||||||
|
labels="${labels%,}]"
|
||||||
|
|
||||||
|
if [ "$labels" != "[]" ]; then
|
||||||
|
echo "Adding labels to PR #${PR_NUMBER}: ${labels}"
|
||||||
|
curl -X POST "${API_URL}/repos/${REPO_OWNER}/${REPO_NAME}/issues/${PR_NUMBER}/labels" \
|
||||||
|
-H "Authorization: token ${GITHUB_TOKEN}" \
|
||||||
|
-H "Content-Type: application/json" \
|
||||||
|
-d "{\"labels\": ${labels}}"
|
||||||
|
else
|
||||||
|
echo "No matching prefixes found in title: ${PR_TITLE}"
|
||||||
|
fi
|
||||||
@@ -10,33 +10,62 @@ A feature-rich, portable CLI color utility for Linux, specializing in color pick
|
|||||||
- **Smart JSON Output:** Generate machine-readable JSON tables for easy integration with other tools.
|
- **Smart JSON Output:** Generate machine-readable JSON tables for easy integration with other tools.
|
||||||
- **Color Naming:** Resolve hex values to human-readable names via thecolorapi.com.
|
- **Color Naming:** Resolve hex values to human-readable names via thecolorapi.com.
|
||||||
- **Visual Previews:** Render color swatches directly in your terminal using truecolor ANSI escapes.
|
- **Visual Previews:** Render color swatches directly in your terminal using truecolor ANSI escapes.
|
||||||
- **Desktop Integration:** Built-in support for clipboard copying (`wl-copy`/`xclip`) and desktop notifications.
|
- **Desktop Integration:** Built-in support for clipboard copying (`wl-copy`/`xclip`) and rich desktop notifications with visual color swatches.
|
||||||
- **Highly Configurable:** Separate configuration profiles for standard terminal usage and desktop/launcher mode.
|
- **Highly Configurable:** Separate configuration profiles for standard terminal usage and desktop/launcher mode.
|
||||||
- **Portable Design:** The entire tool is contained within a single Bash script, with embedded Python helpers.
|
- **Portable Design:** The entire tool is contained within a single Bash script, with embedded Python helpers.
|
||||||
|
|
||||||
## 📋 Prerequisites
|
## 📋 Prerequisites
|
||||||
|
|
||||||
To utilize all features, ensure the following are installed:
|
### Required Core Dependencies
|
||||||
- **KDE Plasma + Wayland:** Required for the `--pick` functionality.
|
These are mandatory for the script to run and perform basic color conversions:
|
||||||
- **Python 3 + PyQt6:** Required for the picker helper.
|
- **Python 3:** Handles all color space math and conversions.
|
||||||
- **curl + jq:** Required for color naming support.
|
- **jq:** Powers JSON formatting and internal data parsing.
|
||||||
- **wl-clipboard** (Wayland) or **xclip** (X11): Required for clipboard support.
|
|
||||||
- **libnotify:** Required for desktop notifications (`notify-send`).
|
### Optional Feature Dependencies
|
||||||
|
These enable specific functionality and can be installed as needed:
|
||||||
|
- **KDE Plasma (Wayland):** Required for screen color picking (`--pick`).
|
||||||
|
- **Python 3 PyQt6:** Required for the KDE color picker DBus interface.
|
||||||
|
- **wl-clipboard** or **xclip:** Enables copying results to the system clipboard (`--copy`).
|
||||||
|
- **curl:** Required for fetching human-readable color names from the web (`--name`).
|
||||||
|
- **libnotify:** Enables desktop notifications (`--notify`).
|
||||||
|
- **ImageMagick:** Generates visual color swatch icons for notifications (`--swatch`).
|
||||||
|
- **Truecolor Terminal Emulator:** Required for accurate visual swatches in terminal output (`--swatch`).
|
||||||
|
|
||||||
|
> **Tip:** Use `color-tool --check-deps` to quickly see which dependencies are currently met on your system.
|
||||||
|
|
||||||
## 🚀 Installation
|
## 🚀 Installation
|
||||||
|
|
||||||
Simply download the `color-tool` script and run the install command:
|
### Automatic Installation (Recommended)
|
||||||
|
|
||||||
|
You can download the script directly from the [latest release](https://git.rootiest.dev/rootiest/color-tool/releases/latest) and install it:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
chmod +x color-tool
|
# Download and install the latest version of the color-tool script
|
||||||
./color-tool --install
|
curl -sSLO https://git.rootiest.dev/rootiest/color-tool/releases/download/latest/color-tool && bash color-tool --install && rm ./color-tool
|
||||||
```
|
```
|
||||||
|
|
||||||
This will:
|
This will:
|
||||||
1. Move the script to `~/.local/share/color-tool/`.
|
1. Install the script to `~/.local/share/color-tool/`.
|
||||||
2. Symlink the binary to `~/.local/bin/color-tool`.
|
2. Symlink the binary to `~/.local/bin/color-tool`.
|
||||||
3. Generate a sample configuration at `~/.config/color-tool/config.toml`.
|
3. Generate a sample configuration at `~/.config/color-tool/config.toml`.
|
||||||
4. Create a `.desktop` entry so you can launch the picker from your application menu.
|
4. Create a `.desktop` entry so you can launch the picker from your application menu.
|
||||||
|
5. Verify and warn about any missing optional or required system dependencies.
|
||||||
|
6. Clean up the downloaded script after installation.
|
||||||
|
|
||||||
|
### Manual Usage
|
||||||
|
|
||||||
|
Alternatively, you can [download the script](https://git.rootiest.dev/rootiest/color-tool/releases/latest) manually.
|
||||||
|
Then run it from any location without installation:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Make script executable
|
||||||
|
chmod +x color-tool
|
||||||
|
# Run script directly
|
||||||
|
./color-tool --pick --output hex,rgb
|
||||||
|
```
|
||||||
|
|
||||||
|
However, this will not add the script to your `$PATH` or create a desktop entry,
|
||||||
|
so you won't be able to launch it from the app menu or use it globally without specifying the path.
|
||||||
|
|
||||||
## 🛠 Usage
|
## 🛠 Usage
|
||||||
|
|
||||||
@@ -55,6 +84,7 @@ Options:
|
|||||||
--get-config Print the current configuration
|
--get-config Print the current configuration
|
||||||
--set-config Update the configuration (e.g. --set-config desktop --copy)
|
--set-config Update the configuration (e.g. --set-config desktop --copy)
|
||||||
--reset-config Restore the configuration file to its default values
|
--reset-config Restore the configuration file to its default values
|
||||||
|
--check-deps Check system dependencies and environment support
|
||||||
--install Install to ~/.local/share/ and symlink to ~/.local/bin/
|
--install Install to ~/.local/share/ and symlink to ~/.local/bin/
|
||||||
--help, -h Show this help message
|
--help, -h Show this help message
|
||||||
```
|
```
|
||||||
@@ -103,6 +133,7 @@ json = false # copy JSON format instead of plain text
|
|||||||
name = false # fetch color name (requires network)
|
name = false # fetch color name (requires network)
|
||||||
copy = true # copy result to clipboard
|
copy = true # copy result to clipboard
|
||||||
notify = true # show desktop notification with the copied value
|
notify = true # show desktop notification with the copied value
|
||||||
|
swatch = true # show color swatch in notification
|
||||||
```
|
```
|
||||||
|
|
||||||
You can also use the CLI to manage your configuration:
|
You can also use the CLI to manage your configuration:
|
||||||
|
|||||||
+261
-67
@@ -25,6 +25,31 @@ SCRIPT_DIR="$(dirname "$(readlink -f "$0")")"
|
|||||||
CONFIG_FILE="$HOME/.config/color-tool/config.toml"
|
CONFIG_FILE="$HOME/.config/color-tool/config.toml"
|
||||||
REPO_URL="https://git.rootiest.dev/rootiest/color-tool"
|
REPO_URL="https://git.rootiest.dev/rootiest/color-tool"
|
||||||
|
|
||||||
|
# Colors and styles
|
||||||
|
BOLD='\033[1m'
|
||||||
|
DIM='\033[2m'
|
||||||
|
RESET='\033[0m'
|
||||||
|
CYAN='\033[36m'
|
||||||
|
YELLOW='\033[33m'
|
||||||
|
GREEN='\033[32m'
|
||||||
|
MAG='\033[35m'
|
||||||
|
RED='\033[31m'
|
||||||
|
|
||||||
|
# Gradient logo colors
|
||||||
|
C1='\033[38;2;255;80;80m' # Red-ish
|
||||||
|
C2='\033[38;2;255;160;0m' # Orange
|
||||||
|
C3='\033[38;2;220;210;0m' # Yellow
|
||||||
|
C4='\033[38;2;80;200;80m' # Green
|
||||||
|
C5='\033[38;2;80;160;255m' # Blue
|
||||||
|
|
||||||
|
# Path constants
|
||||||
|
DATA_DIR="$HOME/.local/share/color-tool"
|
||||||
|
BIN_DIR="$HOME/.local/bin"
|
||||||
|
BIN_PATH="$BIN_DIR/color-tool"
|
||||||
|
CONFIG_DIR="$HOME/.config/color-tool"
|
||||||
|
APP_DIR="$HOME/.local/share/applications"
|
||||||
|
DESKTOP_FILE="$APP_DIR/color-tool.desktop"
|
||||||
|
|
||||||
# ── Initial Defaults ──────────────────────────────────────────────────────────
|
# ── Initial Defaults ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
# Hardcoded base defaults
|
# Hardcoded base defaults
|
||||||
@@ -45,6 +70,7 @@ desktop_name=0
|
|||||||
desktop_notify=1
|
desktop_notify=1
|
||||||
desktop_copy=1
|
desktop_copy=1
|
||||||
desktop_output="hex"
|
desktop_output="hex"
|
||||||
|
desktop_swatch=1
|
||||||
|
|
||||||
# Track which settings were explicitly set via CLI flags to ensure they override config
|
# Track which settings were explicitly set via CLI flags to ensure they override config
|
||||||
cli_json=""
|
cli_json=""
|
||||||
@@ -193,7 +219,33 @@ set_config() {
|
|||||||
|
|
||||||
if [[ -n "$new_val" ]]; then
|
if [[ -n "$new_val" ]]; then
|
||||||
if [[ "$original_line" =~ ^([[:space:]]*[A-Za-z_]+[[:space:]]*=[[:space:]]*)([^[:space:]#]+)(.*)$ ]]; then
|
if [[ "$original_line" =~ ^([[:space:]]*[A-Za-z_]+[[:space:]]*=[[:space:]]*)([^[:space:]#]+)(.*)$ ]]; then
|
||||||
echo "${BASH_REMATCH[1]}$new_val${BASH_REMATCH[3]}" >> "$temp_file"
|
local prefix="${BASH_REMATCH[1]}"
|
||||||
|
local old_val="${BASH_REMATCH[2]}"
|
||||||
|
local suffix="${BASH_REMATCH[3]}"
|
||||||
|
|
||||||
|
# Adjust whitespace to keep comments aligned if the value length changed
|
||||||
|
if [[ "$key" != "output" ]]; then
|
||||||
|
local old_len=${#old_val}
|
||||||
|
local new_len=${#new_val}
|
||||||
|
local diff=$((old_len - new_len))
|
||||||
|
|
||||||
|
if [[ $diff -gt 0 ]]; then
|
||||||
|
# new value is shorter, add spaces to suffix
|
||||||
|
local spaces=""
|
||||||
|
for ((i = 0; i < diff; i++)); do spaces+=" "; done
|
||||||
|
suffix="$spaces$suffix"
|
||||||
|
elif [[ $diff -lt 0 ]]; then
|
||||||
|
# new value is longer, remove spaces from suffix
|
||||||
|
local remove=$((-diff))
|
||||||
|
for ((i = 0; i < remove; i++)); do
|
||||||
|
if [[ "$suffix" == " "* ]]; then
|
||||||
|
suffix="${suffix# }"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "$prefix$new_val$suffix" >>"$temp_file"
|
||||||
continue
|
continue
|
||||||
elif [[ "$original_line" =~ ^([[:space:]]*[A-Za-z_]+[[:space:]]*=[[:space:]]*)(.*)$ ]]; then
|
elif [[ "$original_line" =~ ^([[:space:]]*[A-Za-z_]+[[:space:]]*=[[:space:]]*)(.*)$ ]]; then
|
||||||
echo "${BASH_REMATCH[1]}$new_val" >>"$temp_file"
|
echo "${BASH_REMATCH[1]}$new_val" >>"$temp_file"
|
||||||
@@ -242,6 +294,7 @@ load_config() {
|
|||||||
desktop:alpha) [[ "$val" == "true" ]] && desktop_alpha=1 || desktop_alpha=0 ;;
|
desktop:alpha) [[ "$val" == "true" ]] && desktop_alpha=1 || desktop_alpha=0 ;;
|
||||||
desktop:name) [[ "$val" == "true" ]] && desktop_name=1 || desktop_name=0 ;;
|
desktop:name) [[ "$val" == "true" ]] && desktop_name=1 || desktop_name=0 ;;
|
||||||
desktop:notify) [[ "$val" == "true" ]] && desktop_notify=1 || desktop_notify=0 ;;
|
desktop:notify) [[ "$val" == "true" ]] && desktop_notify=1 || desktop_notify=0 ;;
|
||||||
|
desktop:swatch) [[ "$val" == "true" ]] && desktop_swatch=1 || desktop_swatch=0 ;;
|
||||||
desktop:copy) [[ "$val" == "true" ]] && desktop_copy=1 || desktop_copy=0 ;;
|
desktop:copy) [[ "$val" == "true" ]] && desktop_copy=1 || desktop_copy=0 ;;
|
||||||
desktop:output) desktop_output="$val" ;;
|
desktop:output) desktop_output="$val" ;;
|
||||||
esac
|
esac
|
||||||
@@ -272,6 +325,7 @@ json = false # copy JSON format instead of plain text
|
|||||||
name = false # fetch color name (requires network)
|
name = false # fetch color name (requires network)
|
||||||
copy = true # copy result to clipboard
|
copy = true # copy result to clipboard
|
||||||
notify = true # show desktop notification with the copied value
|
notify = true # show desktop notification with the copied value
|
||||||
|
swatch = true # show color swatch in notification
|
||||||
EOF
|
EOF
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -283,43 +337,37 @@ reset_config() {
|
|||||||
# ── Help ──────────────────────────────────────────────────────────────────────
|
# ── Help ──────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
show_help() {
|
show_help() {
|
||||||
local bold='\033[1m'
|
|
||||||
local dim='\033[2m'
|
|
||||||
local reset='\033[0m'
|
|
||||||
local cyan='\033[36m'
|
|
||||||
local yellow='\033[33m'
|
|
||||||
local mag='\033[35m'
|
|
||||||
|
|
||||||
printf "\n"
|
printf "\n"
|
||||||
# Title: "color" in a rainbow gradient, "-tool" in bold
|
# Title: "color" in a rainbow gradient, "-tool" in bold
|
||||||
printf " ${bold}\033[38;2;255;80;80mc\033[38;2;255;160;0mo\033[38;2;220;210;0ml\033[38;2;80;200;80mo\033[38;2;80;160;255mr${reset}${bold}-tool${reset}"
|
printf " ${BOLD}${C1}c${C2}o${C3}l${C4}o${C5}r${RESET}${BOLD}-tool${RESET}"
|
||||||
printf " ${dim}— pick, convert, and format colors${reset}\n\n"
|
printf " ${DIM}— pick, convert, and format colors${RESET}\n\n"
|
||||||
|
|
||||||
printf " ${bold}${yellow}Usage${reset} ${cyan}${0##*/}${reset} ${dim}[OPTIONS]${reset} [COLOR ...]\n\n"
|
printf " ${BOLD}${YELLOW}Usage${RESET} ${CYAN}${0##*/}${RESET} ${DIM}[OPTIONS]${RESET} [COLOR ...]\n\n"
|
||||||
|
|
||||||
printf " ${bold}${yellow}Options${reset}\n"
|
printf " ${BOLD}${YELLOW}Options${RESET}\n"
|
||||||
printf " ${bold}${cyan}--[no-]pick${reset} Open the KDE Plasma color picker ${dim}(requires KDE + Wayland)${reset}\n"
|
printf " ${BOLD}${CYAN}--[no-]pick${RESET} Open the KDE Plasma color picker ${DIM}(requires KDE + Wayland)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--output${reset} ${dim}FMT${reset} Format(s) to output: ${cyan}hex, rgb, hsl, rgba, hsla, hexa, all${reset} ${dim}(comma-separated)${reset}\n"
|
printf " ${BOLD}${CYAN}--output${RESET} ${DIM}FMT${RESET} Format(s) to output: ${CYAN}hex, rgb, hsl, rgba, hsla, hexa, all${RESET} ${DIM}(comma-separated)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--[no-]json${reset} Output as a JSON table of selected formats\n"
|
printf " ${BOLD}${CYAN}--[no-]json${RESET} Output as a JSON table of selected formats\n"
|
||||||
printf " ${bold}${cyan}--[no-]name${reset} Fetch nearest color name from thecolorapi.com ${dim}(requires curl, jq)${reset}\n"
|
printf " ${BOLD}${CYAN}--[no-]name${RESET} Fetch nearest color name from thecolorapi.com ${DIM}(requires curl, jq)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--[no-]swatch${reset} Include a color swatch in the terminal output\n"
|
printf " ${BOLD}${CYAN}--[no-]swatch${RESET} Include a color swatch in the terminal output and desktop notification\n"
|
||||||
printf " ${bold}${cyan}--[no-]copy${reset} Copy result to clipboard ${dim}(wl-clipboard preferred, xclip as fallback)${reset}\n"
|
printf " ${BOLD}${CYAN}--[no-]copy${RESET} Copy result to clipboard ${DIM}(wl-clipboard preferred, xclip as fallback)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--[no-]notify${reset} Show desktop notification ${dim}(on by default in --desktop)${reset}\n"
|
printf " ${BOLD}${CYAN}--[no-]notify${RESET} Show desktop notification ${DIM}(on by default in --desktop)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--desktop${reset} GUI mode: pick → copy → notify ${dim}(for app menu / .desktop launcher)${reset}\n"
|
printf " ${BOLD}${CYAN}--desktop${RESET} GUI mode: pick → copy → notify ${DIM}(for app menu / .desktop launcher)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--get-config${reset} Print the current configuration\n"
|
printf " ${BOLD}${CYAN}--get-config${RESET} Print the current configuration\n"
|
||||||
printf " ${bold}${cyan}--set-config${reset} Update the configuration ${dim}(e.g. --set-config desktop --copy)${reset}\n"
|
printf " ${BOLD}${CYAN}--set-config${RESET} Update the configuration ${DIM}(e.g. --set-config desktop --copy)${RESET}\n"
|
||||||
printf " ${bold}${cyan}--reset-config${reset} Restore the configuration file to its default values\n"
|
printf " ${BOLD}${CYAN}--reset-config${RESET} Restore the configuration file to its default values\n"
|
||||||
printf " ${bold}${cyan}--install${reset} Install to ~/.local/share/color-tool/ and symlink into ~/.local/bin/\n"
|
printf " ${BOLD}${CYAN}--check-deps${RESET} Check system dependencies and environment support\n"
|
||||||
printf " ${bold}${cyan}--help${reset}, ${bold}${cyan}-h${reset} Show this help message\n\n"
|
printf " ${BOLD}${CYAN}--install${RESET} Install to ~/.local/share/color-tool/ and symlink into ~/.local/bin/\n"
|
||||||
|
printf " ${BOLD}${CYAN}--help${RESET}, ${BOLD}${CYAN}-h${RESET} Show this help message\n\n"
|
||||||
|
|
||||||
printf " ${bold}${yellow}Examples${reset}\n"
|
printf " ${BOLD}${YELLOW}Examples${RESET}\n"
|
||||||
printf " ${cyan}${0##*/}${reset} ${mag}\"#534145\"${reset} --swatch\n"
|
printf " ${CYAN}${0##*/}${RESET} ${MAG}\"#534145\"${RESET} --swatch\n"
|
||||||
printf " ${cyan}${0##*/}${reset} --pick --output rgb,hsl --json\n"
|
printf " ${CYAN}${0##*/}${RESET} --pick --output rgb,hsl --json\n"
|
||||||
printf " ${cyan}${0##*/}${reset} \"rgb(255, 100, 0)\" --output hex,hsla\n\n"
|
printf " ${CYAN}${0##*/}${RESET} \"rgb(255, 100, 0)\" --output hex,hsla\n\n"
|
||||||
|
|
||||||
printf " ${bold}${yellow}Config${reset} ${dim}~/.config/color-tool/config.toml${reset}\n"
|
printf " ${BOLD}${YELLOW}Config${RESET} ${DIM}~/.config/color-tool/config.toml${RESET}\n"
|
||||||
printf " ${dim}[defaults]${reset} keys: ${cyan}output json swatch name copy pick notify${reset}\n"
|
printf " ${DIM}[defaults]${RESET} keys: ${CYAN}output json swatch name copy pick notify${RESET}\n"
|
||||||
printf " ${dim}[desktop]${reset} keys: ${cyan}output json name notify copy${reset} ${dim}(pick always on)${reset}\n\n"
|
printf " ${DIM}[desktop]${RESET} keys: ${CYAN}output json name notify copy swatch${RESET} ${DIM}(pick always on)${RESET}\n\n"
|
||||||
}
|
}
|
||||||
|
|
||||||
# ── Environment detection ─────────────────────────────────────────────────────
|
# ── Environment detection ─────────────────────────────────────────────────────
|
||||||
@@ -336,6 +384,26 @@ check_kde_wayland() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
check_truecolor() {
|
||||||
|
# Standard way to check for truecolor support
|
||||||
|
if [[ "${COLORTERM:-}" == "truecolor" || "${COLORTERM:-}" == "24bit" ]]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
# Some terminals report color count via tput
|
||||||
|
if command -v tput >/dev/null 2>&1; then
|
||||||
|
local colors
|
||||||
|
colors=$(tput colors 2>/dev/null || echo 0)
|
||||||
|
if [[ $colors -ge 16777216 ]]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
# Check for direct color in TERM
|
||||||
|
if [[ "${TERM:-}" == *"-direct"* ]]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
# ── Clipboard & Notifications ─────────────────────────────────────────────────
|
# ── Clipboard & Notifications ─────────────────────────────────────────────────
|
||||||
|
|
||||||
copy_to_clipboard() {
|
copy_to_clipboard() {
|
||||||
@@ -353,8 +421,24 @@ copy_to_clipboard() {
|
|||||||
|
|
||||||
notify_result() {
|
notify_result() {
|
||||||
local value="$1"
|
local value="$1"
|
||||||
|
local hex_color="${2:-}"
|
||||||
command -v notify-send &>/dev/null || return 0
|
command -v notify-send &>/dev/null || return 0
|
||||||
notify-send -i "color-picker" "color-tool" "$value" || true
|
|
||||||
|
local icon_name="color-picker"
|
||||||
|
if [[ $swatch_mode -eq 1 && -n "$hex_color" ]]; then
|
||||||
|
local swatch_path="/tmp/color_swatch_${hex_color//\#/}.png"
|
||||||
|
if command -v magick >/dev/null 2>&1; then
|
||||||
|
if magick -size 64x64 xc:"$hex_color" "$swatch_path" 2>/dev/null; then
|
||||||
|
icon_name="$swatch_path"
|
||||||
|
fi
|
||||||
|
elif command -v convert >/dev/null 2>&1; then
|
||||||
|
if convert -size 64x64 xc:"$hex_color" "$swatch_path" 2>/dev/null; then
|
||||||
|
icon_name="$swatch_path"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
notify-send -i "$icon_name" "color-tool" "$value" || true
|
||||||
}
|
}
|
||||||
|
|
||||||
notify_error() {
|
notify_error() {
|
||||||
@@ -434,69 +518,174 @@ run_color_picker() {
|
|||||||
|
|
||||||
suggest_path_add() {
|
suggest_path_add() {
|
||||||
local bin_dir="$1" is_fish=0
|
local bin_dir="$1" is_fish=0
|
||||||
|
# Direct check
|
||||||
[[ "${SHELL:-}" == */fish ]] && is_fish=1
|
[[ "${SHELL:-}" == */fish ]] && is_fish=1
|
||||||
|
|
||||||
|
# Parent process check (good for sub-shells)
|
||||||
if [[ $is_fish -eq 0 ]]; then
|
if [[ $is_fish -eq 0 ]]; then
|
||||||
local parent
|
local parent
|
||||||
parent=$(ps -p "$PPID" -o comm= 2>/dev/null || true)
|
parent=$(ps -p "$PPID" -o comm= 2>/dev/null || true)
|
||||||
[[ "$parent" == *fish* ]] && is_fish=1
|
[[ "$parent" == *fish* ]] && is_fish=1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ $is_fish -eq 1 ]]; then
|
if [[ $is_fish -eq 1 ]]; then
|
||||||
printf " Run: fish_add_path %s\n" "$bin_dir"
|
printf " ${CYAN}Run:${RESET} fish_add_path %s\n" "$bin_dir"
|
||||||
|
elif [[ "${SHELL:-}" == */zsh ]]; then
|
||||||
|
printf " ${CYAN}Add to ~/.zshrc:${RESET} export PATH=\"%s:\$PATH\"\n" "$bin_dir"
|
||||||
else
|
else
|
||||||
printf " For bash/zsh: export PATH=\"%s:\$PATH\"\n" "$bin_dir"
|
# Fallback for Bash or unknown
|
||||||
printf " For fish: fish_add_path %s\n" "$bin_dir"
|
printf " ${CYAN}Add to ~/.bashrc:${RESET} export PATH=\"%s:\$PATH\"\n" "$bin_dir"
|
||||||
|
printf " ${DIM}(Or use fish_add_path if using Fish)${RESET}\n"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
check_deps() {
|
||||||
|
printf "\n ${BOLD}Dependency Check${RESET}\n\n"
|
||||||
|
|
||||||
|
_check() {
|
||||||
|
local label="$1"
|
||||||
|
local cmd="$2"
|
||||||
|
local desc="$3"
|
||||||
|
local required="${4:-0}"
|
||||||
|
|
||||||
|
printf " "
|
||||||
|
if eval "$cmd"; then
|
||||||
|
printf "${GREEN}✔${RESET} "
|
||||||
|
else
|
||||||
|
if [[ $required -eq 1 ]]; then
|
||||||
|
printf "${RED}✘${RESET} "
|
||||||
|
else
|
||||||
|
printf "${YELLOW}⚠${RESET} "
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
printf "%-25s ${DIM}%s${RESET}\n" "$label" "$desc"
|
||||||
|
}
|
||||||
|
|
||||||
|
_check "python3" "command -v python3 >/dev/null" "Core logic and color conversions" 1
|
||||||
|
_check "python3-pyqt6" "python3 -c 'import PyQt6.QtDBus' 2>/dev/null" "KDE color picker interface" 0
|
||||||
|
_check "jq" "command -v jq >/dev/null" "JSON output and data parsing" 1
|
||||||
|
_check "curl" "command -v curl >/dev/null" "Fetching color names from API" 0
|
||||||
|
_check "wl-clipboard / xclip" "command -v wl-copy >/dev/null || command -v xclip >/dev/null" "System clipboard integration" 0
|
||||||
|
_check "libnotify" "command -v notify-send >/dev/null" "Desktop notifications" 0
|
||||||
|
_check "ImageMagick" "command -v magick >/dev/null || command -v convert >/dev/null" "Color swatches in notifications" 0
|
||||||
|
_check "KDE Plasma (Wayland)" "[[ -n \"${WAYLAND_DISPLAY:-}\" ]] && { local d=\"${XDG_CURRENT_DESKTOP:-}\"; [[ \"\$d\" == *KDE* || -n \"${KDE_FULL_SESSION:-}\" ]]; }" "Required for screen color picking" 0
|
||||||
|
_check "Truecolor Support" "check_truecolor" "Color swatches in terminal output" 0
|
||||||
|
|
||||||
|
printf "\n ${DIM}Legend: ${GREEN}✔${DIM} Found ${YELLOW}⚠${DIM} Missing (Optional) ${RED}✘${DIM} Missing (Required)${RESET}\n\n"
|
||||||
|
}
|
||||||
|
|
||||||
|
check_dependencies() {
|
||||||
|
local missing=()
|
||||||
|
|
||||||
|
if ! command -v magick >/dev/null 2>&1 && ! command -v convert >/dev/null 2>&1; then
|
||||||
|
missing+=("ImageMagick (magick or convert) — required for notification color swatches")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! command -v wl-copy >/dev/null 2>&1 && ! command -v xclip >/dev/null 2>&1; then
|
||||||
|
missing+=("wl-clipboard (wl-copy) or xclip — required for clipboard copy support")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! command -v python3 >/dev/null 2>&1; then
|
||||||
|
missing+=("python3 — required for color conversions and picker GUI")
|
||||||
|
elif ! python3 -c "import PyQt6.QtDBus" >/dev/null 2>&1; then
|
||||||
|
missing+=("python3-pyqt6 — required for the KDE color picker DBus interface")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! command -v jq >/dev/null 2>&1; then
|
||||||
|
missing+=("jq — required for JSON formatting and color names")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! command -v curl >/dev/null 2>&1; then
|
||||||
|
missing+=("curl — required for fetching color names from the API")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! command -v notify-send >/dev/null 2>&1; then
|
||||||
|
missing+=("libnotify (notify-send) — required for desktop notifications")
|
||||||
|
fi
|
||||||
|
|
||||||
|
local env_ok=1
|
||||||
|
if [[ -z "${WAYLAND_DISPLAY:-}" ]]; then
|
||||||
|
env_ok=0
|
||||||
|
fi
|
||||||
|
local desktop="${XDG_CURRENT_DESKTOP:-}"
|
||||||
|
if [[ "$desktop" != *KDE* ]] && [[ -z "${KDE_FULL_SESSION:-}" ]]; then
|
||||||
|
env_ok=0
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $env_ok -eq 0 ]]; then
|
||||||
|
missing+=("KDE Plasma on Wayland — required for the screen color picker feature")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! check_truecolor; then
|
||||||
|
missing+=("Truecolor support not detected — the --swatch feature in the terminal may not work as intended")
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ${#missing[@]} -gt 0 ]]; then
|
||||||
|
printf "\n ${BOLD}${YELLOW}Warnings — Missing optional dependencies:${RESET}\n"
|
||||||
|
for item in "${missing[@]}"; do
|
||||||
|
printf " - %s\n" "$item"
|
||||||
|
done
|
||||||
|
printf " ${YELLOW}Functionality will be limited without these installed.${RESET}\n"
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
do_install() {
|
do_install() {
|
||||||
local data_dir="$HOME/.local/share/color-tool"
|
|
||||||
local bin_dir="$HOME/.local/bin"
|
|
||||||
local bin_link="$bin_dir/color-tool"
|
|
||||||
local src_script
|
local src_script
|
||||||
src_script="$(readlink -f "$0")"
|
src_script="$(readlink -f "$0")"
|
||||||
|
|
||||||
mkdir -p "$data_dir" "$bin_dir"
|
printf "\n ${BOLD}Installing ${C1}c${C2}o${C3}l${C4}o${C5}r${RESET}${BOLD}-tool...${RESET}\n\n"
|
||||||
cp "$src_script" "$data_dir/color-tool"
|
|
||||||
chmod +x "$data_dir/color-tool"
|
|
||||||
generate_picker_script "$data_dir/wl-colorpicker-plasma.py"
|
|
||||||
ln -sf "$data_dir/color-tool" "$bin_link"
|
|
||||||
|
|
||||||
local config_dir="$HOME/.config/color-tool"
|
# Efficient Directory Creation Loop
|
||||||
local config_file="$config_dir/config.toml"
|
local dirs=("$BIN_DIR" "$DATA_DIR" "$CONFIG_DIR" "$APP_DIR")
|
||||||
if [[ ! -f "$config_file" ]]; then
|
for dir in "${dirs[@]}"; do
|
||||||
|
if [[ ! -d "$dir" ]]; then
|
||||||
|
printf " ${DIM}Creating %s...${RESET}\n" "$dir"
|
||||||
|
mkdir -p "$dir"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# Core installation steps
|
||||||
|
cp "$src_script" "$DATA_DIR/color-tool"
|
||||||
|
chmod +x "$DATA_DIR/color-tool"
|
||||||
|
generate_picker_script "$DATA_DIR/wl-colorpicker-plasma.py"
|
||||||
|
ln -sf "$DATA_DIR/color-tool" "$BIN_PATH"
|
||||||
|
|
||||||
|
# Config setup
|
||||||
|
if [[ ! -f "$CONFIG_FILE" ]]; then
|
||||||
write_default_config
|
write_default_config
|
||||||
printf " config %s (sample created)\n" "$config_file"
|
printf " ${BOLD}${GREEN}config${RESET} %s ${DIM}(sample created)${RESET}\n" "$CONFIG_FILE"
|
||||||
else
|
else
|
||||||
printf " config %s\n" "$config_file"
|
printf " ${BOLD}${GREEN}config${RESET} %s\n" "$CONFIG_FILE"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
local app_dir="$HOME/.local/share/applications"
|
# Desktop Entry (using consolidated BIN_PATH)
|
||||||
local desktop_file="$app_dir/color-tool.desktop"
|
cat >"$DESKTOP_FILE" <<EOF
|
||||||
local bin_path="$HOME/.local/bin/color-tool"
|
|
||||||
mkdir -p "$app_dir"
|
|
||||||
cat >"$desktop_file" <<EOF
|
|
||||||
[Desktop Entry]
|
[Desktop Entry]
|
||||||
Version=1.1
|
Version=1.1
|
||||||
Type=Application
|
Type=Application
|
||||||
Name=Color Tool
|
Name=Color Tool
|
||||||
Comment=Pick a color and copy it to the clipboard
|
Comment=Pick a color and copy it to the clipboard
|
||||||
Exec=$bin_path --desktop
|
Exec=$BIN_PATH --desktop
|
||||||
Icon=color-picker
|
Icon=color-picker
|
||||||
Categories=Utility;Graphics;
|
Categories=Utility;Graphics;
|
||||||
Keywords=color;picker;hex;rgb;clipboard;
|
Keywords=color;picker;hex;rgb;clipboard;
|
||||||
Terminal=false
|
Terminal=false
|
||||||
NoDisplay=false
|
NoDisplay=false
|
||||||
EOF
|
EOF
|
||||||
printf " app %s\n" "$desktop_file"
|
|
||||||
|
|
||||||
printf "Installed:\n"
|
# Summary output
|
||||||
printf " data %s/\n" "$data_dir"
|
printf " ${BOLD}${GREEN}app${RESET} %s\n" "$DESKTOP_FILE"
|
||||||
printf " bin %s -> %s\n" "$bin_link" "$data_dir/color-tool"
|
printf " ${BOLD}${GREEN}data${RESET} %s/\n" "$DATA_DIR"
|
||||||
|
printf " ${BOLD}${GREEN}bin${RESET} %s -> %s\n" "$BIN_PATH" "$DATA_DIR/color-tool"
|
||||||
|
|
||||||
if [[ ":$PATH:" != *":$bin_dir:"* ]]; then
|
# Path verification
|
||||||
printf "\nNote: %s is not in your PATH.\n" "$bin_dir"
|
if [[ ":$PATH:" != *":$BIN_DIR:"* ]]; then
|
||||||
suggest_path_add "$bin_dir"
|
printf "\n${BOLD}${YELLOW}Warning:${RESET} %s is not in your PATH.\n" "$BIN_DIR"
|
||||||
|
suggest_path_add "$BIN_DIR"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
check_dependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
# ── Color conversion ──────────────────────────────────────────────────────────
|
# ── Color conversion ──────────────────────────────────────────────────────────
|
||||||
@@ -571,10 +760,12 @@ process_color() {
|
|||||||
return 1
|
return 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
local hex_color
|
||||||
|
hex_color=$(echo "$formats_json" | jq -r '.hex // empty')
|
||||||
|
|
||||||
local name=""
|
local name=""
|
||||||
if [[ $name_mode -eq 1 ]]; then
|
if [[ $name_mode -eq 1 ]]; then
|
||||||
local hex_for_api
|
local hex_for_api="${hex_color//#/}"
|
||||||
hex_for_api=$(echo "$formats_json" | jq -r '.hex' | tr -d '#')
|
|
||||||
name=$(curl -s "https://www.thecolorapi.com/id?hex=$hex_for_api" | jq -r '.name.value // empty')
|
name=$(curl -s "https://www.thecolorapi.com/id?hex=$hex_for_api" | jq -r '.name.value // empty')
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -640,7 +831,7 @@ process_color() {
|
|||||||
|
|
||||||
Value: $output_text"
|
Value: $output_text"
|
||||||
else
|
else
|
||||||
notify_result "$output_text"
|
notify_result "$output_text" "$hex_color"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
@@ -697,6 +888,10 @@ while [[ $# -gt 0 ]]; do
|
|||||||
reset_config
|
reset_config
|
||||||
exit 0
|
exit 0
|
||||||
;;
|
;;
|
||||||
|
--check-deps)
|
||||||
|
check_deps
|
||||||
|
exit 0
|
||||||
|
;;
|
||||||
-*)
|
-*)
|
||||||
echo "Unknown option: $1" >&2
|
echo "Unknown option: $1" >&2
|
||||||
exit 1
|
exit 1
|
||||||
@@ -706,7 +901,6 @@ while [[ $# -gt 0 ]]; do
|
|||||||
shift
|
shift
|
||||||
done
|
done
|
||||||
|
|
||||||
# Apply final hierarchy: CLI Flags > Desktop Config (if --desktop) > Default Config
|
|
||||||
if [[ $desktop_mode -eq 1 ]]; then
|
if [[ $desktop_mode -eq 1 ]]; then
|
||||||
json_mode=${cli_json:-$desktop_json}
|
json_mode=${cli_json:-$desktop_json}
|
||||||
alpha_mode=${cli_alpha:-$desktop_alpha}
|
alpha_mode=${cli_alpha:-$desktop_alpha}
|
||||||
@@ -714,8 +908,8 @@ if [[ $desktop_mode -eq 1 ]]; then
|
|||||||
notify_mode=${cli_notify:-$desktop_notify}
|
notify_mode=${cli_notify:-$desktop_notify}
|
||||||
output_formats=${cli_output:-$desktop_output}
|
output_formats=${cli_output:-$desktop_output}
|
||||||
copy_mode=${cli_copy:-$desktop_copy}
|
copy_mode=${cli_copy:-$desktop_copy}
|
||||||
|
swatch_mode=${cli_swatch:-$desktop_swatch}
|
||||||
do_pick=${cli_pick:-1}
|
do_pick=${cli_pick:-1}
|
||||||
swatch_mode=${cli_swatch:-0} # swatch usually off in desktop mode
|
|
||||||
else
|
else
|
||||||
json_mode=${cli_json:-$json_mode}
|
json_mode=${cli_json:-$json_mode}
|
||||||
alpha_mode=${cli_alpha:-$alpha_mode}
|
alpha_mode=${cli_alpha:-$alpha_mode}
|
||||||
|
|||||||
@@ -206,6 +206,32 @@ it "rejects invalid formats"
|
|||||||
output=$("$COLOR_TOOL" "#000000" --output badfmt --no-copy --no-notify 2>&1 || true)
|
output=$("$COLOR_TOOL" "#000000" --output badfmt --no-copy --no-notify 2>&1 || true)
|
||||||
assert_contains "$output" "Error: Invalid output format: badfmt"
|
assert_contains "$output" "Error: Invalid output format: badfmt"
|
||||||
|
|
||||||
|
# 14. Installation Dependency Warnings
|
||||||
|
it "warns about missing dependencies during install"
|
||||||
|
mkdir -p "$TEST_DIR/install_bin"
|
||||||
|
for cmd in bash cat echo printf mkdir mktemp chmod rm cp ln dirname readlink ps; do
|
||||||
|
if [[ -e "/usr/bin/$cmd" || -L "/usr/bin/$cmd" ]]; then
|
||||||
|
/usr/bin/ln -s "/usr/bin/$cmd" "$TEST_DIR/install_bin/$cmd"
|
||||||
|
elif [[ -e "/bin/$cmd" || -L "/bin/$cmd" ]]; then
|
||||||
|
/usr/bin/ln -s "/bin/$cmd" "$TEST_DIR/install_bin/$cmd"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
output=$(PATH="$TEST_DIR/install_bin" HOME="$TEST_DIR" WAYLAND_DISPLAY="" XDG_CURRENT_DESKTOP="" KDE_FULL_SESSION="" "$COLOR_TOOL" --install 2>&1 || true)
|
||||||
|
|
||||||
|
if [[ "$output" == *"python3 — required"* ]] &&
|
||||||
|
[[ "$output" == *"jq — required"* ]] &&
|
||||||
|
[[ "$output" == *"curl — required"* ]] &&
|
||||||
|
[[ "$output" == *"libnotify (notify-send)"* ]] &&
|
||||||
|
[[ "$output" == *"KDE Plasma on Wayland"* ]] &&
|
||||||
|
[[ "$output" == *"ImageMagick"* ]] &&
|
||||||
|
[[ "$output" == *"wl-clipboard"* ]]; then
|
||||||
|
echo "PASS"
|
||||||
|
passed=$((passed + 1))
|
||||||
|
else
|
||||||
|
echo "FAIL"
|
||||||
|
echo " Actual output: $output"
|
||||||
|
fi
|
||||||
|
|
||||||
# ── Cleanup ───────────────────────────────────────────────────────────────────
|
# ── Cleanup ───────────────────────────────────────────────────────────────────
|
||||||
echo "---------------------------------------"
|
echo "---------------------------------------"
|
||||||
echo "Result: $passed/$total tests passed."
|
echo "Result: $passed/$total tests passed."
|
||||||
|
|||||||
Reference in New Issue
Block a user