教你用.Net來玩微信跳一跳

来源:https://www.cnblogs.com/liuju150/archive/2018/01/12/WeChat-AutoJump_Net-OpenCV.html
-Advertisement-
Play Games

目前開發的所有代碼都已經上傳到了GitHub。歡迎大家來Star https://github.com/GiantLiu/AutoJump 目前程式分為“全自動版本”和“半自動版本” 全自動版本 WeChat.AutoJump.CMDApp 當手機連接好後,打開微信跳一跳 點擊"開始游戲"後。運行此 ...


目前開發的所有代碼都已經上傳到了GitHub。歡迎大家來Star

https://github.com/GiantLiu/AutoJump

目前程式分為“全自動版本”和“半自動版本”

全自動版本

WeChat.AutoJump.CMDApp

當手機連接好後,打開微信跳一跳

點擊"開始游戲"後。運行此程式。就可以實現自動跳了

半自動版本

WeChat.AutoJump.WinApp

此版本需要滑鼠左鍵點小黑人的底部,滑鼠右鍵點目標位的中心

然後程式就會自動跳到相應的位置

 

程式原理
1。將手機點擊到《跳一跳》小程式界面;點擊“開始游戲”後
2。用Adb工具獲取當前手機的截圖,半下載到本地
3.1。如果是半自動版本,那麼就要用滑鼠左右鍵來點擊起始和目標位置
然後程式會自動算出要跳動的距離與要點擊屏幕的時間。
3.2。如果是全自動版本,那麼程式會自動算出小黑人的位置與目標的中心點,
然後自動算距離與點擊屏幕的時間。

4。用Adb工具向手機發送點擊屏幕蓄力命令,完成一次跳動

目前程式只能支持Android設備,IOS設備只寫了介面,還沒有實現
步驟:

  • 安卓手機打開USB調試,設置》開發者選項》USB調試
  • 電腦與手機USB線連接,確保執行adb devices可以找到設備id

  • 界面轉至微信跳一跳游戲,點擊開始游戲
    運行自動/半自動版本程式,就可以開始游戲之路

  •  

代碼關鍵實現
1。通過adb拿到手機的屏幕截圖,其實就是向手機發送相關的命令

  第一條命令是把屏幕的截圖以png格式保存到手機SD卡
  第二條命令是把手機SD卡裡面的圖片下載到本地硬碟對應的目錄
  第三條命令是把手機里的截圖刪除
  第四條命令是發送屏幕按壓命令 從X:100,Y:100這個位置向X200,Y:200這個位置移動,其中時間為500毫秒

adb shell screencap -p /sdcard/1.png
adb pull /sdcard/1.png D:/Download/
adb shell rm /sdcard/1.png
adb shell input swipe 100 100 200 200 500

這裡是.net發送命令相關代碼

public string AdbCommand(string arg)
        {
            using (Process process = new Process())
            {
                var adbDirectoryPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "AndoridAdb");
                var adbPath = Path.Combine(adbDirectoryPath, "adb.exe");
                process.StartInfo.FileName = adbPath;
                process.StartInfo.Arguments = arg;
                process.StartInfo.UseShellExecute = false;
                process.StartInfo.RedirectStandardInput = true;   //重定向標準輸入   
                process.StartInfo.RedirectStandardOutput = true;  //重定向標準輸出   
                process.StartInfo.RedirectStandardError = true;   //重定向錯誤輸出
                process.StartInfo.CreateNoWindow = true;
                process.Start();
                var result = process.StandardOutput.ReadToEnd();
                process.WaitForExit();
                process.Close();
                return result;
            }
        }
View Code

2。如果是半自動版本,那麼要先滑鼠左鍵點小黑人的底部,然後滑鼠右鍵點目標位置的中間。
點完右鍵後。程式會自動算出兩點之間距離與時間。然後就跳一步了。這個沒有什麼技術問題

