钉钉全自动打卡

声明

本篇文章为摘录自GitHub,自己修改了部分代码,仅作为参考,详见原文

⏰ 定时脚本

同时兼容定时和通知两种功能

  • 发送 “开启循环” 开启定时功能
  • 发送 “关闭循环” 关闭定时功能
  • 修正 “日志” 功能BUG,发送日志文件到邮箱中
  • 添加 “连接” 功能,实现远程连接电脑 AutoJs Server
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
/*
* @Author: George Huan
* @Date: 2020-08-03 09:30:30
* @LastEditTime: 2022-03-26 10:56:25
* @Description: DingDing-Automatic-Clock-in (Run on AutoJs)
* @URL: https://github.com/georgehuan1994/DingDing-Automatic-Clock-in
*/

const ACCOUNT = "15868141080"
const PASSWORD = ""

const QQ = "2757412961"
const EMAILL_ADDRESS = "2757412961@qq.com"
const SERVER_CHAN = "SCT171904TfSMLDvHz2oGuB8NU9NdMg1W0"
const PUSH_DEER = "PushDeer发送密钥"

const PUSH_METHOD = {QQ: 1, Email: 2, ServerChan: 3, PushDeer: 4}

// 默认通信方式:
// PUSH_METHOD.QQ -- QQ
// PUSH_METHOD.Email -- Email
// PUSH_METHOD.ServerChan -- Server酱
// PUSH_METHOD.PushDeer -- Push Deer
var DEFAULT_MESSAGE_DELIVER = PUSH_METHOD.QQ;

const PACKAGE_ID_QQ = "com.tencent.mobileqq" // QQ
// const PACKAGE_ID_DD = "com.alibaba.android.rimet" // 钉钉
const PACKAGE_ID_DD = "com.alibaba.android.rimet.zju" // 浙大钉
const PACKAGE_ID_XMSF = "com.xiaomi.xmsf" // 小米推送服务
const PACKAGE_ID_TASKER = "net.dinglisch.android.taskerm" // Tasker
const PACKAGE_ID_MAIL_163 = "com.netease.mail" // 网易邮箱大师
const PACKAGE_ID_MAIL_ANDROID = "com.android.email" // 系统内置邮箱
const PACKAGE_ID_PUSHDEER = "com.pushdeer.os" // Push Deer

const LOWER_BOUND = 1 * 60 * 1000 // 最小等待时间:1min
const UPPER_BOUND = 5 * 60 * 1000 // 最大等待时间:5min
const ONE_MIN = 1 * 60 * 1000 // 等待时间:1min
const FIF_MIN = 15 * 60 * 1000 // 等待时间:0.5h
const HALF_AN_HOUR = 30 * 60 * 1000 // 等待时间:0.5h
const AN_HOUR = 1 * 60 * 60 * 1000 // 等待时间:1h

// 执行时的屏幕亮度(0-255), 需要"修改系统设置"权限
const SCREEN_BRIGHTNESS = 20

// 是否过滤通知
const NOTIFICATIONS_FILTER = true

// PackageId白名单
const PACKAGE_ID_WHITE_LIST = [PACKAGE_ID_QQ,PACKAGE_ID_DD,PACKAGE_ID_XMSF,PACKAGE_ID_MAIL_163,PACKAGE_ID_TASKER,PACKAGE_ID_PUSHDEER]

// 公司的钉钉CorpId, 获取方法见 2020-09-24 更新日志。如果只加入了一家公司, 可以不填
// dingtalk://dingtalkclient/page/link?url='https://attend.dingtalk.com/attend/index.html?corpId=dingca07f56daee3def1bc961a6cb783455b'
const CORP_ID = "dingca07f56daee3def1bc961a6cb783455b"

// 锁屏意图, 配合 Tasker 完成锁屏动作, 具体配置方法见 2021-03-09 更新日志
const ACTION_LOCK_SCREEN = "autojs.intent.action.LOCK_SCREEN"

// 监听音量+键, 开启后无法通过音量+键调整音量, 按下音量+键:结束所有子线程
const OBSERVE_VOLUME_KEY = true

const WEEK_DAY = ["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday",]


// =================== ↓↓↓ 主线程:监听通知 ↓↓↓ ====================

var currentDate = new Date()

// 是否暂停循环打卡
var recursiveCard = false
var timeAnchorPoints = [8,11,13,17,19,22]

// 是否暂停定时打卡
var suspend = false

// 本次打开钉钉前是否需要等待
var needWaiting = false

// 运行日志路径
var globalLogFilePath = "/sdcard/脚本/Archive/" + getCurrentDate() + "-log.txt"

// 检查无障碍权限
auto.waitFor("normal")

// 检查Autojs版本
requiresAutojsVersion("4.1.0")

// 创建运行日志
console.setGlobalLogConfig({
file: "/sdcard/脚本/Archive/" + getCurrentDate() + "-log.txt"
});

// 监听本机通知
events.observeNotification()
events.on("notification", function(n) {
notificationHandler(n)
});

events.setKeyInterceptionEnabled("volume_up", OBSERVE_VOLUME_KEY)

if (OBSERVE_VOLUME_KEY) {
events.observeKey()
};

// 监听音量+键
events.onKeyDown("volume_up", function(event){
threads.shutDownAll()
device.setBrightnessMode(1)
device.cancelKeepingAwake()
toast("已中断所有子线程!")

// 可以在此调试各个方法
// doClock()
// sendQQMsg("测试文本")
// sendEmail("测试主题", "测试文本", null)
// sendServerChan(测试主题, 测试文本)
// sendPushDeer(测试主题, 测试文本)
});

toastLog("监听中, 请在日志中查看记录的通知及其内容")

// =================== ↑↑↑ 主线程:监听通知 ↑↑↑ =====================


// =================== ↑↑↑ 副线程:循环打卡 ↑↑↑ =====================
startRecursiveCard();

function startRecursiveCard(){
while(recursiveCard){
var hour = new Date().getHours()
// 直到打卡时间点 启动钉钉
while(recursiveCard && !existsInArray(timeAnchorPoints, hour)){
var randomTime = random(ONE_MIN, FIF_MIN)
console.log("【循环打卡】未到时间点,进行" + Math.floor(randomTime / 1000) + "秒随机睡眠...")
sleep(randomTime)
hour = new Date().getHours()
}

doClock(); // 打卡
console.log(new Date() + " 【循环打卡】打卡成功")
if(DEFAULT_MESSAGE_DELIVER == PUSH_METHOD.QQ){
sleep(10000) // 等待默认消息通知程序发送
sendQQMsg(new Date() + " 【循环打卡】打卡成功")
sendServerChan("【循环打卡】打卡结果", new Date() + " 打卡成功")
console.log("【循环打卡】QQ&ServerChan 消息发送成功...")
}

// 直到打卡时间点 启动钉钉
while(recursiveCard && existsInArray(timeAnchorPoints, hour)){
var randomTime = random(ONE_MIN, FIF_MIN)
console.log("【循环打卡】已打卡,进行" + Math.floor(randomTime / 1000) + "秒随机睡眠...")
sleep(randomTime)
hour = new Date().getHours()
}
};
}
// =================== ↑↑↑ 副线程:循环打卡 ↑↑↑ =====================

