@charset "UTF-8";
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(255, 255, 255, 1); color: rgba(0, 0, 0, 1) }
.hljs-comment, .hljs-quote, .hljs-variable { color: rgba(0, 128, 0, 1) }
.hljs-built_in, .hljs-keyword, .hljs-name, .hljs-selector-tag, .hljs-tag { color: rgba(0, 0, 255, 1) }
.hljs-addition, .hljs-attribute, .hljs-literal, .hljs-section, .hljs-string, .hljs-template-tag, .hljs-template-variable, .hljs-title, .hljs-type { color: rgba(163, 21, 21, 1) }
.hljs-deletion, .hljs-meta, .hljs-selector-attr, .hljs-selector-pseudo { color: rgba(43, 145, 175, 1) }
.hljs-doctag { color: rgba(128, 128, 128, 1) }
.hljs-attr { color: rgba(255, 0, 0, 1) }
.hljs-bullet, .hljs-link, .hljs-symbol { color: rgba(0, 176, 232, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }

介绍

编程是一项令人兴奋的工作,然而每天都在代码编写中度过很长时间后,我们也希望能够进行一些娱乐活动来放松身心,例如看电影。那么今天,我想要介绍的是一款能够将VS Code变成你的私人影院的插件!这个插件将可以让你在VS Code的代码编辑区中轻松观看电影,不再需要切换电影应用或浏览器。本篇文章将会给你展示如何在VS Code中开发这款插件,采用的技术包括Node.js、axios、JSdom和VS Code的webview等。

准备工作及开发环境搭建

首先,你需要在电脑上安装最新的VS Code和Node.js版本。然后,在VS Code中打开命令行终端,创建一个新的文件夹,用来存储插件代码:

mkdir my-video-plugin
cd my-video-plugin
code .

爬取腾讯视频数据:如何使用axios和jsdom发送请求并解析HTML

我们需要爬取腾讯视频的数据来获取电影列表。我们将使用axios和jsdom插件实现该功能。首先,我们需要安装这些插件:

npm install axios jsdom

接下来,在你的代码中引入它们:

const axios = require('axios');
const jsdom = require('jsdom');
const { JSDOM } = jsdom;

使用axios发送一个GET请求,然后使用jsdom解析HTML文档。以下是解析腾讯视频网站上视频标题的示例代码:

axios.get('https://v.qq.com/')
.then((response) => {
const dom = new JSDOM(response.data);
const document = dom.window.document; // 从腾讯视频提取视频标题
// 注意:class类名可能会被调整修改,掘友们在自己手动做的时候记得自己检查网页源码
const videoTitles = document.querySelectorAll('.item_title .needsclick'); videoTitles.forEach((title) => {
console.log(title.textContent);
});
})
.catch((error) => {
console.log(error);
});

具体的视频链接提取与标题提取原理是一样的,因此这里不多做赘述,免的被掘友们说我水文。

可视化搜索:如何在VS Code中为插件提供搜索界面

我们需要让用户能够搜索他们想要看的电影。我们将添加一个搜索框,用户可以在其中输入搜索关键字。当用户按下回车键时,我们将搜索该关键字并显示结果。

在VS Code中使用webview来实现可视化搜索框,我们需要首先创建一个HTML文件并添加一个搜索框和一个结果列表。接下来,我们需要将其加载到webview中。

以下是加载HTML文件到webview的示例代码:

const panel = vscode.window.createWebviewPanel(
'myVideoPlugin',
'My Video Plugin',
vscode.ViewColumn.One,
{}
); panel.webview.html = `<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>My Video Plugin</title>
</head>
<body>
<input type="text" id="search-box">
<ul id="results"></ul>
</body>
</html>`;

我们还需要指定webview将在哪些情况下显示和隐藏。以下是在搜索命令被触发时显示webview的示例代码:

const disposable = vscode.commands.registerCommand('myVideoPlugin.search', () => {
panel.reveal();
});

可视化账号配置:如何让用户方便地在VS Code中配置腾讯视频账号

我们将添加一个配置页面,让用户可以配置他们的腾讯视频账号。我们将使用VS Code提供的配置API。在你的代码中添加以下代码来设置一个配置项:

const configuration = vscode.workspace.getConfiguration('myVideoPlugin');
const defaultValue = '';
const username = configuration.get('username', defaultValue);
const password = configuration.get('password', defaultValue);

我们需要为配置页面在webview中创建一个表单,让用户可以输入他们的用户名和密码。以下是创建表单的示例代码:

panel.webview.html = `<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>My Video Plugin</title>
</head>
<body>
<h1>Configuration</h1>
<form>
<label for="username">Username:</label>
<input type="text" id="username" name="username" value="${username}">
<br>
<label for="password">Password:</label>
<input type="password" id="password" name="password" value="${password}">
<br>
<button type="submit">Save</button>
</form>
</body>
</html>`;

我们可以在表单提交时保存用户的配置信息。以下是保存配置信息的示例代码:

panel.webview.onDidReceiveMessage((message) => {
switch (message.command) {
case 'saveConfig':
configuration.update('username', message.username, true);
configuration.update('password', message.password, true);
break;
}
});

视频播放:如何在VS Code中使用webview播放腾讯视频

我们将为每个视频创建一个webview以播放视频。当用户将鼠标悬停在视频标题上时,我们将显示一个播放按钮。当用户点击该按钮时,我们将创建一个新的webview来播放视频。

以下是创建视频播放窗口并加载视频的示例代码:

vscode.window.createWebviewPanel(
'myVideoPluginPlayback',
title,
vscode.ViewColumn.Active,
{
enableScripts: true,
retainContextWhenHidden: true,
}
)
.webview.html = `
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<style>
body {
margin: 0;
background-color: black;
}
</style>
</head>
<body>
<video controls autoplay muted>
<source src="${videoUrl}" type="video/mp4">
Your browser does not support the video tag.
</video>
</body>
</html>
`;

我们需要在搜索结果中为每个视频创建一个播放按钮。以下是为每个视频添加播放按钮的示例代码:

videoTitles.forEach((title) => {
const listItem = document.createElement('li');
const button = document.createElement('button');
button.textContent = 'Play';
button.onclick = () => {
playVideo(title.textContent);
};
listItem.appendChild(title);
listItem.appendChild(button);
resultList.appendChild(listItem);
});

打包和发布:将插件打包并发布到VS Code Marketplace

最后,我们需要将插件打包并发布到VS Code Marketplace。我们可以使用VS Code的命令行接口来完成这项任务。首先,我们需要在你的插件项目的根目录中创建一个package.json文件:

{
"name": "my-video-plugin",
"displayName": "My Video Plugin",
"description": "A plugin to turn VS Code into your personal movie theater.",
"version": "0.0.1",
"publisher": "your-publisher-name",
"engines": {
"vscode": "^1.60.0"
},
"categories": [
"Other"
],
"activationEvents": [
"onCommand:myVideoPlugin.search",
"onCommand:myVideoPlugin.config"
],
"main": "./out/extension.js",
"contributes": {
"commands": [
{
"command": "myVideoPlugin.search",
"title": "Search videos"
},
{
"command": "myVideoPlugin.config",
"title": "Configure My Video Plugin"
}
],
"configuration": {
"type": "object",
"title": "My Video Plugin",
"properties": {
"username": {
"type": "string",
"default": ""
},
"password": {
"type": "string",
"default": ""
}
}
}
},
"scripts": {
"vscode:prepublish": "npm run compile",
"compile": "tsc -p ."
},
"devDependencies": {
"@types/node": "^16.7.13",
"@types/vscode": "^1.60.0",
"typescript": "^4.4.4",
"vsce": "^1.123.1"
}
}

接下来,我们需要在终端中运行以下命令来打包插件并发布到VS Code Marketplace:

vsce package
vsce publish

总结:回顾插件实现过程

通过这篇文章,我们学习了如何开发一个让VS Code成为你私人影院的插件。我们使用了Node.js、axios、JSdom以及VS Code的webview等技术实现了爬取电影列表,可视化搜索,可视化账号配置以及视频播放等多种功能。最后,我们还学习了如何打包并发布插件到VS Code Marketplace。

最终的成果将是一个与其他VS Code插件一样易于使用的插件,使得看电影成为了愉悦的工作体验的一部分。皆大欢喜!

注意:本文代码并非完整代码,只是给大家讲解一下开发一个插件的具体步骤以及实现的核心方法和思路,若有需要,可以在评论区留言,如果人数较多的话,我会出一版完整的开源VS code视频播放插件代码,扩展其他功能,供大家开箱即用,让大家打工路上快乐摸鱼。

摸鱼神器:打造一款让VS Code成为你私人影院的插件的更多相关文章

  1. Thief-Book 上班摸鱼神器

    Thief-Book 上班摸鱼神器 介绍 Thief-Book 是一款真正的摸鱼神器,可以更加隐秘性大胆的看小说. 隐蔽性 自定义透明背景,随意调整大小,完美融入各种软件界面 快捷性 三个快捷键,实现 ...

  2. vscode插件(摸鱼神器-小霸王游戏机

    vscode插件(摸鱼神器-小霸王游戏机 步骤 vscode扩展搜索小霸王,点击下载即可. 使用 默认有一个demo小游戏,即超级玛丽. 本地仓库 可以通过local菜单上的添加按钮添加本地nes r ...

  3. 【摸鱼神器】UI库秒变LowCode工具——列表篇(一)设计与实现

    内容摘要: 需求分析 定义 interface 定义 json 文件 定义列表控件的 props 基于 el-table 封装,实现依赖 json 渲染 实现内置功能:选择行(单选.多选),格式化.锁 ...

  4. 【转】让Chrome化身成为摸鱼神器,利用Chorme运行布卡漫画以及其他安卓APK应用教程

    下周就是十一了,无论是学生党还是工作党,大家的大概都会有点心不在焉,为了让大家更好的心不在焉,更好的在十一前最后一周愉快的摸鱼,今天就写一个如何让Chrome(google浏览器)运行安卓APK应用的 ...

  5. 【摸鱼神器】基于SSM风格的Java源代码生成器 单表生成 一对一、一对多、多对多连接查询生成

    一.序言 UCode Cms 是一款Maven版的Java源代码生成器,是快速构建项目的利器.代码生成器模块属于可拆卸模块,即按需引入.代码生成器生成SSM(Spring.SpringBoot.Myb ...

  6. 【摸鱼神器】UCode Cms管理系统 内置超好用的代码生成器 解决多表连接痛点

    一.序言 UCode Cms管理系统是面向企业级应用软件开发的脚手架.当前版本1.3.4.快速体验: git clone https://gitee.com/decsa/demo-cms.git (一 ...

  7. 【摸鱼神器】一次搞定 vue3的 路由 + 菜单 + tabs

    做一个管理后台,首先要设置路由,然后配置菜单(有时候还需要导航),再来一个动态tabs,最后加上权限判断. 这个是不是有点繁琐?尤其是路由的设置和菜单的配置,是不是很雷同?那么能不能简单一点呢?如果可 ...

  8. 【摸鱼神器】UI库秒变LowCode工具——列表篇(二)维护json的小工具

    上一篇介绍了一下如何实现一个可以依赖 json 渲染的列表控件,既然需要 json 文件,那么要如何维护这个 json 文件就成了重点,如果没有好的维护方案的话,那么还不如直接用UI库. 所以需要我们 ...

  9. 【摸鱼神器】UI库秒变低代码工具——表单篇(一)设计

    前面说了列表的低代码化的方法,本篇介绍一下表单的低代码化. 内容摘要 需求分析. 定义 interface. 定义表单控件的 props. 定义 json 文件. 基于 el-form 封装,实现依赖 ...

  10. 【摸鱼神器】UI库秒变低代码工具——表单篇(二)子控件

    上一篇介绍了表单控件,这一篇介绍一下表单里面的各种子控件的封装方式. 主要内容 需求分析 子控件的分类 子控件属性的分类 定义 interface. 定义子控件的的 props. 定义 json 文件 ...

随机推荐

  1. bin格式转safetensors

    技术背景 本文主要介绍在Hugging Face上把bin格式的模型文件转为safetensors格式的模型文件,并下载到本地的方法. bin转safetensors 首先安装safetensors: ...

  2. Linux - crontab 详解

    linux 系统由(crond)这个系统服务来控制的,crond 是 linux 下用来周期性的执行某种任务或等待处理某些事件的一个守护进程,Linux 系统也提供了使用者控制计划任务的命令:cron ...

  3. Vuex:让状态管理不再头疼的“管家”

    如果你正在开发一个 Vue.js 应用程序,但发现自己被各种组件之间的状态共享问题搞得焦头烂额,那么 Vuex 就是你需要的"超级管家".Vuex 是专门为 Vue.js 设计的状 ...

  4. Vulnhub-Zico2靶机-漏扫弱口令数据库+文件包含反弹shell+zip,tar提权

    一.靶机搭建 选择扫描虚拟机 选择路径即可 二.信息收集 官方信息 先看一下官网的信息 级别:中级 目标:获取root并读取标志文件 说明: Zico正试图建立自己的网站,但在选择使用什么CMS时遇到 ...

  5. git安装教程以及生成git ssh key

    问题 GitHub拉取代码需要SSH,总是忘记命令,现网百度.记录下来,以后靠自己的博文 第一步 安装git 地址: https://git-scm.com/download/win 安装它 第二步 ...

  6. typora编辑数学公式

    最后,需要补充两点: 1.如果要导出为docx,需要安装pandoc https://github.com/jgm/pandoc/releases/tag/3.6.3 2.如果要自己写代码,比如输入\ ...

  7. 腾讯云锐驰型轻量服务器搭建开源远程桌面软件RustDesk中继服务器小记

    RustDesk是一个基于Rust编写的全平台开源远程桌面软件,其最大的特点为开箱即用,且数据完全自主掌控,甚至可以依托此项目定制化开发自己专属的远程桌面软件. 一.前言 由于我个人经常性出差,对远程 ...

  8. FastAPI复杂查询终极指南:告别if-else的现代化过滤架构

    title: FastAPI复杂查询终极指南:告别if-else的现代化过滤架构 date: 2025/3/14 updated: 2025/3/14 author: cmdragon excerpt ...

  9. Suspense和vue-async-manager

    Suspense Suspense是 Vue3.x 中新增的特性, 那它有什么用呢?别急,我们通过 Vue2.x 中的一些场景来认识它的作用. Vue2.x 中应该经常遇到这样的场景: <tem ...

  10. ubuntu 刷新 hosts 命令

    systemd-resolved 服务 sudo systemctl restart systemd-resolved 这个命令将重启 systemd-resolved 服务,该服务负责 DNS 解析 ...