3。如果是全自動版本,那反第一步,你拿到屏幕截圖後。要分析出小黑人的位置
我這裡的話。就用了EmguCV (OpenCV的.net調用)。
我們可以用到OpenCV的模板匹配。MatchTemplate方法
模板的話。隨便找一張屏幕截圖,用PS把小黑人扣出來。保存為圖片就可以了
MatchTemplate會找出匹配最高的點。然後給出坐標,這樣,我們就可以算出小黑人的中心位置了

var tempGrayPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Template", "Current.png");

            var tempGrayImg = new Image<Rgb, byte>(tempGrayPath);

            var match = img.MatchTemplate(tempGrayImg, TemplateMatchingType.CcorrNormed);

            double min = 0, max = 0;
            Point maxp = new Point(0, 0);//最好匹配的點
            Point minp = new Point(0, 0);
            CvInvoke.MinMaxLoc(match, ref min, ref max, ref minp, ref maxp);
            Console.WriteLine(min + " " + max);
            CvInvoke.Rectangle(img, new Rectangle(maxp, new Size(tempGrayImg.Width, tempGrayImg.Height)), new MCvScalar(0, 0, 255), 3);

            var startPoint = new Point();
            startPoint.X = maxp.X + (int)(tempGrayImg.Width / 2.0);
            startPoint.Y = maxp.Y + tempGrayImg.Height - 2;
            CvInvoke.Rectangle(img, new Rectangle(startPoint, new Size(1, 1)), new MCvScalar(0, 0, 0), 3);
View Code


4。目標位置計算,
這也是程式最複雜的部分了,
我的實現步驟為
1:先把圖片裁剪到只保留中間的1/3有效分析區域
2:看小黑人在屏幕的左邊還是右邊,那麼目標就會在相反的區域。這樣我們就可以把目標區域的圖片剪切下來

////裁剪查找區域
            ////原圖片1/3以下,小黑人以上
            var newImgStart = imgHeightSplit;
            var newImgEnd = maxp.Y + tempGrayImg.Height;
            var newImgHeight = newImgEnd - newImgStart;
            Rectangle rect = new Rectangle(0, newImgStart, img.Width, newImgHeight);

            CvInvoke.cvSetImageROI(sourceImg, rect);
            var newImg = new Image<Rgb, byte>(sourceImg.Width, newImgHeight);
            CvInvoke.cvCopy(sourceImg, newImg, IntPtr.Zero);



            ////看小黑人在程式的左邊還是右邊
            ////如果在左邊,那目標點就在圖片的右邊
            bool targetInLeft = true;
            if (maxp.X < imgWidthCenter) targetInLeft = false;

            Rectangle halfRect;
            if (targetInLeft)
                halfRect = new Rectangle(0, 0, imgWidthCenter, newImgHeight);
            else
                halfRect = new Rectangle(imgWidthCenter, 0, imgWidthCenter, newImgHeight);

            CvInvoke.cvSetImageROI(newImg, halfRect);
            var halfImg = new Image<Rgb, byte>(imgWidthCenter, newImgHeight);
            CvInvoke.cvCopy(newImg, halfImg, IntPtr.Zero);
View Code

 

5。然後我們通過像素分析,找到目標的頂點
原理是:第一個點與後一個點對比,看變化大小
如果變化大小超過一個值。就認為是目標位了(跳一跳背景是漸變的)
這裡是方塊點。頂點就是一個點。當如果目標為圓體的時候
那頂度也能有幾個像素的Y軸都是相同的。那麼我們要把有幾個相同的找出來。取中間位置,算為頂點