/**
* @description 处理通知
*/
function notificationHandler(n) {

var packageId = n.getPackageName() // 获取通知包名
var abstract = n.tickerText // 获取通知摘要
var text = n.getText() // 获取通知文本

// 过滤 PackageId 白名单之外的应用所发出的通知
if (!filterNotification(packageId, abstract, text)) {
return;
}

// 监听摘要为 "定时打卡" 的通知, 不一定要从 Tasker 中发出通知, 日历、定时器等App均可实现
if (abstract == "定时打卡" && !suspend) {
needWaiting = true
threads.shutDownAll()
threads.start(function(){
doClock()
})
return;
}

switch(text) {

case "打卡": // 监听文本为 "打卡" 的通知
needWaiting = false
// threads.shutDownAll()
threads.start(function(){
doClock()
})
break;

case "查询": // 监听文本为 "查询" 的通知
// threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg(getStorageData("dingding", "clockResult"))
break;
case PUSH_METHOD.Email:
sendEmail("考勤结果", getStorageData("dingding", "clockResult"), null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("考勤结果", getStorageData("dingding", "clockResult"))
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("考勤结果", getStorageData("dingding", "clockResult"))
break;
}
})
break;

case "暂停": // 监听文本为 "暂停" 的通知
suspend = true
console.warn("暂停定时打卡")
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg("修改成功, 已暂停定时打卡功能")
break;
case PUSH_METHOD.Email:
sendEmail("修改成功", "已暂停定时打卡功能", null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("修改成功", "已暂停定时打卡功能")
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("修改成功", "已暂停定时打卡功能")
break;
}
})
break;

case "恢复": // 监听文本为 "恢复" 的通知
suspend = false
console.warn("恢复定时打卡")
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg("修改成功, 已恢复定时打卡功能")
break;
case PUSH_METHOD.Email:
sendEmail("修改成功", "已恢复定时打卡功能", null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("修改成功", "已恢复定时打卡功能")
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("修改成功", "已恢复定时打卡功能")
break;
}
})
break;

case "日志": // 监听文本为 "日志" 的通知
// threads.shutDownAll()
threads.start(function(){
sendEmail("获取日志", globalLogFilePath, globalLogFilePath)
})
break;


case "连接": // 连接到 Autojs Server
// threads.shutDownAll()
threads.start(function(){
connectServer()
})
break;

case "开启循环": // 每天循环打卡 早八晚十
recursiveCard = true
console.warn("每天循环打卡")
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg("开启每天循环打卡功能(早八晚十),时间点:" + timeAnchorPoints)
break;
case PUSH_METHOD.Email:
sendEmail("开启每天循环打卡功能(早八晚十),时间点:" + timeAnchorPoints, null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("开启循环打卡成功", "开启每天循环打卡功能(早八晚十),时间点:" + timeAnchorPoints)
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("开启循环打卡成功", "开启每天循环打卡功能(早八晚十),时间点:" + timeAnchorPoints)
break;
}
startRecursiveCard();
})
break;

case "关闭循环": // 每天循环打卡 早八晚十
recursiveCard = false
console.warn("关闭每天循环打卡")
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg("关闭每天循环打卡功能")
break;
case PUSH_METHOD.Email:
sendEmail("关闭每天循环打卡功能", null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("关闭循环打卡成功", "关闭每天循环打卡功能")
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("关闭循环打卡成功", "关闭每天循环打卡功能")
break;
}
})
break;

default:
break;
}

if (text == null)
return;

// 监听钉钉返回的考勤结果
// if (packageId == PACKAGE_ID_DD && text.indexOf("考勤打卡") >= 0) {
if (packageId == PACKAGE_ID_DD) {
setStorageData("dingding", "clockResult", text)
console.warn("监听钉钉返回的消息...")
// threads.shutDownAll()
threads.start(function() {
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg(text)
break;
case PUSH_METHOD.Email:
sendEmail("考勤结果", text, cameraFilePath)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("考勤结果", text)
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("考勤结果", text)
break;
}
})
return;
}
}


/**
* @description 打卡流程
*/
function doClock() {

currentDate = new Date()
console.log("本地时间: " + getCurrentDate() + " " + getCurrentTime())
console.log("开始打卡流程!")

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕
holdOn() // 随机等待
signIn() // 自动登录
handleLate() // 处理迟到
attendKaoqin() // 考勤打卡

if (null != textContains("上班打卡").findOne(1000))
clockIn() // 上班打卡
else if (null != textContains("下班打卡").findOne(1000))
clockOut() // 下班打卡

lockScreen() // 关闭屏幕
}


/**
* @description 发送邮件流程
* @param {string} title 邮件主题
* @param {string} message 邮件正文
* @param {string} attachFilePath 要发送的附件路径
*/
function sendEmail(title, message, attachFilePath) {

console.log("开始发送邮件流程!")

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕


// 内置电子邮件
app.launch("com.android.email");
console.log("等待邮箱启动...")
sleep(3000) // 等待邮箱启动

if(attachFilePath != null && files.exists(attachFilePath)) {
console.info(attachFilePath)
app.sendEmail({
email: [EMAILL_ADDRESS], subject: title, text: message, attachment: "file://" + attachFilePath
})
}
else {
console.error(attachFilePath)
app.sendEmail({
email: [EMAILL_ADDRESS], subject: title, text: message
})
}
sleep(1000)

// 选择邮箱应用,系统默认应用即可
var emailText = text("电子邮件").findOne(1000);
if(null == emailText){
console.log("邮箱应用选择失败...")
return;
}
emailText.parent().click();
console.log("邮箱应用选择成功...")
sleep(1000)

// 点击发送按钮
var sendBtn = desc("发送").findOne(1000);
if(null == sendBtn){
console.log("邮箱发送失败...")
return;
}
sendBtn.click()
console.log("正在发送邮件...")

// =============================================================================================================
// console.log("选择邮件应用")
// waitForActivity("com.android.internal.app.ChooserActivity") // 等待选择应用界面弹窗出现, 如果设置了默认应用就注释掉
// id("compose_send_btn").findOne().click()

// var emailAppName = app.getAppName(PACKAGE_ID_MAIL_163)
// if (null != emailAppName) {
// if (null != textMatches(emailAppName).findOne(1000)) {
// btn_email = textMatches(emailAppName).findOnce().parent()
// btn_email.click()
// }
// }
// else {
// console.error("不存在应用: " + PACKAGE_ID_MAIL_163)
// lockScreen()
// return;
// }

// // 网易邮箱大师
// var versoin = getPackageVersion(PACKAGE_ID_MAIL_163)
// console.log("应用版本: " + versoin)
// var sp = versoin.split(".")
// if (sp[0] == 6) {
// // 网易邮箱大师 6
// waitForActivity("com.netease.mobimail.activity.MailComposeActivity")
// id("send").findOne().click()
// }
// else {
// // 网易邮箱大师 7
// waitForActivity("com.netease.mobimail.module.mailcompose.MailComposeActivity")
// var input_address = id("input").findOne()
// if (null == input_address.getText()) {
// input_address.setText(EMAILL_ADDRESS)
// }
// id("iv_arrow").findOne().click()
// sleep(1000)
// id("img_send_bg").findOne().click()
// }


home()
sleep(2000)
lockScreen() // 关闭屏幕
}


/**
* @description 发送QQ消息
* @param {string} message 消息内容
*/
function sendQQMsg(message) {

console.log("发送QQ消息")

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕

app.startActivity({
action: "android.intent.action.VIEW",
data:"mqq://im/chat?chat_type=wpa&version=1&src_type=web&uin=" + QQ,
packageName: "com.tencent.mobileqq",
});

// waitForActivity("com.tencent.mobileqq.activity.SplashActivity")

id("input").findOne().setText(message)
id("fun_btn").findOne().click()

home()
sleep(1000)
lockScreen() // 关闭屏幕
}


/**
* @description ServerChan推送
* @param {string} title 标题
* @param {string} message 消息
*/
function sendServerChan(title, message) {

console.log("向 ServerChan 发起推送请求")

url = "https://sctapi.ftqq.com/" + SERVER_CHAN + ".send";

res = http.post(encodeURI(url), {
"title": title,
"desp": message
});

console.log(res)
sleep(1000)
lockScreen() // 关闭屏幕
}


