话不多说,直接上代码。。。
html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src="../js/popup.js" type="module"></script>
<link rel="stylesheet",type="text/css" href="../css/popup.css">
</head>
<body>
<div class="switchDiv">
<label>礼物弹幕</label>
<label class="switch">
<input type="checkbox" id="gift"/>
<span class="slider round"></span>
</label>
</div>
<div class="switchDiv">
<label>评论弹幕</label>
<label class="switch">
<input type="checkbox" id="comment"/>
<span class="slider round"></span>
</label>
</div>
</body>
</html>
bacground.js
console.log('background.js');
chrome.tabs.onUpdated.addListener((tabId,changeInfo,tab)=>{
if(changeInfo.status==='loading' && tab.url && !tab.url.includes("chrome://")){
if(tab.url.includes('live.douyin.com')){
scriptingRegisterContentScripts("douyinbarrage",'./js/douyinbarrage.js');
}else{
scriptingRegisterContentScripts('empty','./js/empty.js');
}
}
});
async function scriptingRegisterContentScripts(...args){
const content_script_id=args[0];
const js_file_name=args[1];
const existingContentScripts=await chrome.scripting.getRegisteredContentScripts({
ids:[content_script_id]
});
if(existingContentScripts.length>0){
await chrome.scripting.updateContentScripts([{
id: content_script_id,
matches:["<all_urls>"],
js:[js_file_name]
}])
.then(() => console.log("registration updated..."+js_file_name));
}else{
await chrome.scripting.registerContentScripts([{
id:content_script_id,
js:[js_file_name],
matches:["<all_urls>"],
runAt:"document_idle",
}])
.then(() => console.log("registration complete..."+js_file_name))
.catch((err) => console.warn("unexpected error", err))
}
}
douyinbarrage.js
console.log("douyinbarrage.js");
//拖动图标
const dragDiv=document.createElement('div');
dragDiv.style.backgroundColor='yellowgreen';
dragDiv.style.width='30px';
dragDiv.style.height='30px';
dragDiv.style.left='20px';
dragDiv.style.top='20px';
dragDiv.style.position='absolute';
dragDiv.style.zIndex='200';
dragDiv.style.cursor='move';
dragDiv.style.borderRadius='50%';
//创建画布
const canvas=document.createElement('canvas');
canvas.width=300;
canvas.height=500;
canvas.style.backgroundColor='yellowgreen';
canvas.style.position='absolute';
canvas.style.left='20px';
canvas.style.top='20px';
canvas.style.zIndex='199';
canvas.style.border='none';
//可拖动画布
let isDragging = false;
dragDiv.addEventListener('mousedown', function(e) {
isDragging = true;
const offsetX = e.clientX - canvas.offsetLeft;
const offsetY = e.clientY - canvas.offsetTop;
document.addEventListener('mousemove', drag);
function drag(e) {
if (isDragging) {
e.preventDefault();
canvas.style.left = e.clientX - offsetX + 'px';
canvas.style.top = e.clientY - offsetY + 'px';
dragDiv.style.left = e.clientX - offsetX + 'px';
dragDiv.style.top = e.clientY - offsetY + 'px';
}
}
document.addEventListener('mouseup', function() {
isDragging = false;
document.removeEventListener('mousemove', drag);
});
});
document.body.appendChild(dragDiv);
document.body.appendChild(canvas);
//画布内容初始化
//图片数据
const fish_img_src='';
const src1='';
const src2='';
const src3='';
const src4='';
const src5='';
const src6='';
const src7='';
const src8='';
var ctx = canvas.getContext('2d');
var [cw,ch]=[canvas.width,canvas.height];
//左上角的分数 text,x,y
var [up_text,up_text_x,up_text_y]=[0,10,ch/2-25];
//左下角的分数 text,x,y
var [down_text,down_text_x,down_text_y]=[0,10,ch/2+35]
//中间的河流 x,y,w,h
var [river_x,river_y,river_w,river_h]=[0,(ch/2-100/2),cw,100];
//鱼
var fish_img=new Image();
var is_fish=false;
var fish_i=0;
var fish_fc = 0; //调速变量
fish_img.src=fish_img_src;
fish_img.onload=()=>{
is_fish=true;
}
var fish_step=5; //移动步数
var [fish_x,fish_y,fish_w,fish_h]=[(cw/2-55/2),(ch/2-37/2),55,37];
var [fish_shake_x,fish_shake_y]=[0,0];
//角色
var player_init_imgs=[src1,src2,src3,src4,src5,src6,src7,src8] //预设角色头像
var player_list=[]; //加载完成的角色
var [player_shake_x,player_shake_y]=[0,0];
var is_player_img=false;
var psz=15;
var player_fc=0 //调速变量
//草
var grass_list=[];
var gr=3; //半径
var grass_tickness=.4; //粗细
var grNum=600; //草的数量
for(let i=0;i<grNum;i++){
grass_list.push({
'gx':randomX(),
'gy':randomY()
});
}
//游戏方法
function moveUp(){
if(fish_y>0){
fish_y-=fish_step;
}
if(fish_y<200){
s1();
}
}
function moveDown(){
if(fish_y<=ch-fish_h){
fish_y+=fish_step;
}
if(fish_y>270){
s2();
}
}
function s1(){
up_text+=1;
down_text-=1;
}
function s2(){
up_text-=1;
down_text+=1;
}
document.addEventListener('keydown',(e)=> {
if (e.key==='ArrowUp'){
moveUp()
}else if(e.key==='ArrowDown'){
moveDown()
}else if(e.key==='ArrowLeft'){
loadPlayerImg();
}else if(e.key==='ArrowRight'){
player_list=[];
[up_text,down_text]=[0,0];
[fish_x,fish_y]=[(cw/2-55/2),(ch/2-37/2)];
}
});
function gameLoop() {
ctx.clearRect(0,0,cw,ch);
drawGrass();
drawRiver();
drawScore();
drawFish();
drawPlayer();
requestAnimationFrame(gameLoop);
}
gameLoop();
//工具
function getRandomFromArray(arr) {
if (!Array.isArray(arr) || arr.length === 0) return undefined;
const randomIndex = Math.floor(Math.random() * arr.length);
return arr[randomIndex];
}
function randomY(){
//随机阵营 上 下
let campsite=Math.floor(Math.random() * 2) + 1
let campsite_y;
if(campsite===1){
campsite_y=Math.floor(Math.random() * 160) + 15; //上方
}else{
campsite_y=Math.floor(Math.random() * 170) + 310 //下方
}
return campsite_y;
}
function randomX(){
//随机x轴
let campsite_x=Math.floor(Math.random() * 265) + 15;
return campsite_x;
}
//绘制草
function drawGrass(){
grass_list.forEach((grass)=>{
let gx=grass['gx'];
let gy=grass['gy'];
ctx.beginPath();
ctx.lineWidth=grass_tickness;
ctx.strokeStyle='green';
// 0 90(0.5*Math.PI) 180(1*Math.PI) 270(1.5*Math.PI)
ctx.arc(gx, gy, gr, 1.5*Math.PI, 0); //270-0度 左部分
ctx.arc(gx+gr*2, gy, gr, 1*Math.PI, 1.5*Math.PI); //180-270度 右部分
ctx.stroke();
});
}
//绘制河流
function drawRiver(){
ctx.fillStyle='skyblue';
ctx.fillRect(river_x,river_y,river_w,river_h);
}
//绘制分数
function drawScore(){
//左上角分数
ctx.fillStyle='red';
ctx.font='16px Arial';
ctx.fillText(up_text,up_text_x,up_text_y);
//左下角分数
ctx.fillText(down_text,down_text_x,down_text_y);
}
//绘制鱼
function drawFish(){
if(is_fish){
//每5帧fish_i++
if(fish_fc%5==0){
fish_i++;
if(fish_i==4)fish_i=0;
fish_shake_x=Math.floor(Math.random() * 3) + 1
fish_shake_y=Math.floor(Math.random() * 3) + 1
}
ctx.drawImage(fish_img, 0,fish_h*fish_i, fish_w,fish_h, fish_x+fish_shake_x,fish_y+fish_shake_y, fish_w,fish_h);
fish_fc++;
}
}
//绘制角色
function drawPlayer(){
if(is_player_img){
//每10帧浮动一下
if(player_fc%10==0){
//随机一个小幅度移动
player_shake_x=Math.floor(Math.random() * 2) + 1
player_shake_y=Math.floor(Math.random() * 2) + 1
}
player_list.forEach((player,index)=>{
let pim=player['player_img'];
let px=player['player_x'];
let py=player['player_y'];
if(index%2==0){ //偶数
//鱼线
ctx.beginPath();
ctx.lineWidth=.15;
ctx.strokeStyle='#999999';
ctx.moveTo(px+psz/2+player_shake_x,py+psz/2+player_shake_y);
ctx.lineTo(fish_x+fish_w/2,fish_y+fish_h/2);
ctx.stroke();
ctx.drawImage(pim,px+player_shake_x,py+player_shake_y,psz,psz);
}else{
//鱼线
ctx.beginPath();
ctx.lineWidth=.15;
ctx.strokeStyle='#999999';
ctx.moveTo(px+psz/2-player_shake_x,py+psz/2-player_shake_y);
ctx.lineTo(fish_x+fish_w/2,fish_y+fish_h/2);
ctx.stroke();
ctx.drawImage(pim,px-player_shake_x,py-player_shake_y,psz,psz);
}
})
player_fc++;
}
}
//加载角色
function loadPlayerImg(){
let player_img=new Image();
//随机一个头像
player_img.src=getRandomFromArray(player_init_imgs);
player_img.onload=()=>{
//加载完成标识
is_player_img=true;
let player_dict={
'player_img':player_img,
'player_x':randomX(),
'player_y':randomY()
};
player_list.push(player_dict);
}
}
//抖音方法
var isGift=false;
var isComment=false;
//你自己可以增加一些规则
function findtext(){
setInterval(function(){
if(isGift){
const giftInfo=document.getElementsByClassName('webcast-chatroom___bottom-message');
if(giftInfo){
const giftI=giftInfo[giftInfo.length-1];
try {
chrome.storage.local.set({douyinGift: giftI.textContent }).then(() => {
});
} catch (error) {
}
}
}
if(isComment){
const commentInfo=document.getElementsByClassName('webcast-chatroom___item webcast-chatroom___enter-done');
if(commentInfo){
const commentI=commentInfo[commentInfo.length-1];
try {
chrome.storage.local.set({douyinComment: commentI.textContent }).then(() => {
});
} catch (error) {
}
}
}
},300);
}
findtext();
chrome.storage.onChanged.addListener((changes, namespace) => {
for (let [key, { oldValue, newValue }] of Object.entries(changes)) {
if(key==='douyinGift'){
if(oldValue!=newValue){
console.log('gift...newValue...'+newValue);
/*if(newValue.includes('来了')){
loadPlayerImg(); //增加角色
}else if(newValue.includes('主播点赞')){
//
}*/
}
}else if(key==='douyinComment'){
if(oldValue!=newValue){
//这里设置条件来判断鱼是上移还是下移
/*if(){
moveUp();
}else if(){
moveDown();
}*/
console.log('comment...newValue...'+newValue);
}
}
}
});
chrome.runtime.onMessage.addListener(async (request,sender,sendResponse)=>{
if(request.action==='gift'){
isGift=request.message;
console.log('gift...request message...'+request.message);
}else if(request.action==='comment'){
isComment=request.message;
console.log('comment...request message...'+request.message);
}
return true;
});
popup.js
console.log('popup.js');
document.addEventListener("DOMContentLoaded", function () {
chrome.tabs.query({active: true, currentWindow: true}, (tabs) => {
if(tabs){
if(tabs.length>0){
const gift=document.getElementById('gift');
const comment=document.getElementById('comment');
chrome.storage.sync.get(['isGift','isComment'],function(data){
gift.checked=data.isGift;
comment.checked=data.isComment;
})
gift.addEventListener('click',function(){
sendMsg(tabs[0].id,"gift",this.checked);
chrome.storage.sync.set({
isGift:this.checked
})
});
comment.addEventListener('click',function(){
sendMsg(tabs[0].id,"comment",this.checked);
chrome.storage.sync.set({
isComment:this.checked
})
});
}
}
});
});
function sendMsg(...args){
chrome.tabs.sendMessage(args[0],{
action:args[1],
message:args[2]
},function(response){
if(chrome.runtime.lastError||!response){
}
})
}
empty.js
console.log('empty.js');
popup.css
body{
width: 230px;
height: auto;
display: flex;
justify-content: center; /*水平居中*/
align-items: center; /*垂直居中*/
}
/* The switch - the box around the slider */
.switchDiv{
display: flex;
width: 48%;
float: left;
justify-content: center; /* 水平居中 */
align-items: center; /* 垂直居中 */
}
.switchDiv label{
margin: 5px;
}
.switch {
position: relative;
display: inline-block;
width: 30px;
height: 15px;
}
/* Hide default HTML checkbox */
.switch input {
opacity: 0;
width: 0;
height: 0;
}
/* The slider */
.slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
-webkit-transition: .2s;
transition: .2s;
}
.slider:before {
position: absolute;
content: "";
height: 13px;
width: 13px;
left: 1px;
bottom: 1px;
background-color: white;
-webkit-transition: .2s;
transition: .2s;
}
input:checked + .slider {
background-color: #2196F3;
}
input:checked + .slider:before {
-webkit-transform: translateX(14px);
-ms-transform: translateX(14px);
transform: translateX(14px);
}
/* Rounded sliders */
.slider.round {
border-radius: 34px;
}
.slider.round:before {
border-radius: 50%;
}
/* The switch - the box around the slider */
manifest.json
{
"manifest_version": 3,
"name": "弹幕游戏",
"author":"atomic",
"version": "1.0.0",
"description": "弹幕游戏",
"icons": {
"16": "./imgs/icon16.png",
"32": "./imgs/icon32.png",
"48": "./imgs/icon48.png"
},
"action": {
"default_title": "弹幕游戏",
"default_popup": "./html/popup.html",
"default_icon": "./imgs/icon32.png"
},
"background": {
"service_worker": "./js/background.js",
"type": "module"
},
"host_permissions": [
"<all_urls>"
],
"permissions": [
"activeTab","scripting",
"storage",
"userScripts","tabs"
]
}
完整的结构
不会写谷歌扩展的,自己上谷歌官网https://developer.chrome.google.cn/docs/extensions/reference/api?hl=zh-cn 看一下