Greasy Fork镜像 支持简体中文。

自动上滑脚本

自用库

目前為 2024-12-30 提交的版本,檢視 最新版本

此腳本不應該直接安裝,它是一個供其他腳本使用的函式庫。欲使用本函式庫,請在腳本 metadata 寫上: // @require https://update.gf.qytechs.cn/scripts/521999/1512055/%E8%87%AA%E5%8A%A8%E4%B8%8A%E6%BB%91%E8%84%9A%E6%9C%AC.js

  1. importClass(android.content.Context);
  2. importClass(android.provider.Settings);
  3. importClass(android.app.KeyguardManager);
  4. try {
  5. var km = context.getSystemService(Context.KEYGUARD_SERVICE);//km.isKeyguardLocked(),km.isKeyguardSecure()
  6. let enabledServices = Settings.Secure.getString(context.getContentResolver(), Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES);
  7. //log('当前已启用的辅助服务\n', enabledServices);
  8. if (!enabledServices.match(/.*org\.autojs\.autoxjs\.v6\/com\.stardust\.autojs\.core\.accessibility\.AccessibilityService.*/g)) {
  9. let Services = (enabledServices ? enabledServices + ":" : "") + "org.autojs.autoxjs.v6/com.stardust.autojs.core.accessibility.AccessibilityService";
  10. Settings.Secure.putString(context.getContentResolver(), Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES, Services);
  11. Settings.Secure.putString(context.getContentResolver(), Settings.Secure.ACCESSIBILITY_ENABLED, '1');
  12. sleep(3000);
  13. }
  14. toastLog("成功开启AutoJS的辅助服务");
  15. } catch (error) {
  16. //受权方法:开启usb调试并使用adb工具链接手机,执行 adb shell pm grant org.autojs.autoxjs.v6 android.permission.WRITE_SECURE_SETTING
  17. toastLog("请受权AutoJS启用辅助服务");
  18. }
  19. auto.waitFor();
  20. //停止其它脚本
  21. engines.all().map((ScriptEngine) => {
  22. if (engines.myEngine().toString() !== ScriptEngine.toString()) {
  23. ScriptEngine.forceStop();
  24. }
  25. });
  26. //============================================================
  27. var AppName = ["抖音极速版","快手极速版"];
  28. var runAppName = AppName[0], times = 100; //滑动次数
  29. toastLog('当前分辨率:'+device.width+'X'+device.height);
  30. //toastLog('唯一标识码:'+device.fingerprint);
  31. const img_block = '';
  32.  
  33. //息屏状态将屏幕唤醒
  34. global.opentimes=0;
  35. while (!device.isScreenOn() || km.isKeyguardLocked()) {
  36. opentimes++;
  37. device.wakeUp();//唤醒设备
  38. toastLog('屏幕唤醒');
  39. sleep(1500); //等待屏幕亮起
  40. back();//如果锁屏后收到新消息,上滑不能解锁屏幕,需要返回一次后上滑
  41. device.keepScreenOn();//一直保持屏幕常亮
  42. sleep(1500);
  43. if (km.isKeyguardSecure()) {
  44. toastLog('密码解锁');
  45. //待开发
  46. break;
  47. } else {
  48. toastLog('上滑解锁');
  49. swipe(device.width / 2, device.height * 0.8, device.width / 2, device.height * 0.3, 400);
  50. }
  51. sleep(1500);
  52. if(opentimes>3){
  53. toastLog('解锁失败,请尝试重启本软件并开启无障碍服务');
  54. break;
  55. }
  56. }
  57. global.oledwin=null;
  58. global.looptimes=times;
  59. function Main() {
  60. toastLog('进入主程序');
  61. for (i = 0; i < AppName.length; i++) {
  62. looptimes=times;
  63. var packageName = getPackageName(AppName[i]);
  64. if (packageName) {
  65. toastLog('启动应用:' + AppName[i]);
  66. var appstate = launchApp(AppName[i]);
  67. sleep(5000);
  68. if (appstate) {
  69. toastLog("应用正在运行");
  70. } else {
  71. toastLog("无法自启动,需模拟点击");
  72. home();
  73. sleep(3000);
  74. var app = id("item_title").text(AppName[i]).visibleToUser(true).findOne(2000);
  75. if (app) {
  76. click(app.bounds().centerX(), app.bounds().top - 50);
  77. sleep(10000);
  78. }
  79. }
  80. runAppName=AppName[i];//安全线程中使用
  81. while (looptimes > 0) {
  82. //toastLog(currentActivity());
  83. var tiktokhomepage = className("Button").descStartsWith("侧边栏").clickable(true).boundsInside(0, 0, 500, 500).visibleToUser(true).findOne(1000);
  84. var giftshowhomepg = idMatches(/.*\/left_btn|.*\/thanos_home_top_search/).boundsInside(0, 0, 500, 500).visibleToUser(true).findOne(1000);
  85. if (tiktokhomepage||giftshowhomepg) {
  86. looptimes--;
  87. var videoDuration = 0;
  88. if(tiktokhomepage){
  89. var seekBar=className('android.widget.SeekBar').desc('进度条').findOne(1000);
  90. }else{
  91. var seekBar=className('android.widget.HorizontalScrollView').idMatches('.*\/tab_layout').findOne(1000);
  92. }
  93. if (seekBar) {
  94. isvideo = true;
  95. let y1 = tiktokhomepage?(seekBar.bounds().centerY()-5):(seekBar.bounds().top-5);
  96. let x1 = random(100, 300);
  97. let x2 = random(500, 800);
  98. let duration_thread = threads.start(function () {
  99. var durationText = className('TextView').textMatches(/[0-9]+:[0-9]+/).boundsInside(device.width/2, 2 * device.height / 3, device.width, device.height).findOne(2000);
  100. if(durationText){
  101. videoDuration = getDouyinVideoDuration(durationText.text());
  102. }
  103. duration_thread.interrupt();
  104. });
  105. gesture(random(800, 1200), [ [x1, y1],[x2, y1],[x1, y1] ]);
  106. console.log("视频时长:",videoDuration+'s');
  107. }
  108. var sleepTime=(videoDuration>0&&videoDuration<180)?videoDuration:random(6, 30);//每个视频随机时间 6-30秒
  109. console.verbose("浏览:" + (times-looptimes), "停留:" + sleepTime + "s");
  110. cutDownBySleep(sleepTime,'观看视频');
  111. randomHeart();//拟人化
  112. } else {
  113. var living = id("root").desc("关闭").clickable(true).boundsInside(device.width-300, 0, device.width, 300).visibleToUser(true).findOne(1000);//直播间
  114. if (living) {
  115. isvideo = true;
  116. toastLog("1.退出直播间");
  117. click(living.bounds().centerX(), living.bounds().centerY());
  118. sleep(2000);
  119. }
  120. if (currentActivity() == 'com.ss.android.ugc.aweme.live.LivePlayActivity') {
  121. toastLog("2.退出直播间");
  122. isvideo = true;
  123. back();
  124. }
  125. toast('不在抖音或快手页面');
  126. oledwin=null;
  127. sleep(3000);
  128. }
  129. }
  130. closeApp(runAppName);
  131. } else {
  132. toastLog("未安装:" + AppName[i]);
  133. }
  134. }
  135. toastLog("自动刷屏完成");
  136. try {
  137. device.cancelKeepingAwake();
  138. //熄屏
  139. runtime.accessibilityBridge.getService().performGlobalAction(android.accessibilityservice.AccessibilityService.GLOBAL_ACTION_LOCK_SCREEN);
  140. } catch (e) {
  141.  
  142. }
  143. //停止本脚本
  144. engines.myEngine().forceStop();
  145. }
  146.  
  147.  
  148. /**
  149. * 倒计时
  150. * @param {倒计时秒} lasterSecend
  151. * @param {显示提示} message
  152. */
  153. function cutDownBySleep(lasterSecend, message) {
  154. message = message || "";
  155. floaty.closeAll();
  156. var fwin = floaty.rawWindow(
  157. `<vertical id="frame" alpha="0" w="{{device.width-500}}px" h="150px">
  158. <card w="auto" h="auto" layout_gravity="center" cardCornerRadius="5dp" cardBackgroundColor="#eeeeee" >
  159. <text id="title" text="" textColor="#333333" textSize="13sp" padding="12 8" />
  160. </card>
  161. </vertical>`
  162. );
  163. fwin.setTouchable(false);
  164. sleep(500);
  165. for (let i = lasterSecend; i > 0; i--) {
  166. if (oledwin) { break; }
  167. if (!fwin || !fwin.title) { break; }
  168. ui.run(() => {
  169. fwin.title.setText(message + "剩余" + i + "秒");
  170. fwin.frame.attr("alpha", 0.8);
  171. fwin.setTouchable(false);
  172. let x = parseInt((device.width - fwin.width) / 2);
  173. let y = device.height-550;
  174. fwin.setPosition(x, y);
  175. });
  176. sleep(1000);
  177. }
  178. fwin=null;
  179. floaty.closeAll();
  180. sleep(500);
  181. }
  182. //视频时长(秒)获取函数
  183. function getDouyinVideoDuration(durationStr) {
  184. if (durationStr) {
  185. //log('1',durationStr);
  186. var durationMatch = durationStr.match(/[0-9]+:[0-9]+/);
  187. if (durationMatch) {
  188. //log('2',durationMatch);
  189. var minutes = 0,seconds = 0;
  190. var parts = durationMatch[0].split(":");
  191. if (parts.length === 2) {
  192. //log('3',parts);
  193. minutes = parseInt(parts[0], 10);
  194. seconds = parseInt(parts[1], 10);
  195. return minutes * 60 + seconds + 3;
  196. }
  197. }
  198. }
  199. return 0;
  200. }
  201. //=============================================================
  202. /**随机拟人化(防止被判定是机器)**/
  203. function randomHeart(num) {
  204. if (7!=num&&text('当前无新视频').visibleToUser(true).findOne(1000)) {
  205. console.log("当前无新视频");
  206. click(device.right - 100, device.top - 100);
  207. randomHeart(7);//切换频道
  208. sleep(1000);
  209. return;
  210. }
  211. let randomIndex = num ? num : random(1, 30);
  212. //加速播放
  213. if(randomIndex % 9 == 0){
  214. var seekBar1=className('android.widget.SeekBar').descMatches(/.*进度条.*/).findOne(1000);
  215. var seekBar2=className('android.widget.HorizontalScrollView').idMatches('.*\/tab_layout').findOne(1000);
  216. if(seekBar1||seekBar2){
  217. let x1=random(90, 120);
  218. let y1=device.height/3;
  219. gestures([0, 1500, [x1,y1], [x1,y1]],[1400, 1500, [x1,y1], [1.1*x1, 2*y1]]);
  220. return;
  221. }
  222. }
  223. //随机下滑
  224. if (randomIndex % 8 == 0) {
  225. console.log('拟人:随机下滑');
  226. swipe(device.width / 2, device.height * 0.1 + randomIndex, device.width / 2, device.height * 0.9 - randomIndex, random(500, 1500));
  227. return;
  228. }
  229. //随机切换频道
  230. if (randomIndex == 7) {
  231. var idList = [];
  232. className('TextView').clickable(true).descEndsWith(',按钮').boundsInside(0, 0, device.width, 300).find().forEach(function(tv){
  233. if(!tv.desc().match(/.*已选中.*|.*团购.*|.*商城.*|.*热点.*|.*直播中.*/)){
  234. //log(tv.desc());
  235. idList.push(tv);
  236. }
  237. });
  238. if(idList.length>0){
  239. let index = random(1, idList.length) - 1;
  240. console.log('拟人:切换频道:' + idList[index].desc());
  241. idList[index].click();
  242. sleep(2000);
  243. if (text('发现通讯录朋友').visibleToUser(true).findOne(1000)) {
  244. console.log("关注中没有视频");
  245. randomHeart(7);//再次切换频道
  246. sleep(2000);
  247. slidingByCurve();
  248. }
  249. return;
  250. }
  251. }
  252. //连续上滑
  253. if (randomIndex == 6) {
  254. console.log('拟人:连续上滑');
  255. var k = random(2, 4);
  256. for (var i = 0; i < k; i++) {
  257. var j = random(2, 5);
  258. if (j == 3) {
  259. swipe(device.width / j, device.height * 0.1 + j * k, device.width / j, device.height * 0.9 - j * k, j * 50);
  260. } else {
  261. swipe(device.width / j, device.height * 0.9 - j * k, device.width / j, device.height * 0.1 + j * k, j * 50);
  262. }
  263. sleep(j * 250);
  264. }
  265. return;
  266. }
  267. //随机恢复到首页
  268. if(randomIndex % 4 == 0){
  269. console.log('拟人:随机回首页');
  270. sleep(3000);back();sleep(3000);back();sleep(3000);
  271. return;
  272. }
  273. //向上滑
  274. slidingByCurve();
  275. }
  276.  
  277. /**------------------------------------------------------------
  278. * 从下往上滑动,随机坐标
  279. */
  280. function slidingByLine() {
  281. // top X,Y范围
  282. tx = randomPointLoc(parseInt(device.width / 3), parseInt(device.width / 2));
  283. ty = randomPointLoc(parseInt(device.height / 5), parseInt(device.height / 4));
  284. // bottom X,Y 范围
  285. bx = randomPointLoc(parseInt(device.width / 3), parseInt(device.width / 2));
  286. by = randomPointLoc(parseInt(3 * device.height / 4), parseInt(4 * device.height / 5));
  287.  
  288. slidingTime = randomRangeTime(0.8, 1.3);
  289. log("上滑:随机直线");
  290. //log("X: "+ Math.abs(bx-tx) + " Y: "+ Math.abs(by - ty));
  291. swipe(bx, by, tx, ty, slidingTime);
  292. }
  293.  
  294. /*从下往上滑动,曲线滑动,随机坐标*/
  295. function slidingByCurve() {
  296. // top X,Y范围
  297. tx = randomPointLoc(parseInt(device.width / 3), parseInt(device.width / 2));
  298. ty = randomPointLoc(parseInt(device.height / 5), parseInt(device.height / 4));
  299. // bottom X,Y 范围
  300. bx = randomPointLoc(parseInt(device.width / 3), parseInt(device.width / 2));
  301. by = randomPointLoc(parseInt(3 * device.height / 4), parseInt(4 * device.height / 5));
  302.  
  303. slidingTime = randomRangeTime(0.2, 0.6);
  304. log("上滑:仿真曲线");
  305. //log("X: "+ Math.abs(bx-tx) + " Y: "+ Math.abs(by - ty));
  306. sml_move(bx, by, tx, ty, slidingTime);
  307. }
  308.  
  309. /**
  310. * 上部左右滑动
  311. */
  312. function left2right(direction) {
  313. var intX=parseInt(Math.random()*200+400);
  314. var intY=parseInt(Math.random()*200+200);
  315. var distance=parseInt(Math.random()*100+device.height/4);
  316. switch (direction) {
  317. case 1:
  318. //向上小距离
  319. sml_move(intX, intY + distance, intX, intY, 400);
  320. break;
  321. case 2:
  322. //向下小距离
  323. sml_move(intX, intY, intX, intY + distance, 400);
  324. break;
  325. case 3:
  326. //向左翻屏
  327. sml_move(
  328. device.width / 2 + parseInt(Math.random() * 100) + 300,
  329. device.height / 4 - parseInt(Math.random() * 200) + 100,
  330. 0 + parseInt(Math.random() * 100),
  331. device.height / 5 + parseInt(Math.random() * 100),
  332. 500
  333. );
  334. break;
  335. case 4:
  336. //向右翻屏
  337. sml_move(
  338. device.width / 2 - parseInt(Math.random() * 100) - 300,
  339. device.height / 5 - parseInt(Math.random() * 200) + 100,
  340. device.width - parseInt(Math.random() * 100),
  341. device.height / 4 + parseInt(Math.random() * 100),
  342. 500
  343. );
  344. break;
  345. }
  346. sleep(1000);
  347. }
  348.  
  349. /**
  350. * 随机位置点
  351. * @param {起始值} start
  352. * @param {结束值} end
  353. * @returns
  354. */
  355. function randomPointLoc(start,end){
  356. len = Math.abs(end - start);
  357. loc = Math.floor(Math.random() * len) + start;
  358. return loc;
  359. }
  360.  
  361. /**
  362. * 从几秒到几秒
  363. * @param {开始秒} start
  364. * @param {结束秒} end
  365. * @returns
  366. */
  367. function randomRangeTime(start,end){
  368. len = Math.abs(end -start)*1000;
  369. ms = Math.floor(Math.random() * len) + start*1000;
  370. return ms;
  371. }
  372.  
  373. /**
  374. * 仿真随机带曲线滑动
  375. * @param {起点x} qx
  376. * @param {起点y} qy
  377. * @param {终点x} zx
  378. * @param {终点y} zy
  379. * @param {滑动时间,单位毫秒} time
  380. */
  381. function sml_move(qx, qy, zx, zy, time) {
  382. var xxy = [time];
  383. var point = [];
  384. var dx0 = {
  385. "x": qx,
  386. "y": qy
  387. };
  388. var dx1 = {
  389. "x": random(qx - 150, qx + 150),
  390. "y": random(qy, qy + 50)
  391. };
  392. var dx2 = {
  393. "x": random(zx - 150, zx + 150),
  394. "y": random(zy, zy + 50),
  395. };
  396. var dx3 = {
  397. "x": zx,
  398. "y": zy
  399. };
  400. for (var i = 0; i < 4; i++) {
  401. eval("point.push(dx" + i + ")");
  402. }
  403. // log(point[3].x)
  404. for (let i = 0; i < 1; i += 0.08) {
  405. let newPoint=bezier_curves(point, i);
  406. xxyy = [parseInt(newPoint.x), parseInt(newPoint.y)]
  407. xxy.push(xxyy);
  408. }
  409. try {
  410. gesture.apply(null, xxy);
  411. } catch (e) {
  412. log(xxy);
  413. }
  414. }
  415. function bezier_curves(cp, t) {
  416. cx = 3.0 * (cp[1].x - cp[0].x);
  417. bx = 3.0 * (cp[2].x - cp[1].x) - cx;
  418. ax = cp[3].x - cp[0].x - cx - bx;
  419. cy = 3.0 * (cp[1].y - cp[0].y);
  420. by = 3.0 * (cp[2].y - cp[1].y) - cy;
  421. ay = cp[3].y - cp[0].y - cy - by;
  422. tSquared = t * t;
  423. tCubed = tSquared * t;
  424. result = {
  425. "x": 0,
  426. "y": 0
  427. };
  428. result.x = (ax * tCubed) + (bx * tSquared) + (cx * t) + cp[0].x;
  429. result.y = (ay * tCubed) + (by * tSquared) + (cy * t) + cp[0].y;
  430. return result;
  431. }
  432.  
  433. //关闭软件
  434. function closeApp(appname) {
  435. let packageName = getPackageName(appname);
  436. // 使用ADB命令强行结束进程
  437. //shell("adb shell am force-stop " + packageName);
  438. console.warn('关闭应用:' + appname);
  439. app.openAppSetting(packageName);
  440. text(app.getAppName(packageName)).waitFor();
  441. let is_sure = textMatches(/.*强行停止.*/).visibleToUser(true).findOne(1000);
  442. if (is_sure&&is_sure.enabled()) {
  443. try {
  444. var btn = className("Button").text('强行停止').visibleToUser(true).findOne(1000);
  445. if (btn) btn.click();
  446. sleep(1000);
  447. btn = className("Button").text('强行停止').visibleToUser(true).findOne(1000);
  448. if (btn) btn.click();
  449. sleep(1000);
  450. btn = className("Button").text('确定').visibleToUser(true).findOne(1000);
  451. if (btn) btn.click();
  452. back(); back(); back();
  453. home();
  454. } catch (e) {
  455. log(app.getAppName(packageName) + "应用已被关闭");
  456. sleep(1000);
  457. back(); back(); back();
  458. home();
  459. }
  460. } else {
  461. log(app.getAppName(packageName) + "应用不能被正常关闭");
  462. back(); back(); back();
  463. home();
  464. }
  465. }
  466.  
  467. importClass(org.opencv.imgproc.Imgproc);
  468. importClass(org.opencv.core.Core);
  469. importClass(org.opencv.core.Rect);
  470. importClass(org.opencv.core.Mat);
  471. importClass(org.opencv.core.Point);
  472. importClass(org.opencv.core.Size);
  473. importClass(org.opencv.core.CvType);
  474. importClass(org.opencv.core.Scalar);
  475. importClass(org.opencv.imgcodecs.Imgcodecs);
  476.  
  477. /**
  478. * @param {number[]} region 是一个两个或四个元素的数组。
  479. * (region[0], region[1])表示找色区域的左上角;region[2]*region[3]表示找色区域的宽高。如果只有region只有两个元素,则找色区域为(region[0], region[1])到屏幕右下角。
  480. * 如果不指定region选项,则找色区域为整张图片。
  481. * @param {*} img
  482. * @returns {org.opencv.core.Rect}
  483. */
  484. function buildRegion(region, img) {
  485. if (region == undefined) { region = []; }
  486. let x = region[0] === undefined ? 0 : region[0];
  487. let y = region[1] === undefined ? 0 : region[1];
  488. let width = region[2] === undefined ? img.getWidth() - x : region[2];
  489. let height = region[3] === undefined ? img.getHeight() - y : region[3];
  490. if (x < 0 || y < 0 || x + width > img.width || y + height > img.height) {
  491. throw new Error(
  492. 'out of region: region = [' + [x, y, width, height] + '], image.size = [' + [img.width, img.height] + ']'
  493. );
  494. }
  495. return new Rect(x, y, width, height);
  496. }
  497.  
  498. /**
  499. * @param {number} threshold 图片相似度。取值范围为0~1的浮点数。默认值为0.9
  500. * @param {number[]} region 找图区域
  501. * @param {number[]} scaleFactors 大图的宽高缩放因子,默认为 [1, 0.9, 1.1, 0.8, 1.2]
  502. * @param {number} max 找图结果最大数量,默认为5
  503. * @param {boolean} grayTransform 是否进行灰度化预处理,默认为true。
  504. * 通常情况下将图像转换为灰度图可以简化匹配过程并提高匹配的准确性,当然,如果你的匹配任务中颜色信息对匹配结果具有重要意义,
  505. * 可以跳过灰度化步骤,直接在彩色图像上进行模板匹配。
  506. */
  507. function MatchOptions(threshold, region, scaleFactors, max, grayTransform) {
  508. this.threshold = threshold;
  509. this.region = region;
  510. this.scaleFactors = scaleFactors;
  511. this.max = max;
  512. this.grayTransform = grayTransform;
  513. }
  514.  
  515. const defaultMatchOptions = new MatchOptions(0.9, undefined, [[1, 1], [0.9, 0.9], [1.1, 1.1], [0.8, 0.8], [1.2, 1.2]], 5, true);
  516. // 校验参数
  517. MatchOptions.check = function (options) {
  518. if (options == undefined) {
  519. return defaultMatchOptions;
  520. }
  521. // deep copy
  522. let newOptions = JSON.parse(JSON.stringify(options));
  523. if (newOptions.threshold == undefined) {
  524. newOptions.threshold = defaultMatchOptions.threshold;
  525. }
  526. if (newOptions.region && !Array.isArray(newOptions.region)) {
  527. throw new TypeError('region type is number[]');
  528. }
  529. if (newOptions.max == undefined) {
  530. newOptions.max = defaultMatchOptions.max;
  531. }
  532. if (newOptions.scaleFactors == undefined) {
  533. newOptions.scaleFactors = defaultMatchOptions.scaleFactors;
  534. } else if (!Array.isArray(newOptions.scaleFactors)) {
  535. throw new TypeError('scaleFactors');
  536. }
  537. for (let index = 0; index < newOptions.scaleFactors.length; index++) {
  538. let factor = newOptions.scaleFactors[index];
  539. if (Array.isArray(factor) && factor[0] > 0 && factor[1] > 0) {
  540. // nothing
  541. } else if (typeof factor === 'number') {
  542. newOptions.scaleFactors[index] = [factor, factor];
  543. } else {
  544. throw new TypeError('scaleFactors');
  545. }
  546. }
  547. if (newOptions.grayTransform === undefined) {
  548. newOptions.grayTransform = defaultMatchOptions.grayTransform;
  549. }
  550.  
  551. return newOptions;
  552. };
  553.  
  554. function Match(point, similarity, scaleX, scaleY) {
  555. this.point = point;
  556. this.similarity = similarity;
  557. this.scaleX = scaleX;
  558. this.scaleY = scaleY;
  559. }
  560.  
  561. /**
  562. * 找图,在图中找出所有匹配的位置
  563. * @param {Image} img
  564. * @param {Image} template
  565. * @param {MatchOptions} options 参数见上方定义
  566. * @returns {Match[]}
  567. */
  568. function matchTemplate(img, template, options) {
  569. if (img == null || template == null) {
  570. throw new Error('ParamError');
  571. }
  572. options = MatchOptions.check(options);
  573. //console.log('参数:', options);
  574.  
  575. let largeMat = img.mat;
  576. let templateMat = template.mat;
  577. let largeGrayMat;
  578. let templateGrayMat;
  579. if (options.region) {
  580. options.region = buildRegion(options.region, img);
  581. largeMat = new Mat(largeMat, options.region);
  582. }
  583. // 灰度处理
  584. if (options.grayTransform) {
  585. largeGrayMat = new Mat();
  586. Imgproc.cvtColor(largeMat, largeGrayMat, Imgproc.COLOR_BGR2GRAY);
  587. templateGrayMat = new Mat();
  588. Imgproc.cvtColor(templateMat, templateGrayMat, Imgproc.COLOR_BGR2GRAY);
  589. }
  590. // =================================================
  591. let finalMatches = [];
  592. for (let factor of options.scaleFactors) {
  593. let [fx, fy] = factor;
  594. let resizedTemplate = new Mat();
  595. Imgproc.resize(templateGrayMat || templateMat, resizedTemplate, new Size(), fx, fy, Imgproc.INTER_LINEAR);
  596. // 执行模板匹配,标准化相关性系数匹配法
  597. let matchMat = new Mat();
  598. Imgproc.matchTemplate(largeGrayMat || largeMat, resizedTemplate, matchMat, Imgproc.TM_CCOEFF_NORMED);
  599.  
  600. let currentMatches = _getAllMatch(matchMat, resizedTemplate, options.threshold, factor, options.region);
  601. //console.log('缩放比:', factor, '可疑目标数:', currentMatches.length);
  602. for (let match of currentMatches) {
  603. if (finalMatches.length === 0) {
  604. finalMatches = currentMatches.slice(0, options.max);
  605. break;
  606. }
  607. if (!isOverlapping(finalMatches, match)) {
  608. finalMatches.push(match);
  609. }
  610. if (finalMatches.length >= options.max) {
  611. break;
  612. }
  613. }
  614. resizedTemplate.release();
  615. matchMat.release();
  616. if (finalMatches.length >= options.max) {
  617. break;
  618. }
  619. }
  620. largeMat !== img.mat && largeMat.release();
  621. largeGrayMat && largeGrayMat.release();
  622. templateGrayMat && templateGrayMat.release();
  623.  
  624. return finalMatches;
  625. }
  626.  
  627. function _getAllMatch(tmResult, templateMat, threshold, factor, rect) {
  628. let currentMatches = [];
  629. let mmr = Core.minMaxLoc(tmResult);
  630.  
  631. while (mmr.maxVal >= threshold) {
  632. // 每次取匹配结果中的最大值和位置,从而使结果按相似度指标从高到低排序
  633. let pos = mmr.maxLoc; // Point
  634. let value = mmr.maxVal;
  635.  
  636. let start = new Point(Math.max(0, pos.x - templateMat.width() / 2), Math.max(0, pos.y - templateMat.height() / 2));
  637. let end = new Point(
  638. Math.min(tmResult.width() - 1, pos.x + templateMat.width() / 2),
  639. Math.min(tmResult.height() - 1, pos.y + templateMat.height() / 2)
  640. );
  641. // 屏蔽已匹配到的区域
  642. Imgproc.rectangle(tmResult, start, end, new Scalar(0), -1);
  643. mmr = Core.minMaxLoc(tmResult);
  644.  
  645. if (rect) {
  646. pos.x += rect.x;
  647. pos.y += rect.y;
  648. start.x += rect.x;
  649. start.y += rect.y;
  650. end.x += rect.x;
  651. end.y += rect.y;
  652. }
  653. let match = new Match(pos, value, factor[0], factor[1]);
  654. // 保存匹配点的大致范围,用于后续去重。设置enumerable为false相当于声明其为私有属性
  655. Object.defineProperty(match, 'matchAroundRect', { value: new Rect(start, end), writable: true, enumerable: false });
  656. currentMatches.push(match);
  657. }
  658.  
  659. return currentMatches;
  660. }
  661.  
  662. /**
  663. * 判断新检测到的点位是否与之前的某个点位重合。
  664. * @param {Match[]} matches
  665. * @param {Match} newMatch
  666. * @returns {boolean}
  667. */
  668. function isOverlapping(matches, newMatch) {
  669. for (let existingMatch of matches) {
  670. // 也可判断两点间的距离,但是平方、开方运算不如比较范围简单高效
  671. if (existingMatch.matchAroundRect.contains(newMatch.point)) {
  672. if (newMatch.similarity > existingMatch.similarity) {
  673. existingMatch.point = newMatch.point;
  674. existingMatch.similarity = newMatch.similarity;
  675. existingMatch.scaleX = newMatch.scaleX;
  676. existingMatch.scaleY = newMatch.scaleY;
  677. existingMatch.matchAroundRect = newMatch.matchAroundRect;
  678. }
  679. return true;
  680. }
  681. }
  682. return false;
  683. }
  684. /**
  685. * 根据搜图结果在原图上画框
  686. * @param {Match[]} matches
  687. * @param {*} srcMat
  688. * @param {*} templateMat
  689. */
  690. function showMatchRectangle(matches, srcMat, templateMat) {
  691. for (let match of matches) {
  692. let start = match.point;
  693. let end = new Point(
  694. match.point.x + templateMat.width() * match.scaleX,
  695. match.point.y + templateMat.height() * match.scaleY
  696. );
  697. Imgproc.rectangle(srcMat, start, end, new Scalar(0, 0, 255), 3);
  698. }
  699.  
  700. const saveName = '/sdcard/Download/temp.jpg';
  701. let img2 = images.matToImage(srcMat);
  702. images.save(img2, saveName);
  703. app.viewFile(saveName);
  704. img2.recycle();
  705. }
  706.  
  707. /** 图像识别定位
  708. * @param {images} templateImage64 预定的搜索模板图片base64
  709. * @param {images} ScreenImage 被搜索图片,如果不提供则全屏截图
  710. * @param {number[]} option 配置参数
  711. * @return {result[]}[ { point: {27.0, 1615.0}, similarity: 1, scaleX: 1, scaleY: 1 } ]
  712. * @return {bounds{}}{ left: 27,top: 1615,right: 157,bottom: 1745,centerX: 92,centerY: 1680 }
  713. //option:{ threshold: 0.85, region: [0, 0], grayTransform: true, max: 3 }
  714. //img_block region: [device.width / 2, 200, device.width / 2, device.height / 3],grayTransform:true,
  715. */
  716. function FindPicture(tempBase64,ScreenImage) {
  717. let templateImage64 = '', options = null;
  718. if (tempBase64 == 'img_block') {
  719. if (!img_block) return;
  720. templateImage64 = img_block;
  721. options = { threshold: 0.8, region: [device.width / 2, 200, device.width / 2, device.height / 3], grayTransform: true, max: 3 };
  722. } else {
  723. return;
  724. }
  725.  
  726. let largeImage = ScreenImage ? ScreenImage : captureScreen();
  727. let template = images.fromBase64(templateImage64);
  728. //images.read(files.getSdcardPath() + '/脚本/template.jpg');
  729.  
  730. //console.log('大图尺寸:', [largeImage.getWidth(), largeImage.getHeight()]);
  731. //console.log('模板尺寸:', [template.getWidth(), template.getHeight()]);
  732. let bounds = null;
  733. let startTs = Date.now();
  734. let result = matchTemplate(largeImage, template, options);
  735. if (result.length > 0) {
  736. let left = result[0].point.x;
  737. let top = result[0].point.y;
  738. let right = parseInt(left + template.getWidth() * result[0].scaleX);
  739. let bottom = parseInt(top + template.getHeight() * result[0].scaleY);
  740. let centerX = parseInt(left + (right - left) / 2);
  741. let centerY = parseInt(top + (bottom - top) / 2);
  742. bounds = { left: left, top: top, right: right, bottom: bottom, centerX: centerX, centerY: centerY };
  743. }
  744. //console.log('识别耗时:', (Date.now() - startTs) / 1000);
  745. console.log(tempBase64, result.length==0?{}:result[0].point);
  746. // 将结果画框展示
  747. //showMatchRectangle(result, largeImage.mat, template.mat);
  748. template.recycle();
  749. setTimeout(function () { largeImage.recycle(); }, 6000);
  750. return bounds;
  751. }
  752.  
  753. /** 拖动滑块*/
  754. function dragSlider(ax,by,br,bigimg) {
  755. var img = bigimg;
  756. if(!bigimg){
  757. var pic = images.grayscale(captureScreen());
  758. //img = images.adaptiveThreshold(images.grayscale(pic), 255, "MEAN_C", "BINARY", 5, 10);
  759. img = images.inRange(pic, '#000000', '#666666');
  760. }
  761. var newimg=images.cvtColor(img, "GRAY2RGBA");
  762. //if(bigimg){
  763. // images.save(newimg, files.getSdcardPath() + '/脚本/2.jpg', "jpg", 100);
  764. // app.viewFile(files.getSdcardPath() + '/脚本/2.jpg');
  765. //}
  766. var t = random(1555,2222);
  767. var xy= FindPicture('img_block',newimg);
  768. //console.info("识别结果:" , xy);
  769. if (xy) {
  770. toastLog("识别成功:" +ax+", "+by);
  771. sml_move(ax, by, xy.centerX, by, t);
  772. } else if(!bigimg) {
  773. toastLog("识别有误,二次识别");
  774. bigimg = images.inRange(pic, '#999999', '#ffffff');
  775. //bigimg = images.adaptiveThreshold(images.grayscale(pic), 255, "MEAN_C", "BINARY", 5, 10);
  776. dragSlider(ax, by, br, bigimg);
  777. }else{
  778. toastLog("识别有误,尝试滑动");
  779. sml_move(ax, by, br-150, by, t);
  780. }
  781. }
  782.  
  783. requestScreenCapture(false);//请求截图权限
  784. runtime.getImages().initOpenCvIfNeeded();//初始化OpenCv
  785.  
  786. /**
  787. *监控脚本是否卡在某界面不动,发现此情况重启脚本
  788. */
  789. function Observer() {
  790. if (oledwin) {return true;}
  791. function unique(arr) {
  792. let newArr = [arr[0]];
  793. for (let i = 1; i < arr.length; i++) {
  794. let flag = false;
  795. for (var j = 0; j < newArr.length; j++) {
  796. if (arr[i] == newArr[j]) {
  797. flag = true;
  798. break;
  799. }
  800. }
  801. if (!flag) {
  802. newArr.push(arr[i]);
  803. }
  804. }
  805. return newArr;
  806. }
  807. currentActis = new Array();
  808. for (let c = 0; c < 59; c++) {//连续扫描60秒后返回结果,如果60秒停留在同一活动页面,则就要重启线程了
  809. //关闭自动弹出的层
  810. var btn=idMatches(/.*\/close/).visibleToUser(true).findOne(1000);
  811. if (btn) {
  812. console.log('点左上角关闭弹出层Observer');
  813. click(btn.bounds().centerX(), btn.bounds().centerY());
  814. }
  815. var accept = textMatches(/立即邀请/).visibleToUser(true).findOne(1000);
  816. if (accept) {
  817. console.log('点立即邀请');
  818. back();
  819. }
  820. //var mypg = desc("返回").visibleToUser(true).findOne(1000);
  821. //if(mypg){
  822. //console.log("点击返回");
  823. //click(mypg.bounds().centerX(), mypg.bounds().centerY());
  824. //}
  825. var btntxt = textMatches(/同意|满意|关闭|关闭应用|不在提醒|我知道了|以后再说|暂不使用|忽略提醒|仍要退出|不感兴趣|去验证|立即领取|提醒我每天来领/).visibleToUser(true).findOne(1000);
  826. if (btntxt) {
  827. console.warn('点击:', btntxt.text());
  828. //截图保存界面,以备后续查看
  829. //captureScreen(files.getSdcardPath() + '/脚本/Observer1_' + currentActivity() + '.jpg');
  830. click(btntxt.bounds().centerX(),btntxt.bounds().centerY());
  831. }
  832. //currentActivity()='.*FaceRecognitionActivity';
  833. var block=className('android.view.View').text('向右拖动滑块填充拼图').visibleToUser(true).findOne(1000);
  834. if(block){
  835. var aa=className('android.widget.Image').textMatches(/cutPic\?captchaSn.*/).visibleToUser(true).findOne(1000);
  836. //log('向右拖动滑块填充拼图',block.bounds().left);
  837. var ax = aa.bounds().centerX();
  838. var by = block.bounds().centerY();
  839. var br = block.bounds().right;
  840. dragSlider(ax,by,br);
  841. }
  842. //toastLog(currentActivity());
  843. currentActis[c] = currentActivity();
  844. sleep(1000);//这是每秒扫描一次活动页
  845. }
  846. //toastLog(currentActivity());
  847. let ac = unique(currentActis);
  848. let cc = currentActivity().match(/.*HomeActivity|.*PhotoDetailActivity|.*AwardVideoPlayActivity|.*AdKwaiRnActivity|.*app\.Dialog|android\.widget\.FrameLayout|.*ToastDialog|.*ScreenCaptureRequestActivity/);
  849. if (ac.length == 1 && !cc) {
  850. console.log(ac,currentActivity());
  851. return false
  852. }
  853. return true
  854. }
  855. //let times = rawInput("请输入要自动刷的视频次数:","50");
  856. // 》》》》》》》》》》》》》》》》》》》 START
  857. work_thread = threads.start(function () {
  858. Main();
  859. });
  860. observer_thread = threads.start(function () {
  861. setInterval(function () {
  862. console.verbose('--------多线程安全检测---------');
  863. if (!Observer()&&looptimes>0) {
  864. work_thread.interrupt();
  865. work_thread = threads.start(function () {
  866. console.warn("Main线程在5秒后重启!",currentActivity());
  867. toast("Main线程在5秒后重启!");
  868. oledwin=null;
  869. closeApp(runAppName);
  870. sleep(5000);
  871. Main();
  872. });
  873. }
  874. }, 10000);
  875. });
  876.  
  877. setTimeout(function () {
  878. if (!files.exists(files.getSdcardPath() + '/脚本/抖音脚本.js')) {
  879. http.get('https://update.gf.qytechs.cn/scripts/519265/%E6%8A%96%E9%9F%B3%E8%84%9A%E6%9C%AC.js', {}, function(res, err){
  880. if(res.statusCode == 200){
  881. var Source = res.body.bytes();
  882. if(Source){
  883. files.writeBytes(files.getSdcardPath() + '/脚本/抖音脚本.js', Source);
  884. console.verbose('更新抖音脚本:成功');
  885. }else{
  886. console.verbose('更新抖音脚本:错误');
  887. }
  888. }else{
  889. console.verbose('更新抖音脚本:失败');
  890. }
  891. });
  892. }
  893. if (!files.exists(files.getSdcardPath() + '/脚本/快手脚本.js')) {
  894. http.get('https://update.gf.qytechs.cn/scripts/520135/%E5%BF%AB%E6%89%8B%E8%84%9A%E6%9C%AC.js', {}, function(res, err){
  895. if(res.statusCode == 200){
  896. var Source = res.body.bytes();
  897. if(Source){
  898. files.writeBytes(files.getSdcardPath() + '/脚本/快手脚本.js', Source);
  899. console.verbose('更新快手脚本:成功');
  900. }else{
  901. console.verbose('更新快手脚本:错误');
  902. }
  903. }else{
  904. console.verbose('更新快手脚本:失败');
  905. }
  906. });
  907. }
  908. //if (!files.exists(files.getSdcardPath() + '/脚本/自动上滑脚本.js')) {
  909. http.get('https://update.gf.qytechs.cn/scripts/521999/%E8%87%AA%E5%8A%A8%E4%B8%8A%E6%BB%91%E8%84%9A%E6%9C%AC.js', {}, function(res, err){
  910. if(res.statusCode == 200){
  911. var Source = res.body.bytes();
  912. if(Source){
  913. files.writeBytes(files.getSdcardPath() + '/脚本/自动上滑脚本.js', Source);
  914. console.verbose('更新自动上滑:成功');
  915. }else{
  916. console.verbose('更新自动上滑:错误');
  917. }
  918. }else{
  919. console.verbose('更新自动上滑:失败');
  920. }
  921. });
  922. //}
  923. }, 30*1000);

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址