/**
* @description PushDeer推送
* @param {string} title 标题
* @param {string} message 消息
*/
function sendPushDeer(title, message) {

console.log("向 PushDeer 发起推送请求")

url = "https://api2.pushdeer.com/message/push"

res = http.post(encodeURI(url), {
"pushkey": PUSH_DEER,
"text": title,
"desp": message,
"type": "markdown",
});

console.log(res)
sleep(1000)
lockScreen() // 关闭屏幕
}


/**
* @description 唤醒设备
*/
function brightScreen() {

console.log("唤醒设备")

device.setBrightnessMode(0) // 手动亮度模式
device.setBrightness(SCREEN_BRIGHTNESS)
device.wakeUpIfNeeded() // 唤醒设备
device.keepScreenOn() // 保持亮屏
sleep(1000) // 等待屏幕亮起

if (!device.isScreenOn()) {
console.warn("设备未唤醒, 重试")
device.wakeUpIfNeeded()
brightScreen()
}
else {
console.info("设备已唤醒")
}
sleep(1000)
}


/**
* @description 解锁屏幕
*/
function unlockScreen() {

console.log("解锁屏幕")

if (isDeviceLocked()) {

gesture(
320, // 滑动时间:毫秒
[
device.width * 0.5, // 滑动起点 x 坐标:屏幕宽度的一半
device.height * 0.9 // 滑动起点 y 坐标:距离屏幕底部 10% 的位置, 华为系统需要往上一些
],
[
device.width / 2, // 滑动终点 x 坐标:屏幕宽度的一半
device.height * 0.1 // 滑动终点 y 坐标:距离屏幕顶部 10% 的位置
]
)

sleep(1000) // 等待解锁动画完成
home()
sleep(1000) // 等待返回动画完成
}

if (isDeviceLocked()) {
console.error("上滑解锁失败, 请按脚本中的注释调整 gesture(time, [x1,y1], [x2,y2]) 方法的参数!")
return;
}
console.info("屏幕已解锁")
}


/**
* @description 随机等待
*/
function holdOn(){

if (!needWaiting) {
return;
}

var randomTime = random(LOWER_BOUND, UPPER_BOUND)
toastLog(Math.floor(randomTime / 1000) + "秒后启动" + app.getAppName(PACKAGE_ID_DD) + "...")
sleep(randomTime)
}


/**
* @description 启动并登陆钉钉
*/
function signIn() {

app.launchPackage(PACKAGE_ID_DD)
console.log("正在启动" + app.getAppName(PACKAGE_ID_DD) + "...")

setVolume(0) // 设备静音

sleep(10000) // 等待钉钉启动

if (currentPackage() == PACKAGE_ID_DD &&
currentActivity() == "com.alibaba.android.user.login.SignUpWithPwdActivity") {
console.info("账号未登录")

var account = id("et_phone_input").findOne()
account.setText(ACCOUNT)
console.log("输入账号")

var password = id("et_pwd_login").findOne()
password.setText(PASSWORD)
console.log("输入密码")

var privacy = id("cb_privacy").findOne(1000)
if(null != privacy){
privacy.click()
console.log("同意隐私协议")
}

var btn_login = id("btn_next").findOne()
btn_login.click()
console.log("正在登陆...")

sleep(3000)
}

if (currentPackage() == PACKAGE_ID_DD &&
currentActivity() != "com.alibaba.android.user.login.SignUpWithPwdActivity") {
console.info("账号已登录")
sleep(1000)
}
}


/**
* @description 处理迟到打卡
*/
function handleLate(){

if (null != textMatches("迟到打卡").clickable(true).findOne(1000)) {
btn_late = textMatches("迟到打卡").clickable(true).findOnce()
btn_late.click()
console.warn("迟到打卡")
}
if (null != descMatches("迟到打卡").clickable(true).findOne(1000)) {
btn_late = descMatches("迟到打卡").clickable(true).findOnce()
btn_late.click()
console.warn("迟到打卡")
}
}


/**
* @description 使用 URL Scheme 进入考勤界面
*/
function attendKaoqin(){

var url_scheme = "dingtalk://dingtalkclient/page/link?url=https://attend.dingtalk.com/attend/index.html"

if(CORP_ID != "") {
url_scheme = url_scheme + "?corpId=" + CORP_ID
}

var a = app.intent({
// action: "VIEW",
action: "android.intent.action.VIEW",
data: url_scheme,
//flags: [Intent.FLAG_ACTIVITY_NEW_TASK]
});
app.startActivity(a);
console.log("正在进入考勤界面...")

textContains("申请").waitFor()
console.info("已进入考勤界面")
sleep(8000)
}


/**
* @description 上班打卡
*/
function clockIn() {

console.log("上班打卡...")

// if (null != textContains("已打卡").findOne(1000)) {
// console.info("已打卡")
// toast("已打卡")
// home()
// sleep(1000)
// return;
// }

console.log("等待连接到考勤机...")
sleep(2000)

if (null != textContains("未连接").findOne(1000)) {
console.error("未连接考勤机, 重新进入考勤界面!")
back()
sleep(2000)
attendKaoqin()
return;
}

textContains("已连接").waitFor()
console.info("已连接考勤机")
sleep(1000)

if (null != textMatches("上班打卡").clickable(true).findOne(1000)) {
btn_clockin = textMatches("上班打卡").clickable(true).findOnce()
btn_clockin.click()
console.log("按下打卡按钮")
}
else {
click(device.width / 2, device.height * 0.560)
console.log("点击打卡按钮坐标")
}
sleep(1000)
handleLate() // 处理迟到打卡

home()
sleep(1000)
}


/**
* @description 下班打卡
*/
function clockOut() {

console.log("下班打卡...")
console.log("等待连接到考勤机...")
sleep(2000)

if (null != textContains("未连接").findOne(1000)) {
console.error("未连接考勤机, 重新进入考勤界面!")
back()
sleep(2000)
attendKaoqin()
return;
}

textContains("已连接").waitFor()
console.info("已连接考勤机")
sleep(1000)

if (null != textMatches("下班打卡").clickable(true).findOne(1000)) {
btn_clockout = textMatches("下班打卡").clickable(true).findOnce()
btn_clockout.click()
console.log("按下打卡按钮")
sleep(1000)
}
else {
click(device.width / 2, device.height * 0.560)
console.log("点击打卡按钮坐标")
}

if (null != textContains("早退打卡").clickable(true).findOne(1000)) {
className("android.widget.Button").text("早退打卡").clickable(true).findOnce().parent().click()
console.warn("早退打卡")
}

home()
sleep(1000)
}


/**
* @description 锁屏
*/
function lockScreen(){

console.log("关闭屏幕")

// 锁屏方案1:Root
// Power()

// 锁屏方案2:No Root
// press(Math.floor(device.width / 2), Math.floor(device.height * 0.973), 1000) // 小米的快捷手势:长按Home键锁屏

// 万能锁屏方案:向Tasker发送广播, 触发系统锁屏动作。配置方法见 2021-03-09 更新日志
app.sendBroadcast({action: ACTION_LOCK_SCREEN});

device.setBrightnessMode(1) // 自动亮度模式
device.cancelKeepingAwake() // 取消设备常亮

if (isDeviceLocked()) {
console.info("屏幕已关闭")
}
else {
console.error("屏幕未关闭, 请尝试其他锁屏方案, 或等待屏幕自动关闭")
}
}


/**
* @description 连接 AutoJs Server
*/
function connectServer(){

console.log("开始连接 AutoJs Server!")
console.log("本地时间: " + getCurrentDate() + " " + getCurrentTime())

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕

connectAutoJsServer() // 连接 AutoJs Server

lockScreen() // 关闭屏幕

}

