Como executar um arrasto (baseado nas coordenadas X, Y do mouse) no Android usando o AccessibilityService?

39

Quero saber como executar um arrasto no Android com base nas coordenadas X, Y do mouse? considere como dois exemplos simples: o Team Viewer / QuickSupport desenha o "padrão de senha" no smartphone remoto e a Caneta do Windows Paint, respectivamente.

insira a descrição da imagem aqui

insira a descrição da imagem aqui

Tudo o que eu posso fazer é simular o toque (com dispatchGesture()e também AccessibilityNodeInfo.ACTION_CLICK).

Encontrei esses links relevantes, mas não sei se eles podem ser úteis:

Abaixo está o meu código de trabalho usado para enviar coordenadas do mouse (dentro de PictureBox controle) para o telefone remoto e simular o toque.

Aplicativo Windows Forms:

private void pictureBox1_MouseDown(object sender, MouseEventArgs e)
{
    foreach (ListViewItem item in lvConnections.SelectedItems)
    {
        // Remote screen resolution
        string[] tokens = item.SubItems[5].Text.Split('x'); // Ex: 1080x1920

        int xClick = (e.X * int.Parse(tokens[0].ToString())) / (pictureBox1.Size.Width);
        int yClick = (e.Y * int.Parse(tokens[1].ToString())) / (pictureBox1.Size.Height);

        Client client = (Client)item.Tag;

        if (e.Button == MouseButtons.Left)
            client.sock.Send(Encoding.UTF8.GetBytes("TOUCH" + xClick + "<|>" + yClick + Environment.NewLine));
    }
}

Editar:

Minha última tentativa foi uma "tela de furto" usando as coordenadas do mouse (C # Windows Forms Application) e uma rotina android personalizada (com referência ao código da "tela de furto" vinculada acima), respectivamente:

private Point mdownPoint = new Point();

private void pictureBox1_MouseDown(object sender, MouseEventArgs e)
{
    foreach (ListViewItem item in lvConnections.SelectedItems)
    {
        // Remote screen resolution
        string[] tokens = item.SubItems[5].Text.Split('x'); // Ex: 1080x1920

        Client client = (Client)item.Tag;

        if (e.Button == MouseButtons.Left)
        {
            xClick = (e.X * int.Parse(tokens[0].ToString())) / (pictureBox1.Size.Width); 
            yClick = (e.Y * int.Parse(tokens[1].ToString())) / (pictureBox1.Size.Height);

            // Saving start position:

            mdownPoint.X = xClick; 
            mdownPoint.Y = yClick; 

            client.sock.Send(Encoding.UTF8.GetBytes("TOUCH" + xClick + "<|>" + yClick + Environment.NewLine));
        }
    }
}

private void PictureBox1_MouseMove(object sender, MouseEventArgs e)
{
    foreach (ListViewItem item in lvConnections.SelectedItems)
    {
        // Remote screen resolution
        string[] tokens = item.SubItems[5].Text.Split('x'); // Ex: 1080x1920

        Client client = (Client)item.Tag;

        if (e.Button == MouseButtons.Left)
        {
            xClick = (e.X * int.Parse(tokens[0].ToString())) / (pictureBox1.Size.Width);
            yClick = (e.Y * int.Parse(tokens[1].ToString())) / (pictureBox1.Size.Height);

            client.sock.Send(Encoding.UTF8.GetBytes("MOUSESWIPESCREEN" + mdownPoint.X + "<|>" + mdownPoint.Y + "<|>" + xClick + "<|>" + yClick + Environment.NewLine));
        }
    }
}

android AccessibilityService :

public void Swipe(int x1, int y1, int x2, int y2, int time) {

if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
    System.out.println(" ======= Swipe =======");

    GestureDescription.Builder gestureBuilder = new GestureDescription.Builder();
    Path path = new Path();
    path.moveTo(x1, y1);
    path.lineTo(x2, y2);

    gestureBuilder.addStroke(new GestureDescription.StrokeDescription(path, 100, time));
    dispatchGesture(gestureBuilder.build(), new GestureResultCallback() {
        @Override
        public void onCompleted(GestureDescription gestureDescription) {
            System.out.println("SWIPE Gesture Completed :D");
            super.onCompleted(gestureDescription);
        }
    }, null);
}

}