Point topPoint = new Point();
            for (int i = 0; i < halfImg.Rows; i++)
            {
                for (int j = 0; j < halfImg.Cols - 1; j++)
                {
                    var cur = halfImg[i, j];
                    var next = halfImg[i, j + 1];
                    if (Math.Abs(RgbHelp.GetDiff(cur, next)) > 2)
                    {
                        var x = 2;
                        next = halfImg[i, j + x];
                        while (Math.Abs(RgbHelp.GetDiff(cur, next)) > 2)
                        {
                            x++;
                            next = halfImg[i, j + x];
                        }
                        topPoint.Y = i;
                        topPoint.X = j + (int)(x / 2.0);
                        break;
                    }
                }
                if (!topPoint.IsEmpty) break;
            }
            CvInvoke.Rectangle(halfImg, new Rectangle(topPoint, new Size(1, 1)), new MCvScalar(0, 0, 255), 3);

            ////這個頂點在原圖中的位置
            var oldTopX = topPoint.X;
            if (!targetInLeft) oldTopX += imgWidthCenter;
            var oldTopY = topPoint.Y + imgHeightSplit;
            var oldTopPoint = new Point(oldTopX, oldTopY);
            CvInvoke.Rectangle(img, new Rectangle(oldTopPoint, new Size(1, 1)), new MCvScalar(0, 0, 255), 3);
View Code

 

找到了相關的點。計算小黑人與目標的距離就不是難事了
然後就是發送跳的命令,一個步驟就完成了

 




您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • BlockingCollection集合是一個擁有阻塞功能的集合,它就是完成了經典生產者消費者的演算法功能。一般情況下,我們可以基於 生產者 - 消費者模式來實現併發。BlockingCollection<T> 類是最好的解決方案 剛結束的物聯網卡項目,我需要調用移動的某個具有批量獲取物聯網卡數據的接 ...
  • 概述 在之前寫的一篇關於async和await的前世今生的文章之後,大家似乎在async和await提高網站處理能力方面還有一些疑問,博客園本身也做了不少的嘗試。今天我們再來回答一下這個問題,同時我們會做一個async和await在WinForm中的嘗試,並且對比在4.5之前的非同步編程模式APM/E ...
  • 一個網頁,它是顯示圖片,但在一些瀏覽器,它卻顯示如下: Insus.NET猜,不是瀏覽器不相容,就是代碼有問題。 在代碼中,只是輸出數據流,圖片格式很多種,如jpg,png,bmp等,沒有指定,程式也不清楚要顯示什麼格式的圖片。因此,Insus.NET把代碼改為如下: context.Respons ...
  • 為什麼要學習表達式樹?表達式樹是將我們原來可以直接由代碼編寫的邏輯以表達式的方式存儲在樹狀的結構里,從而可以在運行時去解析這個樹,然後執行,實現動態的編輯和執行代碼。LINQ to SQL就是通過把表達式樹翻譯成SQL來實現的,所以瞭解表達樹有助於我們更好的理解 LINQ to SQL,同時如果你有 ...
  • 快樂的Lambda表達式 上一篇 背後的故事之 - 快樂的Lambda表達式(一)我們由淺入深的分析了一下Lambda表達式。知道了它和委托以及普通方法的區別,並且通過測試對比他們之間的性能,然後我們通過IL代碼深入瞭解了Lambda表達式,以及介紹瞭如何在.NET中用Lambda表達式來實現Jav ...
  • 快樂的Lambda表達式(二) 自從Lambda隨.NET Framework3.5出現在.NET開發者眼前以來,它已經給我們帶來了太多的欣喜。它優雅,對開發者更友好,能提高開發效率,天啊!它還有可能降低發生一些潛在錯誤的可能。LINQ包括ASP.NET MVC中的很多功能都是用Lambda實現的。 ...
  • C#集體類型( Collections in C#) 集合是.NET FCL(Framework Class Library)中很重要的一部分,也是我們開發當中最常用到的功能之一,幾乎是無處不在。俗話說知其然,知其所以然,平常看到IEnumerable,IEnumerator,ICollection ...
  • NanUI文檔目錄 "NanUI簡介" "開始使用NanUI" "打包並使用內嵌式的HTML/CSS/JS資源" "使用網頁來設計整個視窗" "如何實現C 與Javascript的相互通信" 如何處理NanUI中的下載過程 DonwloadHandler的使用 (待更新。。。) 如何處理NanUI中 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...