/**
*
* @description 启动 AutoJs Server 整体流程
*/
function connectAutoJsServer(){

launchApp("Auto.js")
console.log("正在启动 Auto.js...")
sleep(10000) // 等待 Auto.js 启动

// 打开侧拉菜单
var menuBtn = className("android.widget.ImageButton").desc("打开侧拉菜单").clickable(true).findOne(1000);
if(null != menuBtn){
menuBtn.click();
}
console.log("打开侧拉菜单...")
sleep(1000)

// 滚动屏幕找到连接电脑按钮
var menu = id("drawer_menu").findOne(1000);
if(null == menu){
console.log("滚动屏幕未找到连接电脑按钮...")
return;
}

menu.scrollForward()
console.log("滚动屏幕找到连接电脑按钮...")
sleep(1000)

// 连接电脑
var textCon = className("android.widget.TextView").text("连接电脑").findOne(1000);
if(null != textCon){
// 寻找按钮(Switch)
var layerCon = textCon.parent();
var switchBtn = layerCon.children().findOne(className("android.widget.Switch"));
console.log("找到连接电脑按钮,按钮状态为:" + switchBtn.text() + " " + switchBtn.checked() + "...")
if(switchBtn.checked() == false){
// 按下按钮
switchBtn.click();
console.log("开始连接电脑...")
sleep(2000)

// 点击确定
var okBtn = text("确定").findOne(1000);
okBtn.click();
console.log("连接电脑成功...")
}
}

sleep(1000) // 等待解锁动画完成
home()
sleep(1000) // 等待返回动画完成

// console.log("==========================================================");
// id("drawer_menu").findOne().children().forEach(child => {
// var target = child.findOne(id("sw"));
// console.log(target);
// });
}


// ===================== ↓↓↓ 功能函数 ↓↓↓ =======================

function dateDigitToString(num){
return num < 10 ? '0' + num : num
}

function getCurrentTime(){
var currentDate = new Date()
var hours = dateDigitToString(currentDate.getHours())
var minute = dateDigitToString(currentDate.getMinutes())
var second = dateDigitToString(currentDate.getSeconds())
var formattedTimeString = hours + ':' + minute + ':' + second
return formattedTimeString
}

function getCurrentDate(){
var currentDate = new Date()
var year = dateDigitToString(currentDate.getFullYear())
var month = dateDigitToString(currentDate.getMonth() + 1)
var date = dateDigitToString(currentDate.getDate())
var week = currentDate.getDay()
var formattedDateString = year + '-' + month + '-' + date + '-' + WEEK_DAY[week]
return formattedDateString
}

function existsInArray(arr, element){
for(var i=0; i<arr.length; i++){
if(arr[i] == element) return true;
}
return false;
}

// 通知过滤器
function filterNotification(bundleId, abstract, text) {
var check = PACKAGE_ID_WHITE_LIST.some(function(item) {return bundleId == item})
if (!NOTIFICATIONS_FILTER || check) {
console.verbose(bundleId)
console.verbose(abstract)
console.verbose(text)
console.verbose("---------------------------")
return true
}
else {
return false
}
}

// 保存本地数据
function setStorageData(name, key, value) {
const storage = storages.create(name) // 创建storage对象
storage.put(key, value)
}

// 读取本地数据
function getStorageData(name, key) {
const storage = storages.create(name)
if (storage.contains(key)) {
return storage.get(key, "")
}
// 默认返回undefined
}

// 删除本地数据
function delStorageData(name, key) {
const storage = storages.create(name)
if (storage.contains(key)) {
storage.remove(key)
}
}

// 获取应用版本号
function getPackageVersion(bundleId) {
importPackage(android.content)
var pckMan = context.getPackageManager()
var packageInfo = pckMan.getPackageInfo(bundleId, 0)
return packageInfo.versionName
}

// 屏幕是否为锁定状态
function isDeviceLocked() {
importClass(android.app.KeyguardManager)
importClass(android.content.Context)
var km = context.getSystemService(Context.KEYGUARD_SERVICE)
return km.isKeyguardLocked()
}

// 设置媒体和通知音量
function setVolume(volume) {
device.setMusicVolume(volume)
device.setNotificationVolume(volume)
console.verbose("媒体音量:" + device.getMusicVolume())
console.verbose("通知音量:" + device.getNotificationVolume())
}

Auto.js-VSCodeExt README

桌面编辑器Visual Studio Code的插件。可以让Visual Studio Code支持Auto.js开发。

Install

在VS Code中菜单”查看”->”扩展”->输入”Auto.js”或”hyb1996”搜索,即可看到”Auto.js-VSCodeExt”插件,安装即可。插件的更新也可以在这里更新。

Features

目前功能比较基础,仅支持:

  • 在VS Code的开发者工具实时显示Auto.js的日志与输出
  • 在VS Code命令中增加Run, Stop, Rerun, Stop all等选项。可以在手机与电脑连接后把Sublime编辑器中的脚本推送到AutoJs中执行,或者停止AutoJs中运行的脚本。

Usage

Step 1

Ctrl+Shift+P 或点击”查看”->”命令面板”可调出命令面板,输入 Auto.js 可以看到几个命令,移动光标到命令Auto.js: Start Server,按回车键执行该命令。

此时VS Code会在右上角显示”Auto.js server running”,即开启服务成功。

Step 2

将手机连接到电脑启用的Wifi或者同一局域网中。通过命令行ipconfig(或者其他操作系统的相同功能命令)查看电脑的IP地址。在Auto.js的侧拉菜单中启用调试服务,并输入IP地址,等待连接成功。

Step 3

之后就可以在电脑上编辑JavaScript文件并通过命令Run或者按键F5在手机上运行了。

Commands

Ctrl+Shift+P 或点击”查看”->”命令面板”可调出命令面板,输入 Auto.js 可以看到几个命令:

  • Start Server: 启动插件服务。之后在确保手机和电脑在同一区域网的情况下,在Auto.js的侧拉菜单中使用连接电脑功能连接。
  • Stop Server: 停止插件服务。
  • Run 运行当前编辑器的脚本。如果有多个设备连接,则在所有设备运行。
  • Rerun 停止当前文件对应的脚本并重新运行。如果有多个设备连接,则在所有设备重新运行。
  • Stop 停止当前文件对应的脚本。如果有多个设备连接,则在所有设备停止。
  • StopAll 停止所有正在运行的脚本。如果有多个设备连接,则在所有设备运行所有脚本。
  • Save 保存当前文件到手机的脚本默认目录(文件名会加上前缀remote)。如果有多个设备连接,则在所有设备保存。
  • RunOnDevice: 弹出设备菜单并在指定设备运行脚本。
  • SaveToDevice: 弹出设备菜单并在指定设备保存脚本。
  • New Project(新建项目):选择一个空文件夹(或者在文件管理器中新建一个空文件夹),将会自动创建一个项目
  • Run Project(运行项目):运行一个项目,需要Auto.js 4.0.4Alpha5以上支持
  • Save Project(保存项目):保存一个项目,需要Auto.js 4.0.4Alpha5以上支持

以上命令一些有对应的快捷键,参照命令后面的说明即可。

Log

要显示来自Auto.js的日志,打开 VS Code上面菜单的”帮助”->”切换开发人员工具”->”Console”即可。

钉钉打卡脚本原文

⏰ DingDing-Automatic-Clock-in

📖 简介

钉钉全自动打卡 + 远程打卡脚本,无需 root,基于 auto.js,适用于蓝牙考勤机。

💥 功能

  • 定时打卡
  • 远程打卡
  • 发送考勤结果

⚙️ 工具

  • auto.js
  • Tasker
  • 一款通讯应用(示例脚本中使用的是 QQ / 网易邮箱大师 / ServerChan / PushDeer,彼此互为备用方案)

💡 原理

通过 auto.js 脚本监听本机通知,在 Tasker 中创建定时任务,发出通知,或在另一设备上发送消息到本机,即可触发脚本中的打卡进程,实现定时打卡和远程打卡。

image

