Autovisor
智慧树视频课辅助工具,开启挂机摸鱼时代~
新学期必备干货, 建议收藏备用 !!
程序介绍:
这是一个可无人监督的自动化程序,由Python和JavaScript编写而成。相对于纯JS脚本,本程序可有效防止被网页检测。
核心原理是使浏览器模拟用户的点击操作, 不会导致封号等问题
程序功能:
- 可以自动登录
- 自动播放和切换下一集
- 跳过弹窗和弹出的题目
- 自动静音播放视频
- 检测当前学习进度并后台实时更新
- 检测视频是否暂停并续播 (不用担心视频意外暂停了~)
- 根据当前时间自动设置背景颜色(白昼/暗夜)
使用须知:
0.请确保系统为windows (最好有内置Edge浏览器,也可以修改account文件指定为Chorme )
注:默认为Edge,如果 设为Chrome,程序会先自动下载,在此期间请耐心等待;
1.文件夹内有**"account.json"**,请用文本编辑器打开;
2.前两项请输入你的 用户名,密码;第三项指定启动的 浏览器;第四项输入网课的 具体网址,保存后关闭,例如:
3.运行程序,会自动打开浏览器界面,手动完成滑块验证后,进入网课界面后就可以自动刷课啦~ (~ ^-^ ~)
发行版下载:
Github: Releases · CXRunfree/Autovisor (github.com) (留下一个免费的 star吧?)
网盘备用: [蓝奏云] Autovisor-for-windows 密码: 492l
为便于阅读, 源码已放在文末
如有疑问,可以在评论区留言, 每条留言作者都会认真看的 !
常见问题 :
1.为什么会出现这样一个黑框?
---这是程序运行的后台,你可以查看当前运行的状态
2.为什么网页还没加载出来程序就退出了?
---可能是网速太慢,也可能出现偶然异常
不支持功能:
- 由于习惯分机制对倍速检测严格, 程序不提供自动调节倍速功能
碎碎念:
觉得体验还不错? 来给开源项目发电支持一下吧~!
源代码:
import traceback
import os
import time
import json
from selenium import webdriver
from selenium.common import NoSuchWindowException, JavascriptException, TimeoutException
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from json import JSONDecodeError
def add_opts(option):
option.add_argument('--ignore-certificate-errors')
option.add_argument("disable-extensions")
option.add_experimental_option("excludeSwitches", ["enable-logging"])
def driver_init(driver_type):
if driver_type == "Edge":
options = webdriver.EdgeOptions()
add_opts(options)
_driver = webdriver.Edge(options)
elif driver_type == "Chrome":
options = webdriver.ChromeOptions()
add_opts(options)
_driver = webdriver.Chrome(options)
else:
_driver = None
return _driver
# 图片验证功能已弃用
'''
def auto_verify():
# 等待图片加载
WebDriverWait(driver, 80).until(
EC.presence_of_element_located((By.CLASS_NAME, "yidun_bg-img")))
time.sleep(1)
# 获取图片链接
url1 = driver.execute_script(block_js)
url2 = driver.execute_script(bg_js)
# 获取图片二进制数据
block_bytes = requests.get(url1).content
background_bytes = requests.get(url2).content
# 匹配图形位置,手动修订误差
x_offset = verify.slide_match(block_bytes, background_bytes)["target"][0] + 7
# 定位要拖动的元素
draggable_element = driver.find_element(By.XPATH, drag_bar)
# 拖动元素到指定的偏移位置
ActionChains(driver).drag_and_drop_by_offset(draggable_element, x_offset, 0).perform()
'''
def get_progress():
curt = "进度获取中..."
text = WebDriverWait(driver, 5).until(EC.presence_of_element_located((By.CLASS_NAME, "passTime")))
try:
# 最多保留2位小数;不足则输出原形
s = text.get_attribute('style').split(": ")[1][:-1]
if len(s) > 5:
s = s.split(".")
curt = f"{s[0]}.{s[1][:2]}%"
else:
curt = s
finally:
return curt
def check_play():
canvas = WebDriverWait(driver, 20).until(EC.element_to_be_clickable((By.CLASS_NAME, "videoArea")))
ActionChains(driver).move_to_element(canvas).click().perform()
def shut_volume():
driver.execute_script(show_controlsBar)
WebDriverWait(driver, 5).until(EC.element_to_be_clickable((By.CLASS_NAME, "volumeBox"))).click()
def play_next():
driver.execute_script(show_controlsBar)
next_but = WebDriverWait(driver, 5).until(EC.element_to_be_clickable((By.ID, "nextBtn")))
next_but.click()
def get_lesson_name():
title_ = WebDriverWait(driver, 5).until(EC.element_to_be_clickable((By.ID, "lessonOrder"))).get_attribute("title")
return title_
def skip_question():
WebDriverWait(driver, 1.5).until(EC.element_to_be_clickable((By.XPATH, option2))).click()
driver.find_element(By.XPATH, option1).click()
time.sleep(0.5)
driver.find_element(By.XPATH, x).click()
# 初始化配置>>>>>>>>>>>>>>>>>>>>
print("载入数据...")
# 读取用户信息
try:
with open('./account.json', 'r', encoding='utf-8') as f:
account = json.load(f)
except JSONDecodeError:
print("account文件内容有误!")
exit(-1)
# 启动driver
driver = driver_init(account["Driver"].strip())
if not driver:
print("暂不支持该浏览器,请检查配置!")
exit(-1)
# 隐藏浏览器指纹
with open("Res/stealth.min.js", "r") as f:
js = f.read()
driver.execute_cdp_cmd("Page.addScriptToEvaluateOnNewDocument", {"source": js})
# >>>>>>>>>>>>>>>>>>>>>>>>
# constants
login_url = "https://passport.zhihuishu.com/login"
lesson_finish = True
user = account['User'].strip()
pwd = account['Password'].strip()
# ElementXpath
drag_bar = "/html/body/div[31]/div[2]/div/div/div[2]/div/div[2]/div[2]"
option1 = '//*[@id="playTopic-dialog"]/div/div[2]/div/div[1]/div/div/div[2]/ul/li[1]/div[2]'
option2 = '//*[@id="playTopic-dialog"]/div/div[2]/div/div[1]/div/div/div[2]/ul/li[2]/div[2]'
x = '/html/body/div[1]/div/div[2]/div[1]/div[2]/div[1]/div/div[3]/span/div'
# javascript
# 登录
user_js = f'''document.getElementById("lUsername").value="{user}";'''
pwd_js = f'''document.getElementById("lPassword").value="{pwd}";'''
login_js = '''document.getElementsByClassName("wall-sub-btn")[0].click();'''
block_js = '''return document.getElementsByClassName("yidun_jigsaw")[0].src'''
bg_js = '''return document.getElementsByClassName("yidun_bg-img")[0].src'''
# 弹窗
pop_late = '''document.getElementsByClassName("el-dialog__close el-icon el-icon-close")[2].click()'''
pop_js = '''document.getElementsByClassName("iconfont iconguanbi")[0].click();'''
pop2_js = '''document.evaluate('//*[@id="app"]/div/div[1]/div[1]/span/a',document).iterateNext().click();'''
# 其他
night_js = '''document.getElementsByClassName("Patternbtn-div")[0].click()'''
show_controlsBar = '''document.getElementsByClassName("controlsBar")[0].setAttribute("style","z-index: 2; overflow: inherit;")'''
try:
print("正在登录...")
# 打开登录页面,自动填充账户密码
if not user or not pwd:
raise UserWarning
driver.get(login_url)
driver.execute_script(user_js)
driver.execute_script(pwd_js)
time.sleep(0.5)
driver.execute_script(login_js)
# 等待完成滑块验证,已设置80s等待时间
WebDriverWait(driver, 80).until(
EC.invisibility_of_element((By.CLASS_NAME, "wall-main")))
# 遍历所有课程,加载网页
for course_url in account["Url"]:
if course_url.strip()[:4] != "http":
continue
print("加载播放页...")
driver.get(course_url)
# 等待开屏弹窗出现
WebDriverWait(driver, 80).until(
EC.presence_of_element_located((By.CLASS_NAME, "studytime-div")))
time.sleep(0.5)
try:
# 关闭课程结束提示
driver.execute_script(pop_late)
# 关闭上方横幅
driver.execute_script(pop2_js)
except JavascriptException:
pass # 不是每次都有此类弹窗
# 关闭学习须知
driver.execute_script(pop_js)
# 根据当前时间切换夜间模式
hour = time.localtime().tm_hour
if hour >= 18 or hour < 7:
driver.execute_script(night_js)
# 遍历课程小节
while True:
# 获取课程名
title = get_lesson_name()
print("正在学习:%s" % title)
# 开始播放
curtime = "0"
check_play()
# 设置为静音
shut_volume()
while curtime != "100%":
try:
# 跳过中途弹题(只支持选择题)
skip_question()
except TimeoutException:
pass # 并非每时都要跳过答题
curtime2 = get_progress()
if curtime2 != curtime: # 进度条若相同则说明视频已暂停
curtime = curtime2
print('已完成:%s' % curtime)
else:
check_play()
print("检测到视频暂停,已自动继续..")
title = get_lesson_name()
print("正在学习:%s" % title)
# 进度100%时开始下一集
try:
play_next()
except Exception: # 捕获有时点击失败的情况
continue
# 延时获取新数据
time.sleep(1.5)
title2 = get_lesson_name()
# 点完下一集后若标题不改变,判定为课程结束
if title2 != title:
print(f"\"{title}\" Done !")
else:
print("已学完本课程全部内容!")
print(">>" * 10)
break
time.sleep(1)
# 特殊异常捕获
except Exception as e:
print(f"Error:{type(e)}")
if type(e) == UserWarning:
print("哎呀,是不是忘填账号密码了~")
elif type(e) == NoSuchWindowException:
print("糟糕!网页关闭了~")
else:
log = traceback.format_exc()
print("错误日志已保存至:log.txt")
with open("./log.txt", "w", encoding="utf-8") as doc:
doc.write(log)
print("系统异常,要不重启一下?")
lesson_finish = False
# 程序出口
driver.quit()
if lesson_finish:
print("所有课程学习完毕!")
os.system("pause")
注意:本程序只可用于学习和研究计算机原理(你懂的)
还等什么? 快开始愉快的刷课吧~ !
版权归原作者 Runfreeone 所有, 如有侵权,请联系我们删除。