mirror of
https://github.com/kunkundi/crossdesk.git
synced 2025-12-19 22:09:09 +08:00
Compare commits
1 Commits
eedbf1f64a
...
v1.1.5-202
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a3c905d175 |
95
.github/workflows/update-version-json.yml
vendored
95
.github/workflows/update-version-json.yml
vendored
@@ -73,59 +73,68 @@ jobs:
|
||||
curl -f -s "https://version.crossdesk.cn/version.json" -o version.json || echo "Failed to download, will create new one"
|
||||
|
||||
- name: Generate or update version.json
|
||||
shell: bash
|
||||
run: |
|
||||
TAG_NAME="${{ steps.version.outputs.TAG_NAME }}"
|
||||
VERSION_ONLY="${{ steps.version.outputs.VERSION_ONLY }}"
|
||||
BUILD_DATE_ISO="${{ steps.version.outputs.BUILD_DATE_ISO }}"
|
||||
RELEASE_NAME_JSON="${{ steps.release_info.outputs.RELEASE_NAME }}"
|
||||
RELEASE_BODY_JSON="${{ steps.release_info.outputs.RELEASE_BODY }}"
|
||||
|
||||
# Default downloads structure - use jq to build JSON safely
|
||||
DEFAULT_DOWNLOADS=$(jq -n \
|
||||
--arg tag "${TAG_NAME}" \
|
||||
'{
|
||||
"windows-x64": {
|
||||
"url": ("https://downloads.crossdesk.cn/crossdesk-win-x64-" + $tag + ".exe"),
|
||||
"filename": ("crossdesk-win-x64-" + $tag + ".exe")
|
||||
},
|
||||
"macos-x64": {
|
||||
"url": ("https://downloads.crossdesk.cn/crossdesk-macos-x64-" + $tag + ".pkg"),
|
||||
"filename": ("crossdesk-macos-x64-" + $tag + ".pkg")
|
||||
},
|
||||
"macos-arm64": {
|
||||
"url": ("https://downloads.crossdesk.cn/crossdesk-macos-arm64-" + $tag + ".pkg"),
|
||||
"filename": ("crossdesk-macos-arm64-" + $tag + ".pkg")
|
||||
},
|
||||
"linux-amd64": {
|
||||
"url": ("https://downloads.crossdesk.cn/crossdesk-linux-amd64-" + $tag + ".deb"),
|
||||
"filename": ("crossdesk-linux-amd64-" + $tag + ".deb")
|
||||
},
|
||||
"linux-arm64": {
|
||||
"url": ("https://downloads.crossdesk.cn/crossdesk-linux-arm64-" + $tag + ".deb"),
|
||||
"filename": ("crossdesk-linux-arm64-" + $tag + ".deb")
|
||||
}
|
||||
}')
|
||||
|
||||
# If version.json exists, try to preserve downloads section
|
||||
if [ -f version.json ] && jq -e '.downloads' version.json > /dev/null 2>&1; then
|
||||
EXISTING_DOWNLOADS=$(jq -c '.downloads' version.json)
|
||||
if [ "$EXISTING_DOWNLOADS" != "null" ] && [ "$EXISTING_DOWNLOADS" != "{}" ]; then
|
||||
DOWNLOADS_JSON="$EXISTING_DOWNLOADS"
|
||||
DOWNLOADS="$EXISTING_DOWNLOADS"
|
||||
else
|
||||
DOWNLOADS_JSON=""
|
||||
DOWNLOADS="$DEFAULT_DOWNLOADS"
|
||||
fi
|
||||
else
|
||||
DOWNLOADS_JSON=""
|
||||
DOWNLOADS="$DEFAULT_DOWNLOADS"
|
||||
fi
|
||||
|
||||
# If downloads is empty, use default structure
|
||||
if [ -z "$DOWNLOADS_JSON" ]; then
|
||||
DOWNLOADS_JSON=$(cat << DOWNLOADS_EOF
|
||||
{
|
||||
"windows-x64": {
|
||||
"url": "https://downloads.crossdesk.cn/crossdesk-win-x64-${{ steps.version.outputs.TAG_NAME }}.exe",
|
||||
"filename": "crossdesk-win-x64-${{ steps.version.outputs.TAG_NAME }}.exe"
|
||||
},
|
||||
"macos-x64": {
|
||||
"url": "https://downloads.crossdesk.cn/crossdesk-macos-x64-${{ steps.version.outputs.TAG_NAME }}.pkg",
|
||||
"filename": "crossdesk-macos-x64-${{ steps.version.outputs.TAG_NAME }}.pkg"
|
||||
},
|
||||
"macos-arm64": {
|
||||
"url": "https://downloads.crossdesk.cn/crossdesk-macos-arm64-${{ steps.version.outputs.TAG_NAME }}.pkg",
|
||||
"filename": "crossdesk-macos-arm64-${{ steps.version.outputs.TAG_NAME }}.pkg"
|
||||
},
|
||||
"linux-amd64": {
|
||||
"url": "https://downloads.crossdesk.cn/crossdesk-linux-amd64-${{ steps.version.outputs.TAG_NAME }}.deb",
|
||||
"filename": "crossdesk-linux-amd64-${{ steps.version.outputs.TAG_NAME }}.deb"
|
||||
},
|
||||
"linux-arm64": {
|
||||
"url": "https://downloads.crossdesk.cn/crossdesk-linux-arm64-${{ steps.version.outputs.TAG_NAME }}.deb",
|
||||
"filename": "crossdesk-linux-arm64-${{ steps.version.outputs.TAG_NAME }}.deb"
|
||||
}
|
||||
}
|
||||
DOWNLOADS_EOF
|
||||
)
|
||||
fi
|
||||
|
||||
# Generate version.json using cat and heredoc
|
||||
cat > version.json << EOF
|
||||
{
|
||||
"version": "${{ steps.version.outputs.VERSION_ONLY }}",
|
||||
"releaseDate": "${{ steps.version.outputs.BUILD_DATE_ISO }}",
|
||||
"releaseName": ${{ steps.release_info.outputs.RELEASE_NAME }},
|
||||
"releaseNotes": ${{ steps.release_info.outputs.RELEASE_BODY }},
|
||||
"tagName": "${{ steps.version.outputs.TAG_NAME }}",
|
||||
"downloads": ${DOWNLOADS_JSON}
|
||||
}
|
||||
EOF
|
||||
# Create version.json using jq for proper JSON formatting
|
||||
jq -n \
|
||||
--arg version "$VERSION_ONLY" \
|
||||
--arg releaseDate "$BUILD_DATE_ISO" \
|
||||
--argjson releaseName "$RELEASE_NAME_JSON" \
|
||||
--argjson releaseNotes "$RELEASE_BODY_JSON" \
|
||||
--arg tagName "$TAG_NAME" \
|
||||
--argjson downloads "$DOWNLOADS" \
|
||||
'{
|
||||
version: $version,
|
||||
releaseDate: $releaseDate,
|
||||
releaseName: $releaseName,
|
||||
releaseNotes: $releaseNotes,
|
||||
tagName: $tagName,
|
||||
downloads: $downloads
|
||||
}' > version.json
|
||||
|
||||
cat version.json
|
||||
|
||||
|
||||
@@ -315,10 +315,10 @@ Generation complete. Deployment files::
|
||||
Server certificate: crossdesk.cn_bundle.crt
|
||||
```
|
||||
|
||||
### 服务端
|
||||
#### 服务端
|
||||
将 **crossdesk.cn.key** 和 **crossdesk.cn_bundle.crt** 放置到 **/path/to/your/certs** 目录下。
|
||||
|
||||
### 客户端
|
||||
#### 客户端
|
||||
1. 点击右上角设置进入设置页面。<br>
|
||||
<img width="600" height="210" alt="image" src="https://github.com/user-attachments/assets/6431131d-b32a-4726-8783-6788f47baa3b" /><br><br>
|
||||
|
||||
@@ -331,8 +331,5 @@ Generation complete. Deployment files::
|
||||
7. 勾选使用**自托管服务器配置**,点击确认配置生效。<br><br>
|
||||
<img width="600" height="160" alt="image" src="https://github.com/user-attachments/assets/1e455dc3-4087-4f37-a544-1ff9f8789383" /><br><br>
|
||||
|
||||
### Web 客户端
|
||||
详情见项目 [CrossDesk Web Client](https://github.com/kunkundi/crossdesk-web-client)。
|
||||
|
||||
# 常见问题
|
||||
见 [常见问题](https://github.com/kunkundi/crossdesk/blob/self-hosted-server/docs/FAQ.md) 。
|
||||
|
||||
@@ -323,10 +323,10 @@ Generation complete. Deployment files::
|
||||
Server certificate: crossdesk.cn_bundle.crt
|
||||
```
|
||||
|
||||
### Server Side
|
||||
#### Server Side
|
||||
Place **crossdesk.cn.key** and **crossdesk.cn_bundle.crt** into the **/path/to/your/certs** directory.
|
||||
|
||||
### Client Side
|
||||
#### Client Side
|
||||
1. Click the settings icon in the top-right corner to enter the settings page.<br>
|
||||
<img width="600" height="210" alt="image" src="https://github.com/user-attachments/assets/6431131d-b32a-4726-8783-6788f47baa3b" /><br><br>
|
||||
|
||||
@@ -339,8 +339,5 @@ Place **crossdesk.cn.key** and **crossdesk.cn_bundle.crt** into the **/path/to/y
|
||||
4. Check the option to use **Self-Hosted Server Configuration**.<br><br>
|
||||
<img width="600" height="160" alt="image" src="https://github.com/user-attachments/assets/1e455dc3-4087-4f37-a544-1ff9f8789383" /><br><br>
|
||||
|
||||
### Web Client
|
||||
See [CrossDesk Web Client](https://github.com/kunkundi/crossdesk-web-client)。
|
||||
|
||||
# FAQ
|
||||
See [FAQ](https://github.com/kunkundi/crosssesk/blob/self-hosted-server/docs/FAQ.md) .
|
||||
|
||||
@@ -78,14 +78,6 @@ EOF
|
||||
|
||||
echo ".app created successfully."
|
||||
|
||||
SIGN_IDENTITY="${CROSSDESK_SIGN_IDENTITY:--}"
|
||||
|
||||
if command -v codesign &> /dev/null; then
|
||||
codesign --force --deep --sign "$SIGN_IDENTITY" "${APP_BUNDLE}" 2>/dev/null || {
|
||||
echo "Warning: Code signing failed. The app may not work properly."
|
||||
}
|
||||
fi
|
||||
|
||||
echo "building pkg..."
|
||||
pkgbuild \
|
||||
--identifier "${IDENTIFIER}" \
|
||||
@@ -98,38 +90,13 @@ mkdir -p scripts
|
||||
|
||||
cat > scripts/postinstall <<'EOF'
|
||||
#!/bin/bash
|
||||
|
||||
IDENTIFIER="cn.crossdesk.app"
|
||||
APP_NAME="CrossDesk"
|
||||
APP_PATH="/Applications/${APP_NAME}.app"
|
||||
|
||||
USER_HOME=$( /usr/bin/stat -f "%Su" /dev/console )
|
||||
HOME_DIR=$( /usr/bin/dscl . -read /Users/$USER_HOME NFSHomeDirectory | awk '{print $2}' )
|
||||
|
||||
DEST="$HOME_DIR/Library/Application Support/CrossDesk/certs"
|
||||
mkdir -p "$DEST"
|
||||
if [ -d "/Library/Application Support/CrossDesk/certs" ]; then
|
||||
cp -R "/Library/Application Support/CrossDesk/certs/"* "$DEST/" 2>/dev/null || true
|
||||
fi
|
||||
|
||||
if [ -d "$APP_PATH" ]; then
|
||||
touch "$APP_PATH"
|
||||
/System/Library/Frameworks/CoreServices.framework/Frameworks/LaunchServices.framework/Support/lsregister -f "$APP_PATH" 2>/dev/null || true
|
||||
|
||||
USER_ID=$(dscl . -read /Users/$USER_HOME UniqueID 2>/dev/null | awk '{print $2}')
|
||||
|
||||
if [ -n "$USER_ID" ]; then
|
||||
sleep 1
|
||||
launchctl asuser $USER_ID open "$APP_PATH" 2>/dev/null &
|
||||
APP_PID=$!
|
||||
sleep 3
|
||||
kill "$APP_PID" 2>/dev/null || pkill -f "${APP_NAME}" 2>/dev/null || true
|
||||
fi
|
||||
|
||||
open "x-apple.systempreferences:com.apple.preference.security?Privacy_ScreenCapture" 2>/dev/null &
|
||||
sleep 1
|
||||
open "x-apple.systempreferences:com.apple.preference.security?Privacy_Accessibility" 2>/dev/null &
|
||||
fi
|
||||
mkdir -p "$DEST"
|
||||
cp -R "/Library/Application Support/CrossDesk/certs/"* "$DEST/"
|
||||
|
||||
exit 0
|
||||
EOF
|
||||
|
||||
@@ -78,14 +78,6 @@ EOF
|
||||
|
||||
echo ".app created successfully."
|
||||
|
||||
SIGN_IDENTITY="${CROSSDESK_SIGN_IDENTITY:--}"
|
||||
|
||||
if command -v codesign &> /dev/null; then
|
||||
codesign --force --deep --sign "$SIGN_IDENTITY" "${APP_BUNDLE}" 2>/dev/null || {
|
||||
echo "Warning: Code signing failed. The app may not work properly."
|
||||
}
|
||||
fi
|
||||
|
||||
echo "building pkg..."
|
||||
pkgbuild \
|
||||
--identifier "${IDENTIFIER}" \
|
||||
@@ -98,38 +90,13 @@ mkdir -p scripts
|
||||
|
||||
cat > scripts/postinstall <<'EOF'
|
||||
#!/bin/bash
|
||||
|
||||
IDENTIFIER="cn.crossdesk.app"
|
||||
APP_NAME="CrossDesk"
|
||||
APP_PATH="/Applications/${APP_NAME}.app"
|
||||
|
||||
USER_HOME=$( /usr/bin/stat -f "%Su" /dev/console )
|
||||
HOME_DIR=$( /usr/bin/dscl . -read /Users/$USER_HOME NFSHomeDirectory | awk '{print $2}' )
|
||||
|
||||
DEST="$HOME_DIR/Library/Application Support/CrossDesk/certs"
|
||||
mkdir -p "$DEST"
|
||||
if [ -d "/Library/Application Support/CrossDesk/certs" ]; then
|
||||
cp -R "/Library/Application Support/CrossDesk/certs/"* "$DEST/" 2>/dev/null || true
|
||||
fi
|
||||
|
||||
if [ -d "$APP_PATH" ]; then
|
||||
touch "$APP_PATH"
|
||||
/System/Library/Frameworks/CoreServices.framework/Frameworks/LaunchServices.framework/Support/lsregister -f "$APP_PATH" 2>/dev/null || true
|
||||
|
||||
USER_ID=$(dscl . -read /Users/$USER_HOME UniqueID 2>/dev/null | awk '{print $2}')
|
||||
|
||||
if [ -n "$USER_ID" ]; then
|
||||
sleep 1
|
||||
launchctl asuser $USER_ID open "$APP_PATH" 2>/dev/null &
|
||||
APP_PID=$!
|
||||
sleep 3
|
||||
kill "$APP_PID" 2>/dev/null || pkill -f "${APP_NAME}" 2>/dev/null || true
|
||||
fi
|
||||
|
||||
open "x-apple.systempreferences:com.apple.preference.security?Privacy_ScreenCapture" 2>/dev/null &
|
||||
sleep 1
|
||||
open "x-apple.systempreferences:com.apple.preference.security?Privacy_Accessibility" 2>/dev/null &
|
||||
fi
|
||||
mkdir -p "$DEST"
|
||||
cp -R "/Library/Application Support/CrossDesk/certs/"* "$DEST/"
|
||||
|
||||
exit 0
|
||||
EOF
|
||||
|
||||
@@ -1,56 +0,0 @@
|
||||
#!/bin/bash
|
||||
# Helper script to sign macOS app with different certificate options
|
||||
# Usage: ./sign_app.sh <app_path> [certificate_name]
|
||||
|
||||
APP_PATH="$1"
|
||||
CERT_NAME="$2"
|
||||
|
||||
if [ -z "$APP_PATH" ]; then
|
||||
echo "Usage: $0 <app_path> [certificate_name]"
|
||||
echo ""
|
||||
echo "Examples:"
|
||||
echo " $0 CrossDesk.app # Ad-hoc signing"
|
||||
echo " $0 CrossDesk.app \"CrossDesk Dev\" # Sign with certificate name"
|
||||
echo " $0 CrossDesk.app \"-\" # Explicit ad-hoc signing"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ ! -d "$APP_PATH" ]; then
|
||||
echo "Error: App bundle not found: $APP_PATH"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "$CERT_NAME" ]; then
|
||||
# Default to ad-hoc signing
|
||||
CERT_NAME="-"
|
||||
echo "Using ad-hoc signing (no certificate specified)"
|
||||
else
|
||||
echo "Using certificate: $CERT_NAME"
|
||||
fi
|
||||
|
||||
# Check if codesign is available
|
||||
if ! command -v codesign &> /dev/null; then
|
||||
echo "Error: codesign not found. Please install Xcode Command Line Tools."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
# Sign the app
|
||||
echo "Signing $APP_PATH..."
|
||||
codesign --force --deep --sign "$CERT_NAME" "$APP_PATH"
|
||||
|
||||
if [ $? -eq 0 ]; then
|
||||
echo "Signing successful!"
|
||||
|
||||
# Verify signature
|
||||
echo ""
|
||||
echo "Verifying signature..."
|
||||
codesign -dv --verbose=4 "$APP_PATH" 2>&1 | head -20
|
||||
|
||||
echo ""
|
||||
echo "Checking entitlements..."
|
||||
codesign -d --entitlements - "$APP_PATH" 2>&1 || echo "No entitlements found"
|
||||
else
|
||||
echo "Error: Signing failed"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
@@ -1,60 +0,0 @@
|
||||
#!/bin/bash
|
||||
# macOS uninstall script for CrossDesk
|
||||
# This script removes the application and cleans up permissions
|
||||
|
||||
APP_NAME="CrossDesk"
|
||||
IDENTIFIER="cn.crossdesk.app"
|
||||
|
||||
echo "Uninstalling ${APP_NAME}..."
|
||||
|
||||
# Remove application
|
||||
if [ -d "/Applications/${APP_NAME}.app" ]; then
|
||||
echo "Removing application..."
|
||||
rm -rf "/Applications/${APP_NAME}.app"
|
||||
fi
|
||||
|
||||
# Remove user data
|
||||
USER_HOME=$(/usr/bin/stat -f "%Su" /dev/console 2>/dev/null)
|
||||
if [ -n "$USER_HOME" ]; then
|
||||
HOME_DIR=$(/usr/bin/dscl . -read "/Users/$USER_HOME" NFSHomeDirectory 2>/dev/null | awk '{print $2}')
|
||||
if [ -n "$HOME_DIR" ]; then
|
||||
echo "Removing user data..."
|
||||
rm -rf "$HOME_DIR/Library/Application Support/${APP_NAME}"
|
||||
rm -rf "$HOME_DIR/Library/Logs/${APP_NAME}"
|
||||
rm -rf "$HOME_DIR/Library/Caches/${APP_NAME}"
|
||||
rm -rf "$HOME_DIR/Library/Preferences/${IDENTIFIER}.plist"
|
||||
rm -rf "$HOME_DIR/Library/LaunchAgents/${APP_NAME}.plist"
|
||||
fi
|
||||
fi
|
||||
|
||||
# Remove system-wide certificates
|
||||
if [ -d "/Library/Application Support/${APP_NAME}/certs" ]; then
|
||||
echo "Removing system certificates..."
|
||||
rm -rf "/Library/Application Support/${APP_NAME}"
|
||||
fi
|
||||
|
||||
# Clean up permissions using tccutil (macOS 10.14+)
|
||||
# Note: This requires user interaction and may not work in all cases
|
||||
if command -v tccutil &> /dev/null; then
|
||||
echo "Cleaning up permissions..."
|
||||
# Reset screen recording permission
|
||||
tccutil reset ScreenCapture "$IDENTIFIER" 2>/dev/null || true
|
||||
# Reset accessibility permission
|
||||
tccutil reset Accessibility "$IDENTIFIER" 2>/dev/null || true
|
||||
# Reset camera permission
|
||||
tccutil reset Camera "$IDENTIFIER" 2>/dev/null || true
|
||||
# Reset microphone permission
|
||||
tccutil reset Microphone "$IDENTIFIER" 2>/dev/null || true
|
||||
echo "Permissions cleaned up. You may need to manually remove them from System Settings > Privacy & Security if they still appear."
|
||||
else
|
||||
echo "tccutil not available. Please manually remove permissions from System Settings > Privacy & Security"
|
||||
fi
|
||||
|
||||
echo "Uninstallation complete."
|
||||
echo ""
|
||||
echo "Note: If permissions still appear in System Settings, please remove them manually:"
|
||||
echo " 1. Open System Settings > Privacy & Security"
|
||||
echo " 2. Remove ${APP_NAME} from Screen Recording and Accessibility"
|
||||
|
||||
exit 0
|
||||
|
||||
@@ -199,12 +199,10 @@ int ScreenCapturerSckImpl::Init(const int fps, cb_desktop_data cb) {
|
||||
|
||||
dispatch_semaphore_t sema = dispatch_semaphore_create(0);
|
||||
__block SCShareableContent *content = nil;
|
||||
__block NSError *capture_error = nil;
|
||||
|
||||
[SCShareableContent
|
||||
getShareableContentWithCompletionHandler:^(SCShareableContent *result, NSError *error) {
|
||||
if (error) {
|
||||
capture_error = error;
|
||||
NSLog(@"Failed to get shareable content: %@", error);
|
||||
} else {
|
||||
content = result;
|
||||
@@ -213,15 +211,8 @@ int ScreenCapturerSckImpl::Init(const int fps, cb_desktop_data cb) {
|
||||
}];
|
||||
dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
|
||||
|
||||
if (capture_error) {
|
||||
NSString *error_desc = capture_error.localizedDescription ?: @"Unknown error";
|
||||
LOG_ERROR("Failed to get shareable content: {}",
|
||||
std::string([error_desc UTF8String]));
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (!content || content.displays.count == 0) {
|
||||
LOG_ERROR("Failed to get display info: content is nil or no displays available");
|
||||
LOG_ERROR("Failed to get display info");
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user