同理,监听到钉钉发出的打卡成功通知后,将通知文本通过 QQ消息 或 邮件正文 发送,实现发送考勤结果的功能。

📝 脚本

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
/*
* @Author: George Huan
* @Date: 2020-08-03 09:30:30
* @LastEditTime: 2022-03-26 10:56:25
* @Description: DingDing-Automatic-Clock-in (Run on AutoJs)
* @URL: https://github.com/georgehuan1994/DingDing-Automatic-Clock-in
*/

const ACCOUNT = "钉钉账号"
const PASSWORD = "钉钉密码"

const QQ = "用于接收打卡结果的QQ号"
const EMAILL_ADDRESS = "用于接收打卡结果的邮箱地址"
const SERVER_CHAN = "Server酱发送密钥"
const PUSH_DEER = "PushDeer发送密钥"

const PUSH_METHOD = {QQ: 1, Email: 2, ServerChan: 3, PushDeer: 4}

// 默认通信方式:
// PUSH_METHOD.QQ -- QQ
// PUSH_METHOD.Email -- Email
// PUSH_METHOD.ServerChan -- Server酱
// PUSH_METHOD.PushDeer -- Push Deer
var DEFAULT_MESSAGE_DELIVER = PUSH_METHOD.QQ;

const PACKAGE_ID_QQ = "com.tencent.mobileqq" // QQ
const PACKAGE_ID_DD = "com.alibaba.android.rimet" // 钉钉
const PACKAGE_ID_XMSF = "com.xiaomi.xmsf" // 小米推送服务
const PACKAGE_ID_TASKER = "net.dinglisch.android.taskerm" // Tasker
const PACKAGE_ID_MAIL_163 = "com.netease.mail" // 网易邮箱大师
const PACKAGE_ID_MAIL_ANDROID = "com.android.email" // 系统内置邮箱
const PACKAGE_ID_PUSHDEER = "com.pushdeer.os" // Push Deer

const LOWER_BOUND = 1 * 60 * 1000 // 最小等待时间:1min
const UPPER_BOUND = 5 * 60 * 1000 // 最大等待时间:5min

// 执行时的屏幕亮度(0-255), 需要"修改系统设置"权限
const SCREEN_BRIGHTNESS = 20

// 是否过滤通知
const NOTIFICATIONS_FILTER = true

// PackageId白名单
const PACKAGE_ID_WHITE_LIST = [PACKAGE_ID_QQ,PACKAGE_ID_DD,PACKAGE_ID_XMSF,PACKAGE_ID_MAIL_163,PACKAGE_ID_TASKER,PACKAGE_ID_PUSHDEER]

// 公司的钉钉CorpId, 获取方法见 2020-09-24 更新日志。如果只加入了一家公司, 可以不填
const CORP_ID = ""

// 锁屏意图, 配合 Tasker 完成锁屏动作, 具体配置方法见 2021-03-09 更新日志
const ACTION_LOCK_SCREEN = "autojs.intent.action.LOCK_SCREEN"

// 监听音量+键, 开启后无法通过音量+键调整音量, 按下音量+键:结束所有子线程
const OBSERVE_VOLUME_KEY = true

const WEEK_DAY = ["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday",]


// =================== ↓↓↓ 主线程:监听通知 ↓↓↓ ====================

var currentDate = new Date()

// 是否暂停定时打卡
var suspend = false

// 本次打开钉钉前是否需要等待
var needWaiting = true

// 运行日志路径
var globalLogFilePath = "/sdcard/脚本/Archive/" + getCurrentDate() + "-log.txt"

// 检查无障碍权限
auto.waitFor("normal")

// 检查Autojs版本
requiresAutojsVersion("4.1.0")

// 创建运行日志
console.setGlobalLogConfig({
file: "/sdcard/脚本/Archive/" + getCurrentDate() + "-log.txt"
});

// 监听本机通知
events.observeNotification()
events.on("notification", function(n) {
notificationHandler(n)
});

events.setKeyInterceptionEnabled("volume_up", OBSERVE_VOLUME_KEY)

if (OBSERVE_VOLUME_KEY) {
events.observeKey()
};

// 监听音量+键
events.onKeyDown("volume_up", function(event){
threads.shutDownAll()
device.setBrightnessMode(1)
device.cancelKeepingAwake()
toast("已中断所有子线程!")

// 可以在此调试各个方法
// doClock()
// sendQQMsg("测试文本")
// sendEmail("测试主题", "测试文本", null)
// sendServerChan(测试主题, 测试文本)
// sendPushDeer(测试主题, 测试文本)
});

toastLog("监听中, 请在日志中查看记录的通知及其内容")

// =================== ↑↑↑ 主线程:监听通知 ↑↑↑ =====================



/**
* @description 处理通知
*/
function notificationHandler(n) {

var packageId = n.getPackageName() // 获取通知包名
var abstract = n.tickerText // 获取通知摘要
var text = n.getText() // 获取通知文本

// 过滤 PackageId 白名单之外的应用所发出的通知
if (!filterNotification(packageId, abstract, text)) {
return;
}

// 监听摘要为 "定时打卡" 的通知, 不一定要从 Tasker 中发出通知, 日历、定时器等App均可实现
if (abstract == "定时打卡" && !suspend) {
needWaiting = true
threads.shutDownAll()
threads.start(function(){
doClock()
})
return;
}

switch(text) {

case "打卡": // 监听文本为 "打卡" 的通知
needWaiting = false
threads.shutDownAll()
threads.start(function(){
doClock()
})
break;

case "查询": // 监听文本为 "查询" 的通知
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg(getStorageData("dingding", "clockResult"))
break;
case PUSH_METHOD.Email:
sendEmail("考勤结果", getStorageData("dingding", "clockResult"), null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("考勤结果", getStorageData("dingding", "clockResult"))
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("考勤结果", getStorageData("dingding", "clockResult"))
break;
}
})
break;

case "暂停": // 监听文本为 "暂停" 的通知
suspend = true
console.warn("暂停定时打卡")
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg("修改成功, 已暂停定时打卡功能")
break;
case PUSH_METHOD.Email:
sendEmail("修改成功", "已暂停定时打卡功能", null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("修改成功", "已暂停定时打卡功能")
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("修改成功", "已暂停定时打卡功能")
break;
}
})
break;

case "恢复": // 监听文本为 "恢复" 的通知
suspend = false
console.warn("恢复定时打卡")
threads.shutDownAll()
threads.start(function(){
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg("修改成功, 已恢复定时打卡功能")
break;
case PUSH_METHOD.Email:
sendEmail("修改成功", "已恢复定时打卡功能", null)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("修改成功", "已恢复定时打卡功能")
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("修改成功", "已恢复定时打卡功能")
break;
}
})
break;

case "日志": // 监听文本为 "日志" 的通知
threads.shutDownAll()
threads.start(function(){
sendEmail("获取日志", globalLogFilePath, globalLogFilePath)
})
break;

default:
break;
}

if (text == null)
return;

// 监听钉钉返回的考勤结果
if (packageId == PACKAGE_ID_DD && text.indexOf("考勤打卡") >= 0) {
setStorageData("dingding", "clockResult", text)
threads.shutDownAll()
threads.start(function() {
switch(DEFAULT_MESSAGE_DELIVER) {
case PUSH_METHOD.QQ:
sendQQMsg(text)
break;
case PUSH_METHOD.Email:
sendEmail("考勤结果", text, cameraFilePath)
break;
case PUSH_METHOD.ServerChan:
sendServerChan("考勤结果", text)
break;
case PUSH_METHOD.PushDeer:
sendPushDeer("考勤结果", text)
break;
}
})
return;
}
}


