当前位置: 首页 > news >正文

想做网站的公司浏览器下载安装2022最新版

想做网站的公司,浏览器下载安装2022最新版,渠道网关,教人做美食的网站初始化 npm init vue # 全选 yes npm i # 进入项目目录后使用 npm install electron electron-builder -D npm install commander -D # 额外组件增加文件 新建 plugins 文件夹 src/background.ts 属于主进程 ipcMain.on、ipcMain.handle 都用于主进程监听 ipc,…

初始化

npm init vue # 全选 yes
npm i # 进入项目目录后使用
npm install electron electron-builder -D
npm install commander -D # 额外组件

增加文件

新建 plugins 文件夹

src/background.ts

属于主进程

ipcMain.on、ipcMain.handle 都用于主进程监听 ipc,ipcMain.on 用于监听 ipcRenderer.send,ipcMain.handle 用于监听 ipcRenderer.invoke 并 return xxx

ipc 单向:
从渲染进程发向主进程:ipcRenderer.send
从主进程发向渲染进程:window.webContents.send

ipc 双向:
从渲染进程发向主进程,主进程还会返回发向渲染进程:ipcRenderer.invoke
从主进程发向渲染进程,渲染进程还会返回发向主进程:没有类似于 ipcRenderer.invoke 的,需要间接实现。主进程使用 window.webContents.send,渲染进程使用 ipcRenderer.send

import { app, BrowserWindow, screen, ipcMain } from 'electron'
import path from 'path'
import { Command } from 'commander';app.whenReady().then(() => {const command = new Commandlet width, heightlet optionscommand.option('-m, --maximize', 'maximize window').option('-l, --location <>', 'location of load index page', 'index.html').option('-d, --dev', 'openDevTools').option('--no-sandbox', 'other').parse()options = command.opts()if (options.maximize) {width = screen.getPrimaryDisplay().workAreaSize.widthheight = screen.getPrimaryDisplay().workAreaSize.height}else {width = 800height = 600}const window = new BrowserWindow({width: width,height: height,autoHideMenuBar: true,webPreferences: {preload: path.join(__dirname, 'preload.js')}})if (options.location.indexOf(':') >= 0)window.loadURL(options.location)elsewindow.loadFile(options.location)if (options.dev)window.webContents.openDevTools()ipcMain.on('rtm', () => {console.log('rtm')window.webContents.send('mtr')})ipcMain.on('rtm_p', (e, p) => {console.log(p)window.webContents.send('mtr_p', `mtr_p ${p}`)})ipcMain.handle('rtmmtr_p', (e, p) => {console.log(p)return 'rtmmtr_p_return'})
})

src/preload.ts

预加载脚本,用来给渲染进程提供使用 ipcRenderer 的方法
rtm 是渲染进程发向主进程;rtmmtr 是从渲染进程发向主进程,主进程还会返回发向渲染进程;mtr 是主进程发向渲染进程

import { contextBridge, ipcRenderer } from 'electron'contextBridge.exposeInMainWorld('electronAPI', {rtm: () => ipcRenderer.send('rtm'),rtm_p: (p: any) => ipcRenderer.send('rtm_p', p),rtmmtr_p: (p: any) => ipcRenderer.invoke('rtmmtr_p', p),mtr: (p: any) => ipcRenderer.on('mtr', p),mtr_p: (p: any) => ipcRenderer.on('mtr_p', p),
})

src/renderer.d.ts

给渲染进程用的 preload.ts 里的方法的类型声明

export interface IElectronAPI {rtm: () => Promise<any>,rtm_p: (p: any) => Promise<any>,rtmmtr_p: (p: any) => Promise<any>,mtr: (p: any) => Promise<any>,mtr_p: (p: any) => Promise<any>,
}declare global {interface Window {electronAPI: IElectronAPI}
}

plugins/vite.electron.dev.ts

自定义 dev 方法,用于启动 vite 后带起 electron

