* Add sidebar blur effect with withinWindow blending - Add NSVisualEffectView-based blur backdrop for sidebar - Support withinWindow blending mode to blur terminal content behind sidebar - Auto-switch to overlay layout when withinWindow mode is selected - Add sidebar debug panel with material, blending, tint, and opacity controls - Add preset options (HUD Glass, Popover Glass, etc.) - Default to HUD Glass preset with withinWindow blur * Simplify tab close button visibility and remove hover background Show close button only on hover instead of when active/multi-selected. Remove the hover background color from tabs for cleaner appearance. * Add config reload support with notification system - Add reloadConfiguration() methods for app-wide and per-surface reload - Handle GHOSTTY_ACTION_RELOAD_CONFIG action from Ghostty - Add ghosttyConfigDidReload notification for views to react - TerminalSplitTreeView reloads GhosttyConfig on notification - Add openConfigurationInTextEdit() helper - Fix activeMainWindow() to correctly find main window * Add custom tab titles and pinned tabs support - Add customTitle and isPinned properties to Tab - Separate process title from custom title with applyProcessTitle() - Add setCustomTitle()/clearCustomTitle() for user-defined tab names - Add togglePin()/setPinned() with automatic reordering - Pinned tabs stay at the top, new tabs insert after pinned section - moveTabToTop/moveTabsToTop respect pinned tab ordering * Add --panel option to new-split command - CLI: Parse --panel <id|index> option for new-split - Controller: Resolve panel argument to split specific surface - Return new panel UUID on successful split creation * Fix notifications popover positioning with layout-aware anchor - Use AnchorNSView with layout callback for reliable positioning - Force layout before showing popover to ensure current geometry - Convert anchor bounds to window content view coordinates - Add fallback positioning near top-left when anchor unavailable - Fix button hit testing with explicit frame and contentShape * Improve app termination in reload script Use osascript to gracefully quit by bundle ID before pkill fallback. Add more robust pkill patterns to catch instances from any DerivedData path. * Add sidebar blur effect with live-adjustable glass settings - Add WindowGlassEffect for window-level NSGlassEffectView (macOS 26+) - Add SidebarBackdrop with configurable material, blend mode, tint, and opacity - Add Sidebar Debug panel (Debug menu) for live adjustment of sidebar appearance - Add Background Debug panel for window glass tint settings - Support both behindWindow and withinWindow blur modes - Live tint updates without requiring window reload * Align titlebar text to left edge of content area
245 lines
7.5 KiB
Bash
Executable file
245 lines
7.5 KiB
Bash
Executable file
#!/usr/bin/env bash
|
|
set -euo pipefail
|
|
|
|
APP_NAME="cmuxterm DEV"
|
|
BUNDLE_ID="com.cmuxterm.app.debug"
|
|
BASE_APP_NAME="cmuxterm DEV"
|
|
DERIVED_DATA=""
|
|
NAME_SET=0
|
|
BUNDLE_SET=0
|
|
DERIVED_SET=0
|
|
TAG=""
|
|
|
|
usage() {
|
|
cat <<'EOF'
|
|
Usage: ./scripts/reload.sh [options]
|
|
|
|
Options:
|
|
--tag <name> Short tag for parallel builds (e.g., feature-xyz-lol).
|
|
Sets app name, bundle id, and derived data path unless overridden.
|
|
--name <app name> Override app display/bundle name.
|
|
--bundle-id <id> Override bundle identifier.
|
|
--derived-data <path> Override derived data path.
|
|
-h, --help Show this help.
|
|
EOF
|
|
}
|
|
|
|
sanitize_bundle() {
|
|
local raw="$1"
|
|
local cleaned
|
|
cleaned="$(echo "$raw" | tr '[:upper:]' '[:lower:]' | sed -E 's/[^a-z0-9]+/./g; s/^\\.+//; s/\\.+$//; s/\\.+/./g')"
|
|
if [[ -z "$cleaned" ]]; then
|
|
cleaned="agent"
|
|
fi
|
|
echo "$cleaned"
|
|
}
|
|
|
|
sanitize_path() {
|
|
local raw="$1"
|
|
local cleaned
|
|
cleaned="$(echo "$raw" | tr '[:upper:]' '[:lower:]' | sed -E 's/[^a-z0-9]+/-/g; s/^-+//; s/-+$//; s/-+/-/g')"
|
|
if [[ -z "$cleaned" ]]; then
|
|
cleaned="agent"
|
|
fi
|
|
echo "$cleaned"
|
|
}
|
|
|
|
while [[ $# -gt 0 ]]; do
|
|
case "$1" in
|
|
--tag)
|
|
TAG="${2:-}"
|
|
if [[ -z "$TAG" ]]; then
|
|
echo "error: --tag requires a value" >&2
|
|
exit 1
|
|
fi
|
|
shift 2
|
|
;;
|
|
--name)
|
|
APP_NAME="${2:-}"
|
|
if [[ -z "$APP_NAME" ]]; then
|
|
echo "error: --name requires a value" >&2
|
|
exit 1
|
|
fi
|
|
NAME_SET=1
|
|
shift 2
|
|
;;
|
|
--bundle-id)
|
|
BUNDLE_ID="${2:-}"
|
|
if [[ -z "$BUNDLE_ID" ]]; then
|
|
echo "error: --bundle-id requires a value" >&2
|
|
exit 1
|
|
fi
|
|
BUNDLE_SET=1
|
|
shift 2
|
|
;;
|
|
--derived-data)
|
|
DERIVED_DATA="${2:-}"
|
|
if [[ -z "$DERIVED_DATA" ]]; then
|
|
echo "error: --derived-data requires a value" >&2
|
|
exit 1
|
|
fi
|
|
DERIVED_SET=1
|
|
shift 2
|
|
;;
|
|
-h|--help)
|
|
usage
|
|
exit 0
|
|
;;
|
|
*)
|
|
echo "error: unknown option $1" >&2
|
|
usage
|
|
exit 1
|
|
;;
|
|
esac
|
|
done
|
|
|
|
if [[ -n "$TAG" ]]; then
|
|
TAG_ID="$(sanitize_bundle "$TAG")"
|
|
TAG_SLUG="$(sanitize_path "$TAG")"
|
|
if [[ "$NAME_SET" -eq 0 ]]; then
|
|
APP_NAME="cmuxterm DEV ${TAG}"
|
|
fi
|
|
if [[ "$BUNDLE_SET" -eq 0 ]]; then
|
|
BUNDLE_ID="com.cmuxterm.app.debug.${TAG_ID}"
|
|
fi
|
|
if [[ "$DERIVED_SET" -eq 0 ]]; then
|
|
DERIVED_DATA="/tmp/cmuxterm-${TAG_SLUG}"
|
|
fi
|
|
fi
|
|
|
|
XCODEBUILD_ARGS=(
|
|
-project GhosttyTabs.xcodeproj
|
|
-scheme cmux
|
|
-configuration Debug
|
|
-destination 'platform=macOS'
|
|
)
|
|
if [[ -n "$DERIVED_DATA" ]]; then
|
|
XCODEBUILD_ARGS+=(-derivedDataPath "$DERIVED_DATA")
|
|
fi
|
|
if [[ -z "$TAG" ]]; then
|
|
XCODEBUILD_ARGS+=(
|
|
INFOPLIST_KEY_CFBundleName="$APP_NAME"
|
|
INFOPLIST_KEY_CFBundleDisplayName="$APP_NAME"
|
|
PRODUCT_BUNDLE_IDENTIFIER="$BUNDLE_ID"
|
|
)
|
|
fi
|
|
XCODEBUILD_ARGS+=(build)
|
|
|
|
xcodebuild "${XCODEBUILD_ARGS[@]}"
|
|
sleep 0.2
|
|
|
|
FALLBACK_APP_NAME="$BASE_APP_NAME"
|
|
SEARCH_APP_NAME="$APP_NAME"
|
|
if [[ -n "$TAG" ]]; then
|
|
SEARCH_APP_NAME="$BASE_APP_NAME"
|
|
fi
|
|
if [[ -n "$DERIVED_DATA" ]]; then
|
|
APP_PATH="${DERIVED_DATA}/Build/Products/Debug/${SEARCH_APP_NAME}.app"
|
|
if [[ ! -d "${APP_PATH}" && "$SEARCH_APP_NAME" != "$FALLBACK_APP_NAME" ]]; then
|
|
APP_PATH="${DERIVED_DATA}/Build/Products/Debug/${FALLBACK_APP_NAME}.app"
|
|
fi
|
|
else
|
|
APP_BINARY="$(
|
|
find "$HOME/Library/Developer/Xcode/DerivedData" -path "*/Build/Products/Debug/${SEARCH_APP_NAME}.app/Contents/MacOS/${SEARCH_APP_NAME}" -print0 \
|
|
| xargs -0 /usr/bin/stat -f "%m %N" 2>/dev/null \
|
|
| sort -nr \
|
|
| head -n 1 \
|
|
| cut -d' ' -f2-
|
|
)"
|
|
if [[ -n "${APP_BINARY}" ]]; then
|
|
APP_PATH="$(dirname "$(dirname "$(dirname "$APP_BINARY")")")"
|
|
fi
|
|
if [[ -z "${APP_PATH}" && "$SEARCH_APP_NAME" != "$FALLBACK_APP_NAME" ]]; then
|
|
APP_BINARY="$(
|
|
find "$HOME/Library/Developer/Xcode/DerivedData" -path "*/Build/Products/Debug/${FALLBACK_APP_NAME}.app/Contents/MacOS/${FALLBACK_APP_NAME}" -print0 \
|
|
| xargs -0 /usr/bin/stat -f "%m %N" 2>/dev/null \
|
|
| sort -nr \
|
|
| head -n 1 \
|
|
| cut -d' ' -f2-
|
|
)"
|
|
if [[ -n "${APP_BINARY}" ]]; then
|
|
APP_PATH="$(dirname "$(dirname "$(dirname "$APP_BINARY")")")"
|
|
fi
|
|
fi
|
|
fi
|
|
if [[ -z "${APP_PATH}" || ! -d "${APP_PATH}" ]]; then
|
|
echo "${APP_NAME}.app not found in DerivedData" >&2
|
|
exit 1
|
|
fi
|
|
|
|
if [[ -n "$TAG" && "$APP_NAME" != "$SEARCH_APP_NAME" ]]; then
|
|
TAG_APP_PATH="$(dirname "$APP_PATH")/${APP_NAME}.app"
|
|
rm -rf "$TAG_APP_PATH"
|
|
cp -R "$APP_PATH" "$TAG_APP_PATH"
|
|
INFO_PLIST="$TAG_APP_PATH/Contents/Info.plist"
|
|
if [[ -f "$INFO_PLIST" ]]; then
|
|
/usr/libexec/PlistBuddy -c "Set :CFBundleName $APP_NAME" "$INFO_PLIST" 2>/dev/null \
|
|
|| /usr/libexec/PlistBuddy -c "Add :CFBundleName string $APP_NAME" "$INFO_PLIST"
|
|
/usr/libexec/PlistBuddy -c "Set :CFBundleDisplayName $APP_NAME" "$INFO_PLIST" 2>/dev/null \
|
|
|| /usr/libexec/PlistBuddy -c "Add :CFBundleDisplayName string $APP_NAME" "$INFO_PLIST"
|
|
/usr/libexec/PlistBuddy -c "Set :CFBundleIdentifier $BUNDLE_ID" "$INFO_PLIST" 2>/dev/null \
|
|
|| /usr/libexec/PlistBuddy -c "Add :CFBundleIdentifier string $BUNDLE_ID" "$INFO_PLIST"
|
|
if [[ -n "${TAG_SLUG:-}" ]]; then
|
|
APP_SUPPORT_DIR="$HOME/Library/Application Support/cmuxterm"
|
|
CMUXD_SOCKET="${APP_SUPPORT_DIR}/cmuxd-dev-${TAG_SLUG}.sock"
|
|
CMUX_SOCKET="/tmp/cmuxterm-debug-${TAG_SLUG}.sock"
|
|
/usr/libexec/PlistBuddy -c "Add :LSEnvironment dict" "$INFO_PLIST" 2>/dev/null || true
|
|
/usr/libexec/PlistBuddy -c "Set :LSEnvironment:CMUXD_UNIX_PATH \"${CMUXD_SOCKET}\"" "$INFO_PLIST" 2>/dev/null \
|
|
|| /usr/libexec/PlistBuddy -c "Add :LSEnvironment:CMUXD_UNIX_PATH string \"${CMUXD_SOCKET}\"" "$INFO_PLIST"
|
|
/usr/libexec/PlistBuddy -c "Set :LSEnvironment:CMUX_SOCKET_PATH \"${CMUX_SOCKET}\"" "$INFO_PLIST" 2>/dev/null \
|
|
|| /usr/libexec/PlistBuddy -c "Add :LSEnvironment:CMUX_SOCKET_PATH string \"${CMUX_SOCKET}\"" "$INFO_PLIST"
|
|
if [[ -S "$CMUXD_SOCKET" ]]; then
|
|
for PID in $(lsof -t "$CMUXD_SOCKET" 2>/dev/null); do
|
|
kill "$PID" 2>/dev/null || true
|
|
done
|
|
rm -f "$CMUXD_SOCKET"
|
|
fi
|
|
if [[ -S "$CMUX_SOCKET" ]]; then
|
|
rm -f "$CMUX_SOCKET"
|
|
fi
|
|
fi
|
|
/usr/bin/codesign --force --sign - --timestamp=none --generate-entitlement-der "$TAG_APP_PATH" >/dev/null 2>&1 || true
|
|
fi
|
|
APP_PATH="$TAG_APP_PATH"
|
|
fi
|
|
|
|
# Ensure any running instance is fully terminated, regardless of DerivedData path.
|
|
/usr/bin/osascript -e "tell application id \"${BUNDLE_ID}\" to quit" >/dev/null 2>&1 || true
|
|
sleep 0.3
|
|
pkill -f "/${BASE_APP_NAME}.app/Contents/MacOS/${BASE_APP_NAME}" || true
|
|
if [[ "${APP_NAME}" != "${BASE_APP_NAME}" ]]; then
|
|
pkill -f "/${APP_NAME}.app/Contents/MacOS/${APP_NAME}" || true
|
|
fi
|
|
sleep 0.3
|
|
CMUXD_SRC="$PWD/cmuxd/zig-out/bin/cmuxd"
|
|
if [[ -d "$PWD/cmuxd" ]]; then
|
|
(cd "$PWD/cmuxd" && zig build -Doptimize=ReleaseFast)
|
|
fi
|
|
if [[ -x "$CMUXD_SRC" ]]; then
|
|
BIN_DIR="$APP_PATH/Contents/Resources/bin"
|
|
mkdir -p "$BIN_DIR"
|
|
cp "$CMUXD_SRC" "$BIN_DIR/cmuxd"
|
|
chmod +x "$BIN_DIR/cmuxd"
|
|
fi
|
|
open "$APP_PATH"
|
|
osascript -e "tell application id \"${BUNDLE_ID}\" to activate" || true
|
|
|
|
# Safety: ensure only one instance is running.
|
|
sleep 0.2
|
|
PIDS=($(pgrep -f "${APP_PATH}/Contents/MacOS/" || true))
|
|
if [[ "${#PIDS[@]}" -gt 1 ]]; then
|
|
NEWEST_PID=""
|
|
NEWEST_AGE=999999
|
|
for PID in "${PIDS[@]}"; do
|
|
AGE="$(ps -o etimes= -p "$PID" | tr -d ' ')"
|
|
if [[ -n "$AGE" && "$AGE" -lt "$NEWEST_AGE" ]]; then
|
|
NEWEST_AGE="$AGE"
|
|
NEWEST_PID="$PID"
|
|
fi
|
|
done
|
|
for PID in "${PIDS[@]}"; do
|
|
if [[ "$PID" != "$NEWEST_PID" ]]; then
|
|
kill "$PID" 2>/dev/null || true
|
|
fi
|
|
done
|
|
fi
|