/**
* @description 打卡流程
*/
function doClock() {

currentDate = new Date()
console.log("本地时间: " + getCurrentDate() + " " + getCurrentTime())
console.log("开始打卡流程!")

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕
holdOn() // 随机等待
signIn() // 自动登录
handleLate() // 处理迟到
attendKaoqin() // 考勤打卡

if (currentDate.getHours() <= 12)
clockIn() // 上班打卡
else
clockOut() // 下班打卡

lockScreen() // 关闭屏幕
}


/**
* @description 发送邮件流程
* @param {string} title 邮件主题
* @param {string} message 邮件正文
* @param {string} attachFilePath 要发送的附件路径
*/
function sendEmail(title, message, attachFilePath) {

console.log("开始发送邮件流程!")

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕

if(attachFilePath != null && files.exists(attachFilePath)) {
console.info(attachFilePath)
app.sendEmail({
email: [EMAILL_ADDRESS], subject: title, text: message, attachment: attachFilePath
})
}
else {
console.error(attachFilePath)
app.sendEmail({
email: [EMAILL_ADDRESS], subject: title, text: message
})
}

console.log("选择邮件应用")
waitForActivity("com.android.internal.app.ChooserActivity") // 等待选择应用界面弹窗出现, 如果设置了默认应用就注释掉

var emailAppName = app.getAppName(PACKAGE_ID_MAIL_163)
if (null != emailAppName) {
if (null != textMatches(emailAppName).findOne(1000)) {
btn_email = textMatches(emailAppName).findOnce().parent()
btn_email.click()
}
}
else {
console.error("不存在应用: " + PACKAGE_ID_MAIL_163)
lockScreen()
return;
}

// 网易邮箱大师
var versoin = getPackageVersion(PACKAGE_ID_MAIL_163)
console.log("应用版本: " + versoin)
var sp = versoin.split(".")
if (sp[0] == 6) {
// 网易邮箱大师 6
waitForActivity("com.netease.mobimail.activity.MailComposeActivity")
id("send").findOne().click()
}
else {
// 网易邮箱大师 7
waitForActivity("com.netease.mobimail.module.mailcompose.MailComposeActivity")
var input_address = id("input").findOne()
if (null == input_address.getText()) {
input_address.setText(EMAILL_ADDRESS)
}
id("iv_arrow").findOne().click()
sleep(1000)
id("img_send_bg").findOne().click()
}

// 内置电子邮件
// waitForActivity("com.kingsoft.mail.compose.ComposeActivity")
// id("compose_send_btn").findOne().click()

console.log("正在发送邮件...")

home()
sleep(2000)
lockScreen() // 关闭屏幕
}


/**
* @description 发送QQ消息
* @param {string} message 消息内容
*/
function sendQQMsg(message) {

console.log("发送QQ消息")

brightScreen() // 唤醒屏幕
unlockScreen() // 解锁屏幕

app.startActivity({
action: "android.intent.action.VIEW",
data:"mqq://im/chat?chat_type=wpa&version=1&src_type=web&uin=" + QQ,
packageName: "com.tencent.mobileqq",
});

// waitForActivity("com.tencent.mobileqq.activity.SplashActivity")

id("input").findOne().setText(message)
id("fun_btn").findOne().click()

home()
sleep(1000)
lockScreen() // 关闭屏幕
}


/**
* @description ServerChan推送
* @param {string} title 标题
* @param {string} message 消息
*/
function sendServerChan(title, message) {

console.log("向 ServerChan 发起推送请求")

url = "https://sctapi.ftqq.com/" + SERVER_CHAN + ".send";

res = http.post(encodeURI(url), {
"title": title,
"desp": message
});

console.log(res)
sleep(1000)
lockScreen() // 关闭屏幕
}


/**
* @description PushDeer推送
* @param {string} title 标题
* @param {string} message 消息
*/
function sendPushDeer(title, message) {

console.log("向 PushDeer 发起推送请求")

url = "https://api2.pushdeer.com/message/push"

res = http.post(encodeURI(url), {
"pushkey": PUSH_DEER,
"text": title,
"desp": message,
"type": "markdown",
});

console.log(res)
sleep(1000)
lockScreen() // 关闭屏幕
}


/**
* @description 唤醒设备
*/
function brightScreen() {

console.log("唤醒设备")

device.setBrightnessMode(0) // 手动亮度模式
device.setBrightness(SCREEN_BRIGHTNESS)
device.wakeUpIfNeeded() // 唤醒设备
device.keepScreenOn() // 保持亮屏
sleep(1000) // 等待屏幕亮起

if (!device.isScreenOn()) {
console.warn("设备未唤醒, 重试")
device.wakeUpIfNeeded()
brightScreen()
}
else {
console.info("设备已唤醒")
}
sleep(1000)
}


/**
* @description 解锁屏幕
*/
function unlockScreen() {

console.log("解锁屏幕")

if (isDeviceLocked()) {

gesture(
320, // 滑动时间:毫秒
[
device.width * 0.5, // 滑动起点 x 坐标:屏幕宽度的一半
device.height * 0.9 // 滑动起点 y 坐标:距离屏幕底部 10% 的位置, 华为系统需要往上一些
],
[
device.width / 2, // 滑动终点 x 坐标:屏幕宽度的一半
device.height * 0.1 // 滑动终点 y 坐标:距离屏幕顶部 10% 的位置
]
)

sleep(1000) // 等待解锁动画完成
home()
sleep(1000) // 等待返回动画完成
}

if (isDeviceLocked()) {
console.error("上滑解锁失败, 请按脚本中的注释调整 gesture(time, [x1,y1], [x2,y2]) 方法的参数!")
return;
}
console.info("屏幕已解锁")
}


/**
* @description 随机等待
*/
function holdOn(){

if (!needWaiting) {
return;
}

var randomTime = random(LOWER_BOUND, UPPER_BOUND)
toastLog(Math.floor(randomTime / 1000) + "秒后启动" + app.getAppName(PACKAGE_ID_DD) + "...")
sleep(randomTime)
}


/**
* @description 启动并登陆钉钉
*/
function signIn() {

app.launchPackage(PACKAGE_ID_DD)
console.log("正在启动" + app.getAppName(PACKAGE_ID_DD) + "...")

setVolume(0) // 设备静音

sleep(10000) // 等待钉钉启动

if (currentPackage() == PACKAGE_ID_DD &&
currentActivity() == "com.alibaba.android.user.login.SignUpWithPwdActivity") {
console.info("账号未登录")

var account = id("et_phone_input").findOne()
account.setText(ACCOUNT)
console.log("输入账号")

var password = id("et_pwd_login").findOne()
password.setText(PASSWORD)
console.log("输入密码")

var privacy = id("cb_privacy").findOne()
privacy.click()
console.log("同意隐私协议")

var btn_login = id("btn_next").findOne()
btn_login.click()
console.log("正在登陆...")

sleep(3000)
}

if (currentPackage() == PACKAGE_ID_DD &&
currentActivity() != "com.alibaba.android.user.login.SignUpWithPwdActivity") {
console.info("账号已登录")
sleep(1000)
}
}


/**
* @description 处理迟到打卡
*/
function handleLate(){

if (null != textMatches("迟到打卡").clickable(true).findOne(1000)) {
btn_late = textMatches("迟到打卡").clickable(true).findOnce()
btn_late.click()
console.warn("迟到打卡")
}
if (null != descMatches("迟到打卡").clickable(true).findOne(1000)) {
btn_late = descMatches("迟到打卡").clickable(true).findOnce()
btn_late.click()
console.warn("迟到打卡")
}
}


/**
* @description 使用 URL Scheme 进入考勤界面
*/
function attendKaoqin(){

var url_scheme = "dingtalk://dingtalkclient/page/link?url=https://attend.dingtalk.com/attend/index.html"

if(CORP_ID != "") {
url_scheme = url_scheme + "?corpId=" + CORP_ID
}

var a = app.intent({
action: "VIEW",
data: url_scheme,
//flags: [Intent.FLAG_ACTIVITY_NEW_TASK]
});
app.startActivity(a);
console.log("正在进入考勤界面...")

textContains("申请").waitFor()
console.info("已进入考勤界面")
sleep(1000)
}


