Compare commits
4 Commits
pr3023
...
open_a_ter
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6a79f44ecc | ||
|
|
d66d708e55 | ||
|
|
0a16272000 | ||
|
|
93ac6a347e |
6
BUILD.md
6
BUILD.md
@@ -270,10 +270,10 @@ install` must be run as root)._
|
|||||||
|
|
||||||
#### Option 2: Use prebuilt server
|
#### Option 2: Use prebuilt server
|
||||||
|
|
||||||
- [`scrcpy-server-v1.22`][direct-scrcpy-server]
|
- [`scrcpy-server-v1.21`][direct-scrcpy-server]
|
||||||
_(SHA-256: c05d273eec7533c0e106282e0254cf04e7f5e8f0c2920ca39448865fab2a419b)_
|
_(SHA-256: dbcccab523ee26796e55ea33652649e4b7af498edae9aa75e4d4d7869c0ab848)_
|
||||||
|
|
||||||
[direct-scrcpy-server]: https://github.com/Genymobile/scrcpy/releases/download/v1.22/scrcpy-server-v1.22
|
[direct-scrcpy-server]: https://github.com/Genymobile/scrcpy/releases/download/v1.21/scrcpy-server-v1.21
|
||||||
|
|
||||||
Download the prebuilt server somewhere, and specify its path during the Meson
|
Download the prebuilt server somewhere, and specify its path during the Meson
|
||||||
configuration:
|
configuration:
|
||||||
|
|||||||
24
FAQ.md
24
FAQ.md
@@ -12,7 +12,7 @@ Here are the common reported problems and their status.
|
|||||||
|
|
||||||
In that case, it will print this error:
|
In that case, it will print this error:
|
||||||
|
|
||||||
> ERROR: "adb get-serialno" returned with value 1
|
> ERROR: "adb push" returned with value 1
|
||||||
|
|
||||||
This is typically not a bug in _scrcpy_, but a problem in your environment.
|
This is typically not a bug in _scrcpy_, but a problem in your environment.
|
||||||
|
|
||||||
@@ -32,38 +32,28 @@ in the release, so it should work out-of-the-box.
|
|||||||
|
|
||||||
### Device unauthorized
|
### Device unauthorized
|
||||||
|
|
||||||
|
Check [stackoverflow][device-unauthorized].
|
||||||
> error: device unauthorized.
|
|
||||||
> This adb server's $ADB_VENDOR_KEYS is not set
|
|
||||||
> Try 'adb kill-server' if that seems wrong.
|
|
||||||
> Otherwise check for a confirmation dialog on your device.
|
|
||||||
|
|
||||||
When connecting, a popup should open on the device. You must authorize USB
|
|
||||||
debugging.
|
|
||||||
|
|
||||||
If it does not open, check [stackoverflow][device-unauthorized].
|
|
||||||
|
|
||||||
[device-unauthorized]: https://stackoverflow.com/questions/23081263/adb-android-device-unauthorized
|
[device-unauthorized]: https://stackoverflow.com/questions/23081263/adb-android-device-unauthorized
|
||||||
|
|
||||||
|
|
||||||
### Device not detected
|
### Device not detected
|
||||||
|
|
||||||
> error: no devices/emulators found
|
> adb: error: failed to get feature set: no devices/emulators found
|
||||||
|
|
||||||
Check that you correctly enabled [adb debugging][enable-adb].
|
Check that you correctly enabled [adb debugging][enable-adb].
|
||||||
|
|
||||||
If your device is not detected, you may need some [drivers] (on Windows). There is a separate [USB driver for Google devices][google-usb-driver].
|
If your device is not detected, you may need some [drivers] (on Windows).
|
||||||
|
|
||||||
[enable-adb]: https://developer.android.com/studio/command-line/adb.html#Enabling
|
[enable-adb]: https://developer.android.com/studio/command-line/adb.html#Enabling
|
||||||
[drivers]: https://developer.android.com/studio/run/oem-usb.html
|
[drivers]: https://developer.android.com/studio/run/oem-usb.html
|
||||||
[google-usb-driver]: https://developer.android.com/studio/run/win-usb
|
|
||||||
|
|
||||||
|
|
||||||
### Several devices connected
|
### Several devices connected
|
||||||
|
|
||||||
If several devices are connected, you will encounter this error:
|
If several devices are connected, you will encounter this error:
|
||||||
|
|
||||||
> error: more than one device/emulator
|
> adb: error: failed to get feature set: more than one device/emulator
|
||||||
|
|
||||||
the identifier of the device you want to mirror must be provided:
|
the identifier of the device you want to mirror must be provided:
|
||||||
|
|
||||||
@@ -71,7 +61,7 @@ the identifier of the device you want to mirror must be provided:
|
|||||||
scrcpy -s 01234567890abcdef
|
scrcpy -s 01234567890abcdef
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that if your device is connected over TCP/IP, you might get this message:
|
Note that if your device is connected over TCP/IP, you'll get this message:
|
||||||
|
|
||||||
> adb: error: more than one device/emulator
|
> adb: error: more than one device/emulator
|
||||||
> ERROR: "adb reverse" returned with value 1
|
> ERROR: "adb reverse" returned with value 1
|
||||||
@@ -305,4 +295,4 @@ This FAQ is available in other languages:
|
|||||||
|
|
||||||
- [Italiano (Italiano, `it`) - v1.19](FAQ.it.md)
|
- [Italiano (Italiano, `it`) - v1.19](FAQ.it.md)
|
||||||
- [한국어 (Korean, `ko`) - v1.11](FAQ.ko.md)
|
- [한국어 (Korean, `ko`) - v1.11](FAQ.ko.md)
|
||||||
- [简体中文 (Simplified Chinese, `zh-Hans`) - v1.22](FAQ.zh-Hans.md)
|
- [简体中文 (Simplified Chinese, `zh-Hans`) - v1.18](FAQ.zh-Hans.md)
|
||||||
|
|||||||
@@ -1,12 +1,7 @@
|
|||||||
_Only the original [FAQ.md](FAQ.md) is guaranteed to be up-to-date._
|
只有原版的[FAQ](FAQ.md)会保持更新。
|
||||||
|
本文根据[d6aaa5]翻译。
|
||||||
|
|
||||||
_只有原版的 [FAQ.md](FAQ.md)是保证最新的。_
|
[d6aaa5]:https://github.com/Genymobile/scrcpy/blob/d6aaa5bf9aa3710660c683b6e3e0ed971ee44af5/FAQ.md
|
||||||
|
|
||||||
Current version is based on [28054cd]
|
|
||||||
|
|
||||||
本文根据[28054cd]进行翻译。
|
|
||||||
|
|
||||||
[28054cd]: https://github.com/Genymobile/scrcpy/blob/28054cd471f848733e11372c9d745cd5d71e6ce7/FAQ.md
|
|
||||||
|
|
||||||
# 常见问题
|
# 常见问题
|
||||||
|
|
||||||
@@ -18,7 +13,7 @@ Current version is based on [28054cd]
|
|||||||
|
|
||||||
在这种情况中,将会输出这个错误:
|
在这种情况中,将会输出这个错误:
|
||||||
|
|
||||||
> ERROR: "adb get-serialno" returned with value 1
|
> ERROR: "adb push" returned with value 1
|
||||||
|
|
||||||
这通常不是 _scrcpy_ 的bug,而是你的环境的问题。
|
这通常不是 _scrcpy_ 的bug,而是你的环境的问题。
|
||||||
|
|
||||||
@@ -38,37 +33,28 @@ adb devices
|
|||||||
|
|
||||||
### 设备未授权
|
### 设备未授权
|
||||||
|
|
||||||
|
参见这里 [stackoverflow][device-unauthorized].
|
||||||
> error: device unauthorized.
|
|
||||||
> This adb server's $ADB_VENDOR_KEYS is not set
|
|
||||||
> Try 'adb kill-server' if that seems wrong.
|
|
||||||
> Otherwise check for a confirmation dialog on your device.
|
|
||||||
|
|
||||||
连接时,在设备上应该会打开一个弹出窗口。 您必须授权 USB 调试。
|
|
||||||
|
|
||||||
如果没有打开,参见[stackoverflow][device-unauthorized].
|
|
||||||
|
|
||||||
[device-unauthorized]: https://stackoverflow.com/questions/23081263/adb-android-device-unauthorized
|
[device-unauthorized]: https://stackoverflow.com/questions/23081263/adb-android-device-unauthorized
|
||||||
|
|
||||||
|
|
||||||
### 未检测到设备
|
### 未检测到设备
|
||||||
|
|
||||||
> error: no devices/emulators found
|
> adb: error: failed to get feature set: no devices/emulators found
|
||||||
|
|
||||||
确认已经正确启用 [adb debugging][enable-adb].
|
确认已经正确启用 [adb debugging][enable-adb].
|
||||||
|
|
||||||
如果你的设备没有被检测到,你可能需要一些[驱动][drivers] (在 Windows上)。这里有一个单独的 [适用于Google设备的USB驱动][google-usb-driver].
|
如果你的设备没有被检测到,你可能需要一些[驱动][drivers] (在 Windows上).
|
||||||
|
|
||||||
[enable-adb]: https://developer.android.com/studio/command-line/adb.html#Enabling
|
[enable-adb]: https://developer.android.com/studio/command-line/adb.html#Enabling
|
||||||
[drivers]: https://developer.android.com/studio/run/oem-usb.html
|
[drivers]: https://developer.android.com/studio/run/oem-usb.html
|
||||||
[google-usb-driver]: https://developer.android.com/studio/run/win-usb
|
|
||||||
|
|
||||||
|
|
||||||
### 已连接多个设备
|
### 已连接多个设备
|
||||||
|
|
||||||
如果连接了多个设备,您将遇到以下错误:
|
如果连接了多个设备,您将遇到以下错误:
|
||||||
|
|
||||||
> error: more than one device/emulator
|
> adb: error: failed to get feature set: more than one device/emulator
|
||||||
|
|
||||||
必须提供要镜像的设备的标识符:
|
必须提供要镜像的设备的标识符:
|
||||||
|
|
||||||
@@ -104,19 +90,19 @@ scrcpy
|
|||||||
### 设备断开连接
|
### 设备断开连接
|
||||||
|
|
||||||
如果 _scrcpy_ 在警告“设备连接断开”的情况下自动中止,那就意味着`adb`连接已经断开了。
|
如果 _scrcpy_ 在警告“设备连接断开”的情况下自动中止,那就意味着`adb`连接已经断开了。
|
||||||
|
请尝试使用另一条USB线或者电脑上的另一个USB接口。
|
||||||
请尝试使用另一条USB线或者电脑上的另一个USB接口。请参看 [#281] 和 [#283]。
|
请参看 [#281] 和 [#283]。
|
||||||
|
|
||||||
[#281]: https://github.com/Genymobile/scrcpy/issues/281
|
[#281]: https://github.com/Genymobile/scrcpy/issues/281
|
||||||
[#283]: https://github.com/Genymobile/scrcpy/issues/283
|
[#283]: https://github.com/Genymobile/scrcpy/issues/283
|
||||||
|
|
||||||
|
|
||||||
## 控制相关问题
|
## 控制相关问题
|
||||||
|
|
||||||
### 鼠标和键盘不起作用
|
### 鼠标和键盘不起作用
|
||||||
|
|
||||||
|
|
||||||
在某些设备上,您可能需要启用一个选项以允许 [模拟输入][simulating input]。
|
在某些设备上,您可能需要启用一个选项以允许 [模拟输入][simulating input]。
|
||||||
|
|
||||||
在开发者选项中,打开:
|
在开发者选项中,打开:
|
||||||
|
|
||||||
> **USB调试 (安全设置)**
|
> **USB调试 (安全设置)**
|
||||||
@@ -129,12 +115,10 @@ scrcpy
|
|||||||
|
|
||||||
可输入的文本[被限制为ASCII字符][text-input]。也可以用一些小技巧输入一些[带重音符号的字符][accented-characters],但是仅此而已。参见[#37]。
|
可输入的文本[被限制为ASCII字符][text-input]。也可以用一些小技巧输入一些[带重音符号的字符][accented-characters],但是仅此而已。参见[#37]。
|
||||||
|
|
||||||
自 Linux 上的 scrcpy v1.20 之后,可以模拟[物理键盘][hid] (HID)。
|
|
||||||
|
|
||||||
[text-input]: https://github.com/Genymobile/scrcpy/issues?q=is%3Aopen+is%3Aissue+label%3Aunicode
|
[text-input]: https://github.com/Genymobile/scrcpy/issues?q=is%3Aopen+is%3Aissue+label%3Aunicode
|
||||||
[accented-characters]: https://blog.rom1v.com/2018/03/introducing-scrcpy/#handle-accented-characters
|
[accented-characters]: https://blog.rom1v.com/2018/03/introducing-scrcpy/#handle-accented-characters
|
||||||
[#37]: https://github.com/Genymobile/scrcpy/issues/37
|
[#37]: https://github.com/Genymobile/scrcpy/issues/37
|
||||||
[hid]: README.md#physical-keyboard-simulation-hid
|
|
||||||
|
|
||||||
|
|
||||||
## 客户端相关问题
|
## 客户端相关问题
|
||||||
@@ -145,6 +129,7 @@ scrcpy
|
|||||||
|
|
||||||
[#40]: https://github.com/Genymobile/scrcpy/issues/40
|
[#40]: https://github.com/Genymobile/scrcpy/issues/40
|
||||||
|
|
||||||
|
|
||||||
为了提升降尺度的质量,如果渲染器是OpenGL并且支持mip映射,就会自动开启三线性过滤。
|
为了提升降尺度的质量,如果渲染器是OpenGL并且支持mip映射,就会自动开启三线性过滤。
|
||||||
|
|
||||||
在Windows上,你可能希望强制使用OpenGL:
|
在Windows上,你可能希望强制使用OpenGL:
|
||||||
@@ -192,7 +177,6 @@ scrcpy
|
|||||||
## 崩溃
|
## 崩溃
|
||||||
|
|
||||||
### 异常
|
### 异常
|
||||||
|
|
||||||
可能有很多原因。一个常见的原因是您的设备无法按给定清晰度进行编码:
|
可能有很多原因。一个常见的原因是您的设备无法按给定清晰度进行编码:
|
||||||
|
|
||||||
> ```
|
> ```
|
||||||
@@ -220,40 +204,12 @@ scrcpy -m 1024
|
|||||||
scrcpy -m 800
|
scrcpy -m 800
|
||||||
```
|
```
|
||||||
|
|
||||||
自 scrcpy v1.22以来,scrcpy 会自动在失败前以更低的分辨率重试。这种行为可以用`--no-downsize-on-error`关闭。
|
|
||||||
|
|
||||||
你也可以尝试另一种 [编码器](README.md#encoder)。
|
你也可以尝试另一种 [编码器](README.md#encoder)。
|
||||||
|
|
||||||
|
|
||||||
如果您在 Android 12 上遇到此异常,则只需升级到 scrcpy >= 1.18 (见 [#2129]):
|
|
||||||
|
|
||||||
```
|
|
||||||
> ERROR: Exception on thread Thread[main,5,main]
|
|
||||||
java.lang.AssertionError: java.lang.reflect.InvocationTargetException
|
|
||||||
at com.genymobile.scrcpy.wrappers.SurfaceControl.setDisplaySurface(SurfaceControl.java:75)
|
|
||||||
...
|
|
||||||
Caused by: java.lang.reflect.InvocationTargetException
|
|
||||||
at java.lang.reflect.Method.invoke(Native Method)
|
|
||||||
at com.genymobile.scrcpy.wrappers.SurfaceControl.setDisplaySurface(SurfaceControl.java:73)
|
|
||||||
... 7 more
|
|
||||||
Caused by: java.lang.IllegalArgumentException: displayToken must not be null
|
|
||||||
at android.view.SurfaceControl$Transaction.setDisplaySurface(SurfaceControl.java:3067)
|
|
||||||
at android.view.SurfaceControl.setDisplaySurface(SurfaceControl.java:2147)
|
|
||||||
... 9 more
|
|
||||||
```
|
|
||||||
|
|
||||||
[#2129]: https://github.com/Genymobile/scrcpy/issues/2129
|
|
||||||
|
|
||||||
|
|
||||||
## Windows命令行
|
## Windows命令行
|
||||||
|
|
||||||
从 v1.22 开始,增加了一个“快捷方式”,可以直接在 scrcpy 目录打开一个终端。双击`open_a_terminal_here.bat`,然后输入你的命令。 例如:
|
一些Windows用户不熟悉命令行。以下是如何打开终端并带参数执行`scrcpy`:
|
||||||
|
|
||||||
```
|
|
||||||
scrcpy --record file.mkv
|
|
||||||
```
|
|
||||||
|
|
||||||
您也可以打开终端并手动转到 scrcpy 文件夹:
|
|
||||||
|
|
||||||
1. 按下 <kbd>Windows</kbd>+<kbd>r</kbd>,打开一个对话框。
|
1. 按下 <kbd>Windows</kbd>+<kbd>r</kbd>,打开一个对话框。
|
||||||
2. 输入 `cmd` 并按 <kbd>Enter</kbd>,这样就打开了一个终端。
|
2. 输入 `cmd` 并按 <kbd>Enter</kbd>,这样就打开了一个终端。
|
||||||
@@ -277,7 +233,7 @@ scrcpy --record file.mkv
|
|||||||
scrcpy --prefer-text --turn-screen-off --stay-awake
|
scrcpy --prefer-text --turn-screen-off --stay-awake
|
||||||
```
|
```
|
||||||
|
|
||||||
然后只需双击刚刚创建的文件。
|
然后双击刚刚创建的文件。
|
||||||
|
|
||||||
你也可以编辑 `scrcpy-console.bat` 或者 `scrcpy-noconsole.vbs`(的副本)来添加参数。
|
你也可以编辑 `scrcpy-console.bat` 或者 `scrcpy-noconsole.vbs`(的副本)来添加参数。
|
||||||
|
|
||||||
|
|||||||
1016
README.de.md
1016
README.de.md
File diff suppressed because it is too large
Load Diff
67
README.md
67
README.md
@@ -1,9 +1,7 @@
|
|||||||
# scrcpy (v1.22)
|
# scrcpy (v1.21)
|
||||||
|
|
||||||
<img src="data/icon.svg" width="128" height="128" alt="scrcpy" align="right" />
|
<img src="data/icon.svg" width="128" height="128" alt="scrcpy" align="right" />
|
||||||
|
|
||||||
_pronounced "**scr**een **c**o**py**"_
|
|
||||||
|
|
||||||
[Read in another language](#translations)
|
[Read in another language](#translations)
|
||||||
|
|
||||||
This application provides display and control of Android devices connected via
|
This application provides display and control of Android devices connected via
|
||||||
@@ -35,7 +33,6 @@ Its features include:
|
|||||||
(Linux-only)
|
(Linux-only)
|
||||||
- [physical mouse simulation (HID)](#physical-mouse-simulation-hid)
|
- [physical mouse simulation (HID)](#physical-mouse-simulation-hid)
|
||||||
(Linux-only)
|
(Linux-only)
|
||||||
- [OTG mode](#otg) (Linux-only)
|
|
||||||
- and more…
|
- and more…
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
@@ -76,12 +73,6 @@ On Debian and Ubuntu:
|
|||||||
apt install scrcpy
|
apt install scrcpy
|
||||||
```
|
```
|
||||||
|
|
||||||
On Arch Linux:
|
|
||||||
|
|
||||||
```
|
|
||||||
pacman -S scrcpy
|
|
||||||
```
|
|
||||||
|
|
||||||
A [Snap] package is available: [`scrcpy`][snap-link].
|
A [Snap] package is available: [`scrcpy`][snap-link].
|
||||||
|
|
||||||
[snap-link]: https://snapstats.org/snaps/scrcpy
|
[snap-link]: https://snapstats.org/snaps/scrcpy
|
||||||
@@ -93,6 +84,10 @@ For Fedora, a [COPR] package is available: [`scrcpy`][copr-link].
|
|||||||
[COPR]: https://fedoraproject.org/wiki/Category:Copr
|
[COPR]: https://fedoraproject.org/wiki/Category:Copr
|
||||||
[copr-link]: https://copr.fedorainfracloud.org/coprs/zeno/scrcpy/
|
[copr-link]: https://copr.fedorainfracloud.org/coprs/zeno/scrcpy/
|
||||||
|
|
||||||
|
For Arch Linux, an [AUR] package is available: [`scrcpy`][aur-link].
|
||||||
|
|
||||||
|
[AUR]: https://wiki.archlinux.org/index.php/Arch_User_Repository
|
||||||
|
[aur-link]: https://aur.archlinux.org/packages/scrcpy/
|
||||||
|
|
||||||
For Gentoo, an [Ebuild] is available: [`scrcpy/`][ebuild-link].
|
For Gentoo, an [Ebuild] is available: [`scrcpy/`][ebuild-link].
|
||||||
|
|
||||||
@@ -108,10 +103,10 @@ process][BUILD_simple]).
|
|||||||
For Windows, for simplicity, a prebuilt archive with all the dependencies
|
For Windows, for simplicity, a prebuilt archive with all the dependencies
|
||||||
(including `adb`) is available:
|
(including `adb`) is available:
|
||||||
|
|
||||||
- [`scrcpy-win64-v1.22.zip`][direct-win64]
|
- [`scrcpy-win64-v1.21.zip`][direct-win64]
|
||||||
_(SHA-256: ce4d9b8cc761e29862c4a72d8ad6f538bdd1f1831d15fd1f36633cd3b403db82)_
|
_(SHA-256: fdab0c1421353b592a9bbcebd6e252675eadccca65cca8105686feaa9c1ded53)_
|
||||||
|
|
||||||
[direct-win64]: https://github.com/Genymobile/scrcpy/releases/download/v1.22/scrcpy-win64-v1.22.zip
|
[direct-win64]: https://github.com/Genymobile/scrcpy/releases/download/v1.21/scrcpy-win64-v1.21.zip
|
||||||
|
|
||||||
It is also available in [Chocolatey]:
|
It is also available in [Chocolatey]:
|
||||||
|
|
||||||
@@ -852,36 +847,6 @@ Special capture keys, either <kbd>Alt</kbd> or <kbd>Super</kbd>, toggle
|
|||||||
the mouse back to the computer.
|
the mouse back to the computer.
|
||||||
|
|
||||||
|
|
||||||
#### OTG
|
|
||||||
|
|
||||||
It is possible to run _scrcpy_ with only physical keyboard and mouse simulation
|
|
||||||
(HID), as if the computer keyboard and mouse were plugged directly to the device
|
|
||||||
via an OTG cable.
|
|
||||||
|
|
||||||
In this mode, _adb_ (USB debugging) is not necessary, and mirroring is disabled.
|
|
||||||
|
|
||||||
To enable OTG mode:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --otg
|
|
||||||
# Pass the serial if several USB devices are available
|
|
||||||
scrcpy --otg -s 0123456789abcdef
|
|
||||||
```
|
|
||||||
|
|
||||||
It is possible to enable only HID keyboard or HID mouse:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --otg --hid-keyboard # keyboard only
|
|
||||||
scrcpy --otg --hid-mouse # mouse only
|
|
||||||
scrcpy --otg --hid-keyboard --hid-mouse # keyboard and mouse
|
|
||||||
# for convenience, enable both by default
|
|
||||||
scrcpy --otg # keyboard and mouse
|
|
||||||
```
|
|
||||||
|
|
||||||
Like `--hid-keyboard` and `--hid-mouse`, it only works if the device is
|
|
||||||
connected by USB, and is currently only supported on Linux.
|
|
||||||
|
|
||||||
|
|
||||||
#### Text injection preference
|
#### Text injection preference
|
||||||
|
|
||||||
There are two kinds of [events][textevents] generated when typing text:
|
There are two kinds of [events][textevents] generated when typing text:
|
||||||
@@ -1002,7 +967,7 @@ _<kbd>[Super]</kbd> is typically the <kbd>Windows</kbd> or <kbd>Cmd</kbd> key._
|
|||||||
| Click on `HOME` | <kbd>MOD</kbd>+<kbd>h</kbd> \| _Middle-click_
|
| Click on `HOME` | <kbd>MOD</kbd>+<kbd>h</kbd> \| _Middle-click_
|
||||||
| Click on `BACK` | <kbd>MOD</kbd>+<kbd>b</kbd> \| _Right-click²_
|
| Click on `BACK` | <kbd>MOD</kbd>+<kbd>b</kbd> \| _Right-click²_
|
||||||
| Click on `APP_SWITCH` | <kbd>MOD</kbd>+<kbd>s</kbd> \| _4th-click³_
|
| Click on `APP_SWITCH` | <kbd>MOD</kbd>+<kbd>s</kbd> \| _4th-click³_
|
||||||
| Click on `MENU` (unlock screen)⁴ | <kbd>MOD</kbd>+<kbd>m</kbd>
|
| Click on `MENU` (unlock screen) | <kbd>MOD</kbd>+<kbd>m</kbd>
|
||||||
| Click on `VOLUME_UP` | <kbd>MOD</kbd>+<kbd>↑</kbd> _(up)_
|
| Click on `VOLUME_UP` | <kbd>MOD</kbd>+<kbd>↑</kbd> _(up)_
|
||||||
| Click on `VOLUME_DOWN` | <kbd>MOD</kbd>+<kbd>↓</kbd> _(down)_
|
| Click on `VOLUME_DOWN` | <kbd>MOD</kbd>+<kbd>↓</kbd> _(down)_
|
||||||
| Click on `POWER` | <kbd>MOD</kbd>+<kbd>p</kbd>
|
| Click on `POWER` | <kbd>MOD</kbd>+<kbd>p</kbd>
|
||||||
@@ -1013,9 +978,9 @@ _<kbd>[Super]</kbd> is typically the <kbd>Windows</kbd> or <kbd>Cmd</kbd> key._
|
|||||||
| Expand notification panel | <kbd>MOD</kbd>+<kbd>n</kbd> \| _5th-click³_
|
| Expand notification panel | <kbd>MOD</kbd>+<kbd>n</kbd> \| _5th-click³_
|
||||||
| Expand settings panel | <kbd>MOD</kbd>+<kbd>n</kbd>+<kbd>n</kbd> \| _Double-5th-click³_
|
| Expand settings panel | <kbd>MOD</kbd>+<kbd>n</kbd>+<kbd>n</kbd> \| _Double-5th-click³_
|
||||||
| Collapse panels | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>n</kbd>
|
| Collapse panels | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>n</kbd>
|
||||||
| Copy to clipboard⁵ | <kbd>MOD</kbd>+<kbd>c</kbd>
|
| Copy to clipboard⁴ | <kbd>MOD</kbd>+<kbd>c</kbd>
|
||||||
| Cut to clipboard⁵ | <kbd>MOD</kbd>+<kbd>x</kbd>
|
| Cut to clipboard⁴ | <kbd>MOD</kbd>+<kbd>x</kbd>
|
||||||
| Synchronize clipboards and paste⁵ | <kbd>MOD</kbd>+<kbd>v</kbd>
|
| Synchronize clipboards and paste⁴ | <kbd>MOD</kbd>+<kbd>v</kbd>
|
||||||
| Inject computer clipboard text | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>
|
| Inject computer clipboard text | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>
|
||||||
| Enable/disable FPS counter (on stdout) | <kbd>MOD</kbd>+<kbd>i</kbd>
|
| Enable/disable FPS counter (on stdout) | <kbd>MOD</kbd>+<kbd>i</kbd>
|
||||||
| Pinch-to-zoom | <kbd>Ctrl</kbd>+_click-and-move_
|
| Pinch-to-zoom | <kbd>Ctrl</kbd>+_click-and-move_
|
||||||
@@ -1025,8 +990,7 @@ _<kbd>[Super]</kbd> is typically the <kbd>Windows</kbd> or <kbd>Cmd</kbd> key._
|
|||||||
_¹Double-click on black borders to remove them._
|
_¹Double-click on black borders to remove them._
|
||||||
_²Right-click turns the screen on if it was off, presses BACK otherwise._
|
_²Right-click turns the screen on if it was off, presses BACK otherwise._
|
||||||
_³4th and 5th mouse buttons, if your mouse has them._
|
_³4th and 5th mouse buttons, if your mouse has them._
|
||||||
_⁴For react-native apps in development, `MENU` triggers development menu._
|
_⁴Only on Android >= 7._
|
||||||
_⁵Only on Android >= 7._
|
|
||||||
|
|
||||||
Shortcuts with repeated keys are executted by releasing and pressing the key a
|
Shortcuts with repeated keys are executted by releasing and pressing the key a
|
||||||
second time. For example, to execute "Expand settings panel":
|
second time. For example, to execute "Expand settings panel":
|
||||||
@@ -1110,14 +1074,13 @@ Read the [developers page].
|
|||||||
|
|
||||||
This README is available in other languages:
|
This README is available in other languages:
|
||||||
|
|
||||||
- [Deutsch (German, `de`) - v1.22](README.de.md)
|
|
||||||
- [Indonesian (Indonesia, `id`) - v1.16](README.id.md)
|
- [Indonesian (Indonesia, `id`) - v1.16](README.id.md)
|
||||||
- [Italiano (Italiano, `it`) - v1.19](README.it.md)
|
- [Italiano (Italiano, `it`) - v1.19](README.it.md)
|
||||||
- [日本語 (Japanese, `jp`) - v1.19](README.jp.md)
|
- [日本語 (Japanese, `jp`) - v1.19](README.jp.md)
|
||||||
- [한국어 (Korean, `ko`) - v1.11](README.ko.md)
|
- [한국어 (Korean, `ko`) - v1.11](README.ko.md)
|
||||||
- [Português Brasileiro (Brazilian Portuguese, `pt-BR`) - v1.19](README.pt-br.md)
|
- [Português Brasileiro (Brazilian Portuguese, `pt-BR`) - v1.19](README.pt-br.md)
|
||||||
- [Español (Spanish, `sp`) - v1.21](README.sp.md)
|
- [Español (Spanish, `sp`) - v1.17](README.sp.md)
|
||||||
- [简体中文 (Simplified Chinese, `zh-Hans`) - v1.22](README.zh-Hans.md)
|
- [简体中文 (Simplified Chinese, `zh-Hans`) - v1.20](README.zh-Hans.md)
|
||||||
- [繁體中文 (Traditional Chinese, `zh-Hant`) - v1.15](README.zh-Hant.md)
|
- [繁體中文 (Traditional Chinese, `zh-Hant`) - v1.15](README.zh-Hant.md)
|
||||||
- [Turkish (Turkish, `tr`) - v1.18](README.tr.md)
|
- [Turkish (Turkish, `tr`) - v1.18](README.tr.md)
|
||||||
|
|
||||||
|
|||||||
341
README.sp.md
341
README.sp.md
@@ -1,36 +1,24 @@
|
|||||||
Solo se garantiza que el archivo [README](README.md) original esté actualizado.
|
Solo se garantiza que el archivo [README](README.md) original esté actualizado.
|
||||||
|
|
||||||
# scrcpy (v1.21)
|
# scrcpy (v1.17)
|
||||||
|
|
||||||
<img src="data/icon.svg" width="128" height="128" alt="scrcpy" align="right" />
|
Esta aplicación proporciona imagen y control de un dispositivo Android conectado
|
||||||
|
por USB (o [por TCP/IP][article-tcpip]). No requiere acceso _root_.
|
||||||
Esta aplicación proporciona control e imagen de un dispositivo Android conectado
|
|
||||||
por USB (o [por TCP/IP](#conexión)). No requiere acceso _root_.
|
|
||||||
Compatible con _GNU/Linux_, _Windows_ y _macOS_.
|
Compatible con _GNU/Linux_, _Windows_ y _macOS_.
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
Se enfoca en:
|
Sus características principales son:
|
||||||
- **ser ligera**: aplicación nativa, solo muestra la imagen del dispositivo
|
|
||||||
- **rendimiento**: 30~120fps, dependiendo del dispositivo
|
- **ligero** (nativo, solo muestra la imagen del dispositivo)
|
||||||
- **calidad**: 1920×1080 o superior
|
- **desempeño** (30~60fps)
|
||||||
- **baja latencia**: [35~70ms][lowlatency]
|
- **calidad** (1920×1080 o superior)
|
||||||
- **inicio rápido**: ~1 segundo para mostrar la primera imagen
|
- **baja latencia** ([35~70ms][lowlatency])
|
||||||
- **no intrusivo**: no deja nada instalado en el dispositivo
|
- **corto tiempo de inicio** (~1 segundo para mostrar la primera imagen)
|
||||||
- **beneficios**: sin cuentas, sin anuncios, no requiere acceso a internet
|
- **no intrusivo** (no se deja nada instalado en el dispositivo)
|
||||||
- **libertad**: software gratis y de código abierto
|
|
||||||
|
|
||||||
[lowlatency]: https://github.com/Genymobile/scrcpy/pull/646
|
[lowlatency]: https://github.com/Genymobile/scrcpy/pull/646
|
||||||
|
|
||||||
Con la aplicación puede:
|
|
||||||
- [grabar la pantalla](#capturas-y-grabaciones)
|
|
||||||
- duplicar la imagen con [la pantalla apagada](#apagar-la-pantalla)
|
|
||||||
- [copiar y pegar](#copiar-y-pegar) en ambos sentidos
|
|
||||||
- [configurar la calidad](#configuración-de-captura)
|
|
||||||
- usar la pantalla del dispositivo [como webcam (V4L2)](#v4l2loopback) (solo en Linux)
|
|
||||||
- [emular un teclado físico (HID)](#emular-teclado-físico-hid)
|
|
||||||
(solo en Linux)
|
|
||||||
- y mucho más…
|
|
||||||
|
|
||||||
## Requisitos
|
## Requisitos
|
||||||
|
|
||||||
@@ -63,7 +51,7 @@ Construir desde la fuente: [BUILD] ([proceso simplificado][BUILD_simple])
|
|||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
|
|
||||||
En Debian y Ubuntu:
|
En Debian (_test_ y _sid_ por ahora) y Ubuntu (20.04):
|
||||||
|
|
||||||
```
|
```
|
||||||
apt install scrcpy
|
apt install scrcpy
|
||||||
@@ -137,7 +125,7 @@ Necesitarás `adb`, accesible desde `PATH`. Si aún no lo tienes:
|
|||||||
brew install android-platform-tools
|
brew install android-platform-tools
|
||||||
```
|
```
|
||||||
|
|
||||||
También está disponible en [MacPorts], que configura el adb automáticamente:
|
También está disponible en [MacPorts], que configurará el adb automáticamente:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
sudo port install scrcpy
|
sudo port install scrcpy
|
||||||
@@ -165,7 +153,7 @@ scrcpy --help
|
|||||||
|
|
||||||
## Características
|
## Características
|
||||||
|
|
||||||
### Configuración de captura
|
### Capturar configuración
|
||||||
|
|
||||||
#### Reducir la definición
|
#### Reducir la definición
|
||||||
|
|
||||||
@@ -220,11 +208,10 @@ Si `--max-size` también está especificado, el cambio de tamaño es aplicado de
|
|||||||
Para fijar la rotación de la transmisión:
|
Para fijar la rotación de la transmisión:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
scrcpy --lock-video-orientation # orientación inicial
|
scrcpy --lock-video-orientation 0 # orientación normal
|
||||||
scrcpy --lock-video-orientation=0 # orientación normal
|
scrcpy --lock-video-orientation 1 # 90° contrarreloj
|
||||||
scrcpy --lock-video-orientation=1 # 90° contrarreloj
|
scrcpy --lock-video-orientation 2 # 180°
|
||||||
scrcpy --lock-video-orientation=2 # 180°
|
scrcpy --lock-video-orientation 3 # 90° sentido de las agujas del reloj
|
||||||
scrcpy --lock-video-orientation=3 # 90° sentido de las agujas del reloj
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Esto afecta la rotación de la grabación.
|
Esto afecta la rotación de la grabación.
|
||||||
@@ -246,10 +233,7 @@ Para listar los codificadores disponibles, puedes pasar un nombre de codificador
|
|||||||
scrcpy --encoder _
|
scrcpy --encoder _
|
||||||
```
|
```
|
||||||
|
|
||||||
### Capturas y grabaciones
|
### Grabación
|
||||||
|
|
||||||
|
|
||||||
#### Grabación
|
|
||||||
|
|
||||||
Es posible grabar la pantalla mientras se transmite:
|
Es posible grabar la pantalla mientras se transmite:
|
||||||
|
|
||||||
@@ -266,117 +250,17 @@ scrcpy -Nr file.mkv
|
|||||||
# interrumpe la grabación con Ctrl+C
|
# interrumpe la grabación con Ctrl+C
|
||||||
```
|
```
|
||||||
|
|
||||||
Los "skipped frames" son grabados, incluso si no se mostrados en tiempo real (por razones de desempeño). Los frames tienen _marcas de tiempo_ en el dispositivo, por lo que el "[packet delay
|
"Skipped frames" son grabados, incluso si no son mostrados en tiempo real (por razones de desempeño). Los frames tienen _marcas de tiempo_ en el dispositivo, por lo que el "[packet delay
|
||||||
variation]" no impacta el archivo grabado.
|
variation]" no impacta el archivo grabado.
|
||||||
|
|
||||||
[packet delay variation]: https://en.wikipedia.org/wiki/Packet_delay_variation
|
[packet delay variation]: https://en.wikipedia.org/wiki/Packet_delay_variation
|
||||||
|
|
||||||
|
|
||||||
#### v4l2loopback
|
|
||||||
|
|
||||||
En Linux se puede mandar el stream del video a un dispositivo loopback v4l2, por
|
|
||||||
lo que se puede abrir el dispositivo Android como una webcam con cualquier
|
|
||||||
programa compatible con v4l2.
|
|
||||||
|
|
||||||
Se debe instalar el modulo `v4l2loopback`:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
sudo apt install v4l2loopback-dkms
|
|
||||||
```
|
|
||||||
|
|
||||||
Para crear un dispositivo v4l2:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
sudo modprobe v4l2loopback
|
|
||||||
```
|
|
||||||
|
|
||||||
Esto va a crear un nuevo dispositivo de video en `/dev/videoN`, donde `N` es un número
|
|
||||||
(hay más [opciones](https://github.com/umlaeute/v4l2loopback#options) disponibles
|
|
||||||
para crear múltiples dispositivos o usar un ID en específico).
|
|
||||||
|
|
||||||
Para ver los dispositivos disponibles:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# requiere el paquete v4l-utils
|
|
||||||
v4l2-ctl --list-devices
|
|
||||||
# simple pero generalmente suficiente
|
|
||||||
ls /dev/video*
|
|
||||||
```
|
|
||||||
|
|
||||||
Para iniciar scrcpy usando una fuente v4l2:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --v4l2-sink=/dev/videoN
|
|
||||||
scrcpy --v4l2-sink=/dev/videoN --no-display # deshabilita la transmisión de imagen
|
|
||||||
scrcpy --v4l2-sink=/dev/videoN -N # más corto
|
|
||||||
```
|
|
||||||
|
|
||||||
(reemplace `N` con el ID del dispositivo, compruebe con `ls /dev/video*`)
|
|
||||||
|
|
||||||
Una vez habilitado, podés abrir el stream del video con una herramienta compatible con v4l2:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
ffplay -i /dev/videoN
|
|
||||||
vlc v4l2:///dev/videoN # VLC puede agregar un delay por buffering
|
|
||||||
```
|
|
||||||
|
|
||||||
Por ejemplo, podrías capturar el video usando [OBS].
|
|
||||||
|
|
||||||
[OBS]: https://obsproject.com/
|
|
||||||
|
|
||||||
|
|
||||||
#### Buffering
|
|
||||||
|
|
||||||
Es posible agregar buffering al video. Esto reduce el ruido en la imagen ("jitter")
|
|
||||||
pero aumenta la latencia (vea [#2464]).
|
|
||||||
|
|
||||||
[#2464]: https://github.com/Genymobile/scrcpy/issues/2464
|
|
||||||
|
|
||||||
La opción de buffering está disponible para la transmisión de imagen:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --display-buffer=50 # agrega 50 ms de buffering a la imagen
|
|
||||||
```
|
|
||||||
|
|
||||||
y las fuentes V4L2:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --v4l2-buffer=500 # agrega 500 ms de buffering a la fuente v4l2
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
### Conexión
|
### Conexión
|
||||||
|
|
||||||
#### TCP/IP (Inalámbrica)
|
#### Inalámbrica
|
||||||
|
|
||||||
_Scrcpy_ usa `adb` para comunicarse con el dispositivo, y `adb` puede [conectarse] vía TCP/IP.
|
_Scrcpy_ usa `adb` para comunicarse con el dispositivo, y `adb` puede [conectarse] vía TCP/IP:
|
||||||
El dispositivo debe estar conectado a la misma red que la computadora:
|
|
||||||
|
|
||||||
##### Automático
|
|
||||||
|
|
||||||
La opción `--tcpip` permite configurar la conexión automáticamente. Hay 2 variables.
|
|
||||||
|
|
||||||
Si el dispositivo (accesible en 192.168.1.1 para este ejemplo) ya está escuchando
|
|
||||||
en un puerto (generalmente 5555) esperando una conexión adb entrante, entonces corré:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --tcpip=192.168.1.1 # el puerto default es 5555
|
|
||||||
scrcpy --tcpip=192.168.1.1:5555
|
|
||||||
```
|
|
||||||
|
|
||||||
Si el dispositivo no tiene habilitado el modo adb TCP/IP (o si no sabés la dirección IP),
|
|
||||||
entonces conectá el dispositivo por USB y corré:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --tcpip # sin argumentos
|
|
||||||
```
|
|
||||||
|
|
||||||
El programa buscará automáticamente la IP del dispositivo, habilitará el modo TCP/IP, y
|
|
||||||
se conectará al dispositivo antes de comenzar a transmitir la imagen.
|
|
||||||
|
|
||||||
##### Manual
|
|
||||||
|
|
||||||
Como alternativa, se puede habilitar la conexión TCP/IP manualmente usando `adb`:
|
|
||||||
|
|
||||||
1. Conecta el dispositivo al mismo Wi-Fi que tu computadora.
|
1. Conecta el dispositivo al mismo Wi-Fi que tu computadora.
|
||||||
2. Obtén la dirección IP del dispositivo, en Ajustes → Acerca del dispositivo → Estado, o ejecutando este comando:
|
2. Obtén la dirección IP del dispositivo, en Ajustes → Acerca del dispositivo → Estado, o ejecutando este comando:
|
||||||
@@ -418,7 +302,7 @@ scrcpy -s 192.168.0.1:5555 # versión breve
|
|||||||
|
|
||||||
Puedes iniciar múltiples instancias de _scrcpy_ para múltiples dispositivos.
|
Puedes iniciar múltiples instancias de _scrcpy_ para múltiples dispositivos.
|
||||||
|
|
||||||
#### Iniciar automáticamente al detectar dispositivo
|
#### Autoiniciar al detectar dispositivo
|
||||||
|
|
||||||
Puedes utilizar [AutoAdb]:
|
Puedes utilizar [AutoAdb]:
|
||||||
|
|
||||||
@@ -428,82 +312,37 @@ autoadb scrcpy -s '{}'
|
|||||||
|
|
||||||
[AutoAdb]: https://github.com/rom1v/autoadb
|
[AutoAdb]: https://github.com/rom1v/autoadb
|
||||||
|
|
||||||
#### Túneles
|
#### Túnel SSH
|
||||||
|
|
||||||
Para conectarse a un dispositivo remoto, es posible conectar un cliente local `adb` a un servidor remoto `adb` (siempre y cuando utilicen la misma versión de protocolos _adb_).
|
Para conectarse a un dispositivo remoto, es posible conectar un cliente local de `adb` a un servidor remoto `adb` (siempre y cuando utilicen la misma versión de protocolos _adb_):
|
||||||
|
|
||||||
##### Servidor ADB remoto
|
|
||||||
|
|
||||||
Para conectarse a un servidor ADB remoto, haz que el servidor escuche en todas las interfaces:
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
adb kill-server
|
adb kill-server # cierra el servidor local adb en 5037
|
||||||
adb -a nodaemon server start
|
ssh -CN -L5037:localhost:5037 -R27183:localhost:27183 your_remote_computer
|
||||||
# conserva este servidor abierto
|
# conserva este servidor abierto
|
||||||
```
|
```
|
||||||
|
|
||||||
**Advertencia: todas las comunicaciones entre los clientes y el servidor ADB están desencriptadas.**
|
Desde otra terminal:
|
||||||
|
|
||||||
Supondremos que este servidor se puede acceder desde 192.168.1.2. Entonces, desde otra
|
|
||||||
terminal, corré scrcpy:
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
export ADB_SERVER_SOCKET=tcp:192.168.1.2:5037
|
|
||||||
scrcpy --tunnel-host=192.168.1.2
|
|
||||||
```
|
|
||||||
|
|
||||||
Por default, scrcpy usa el puerto local que se usó para establecer el tunel
|
|
||||||
`adb forward` (típicamente `27183`, vea `--port`). También es posible forzar un
|
|
||||||
puerto diferente (puede resultar útil en situaciones más complejas, donde haya
|
|
||||||
múltiples redirecciones):
|
|
||||||
|
|
||||||
```
|
|
||||||
scrcpy --tunnel-port=1234
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
##### Túnel SSH
|
|
||||||
|
|
||||||
Para comunicarse con un servidor ADB remoto de forma segura, es preferible usar un túnel SSH.
|
|
||||||
|
|
||||||
Primero, asegurate que el servidor ADB está corriendo en la computadora remota:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
adb start-server
|
|
||||||
```
|
|
||||||
|
|
||||||
Después, establecé el túnel SSH:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# local 5038 --> remoto 5037
|
|
||||||
# local 27183 <-- remoto 27183
|
|
||||||
ssh -CN -L5038:localhost:5037 -R27183:localhost:27183 your_remote_computer
|
|
||||||
# conserva este servidor abierto
|
|
||||||
```
|
|
||||||
|
|
||||||
Desde otra terminal, corré scrcpy:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
export ADB_SERVER_SOCKET=tcp:localhost:5038
|
|
||||||
scrcpy
|
scrcpy
|
||||||
```
|
```
|
||||||
|
|
||||||
Para evitar habilitar "remote port forwarding", puedes forzar una "forward connection" (nótese el argumento `-L` en vez de `-R`):
|
Para evitar habilitar "remote port forwarding", puedes forzar una "forward connection" (nótese el argumento `-L` en vez de `-R`):
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# local 5038 --> remoto 5037
|
adb kill-server # cierra el servidor local adb en 5037
|
||||||
# local 27183 --> remoto 27183
|
ssh -CN -L5037:localhost:5037 -L27183:localhost:27183 your_remote_computer
|
||||||
ssh -CN -L5038:localhost:5037 -L27183:localhost:27183 your_remote_computer
|
|
||||||
# conserva este servidor abierto
|
# conserva este servidor abierto
|
||||||
```
|
```
|
||||||
|
|
||||||
Desde otra terminal, corré scrcpy:
|
Desde otra terminal:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
export ADB_SERVER_SOCKET=tcp:localhost:5038
|
|
||||||
scrcpy --force-adb-forward
|
scrcpy --force-adb-forward
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
Al igual que las conexiones inalámbricas, puede resultar útil reducir la calidad:
|
Al igual que las conexiones inalámbricas, puede resultar útil reducir la calidad:
|
||||||
|
|
||||||
```
|
```
|
||||||
@@ -563,7 +402,7 @@ Se puede rotar la ventana:
|
|||||||
scrcpy --rotation 1
|
scrcpy --rotation 1
|
||||||
```
|
```
|
||||||
|
|
||||||
Los posibles valores son:
|
Los valores posibles son:
|
||||||
- `0`: sin rotación
|
- `0`: sin rotación
|
||||||
- `1`: 90 grados contrarreloj
|
- `1`: 90 grados contrarreloj
|
||||||
- `2`: 180 grados
|
- `2`: 180 grados
|
||||||
@@ -577,7 +416,7 @@ Nótese que _scrcpy_ maneja 3 diferentes rotaciones:
|
|||||||
- `--rotation` (o <kbd>MOD</kbd>+<kbd>←</kbd>/<kbd>MOD</kbd>+<kbd>→</kbd>) rota solo el contenido de la imagen. Esto solo afecta a la imagen mostrada, no a la grabación.
|
- `--rotation` (o <kbd>MOD</kbd>+<kbd>←</kbd>/<kbd>MOD</kbd>+<kbd>→</kbd>) rota solo el contenido de la imagen. Esto solo afecta a la imagen mostrada, no a la grabación.
|
||||||
|
|
||||||
|
|
||||||
### Otras opciones
|
### Otras opciones menores
|
||||||
|
|
||||||
#### Solo lectura ("Read-only")
|
#### Solo lectura ("Read-only")
|
||||||
|
|
||||||
@@ -640,12 +479,14 @@ scrcpy -Sw # versión breve
|
|||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
#### Apagar al cerrar la aplicación
|
#### Renderizar frames vencidos
|
||||||
|
|
||||||
Para apagar la pantalla del dispositivo al cerrar scrcpy:
|
Por defecto, para minimizar la latencia, _scrcpy_ siempre renderiza el último frame disponible decodificado, e ignora cualquier frame anterior.
|
||||||
|
|
||||||
|
Para forzar el renderizado de todos los frames (a costo de posible aumento de latencia), use:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
scrcpy --power-off-on-close
|
scrcpy --render-expired-frames
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Mostrar clicks
|
#### Mostrar clicks
|
||||||
@@ -707,8 +548,6 @@ Además, <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd> permite inyectar el texto
|
|||||||
|
|
||||||
Algunos dispositivos no se comportan como es esperado al establecer el portapapeles programáticamente. La opción `--legacy-paste` está disponible para cambiar el comportamiento de <kbd>Ctrl</kbd>+<kbd>v</kbd> y <kbd>MOD</kbd>+<kbd>v</kbd> para que también inyecten el texto del portapapeles de la computadora como una secuencia de teclas (de la misma forma que <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>).
|
Algunos dispositivos no se comportan como es esperado al establecer el portapapeles programáticamente. La opción `--legacy-paste` está disponible para cambiar el comportamiento de <kbd>Ctrl</kbd>+<kbd>v</kbd> y <kbd>MOD</kbd>+<kbd>v</kbd> para que también inyecten el texto del portapapeles de la computadora como una secuencia de teclas (de la misma forma que <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>).
|
||||||
|
|
||||||
Para deshabilitar la auto-sincronización del portapapeles, use `--no-clipboard-autosync`.
|
|
||||||
|
|
||||||
#### Pellizcar para zoom
|
#### Pellizcar para zoom
|
||||||
|
|
||||||
Para simular "pinch-to-zoom": <kbd>Ctrl</kbd>+_click-y-mover_.
|
Para simular "pinch-to-zoom": <kbd>Ctrl</kbd>+_click-y-mover_.
|
||||||
@@ -717,48 +556,6 @@ Más precisamente, mantén <kbd>Ctrl</kbd> mientras presionas botón izquierdo.
|
|||||||
|
|
||||||
Concretamente, scrcpy genera clicks adicionales con un "dedo virtual" en la posición invertida respecto al centro de la pantalla.
|
Concretamente, scrcpy genera clicks adicionales con un "dedo virtual" en la posición invertida respecto al centro de la pantalla.
|
||||||
|
|
||||||
#### Emular teclado físico (HID)
|
|
||||||
|
|
||||||
Por default, scrcpy usa el sistema de Android para la injección de teclas o texto:
|
|
||||||
funciona en todas partes, pero está limitado a ASCII.
|
|
||||||
|
|
||||||
En Linux, scrcpy puede emular un teclado USB físico en Android para proveer
|
|
||||||
una mejor experiencia al enviar _inputs_ (usando [USB HID vía AOAv2][hid-aoav2]):
|
|
||||||
deshabilita el teclado virtual y funciona para todos los caracteres y IME.
|
|
||||||
|
|
||||||
[hid-aoav2]: https://source.android.com/devices/accessories/aoa2#hid-support
|
|
||||||
|
|
||||||
Sin embargo, solo funciona si el dispositivo está conectado por USB, y por ahora
|
|
||||||
solo funciona en Linux.
|
|
||||||
|
|
||||||
Para habilitar este modo:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --hid-keyboard
|
|
||||||
scrcpy -K # más corto
|
|
||||||
```
|
|
||||||
|
|
||||||
Si por alguna razón falla (por ejemplo si el dispositivo no está conectado vía
|
|
||||||
USB), automáticamente vuelve al modo default (un mensaje se escribirá en la consola).
|
|
||||||
Se puede usar los mismos argumentos en la línea de comandos tanto si se conecta con
|
|
||||||
USB o vía TCP/IP.
|
|
||||||
|
|
||||||
En este modo, los _raw key events_ (_scancodes_) se envían al dispositivo, independientemente
|
|
||||||
del mapeo del teclado en el host. Por eso, si el diseño de tu teclado no concuerda, debe ser
|
|
||||||
configurado en el dispositivo Android, en Ajustes → Sistema → Idioma y Entrada de Texto
|
|
||||||
→ [Teclado Físico].
|
|
||||||
|
|
||||||
Se puede iniciar automáticamente en esta página de ajustes:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
adb shell am start -a android.settings.HARD_KEYBOARD_SETTINGS
|
|
||||||
```
|
|
||||||
|
|
||||||
Sin embargo, la opción solo está disponible cuando el teclado HID está activo
|
|
||||||
(o cuando se conecta un teclado físico).
|
|
||||||
|
|
||||||
[Teclado Físico]: https://github.com/Genymobile/scrcpy/pull/2632#issuecomment-923756915
|
|
||||||
|
|
||||||
|
|
||||||
#### Preferencias de inyección de texto
|
#### Preferencias de inyección de texto
|
||||||
|
|
||||||
@@ -776,23 +573,13 @@ scrcpy --prefer-text
|
|||||||
|
|
||||||
(Pero esto romperá el comportamiento del teclado en los juegos)
|
(Pero esto romperá el comportamiento del teclado en los juegos)
|
||||||
|
|
||||||
Por el contrario, se puede forzar scrcpy para siempre injectar _raw key events_:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --raw-key-events
|
|
||||||
```
|
|
||||||
|
|
||||||
Estas opciones no tienen efecto en los teclados HID (todos los _key events_ son enviados como
|
|
||||||
_scancodes_ en este modo).
|
|
||||||
|
|
||||||
[textevents]: https://blog.rom1v.com/2018/03/introducing-scrcpy/#handle-text-input
|
[textevents]: https://blog.rom1v.com/2018/03/introducing-scrcpy/#handle-text-input
|
||||||
[prefertext]: https://github.com/Genymobile/scrcpy/issues/650#issuecomment-512945343
|
[prefertext]: https://github.com/Genymobile/scrcpy/issues/650#issuecomment-512945343
|
||||||
|
|
||||||
|
|
||||||
#### Repetir tecla
|
#### Repetir tecla
|
||||||
|
|
||||||
Por defecto, mantener una tecla presionada genera múltiples _key events_. Esto puede
|
Por defecto, mantener una tecla presionada genera múltiples _key events_. Esto puede causar problemas de desempeño en algunos juegos, donde estos eventos no tienen sentido de todos modos.
|
||||||
causar problemas de desempeño en algunos juegos, donde estos eventos no tienen sentido de todos modos.
|
|
||||||
|
|
||||||
Para evitar enviar _key events_ repetidos:
|
Para evitar enviar _key events_ repetidos:
|
||||||
|
|
||||||
@@ -800,9 +587,6 @@ Para evitar enviar _key events_ repetidos:
|
|||||||
scrcpy --no-key-repeat
|
scrcpy --no-key-repeat
|
||||||
```
|
```
|
||||||
|
|
||||||
Estas opciones no tienen efecto en los teclados HID (Android maneja directamente
|
|
||||||
las repeticiones de teclas en este modo)
|
|
||||||
|
|
||||||
|
|
||||||
#### Botón derecho y botón del medio
|
#### Botón derecho y botón del medio
|
||||||
|
|
||||||
@@ -824,15 +608,14 @@ No hay respuesta visual, un mensaje se escribirá en la consola.
|
|||||||
|
|
||||||
#### Enviar archivos al dispositivo
|
#### Enviar archivos al dispositivo
|
||||||
|
|
||||||
Para enviar un archivo a `/sdcard/Download/` en el dispositivo, arrastre y suelte
|
Para enviar un archivo a `/sdcard/` en el dispositivo, arrastre y suelte un archivo (no APK) a la ventana de _scrcpy_.
|
||||||
un archivo (no APK) a la ventana de _scrcpy_.
|
|
||||||
|
|
||||||
No hay ninguna respuesta visual, un mensaje se escribirá en la consola.
|
No hay respuesta visual, un mensaje se escribirá en la consola.
|
||||||
|
|
||||||
El directorio de destino puede ser modificado al iniciar:
|
El directorio de destino puede ser modificado al iniciar:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
scrcpy --push-target=/sdcard/Movies/
|
scrcpy --push-target=/sdcard/Download/
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
@@ -870,11 +653,11 @@ _<kbd>[Super]</kbd> es generalmente la tecla <kbd>Windows</kbd> o <kbd>Cmd</kbd>
|
|||||||
| Rotar pantalla hacia la izquierda | <kbd>MOD</kbd>+<kbd>←</kbd> _(izquierda)_
|
| Rotar pantalla hacia la izquierda | <kbd>MOD</kbd>+<kbd>←</kbd> _(izquierda)_
|
||||||
| Rotar pantalla hacia la derecha | <kbd>MOD</kbd>+<kbd>→</kbd> _(derecha)_
|
| Rotar pantalla hacia la derecha | <kbd>MOD</kbd>+<kbd>→</kbd> _(derecha)_
|
||||||
| Ajustar ventana a 1:1 ("pixel-perfect") | <kbd>MOD</kbd>+<kbd>g</kbd>
|
| Ajustar ventana a 1:1 ("pixel-perfect") | <kbd>MOD</kbd>+<kbd>g</kbd>
|
||||||
| Ajustar ventana para quitar los bordes negros| <kbd>MOD</kbd>+<kbd>w</kbd> \| _Doble click izquierdo¹_
|
| Ajustar ventana para quitar los bordes negros| <kbd>MOD</kbd>+<kbd>w</kbd> \| _Doble click¹_
|
||||||
| Click en `INICIO` | <kbd>MOD</kbd>+<kbd>h</kbd> \| _Click medio_
|
| Click en `INICIO` | <kbd>MOD</kbd>+<kbd>h</kbd> \| _Botón del medio_
|
||||||
| Click en `RETROCEDER` | <kbd>MOD</kbd>+<kbd>b</kbd> \| _Click derecho²_
|
| Click en `RETROCEDER` | <kbd>MOD</kbd>+<kbd>b</kbd> \| _Botón derecho²_
|
||||||
| Click en `CAMBIAR APLICACIÓN` | <kbd>MOD</kbd>+<kbd>s</kbd> \| _Cuarto botón³_
|
| Click en `CAMBIAR APLICACIÓN` | <kbd>MOD</kbd>+<kbd>s</kbd>
|
||||||
| Click en `MENÚ` (desbloquear pantalla)⁴ | <kbd>MOD</kbd>+<kbd>m</kbd>
|
| Click en `MENÚ` (desbloquear pantalla) | <kbd>MOD</kbd>+<kbd>m</kbd>
|
||||||
| Click en `SUBIR VOLUMEN` | <kbd>MOD</kbd>+<kbd>↑</kbd> _(arriba)_
|
| Click en `SUBIR VOLUMEN` | <kbd>MOD</kbd>+<kbd>↑</kbd> _(arriba)_
|
||||||
| Click en `BAJAR VOLUME` | <kbd>MOD</kbd>+<kbd>↓</kbd> _(abajo)_
|
| Click en `BAJAR VOLUME` | <kbd>MOD</kbd>+<kbd>↓</kbd> _(abajo)_
|
||||||
| Click en `ENCENDIDO` | <kbd>MOD</kbd>+<kbd>p</kbd>
|
| Click en `ENCENDIDO` | <kbd>MOD</kbd>+<kbd>p</kbd>
|
||||||
@@ -882,30 +665,18 @@ _<kbd>[Super]</kbd> es generalmente la tecla <kbd>Windows</kbd> o <kbd>Cmd</kbd>
|
|||||||
| Apagar pantalla (manteniendo la transmisión)| <kbd>MOD</kbd>+<kbd>o</kbd>
|
| Apagar pantalla (manteniendo la transmisión)| <kbd>MOD</kbd>+<kbd>o</kbd>
|
||||||
| Encender pantalla | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>o</kbd>
|
| Encender pantalla | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>o</kbd>
|
||||||
| Rotar pantalla del dispositivo | <kbd>MOD</kbd>+<kbd>r</kbd>
|
| Rotar pantalla del dispositivo | <kbd>MOD</kbd>+<kbd>r</kbd>
|
||||||
| Abrir panel de notificaciones | <kbd>MOD</kbd>+<kbd>n</kbd> \| _Quinto botón³_
|
| Abrir panel de notificaciones | <kbd>MOD</kbd>+<kbd>n</kbd>
|
||||||
| Abrir panel de configuración | <kbd>MOD</kbd>+<kbd>n</kbd>+<kbd>n</kbd> \| _Doble quinto botón³_
|
| Cerrar panel de notificaciones | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>n</kbd>
|
||||||
| Cerrar paneles | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>n</kbd>
|
| Copiar al portapapeles³ | <kbd>MOD</kbd>+<kbd>c</kbd>
|
||||||
| Copiar al portapapeles⁵ | <kbd>MOD</kbd>+<kbd>c</kbd>
|
| Cortar al portapapeles³ | <kbd>MOD</kbd>+<kbd>x</kbd>
|
||||||
| Cortar al portapapeles⁵ | <kbd>MOD</kbd>+<kbd>x</kbd>
|
| Synchronizar portapapeles y pegar³ | <kbd>MOD</kbd>+<kbd>v</kbd>
|
||||||
| Synchronizar portapapeles y pegar⁵ | <kbd>MOD</kbd>+<kbd>v</kbd>
|
| inyectar texto del portapapeles de la PC | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>
|
||||||
| Inyectar texto del portapapeles de la PC | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>
|
|
||||||
| Habilitar/Deshabilitar contador de FPS (en stdout) | <kbd>MOD</kbd>+<kbd>i</kbd>
|
| Habilitar/Deshabilitar contador de FPS (en stdout) | <kbd>MOD</kbd>+<kbd>i</kbd>
|
||||||
| Pellizcar para zoom | <kbd>Ctrl</kbd>+_click-y-mover_
|
| Pellizcar para zoom | <kbd>Ctrl</kbd>+_click-y-mover_
|
||||||
| Arrastrar y soltar un archivo (APK) | Instalar APK desde la computadora
|
|
||||||
| Arrastrar y soltar un archivo (no APK) | [Mover archivo al dispositivo](#enviar-archivos-al-dispositivo)
|
|
||||||
|
|
||||||
_¹Doble click en los bordes negros para eliminarlos._
|
_¹Doble click en los bordes negros para eliminarlos._
|
||||||
_²Botón derecho enciende la pantalla si estaba apagada, sino ejecuta RETROCEDER._
|
_²Botón derecho enciende la pantalla si estaba apagada, sino ejecuta RETROCEDER._
|
||||||
_³Cuarto y quinto botón del mouse, si tu mouse los tiene._
|
_³Solo en Android >= 7._
|
||||||
_⁴Para las apps react-native en desarrollo, `MENU` activa el menú de desarrollo._
|
|
||||||
_⁵Solo en Android >= 7._
|
|
||||||
|
|
||||||
Los shortcuts con teclas repetidas se ejecutan soltando y volviendo a apretar la tecla
|
|
||||||
por segunda vez. Por ejemplo, para ejecutar "Abrir panel de configuración":
|
|
||||||
|
|
||||||
1. Apretá y mantené apretado <kbd>MOD</kbd>.
|
|
||||||
2. Después apretá dos veces la tecla <kbd>n</kbd>.
|
|
||||||
3. Por último, soltá la tecla <kbd>MOD</kbd>.
|
|
||||||
|
|
||||||
Todos los atajos <kbd>Ctrl</kbd>+_tecla_ son enviados al dispositivo para que sean manejados por la aplicación activa.
|
Todos los atajos <kbd>Ctrl</kbd>+_tecla_ son enviados al dispositivo para que sean manejados por la aplicación activa.
|
||||||
|
|
||||||
@@ -920,8 +691,6 @@ ADB=/path/to/adb scrcpy
|
|||||||
|
|
||||||
Para sobreescribir el path del archivo `scrcpy-server`, configure el path en `SCRCPY_SERVER_PATH`.
|
Para sobreescribir el path del archivo `scrcpy-server`, configure el path en `SCRCPY_SERVER_PATH`.
|
||||||
|
|
||||||
Para sobreescribir el ícono, configure el path en `SCRCPY_ICON_PATH`.
|
|
||||||
|
|
||||||
|
|
||||||
## ¿Por qué _scrcpy_?
|
## ¿Por qué _scrcpy_?
|
||||||
|
|
||||||
|
|||||||
@@ -1,19 +1,17 @@
|
|||||||
_Only the original [README](README.md) is guaranteed to be up-to-date._
|
_Only the original [README](README.md) is guaranteed to be up-to-date._
|
||||||
|
|
||||||
_只有原版的 [README](README.md)是保证最新的。_
|
只有原版的[README](README.md)会保持最新。
|
||||||
|
|
||||||
Current version is based on [f4c7044]
|
Current version is based on [65b023a]
|
||||||
|
|
||||||
本文根据[f4c7044]进行翻译。
|
本文根据[65b023a]进行翻译。
|
||||||
|
|
||||||
[f4c7044]: https://github.com/Genymobile/scrcpy/blob/f4c7044b46ae28eb64cb5e1a15c9649a44023c70/README.md
|
[65b023a]: https://github.com/Genymobile/scrcpy/blob/65b023ac6d586593193fd5290f65e25603b68e02/README.md
|
||||||
|
|
||||||
# scrcpy (v1.22)
|
# scrcpy (v1.20)
|
||||||
|
|
||||||
<img src="data/icon.svg" width="128" height="128" alt="scrcpy" align="right" />
|
<img src="data/icon.svg" width="128" height="128" alt="scrcpy" align="right" />
|
||||||
|
|
||||||
_发音为 "**scr**een **c**o**py**"_
|
|
||||||
|
|
||||||
本应用程序可以显示并控制通过 USB (或 [TCP/IP][article-tcpip]) 连接的安卓设备,且不需要任何 _root_ 权限。本程序支持 _GNU/Linux_, _Windows_ 和 _macOS_。
|
本应用程序可以显示并控制通过 USB (或 [TCP/IP][article-tcpip]) 连接的安卓设备,且不需要任何 _root_ 权限。本程序支持 _GNU/Linux_, _Windows_ 和 _macOS_。
|
||||||
|
|
||||||

|

|
||||||
@@ -38,8 +36,6 @@ _发音为 "**scr**een **c**o**py**"_
|
|||||||
- [可配置显示质量](#采集设置)
|
- [可配置显示质量](#采集设置)
|
||||||
- 以设备屏幕[作为摄像头(V4L2)](#v4l2loopback) (仅限 Linux)
|
- 以设备屏幕[作为摄像头(V4L2)](#v4l2loopback) (仅限 Linux)
|
||||||
- [模拟物理键盘 (HID)](#物理键盘模拟-hid) (仅限 Linux)
|
- [模拟物理键盘 (HID)](#物理键盘模拟-hid) (仅限 Linux)
|
||||||
- [物理鼠标模拟 (HID)](#物理鼠标模拟-hid) (仅限 Linux)
|
|
||||||
- [OTG模式](#otg) (仅限 Linux)
|
|
||||||
- 更多 ……
|
- 更多 ……
|
||||||
|
|
||||||
## 系统要求
|
## 系统要求
|
||||||
@@ -72,18 +68,12 @@ _发音为 "**scr**een **c**o**py**"_
|
|||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
|
|
||||||
在 Debian 和 Ubuntu 上:
|
在 Debian (目前仅支持 _testing_ 和 _sid_ 分支) 和Ubuntu (20.04) 上:
|
||||||
|
|
||||||
```
|
```
|
||||||
apt install scrcpy
|
apt install scrcpy
|
||||||
```
|
```
|
||||||
|
|
||||||
在 Arch Linux 上:
|
|
||||||
|
|
||||||
```
|
|
||||||
pacman -S scrcpy
|
|
||||||
```
|
|
||||||
|
|
||||||
我们也提供 [Snap] 包: [`scrcpy`][snap-link]。
|
我们也提供 [Snap] 包: [`scrcpy`][snap-link]。
|
||||||
|
|
||||||
[snap-link]: https://snapstats.org/snaps/scrcpy
|
[snap-link]: https://snapstats.org/snaps/scrcpy
|
||||||
@@ -95,6 +85,11 @@ pacman -S scrcpy
|
|||||||
[COPR]: https://fedoraproject.org/wiki/Category:Copr
|
[COPR]: https://fedoraproject.org/wiki/Category:Copr
|
||||||
[copr-link]: https://copr.fedorainfracloud.org/coprs/zeno/scrcpy/
|
[copr-link]: https://copr.fedorainfracloud.org/coprs/zeno/scrcpy/
|
||||||
|
|
||||||
|
对 Arch Linux 我们提供 [AUR] 包: [`scrcpy`][aur-link]。
|
||||||
|
|
||||||
|
[AUR]: https://wiki.archlinux.org/index.php/Arch_User_Repository
|
||||||
|
[aur-link]: https://aur.archlinux.org/packages/scrcpy/
|
||||||
|
|
||||||
对 Gentoo 我们提供 [Ebuild] 包:[`scrcpy/`][ebuild-link]。
|
对 Gentoo 我们提供 [Ebuild] 包:[`scrcpy/`][ebuild-link]。
|
||||||
|
|
||||||
[Ebuild]: https://wiki.gentoo.org/wiki/Ebuild
|
[Ebuild]: https://wiki.gentoo.org/wiki/Ebuild
|
||||||
@@ -348,32 +343,9 @@ scrcpy --v4l2-buffer=500 # 为 v4l2 漏增加 500 毫秒的缓冲
|
|||||||
|
|
||||||
### 连接
|
### 连接
|
||||||
|
|
||||||
#### TCP/IP (无线)
|
#### 无线
|
||||||
|
|
||||||
_Scrcpy_ 使用 `adb` 与设备通信,并且 `adb` 支持通过 TCP/IP [连接]到设备(设备必须连接与电脑相同的网络)。
|
_Scrcpy_ 使用 `adb` 与设备通信,并且 `adb` 支持通过 TCP/IP [连接]到设备:
|
||||||
|
|
||||||
##### 自动配置
|
|
||||||
|
|
||||||
参数 `--tcpip` 允许自动配置连接。这里有两种方式。
|
|
||||||
|
|
||||||
对于传入的 adb 连接,如果设备(在这个例子中以192.168.1.1为可用地址)已经监听了一个端口(通常是5555),运行:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --tcpip=192.168.1.1 # 默认端口是5555
|
|
||||||
scrcpy --tcpip=192.168.1.1:5555
|
|
||||||
```
|
|
||||||
|
|
||||||
如果adb TCP/IP(无线) 模式在某些设备上不被启用(或者你不知道IP地址),用USB连接设备,然后运行:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --tcpip # 无需其他参数
|
|
||||||
```
|
|
||||||
|
|
||||||
这将会自动寻找设备IP地址,启用TCP/IP模式,然后在启动之前连接到设备。
|
|
||||||
|
|
||||||
##### 手动配置
|
|
||||||
|
|
||||||
或者,可以通过 `adb` 使用手动启用 TCP/IP 连接:
|
|
||||||
|
|
||||||
1. 将设备和电脑连接至同一 Wi-Fi。
|
1. 将设备和电脑连接至同一 Wi-Fi。
|
||||||
2. 打开 设置 → 关于手机 → 状态信息,获取设备的 IP 地址,也可以执行以下的命令:
|
2. 打开 设置 → 关于手机 → 状态信息,获取设备的 IP 地址,也可以执行以下的命令:
|
||||||
@@ -387,7 +359,7 @@ scrcpy --tcpip # 无需其他参数
|
|||||||
5. 连接到您的设备:`adb connect DEVICE_IP:5555` _(将 `DEVICE_IP` 替换为设备 IP)_。
|
5. 连接到您的设备:`adb connect DEVICE_IP:5555` _(将 `DEVICE_IP` 替换为设备 IP)_。
|
||||||
6. 正常运行 `scrcpy`。
|
6. 正常运行 `scrcpy`。
|
||||||
|
|
||||||
降低比特率和分辨率可能很有用:
|
可能降低码率和分辨率会更好一些:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
scrcpy --bit-rate 2M --max-size 800
|
scrcpy --bit-rate 2M --max-size 800
|
||||||
@@ -425,75 +397,33 @@ autoadb scrcpy -s '{}'
|
|||||||
|
|
||||||
[AutoAdb]: https://github.com/rom1v/autoadb
|
[AutoAdb]: https://github.com/rom1v/autoadb
|
||||||
|
|
||||||
#### 隧道
|
#### SSH 隧道
|
||||||
|
|
||||||
要远程连接到设备,可以将本地的 adb 客户端连接到远程的 adb 服务端 (需要两端的 _adb_ 协议版本相同)。
|
要远程连接到设备,可以将本地的 adb 客户端连接到远程的 adb 服务端 (需要两端的 _adb_ 协议版本相同):
|
||||||
|
|
||||||
##### 远程ADB服务器
|
|
||||||
|
|
||||||
要连接到一个远程ADB服务器,让服务器在所有接口上监听:
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
adb kill-server
|
adb kill-server # 关闭本地 5037 端口上的 adb 服务端
|
||||||
adb -a nodaemon server start
|
ssh -CN -L5037:localhost:5037 -R27183:localhost:27183 your_remote_computer
|
||||||
# 保持该窗口开启
|
# 保持该窗口开启
|
||||||
```
|
```
|
||||||
|
|
||||||
**警告:所有客户端与ADB服务器的交流都是未加密的。**
|
在另一个终端:
|
||||||
|
|
||||||
假设此服务器可在 192.168.1.2 访问。 然后,从另一个终端,运行 scrcpy:
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
export ADB_SERVER_SOCKET=tcp:192.168.1.2:5037
|
|
||||||
scrcpy --tunnel-host=192.168.1.2
|
|
||||||
```
|
|
||||||
|
|
||||||
默认情况下,scrcpy使用用于 `adb forward` 隧道建立的本地端口(通常是 `27183`,见 `--port` )。它也可以强制使用一个不同的隧道端口(当涉及更多的重定向时,这在更复杂的情况下可能很有用):
|
|
||||||
|
|
||||||
```
|
|
||||||
scrcpy --tunnel-port=1234
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
##### SSH 隧道
|
|
||||||
|
|
||||||
为了安全地与远程ADB服务器通信,最好使用SSH隧道。
|
|
||||||
|
|
||||||
首先,确保ADB服务器正在远程计算机上运行:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
adb start-server
|
|
||||||
```
|
|
||||||
|
|
||||||
然后,建立一个SSH隧道:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# 本地 5038 --> 远程 5037
|
|
||||||
# 本地 27183 <-- 远程 27183
|
|
||||||
ssh -CN -L5038:localhost:5037 -R27183:localhost:27183 your_remote_computer
|
|
||||||
# 保持该窗口开启
|
|
||||||
```
|
|
||||||
|
|
||||||
在另一个终端上,运行scrcpy:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
export ADB_SERVER_SOCKET=tcp:localhost:5038
|
|
||||||
scrcpy
|
scrcpy
|
||||||
```
|
```
|
||||||
|
|
||||||
若要不使用远程端口转发,可以强制使用正向连接(注意是 `-L` 而不是 `-R` ):
|
若要不使用远程端口转发,可以强制使用正向连接 (注意 `-L` 和 `-R` 的区别):
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# 本地 5038 --> 远程 5037
|
adb kill-server # 关闭本地 5037 端口上的 adb 服务端
|
||||||
# 本地 27183 <-- 远程 27183
|
ssh -CN -L5037:localhost:5037 -L27183:localhost:27183 your_remote_computer
|
||||||
ssh -CN -L5038:localhost:5037 -L27183:localhost:27183 your_remote_computer
|
|
||||||
# 保持该窗口开启
|
# 保持该窗口开启
|
||||||
```
|
```
|
||||||
|
|
||||||
在另一个终端上,运行scrcpy:
|
在另一个终端:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
export ADB_SERVER_SOCKET=tcp:localhost:5038
|
|
||||||
scrcpy --force-adb-forward
|
scrcpy --force-adb-forward
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -511,7 +441,7 @@ scrcpy -b2M -m800 --max-fps 15
|
|||||||
窗口的标题默认为设备型号。可以通过如下命令修改:
|
窗口的标题默认为设备型号。可以通过如下命令修改:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
scrcpy --window-title "我的设备"
|
scrcpy --window-title 'My device'
|
||||||
```
|
```
|
||||||
|
|
||||||
#### 位置和大小
|
#### 位置和大小
|
||||||
@@ -700,8 +630,6 @@ scrcpy --disable-screensaver
|
|||||||
|
|
||||||
一些设备不支持通过程序设置剪贴板。通过 `--legacy-paste` 选项可以修改 <kbd>Ctrl</kbd>+<kbd>v</kbd> 和 <kbd>MOD</kbd>+<kbd>v</kbd> 的工作方式,使它们通过按键事件 (同 <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>) 来注入电脑剪贴板内容。
|
一些设备不支持通过程序设置剪贴板。通过 `--legacy-paste` 选项可以修改 <kbd>Ctrl</kbd>+<kbd>v</kbd> 和 <kbd>MOD</kbd>+<kbd>v</kbd> 的工作方式,使它们通过按键事件 (同 <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>) 来注入电脑剪贴板内容。
|
||||||
|
|
||||||
要禁用自动剪贴板同步功能,使用`--no-clipboard-autosync`。
|
|
||||||
|
|
||||||
#### 双指缩放
|
#### 双指缩放
|
||||||
|
|
||||||
模拟“双指缩放”:<kbd>Ctrl</kbd>+_按住并移动鼠标_。
|
模拟“双指缩放”:<kbd>Ctrl</kbd>+_按住并移动鼠标_。
|
||||||
@@ -731,60 +659,11 @@ scrcpy -K # 简写
|
|||||||
|
|
||||||
在这种模式下,原始按键事件 (扫描码) 被发送给设备,而与宿主机按键映射无关。因此,若键盘布局不匹配,需要在 Android 设备上进行配置,具体为 设置 → 系统 → 语言和输入法 → [实体键盘]。
|
在这种模式下,原始按键事件 (扫描码) 被发送给设备,而与宿主机按键映射无关。因此,若键盘布局不匹配,需要在 Android 设备上进行配置,具体为 设置 → 系统 → 语言和输入法 → [实体键盘]。
|
||||||
|
|
||||||
[实体键盘]: https://github.com/Genymobile/scrcpy/pull/2632#issuecomment-923756915
|
[Physical keyboard]: https://github.com/Genymobile/scrcpy/pull/2632#issuecomment-923756915
|
||||||
|
|
||||||
#### 物理鼠标模拟 (HID)
|
|
||||||
|
|
||||||
与物理键盘模拟类似,可以模拟一个物理鼠标。 同样,它仅在设备通过 USB 连接时才有效,并且目前仅在 Linux 上受支持。
|
|
||||||
|
|
||||||
默认情况下,scrcpy 使用 Android 鼠标事件注入,使用绝对坐标。 通过模拟物理鼠标,在Android设备上出现鼠标指针,并注入鼠标相对运动、点击和滚动。
|
|
||||||
|
|
||||||
启用此模式:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --hid-mouse
|
|
||||||
scrcpy -M # 简写
|
|
||||||
```
|
|
||||||
|
|
||||||
您还可以将 `--forward-all-clicks` 添加到 [转发所有点击][forward_all_clicks].
|
|
||||||
|
|
||||||
[forward_all_clicks]: #右键和中键
|
|
||||||
|
|
||||||
启用此模式后,计算机鼠标将被“捕获”(鼠标指针从计算机上消失并出现在 Android 设备上)。
|
|
||||||
|
|
||||||
特殊的捕获键,<kbd>Alt</kbd> 或 <kbd>Super</kbd>,切换(禁用或启用)鼠标捕获。 使用其中之一将鼠标的控制权交还给计算机。
|
|
||||||
|
|
||||||
|
|
||||||
#### OTG
|
|
||||||
|
|
||||||
可以仅使用物理键盘和鼠标模拟 (HID) 运行 _scrcpy_,就好像计算机键盘和鼠标通过 OTG 线直接插入设备一样。
|
|
||||||
|
|
||||||
在这个模式下,_adb_ (USB 调试)是不必要的,且镜像被禁用。
|
|
||||||
|
|
||||||
启用 OTG 模式:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --otg
|
|
||||||
# 如果有多个 USB 设备可用,则通过序列号选择
|
|
||||||
scrcpy --otg -s 0123456789abcdef
|
|
||||||
```
|
|
||||||
|
|
||||||
只开启 HID 键盘 或 HID 鼠标 是可行的:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --otg --hid-keyboard # 只开启 HID 键盘
|
|
||||||
scrcpy --otg --hid-mouse # 只开启 HID 鼠标
|
|
||||||
scrcpy --otg --hid-keyboard --hid-mouse # 开启 HID 键盘 和 HID 鼠标
|
|
||||||
# 为了方便,默认两者都开启
|
|
||||||
scrcpy --otg # 开启 HID 键盘 和 HID 鼠标
|
|
||||||
```
|
|
||||||
|
|
||||||
像 `--hid-keyboard` 和 `--hid-mouse` 一样,它只在设备通过 USB 连接时才有效,且目前仅在 Linux 上支持。
|
|
||||||
|
|
||||||
|
|
||||||
#### 文本注入偏好
|
#### 文本注入偏好
|
||||||
|
|
||||||
输入文字的时候,系统会产生两种[事件][textevents]:
|
打字的时候,系统会产生两种[事件][textevents]:
|
||||||
- _按键事件_ ,代表一个按键被按下或松开。
|
- _按键事件_ ,代表一个按键被按下或松开。
|
||||||
- _文本事件_ ,代表一个字符被输入。
|
- _文本事件_ ,代表一个字符被输入。
|
||||||
|
|
||||||
@@ -796,13 +675,7 @@ scrcpy --otg # 开启 HID 键盘 和 HID 鼠标
|
|||||||
scrcpy --prefer-text
|
scrcpy --prefer-text
|
||||||
```
|
```
|
||||||
|
|
||||||
(但这会导致键盘在游戏中工作不正常)
|
(这会导致键盘在游戏中工作不正常)
|
||||||
|
|
||||||
相反,您可以强制始终注入原始按键事件:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
scrcpy --raw-key-events
|
|
||||||
```
|
|
||||||
|
|
||||||
该选项不影响 HID 键盘 (该模式下,所有按键都发送为扫描码)。
|
该选项不影响 HID 键盘 (该模式下,所有按键都发送为扫描码)。
|
||||||
|
|
||||||
@@ -892,7 +765,7 @@ _<kbd>[Super]</kbd> 键通常是指 <kbd>Windows</kbd> 或 <kbd>Cmd</kbd> 键。
|
|||||||
| 点按 `主屏幕` | <kbd>MOD</kbd>+<kbd>h</kbd> \| _中键_
|
| 点按 `主屏幕` | <kbd>MOD</kbd>+<kbd>h</kbd> \| _中键_
|
||||||
| 点按 `返回` | <kbd>MOD</kbd>+<kbd>b</kbd> \| _右键²_
|
| 点按 `返回` | <kbd>MOD</kbd>+<kbd>b</kbd> \| _右键²_
|
||||||
| 点按 `切换应用` | <kbd>MOD</kbd>+<kbd>s</kbd> \| _第4键³_
|
| 点按 `切换应用` | <kbd>MOD</kbd>+<kbd>s</kbd> \| _第4键³_
|
||||||
| 点按 `菜单` (解锁屏幕)⁴ | <kbd>MOD</kbd>+<kbd>m</kbd>
|
| 点按 `菜单` (解锁屏幕) | <kbd>MOD</kbd>+<kbd>m</kbd>
|
||||||
| 点按 `音量+` | <kbd>MOD</kbd>+<kbd>↑</kbd> _(上箭头)_
|
| 点按 `音量+` | <kbd>MOD</kbd>+<kbd>↑</kbd> _(上箭头)_
|
||||||
| 点按 `音量-` | <kbd>MOD</kbd>+<kbd>↓</kbd> _(下箭头)_
|
| 点按 `音量-` | <kbd>MOD</kbd>+<kbd>↓</kbd> _(下箭头)_
|
||||||
| 点按 `电源` | <kbd>MOD</kbd>+<kbd>p</kbd>
|
| 点按 `电源` | <kbd>MOD</kbd>+<kbd>p</kbd>
|
||||||
@@ -903,9 +776,9 @@ _<kbd>[Super]</kbd> 键通常是指 <kbd>Windows</kbd> 或 <kbd>Cmd</kbd> 键。
|
|||||||
| 展开通知面板 | <kbd>MOD</kbd>+<kbd>n</kbd> \| _第5键³_
|
| 展开通知面板 | <kbd>MOD</kbd>+<kbd>n</kbd> \| _第5键³_
|
||||||
| 展开设置面板 | <kbd>MOD</kbd>+<kbd>n</kbd>+<kbd>n</kbd> \| _双击第5键³_
|
| 展开设置面板 | <kbd>MOD</kbd>+<kbd>n</kbd>+<kbd>n</kbd> \| _双击第5键³_
|
||||||
| 收起通知面板 | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>n</kbd>
|
| 收起通知面板 | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>n</kbd>
|
||||||
| 复制到剪贴板⁵ | <kbd>MOD</kbd>+<kbd>c</kbd>
|
| 复制到剪贴板⁴ | <kbd>MOD</kbd>+<kbd>c</kbd>
|
||||||
| 剪切到剪贴板⁵ | <kbd>MOD</kbd>+<kbd>x</kbd>
|
| 剪切到剪贴板⁴ | <kbd>MOD</kbd>+<kbd>x</kbd>
|
||||||
| 同步剪贴板并粘贴⁵ | <kbd>MOD</kbd>+<kbd>v</kbd>
|
| 同步剪贴板并粘贴⁴ | <kbd>MOD</kbd>+<kbd>v</kbd>
|
||||||
| 注入电脑剪贴板文本 | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>
|
| 注入电脑剪贴板文本 | <kbd>MOD</kbd>+<kbd>Shift</kbd>+<kbd>v</kbd>
|
||||||
| 打开/关闭FPS显示 (至标准输出) | <kbd>MOD</kbd>+<kbd>i</kbd>
|
| 打开/关闭FPS显示 (至标准输出) | <kbd>MOD</kbd>+<kbd>i</kbd>
|
||||||
| 捏拉缩放 | <kbd>Ctrl</kbd>+_按住并移动鼠标_
|
| 捏拉缩放 | <kbd>Ctrl</kbd>+_按住并移动鼠标_
|
||||||
@@ -915,8 +788,7 @@ _<kbd>[Super]</kbd> 键通常是指 <kbd>Windows</kbd> 或 <kbd>Cmd</kbd> 键。
|
|||||||
_¹双击黑边可以去除黑边。_
|
_¹双击黑边可以去除黑边。_
|
||||||
_²点击鼠标右键将在屏幕熄灭时点亮屏幕,其余情况则视为按下返回键 。_
|
_²点击鼠标右键将在屏幕熄灭时点亮屏幕,其余情况则视为按下返回键 。_
|
||||||
_³鼠标的第4键和第5键。_
|
_³鼠标的第4键和第5键。_
|
||||||
_⁴对于开发中的 react-native 应用程序,`MENU` 触发开发菜单。_
|
_⁴需要安卓版本 Android >= 7。_
|
||||||
_⁵需要安卓版本 Android >= 7。_
|
|
||||||
|
|
||||||
有重复按键的快捷键通过松开再按下一个按键来进行,如“展开设置面板”:
|
有重复按键的快捷键通过松开再按下一个按键来进行,如“展开设置面板”:
|
||||||
|
|
||||||
@@ -944,7 +816,7 @@ ADB=/path/to/adb scrcpy
|
|||||||
|
|
||||||
一个同事让我找出一个和 [gnirehtet] 一样难以发音的名字。
|
一个同事让我找出一个和 [gnirehtet] 一样难以发音的名字。
|
||||||
|
|
||||||
[`strcpy`] 源于 **str**ing (字符串); `scrcpy` 源于 **scr**een (屏幕)。
|
[`strcpy`] 复制一个 **str**ing (字符串); `scrcpy` 复制一个 **scr**een (屏幕)。
|
||||||
|
|
||||||
[gnirehtet]: https://github.com/Genymobile/gnirehtet
|
[gnirehtet]: https://github.com/Genymobile/gnirehtet
|
||||||
[`strcpy`]: http://man7.org/linux/man-pages/man3/strcpy.3.html
|
[`strcpy`]: http://man7.org/linux/man-pages/man3/strcpy.3.html
|
||||||
|
|||||||
@@ -72,15 +72,12 @@ if v4l2_support
|
|||||||
src += [ 'src/v4l2_sink.c' ]
|
src += [ 'src/v4l2_sink.c' ]
|
||||||
endif
|
endif
|
||||||
|
|
||||||
usb_support = host_machine.system() == 'linux'
|
aoa_hid_support = host_machine.system() == 'linux'
|
||||||
if usb_support
|
if aoa_hid_support
|
||||||
src += [
|
src += [
|
||||||
'src/usb/aoa_hid.c',
|
'src/aoa_hid.c',
|
||||||
'src/usb/hid_keyboard.c',
|
'src/hid_keyboard.c',
|
||||||
'src/usb/hid_mouse.c',
|
'src/hid_mouse.c',
|
||||||
'src/usb/scrcpy_otg.c',
|
|
||||||
'src/usb/screen_otg.c',
|
|
||||||
'src/usb/usb.c',
|
|
||||||
]
|
]
|
||||||
endif
|
endif
|
||||||
|
|
||||||
@@ -102,16 +99,16 @@ if not crossbuild_windows
|
|||||||
dependencies += dependency('libavdevice')
|
dependencies += dependency('libavdevice')
|
||||||
endif
|
endif
|
||||||
|
|
||||||
if usb_support
|
if aoa_hid_support
|
||||||
dependencies += dependency('libusb-1.0')
|
dependencies += dependency('libusb-1.0')
|
||||||
endif
|
endif
|
||||||
|
|
||||||
else
|
else
|
||||||
# cross-compile mingw32 build (from Linux to Windows)
|
# cross-compile mingw32 build (from Linux to Windows)
|
||||||
prebuilt_sdl2 = meson.get_cross_property('prebuilt_sdl2')
|
prebuilt_sdl2 = meson.get_cross_property('prebuilt_sdl2')
|
||||||
sdl2_bin_dir = meson.current_source_dir() + '/../prebuilt-deps/data/' + prebuilt_sdl2 + '/bin'
|
sdl2_bin_dir = meson.current_source_dir() + '/../prebuilt-deps/' + prebuilt_sdl2 + '/bin'
|
||||||
sdl2_lib_dir = meson.current_source_dir() + '/../prebuilt-deps/data/' + prebuilt_sdl2 + '/lib'
|
sdl2_lib_dir = meson.current_source_dir() + '/../prebuilt-deps/' + prebuilt_sdl2 + '/lib'
|
||||||
sdl2_include_dir = '../prebuilt-deps/data/' + prebuilt_sdl2 + '/include'
|
sdl2_include_dir = '../prebuilt-deps/' + prebuilt_sdl2 + '/include'
|
||||||
|
|
||||||
sdl2 = declare_dependency(
|
sdl2 = declare_dependency(
|
||||||
dependencies: [
|
dependencies: [
|
||||||
@@ -122,8 +119,8 @@ else
|
|||||||
)
|
)
|
||||||
|
|
||||||
prebuilt_ffmpeg = meson.get_cross_property('prebuilt_ffmpeg')
|
prebuilt_ffmpeg = meson.get_cross_property('prebuilt_ffmpeg')
|
||||||
ffmpeg_bin_dir = meson.current_source_dir() + '/../prebuilt-deps/data/' + prebuilt_ffmpeg + '/bin'
|
ffmpeg_bin_dir = meson.current_source_dir() + '/../prebuilt-deps/' + prebuilt_ffmpeg + '/bin'
|
||||||
ffmpeg_include_dir = '../prebuilt-deps/data/' + prebuilt_ffmpeg + '/include'
|
ffmpeg_include_dir = '../prebuilt-deps/' + prebuilt_ffmpeg + '/include'
|
||||||
|
|
||||||
# ffmpeg versions are different for win32 and win64 builds
|
# ffmpeg versions are different for win32 and win64 builds
|
||||||
ffmpeg_avcodec = meson.get_cross_property('ffmpeg_avcodec')
|
ffmpeg_avcodec = meson.get_cross_property('ffmpeg_avcodec')
|
||||||
@@ -196,7 +193,7 @@ conf.set('SERVER_DEBUGGER_METHOD_NEW', get_option('server_debugger_method') == '
|
|||||||
conf.set('HAVE_V4L2', v4l2_support)
|
conf.set('HAVE_V4L2', v4l2_support)
|
||||||
|
|
||||||
# enable HID over AOA support (linux only)
|
# enable HID over AOA support (linux only)
|
||||||
conf.set('HAVE_USB', usb_support)
|
conf.set('HAVE_AOA_HID', aoa_hid_support)
|
||||||
|
|
||||||
configure_file(configuration: conf, output: 'config.h')
|
configure_file(configuration: conf, output: 'config.h')
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ BEGIN
|
|||||||
VALUE "LegalCopyright", "Romain Vimont, Genymobile"
|
VALUE "LegalCopyright", "Romain Vimont, Genymobile"
|
||||||
VALUE "OriginalFilename", "scrcpy.exe"
|
VALUE "OriginalFilename", "scrcpy.exe"
|
||||||
VALUE "ProductName", "scrcpy"
|
VALUE "ProductName", "scrcpy"
|
||||||
VALUE "ProductVersion", "1.22"
|
VALUE "ProductVersion", "1.21"
|
||||||
END
|
END
|
||||||
END
|
END
|
||||||
BLOCK "VarFileInfo"
|
BLOCK "VarFileInfo"
|
||||||
|
|||||||
14
app/scrcpy.1
14
app/scrcpy.1
@@ -162,20 +162,6 @@ Do not forward repeated key events when a key is held down.
|
|||||||
.B \-\-no\-mipmaps
|
.B \-\-no\-mipmaps
|
||||||
If the renderer is OpenGL 3.0+ or OpenGL ES 2.0+, then mipmaps are automatically generated to improve downscaling quality. This option disables the generation of mipmaps.
|
If the renderer is OpenGL 3.0+ or OpenGL ES 2.0+, then mipmaps are automatically generated to improve downscaling quality. This option disables the generation of mipmaps.
|
||||||
|
|
||||||
.TP
|
|
||||||
.B \-\-otg
|
|
||||||
Run in OTG mode: simulate physical keyboard and mouse, as if the computer keyboard and mouse were plugged directly to the device via an OTG cable.
|
|
||||||
|
|
||||||
In this mode, adb (USB debugging) is not necessary, and mirroring is disabled.
|
|
||||||
|
|
||||||
LAlt, LSuper or RSuper toggle the mouse capture mode, to give control of the mouse back to the computer.
|
|
||||||
|
|
||||||
If any of \fB\-\-hid\-keyboard\fR or \fB\-\-hid\-mouse\fR is set, only enable keyboard or mouse respectively, otherwise enable both.
|
|
||||||
|
|
||||||
It may only work over USB, and is currently only supported on Linux.
|
|
||||||
|
|
||||||
See \fB\-\-hid\-keyboard\fR and \fB\-\-hid\-mouse\fR.
|
|
||||||
|
|
||||||
.TP
|
.TP
|
||||||
.BI "\-p, \-\-port " port[:port]
|
.BI "\-p, \-\-port " port[:port]
|
||||||
Set the TCP port (range) used by the client to listen.
|
Set the TCP port (range) used by the client to listen.
|
||||||
|
|||||||
@@ -425,7 +425,7 @@ adb_get_serialno(struct sc_intr *intr, unsigned flags) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (r == -1) {
|
if (r == -1) {
|
||||||
return NULL;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
sc_str_truncate(buf, r, " \r\n");
|
sc_str_truncate(buf, r, " \r\n");
|
||||||
@@ -455,7 +455,7 @@ adb_get_device_ip(struct sc_intr *intr, const char *serial, unsigned flags) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (r == -1) {
|
if (r == -1) {
|
||||||
return NULL;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
assert((size_t) r <= sizeof(buf));
|
assert((size_t) r <= sizeof(buf));
|
||||||
|
|||||||
@@ -50,9 +50,79 @@ log_libusb_error(enum libusb_error errcode) {
|
|||||||
LOGW("libusb error: %s", libusb_strerror(errcode));
|
LOGW("libusb error: %s", libusb_strerror(errcode));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
accept_device(libusb_device *device, const char *serial) {
|
||||||
|
// do not log any USB error in this function, it is expected that many USB
|
||||||
|
// devices available on the computer have permission restrictions
|
||||||
|
|
||||||
|
struct libusb_device_descriptor desc;
|
||||||
|
int result = libusb_get_device_descriptor(device, &desc);
|
||||||
|
if (result < 0 || !desc.iSerialNumber) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
libusb_device_handle *handle;
|
||||||
|
result = libusb_open(device, &handle);
|
||||||
|
if (result < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
char buffer[128];
|
||||||
|
result = libusb_get_string_descriptor_ascii(handle, desc.iSerialNumber,
|
||||||
|
(unsigned char *) buffer,
|
||||||
|
sizeof(buffer));
|
||||||
|
libusb_close(handle);
|
||||||
|
if (result < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer[sizeof(buffer) - 1] = '\0'; // just in case
|
||||||
|
|
||||||
|
// accept the device if its serial matches
|
||||||
|
return !strcmp(buffer, serial);
|
||||||
|
}
|
||||||
|
|
||||||
|
static libusb_device *
|
||||||
|
sc_aoa_find_usb_device(const char *serial) {
|
||||||
|
if (!serial) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
libusb_device **list;
|
||||||
|
libusb_device *result = NULL;
|
||||||
|
ssize_t count = libusb_get_device_list(NULL, &list);
|
||||||
|
if (count < 0) {
|
||||||
|
log_libusb_error((enum libusb_error) count);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (size_t i = 0; i < (size_t) count; ++i) {
|
||||||
|
libusb_device *device = list[i];
|
||||||
|
|
||||||
|
if (accept_device(device, serial)) {
|
||||||
|
result = libusb_ref_device(device);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
libusb_free_device_list(list, 1);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
sc_aoa_open_usb_handle(libusb_device *device, libusb_device_handle **handle) {
|
||||||
|
int result = libusb_open(device, handle);
|
||||||
|
if (result < 0) {
|
||||||
|
log_libusb_error((enum libusb_error) result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
sc_aoa_init(struct sc_aoa *aoa, struct sc_usb *usb,
|
sc_aoa_init(struct sc_aoa *aoa, const char *serial,
|
||||||
struct sc_acksync *acksync) {
|
struct sc_acksync *acksync) {
|
||||||
|
assert(acksync);
|
||||||
|
|
||||||
cbuf_init(&aoa->queue);
|
cbuf_init(&aoa->queue);
|
||||||
|
|
||||||
if (!sc_mutex_init(&aoa->mutex)) {
|
if (!sc_mutex_init(&aoa->mutex)) {
|
||||||
@@ -60,15 +130,39 @@ sc_aoa_init(struct sc_aoa *aoa, struct sc_usb *usb,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!sc_cond_init(&aoa->event_cond)) {
|
if (!sc_cond_init(&aoa->event_cond)) {
|
||||||
sc_mutex_destroy(&aoa->mutex);
|
goto error_destroy_mutex;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (libusb_init(&aoa->usb_context) != LIBUSB_SUCCESS) {
|
||||||
|
goto error_destroy_cond;
|
||||||
|
}
|
||||||
|
|
||||||
|
aoa->usb_device = sc_aoa_find_usb_device(serial);
|
||||||
|
if (!aoa->usb_device) {
|
||||||
|
LOGW("USB device of serial %s not found", serial);
|
||||||
|
goto error_exit_libusb;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sc_aoa_open_usb_handle(aoa->usb_device, &aoa->usb_handle) < 0) {
|
||||||
|
LOGW("Open USB handle failed");
|
||||||
|
goto error_unref_device;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
aoa->stopped = false;
|
aoa->stopped = false;
|
||||||
aoa->acksync = acksync;
|
aoa->acksync = acksync;
|
||||||
aoa->usb = usb;
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
error_unref_device:
|
||||||
|
libusb_unref_device(aoa->usb_device);
|
||||||
|
error_exit_libusb:
|
||||||
|
libusb_exit(aoa->usb_context);
|
||||||
|
error_destroy_cond:
|
||||||
|
sc_cond_destroy(&aoa->event_cond);
|
||||||
|
error_destroy_mutex:
|
||||||
|
sc_mutex_destroy(&aoa->mutex);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@@ -79,6 +173,9 @@ sc_aoa_destroy(struct sc_aoa *aoa) {
|
|||||||
sc_hid_event_destroy(&event);
|
sc_hid_event_destroy(&event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
libusb_close(aoa->usb_handle);
|
||||||
|
libusb_unref_device(aoa->usb_device);
|
||||||
|
libusb_exit(aoa->usb_context);
|
||||||
sc_cond_destroy(&aoa->event_cond);
|
sc_cond_destroy(&aoa->event_cond);
|
||||||
sc_mutex_destroy(&aoa->mutex);
|
sc_mutex_destroy(&aoa->mutex);
|
||||||
}
|
}
|
||||||
@@ -95,8 +192,8 @@ sc_aoa_register_hid(struct sc_aoa *aoa, uint16_t accessory_id,
|
|||||||
uint16_t index = report_desc_size;
|
uint16_t index = report_desc_size;
|
||||||
unsigned char *buffer = NULL;
|
unsigned char *buffer = NULL;
|
||||||
uint16_t length = 0;
|
uint16_t length = 0;
|
||||||
int result = libusb_control_transfer(aoa->usb->handle, request_type,
|
int result = libusb_control_transfer(aoa->usb_handle, request_type, request,
|
||||||
request, value, index, buffer, length,
|
value, index, buffer, length,
|
||||||
DEFAULT_TIMEOUT);
|
DEFAULT_TIMEOUT);
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
log_libusb_error((enum libusb_error) result);
|
log_libusb_error((enum libusb_error) result);
|
||||||
@@ -131,8 +228,8 @@ sc_aoa_set_hid_report_desc(struct sc_aoa *aoa, uint16_t accessory_id,
|
|||||||
// libusb_control_transfer expects a pointer to non-const
|
// libusb_control_transfer expects a pointer to non-const
|
||||||
unsigned char *buffer = (unsigned char *) report_desc;
|
unsigned char *buffer = (unsigned char *) report_desc;
|
||||||
uint16_t length = report_desc_size;
|
uint16_t length = report_desc_size;
|
||||||
int result = libusb_control_transfer(aoa->usb->handle, request_type,
|
int result = libusb_control_transfer(aoa->usb_handle, request_type, request,
|
||||||
request, value, index, buffer, length,
|
value, index, buffer, length,
|
||||||
DEFAULT_TIMEOUT);
|
DEFAULT_TIMEOUT);
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
log_libusb_error((enum libusb_error) result);
|
log_libusb_error((enum libusb_error) result);
|
||||||
@@ -173,8 +270,8 @@ sc_aoa_send_hid_event(struct sc_aoa *aoa, const struct sc_hid_event *event) {
|
|||||||
uint16_t index = 0;
|
uint16_t index = 0;
|
||||||
unsigned char *buffer = event->buffer;
|
unsigned char *buffer = event->buffer;
|
||||||
uint16_t length = event->size;
|
uint16_t length = event->size;
|
||||||
int result = libusb_control_transfer(aoa->usb->handle, request_type,
|
int result = libusb_control_transfer(aoa->usb_handle, request_type, request,
|
||||||
request, value, index, buffer, length,
|
value, index, buffer, length,
|
||||||
DEFAULT_TIMEOUT);
|
DEFAULT_TIMEOUT);
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
log_libusb_error((enum libusb_error) result);
|
log_libusb_error((enum libusb_error) result);
|
||||||
@@ -195,8 +292,8 @@ sc_aoa_unregister_hid(struct sc_aoa *aoa, const uint16_t accessory_id) {
|
|||||||
uint16_t index = 0;
|
uint16_t index = 0;
|
||||||
unsigned char *buffer = NULL;
|
unsigned char *buffer = NULL;
|
||||||
uint16_t length = 0;
|
uint16_t length = 0;
|
||||||
int result = libusb_control_transfer(aoa->usb->handle, request_type,
|
int result = libusb_control_transfer(aoa->usb_handle, request_type, request,
|
||||||
request, value, index, buffer, length,
|
value, index, buffer, length,
|
||||||
DEFAULT_TIMEOUT);
|
DEFAULT_TIMEOUT);
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
log_libusb_error((enum libusb_error) result);
|
log_libusb_error((enum libusb_error) result);
|
||||||
@@ -246,11 +343,6 @@ run_aoa_thread(void *data) {
|
|||||||
|
|
||||||
if (ack_to_wait != SC_SEQUENCE_INVALID) {
|
if (ack_to_wait != SC_SEQUENCE_INVALID) {
|
||||||
LOGD("Waiting ack from server sequence=%" PRIu64_, ack_to_wait);
|
LOGD("Waiting ack from server sequence=%" PRIu64_, ack_to_wait);
|
||||||
|
|
||||||
// If some events have ack_to_wait set, then sc_aoa must have been
|
|
||||||
// initialized with a non NULL acksync
|
|
||||||
assert(aoa->acksync);
|
|
||||||
|
|
||||||
// Do not block the loop indefinitely if the ack never comes (it
|
// Do not block the loop indefinitely if the ack never comes (it
|
||||||
// should never happen)
|
// should never happen)
|
||||||
sc_tick deadline = sc_tick_now() + SC_TICK_FROM_MS(500);
|
sc_tick deadline = sc_tick_now() + SC_TICK_FROM_MS(500);
|
||||||
@@ -297,10 +389,8 @@ sc_aoa_stop(struct sc_aoa *aoa) {
|
|||||||
sc_cond_signal(&aoa->event_cond);
|
sc_cond_signal(&aoa->event_cond);
|
||||||
sc_mutex_unlock(&aoa->mutex);
|
sc_mutex_unlock(&aoa->mutex);
|
||||||
|
|
||||||
if (aoa->acksync) {
|
|
||||||
sc_acksync_interrupt(aoa->acksync);
|
sc_acksync_interrupt(aoa->acksync);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
sc_aoa_join(struct sc_aoa *aoa) {
|
sc_aoa_join(struct sc_aoa *aoa) {
|
||||||
@@ -6,7 +6,6 @@
|
|||||||
|
|
||||||
#include <libusb-1.0/libusb.h>
|
#include <libusb-1.0/libusb.h>
|
||||||
|
|
||||||
#include "usb.h"
|
|
||||||
#include "util/acksync.h"
|
#include "util/acksync.h"
|
||||||
#include "util/cbuf.h"
|
#include "util/cbuf.h"
|
||||||
#include "util/thread.h"
|
#include "util/thread.h"
|
||||||
@@ -30,7 +29,9 @@ sc_hid_event_destroy(struct sc_hid_event *hid_event);
|
|||||||
struct sc_hid_event_queue CBUF(struct sc_hid_event, 64);
|
struct sc_hid_event_queue CBUF(struct sc_hid_event, 64);
|
||||||
|
|
||||||
struct sc_aoa {
|
struct sc_aoa {
|
||||||
struct sc_usb *usb;
|
libusb_context *usb_context;
|
||||||
|
libusb_device *usb_device;
|
||||||
|
libusb_device_handle *usb_handle;
|
||||||
sc_thread thread;
|
sc_thread thread;
|
||||||
sc_mutex mutex;
|
sc_mutex mutex;
|
||||||
sc_cond event_cond;
|
sc_cond event_cond;
|
||||||
@@ -41,7 +42,7 @@ struct sc_aoa {
|
|||||||
};
|
};
|
||||||
|
|
||||||
bool
|
bool
|
||||||
sc_aoa_init(struct sc_aoa *aoa, struct sc_usb *usb, struct sc_acksync *acksync);
|
sc_aoa_init(struct sc_aoa *aoa, const char *serial, struct sc_acksync *acksync);
|
||||||
|
|
||||||
void
|
void
|
||||||
sc_aoa_destroy(struct sc_aoa *aoa);
|
sc_aoa_destroy(struct sc_aoa *aoa);
|
||||||
@@ -53,7 +53,6 @@
|
|||||||
#define OPT_TCPIP 1033
|
#define OPT_TCPIP 1033
|
||||||
#define OPT_RAW_KEY_EVENTS 1034
|
#define OPT_RAW_KEY_EVENTS 1034
|
||||||
#define OPT_NO_DOWNSIZE_ON_ERROR 1035
|
#define OPT_NO_DOWNSIZE_ON_ERROR 1035
|
||||||
#define OPT_OTG 1036
|
|
||||||
|
|
||||||
struct sc_option {
|
struct sc_option {
|
||||||
char shortopt;
|
char shortopt;
|
||||||
@@ -277,22 +276,6 @@ static const struct sc_option options[] = {
|
|||||||
"mipmaps are automatically generated to improve downscaling "
|
"mipmaps are automatically generated to improve downscaling "
|
||||||
"quality. This option disables the generation of mipmaps.",
|
"quality. This option disables the generation of mipmaps.",
|
||||||
},
|
},
|
||||||
{
|
|
||||||
.longopt_id = OPT_OTG,
|
|
||||||
.longopt = "otg",
|
|
||||||
.text = "Run in OTG mode: simulate physical keyboard and mouse, "
|
|
||||||
"as if the computer keyboard and mouse were plugged directly "
|
|
||||||
"to the device via an OTG cable.\n"
|
|
||||||
"In this mode, adb (USB debugging) is not necessary, and "
|
|
||||||
"mirroring is disabled.\n"
|
|
||||||
"LAlt, LSuper or RSuper toggle the mouse capture mode, to give "
|
|
||||||
"control of the mouse back to the computer.\n"
|
|
||||||
"If any of --hid-keyboard or --hid-mouse is set, only enable "
|
|
||||||
"keyboard or mouse respectively, otherwise enable both."
|
|
||||||
"It may only work over USB, and is currently only supported "
|
|
||||||
"on Linux.\n"
|
|
||||||
"See --hid-keyboard and --hid-mouse.",
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
.shortopt = 'p',
|
.shortopt = 'p',
|
||||||
.longopt = "port",
|
.longopt = "port",
|
||||||
@@ -1335,7 +1318,7 @@ parse_args_with_getopt(struct scrcpy_cli_args *args, int argc, char *argv[],
|
|||||||
args->help = true;
|
args->help = true;
|
||||||
break;
|
break;
|
||||||
case 'K':
|
case 'K':
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
opts->keyboard_input_mode = SC_KEYBOARD_INPUT_MODE_HID;
|
opts->keyboard_input_mode = SC_KEYBOARD_INPUT_MODE_HID;
|
||||||
break;
|
break;
|
||||||
#else
|
#else
|
||||||
@@ -1354,7 +1337,7 @@ parse_args_with_getopt(struct scrcpy_cli_args *args, int argc, char *argv[],
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'M':
|
case 'M':
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
opts->mouse_input_mode = SC_MOUSE_INPUT_MODE_HID;
|
opts->mouse_input_mode = SC_MOUSE_INPUT_MODE_HID;
|
||||||
break;
|
break;
|
||||||
#else
|
#else
|
||||||
@@ -1517,15 +1500,6 @@ parse_args_with_getopt(struct scrcpy_cli_args *args, int argc, char *argv[],
|
|||||||
case OPT_NO_DOWNSIZE_ON_ERROR:
|
case OPT_NO_DOWNSIZE_ON_ERROR:
|
||||||
opts->downsize_on_error = false;
|
opts->downsize_on_error = false;
|
||||||
break;
|
break;
|
||||||
case OPT_OTG:
|
|
||||||
#ifdef HAVE_USB
|
|
||||||
opts->otg = true;
|
|
||||||
break;
|
|
||||||
#else
|
|
||||||
LOGE("OTG mode (--otg) is not supported on this platform. It "
|
|
||||||
"is only available on Linux.");
|
|
||||||
return false;
|
|
||||||
#endif
|
|
||||||
case OPT_V4L2_SINK:
|
case OPT_V4L2_SINK:
|
||||||
#ifdef HAVE_V4L2
|
#ifdef HAVE_V4L2
|
||||||
opts->v4l2_device = optarg;
|
opts->v4l2_device = optarg;
|
||||||
@@ -1636,43 +1610,6 @@ parse_args_with_getopt(struct scrcpy_cli_args *args, int argc, char *argv[],
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef HAVE_USB
|
|
||||||
if (opts->otg) {
|
|
||||||
// OTG mode is compatible with only very few options.
|
|
||||||
// Only report obvious errors.
|
|
||||||
if (opts->record_filename) {
|
|
||||||
LOGE("OTG mode: could not record");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (opts->turn_screen_off) {
|
|
||||||
LOGE("OTG mode: could not turn screen off");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (opts->stay_awake) {
|
|
||||||
LOGE("OTG mode: could not stay awake");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (opts->show_touches) {
|
|
||||||
LOGE("OTG mode: could not request to show touches");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (opts->power_off_on_close) {
|
|
||||||
LOGE("OTG mode: could not request power off on close");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (opts->display_id) {
|
|
||||||
LOGE("OTG mode: could not select display");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
#ifdef HAVE_V4L2
|
|
||||||
if (opts->v4l2_device) {
|
|
||||||
LOGE("OTG mode: could not sink to V4L2 device");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,4 +2,3 @@
|
|||||||
#define EVENT_STREAM_STOPPED (SDL_USEREVENT + 1)
|
#define EVENT_STREAM_STOPPED (SDL_USEREVENT + 1)
|
||||||
#define EVENT_SERVER_CONNECTION_FAILED (SDL_USEREVENT + 2)
|
#define EVENT_SERVER_CONNECTION_FAILED (SDL_USEREVENT + 2)
|
||||||
#define EVENT_SERVER_CONNECTED (SDL_USEREVENT + 3)
|
#define EVENT_SERVER_CONNECTED (SDL_USEREVENT + 3)
|
||||||
#define EVENT_USB_DEVICE_DISCONNECTED (SDL_USEREVENT + 4)
|
|
||||||
|
|||||||
@@ -51,6 +51,8 @@ swap_frames(AVFrame **lhs, AVFrame **rhs) {
|
|||||||
bool
|
bool
|
||||||
sc_frame_buffer_push(struct sc_frame_buffer *fb, const AVFrame *frame,
|
sc_frame_buffer_push(struct sc_frame_buffer *fb, const AVFrame *frame,
|
||||||
bool *previous_frame_skipped) {
|
bool *previous_frame_skipped) {
|
||||||
|
sc_mutex_lock(&fb->mutex);
|
||||||
|
|
||||||
// Use a temporary frame to preserve pending_frame in case of error.
|
// Use a temporary frame to preserve pending_frame in case of error.
|
||||||
// tmp_frame is an empty frame, no need to call av_frame_unref() beforehand.
|
// tmp_frame is an empty frame, no need to call av_frame_unref() beforehand.
|
||||||
int r = av_frame_ref(fb->tmp_frame, frame);
|
int r = av_frame_ref(fb->tmp_frame, frame);
|
||||||
@@ -59,8 +61,6 @@ sc_frame_buffer_push(struct sc_frame_buffer *fb, const AVFrame *frame,
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
sc_mutex_lock(&fb->mutex);
|
|
||||||
|
|
||||||
// Now that av_frame_ref() succeeded, we can replace the previous
|
// Now that av_frame_ref() succeeded, we can replace the previous
|
||||||
// pending_frame
|
// pending_frame
|
||||||
swap_frames(&fb->pending_frame, &fb->tmp_frame);
|
swap_frames(&fb->pending_frame, &fb->tmp_frame);
|
||||||
|
|||||||
@@ -377,76 +377,4 @@ struct sc_touch_event {
|
|||||||
float pressure;
|
float pressure;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline uint16_t
|
|
||||||
sc_mods_state_from_sdl(uint16_t mods_state) {
|
|
||||||
return mods_state;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline enum sc_keycode
|
|
||||||
sc_keycode_from_sdl(SDL_Keycode keycode) {
|
|
||||||
return (enum sc_keycode) keycode;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline enum sc_scancode
|
|
||||||
sc_scancode_from_sdl(SDL_Scancode scancode) {
|
|
||||||
return (enum sc_scancode) scancode;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline enum sc_action
|
|
||||||
sc_action_from_sdl_keyboard_type(uint32_t type) {
|
|
||||||
assert(type == SDL_KEYDOWN || type == SDL_KEYUP);
|
|
||||||
if (type == SDL_KEYDOWN) {
|
|
||||||
return SC_ACTION_DOWN;
|
|
||||||
}
|
|
||||||
return SC_ACTION_UP;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline enum sc_action
|
|
||||||
sc_action_from_sdl_mousebutton_type(uint32_t type) {
|
|
||||||
assert(type == SDL_MOUSEBUTTONDOWN || type == SDL_MOUSEBUTTONUP);
|
|
||||||
if (type == SDL_MOUSEBUTTONDOWN) {
|
|
||||||
return SC_ACTION_DOWN;
|
|
||||||
}
|
|
||||||
return SC_ACTION_UP;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline enum sc_touch_action
|
|
||||||
sc_touch_action_from_sdl(uint32_t type) {
|
|
||||||
assert(type == SDL_FINGERMOTION || type == SDL_FINGERDOWN ||
|
|
||||||
type == SDL_FINGERUP);
|
|
||||||
if (type == SDL_FINGERMOTION) {
|
|
||||||
return SC_TOUCH_ACTION_MOVE;
|
|
||||||
}
|
|
||||||
if (type == SDL_FINGERDOWN) {
|
|
||||||
return SC_TOUCH_ACTION_DOWN;
|
|
||||||
}
|
|
||||||
return SC_TOUCH_ACTION_UP;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline enum sc_mouse_button
|
|
||||||
sc_mouse_button_from_sdl(uint8_t button) {
|
|
||||||
if (button >= SDL_BUTTON_LEFT && button <= SDL_BUTTON_X2) {
|
|
||||||
// SC_MOUSE_BUTTON_* constants are initialized from SDL_BUTTON(index)
|
|
||||||
return SDL_BUTTON(button);
|
|
||||||
}
|
|
||||||
|
|
||||||
return SC_MOUSE_BUTTON_UNKNOWN;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline uint8_t
|
|
||||||
sc_mouse_buttons_state_from_sdl(uint32_t buttons_state,
|
|
||||||
bool forward_all_clicks) {
|
|
||||||
assert(buttons_state < 0x100); // fits in uint8_t
|
|
||||||
|
|
||||||
uint8_t mask = SC_MOUSE_BUTTON_LEFT;
|
|
||||||
if (forward_all_clicks) {
|
|
||||||
mask |= SC_MOUSE_BUTTON_RIGHT
|
|
||||||
| SC_MOUSE_BUTTON_MIDDLE
|
|
||||||
| SC_MOUSE_BUTTON_X1
|
|
||||||
| SC_MOUSE_BUTTON_X2;
|
|
||||||
}
|
|
||||||
|
|
||||||
return buttons_state & mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -7,6 +7,78 @@
|
|||||||
#include "screen.h"
|
#include "screen.h"
|
||||||
#include "util/log.h"
|
#include "util/log.h"
|
||||||
|
|
||||||
|
static inline uint16_t
|
||||||
|
sc_mods_state_from_sdl(uint16_t mods_state) {
|
||||||
|
return mods_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline enum sc_keycode
|
||||||
|
sc_keycode_from_sdl(SDL_Keycode keycode) {
|
||||||
|
return (enum sc_keycode) keycode;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline enum sc_scancode
|
||||||
|
sc_scancode_from_sdl(SDL_Scancode scancode) {
|
||||||
|
return (enum sc_scancode) scancode;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline enum sc_action
|
||||||
|
sc_action_from_sdl_keyboard_type(uint32_t type) {
|
||||||
|
assert(type == SDL_KEYDOWN || type == SDL_KEYUP);
|
||||||
|
if (type == SDL_KEYDOWN) {
|
||||||
|
return SC_ACTION_DOWN;
|
||||||
|
}
|
||||||
|
return SC_ACTION_UP;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline enum sc_action
|
||||||
|
sc_action_from_sdl_mousebutton_type(uint32_t type) {
|
||||||
|
assert(type == SDL_MOUSEBUTTONDOWN || type == SDL_MOUSEBUTTONUP);
|
||||||
|
if (type == SDL_MOUSEBUTTONDOWN) {
|
||||||
|
return SC_ACTION_DOWN;
|
||||||
|
}
|
||||||
|
return SC_ACTION_UP;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline enum sc_touch_action
|
||||||
|
sc_touch_action_from_sdl(uint32_t type) {
|
||||||
|
assert(type == SDL_FINGERMOTION || type == SDL_FINGERDOWN ||
|
||||||
|
type == SDL_FINGERUP);
|
||||||
|
if (type == SDL_FINGERMOTION) {
|
||||||
|
return SC_TOUCH_ACTION_MOVE;
|
||||||
|
}
|
||||||
|
if (type == SDL_FINGERDOWN) {
|
||||||
|
return SC_TOUCH_ACTION_DOWN;
|
||||||
|
}
|
||||||
|
return SC_TOUCH_ACTION_UP;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline enum sc_mouse_button
|
||||||
|
sc_mouse_button_from_sdl(uint8_t button) {
|
||||||
|
if (button >= SDL_BUTTON_LEFT && button <= SDL_BUTTON_X2) {
|
||||||
|
// SC_MOUSE_BUTTON_* constants are initialized from SDL_BUTTON(index)
|
||||||
|
return SDL_BUTTON(button);
|
||||||
|
}
|
||||||
|
|
||||||
|
return SC_MOUSE_BUTTON_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline uint8_t
|
||||||
|
sc_mouse_buttons_state_from_sdl(uint32_t buttons_state,
|
||||||
|
bool forward_all_clicks) {
|
||||||
|
assert(buttons_state < 0x100); // fits in uint8_t
|
||||||
|
|
||||||
|
uint8_t mask = SC_MOUSE_BUTTON_LEFT;
|
||||||
|
if (forward_all_clicks) {
|
||||||
|
mask |= SC_MOUSE_BUTTON_RIGHT
|
||||||
|
| SC_MOUSE_BUTTON_MIDDLE
|
||||||
|
| SC_MOUSE_BUTTON_X1
|
||||||
|
| SC_MOUSE_BUTTON_X2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return buttons_state & mask;
|
||||||
|
}
|
||||||
|
|
||||||
#define SC_SDL_SHORTCUT_MODS_MASK (KMOD_CTRL | KMOD_ALT | KMOD_GUI)
|
#define SC_SDL_SHORTCUT_MODS_MASK (KMOD_CTRL | KMOD_ALT | KMOD_GUI)
|
||||||
|
|
||||||
static inline uint16_t
|
static inline uint16_t
|
||||||
|
|||||||
@@ -13,25 +13,26 @@
|
|||||||
#include "cli.h"
|
#include "cli.h"
|
||||||
#include "options.h"
|
#include "options.h"
|
||||||
#include "scrcpy.h"
|
#include "scrcpy.h"
|
||||||
#include "usb/scrcpy_otg.h"
|
|
||||||
#include "util/log.h"
|
#include "util/log.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
print_version(void) {
|
print_version(void) {
|
||||||
printf("\ndependencies:\n");
|
fprintf(stderr, "scrcpy %s\n\n", SCRCPY_VERSION);
|
||||||
printf(" - SDL %d.%d.%d\n", SDL_MAJOR_VERSION, SDL_MINOR_VERSION,
|
|
||||||
|
fprintf(stderr, "dependencies:\n");
|
||||||
|
fprintf(stderr, " - SDL %d.%d.%d\n", SDL_MAJOR_VERSION, SDL_MINOR_VERSION,
|
||||||
SDL_PATCHLEVEL);
|
SDL_PATCHLEVEL);
|
||||||
printf(" - libavcodec %d.%d.%d\n", LIBAVCODEC_VERSION_MAJOR,
|
fprintf(stderr, " - libavcodec %d.%d.%d\n", LIBAVCODEC_VERSION_MAJOR,
|
||||||
LIBAVCODEC_VERSION_MINOR,
|
LIBAVCODEC_VERSION_MINOR,
|
||||||
LIBAVCODEC_VERSION_MICRO);
|
LIBAVCODEC_VERSION_MICRO);
|
||||||
printf(" - libavformat %d.%d.%d\n", LIBAVFORMAT_VERSION_MAJOR,
|
fprintf(stderr, " - libavformat %d.%d.%d\n", LIBAVFORMAT_VERSION_MAJOR,
|
||||||
LIBAVFORMAT_VERSION_MINOR,
|
LIBAVFORMAT_VERSION_MINOR,
|
||||||
LIBAVFORMAT_VERSION_MICRO);
|
LIBAVFORMAT_VERSION_MICRO);
|
||||||
printf(" - libavutil %d.%d.%d\n", LIBAVUTIL_VERSION_MAJOR,
|
fprintf(stderr, " - libavutil %d.%d.%d\n", LIBAVUTIL_VERSION_MAJOR,
|
||||||
LIBAVUTIL_VERSION_MINOR,
|
LIBAVUTIL_VERSION_MINOR,
|
||||||
LIBAVUTIL_VERSION_MICRO);
|
LIBAVUTIL_VERSION_MICRO);
|
||||||
#ifdef HAVE_V4L2
|
#ifdef HAVE_V4L2
|
||||||
printf(" - libavdevice %d.%d.%d\n", LIBAVDEVICE_VERSION_MAJOR,
|
fprintf(stderr, " - libavdevice %d.%d.%d\n", LIBAVDEVICE_VERSION_MAJOR,
|
||||||
LIBAVDEVICE_VERSION_MINOR,
|
LIBAVDEVICE_VERSION_MINOR,
|
||||||
LIBAVDEVICE_VERSION_MICRO);
|
LIBAVDEVICE_VERSION_MICRO);
|
||||||
#endif
|
#endif
|
||||||
@@ -89,14 +90,9 @@ main(int argc, char *argv[]) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef HAVE_USB
|
int res = scrcpy(&args.opts) ? 0 : 1;
|
||||||
bool ok = args.opts.otg ? scrcpy_otg(&args.opts)
|
|
||||||
: scrcpy(&args.opts);
|
|
||||||
#else
|
|
||||||
bool ok = scrcpy(&args.opts);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
avformat_network_deinit(); // ignore failure
|
avformat_network_deinit(); // ignore failure
|
||||||
|
|
||||||
return ok ? 0 : 1;
|
return res;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,9 +37,6 @@ const struct scrcpy_options scrcpy_options_default = {
|
|||||||
.display_id = 0,
|
.display_id = 0,
|
||||||
.display_buffer = 0,
|
.display_buffer = 0,
|
||||||
.v4l2_buffer = 0,
|
.v4l2_buffer = 0,
|
||||||
#ifdef HAVE_USB
|
|
||||||
.otg = false,
|
|
||||||
#endif
|
|
||||||
.show_touches = false,
|
.show_touches = false,
|
||||||
.fullscreen = false,
|
.fullscreen = false,
|
||||||
.always_on_top = false,
|
.always_on_top = false,
|
||||||
|
|||||||
@@ -112,9 +112,6 @@ struct scrcpy_options {
|
|||||||
uint32_t display_id;
|
uint32_t display_id;
|
||||||
sc_tick display_buffer;
|
sc_tick display_buffer;
|
||||||
sc_tick v4l2_buffer;
|
sc_tick v4l2_buffer;
|
||||||
#ifdef HAVE_USB
|
|
||||||
bool otg;
|
|
||||||
#endif
|
|
||||||
bool show_touches;
|
bool show_touches;
|
||||||
bool fullscreen;
|
bool fullscreen;
|
||||||
bool always_on_top;
|
bool always_on_top;
|
||||||
|
|||||||
@@ -17,18 +17,16 @@
|
|||||||
#include "decoder.h"
|
#include "decoder.h"
|
||||||
#include "events.h"
|
#include "events.h"
|
||||||
#include "file_pusher.h"
|
#include "file_pusher.h"
|
||||||
|
#ifdef HAVE_AOA_HID
|
||||||
|
# include "hid_keyboard.h"
|
||||||
|
# include "hid_mouse.h"
|
||||||
|
#endif
|
||||||
#include "keyboard_inject.h"
|
#include "keyboard_inject.h"
|
||||||
#include "mouse_inject.h"
|
#include "mouse_inject.h"
|
||||||
#include "recorder.h"
|
#include "recorder.h"
|
||||||
#include "screen.h"
|
#include "screen.h"
|
||||||
#include "server.h"
|
#include "server.h"
|
||||||
#include "stream.h"
|
#include "stream.h"
|
||||||
#ifdef HAVE_USB
|
|
||||||
# include "usb/aoa_hid.h"
|
|
||||||
# include "usb/hid_keyboard.h"
|
|
||||||
# include "usb/hid_mouse.h"
|
|
||||||
# include "usb/usb.h"
|
|
||||||
#endif
|
|
||||||
#include "util/acksync.h"
|
#include "util/acksync.h"
|
||||||
#include "util/log.h"
|
#include "util/log.h"
|
||||||
#include "util/net.h"
|
#include "util/net.h"
|
||||||
@@ -47,21 +45,20 @@ struct scrcpy {
|
|||||||
#endif
|
#endif
|
||||||
struct sc_controller controller;
|
struct sc_controller controller;
|
||||||
struct sc_file_pusher file_pusher;
|
struct sc_file_pusher file_pusher;
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
struct sc_usb usb;
|
|
||||||
struct sc_aoa aoa;
|
struct sc_aoa aoa;
|
||||||
// sequence/ack helper to synchronize clipboard and Ctrl+v via HID
|
// sequence/ack helper to synchronize clipboard and Ctrl+v via HID
|
||||||
struct sc_acksync acksync;
|
struct sc_acksync acksync;
|
||||||
#endif
|
#endif
|
||||||
union {
|
union {
|
||||||
struct sc_keyboard_inject keyboard_inject;
|
struct sc_keyboard_inject keyboard_inject;
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
struct sc_hid_keyboard keyboard_hid;
|
struct sc_hid_keyboard keyboard_hid;
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
union {
|
union {
|
||||||
struct sc_mouse_inject mouse_inject;
|
struct sc_mouse_inject mouse_inject;
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
struct sc_hid_mouse mouse_hid;
|
struct sc_hid_mouse mouse_hid;
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
@@ -286,7 +283,7 @@ scrcpy(struct scrcpy_options *options) {
|
|||||||
bool v4l2_sink_initialized = false;
|
bool v4l2_sink_initialized = false;
|
||||||
#endif
|
#endif
|
||||||
bool stream_started = false;
|
bool stream_started = false;
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
bool aoa_hid_initialized = false;
|
bool aoa_hid_initialized = false;
|
||||||
bool hid_keyboard_initialized = false;
|
bool hid_keyboard_initialized = false;
|
||||||
bool hid_mouse_initialized = false;
|
bool hid_mouse_initialized = false;
|
||||||
@@ -413,7 +410,7 @@ scrcpy(struct scrcpy_options *options) {
|
|||||||
struct sc_mouse_processor *mp = NULL;
|
struct sc_mouse_processor *mp = NULL;
|
||||||
|
|
||||||
if (options->control) {
|
if (options->control) {
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
bool use_hid_keyboard =
|
bool use_hid_keyboard =
|
||||||
options->keyboard_input_mode == SC_KEYBOARD_INPUT_MODE_HID;
|
options->keyboard_input_mode == SC_KEYBOARD_INPUT_MODE_HID;
|
||||||
bool use_hid_mouse =
|
bool use_hid_mouse =
|
||||||
@@ -424,52 +421,9 @@ scrcpy(struct scrcpy_options *options) {
|
|||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
|
||||||
ok = sc_usb_init(&s->usb);
|
ok = sc_aoa_init(&s->aoa, serial, &s->acksync);
|
||||||
if (!ok) {
|
|
||||||
LOGE("Failed to initialize USB");
|
|
||||||
sc_acksync_destroy(&s->acksync);
|
|
||||||
goto aoa_hid_end;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(serial);
|
|
||||||
struct sc_usb_device usb_devices[16];
|
|
||||||
ssize_t count = sc_usb_find_devices(&s->usb, serial, usb_devices,
|
|
||||||
ARRAY_LEN(usb_devices));
|
|
||||||
if (count <= 0) {
|
|
||||||
LOGE("Could not find USB device %s", serial);
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
sc_acksync_destroy(&s->acksync);
|
|
||||||
goto aoa_hid_end;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count > 1) {
|
|
||||||
LOGE("Multiple (%d) devices with serial %s", (int) count, serial);
|
|
||||||
sc_usb_device_destroy_all(usb_devices, count);
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
sc_acksync_destroy(&s->acksync);
|
|
||||||
goto aoa_hid_end;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct sc_usb_device *usb_device = &usb_devices[0];
|
|
||||||
|
|
||||||
LOGI("USB device: %s (%04" PRIx16 ":%04" PRIx16 ") %s %s",
|
|
||||||
usb_device->serial, usb_device->vid, usb_device->pid,
|
|
||||||
usb_device->manufacturer, usb_device->product);
|
|
||||||
|
|
||||||
ok = sc_usb_connect(&s->usb, usb_device->device, NULL, NULL);
|
|
||||||
sc_usb_device_destroy(usb_device);
|
|
||||||
if (!ok) {
|
|
||||||
LOGE("Failed to connect to USB device %s", serial);
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
sc_acksync_destroy(&s->acksync);
|
|
||||||
goto aoa_hid_end;
|
|
||||||
}
|
|
||||||
|
|
||||||
ok = sc_aoa_init(&s->aoa, &s->usb, &s->acksync);
|
|
||||||
if (!ok) {
|
if (!ok) {
|
||||||
LOGE("Failed to enable HID over AOA");
|
LOGE("Failed to enable HID over AOA");
|
||||||
sc_usb_disconnect(&s->usb);
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
sc_acksync_destroy(&s->acksync);
|
sc_acksync_destroy(&s->acksync);
|
||||||
goto aoa_hid_end;
|
goto aoa_hid_end;
|
||||||
}
|
}
|
||||||
@@ -496,8 +450,6 @@ scrcpy(struct scrcpy_options *options) {
|
|||||||
|
|
||||||
if (!need_aoa || !sc_aoa_start(&s->aoa)) {
|
if (!need_aoa || !sc_aoa_start(&s->aoa)) {
|
||||||
sc_acksync_destroy(&s->acksync);
|
sc_acksync_destroy(&s->acksync);
|
||||||
sc_usb_disconnect(&s->usb);
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
sc_aoa_destroy(&s->aoa);
|
sc_aoa_destroy(&s->aoa);
|
||||||
goto aoa_hid_end;
|
goto aoa_hid_end;
|
||||||
}
|
}
|
||||||
@@ -641,7 +593,7 @@ aoa_hid_end:
|
|||||||
end:
|
end:
|
||||||
// The stream is not stopped explicitly, because it will stop by itself on
|
// The stream is not stopped explicitly, because it will stop by itself on
|
||||||
// end-of-stream
|
// end-of-stream
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
if (aoa_hid_initialized) {
|
if (aoa_hid_initialized) {
|
||||||
if (hid_keyboard_initialized) {
|
if (hid_keyboard_initialized) {
|
||||||
sc_hid_keyboard_destroy(&s->keyboard_hid);
|
sc_hid_keyboard_destroy(&s->keyboard_hid);
|
||||||
@@ -650,7 +602,6 @@ end:
|
|||||||
sc_hid_mouse_destroy(&s->mouse_hid);
|
sc_hid_mouse_destroy(&s->mouse_hid);
|
||||||
}
|
}
|
||||||
sc_aoa_stop(&s->aoa);
|
sc_aoa_stop(&s->aoa);
|
||||||
sc_usb_stop(&s->usb);
|
|
||||||
}
|
}
|
||||||
if (acksync) {
|
if (acksync) {
|
||||||
sc_acksync_destroy(acksync);
|
sc_acksync_destroy(acksync);
|
||||||
@@ -683,13 +634,10 @@ end:
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef HAVE_USB
|
#ifdef HAVE_AOA_HID
|
||||||
if (aoa_hid_initialized) {
|
if (aoa_hid_initialized) {
|
||||||
sc_aoa_join(&s->aoa);
|
sc_aoa_join(&s->aoa);
|
||||||
sc_aoa_destroy(&s->aoa);
|
sc_aoa_destroy(&s->aoa);
|
||||||
sc_usb_join(&s->usb);
|
|
||||||
sc_usb_disconnect(&s->usb);
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -769,10 +769,12 @@ run_server(void *data) {
|
|||||||
// Interrupt sockets to wake up socket blocking calls on the server
|
// Interrupt sockets to wake up socket blocking calls on the server
|
||||||
assert(server->video_socket != SC_SOCKET_NONE);
|
assert(server->video_socket != SC_SOCKET_NONE);
|
||||||
net_interrupt(server->video_socket);
|
net_interrupt(server->video_socket);
|
||||||
|
net_close(server->video_socket);
|
||||||
|
|
||||||
if (server->control_socket != SC_SOCKET_NONE) {
|
if (server->control_socket != SC_SOCKET_NONE) {
|
||||||
// There is no control_socket if --no-control is set
|
// There is no control_socket if --no-control is set
|
||||||
net_interrupt(server->control_socket);
|
net_interrupt(server->control_socket);
|
||||||
|
net_close(server->control_socket);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Give some delay for the server to terminate properly
|
// Give some delay for the server to terminate properly
|
||||||
@@ -828,13 +830,6 @@ sc_server_stop(struct sc_server *server) {
|
|||||||
|
|
||||||
void
|
void
|
||||||
sc_server_destroy(struct sc_server *server) {
|
sc_server_destroy(struct sc_server *server) {
|
||||||
if (server->video_socket != SC_SOCKET_NONE) {
|
|
||||||
net_close(server->video_socket);
|
|
||||||
}
|
|
||||||
if (server->control_socket != SC_SOCKET_NONE) {
|
|
||||||
net_close(server->control_socket);
|
|
||||||
}
|
|
||||||
|
|
||||||
sc_server_params_destroy(&server->params);
|
sc_server_params_destroy(&server->params);
|
||||||
sc_intr_destroy(&server->intr);
|
sc_intr_destroy(&server->intr);
|
||||||
sc_cond_destroy(&server->cond_stopped);
|
sc_cond_destroy(&server->cond_stopped);
|
||||||
|
|||||||
@@ -1,232 +0,0 @@
|
|||||||
#include "scrcpy_otg.h"
|
|
||||||
|
|
||||||
#include <SDL2/SDL.h>
|
|
||||||
|
|
||||||
#include "events.h"
|
|
||||||
#include "screen_otg.h"
|
|
||||||
#include "util/log.h"
|
|
||||||
|
|
||||||
struct scrcpy_otg {
|
|
||||||
struct sc_usb usb;
|
|
||||||
struct sc_aoa aoa;
|
|
||||||
struct sc_hid_keyboard keyboard;
|
|
||||||
struct sc_hid_mouse mouse;
|
|
||||||
|
|
||||||
struct sc_screen_otg screen_otg;
|
|
||||||
};
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_usb_on_disconnected(struct sc_usb *usb, void *userdata) {
|
|
||||||
(void) usb;
|
|
||||||
(void) userdata;
|
|
||||||
|
|
||||||
SDL_Event event;
|
|
||||||
event.type = EVENT_USB_DEVICE_DISCONNECTED;
|
|
||||||
int ret = SDL_PushEvent(&event);
|
|
||||||
if (ret < 0) {
|
|
||||||
LOGE("Could not post USB disconnection event: %s", SDL_GetError());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
event_loop(struct scrcpy_otg *s) {
|
|
||||||
SDL_Event event;
|
|
||||||
while (SDL_WaitEvent(&event)) {
|
|
||||||
switch (event.type) {
|
|
||||||
case EVENT_USB_DEVICE_DISCONNECTED:
|
|
||||||
LOGW("Device disconnected");
|
|
||||||
return false;
|
|
||||||
case SDL_QUIT:
|
|
||||||
LOGD("User requested to quit");
|
|
||||||
return true;
|
|
||||||
default:
|
|
||||||
sc_screen_otg_handle_event(&s->screen_otg, &event);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
scrcpy_otg(struct scrcpy_options *options) {
|
|
||||||
static struct scrcpy_otg scrcpy_otg;
|
|
||||||
struct scrcpy_otg *s = &scrcpy_otg;
|
|
||||||
|
|
||||||
const char *serial = options->serial;
|
|
||||||
|
|
||||||
// Minimal SDL initialization
|
|
||||||
if (SDL_Init(SDL_INIT_EVENTS)) {
|
|
||||||
LOGC("Could not initialize SDL: %s", SDL_GetError());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
atexit(SDL_Quit);
|
|
||||||
|
|
||||||
if (!SDL_SetHint(SDL_HINT_MOUSE_FOCUS_CLICKTHROUGH, "1")) {
|
|
||||||
LOGW("Could not enable mouse focus clickthrough");
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ret = false;
|
|
||||||
|
|
||||||
struct sc_hid_keyboard *keyboard = NULL;
|
|
||||||
struct sc_hid_mouse *mouse = NULL;
|
|
||||||
bool usb_device_initialized = false;
|
|
||||||
bool usb_connected = false;
|
|
||||||
bool aoa_started = false;
|
|
||||||
bool aoa_initialized = false;
|
|
||||||
|
|
||||||
static const struct sc_usb_callbacks cbs = {
|
|
||||||
.on_disconnected = sc_usb_on_disconnected,
|
|
||||||
};
|
|
||||||
bool ok = sc_usb_init(&s->usb);
|
|
||||||
if (!ok) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct sc_usb_device usb_devices[16];
|
|
||||||
ssize_t count = sc_usb_find_devices(&s->usb, serial, usb_devices,
|
|
||||||
ARRAY_LEN(usb_devices));
|
|
||||||
if (count < 0) {
|
|
||||||
LOGE("Could not list USB devices");
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count == 0) {
|
|
||||||
if (serial) {
|
|
||||||
LOGE("Could not find USB device %s", serial);
|
|
||||||
} else {
|
|
||||||
LOGE("Could not find any USB device");
|
|
||||||
}
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count > 1) {
|
|
||||||
if (serial) {
|
|
||||||
LOGE("Multiple (%d) USB devices with serial %s:", (int) count,
|
|
||||||
serial);
|
|
||||||
} else {
|
|
||||||
LOGE("Multiple (%d) USB devices:", (int) count);
|
|
||||||
}
|
|
||||||
for (size_t i = 0; i < (size_t) count; ++i) {
|
|
||||||
struct sc_usb_device *d = &usb_devices[i];
|
|
||||||
LOGE(" %-18s (%04" PRIx16 ":%04" PRIx16 ") %s %s",
|
|
||||||
d->serial, d->vid, d->pid, d->manufacturer, d->product);
|
|
||||||
}
|
|
||||||
if (!serial) {
|
|
||||||
LOGE("Specify the device via -s or --serial");
|
|
||||||
}
|
|
||||||
sc_usb_device_destroy_all(usb_devices, count);
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
usb_device_initialized = true;
|
|
||||||
|
|
||||||
struct sc_usb_device *usb_device = &usb_devices[0];
|
|
||||||
|
|
||||||
LOGI("USB device: %s (%04" PRIx16 ":%04" PRIx16 ") %s %s",
|
|
||||||
usb_device->serial, usb_device->vid, usb_device->pid,
|
|
||||||
usb_device->manufacturer, usb_device->product);
|
|
||||||
|
|
||||||
ok = sc_usb_connect(&s->usb, usb_device->device, &cbs, NULL);
|
|
||||||
if (!ok) {
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
usb_connected = true;
|
|
||||||
|
|
||||||
ok = sc_aoa_init(&s->aoa, &s->usb, NULL);
|
|
||||||
if (!ok) {
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
aoa_initialized = true;
|
|
||||||
|
|
||||||
bool enable_keyboard =
|
|
||||||
options->keyboard_input_mode == SC_KEYBOARD_INPUT_MODE_HID;
|
|
||||||
bool enable_mouse =
|
|
||||||
options->mouse_input_mode == SC_MOUSE_INPUT_MODE_HID;
|
|
||||||
|
|
||||||
// If neither --hid-keyboard or --hid-mouse is passed, enable both
|
|
||||||
if (!enable_keyboard && !enable_mouse) {
|
|
||||||
enable_keyboard = true;
|
|
||||||
enable_mouse = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (enable_keyboard) {
|
|
||||||
ok = sc_hid_keyboard_init(&s->keyboard, &s->aoa);
|
|
||||||
if (!ok) {
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
keyboard = &s->keyboard;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (enable_mouse) {
|
|
||||||
ok = sc_hid_mouse_init(&s->mouse, &s->aoa);
|
|
||||||
if (!ok) {
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
mouse = &s->mouse;
|
|
||||||
}
|
|
||||||
|
|
||||||
ok = sc_aoa_start(&s->aoa);
|
|
||||||
if (!ok) {
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
aoa_started = true;
|
|
||||||
|
|
||||||
const char *window_title = options->window_title;
|
|
||||||
if (!window_title) {
|
|
||||||
window_title = usb_device->product ? usb_device->product : "scrcpy";
|
|
||||||
}
|
|
||||||
|
|
||||||
struct sc_screen_otg_params params = {
|
|
||||||
.keyboard = keyboard,
|
|
||||||
.mouse = mouse,
|
|
||||||
.window_title = window_title,
|
|
||||||
.always_on_top = options->always_on_top,
|
|
||||||
.window_x = options->window_x,
|
|
||||||
.window_y = options->window_y,
|
|
||||||
.window_borderless = options->window_borderless,
|
|
||||||
};
|
|
||||||
|
|
||||||
ok = sc_screen_otg_init(&s->screen_otg, ¶ms);
|
|
||||||
if (!ok) {
|
|
||||||
goto end;
|
|
||||||
}
|
|
||||||
|
|
||||||
// usb_device not needed anymore
|
|
||||||
sc_usb_device_destroy(usb_device);
|
|
||||||
usb_device_initialized = false;
|
|
||||||
|
|
||||||
ret = event_loop(s);
|
|
||||||
LOGD("quit...");
|
|
||||||
|
|
||||||
end:
|
|
||||||
if (aoa_started) {
|
|
||||||
sc_aoa_stop(&s->aoa);
|
|
||||||
}
|
|
||||||
sc_usb_stop(&s->usb);
|
|
||||||
|
|
||||||
if (mouse) {
|
|
||||||
sc_hid_mouse_destroy(&s->mouse);
|
|
||||||
}
|
|
||||||
if (keyboard) {
|
|
||||||
sc_hid_keyboard_destroy(&s->keyboard);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (aoa_initialized) {
|
|
||||||
sc_aoa_join(&s->aoa);
|
|
||||||
sc_aoa_destroy(&s->aoa);
|
|
||||||
}
|
|
||||||
|
|
||||||
sc_usb_join(&s->usb);
|
|
||||||
|
|
||||||
if (usb_connected) {
|
|
||||||
sc_usb_disconnect(&s->usb);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (usb_device_initialized) {
|
|
||||||
sc_usb_device_destroy(usb_device);
|
|
||||||
}
|
|
||||||
|
|
||||||
sc_usb_destroy(&s->usb);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
#ifndef SCRCPY_OTG_H
|
|
||||||
#define SCRCPY_OTG_H
|
|
||||||
|
|
||||||
#include "common.h"
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include "options.h"
|
|
||||||
|
|
||||||
bool
|
|
||||||
scrcpy_otg(struct scrcpy_options *options);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -1,267 +0,0 @@
|
|||||||
#include "screen_otg.h"
|
|
||||||
|
|
||||||
#include "icon.h"
|
|
||||||
#include "options.h"
|
|
||||||
#include "util/log.h"
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_screen_otg_capture_mouse(struct sc_screen_otg *screen, bool capture) {
|
|
||||||
assert(screen->mouse);
|
|
||||||
if (SDL_SetRelativeMouseMode(capture)) {
|
|
||||||
LOGE("Could not set relative mouse mode to %s: %s",
|
|
||||||
capture ? "true" : "false", SDL_GetError());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
screen->mouse_captured = capture;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_screen_otg_render(struct sc_screen_otg *screen) {
|
|
||||||
SDL_RenderClear(screen->renderer);
|
|
||||||
if (screen->texture) {
|
|
||||||
SDL_RenderCopy(screen->renderer, screen->texture, NULL, NULL);
|
|
||||||
}
|
|
||||||
SDL_RenderPresent(screen->renderer);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
sc_screen_otg_init(struct sc_screen_otg *screen,
|
|
||||||
const struct sc_screen_otg_params *params) {
|
|
||||||
screen->keyboard = params->keyboard;
|
|
||||||
screen->mouse = params->mouse;
|
|
||||||
|
|
||||||
screen->mouse_captured = false;
|
|
||||||
screen->mouse_capture_key_pressed = 0;
|
|
||||||
|
|
||||||
const char *title = params->window_title;
|
|
||||||
assert(title);
|
|
||||||
|
|
||||||
int x = params->window_x != SC_WINDOW_POSITION_UNDEFINED
|
|
||||||
? params->window_x : (int) SDL_WINDOWPOS_UNDEFINED;
|
|
||||||
int y = params->window_y != SC_WINDOW_POSITION_UNDEFINED
|
|
||||||
? params->window_y : (int) SDL_WINDOWPOS_UNDEFINED;
|
|
||||||
int width = 256;
|
|
||||||
int height = 256;
|
|
||||||
|
|
||||||
uint32_t window_flags = 0;
|
|
||||||
if (params->always_on_top) {
|
|
||||||
window_flags |= SDL_WINDOW_ALWAYS_ON_TOP;
|
|
||||||
}
|
|
||||||
if (params->window_borderless) {
|
|
||||||
window_flags |= SDL_WINDOW_BORDERLESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
screen->window = SDL_CreateWindow(title, x, y, width, height, window_flags);
|
|
||||||
if (!screen->window) {
|
|
||||||
LOGE("Could not create window: %s", SDL_GetError());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
screen->renderer = SDL_CreateRenderer(screen->window, -1, 0);
|
|
||||||
if (!screen->renderer) {
|
|
||||||
LOGE("Could not create renderer: %s", SDL_GetError());
|
|
||||||
goto error_destroy_window;
|
|
||||||
}
|
|
||||||
|
|
||||||
SDL_Surface *icon = scrcpy_icon_load();
|
|
||||||
|
|
||||||
if (icon) {
|
|
||||||
SDL_SetWindowIcon(screen->window, icon);
|
|
||||||
|
|
||||||
screen->texture = SDL_CreateTextureFromSurface(screen->renderer, icon);
|
|
||||||
scrcpy_icon_destroy(icon);
|
|
||||||
if (!screen->texture) {
|
|
||||||
goto error_destroy_renderer;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
screen->texture = NULL;
|
|
||||||
LOGW("Could not load icon");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (screen->mouse) {
|
|
||||||
// Capture mouse on start
|
|
||||||
sc_screen_otg_capture_mouse(screen, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
|
|
||||||
error_destroy_window:
|
|
||||||
SDL_DestroyWindow(screen->window);
|
|
||||||
error_destroy_renderer:
|
|
||||||
SDL_DestroyRenderer(screen->renderer);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_screen_otg_destroy(struct sc_screen_otg *screen) {
|
|
||||||
if (screen->texture) {
|
|
||||||
SDL_DestroyTexture(screen->texture);
|
|
||||||
}
|
|
||||||
SDL_DestroyRenderer(screen->renderer);
|
|
||||||
SDL_DestroyWindow(screen->window);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline bool
|
|
||||||
sc_screen_otg_is_mouse_capture_key(SDL_Keycode key) {
|
|
||||||
return key == SDLK_LALT || key == SDLK_LGUI || key == SDLK_RGUI;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_screen_otg_process_key(struct sc_screen_otg *screen,
|
|
||||||
const SDL_KeyboardEvent *event) {
|
|
||||||
assert(screen->keyboard);
|
|
||||||
struct sc_key_processor *kp = &screen->keyboard->key_processor;
|
|
||||||
|
|
||||||
struct sc_key_event evt = {
|
|
||||||
.action = sc_action_from_sdl_keyboard_type(event->type),
|
|
||||||
.keycode = sc_keycode_from_sdl(event->keysym.sym),
|
|
||||||
.scancode = sc_scancode_from_sdl(event->keysym.scancode),
|
|
||||||
.repeat = event->repeat,
|
|
||||||
.mods_state = sc_mods_state_from_sdl(event->keysym.mod),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert(kp->ops->process_key);
|
|
||||||
kp->ops->process_key(kp, &evt, SC_SEQUENCE_INVALID);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_screen_otg_process_mouse_motion(struct sc_screen_otg *screen,
|
|
||||||
const SDL_MouseMotionEvent *event) {
|
|
||||||
assert(screen->mouse);
|
|
||||||
struct sc_mouse_processor *mp = &screen->mouse->mouse_processor;
|
|
||||||
|
|
||||||
struct sc_mouse_motion_event evt = {
|
|
||||||
// .position not used for HID events
|
|
||||||
.xrel = event->xrel,
|
|
||||||
.yrel = event->yrel,
|
|
||||||
.buttons_state = sc_mouse_buttons_state_from_sdl(event->state, true),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert(mp->ops->process_mouse_motion);
|
|
||||||
mp->ops->process_mouse_motion(mp, &evt);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_screen_otg_process_mouse_button(struct sc_screen_otg *screen,
|
|
||||||
const SDL_MouseButtonEvent *event) {
|
|
||||||
assert(screen->mouse);
|
|
||||||
struct sc_mouse_processor *mp = &screen->mouse->mouse_processor;
|
|
||||||
|
|
||||||
uint32_t sdl_buttons_state = SDL_GetMouseState(NULL, NULL);
|
|
||||||
|
|
||||||
struct sc_mouse_click_event evt = {
|
|
||||||
// .position not used for HID events
|
|
||||||
.action = sc_action_from_sdl_mousebutton_type(event->type),
|
|
||||||
.button = sc_mouse_button_from_sdl(event->button),
|
|
||||||
.buttons_state =
|
|
||||||
sc_mouse_buttons_state_from_sdl(sdl_buttons_state, true),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert(mp->ops->process_mouse_click);
|
|
||||||
mp->ops->process_mouse_click(mp, &evt);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_screen_otg_process_mouse_wheel(struct sc_screen_otg *screen,
|
|
||||||
const SDL_MouseWheelEvent *event) {
|
|
||||||
assert(screen->mouse);
|
|
||||||
struct sc_mouse_processor *mp = &screen->mouse->mouse_processor;
|
|
||||||
|
|
||||||
uint32_t sdl_buttons_state = SDL_GetMouseState(NULL, NULL);
|
|
||||||
|
|
||||||
struct sc_mouse_scroll_event evt = {
|
|
||||||
// .position not used for HID events
|
|
||||||
.hscroll = event->x,
|
|
||||||
.vscroll = event->y,
|
|
||||||
.buttons_state =
|
|
||||||
sc_mouse_buttons_state_from_sdl(sdl_buttons_state, true),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert(mp->ops->process_mouse_scroll);
|
|
||||||
mp->ops->process_mouse_scroll(mp, &evt);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_screen_otg_handle_event(struct sc_screen_otg *screen, SDL_Event *event) {
|
|
||||||
switch (event->type) {
|
|
||||||
case SDL_WINDOWEVENT:
|
|
||||||
switch (event->window.event) {
|
|
||||||
case SDL_WINDOWEVENT_EXPOSED:
|
|
||||||
sc_screen_otg_render(screen);
|
|
||||||
break;
|
|
||||||
case SDL_WINDOWEVENT_FOCUS_LOST:
|
|
||||||
if (screen->mouse) {
|
|
||||||
sc_screen_otg_capture_mouse(screen, false);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
case SDL_KEYDOWN:
|
|
||||||
if (screen->mouse) {
|
|
||||||
SDL_Keycode key = event->key.keysym.sym;
|
|
||||||
if (sc_screen_otg_is_mouse_capture_key(key)) {
|
|
||||||
if (!screen->mouse_capture_key_pressed) {
|
|
||||||
screen->mouse_capture_key_pressed = key;
|
|
||||||
} else {
|
|
||||||
// Another mouse capture key has been pressed, cancel
|
|
||||||
// mouse (un)capture
|
|
||||||
screen->mouse_capture_key_pressed = 0;
|
|
||||||
}
|
|
||||||
// Mouse capture keys are never forwarded to the device
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (screen->keyboard) {
|
|
||||||
sc_screen_otg_process_key(screen, &event->key);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SDL_KEYUP:
|
|
||||||
if (screen->mouse) {
|
|
||||||
SDL_Keycode key = event->key.keysym.sym;
|
|
||||||
SDL_Keycode cap = screen->mouse_capture_key_pressed;
|
|
||||||
screen->mouse_capture_key_pressed = 0;
|
|
||||||
if (sc_screen_otg_is_mouse_capture_key(key)) {
|
|
||||||
if (key == cap) {
|
|
||||||
// A mouse capture key has been pressed then released:
|
|
||||||
// toggle the capture mouse mode
|
|
||||||
sc_screen_otg_capture_mouse(screen,
|
|
||||||
!screen->mouse_captured);
|
|
||||||
}
|
|
||||||
// Mouse capture keys are never forwarded to the device
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (screen->keyboard) {
|
|
||||||
sc_screen_otg_process_key(screen, &event->key);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SDL_MOUSEMOTION:
|
|
||||||
if (screen->mouse && screen->mouse_captured) {
|
|
||||||
sc_screen_otg_process_mouse_motion(screen, &event->motion);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SDL_MOUSEBUTTONDOWN:
|
|
||||||
if (screen->mouse && screen->mouse_captured) {
|
|
||||||
sc_screen_otg_process_mouse_button(screen, &event->button);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SDL_MOUSEBUTTONUP:
|
|
||||||
if (screen->mouse) {
|
|
||||||
if (screen->mouse_captured) {
|
|
||||||
sc_screen_otg_process_mouse_button(screen, &event->button);
|
|
||||||
} else {
|
|
||||||
sc_screen_otg_capture_mouse(screen, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SDL_MOUSEWHEEL:
|
|
||||||
if (screen->mouse && screen->mouse_captured) {
|
|
||||||
sc_screen_otg_process_mouse_wheel(screen, &event->wheel);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
#ifndef SC_SCREEN_OTG_H
|
|
||||||
#define SC_SCREEN_OTG_H
|
|
||||||
|
|
||||||
#include "common.h"
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <SDL2/SDL.h>
|
|
||||||
|
|
||||||
#include "hid_keyboard.h"
|
|
||||||
#include "hid_mouse.h"
|
|
||||||
|
|
||||||
struct sc_screen_otg {
|
|
||||||
struct sc_hid_keyboard *keyboard;
|
|
||||||
struct sc_hid_mouse *mouse;
|
|
||||||
|
|
||||||
SDL_Window *window;
|
|
||||||
SDL_Renderer *renderer;
|
|
||||||
SDL_Texture *texture;
|
|
||||||
|
|
||||||
// See equivalent mechanism in screen.h
|
|
||||||
bool mouse_captured;
|
|
||||||
SDL_Keycode mouse_capture_key_pressed;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct sc_screen_otg_params {
|
|
||||||
struct sc_hid_keyboard *keyboard;
|
|
||||||
struct sc_hid_mouse *mouse;
|
|
||||||
|
|
||||||
const char *window_title;
|
|
||||||
bool always_on_top;
|
|
||||||
int16_t window_x; // accepts SC_WINDOW_POSITION_UNDEFINED
|
|
||||||
int16_t window_y; // accepts SC_WINDOW_POSITION_UNDEFINED
|
|
||||||
bool window_borderless;
|
|
||||||
};
|
|
||||||
|
|
||||||
bool
|
|
||||||
sc_screen_otg_init(struct sc_screen_otg *screen,
|
|
||||||
const struct sc_screen_otg_params *params);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_screen_otg_destroy(struct sc_screen_otg *screen);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_screen_otg_handle_event(struct sc_screen_otg *screen, SDL_Event *event);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -1,264 +0,0 @@
|
|||||||
#include "usb.h"
|
|
||||||
|
|
||||||
#include <assert.h>
|
|
||||||
|
|
||||||
#include "util/log.h"
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
log_libusb_error(enum libusb_error errcode) {
|
|
||||||
LOGW("libusb error: %s", libusb_strerror(errcode));
|
|
||||||
}
|
|
||||||
|
|
||||||
static char *
|
|
||||||
read_string(libusb_device_handle *handle, uint8_t desc_index) {
|
|
||||||
char buffer[128];
|
|
||||||
int result =
|
|
||||||
libusb_get_string_descriptor_ascii(handle, desc_index,
|
|
||||||
(unsigned char *) buffer,
|
|
||||||
sizeof(buffer));
|
|
||||||
if (result < 0) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert((size_t) result <= sizeof(buffer));
|
|
||||||
|
|
||||||
// When non-negative, 'result' contains the number of bytes written
|
|
||||||
char *s = malloc(result + 1);
|
|
||||||
memcpy(s, buffer, result);
|
|
||||||
s[result] = '\0';
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
accept_device(libusb_device *device, const char *serial,
|
|
||||||
struct sc_usb_device *out) {
|
|
||||||
// Do not log any USB error in this function, it is expected that many USB
|
|
||||||
// devices available on the computer have permission restrictions
|
|
||||||
|
|
||||||
struct libusb_device_descriptor desc;
|
|
||||||
int result = libusb_get_device_descriptor(device, &desc);
|
|
||||||
if (result < 0 || !desc.iSerialNumber) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
libusb_device_handle *handle;
|
|
||||||
result = libusb_open(device, &handle);
|
|
||||||
if (result < 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
char *device_serial = read_string(handle, desc.iSerialNumber);
|
|
||||||
if (!device_serial) {
|
|
||||||
libusb_close(handle);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (serial) {
|
|
||||||
// Filter by serial
|
|
||||||
bool matches = !strcmp(serial, device_serial);
|
|
||||||
if (!matches) {
|
|
||||||
free(device_serial);
|
|
||||||
libusb_close(handle);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
out->device = libusb_ref_device(device);
|
|
||||||
out->serial = device_serial;
|
|
||||||
out->vid = desc.idVendor;
|
|
||||||
out->pid = desc.idProduct;
|
|
||||||
out->manufacturer = read_string(handle, desc.iManufacturer);
|
|
||||||
out->product = read_string(handle, desc.iProduct);
|
|
||||||
|
|
||||||
libusb_close(handle);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_device_destroy(struct sc_usb_device *usb_device) {
|
|
||||||
libusb_unref_device(usb_device->device);
|
|
||||||
free(usb_device->serial);
|
|
||||||
free(usb_device->manufacturer);
|
|
||||||
free(usb_device->product);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_device_destroy_all(struct sc_usb_device *usb_devices, size_t count) {
|
|
||||||
for (size_t i = 0; i < count; ++i) {
|
|
||||||
sc_usb_device_destroy(&usb_devices[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ssize_t
|
|
||||||
sc_usb_find_devices(struct sc_usb *usb, const char *serial,
|
|
||||||
struct sc_usb_device *devices, size_t len) {
|
|
||||||
libusb_device **list;
|
|
||||||
ssize_t count = libusb_get_device_list(usb->context, &list);
|
|
||||||
if (count < 0) {
|
|
||||||
log_libusb_error((enum libusb_error) count);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t idx = 0;
|
|
||||||
for (size_t i = 0; i < (size_t) count && idx < len; ++i) {
|
|
||||||
libusb_device *device = list[i];
|
|
||||||
|
|
||||||
if (accept_device(device, serial, &devices[idx])) {
|
|
||||||
++idx;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
libusb_free_device_list(list, 1);
|
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
static libusb_device_handle *
|
|
||||||
sc_usb_open_handle(libusb_device *device) {
|
|
||||||
libusb_device_handle *handle;
|
|
||||||
int result = libusb_open(device, &handle);
|
|
||||||
if (result < 0) {
|
|
||||||
log_libusb_error((enum libusb_error) result);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
return handle;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
sc_usb_init(struct sc_usb *usb) {
|
|
||||||
usb->handle = NULL;
|
|
||||||
return libusb_init(&usb->context) == LIBUSB_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_destroy(struct sc_usb *usb) {
|
|
||||||
libusb_exit(usb->context);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
sc_usb_libusb_callback(libusb_context *ctx, libusb_device *device,
|
|
||||||
libusb_hotplug_event event, void *userdata) {
|
|
||||||
(void) ctx;
|
|
||||||
(void) device;
|
|
||||||
(void) event;
|
|
||||||
|
|
||||||
struct sc_usb *usb = userdata;
|
|
||||||
|
|
||||||
libusb_device *dev = libusb_get_device(usb->handle);
|
|
||||||
assert(dev);
|
|
||||||
if (dev != device) {
|
|
||||||
// Not the connected device
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(usb->cbs && usb->cbs->on_disconnected);
|
|
||||||
usb->cbs->on_disconnected(usb, usb->cbs_userdata);
|
|
||||||
|
|
||||||
// Do not automatically deregister the callback by returning 1. Instead,
|
|
||||||
// manually deregister to interrupt libusb_handle_events() from the libusb
|
|
||||||
// event thread: <https://stackoverflow.com/a/60119225/1987178>
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
run_libusb_event_handler(void *data) {
|
|
||||||
struct sc_usb *usb = data;
|
|
||||||
while (!atomic_load(&usb->stopped)) {
|
|
||||||
// Interrupted by events or by libusb_hotplug_deregister_callback()
|
|
||||||
libusb_handle_events(usb->context);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
sc_usb_register_callback(struct sc_usb *usb) {
|
|
||||||
if (!libusb_has_capability(LIBUSB_CAP_HAS_HOTPLUG)) {
|
|
||||||
LOGW("libusb does not have hotplug capability");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
libusb_device *device = libusb_get_device(usb->handle);
|
|
||||||
assert(device);
|
|
||||||
|
|
||||||
struct libusb_device_descriptor desc;
|
|
||||||
int result = libusb_get_device_descriptor(device, &desc);
|
|
||||||
if (result < 0) {
|
|
||||||
log_libusb_error((enum libusb_error) result);
|
|
||||||
LOGW("Could not read USB device descriptor");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
int events = LIBUSB_HOTPLUG_EVENT_DEVICE_LEFT;
|
|
||||||
int flags = LIBUSB_HOTPLUG_NO_FLAGS;
|
|
||||||
int vendor_id = desc.idVendor;
|
|
||||||
int product_id = desc.idProduct;
|
|
||||||
int dev_class = LIBUSB_HOTPLUG_MATCH_ANY;
|
|
||||||
result = libusb_hotplug_register_callback(usb->context, events, flags,
|
|
||||||
vendor_id, product_id, dev_class,
|
|
||||||
sc_usb_libusb_callback, usb,
|
|
||||||
&usb->callback_handle);
|
|
||||||
if (result < 0) {
|
|
||||||
log_libusb_error((enum libusb_error) result);
|
|
||||||
LOGW("Could not register USB callback");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
usb->has_callback_handle = true;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
sc_usb_connect(struct sc_usb *usb, libusb_device *device,
|
|
||||||
const struct sc_usb_callbacks *cbs, void *cbs_userdata) {
|
|
||||||
usb->handle = sc_usb_open_handle(device);
|
|
||||||
if (!usb->handle) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
usb->has_callback_handle = false;
|
|
||||||
usb->has_libusb_event_thread = false;
|
|
||||||
|
|
||||||
// If cbs is set, then cbs->on_disconnected must be set
|
|
||||||
assert(!cbs || cbs->on_disconnected);
|
|
||||||
usb->cbs = cbs;
|
|
||||||
usb->cbs_userdata = cbs_userdata;
|
|
||||||
|
|
||||||
if (cbs) {
|
|
||||||
atomic_init(&usb->stopped, false);
|
|
||||||
if (sc_usb_register_callback(usb)) {
|
|
||||||
// Create a thread to process libusb events, so that device
|
|
||||||
// disconnection could be detected immediately
|
|
||||||
usb->has_libusb_event_thread =
|
|
||||||
sc_thread_create(&usb->libusb_event_thread,
|
|
||||||
run_libusb_event_handler, "scrcpy-usbev", usb);
|
|
||||||
if (!usb->has_libusb_event_thread) {
|
|
||||||
LOGW("Libusb event thread handler could not be created, USB "
|
|
||||||
"device disconnection might not be detected immediately");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
LOGW("Could not register USB device disconnection callback");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_disconnect(struct sc_usb *usb) {
|
|
||||||
libusb_close(usb->handle);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_stop(struct sc_usb *usb) {
|
|
||||||
if (usb->has_callback_handle) {
|
|
||||||
atomic_store(&usb->stopped, true);
|
|
||||||
libusb_hotplug_deregister_callback(usb->context, usb->callback_handle);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_join(struct sc_usb *usb) {
|
|
||||||
if (usb->has_libusb_event_thread) {
|
|
||||||
sc_thread_join(&usb->libusb_event_thread, NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
#ifndef SC_USB_H
|
|
||||||
#define SC_USB_H
|
|
||||||
|
|
||||||
#include "common.h"
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <libusb-1.0/libusb.h>
|
|
||||||
|
|
||||||
#include "util/thread.h"
|
|
||||||
|
|
||||||
struct sc_usb {
|
|
||||||
libusb_context *context;
|
|
||||||
libusb_device_handle *handle;
|
|
||||||
|
|
||||||
const struct sc_usb_callbacks *cbs;
|
|
||||||
void *cbs_userdata;
|
|
||||||
|
|
||||||
bool has_callback_handle;
|
|
||||||
libusb_hotplug_callback_handle callback_handle;
|
|
||||||
|
|
||||||
bool has_libusb_event_thread;
|
|
||||||
sc_thread libusb_event_thread;
|
|
||||||
|
|
||||||
atomic_bool stopped; // only used if cbs != NULL
|
|
||||||
};
|
|
||||||
|
|
||||||
struct sc_usb_callbacks {
|
|
||||||
void (*on_disconnected)(struct sc_usb *usb, void *userdata);
|
|
||||||
};
|
|
||||||
|
|
||||||
struct sc_usb_device {
|
|
||||||
libusb_device *device;
|
|
||||||
char *serial;
|
|
||||||
char *manufacturer;
|
|
||||||
char *product;
|
|
||||||
uint16_t vid;
|
|
||||||
uint16_t pid;
|
|
||||||
};
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_device_destroy(struct sc_usb_device *usb_device);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_device_destroy_all(struct sc_usb_device *usb_devices, size_t count);
|
|
||||||
|
|
||||||
bool
|
|
||||||
sc_usb_init(struct sc_usb *usb);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_destroy(struct sc_usb *usb);
|
|
||||||
|
|
||||||
ssize_t
|
|
||||||
sc_usb_find_devices(struct sc_usb *usb, const char *serial,
|
|
||||||
struct sc_usb_device *devices, size_t len);
|
|
||||||
|
|
||||||
bool
|
|
||||||
sc_usb_connect(struct sc_usb *usb, libusb_device *device,
|
|
||||||
const struct sc_usb_callbacks *cbs, void *cbs_userdata);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_disconnect(struct sc_usb *usb);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_stop(struct sc_usb *usb);
|
|
||||||
|
|
||||||
void
|
|
||||||
sc_usb_join(struct sc_usb *usb);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -1,8 +1,5 @@
|
|||||||
#include "log.h"
|
#include "log.h"
|
||||||
|
|
||||||
#if _WIN32
|
|
||||||
# include <windows.h>
|
|
||||||
#endif
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
static SDL_LogPriority
|
static SDL_LogPriority
|
||||||
@@ -54,24 +51,3 @@ sc_get_log_level(void) {
|
|||||||
SDL_LogPriority sdl_log = SDL_LogGetPriority(SDL_LOG_CATEGORY_APPLICATION);
|
SDL_LogPriority sdl_log = SDL_LogGetPriority(SDL_LOG_CATEGORY_APPLICATION);
|
||||||
return log_level_sdl_to_sc(sdl_log);
|
return log_level_sdl_to_sc(sdl_log);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
bool
|
|
||||||
sc_log_windows_error(const char *prefix, int error) {
|
|
||||||
assert(prefix);
|
|
||||||
|
|
||||||
char *message;
|
|
||||||
DWORD flags = FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM;
|
|
||||||
DWORD lang_id = MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US);
|
|
||||||
int ret =
|
|
||||||
FormatMessage(flags, NULL, error, lang_id, (char *) &message, 0, NULL);
|
|
||||||
if (ret <= 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Note: message already contains a trailing '\n'
|
|
||||||
LOGE("%s: [%d] %s", prefix, error, message);
|
|
||||||
LocalFree(message);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|||||||
@@ -26,10 +26,4 @@ sc_set_log_level(enum sc_log_level level);
|
|||||||
enum sc_log_level
|
enum sc_log_level
|
||||||
sc_get_log_level(void);
|
sc_get_log_level(void);
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
// Log system error (typically returned by GetLastError() or similar)
|
|
||||||
bool
|
|
||||||
sc_log_windows_error(const char *prefix, int error);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -117,7 +117,14 @@ set_cloexec_flag(sc_raw_socket raw_sock) {
|
|||||||
static void
|
static void
|
||||||
net_perror(const char *s) {
|
net_perror(const char *s) {
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
sc_log_windows_error(s, WSAGetLastError());
|
int error = WSAGetLastError();
|
||||||
|
char *wsa_message;
|
||||||
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
|
||||||
|
NULL, error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
||||||
|
(char *) &wsa_message, 0, NULL);
|
||||||
|
// no explicit '\n', wsa_message already contains a trailing '\n'
|
||||||
|
fprintf(stderr, "%s: [%d] %s", s, error, wsa_message);
|
||||||
|
LocalFree(wsa_message);
|
||||||
#else
|
#else
|
||||||
perror(s);
|
perror(s);
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
// <https://stackoverflow.com/a/44383330/1987178>
|
// <https://stackoverflow.com/a/44383330/1987178>
|
||||||
# define SC_PRIsizet "Iu"
|
# define SC_PRIsizet "Iu"
|
||||||
# define SC_PROCESS_NONE NULL
|
# define SC_PROCESS_NONE NULL
|
||||||
# define SC_EXIT_CODE_NONE -1UL // max value as unsigned long
|
# define SC_EXIT_CODE_NONE -1u // max value as unsigned
|
||||||
typedef HANDLE sc_pid;
|
typedef HANDLE sc_pid;
|
||||||
typedef DWORD sc_exit_code;
|
typedef DWORD sc_exit_code;
|
||||||
typedef HANDLE sc_pipe;
|
typedef HANDLE sc_pipe;
|
||||||
|
|||||||
@@ -19,5 +19,5 @@ endian = 'little'
|
|||||||
ffmpeg_avcodec = 'avcodec-58'
|
ffmpeg_avcodec = 'avcodec-58'
|
||||||
ffmpeg_avformat = 'avformat-58'
|
ffmpeg_avformat = 'avformat-58'
|
||||||
ffmpeg_avutil = 'avutil-56'
|
ffmpeg_avutil = 'avutil-56'
|
||||||
prebuilt_ffmpeg = 'ffmpeg-win32-4.3.1'
|
prebuilt_ffmpeg = 'ffmpeg-4.3.1-win32-shared'
|
||||||
prebuilt_sdl2 = 'SDL2-2.0.20/i686-w64-mingw32'
|
prebuilt_sdl2 = 'SDL2-2.0.18/i686-w64-mingw32'
|
||||||
|
|||||||
@@ -19,5 +19,5 @@ endian = 'little'
|
|||||||
ffmpeg_avcodec = 'avcodec-59'
|
ffmpeg_avcodec = 'avcodec-59'
|
||||||
ffmpeg_avformat = 'avformat-59'
|
ffmpeg_avformat = 'avformat-59'
|
||||||
ffmpeg_avutil = 'avutil-57'
|
ffmpeg_avutil = 'avutil-57'
|
||||||
prebuilt_ffmpeg = 'ffmpeg-win64-5.0'
|
prebuilt_ffmpeg = 'ffmpeg-5.0-full_build-shared'
|
||||||
prebuilt_sdl2 = 'SDL2-2.0.20/x86_64-w64-mingw32'
|
prebuilt_sdl2 = 'SDL2-2.0.18/x86_64-w64-mingw32'
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
set -e
|
set -e
|
||||||
|
|
||||||
BUILDDIR=build-auto
|
BUILDDIR=build-auto
|
||||||
PREBUILT_SERVER_URL=https://github.com/Genymobile/scrcpy/releases/download/v1.22/scrcpy-server-v1.22
|
PREBUILT_SERVER_URL=https://github.com/Genymobile/scrcpy/releases/download/v1.21/scrcpy-server-v1.21
|
||||||
PREBUILT_SERVER_SHA256=c05d273eec7533c0e106282e0254cf04e7f5e8f0c2920ca39448865fab2a419b
|
PREBUILT_SERVER_SHA256=dbcccab523ee26796e55ea33652649e4b7af498edae9aa75e4d4d7869c0ab848
|
||||||
|
|
||||||
echo "[scrcpy] Downloading prebuilt server..."
|
echo "[scrcpy] Downloading prebuilt server..."
|
||||||
wget "$PREBUILT_SERVER_URL" -O scrcpy-server
|
wget "$PREBUILT_SERVER_URL" -O scrcpy-server
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
project('scrcpy', 'c',
|
project('scrcpy', 'c',
|
||||||
version: '1.22',
|
version: '1.21',
|
||||||
meson_version: '>= 0.48',
|
meson_version: '>= 0.48',
|
||||||
default_options: [
|
default_options: [
|
||||||
'c_std=c11',
|
'c_std=c11',
|
||||||
|
|||||||
5
prebuilt-deps/.gitignore
vendored
5
prebuilt-deps/.gitignore
vendored
@@ -1 +1,4 @@
|
|||||||
/data
|
*
|
||||||
|
!/.gitignore
|
||||||
|
!/Makefile
|
||||||
|
!/prepare-dep
|
||||||
|
|||||||
33
prebuilt-deps/Makefile
Normal file
33
prebuilt-deps/Makefile
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
.PHONY: prepare-win32 prepare-win64 \
|
||||||
|
prepare-ffmpeg-win32 \
|
||||||
|
prepare-ffmpeg-win64 \
|
||||||
|
prepare-sdl2 \
|
||||||
|
prepare-adb
|
||||||
|
|
||||||
|
prepare-win32: prepare-sdl2 prepare-ffmpeg-win32 prepare-adb
|
||||||
|
prepare-win64: prepare-sdl2 prepare-ffmpeg-win64 prepare-adb
|
||||||
|
|
||||||
|
# Use old FFmpeg version for win32, there are no new prebuilts
|
||||||
|
prepare-ffmpeg-win32:
|
||||||
|
@./prepare-dep https://github.com/Genymobile/scrcpy/releases/download/v1.16/ffmpeg-4.3.1-win32-shared.zip \
|
||||||
|
357af9901a456f4dcbacd107e83a934d344c9cb07ddad8aaf80612eeab7d26d2 \
|
||||||
|
ffmpeg-4.3.1-win32-shared
|
||||||
|
@./prepare-dep https://github.com/Genymobile/scrcpy/releases/download/v1.16/ffmpeg-4.3.1-win32-dev.zip \
|
||||||
|
230efb08e9bcf225bd474da29676c70e591fc94d8790a740ca801408fddcb78b \
|
||||||
|
ffmpeg-4.3.1-win32-dev
|
||||||
|
ln -sf ../ffmpeg-4.3.1-win32-dev/include ffmpeg-4.3.1-win32-shared/
|
||||||
|
|
||||||
|
prepare-ffmpeg-win64:
|
||||||
|
@./prepare-dep https://github.com/GyanD/codexffmpeg/releases/download/5.0/ffmpeg-5.0-full_build-shared.7z \
|
||||||
|
e5900f6cecd4c438d398bd2fc308736c10b857cd8dd61c11bcfb05bff5d1211a \
|
||||||
|
ffmpeg-5.0-full_build-shared
|
||||||
|
|
||||||
|
prepare-sdl2:
|
||||||
|
@./prepare-dep https://libsdl.org/release/SDL2-devel-2.0.18-mingw.tar.gz \
|
||||||
|
bbad7c6947f6ca3e05292f065852ed8b62f319fc5533047e7708769c4dbae394 \
|
||||||
|
SDL2-2.0.18
|
||||||
|
|
||||||
|
prepare-adb:
|
||||||
|
@./prepare-dep https://dl.google.com/android/repository/platform-tools_r31.0.3-windows.zip \
|
||||||
|
0f4b8fdd26af2c3733539d6eebb3c2ed499ea1d4bb1f4e0ecc2d6016961a6e24 \
|
||||||
|
platform-tools
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
PREBUILT_DATA_DIR=data
|
|
||||||
|
|
||||||
checksum() {
|
|
||||||
local file="$1"
|
|
||||||
local sum="$2"
|
|
||||||
echo "$file: verifying checksum..."
|
|
||||||
echo "$sum $file" | sha256sum -c
|
|
||||||
}
|
|
||||||
|
|
||||||
get_file() {
|
|
||||||
local url="$1"
|
|
||||||
local file="$2"
|
|
||||||
local sum="$3"
|
|
||||||
if [[ -f "$file" ]]
|
|
||||||
then
|
|
||||||
echo "$file: found"
|
|
||||||
else
|
|
||||||
echo "$file: not found, downloading..."
|
|
||||||
wget "$url" -O "$file"
|
|
||||||
fi
|
|
||||||
checksum "$file" "$sum"
|
|
||||||
}
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -e
|
|
||||||
DIR=$(dirname ${BASH_SOURCE[0]})
|
|
||||||
cd "$DIR"
|
|
||||||
. common
|
|
||||||
mkdir -p "$PREBUILT_DATA_DIR"
|
|
||||||
cd "$PREBUILT_DATA_DIR"
|
|
||||||
|
|
||||||
DEP_DIR=platform-tools-31.0.3
|
|
||||||
|
|
||||||
FILENAME=platform-tools_r31.0.3-windows.zip
|
|
||||||
SHA256SUM=0f4b8fdd26af2c3733539d6eebb3c2ed499ea1d4bb1f4e0ecc2d6016961a6e24
|
|
||||||
|
|
||||||
if [[ -d "$DEP_DIR" ]]
|
|
||||||
then
|
|
||||||
echo "$DEP_DIR" found
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
get_file "https://dl.google.com/android/repository/$FILENAME" \
|
|
||||||
"$FILENAME" "$SHA256SUM"
|
|
||||||
|
|
||||||
mkdir "$DEP_DIR"
|
|
||||||
cd "$DEP_DIR"
|
|
||||||
|
|
||||||
ZIP_PREFIX=platform-tools
|
|
||||||
unzip "../$FILENAME" \
|
|
||||||
"$ZIP_PREFIX"/AdbWinApi.dll \
|
|
||||||
"$ZIP_PREFIX"/AdbWinUsbApi.dll \
|
|
||||||
"$ZIP_PREFIX"/adb.exe
|
|
||||||
mv "$ZIP_PREFIX"/* .
|
|
||||||
rmdir "$ZIP_PREFIX"
|
|
||||||
61
prebuilt-deps/prepare-dep
Executable file
61
prebuilt-deps/prepare-dep
Executable file
@@ -0,0 +1,61 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -e
|
||||||
|
url="$1"
|
||||||
|
sum="$2"
|
||||||
|
dir="$3"
|
||||||
|
|
||||||
|
checksum() {
|
||||||
|
local file="$1"
|
||||||
|
local sum="$2"
|
||||||
|
echo "$file: verifying checksum..."
|
||||||
|
echo "$sum $file" | sha256sum -c
|
||||||
|
}
|
||||||
|
|
||||||
|
get_file() {
|
||||||
|
local url="$1"
|
||||||
|
local file="$2"
|
||||||
|
local sum="$3"
|
||||||
|
if [[ -f "$file" ]]
|
||||||
|
then
|
||||||
|
echo "$file: found"
|
||||||
|
else
|
||||||
|
echo "$file: not found, downloading..."
|
||||||
|
wget "$url" -O "$file"
|
||||||
|
fi
|
||||||
|
checksum "$file" "$sum"
|
||||||
|
}
|
||||||
|
|
||||||
|
extract() {
|
||||||
|
local file="$1"
|
||||||
|
echo "Extracting $file..."
|
||||||
|
if [[ "$file" == *.zip ]]
|
||||||
|
then
|
||||||
|
unzip -q "$file"
|
||||||
|
elif [[ "$file" == *.tar.gz ]]
|
||||||
|
then
|
||||||
|
tar xf "$file"
|
||||||
|
elif [[ "$file" == *.7z ]]
|
||||||
|
then
|
||||||
|
7z x "$file"
|
||||||
|
else
|
||||||
|
echo "Unsupported file: $file"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
get_dep() {
|
||||||
|
local url="$1"
|
||||||
|
local sum="$2"
|
||||||
|
local dir="$3"
|
||||||
|
local file="${url##*/}"
|
||||||
|
if [[ -d "$dir" ]]
|
||||||
|
then
|
||||||
|
echo "$dir: found"
|
||||||
|
else
|
||||||
|
echo "$dir: not found"
|
||||||
|
get_file "$url" "$file" "$sum"
|
||||||
|
extract "$file"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
get_dep "$url" "$sum" "$dir"
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -e
|
|
||||||
DIR=$(dirname ${BASH_SOURCE[0]})
|
|
||||||
cd "$DIR"
|
|
||||||
. common
|
|
||||||
mkdir -p "$PREBUILT_DATA_DIR"
|
|
||||||
cd "$PREBUILT_DATA_DIR"
|
|
||||||
|
|
||||||
DEP_DIR=ffmpeg-win32-4.3.1
|
|
||||||
|
|
||||||
FILENAME_SHARED=ffmpeg-4.3.1-win32-shared.zip
|
|
||||||
SHA256SUM_SHARED=357af9901a456f4dcbacd107e83a934d344c9cb07ddad8aaf80612eeab7d26d2
|
|
||||||
|
|
||||||
FILENAME_DEV=ffmpeg-4.3.1-win32-dev.zip
|
|
||||||
SHA256SUM_DEV=230efb08e9bcf225bd474da29676c70e591fc94d8790a740ca801408fddcb78b
|
|
||||||
|
|
||||||
if [[ -d "$DEP_DIR" ]]
|
|
||||||
then
|
|
||||||
echo "$DEP_DIR" found
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
get_file "https://github.com/Genymobile/scrcpy/releases/download/v1.16/$FILENAME_SHARED" \
|
|
||||||
"$FILENAME_SHARED" "$SHA256SUM_SHARED"
|
|
||||||
get_file "https://github.com/Genymobile/scrcpy/releases/download/v1.16/$FILENAME_DEV" \
|
|
||||||
"$FILENAME_DEV" "$SHA256SUM_DEV"
|
|
||||||
|
|
||||||
mkdir "$DEP_DIR"
|
|
||||||
cd "$DEP_DIR"
|
|
||||||
|
|
||||||
ZIP_PREFIX_SHARED=ffmpeg-4.3.1-win32-shared
|
|
||||||
unzip "../$FILENAME_SHARED" \
|
|
||||||
"$ZIP_PREFIX_SHARED"/bin/avutil-56.dll \
|
|
||||||
"$ZIP_PREFIX_SHARED"/bin/avcodec-58.dll \
|
|
||||||
"$ZIP_PREFIX_SHARED"/bin/avformat-58.dll \
|
|
||||||
"$ZIP_PREFIX_SHARED"/bin/swresample-3.dll \
|
|
||||||
"$ZIP_PREFIX_SHARED"/bin/swscale-5.dll
|
|
||||||
|
|
||||||
ZIP_PREFIX_DEV=ffmpeg-4.3.1-win32-dev
|
|
||||||
unzip "../$FILENAME_DEV" \
|
|
||||||
"$ZIP_PREFIX_DEV/include/*"
|
|
||||||
|
|
||||||
mv "$ZIP_PREFIX_SHARED"/* .
|
|
||||||
mv "$ZIP_PREFIX_DEV"/* .
|
|
||||||
rmdir "$ZIP_PREFIX_SHARED" "$ZIP_PREFIX_DEV"
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -e
|
|
||||||
DIR=$(dirname ${BASH_SOURCE[0]})
|
|
||||||
cd "$DIR"
|
|
||||||
. common
|
|
||||||
mkdir -p "$PREBUILT_DATA_DIR"
|
|
||||||
cd "$PREBUILT_DATA_DIR"
|
|
||||||
|
|
||||||
DEP_DIR=ffmpeg-win64-5.0
|
|
||||||
|
|
||||||
FILENAME=ffmpeg-5.0-full_build-shared.7z
|
|
||||||
SHA256SUM=e5900f6cecd4c438d398bd2fc308736c10b857cd8dd61c11bcfb05bff5d1211a
|
|
||||||
|
|
||||||
if [[ -d "$DEP_DIR" ]]
|
|
||||||
then
|
|
||||||
echo "$DEP_DIR" found
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
get_file "https://github.com/GyanD/codexffmpeg/releases/download/5.0/$FILENAME" \
|
|
||||||
"$FILENAME" "$SHA256SUM"
|
|
||||||
|
|
||||||
mkdir "$DEP_DIR"
|
|
||||||
cd "$DEP_DIR"
|
|
||||||
|
|
||||||
ZIP_PREFIX=ffmpeg-5.0-full_build-shared
|
|
||||||
7z x "../$FILENAME" \
|
|
||||||
"$ZIP_PREFIX"/bin/avutil-57.dll \
|
|
||||||
"$ZIP_PREFIX"/bin/avcodec-59.dll \
|
|
||||||
"$ZIP_PREFIX"/bin/avformat-59.dll \
|
|
||||||
"$ZIP_PREFIX"/bin/swresample-4.dll \
|
|
||||||
"$ZIP_PREFIX"/bin/swscale-6.dll \
|
|
||||||
"$ZIP_PREFIX"/include
|
|
||||||
mv "$ZIP_PREFIX"/* .
|
|
||||||
rmdir "$ZIP_PREFIX"
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -e
|
|
||||||
DIR=$(dirname ${BASH_SOURCE[0]})
|
|
||||||
cd "$DIR"
|
|
||||||
. common
|
|
||||||
mkdir -p "$PREBUILT_DATA_DIR"
|
|
||||||
cd "$PREBUILT_DATA_DIR"
|
|
||||||
|
|
||||||
DEP_DIR=SDL2-2.0.20
|
|
||||||
|
|
||||||
FILENAME=SDL2-devel-2.0.20-mingw.tar.gz
|
|
||||||
SHA256SUM=38094d82a857d6c62352e5c5cdec74948c5b4d25c59cbd298d6d233568976bd1
|
|
||||||
|
|
||||||
if [[ -d "$DEP_DIR" ]]
|
|
||||||
then
|
|
||||||
echo "$DEP_DIR" found
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
get_file "https://libsdl.org/release/$FILENAME" "$FILENAME" "$SHA256SUM"
|
|
||||||
|
|
||||||
mkdir "$DEP_DIR"
|
|
||||||
cd "$DEP_DIR"
|
|
||||||
|
|
||||||
TAR_PREFIX="$DEP_DIR" # root directory inside the tar has the same name
|
|
||||||
tar xf "../$FILENAME" --strip-components=1 \
|
|
||||||
"$TAR_PREFIX"/i686-w64-mingw32/bin/SDL2.dll \
|
|
||||||
"$TAR_PREFIX"/i686-w64-mingw32/include/ \
|
|
||||||
"$TAR_PREFIX"/i686-w64-mingw32/lib/ \
|
|
||||||
"$TAR_PREFIX"/x86_64-w64-mingw32/bin/SDL2.dll \
|
|
||||||
"$TAR_PREFIX"/x86_64-w64-mingw32/include/ \
|
|
||||||
"$TAR_PREFIX"/x86_64-w64-mingw32/lib/ \
|
|
||||||
44
release.mk
44
release.mk
@@ -63,9 +63,7 @@ build-server:
|
|||||||
ninja -C "$(SERVER_BUILD_DIR)"
|
ninja -C "$(SERVER_BUILD_DIR)"
|
||||||
|
|
||||||
prepare-deps-win32:
|
prepare-deps-win32:
|
||||||
@prebuilt-deps/prepare-adb.sh
|
-$(MAKE) -C prebuilt-deps prepare-win32
|
||||||
@prebuilt-deps/prepare-sdl.sh
|
|
||||||
@prebuilt-deps/prepare-ffmpeg-win32.sh
|
|
||||||
|
|
||||||
build-win32: prepare-deps-win32
|
build-win32: prepare-deps-win32
|
||||||
[ -d "$(WIN32_BUILD_DIR)" ] || ( mkdir "$(WIN32_BUILD_DIR)" && \
|
[ -d "$(WIN32_BUILD_DIR)" ] || ( mkdir "$(WIN32_BUILD_DIR)" && \
|
||||||
@@ -77,9 +75,7 @@ build-win32: prepare-deps-win32
|
|||||||
ninja -C "$(WIN32_BUILD_DIR)"
|
ninja -C "$(WIN32_BUILD_DIR)"
|
||||||
|
|
||||||
prepare-deps-win64:
|
prepare-deps-win64:
|
||||||
@prebuilt-deps/prepare-adb.sh
|
-$(MAKE) -C prebuilt-deps prepare-win64
|
||||||
@prebuilt-deps/prepare-sdl.sh
|
|
||||||
@prebuilt-deps/prepare-ffmpeg-win64.sh
|
|
||||||
|
|
||||||
build-win64: prepare-deps-win64
|
build-win64: prepare-deps-win64
|
||||||
[ -d "$(WIN64_BUILD_DIR)" ] || ( mkdir "$(WIN64_BUILD_DIR)" && \
|
[ -d "$(WIN64_BUILD_DIR)" ] || ( mkdir "$(WIN64_BUILD_DIR)" && \
|
||||||
@@ -98,15 +94,15 @@ dist-win32: build-server build-win32
|
|||||||
cp data/scrcpy-noconsole.vbs "$(DIST)/$(WIN32_TARGET_DIR)"
|
cp data/scrcpy-noconsole.vbs "$(DIST)/$(WIN32_TARGET_DIR)"
|
||||||
cp data/icon.png "$(DIST)/$(WIN32_TARGET_DIR)"
|
cp data/icon.png "$(DIST)/$(WIN32_TARGET_DIR)"
|
||||||
cp data/open_a_terminal_here.bat "$(DIST)/$(WIN32_TARGET_DIR)"
|
cp data/open_a_terminal_here.bat "$(DIST)/$(WIN32_TARGET_DIR)"
|
||||||
cp prebuilt-deps/data/ffmpeg-win32-4.3.1/bin/avutil-56.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-4.3.1-win32-shared/bin/avutil-56.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win32-4.3.1/bin/avcodec-58.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-4.3.1-win32-shared/bin/avcodec-58.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win32-4.3.1/bin/avformat-58.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-4.3.1-win32-shared/bin/avformat-58.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win32-4.3.1/bin/swresample-3.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-4.3.1-win32-shared/bin/swresample-3.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win32-4.3.1/bin/swscale-5.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-4.3.1-win32-shared/bin/swscale-5.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/platform-tools-31.0.3/adb.exe "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/platform-tools/adb.exe "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/platform-tools-31.0.3/AdbWinApi.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/platform-tools/AdbWinApi.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/platform-tools-31.0.3/AdbWinUsbApi.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/platform-tools/AdbWinUsbApi.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/SDL2-2.0.20/i686-w64-mingw32/bin/SDL2.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
cp prebuilt-deps/SDL2-2.0.18/i686-w64-mingw32/bin/SDL2.dll "$(DIST)/$(WIN32_TARGET_DIR)/"
|
||||||
|
|
||||||
dist-win64: build-server build-win64
|
dist-win64: build-server build-win64
|
||||||
mkdir -p "$(DIST)/$(WIN64_TARGET_DIR)"
|
mkdir -p "$(DIST)/$(WIN64_TARGET_DIR)"
|
||||||
@@ -116,15 +112,15 @@ dist-win64: build-server build-win64
|
|||||||
cp data/scrcpy-noconsole.vbs "$(DIST)/$(WIN64_TARGET_DIR)"
|
cp data/scrcpy-noconsole.vbs "$(DIST)/$(WIN64_TARGET_DIR)"
|
||||||
cp data/icon.png "$(DIST)/$(WIN64_TARGET_DIR)"
|
cp data/icon.png "$(DIST)/$(WIN64_TARGET_DIR)"
|
||||||
cp data/open_a_terminal_here.bat "$(DIST)/$(WIN64_TARGET_DIR)"
|
cp data/open_a_terminal_here.bat "$(DIST)/$(WIN64_TARGET_DIR)"
|
||||||
cp prebuilt-deps/data/ffmpeg-win64-5.0/bin/avutil-57.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-5.0-full_build-shared/bin/avutil-57.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win64-5.0/bin/avcodec-59.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-5.0-full_build-shared/bin/avcodec-59.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win64-5.0/bin/avformat-59.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-5.0-full_build-shared/bin/avformat-59.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win64-5.0/bin/swresample-4.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-5.0-full_build-shared/bin/swresample-4.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/ffmpeg-win64-5.0/bin/swscale-6.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/ffmpeg-5.0-full_build-shared/bin/swscale-6.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/platform-tools-31.0.3/adb.exe "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/platform-tools/adb.exe "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/platform-tools-31.0.3/AdbWinApi.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/platform-tools/AdbWinApi.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/platform-tools-31.0.3/AdbWinUsbApi.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/platform-tools/AdbWinUsbApi.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
cp prebuilt-deps/data/SDL2-2.0.20/x86_64-w64-mingw32/bin/SDL2.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
cp prebuilt-deps/SDL2-2.0.18/x86_64-w64-mingw32/bin/SDL2.dll "$(DIST)/$(WIN64_TARGET_DIR)/"
|
||||||
|
|
||||||
zip-win32: dist-win32
|
zip-win32: dist-win32
|
||||||
cd "$(DIST)/$(WIN32_TARGET_DIR)"; \
|
cd "$(DIST)/$(WIN32_TARGET_DIR)"; \
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ android {
|
|||||||
applicationId "com.genymobile.scrcpy"
|
applicationId "com.genymobile.scrcpy"
|
||||||
minSdkVersion 21
|
minSdkVersion 21
|
||||||
targetSdkVersion 31
|
targetSdkVersion 31
|
||||||
versionCode 12200
|
versionCode 12100
|
||||||
versionName "1.22"
|
versionName "1.21"
|
||||||
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
||||||
}
|
}
|
||||||
buildTypes {
|
buildTypes {
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
set -e
|
set -e
|
||||||
|
|
||||||
SCRCPY_DEBUG=false
|
SCRCPY_DEBUG=false
|
||||||
SCRCPY_VERSION_NAME=1.22
|
SCRCPY_VERSION_NAME=1.21
|
||||||
|
|
||||||
PLATFORM=${ANDROID_PLATFORM:-31}
|
PLATFORM=${ANDROID_PLATFORM:-31}
|
||||||
BUILD_TOOLS=${ANDROID_BUILD_TOOLS:-31.0.0}
|
BUILD_TOOLS=${ANDROID_BUILD_TOOLS:-31.0.0}
|
||||||
|
|||||||
@@ -52,8 +52,7 @@ public final class Server {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
CleanUp.configure(options.getDisplayId(), restoreStayOn, mustDisableShowTouchesOnCleanUp, restoreNormalPowerMode,
|
CleanUp.configure(options.getDisplayId(), restoreStayOn, mustDisableShowTouchesOnCleanUp, restoreNormalPowerMode, options.getPowerOffScreenOnClose());
|
||||||
options.getPowerOffScreenOnClose());
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
Ln.e("Could not configure cleanup", e);
|
Ln.e("Could not configure cleanup", e);
|
||||||
}
|
}
|
||||||
@@ -262,7 +261,6 @@ public final class Server {
|
|||||||
options.setSendFrameMeta(false);
|
options.setSendFrameMeta(false);
|
||||||
options.setSendDummyByte(false);
|
options.setSendDummyByte(false);
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
default:
|
default:
|
||||||
Ln.w("Unknown server option: " + key);
|
Ln.w("Unknown server option: " + key);
|
||||||
break;
|
break;
|
||||||
|
|||||||
Reference in New Issue
Block a user