que produz o seguinte resultado (mas ainda não é capaz de desenhar "senha padrão" como o TeamViewer, por exemplo). Mas, como dito no comentário abaixo, acho que com uma abordagem semelhante, isso pode ser alcançado usando gestos contínuos provavelmente. Todas as sugestões nessa direção serão bem-vindas.

insira a descrição da imagem aqui

insira a descrição da imagem aqui


Edição 2:

Definitivamente, a solução são gestos contínuos, como dito na edição anterior .

E abaixo está um suposto código fixo que eu encontrei aqui =>

android AccessibilityService:

// Simulates an L-shaped drag path: 200 pixels right, then 200 pixels down.
Path path = new Path();
path.moveTo(200,200);
path.lineTo(400,200);

final GestureDescription.StrokeDescription sd = new GestureDescription.StrokeDescription(path, 0, 500, true);

// The starting point of the second path must match
// the ending point of the first path.
Path path2 = new Path();
path2.moveTo(400,200);
path2.lineTo(400,400);

final GestureDescription.StrokeDescription sd2 = sd.continueStroke(path2, 0, 500, false); // 0.5 second

HongBaoService.mService.dispatchGesture(new GestureDescription.Builder().addStroke(sd).build(), new AccessibilityService.GestureResultCallback(){

@Override
public void onCompleted(GestureDescription gestureDescription){
super.onCompleted(gestureDescription);
HongBaoService.mService.dispatchGesture(new GestureDescription.Builder().addStroke(sd2).build(),null,null);
}

@Override
public void onCancelled(GestureDescription gestureDescription){
super.onCancelled(gestureDescription);
}
},null);

Então, minha dúvida é: como enviar corretamente as coordenadas do mouse para o código acima, do modo que pode ser executado em qualquer direção? Alguma idéia?


Edição 3:

Eu encontrei duas rotinas que são usadas para executar o arrasto, mas elas estão usando o UiAutomation + injectInputEvent(). AFAIK, a injeção de evento funciona apenas em um aplicativo de sistema como o dito aqui e aqui e eu não quero isso.

Estas são as rotinas encontradas:

Então, para alcançar meu objetivo, acho que a 2ª rotina é mais apropriada para usar (seguindo a lógica, excluindo a injeção de eventos) com o código mostrado no Edit 2 e enviar todos os pontos pictureBox1_MouseDowne pictureBox1_MouseMove(C # Windows Forms Application), respectivamente, para preencherPoint[] dinamicamente e ao pictureBox1_MouseUpenviar cmd para executar a rotina e usar essa matriz preenchida. Se você tem uma idéia da 1ª rotina, me informe: D.

Se, depois de ler este Edit, você tiver uma solução possível, mostre-me uma resposta, por favor, enquanto tentarei testar essa idéia.

BrowJr
fonte
11
O TeamViewer não está usando a estrutura de acessibilidade, provavelmente. Eles têm acordos especiais com fabricantes de dispositivos, e é por isso que o produto não está disponível para todos os dispositivos.
CommonsWare
@CommonsWare obrigado. Mas acho que StrokeDescription.continueStroke()pode ser uma solução provável. Veja a seção Gestos continuados aqui .
BrowJr
2
Em relação à sua primeira abordagem. pictureBox1_MouseDownnão deve enviar as coordenadas. Ele deve armazenar apenas as coordenadas iniciais e, depois, pictureBox1_MouseUpenviá-las, porque isso marca o fim do movimento do mouse.
Greggz

Respostas:

1

Aqui está um exemplo de uma solução baseada no Edit 3 da pergunta.


Aplicativo Cs Windows Froms " formMain.cs ":

using System.Net.Sockets;

private List<Point> lstPoints;

private void pictureBox1_MouseDown(object sender, MouseEventArgs e) 
{
    if (e.Button == MouseButtons.Left)
    {
        lstPoints = new List<Point>();
        lstPoints.Add(new Point(e.X, e.Y));
    }
}

private void PictureBox1_MouseMove(object sender, MouseEventArgs e)
{
    if (e.Button == MouseButtons.Left)
    {
        lstPoints.Add(new Point(e.X, e.Y));
    }
}

private void PictureBox1_MouseUp(object sender, MouseEventArgs e)
{
    lstPoints.Add(new Point(e.X, e.Y));

    StringBuilder sb = new StringBuilder();

    foreach (Point obj in lstPoints)
    {
        sb.Append(Convert.ToString(obj) + ":");
    }

    serverSocket.Send("MDRAWEVENT" + sb.ToString() + Environment.NewLine);
}

serviço android " SocketBackground.java ":

import java.net.Socket;

String xline;

while (clientSocket.isConnected()) {

    BufferedReader xreader = new BufferedReader(new InputStreamReader(clientSocket.getInputStream(), StandardCharsets.UTF_8));

    if (xreader.ready()) {

        while ((xline = xreader.readLine()) != null) {
                xline = xline.trim();

            if (xline != null && !xline.trim().isEmpty()) {

                if (xline.contains("MDRAWEVENT")) {

                    String coordinates = xline.replace("MDRAWEVENT", "");
                    String[] tokens = coordinates.split(Pattern.quote(":"));
                    Point[] moviments = new Point[tokens.length];

                    for (int i = 0; i < tokens.length; i++) {
                       String[] coordinates = tokens[i].replace("{", "").replace("}", "").split(",");

                       int x = Integer.parseInt(coordinates[0].split("=")[1]);
                       int y = Integer.parseInt(coordinates[1].split("=")[1]);

                       moviments[i] = new Point(x, y);
                    }

                    MyAccessibilityService.instance.mouseDraw(moviments, 2000);
                }
            }
        }
    }
}

android AccessibilityService" MyAccessibilityService.java ":

public void mouseDraw(Point[] segments, int time) {
    if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

        Path path = new Path();
        path.moveTo(segments[0].x, segments[0].y);

        for (int i = 1; i < segments.length; i++) {

            path.lineTo(segments[i].x, segments[i].y);

            GestureDescription.StrokeDescription sd = new GestureDescription.StrokeDescription(path, 0, time);

            dispatchGesture(new GestureDescription.Builder().addStroke(sd).build(), new AccessibilityService.GestureResultCallback() {

                @Override
                public void onCompleted(GestureDescription gestureDescription) {
                    super.onCompleted(gestureDescription);
                }

                @Override
                public void onCancelled(GestureDescription gestureDescription) {
                    super.onCancelled(gestureDescription);
                }
            }, null);
        }
    }
}
BrowJr
fonte
0

Você já tentou usar o script AutoIt ?

Você pode salvar coordenadas em janelas / telas específicas. Você pode manter o mouse pressionado enquanto desenha o padrão.

Eu também tenho alguns códigos / scripts de exemplo para você, se você quiser!


EDITAR:

De acordo com este tutorial, você pode usar o Auto-IT em C #.

Siga esses passos:

  1. Instale o Auto-IT
  2. Adicione o Auto-IT como referência no gerenciador de referência (AutoItX3.dll)
  3. Em seguida, importe a biblioteca que você adicionou: Using AutoItX3Lib;
  4. Crie um novo objeto AutoItX3 chamado 'auto': AutoItX3 auto = new AutoItX3();
  5. Agora você pode executar comandos Auto It.

Este é o exemplo completo para executar um clique de mouse:

Using AutoItX3Lib;
AutoItX3 auto = new AutoItX3();
auto.MouseClick("left", 78, 1133, 1, 35)


Com o AutoIt Window Info Toolvocê pode verificar as coordenadas que deseja usar.

Observe que existem diferenças entre os modos de coordenadas do mouse:

por exemplo: auto.AutoItSetOption("MouseCoordMode", 1)usará coordenadas absolutas da tela. Veja a fonte aqui .


Para manter o mouse pressionado, você pode verificar a Função MouseDown

JaFizz
fonte
11
Isso não ajudou. Sua sugestão é exatamente o que meu C # Windows Form Application já faz.
BrowJr