/**
* @description 上班打卡
*/
function clockIn() {

console.log("上班打卡...")

if (null != textContains("已打卡").findOne(1000)) {
console.info("已打卡")
toast("已打卡")
home()
sleep(1000)
return;
}

console.log("等待连接到考勤机...")
sleep(2000)

if (null != textContains("未连接").findOne(1000)) {
console.error("未连接考勤机, 重新进入考勤界面!")
back()
sleep(2000)
attendKaoqin()
return;
}

textContains("已连接").waitFor()
console.info("已连接考勤机")
sleep(1000)

if (null != textMatches("上班打卡").clickable(true).findOne(1000)) {
btn_clockin = textMatches("上班打卡").clickable(true).findOnce()
btn_clockin.click()
console.log("按下打卡按钮")
}
else {
click(device.width / 2, device.height * 0.560)
console.log("点击打卡按钮坐标")
}
sleep(1000)
handleLate() // 处理迟到打卡

home()
sleep(1000)
}


/**
* @description 下班打卡
*/
function clockOut() {

console.log("下班打卡...")
console.log("等待连接到考勤机...")
sleep(2000)

if (null != textContains("未连接").findOne(1000)) {
console.error("未连接考勤机, 重新进入考勤界面!")
back()
sleep(2000)
attendKaoqin()
return;
}

textContains("已连接").waitFor()
console.info("已连接考勤机")
sleep(1000)

if (null != textMatches("下班打卡").clickable(true).findOne(1000)) {
btn_clockout = textMatches("下班打卡").clickable(true).findOnce()
btn_clockout.click()
console.log("按下打卡按钮")
sleep(1000)
}
else {
click(device.width / 2, device.height * 0.560)
console.log("点击打卡按钮坐标")
}

if (null != textContains("早退打卡").clickable(true).findOne(1000)) {
className("android.widget.Button").text("早退打卡").clickable(true).findOnce().parent().click()
console.warn("早退打卡")
}

home()
sleep(1000)
}


/**
* @description 锁屏
*/
function lockScreen(){

console.log("关闭屏幕")

// 锁屏方案1:Root
// Power()

// 锁屏方案2:No Root
// press(Math.floor(device.width / 2), Math.floor(device.height * 0.973), 1000) // 小米的快捷手势:长按Home键锁屏

// 万能锁屏方案:向Tasker发送广播, 触发系统锁屏动作。配置方法见 2021-03-09 更新日志
app.sendBroadcast({action: ACTION_LOCK_SCREEN});

device.setBrightnessMode(1) // 自动亮度模式
device.cancelKeepingAwake() // 取消设备常亮

if (isDeviceLocked()) {
console.info("屏幕已关闭")
}
else {
console.error("屏幕未关闭, 请尝试其他锁屏方案, 或等待屏幕自动关闭")
}
}



// ===================== ↓↓↓ 功能函数 ↓↓↓ =======================

function dateDigitToString(num){
return num < 10 ? '0' + num : num
}

function getCurrentTime(){
var currentDate = new Date()
var hours = dateDigitToString(currentDate.getHours())
var minute = dateDigitToString(currentDate.getMinutes())
var second = dateDigitToString(currentDate.getSeconds())
var formattedTimeString = hours + ':' + minute + ':' + second
return formattedTimeString
}

function getCurrentDate(){
var currentDate = new Date()
var year = dateDigitToString(currentDate.getFullYear())
var month = dateDigitToString(currentDate.getMonth() + 1)
var date = dateDigitToString(currentDate.getDate())
var week = currentDate.getDay()
var formattedDateString = year + '-' + month + '-' + date + '-' + WEEK_DAY[week]
return formattedDateString
}

// 通知过滤器
function filterNotification(bundleId, abstract, text) {
var check = PACKAGE_ID_WHITE_LIST.some(function(item) {return bundleId == item})
if (!NOTIFICATIONS_FILTER || check) {
console.verbose(bundleId)
console.verbose(abstract)
console.verbose(text)
console.verbose("---------------------------")
return true
}
else {
return false
}
}

// 保存本地数据
function setStorageData(name, key, value) {
const storage = storages.create(name) // 创建storage对象
storage.put(key, value)
}

// 读取本地数据
function getStorageData(name, key) {
const storage = storages.create(name)
if (storage.contains(key)) {
return storage.get(key, "")
}
// 默认返回undefined
}

// 删除本地数据
function delStorageData(name, key) {
const storage = storages.create(name)
if (storage.contains(key)) {
storage.remove(key)
}
}

// 获取应用版本号
function getPackageVersion(bundleId) {
importPackage(android.content)
var pckMan = context.getPackageManager()
var packageInfo = pckMan.getPackageInfo(bundleId, 0)
return packageInfo.versionName
}

// 屏幕是否为锁定状态
function isDeviceLocked() {
importClass(android.app.KeyguardManager)
importClass(android.content.Context)
var km = context.getSystemService(Context.KEYGUARD_SERVICE)
return km.isKeyguardLocked()
}

// 设置媒体和通知音量
function setVolume(volume) {
device.setMusicVolume(volume)
device.setNotificationVolume(volume)
console.verbose("媒体音量:" + device.getMusicVolume())
console.verbose("通知音量:" + device.getNotificationVolume())
}

📐 工具介绍

Auto.js

Auto.js 是利用安卓系统的 「无障碍服务」 实现类似于按键精灵一样,可以通过代码模拟一系列界面动作的辅助工具。

与 「按键精灵」 不同的是,它的模拟动作并不是简单的使用在界面定坐标点来实现,而是找控件来实现的。

免费版:Auto.js 4.1.1a Alpha2-armeabi-v7a-release

github:GitHub - hyb1996/Auto.js

官方文档:首页 - Auto.js

推荐使用VS Code 插件进行调试,调试完成后,还能通过此插件将脚本保存到手机上。

Tasker

Tasker 也是一个安卓自动化神器,与 Auto.js 结合使用可胜任日常工作流。

此处仅提供 Tasker 5.0 及以下的官方原版,原版不含正版验证,使用不受限制:

Tasker.4.9u4m.apk

Tasker.5.0u7m.apk

Tasker 定时打卡配置:

  1. 添加一个 「通知」 操作任务,通知标题修改为 「定时打卡」,通知文字随意,通知优先级设为 1。
  2. 添加两个配置文件,使用日期和时间作为条件,分别在上班前和下班后触发。

你也可以下载配置文件,导入到 Tasker 中使用,方法如下:

  1. 长按 菜单栏-任务,导入”发送通知.tsk.xml”。
  2. 长按 菜单栏-配置文件,导入”上班打卡.prf.xml” 和 “下班打卡.prf.xml”。
  3. 在任务编辑界面左下方有一个三角形的播放按钮,点击即可发送通知,方便调试。

🕹️ 使用方法

远程打卡

  • 向本机的 QQ 发送消息 「打卡」,或回复标题为 「打卡」 的邮件,或向 PushDeer 发送标题为「打卡」 的推送请求,即可触发打卡进程。
  • 向本机的 QQ 发送消息 「查询」,或回复标题为 「查询」 的邮件,或向 PushDeer 发送标题为「查询」 的推送请求,即可查询最新一次打卡结果。

暂停/恢复定时打卡

  • 向本机的 QQ 发送消息 「暂停」,或回复标题为 「暂停」 的邮件,或向 PushDeer 发送标题为「暂停」 的推送请求,即可暂停定时打卡功能(仅暂停定时打卡,不影响远程打卡功能)
  • 向本机的 QQ 发送消息 「恢复」,或回复标题为 「恢复」 的邮件,或向 PushDeer 发送标题为「恢复」 的推送请求,即可恢复定时打卡功能。