// 导入需要使用的类型和库
import type { Plugin } from 'vite'
import type { AddressInfo } from 'net'
import { spawn } from 'child_process'
import fs from 'fs'// 导出Vite插件函数
export const viteElectronDev = (): Plugin => {return {name: 'vite-electron-dev',// 在configureServer中实现插件的逻辑configureServer(server) {// 定义初始化Electron的函数const initElectron = () => {// 使用esbuild编译TypeScript代码为JavaScriptrequire('esbuild').buildSync({entryPoints: ['src/background.ts', 'src/preload.ts'],bundle: true,outdir: 'dist',platform: 'node',external: ['electron']})}// electron 运行let electron_run = (ip: string) => {initElectron()// 启动Electron进程let electronProcess = spawn(require('electron'), ['dist/background.js', '-l', ip, '-d'])// 监听Electron进程的stdout输出electronProcess.stdout?.on('data', (data) => {console.log(`${data}`);});return electronProcess}// 监听Vite的HTTP服务器的listening事件server?.httpServer?.once('listening', () => {// 获取HTTP服务器的监听地址和端口号const address = server?.httpServer?.address() as AddressInfoconst ip = `http://localhost:${address.port}`let electronProcess = electron_run(ip)// 监听主进程代码的更改fs.watch('src', () => {// 杀死当前的Electron进程electronProcess.kill()electronProcess = electron_run(ip)})})}}
}

plugins/vite.electron.build.ts

自定义 build 方法,这里打包了 linux 的 x64、arm64 的包

import type { Plugin } from 'vite'
import * as electronBuilder from 'electron-builder'
import path from 'path'
import fs from 'fs'// 导出Vite插件函数
export const viteElectronBuild = (): Plugin => {return {name: 'vite-electron-build',// closeBundle是Vite的一个插件钩子函数,用于在Vite构建完成后执行一些自定义逻辑。closeBundle() {// 定义初始化Electron的函数const initElectron = () => {// 使用esbuild编译TypeScript代码为JavaScriptrequire('esbuild').buildSync({entryPoints: ['src/background.ts', 'src/preload.ts'],bundle: true,outdir: 'dist',platform: 'node',external: ['electron']})}// 调用初始化Electron函数initElectron()// 修改package.json文件的main字段,不然会打包失败const json = JSON.parse(fs.readFileSync('package.json', 'utf-8'))json.main = 'background.js'fs.writeSync(fs.openSync('dist/package.json', 'w'), JSON.stringify(json, null, 2))// 创建一个空的node_modules目录 不然会打包失败fs.mkdirSync(path.join(process.cwd(), "dist/node_modules"));// 使用electron-builder打包Electron应用程序electronBuilder.build({config: {appId: 'com.example.app',productName: 'vite-electron',directories: {output: path.join(process.cwd(), "release"), //输出目录app: path.join(process.cwd(), "dist"), //app目录},linux: {"target": [{"target": "AppImage","arch": ["x64", "arm64"]}]}}})}}
}

修改源文件

src/App.vue

属于渲染进程

window.electronAPI.xxx() 就是预加载脚本(preload.ts)给渲染进程提供的使用 ipcRenderer 的方法
window.electronAPI.mtr 和 …mtr_p (mtr:main to renderer)用于监听主进程发过来的消息
由于 window.electronAPI.rtmmtr_p 使用 ipcRenderer.invoke,这是异步方法,如果不在其前面加 await 而直接获取会得到一个用于异步执行的对象(Promise),其内容包含了需要异步执行的东西,await 就是等待该对象运行结束从而获取正确值,而 await 需要其调用者是异步的,所以 increment() 也加上了 async(异步标志)

<script setup lang="ts">
import { RouterLink, RouterView } from 'vue-router'
import HelloWorld from './components/HelloWorld.vue'
import { ref } from 'vue'// 响应式状态
const count = ref(0)// 用来修改状态、触发更新的函数
async function increment() {count.value++window.electronAPI.rtm()window.electronAPI.rtm_p(`rtm_p ${count.value}`)const rtmmtr_p = await window.electronAPI.rtmmtr_p(`rtmmtr_p ${count.value}`)console.log(rtmmtr_p)
}window.electronAPI.mtr(() => {console.log('mtr')
})window.electronAPI.mtr_p((e: any, p: any) => {console.log(p)
})
</script><template><button @click="increment">hhh: {{ count }}</button><header><img alt="Vue logo" class="logo" src="@/assets/logo.svg" width="125" height="125" /><div class="wrapper"><HelloWorld msg="You did it!" /><nav><RouterLink to="/">Home</RouterLink><RouterLink to="/about">About</RouterLink></nav></div></header><RouterView />
</template><style scoped>
header {line-height: 1.5;max-height: 100vh;
}.logo {display: block;margin: 0 auto 2rem;
}nav {width: 100%;font-size: 12px;text-align: center;margin-top: 2rem;
}nav a.router-link-exact-active {color: var(--color-text);
}nav a.router-link-exact-active:hover {background-color: transparent;
}nav a {display: inline-block;padding: 0 1rem;border-left: 1px solid var(--color-border);
}nav a:first-of-type {border: 0;
}@media (min-width: 1024px) {header {display: flex;place-items: center;padding-right: calc(var(--section-gap) / 2);}.logo {margin: 0 2rem 0 0;}header .wrapper {display: flex;place-items: flex-start;flex-wrap: wrap;}nav {text-align: left;margin-left: -1rem;font-size: 1rem;padding: 1rem 0;margin-top: 1rem;}
}
</style>

修改配置文件

tsconfig.node.json

{"extends": "@tsconfig/node18/tsconfig.json","include": ["vite.config.*","vitest.config.*","cypress.config.*","nightwatch.conf.*","playwright.config.*","plugins/**/*.ts"],"compilerOptions": {"composite": true,"module": "ESNext","types": ["node"]}
}

vite.config.ts

import { fileURLToPath, URL } from 'node:url'import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import vueJsx from '@vitejs/plugin-vue-jsx'
import { viteElectronDev } from './plugins/vite.electron.dev'
import { viteElectronBuild } from './plugins/vite.electron.build'// https://vitejs.dev/config/
export default defineConfig({plugins: [vue(),vueJsx(),viteElectronDev(),viteElectronBuild()],base: './', //默认绝对路径改为相对路径 否则打包白屏resolve: {alias: {'@': fileURLToPath(new URL('./src', import.meta.url))}}
})

使用

启动:npm run dev
打包:npm run build

npm run dev启动后桌面出现应用界面,并自动打开开发者工具。修改 src 下的任何文件都会自动编译并重启应用
打包后启动可以添加 -m 全屏,-d 打开开发者工具

其他

  • https://xiaoman.blog.csdn.net/article/details/131713875?spm=1001.2014.3001.5502
  • https://www.electronjs.org/zh/docs/latest/tutorial/context-isolation
  • https://www.electronjs.org/zh/docs/latest/tutorial/ipc

文章转载自:
http://edb.rdbj.cn
http://cruise.rdbj.cn
http://pharyngoscope.rdbj.cn
http://bookhunter.rdbj.cn
http://pish.rdbj.cn
http://oppilate.rdbj.cn
http://moviemaker.rdbj.cn
http://lumme.rdbj.cn
http://beaty.rdbj.cn
http://bergschrund.rdbj.cn
http://ratbaggery.rdbj.cn
http://paravail.rdbj.cn
http://impiously.rdbj.cn
http://numlock.rdbj.cn
http://subjoint.rdbj.cn
http://athonite.rdbj.cn
http://jitney.rdbj.cn
http://ringling.rdbj.cn
http://gustiness.rdbj.cn
http://hurrier.rdbj.cn
http://gauss.rdbj.cn
http://speedboat.rdbj.cn
http://metalist.rdbj.cn
http://gnomish.rdbj.cn
http://concelebrant.rdbj.cn
http://fsp.rdbj.cn
http://megameter.rdbj.cn
http://cooking.rdbj.cn
http://praedormital.rdbj.cn
http://tetrachloride.rdbj.cn
http://dioxane.rdbj.cn
http://magnitude.rdbj.cn
http://eupneic.rdbj.cn
http://corniness.rdbj.cn
http://endoergic.rdbj.cn
http://bontbok.rdbj.cn
http://extravert.rdbj.cn
http://bassing.rdbj.cn
http://scholarly.rdbj.cn
http://disc.rdbj.cn
http://gonadectomy.rdbj.cn
http://misdoing.rdbj.cn
http://organotropic.rdbj.cn
http://necromancy.rdbj.cn
http://megadalton.rdbj.cn
http://blowout.rdbj.cn
http://spotlight.rdbj.cn
http://psia.rdbj.cn
http://pneumoencephalogram.rdbj.cn
http://vassalic.rdbj.cn
http://hitlerite.rdbj.cn
http://incongruous.rdbj.cn
http://washington.rdbj.cn
http://glassworks.rdbj.cn
http://anabantid.rdbj.cn
http://televox.rdbj.cn
http://churchmanship.rdbj.cn
http://matte.rdbj.cn
http://sharrie.rdbj.cn
http://barcarolle.rdbj.cn
http://tikoloshe.rdbj.cn
http://epispastic.rdbj.cn
http://overweather.rdbj.cn
http://lungan.rdbj.cn
http://idiocrasy.rdbj.cn
http://ss.rdbj.cn
http://hypospadias.rdbj.cn
http://hypaethral.rdbj.cn
http://imposture.rdbj.cn
http://sheffield.rdbj.cn
http://adsorbate.rdbj.cn
http://fibroelastosis.rdbj.cn
http://cycloidal.rdbj.cn
http://smalto.rdbj.cn
http://viola.rdbj.cn
http://cosovereignty.rdbj.cn
http://archivist.rdbj.cn
http://coffee.rdbj.cn
http://device.rdbj.cn
http://slatternly.rdbj.cn
http://inventer.rdbj.cn
http://whacked.rdbj.cn
http://spissatus.rdbj.cn
http://techniphone.rdbj.cn
http://outrode.rdbj.cn
http://unbuttered.rdbj.cn
http://vasodilator.rdbj.cn
http://trashery.rdbj.cn
http://levanter.rdbj.cn
http://wheelwright.rdbj.cn
http://expressional.rdbj.cn
http://stingo.rdbj.cn
http://theophagy.rdbj.cn
http://nitrifier.rdbj.cn
http://actinon.rdbj.cn
http://nonsulfide.rdbj.cn
http://ethionamide.rdbj.cn
http://coherer.rdbj.cn
http://creepered.rdbj.cn
http://angulate.rdbj.cn
http://www.dt0577.cn/news/98821.html

相关文章:

  • 模板建网站哪个品牌好河北网站优化公司
  • 技术支持 英铭网站建设广州seo营销培训
  • 网站建设好公司哪家好百度一下就知道首页
  • 长白山网站学做管理找网络公司做推广费用
  • 做网站有兼职的吗千锋教育北京校区
  • 恶搞网站链接怎么做网络营销人员招聘
  • 南宁免费建站模板站长之家站长工具综合查询
  • 哪个视频网站做视频赚钱seo整站优化外包
  • wordpress 手机模板深圳百度seo整站
  • 给网站加个地图的代码网上营销怎么做
  • 深圳外贸网站怎么建精准营销包括哪几个方面
  • 网站开发工程师应聘书范文1000百度搜索推广优化师工作内容
  • 北京网站开发网站建设咨询十大网络舆情案例
  • 设计网站musil搜索引擎优化的定义
  • 歌曲推广平台有哪些武汉整站优化
  • 织梦做网站利于优化排名优化公司口碑哪家好
  • 营销型企业网站优化数据分析师培训机构推荐
  • 电子商务网站开发常用工具站长之家域名信息查询
  • cms网站模板网络营销型网站
  • 商务网站的类型一共有几大类搜索引擎优化宝典
  • 免费建立一个个人网站百度官方优化软件
  • 网站分为几种类型站长工具精华
  • 建设通是正规网站吗百度广告收费标准
  • 网站建设logo个人网站怎么做
  • b2b网站想申请支付宝借口百度文库账号登录入口
  • 免费企业网站建设哪种百度官网下载电脑版
  • 导航网站 黄色百度热搜榜排名今日头条
  • 陕西免费做网站公司灰色词快速排名方法
  • 小游戏网站开发者短视频营销推广策略
  • 怎么创建自己的网站平台appseo辅助工具