⚠️ 注意事项 (必读!!!)

  • AutoJs Pro 版本屏蔽了一些主流应用,如果要使用 QQ 作为回复方式,不要使用 AutoJs Pro 版!
  • 首次启动 AutoJs,需要为其开启无障碍权限。
  • 运行脚本前,请在 AutoJs 菜单栏中(从屏幕左边划出),开启 「通知读取权限」。
  • 若无法通过 app.launchPackage() 方法启动应用,请开启该应用的「自启动」「允许后台弹窗」。
  • AutoJs、Tasker 可息屏运行,需要在系统设置中开启通知亮屏。
  • 为保证 AutoJs、Tasker 进程不被系统清理,可调整它们的电池管理策略、加入管理应用的白名单,为其开启前台服务、添加应用锁…
  • 虽然脚本可执行完整的打卡步骤,但推荐开启钉钉的极速打卡功能,在钉钉启动时即可完成打卡,应把后续的步骤视为极速打卡失败后的保险措施。

📜 更新日志

2022-03-26

  1. 可以通过 PushDeer 接收通知、推送考勤结果

2022-03-01

  1. 可以通过Server酱来推送考勤结果

2021-10-23

  1. 适配网易邮箱大师7.0

2021-09-02

  1. 新增获取日志功能,发送 「日志」,可将运行日志作为邮件附件发送(最好使用内置邮件)
  2. 优化通知过滤器,过滤 Tasker 发出的无效通知

2021-07-07

  1. 登录流程自动同意隐私协议

2021-05-27

  1. 修改了部分常量的命名
  2. 移除了休息日不打卡的判断
  3. 在邮件的基础上,增加QQ作为新的通讯方式。除发送考勤结果需要手动指定应用外,使用QQ向本机发送 「查询、暂停、恢复」 指令,则会用QQ来回复查询或操作结果;使用邮件向本机发送指令则用邮件回复。

2021-05-06

  1. 增加音量上键监听,按下后中断所有子线程,也可以利用回调来进行调试
  2. 不再使用考勤机名称来判断连接状态
  3. 重新进入打卡界面前,先返回上级菜单,以解决顶号登录无法正常连接到考勤机的问题
  4. 启动钉钉时,将媒体音量和通知音量设为0

2021-03-15

  1. 运行时检查Auto.js版本,脚本需要在Auto.js 4.1.0及以上版本中运行
  2. 新增解锁是否成功的判断,若解锁失败则停止运行脚本
  3. 优化 signIn() 方法,使用 bundleId + activity 来判断登录情况
  4. 优化部分控件和信息的获取方式

2021-03-09

  1. 移除 「结束钉钉」、「检查更新」 这个两个过程,使用最近一次监测到的正在运行的应用的包名进行判断

  2. 补充一个万能锁屏方案:向Tasker发送广播,触发Tasker中的系统锁屏操作。

    • 在Tasker中添加一个任务,在任务中添加操作 「系统锁屏(关闭屏幕)」
    • 在Tasker中添加一个事件类型的配置文件,事件类别:系统-收到的意图
    • 在事件操作中填写:autojs.intent.action.LOCK_SCREEN ,保持发送方与接收方的action一致即可
1
2
3
app.sendBroadcast({
action: 'autojs.intent.action.LOCK_SCREEN'
});

2021-02-07

  1. 防止监听事件被耗时操作阻塞。

2021-01-15

  1. 移除 「进入工作台」 以及 「进入考勤打卡界面」 这两个过程
  2. 启动并成功登录钉钉后,直接使用intent拉起考勤打卡界面

2021-01-08

  1. 修复:通知过滤器报错

2020-12-30

  1. 优化:现在可以通过邮件来 暂停/恢复 定时打卡功能,以应对停工停产,或其他需要暂时停止定时打卡的特殊情况

2020-12-04

  1. 优化:打卡过程在子线程中执行,钉钉返回打卡结果后,直接中断子线程,减少无效操作

2020-10-27

  1. 修复:当钉钉的通知文本为null时,indexOf()方法无法正常执行

2020-09-24

  1. 优化:使用URL Scheme直接拉起考勤打卡界面
1
2
3
4
5
6
7
8
function attendKaoqin(){
var a = app.intent({
action: "VIEW",
data: "dingtalk://dingtalkclient/page/link?url=https://attend.dingtalk.com/attend/index.html"
});
app.startActivity(a);
sleep(5000)
}

获取URL的方式如下:

  1. 在PC端找到 「智能工作助理」 联系人
  2. 发送消息 “打卡” ,点击 「立即打卡」
  3. 弹出一个二维码。此二维码就是拉起考勤打卡界面的 URL,用自带的相机或其他应用扫描,并在浏览器中打开,即可获得完整URL
  4. 观察获取到的URL,找到 CorpId=xxxxxxxxxxxxxxxxxxx ,将CorpId的值填写到的脚本开头的CORP_ID这个常量中
  5. 仅使用 dingtalk://dingtalkclient/page/link?url=https://attend.dingtalk.com/attend/index.html,也可以拉起旧版打卡界面,钉钉会自动获取企业的CorpId。如果加入了多个组织,且没有填写CorpId,则在拉起考勤界面时会弹出一个选择组织的对话框。

2020-09-11

  1. 将上次考勤结果储存在本地
  2. 将运行日志储存在本地 /sdcard/脚本/Archive/
  3. 修复在下班极速打卡之后,重复打卡的问题

2020-09-04

  1. 将 “打卡” 与 “发送邮件” 分离成两个过程,打卡完成后,将钉钉返回的考勤结果作为邮件正文发送

2020-09-02

  1. 改为使用 “去打卡” 文本获取按钮。若找不到 “去打卡” 按钮,则直接点击 “考勤打卡” 的屏幕坐标

📢 声明

此仓库及脚本仅供学习交流,欢迎转载。旨在让人们关注996制度的存在和非法性,并尝试改变这种现象。

根据1994年第八届全国人大常委会通过和2018年第十三届全国人大常委会修正的《中华人民共和国劳动法》规定,劳动者每日工作时间不超过8小时,平均每周工作时间不超过44小时,而996工作制每周至少要工作72个小时,远超法律标准,因此996工作制度违反劳动法。

而钉钉却允许企业管理者违反法律,非法排班!

第三十六条 国家实行劳动者每日工作时间不超过八小时、平均每周工作时间不超过四十四小时的工时制度。

第四十一条 用人单位由于生产经营需要,经与工会和劳动者协商后可以延长工作时间,一般每日不得超过一小时;因特殊原因需要延长工作时间的,在保障劳动者身体健康的条件下延长工作时间每日不得超过三小时,但是每月不得超过三十六小时。

第四十四条 有下列情形之一的,用人单位应当按照下列标准支付高于劳动者正常工作时间工资的工资报酬:

(一)安排劳动者延长工作时间的,支付不低于工资的百分之一百五十的工资报酬;
(二)休息日安排劳动者工作又不能安排补休的,支付不低于工资的百分之二百的工资报酬;
(三)法定休假日安排劳动者工作的,支付不低于工资的百分之三百的工资报酬。

第九十条 用人单位违反本法规定,延长劳动者工作时间的,由劳动行政部门给予警告,责令改正,并可以处以罚款。

第九十一条 用人单位有下列侵害劳动者合法权益情形之一的,由劳动行政部门责令支付劳动者的工资报酬、经济补偿,并可以责令支付赔偿金:

(二)拒不支付劳动者延长工作时间工资报酬的;

相关项目:996 薪资计算助手


如果觉得还不错的话,就点击右上角, 给我个Star ⭐️ 鼓励一下我吧~
文章作者: ZJH
文章链接: http://example.com/2022/09/16/脚本/DingDing-Automatic-Clock-in-master/DingDing-Automatic-Clock-in/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 Zany's Blog