🔨 Refactor: most config handle by picgo-core

for config syncing with picgo-core && picgo-plugins
This commit is contained in:
Molunerfinn 2019-12-26 20:15:41 +08:00
parent fec4360043
commit 3418560c63
30 changed files with 294 additions and 232 deletions

View File

@ -42,7 +42,7 @@
"keycode": "^2.2.0", "keycode": "^2.2.0",
"lodash-id": "^0.14.0", "lodash-id": "^0.14.0",
"lowdb": "^1.0.0", "lowdb": "^1.0.0",
"picgo": "^1.3.7", "picgo": "^1.4.2",
"vue": "^2.6.10", "vue": "^2.6.10",
"vue-gallery": "^2.0.1", "vue-gallery": "^2.0.1",
"vue-lazyload": "^1.2.6", "vue-lazyload": "^1.2.6",

View File

@ -1,6 +1,5 @@
'use strict' 'use strict'
import Uploader from '~/main/utils/uploader'
import { import {
app, app,
BrowserWindow, BrowserWindow,
@ -21,6 +20,8 @@ import {
installVueDevtools installVueDevtools
} from 'vue-cli-plugin-electron-builder/lib' } from 'vue-cli-plugin-electron-builder/lib'
import db from '#/datastore' import db from '#/datastore'
import picgo from '~/main/utils/picgo'
import uploader from '~/main/utils/uploader'
import beforeOpen from '~/main/utils/beforeOpen' import beforeOpen from '~/main/utils/beforeOpen'
import pasteTemplate from '#/utils/pasteTemplate' import pasteTemplate from '#/utils/pasteTemplate'
import updateChecker from '~/main/utils/updateChecker' import updateChecker from '~/main/utils/updateChecker'
@ -63,14 +64,16 @@ const miniWinURL = isDevelopment
fixPath() fixPath()
function createContextMenu () { function createContextMenu () {
const picBeds = getPicBeds(app) const picBeds = getPicBeds()
const submenu = picBeds.map(item => { const submenu = picBeds.map(item => {
return { return {
label: item.name, label: item.name,
type: 'radio', type: 'radio',
checked: db.get('picBed.current') === item.type, checked: db.get('picBed.current') === item.type,
click () { click () {
db.set('picBed.current', item.type) picgo.saveConfig({
'picBed.current': item.type
})
if (settingWindow) { if (settingWindow) {
settingWindow.webContents.send('syncPicBed') settingWindow.webContents.send('syncPicBed')
} }
@ -193,7 +196,7 @@ function createTray () {
tray.on('drop-files', async (event: Event, files: string[]) => { tray.on('drop-files', async (event: Event, files: string[]) => {
const pasteStyle = db.get('settings.pasteStyle') || 'markdown' const pasteStyle = db.get('settings.pasteStyle') || 'markdown'
const imgs = await new Uploader(files, window!.webContents).upload() const imgs = await uploader.setWebContents(window!.webContents).upload(files)
if (imgs !== false) { if (imgs !== false) {
for (let i = 0; i < imgs.length; i++) { for (let i = 0; i < imgs.length; i++) {
clipboard.writeText(pasteTemplate(pasteStyle, imgs[i])) clipboard.writeText(pasteTemplate(pasteStyle, imgs[i]))
@ -374,7 +377,7 @@ const uploadClipboardFiles = async () => {
} else { } else {
win = settingWindow || window || createSettingWindow() win = settingWindow || window || createSettingWindow()
} }
let img = await new Uploader(undefined, win!.webContents).upload() let img = await uploader.setWebContents(win!.webContents).upload()
if (img !== false) { if (img !== false) {
if (img.length > 0) { if (img.length > 0) {
const pasteStyle = db.get('settings.pasteStyle') || 'markdown' const pasteStyle = db.get('settings.pasteStyle') || 'markdown'
@ -403,7 +406,7 @@ const uploadClipboardFiles = async () => {
const uploadChoosedFiles = async (webContents: WebContents, files: IFileWithPath[]) => { const uploadChoosedFiles = async (webContents: WebContents, files: IFileWithPath[]) => {
const input = files.map(item => item.path) const input = files.map(item => item.path)
const imgs = await new Uploader(input, webContents).upload() const imgs = await uploader.setWebContents(webContents).upload(input)
if (imgs !== false) { if (imgs !== false) {
const pasteStyle = db.get('settings.pasteStyle') || 'markdown' const pasteStyle = db.get('settings.pasteStyle') || 'markdown'
let pasteText = '' let pasteText = ''
@ -427,11 +430,11 @@ const uploadChoosedFiles = async (webContents: WebContents, files: IFileWithPath
} }
} }
picgoCoreIPC(app, ipcMain) picgoCoreIPC()
// from macOS tray // from macOS tray
ipcMain.on('uploadClipboardFiles', async () => { ipcMain.on('uploadClipboardFiles', async () => {
const img = await new Uploader(undefined, window!.webContents).upload() const img = await uploader.setWebContents(window!.webContents).upload()
if (img !== false) { if (img !== false) {
const pasteStyle = db.get('settings.pasteStyle') || 'markdown' const pasteStyle = db.get('settings.pasteStyle') || 'markdown'
clipboard.writeText(pasteTemplate(pasteStyle, img[0])) clipboard.writeText(pasteTemplate(pasteStyle, img[0]))
@ -536,7 +539,7 @@ ipcMain.on('syncPicBed', () => {
}) })
ipcMain.on('getPicBeds', (evt: IpcMainEvent) => { ipcMain.on('getPicBeds', (evt: IpcMainEvent) => {
const picBeds = getPicBeds(app) const picBeds = getPicBeds()
evt.sender.send('getPicBeds', picBeds) evt.sender.send('getPicBeds', picBeds)
evt.returnValue = picBeds evt.returnValue = picBeds
}) })

View File

@ -6,6 +6,8 @@ import ElementUI from 'element-ui'
import { webFrame } from 'electron' import { webFrame } from 'electron'
import 'element-ui/lib/theme-chalk/index.css' import 'element-ui/lib/theme-chalk/index.css'
import VueLazyLoad from 'vue-lazyload' import VueLazyLoad from 'vue-lazyload'
import axios from 'axios'
import mainMixin from './renderer/utils/mainMixin'
webFrame.setVisualZoomLevelLimits(1, 1) webFrame.setVisualZoomLevelLimits(1, 1)
webFrame.setLayoutZoomLevelLimits(0, 0) webFrame.setLayoutZoomLevelLimits(0, 0)
@ -22,9 +24,11 @@ Vue.prototype.$builtInPicBed = [
'github' 'github'
] ]
Vue.prototype.$db = db Vue.prototype.$db = db
Vue.prototype.$http = axios
Vue.use(ElementUI) Vue.use(ElementUI)
Vue.use(VueLazyLoad) Vue.use(VueLazyLoad)
Vue.mixin(mainMixin)
new Vue({ new Vue({
router, router,

View File

@ -1,14 +1,7 @@
import path from 'path'
import db from '#/datastore' import db from '#/datastore'
import { App } from 'electron' import picgo from './picgo'
// eslint-disable-next-line
const requireFunc = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require
const getPicBeds = (app: App) => { const getPicBeds = () => {
const PicGo = requireFunc('picgo')
const STORE_PATH = app.getPath('userData')
const CONFIG_PATH = path.join(STORE_PATH, '/data.json')
const picgo = new PicGo(CONFIG_PATH)
const picBedTypes = picgo.helper.uploader.getIdList() const picBedTypes = picgo.helper.uploader.getIdList()
const picBedFromDB = db.get('picBed.list') || [] const picBedFromDB = db.get('picBed.list') || []
const picBeds = picBedTypes.map((item: string) => { const picBeds = picBedTypes.map((item: string) => {

View File

@ -7,7 +7,7 @@ import {
ipcMain ipcMain
} from 'electron' } from 'electron'
import db from '#/datastore' import db from '#/datastore'
import Uploader from './uploader' import uploader from './uploader'
import pasteTemplate from '#/utils/pasteTemplate' import pasteTemplate from '#/utils/pasteTemplate'
const WEBCONTENTS = Symbol('WEBCONTENTS') const WEBCONTENTS = Symbol('WEBCONTENTS')
@ -52,7 +52,7 @@ class GuiApi implements IGuiApi {
} }
async upload (input: IUploadOption) { async upload (input: IUploadOption) {
const imgs = await new Uploader(input, this[WEBCONTENTS]).upload() const imgs = await uploader.setWebContents(this[WEBCONTENTS]).upload(input)
if (imgs !== false) { if (imgs !== false) {
const pasteStyle = db.get('settings.pasteStyle') || 'markdown' const pasteStyle = db.get('settings.pasteStyle') || 'markdown'
let pasteText = '' let pasteText = ''

19
src/main/utils/picgo.ts Normal file
View File

@ -0,0 +1,19 @@
import PicGoCore from '~/universal/types/picgo'
import {
app
} from 'electron'
import path from 'path'
// eslint-disable-next-line
const requireFunc = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require
const PicGo = requireFunc('picgo') as typeof PicGoCore
const STORE_PATH = app.getPath('userData')
const CONFIG_PATH = path.join(STORE_PATH, '/data.json')
const picgo = new PicGo(CONFIG_PATH)
picgo.setConfig({
debug: true,
PICGO_ENV: 'GUI'
})
export default picgo as PicGoCore

View File

@ -1,14 +1,24 @@
import path from 'path' import path from 'path'
import GuiApi from './guiApi' import GuiApi from './guiApi'
import { dialog, shell, IpcMain, IpcMainEvent, App } from 'electron' import {
import db from '#/datastore' dialog,
shell,
IpcMain,
IpcMainEvent,
App,
ipcMain,
app
} from 'electron'
import PicGoCore from '~/universal/types/picgo' import PicGoCore from '~/universal/types/picgo'
import { IPicGoHelperType } from '#/types/enum' import { IPicGoHelperType } from '#/types/enum'
import shortKeyHandler from './shortKeyHandler'
import picgo from '~/main/utils/picgo'
// eslint-disable-next-line // eslint-disable-next-line
const requireFunc = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require const requireFunc = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require
const PicGo = requireFunc('picgo') as typeof PicGoCore // const PluginHandler = requireFunc('picgo/dist/lib/PluginHandler').default
const PluginHandler = requireFunc('picgo/dist/lib/PluginHandler').default const STORE_PATH = app.getPath('userData')
// const CONFIG_PATH = path.join(STORE_PATH, '/data.json')
type PicGoNotice = { type PicGoNotice = {
title: string, title: string,
@ -48,9 +58,8 @@ const handleConfigWithFunction = (config: any[]) => {
return config return config
} }
const handleGetPluginList = (ipcMain: IpcMain, STORE_PATH: string, CONFIG_PATH: string) => { const handleGetPluginList = () => {
ipcMain.on('getPluginList', (event: IpcMainEvent) => { ipcMain.on('getPluginList', (event: IpcMainEvent) => {
const picgo = new PicGo(CONFIG_PATH)
const pluginList = picgo.pluginLoader.getList() const pluginList = picgo.pluginLoader.getList()
const list = [] const list = []
for (let i in pluginList) { for (let i in pluginList) {
@ -102,48 +111,49 @@ const handleGetPluginList = (ipcMain: IpcMain, STORE_PATH: string, CONFIG_PATH:
}) })
} }
const handlePluginInstall = (ipcMain: IpcMain, CONFIG_PATH: string) => { const handlePluginInstall = () => {
ipcMain.on('installPlugin', async (event: IpcMainEvent, msg: string) => { ipcMain.on('installPlugin', async (event: IpcMainEvent, msg: string) => {
const picgo = new PicGo(CONFIG_PATH) picgo.once('installSuccess', (notice: PicGoNotice) => {
const pluginHandler = new PluginHandler(picgo)
picgo.on('installSuccess', (notice: PicGoNotice) => {
event.sender.send('installSuccess', notice.body[0].replace(/picgo-plugin-/, '')) event.sender.send('installSuccess', notice.body[0].replace(/picgo-plugin-/, ''))
shortKeyHandler.registerPluginShortKey(notice.body[0])
picgo.removeAllListeners('installFailed')
}) })
picgo.on('failed', () => { picgo.once('installFailed', () => {
handleNPMError() handleNPMError()
picgo.removeAllListeners('installSuccess')
}) })
await pluginHandler.uninstall([msg]) await picgo.pluginHandler.install([msg])
pluginHandler.install([msg])
picgo.cmd.program.removeAllListeners() picgo.cmd.program.removeAllListeners()
}) })
} }
const handlePluginUninstall = (ipcMain: IpcMain, CONFIG_PATH: string) => { const handlePluginUninstall = () => {
ipcMain.on('uninstallPlugin', async (event: IpcMainEvent, msg: string) => { ipcMain.on('uninstallPlugin', async (event: IpcMainEvent, msg: string) => {
const picgo = new PicGo(CONFIG_PATH) picgo.once('uninstallSuccess', (notice: PicGoNotice) => {
const pluginHandler = new PluginHandler(picgo)
picgo.on('uninstallSuccess', (notice: PicGoNotice) => {
event.sender.send('uninstallSuccess', notice.body[0].replace(/picgo-plugin-/, '')) event.sender.send('uninstallSuccess', notice.body[0].replace(/picgo-plugin-/, ''))
shortKeyHandler.unregisterPluginShortKey(notice.body[0])
picgo.removeAllListeners('uninstallFailed')
}) })
picgo.on('failed', () => { picgo.once('uninstallFailed', () => {
handleNPMError() handleNPMError()
picgo.removeAllListeners('uninstallSuccess')
}) })
await pluginHandler.uninstall([msg]) await picgo.pluginHandler.uninstall([msg])
picgo.cmd.program.removeAllListeners() picgo.cmd.program.removeAllListeners()
}) })
} }
const handlePluginUpdate = (ipcMain: IpcMain, CONFIG_PATH: string) => { const handlePluginUpdate = () => {
ipcMain.on('updatePlugin', async (event: IpcMainEvent, msg: string) => { ipcMain.on('updatePlugin', async (event: IpcMainEvent, msg: string) => {
const picgo = new PicGo(CONFIG_PATH) picgo.once('updateSuccess', (notice: { body: string[], title: string }) => {
const pluginHandler = new PluginHandler(picgo)
picgo.on('updateSuccess', notice => {
event.sender.send('updateSuccess', notice.body[0].replace(/picgo-plugin-/, '')) event.sender.send('updateSuccess', notice.body[0].replace(/picgo-plugin-/, ''))
picgo.removeAllListeners('updateFailed')
}) })
picgo.on('failed', () => { picgo.once('updateFailed', () => {
handleNPMError() handleNPMError()
picgo.removeAllListeners('updateSuccess')
}) })
await pluginHandler.update([msg]) await picgo.pluginHandler.update([msg])
picgo.cmd.program.removeAllListeners() picgo.cmd.program.removeAllListeners()
}) })
} }
@ -160,9 +170,8 @@ const handleNPMError = () => {
}) })
} }
const handleGetPicBedConfig = (ipcMain: IpcMain, CONFIG_PATH: string) => { const handleGetPicBedConfig = () => {
ipcMain.on('getPicBedConfig', (event: IpcMainEvent, type: string) => { ipcMain.on('getPicBedConfig', (event: IpcMainEvent, type: string) => {
const picgo = new PicGo(CONFIG_PATH)
const name = picgo.helper.uploader.get(type).name || type const name = picgo.helper.uploader.get(type).name || type
if (picgo.helper.uploader.get(type).config) { if (picgo.helper.uploader.get(type).config) {
const config = handleConfigWithFunction(picgo.helper.uploader.get(type).config(picgo)) const config = handleConfigWithFunction(picgo.helper.uploader.get(type).config(picgo))
@ -174,9 +183,8 @@ const handleGetPicBedConfig = (ipcMain: IpcMain, CONFIG_PATH: string) => {
}) })
} }
const handlePluginActions = (ipcMain: IpcMain, CONFIG_PATH: string) => { const handlePluginActions = () => {
ipcMain.on('pluginActions', (event: IpcMainEvent, name: string, label: string) => { ipcMain.on('pluginActions', (event: IpcMainEvent, name: string, label: string) => {
const picgo = new PicGo(CONFIG_PATH)
const plugin = picgo.pluginLoader.getPlugin(`picgo-plugin-${name}`) const plugin = picgo.pluginLoader.getPlugin(`picgo-plugin-${name}`)
const guiApi = new GuiApi(event.sender) const guiApi = new GuiApi(event.sender)
if (plugin.guiMenu && plugin.guiMenu(picgo).length > 0) { if (plugin.guiMenu && plugin.guiMenu(picgo).length > 0) {
@ -190,9 +198,8 @@ const handlePluginActions = (ipcMain: IpcMain, CONFIG_PATH: string) => {
}) })
} }
const handleRemoveFiles = (ipcMain: IpcMain, CONFIG_PATH: string) => { const handleRemoveFiles = () => {
ipcMain.on('removeFiles', (event: IpcMainEvent, files: ImgInfo[]) => { ipcMain.on('removeFiles', (event: IpcMainEvent, files: ImgInfo[]) => {
const picgo = new PicGo(CONFIG_PATH)
const guiApi = new GuiApi(event.sender) const guiApi = new GuiApi(event.sender)
setTimeout(() => { setTimeout(() => {
picgo.emit('remove', files, guiApi) picgo.emit('remove', files, guiApi)
@ -200,26 +207,19 @@ const handleRemoveFiles = (ipcMain: IpcMain, CONFIG_PATH: string) => {
}) })
} }
// const handlePluginShortKeyRegister = (plugin) => { const handlePicGoSaveData = () => {
// if (plugin.shortKeys && plugin.shortKeys.length > 0) { ipcMain.on('picgoSaveData', (event: IpcMainEvent, data: IObj) => {
// let shortKeyConfig = db.get('settings.shortKey') picgo.saveConfig(data)
// plugin.shortKeys.forEach(item => { })
// if (!shortKeyConfig[item.name]) { }
// shortKeyConfig[item.name] = item
// } export default () => {
// }) handleGetPluginList()
// } handlePluginInstall()
// } handlePluginUninstall()
handlePluginUpdate()
export default (app: App, ipcMain: IpcMain) => { handleGetPicBedConfig()
const STORE_PATH = app.getPath('userData') handlePluginActions()
const CONFIG_PATH = path.join(STORE_PATH, '/data.json') handleRemoveFiles()
handleGetPluginList(ipcMain, STORE_PATH, CONFIG_PATH) handlePicGoSaveData()
handlePluginInstall(ipcMain, CONFIG_PATH)
handlePluginUninstall(ipcMain, CONFIG_PATH)
handlePluginUpdate(ipcMain, CONFIG_PATH)
handleGetPicBedConfig(ipcMain, CONFIG_PATH)
handlePluginActions(ipcMain, CONFIG_PATH)
handleRemoveFiles(ipcMain, CONFIG_PATH)
// handlePluginShortKeyRegister({})
} }

View File

@ -10,12 +10,7 @@ import logger from './logger'
import GuiApi from './guiApi' import GuiApi from './guiApi'
import db from '#/datastore' import db from '#/datastore'
import shortKeyService from './shortKeyService' import shortKeyService from './shortKeyService'
// eslint-disable-next-line import picgo from './picgo'
const requireFunc = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require
const PicGo = requireFunc('picgo') as typeof PicGoCore
const STORE_PATH = app.getPath('userData')
const CONFIG_PATH = path.join(STORE_PATH, '/data.json')
class ShortKeyHandler { class ShortKeyHandler {
private isInModifiedMode: boolean = false private isInModifiedMode: boolean = false
@ -40,7 +35,6 @@ class ShortKeyHandler {
}) })
} }
private initPluginsShortKey () { private initPluginsShortKey () {
const picgo = new PicGo(CONFIG_PATH)
const pluginList = picgo.pluginLoader.getList() const pluginList = picgo.pluginLoader.getList()
for (let item of pluginList) { for (let item of pluginList) {
const plugin = picgo.pluginLoader.getPlugin(item) const plugin = picgo.pluginLoader.getPlugin(item)
@ -75,12 +69,14 @@ class ShortKeyHandler {
logger.warn(`${command} do not provide a key to bind`) logger.warn(`${command} do not provide a key to bind`)
} }
if (writeFlag) { if (writeFlag) {
db.set(`settings.shortKey.${command}`, { picgo.saveConfig({
[`settings.shortKey.${command}`]: {
enable: true, enable: true,
name: config.name, name: config.name,
label: config.label, label: config.label,
key: config.key key: config.key
} as IShortKeyConfig) }
})
} }
} }
// enable or disable shortKey // enable or disable shortKey
@ -88,13 +84,17 @@ class ShortKeyHandler {
const command = `${from}:${item.name}` const command = `${from}:${item.name}`
if (item.enable === false) { if (item.enable === false) {
globalShortcut.unregister(item.key) globalShortcut.unregister(item.key)
db.set(`settings.shortKey.${command}.enable`, false) picgo.saveConfig({
[`settings.shortKey.${command}.enable`]: false
})
return true return true
} else { } else {
if (globalShortcut.isRegistered(item.key)) { if (globalShortcut.isRegistered(item.key)) {
return false return false
} else { } else {
db.set(`settings.shortKey.${command}.enable`, true) picgo.saveConfig({
[`settings.shortKey.${command}.enable`]: true
})
globalShortcut.register(item.key, () => { globalShortcut.register(item.key, () => {
this.handler(command) this.handler(command)
}) })
@ -107,7 +107,9 @@ class ShortKeyHandler {
const command = `${from}:${item.name}` const command = `${from}:${item.name}`
if (globalShortcut.isRegistered(item.key)) return false if (globalShortcut.isRegistered(item.key)) return false
globalShortcut.unregister(oldKey) globalShortcut.unregister(oldKey)
db.set(`settings.shortKey.${command}.key`, item.key) picgo.saveConfig({
[`settings.shortKey.${command}.key`]: item.key
})
globalShortcut.register(item.key, () => { globalShortcut.register(item.key, () => {
this.handler(`${from}:${item.name}`) this.handler(`${from}:${item.name}`)
}) })
@ -122,7 +124,6 @@ class ShortKeyHandler {
} else if (command.includes('picgo-plugin-')) { } else if (command.includes('picgo-plugin-')) {
const handler = shortKeyService.getShortKeyHandler(command) const handler = shortKeyService.getShortKeyHandler(command)
if (handler) { if (handler) {
const picgo = new PicGo(CONFIG_PATH)
// make sure settingWindow is created // make sure settingWindow is created
bus.once('createSettingWindowDone', (cmd: string, settingWindowId: number) => { bus.once('createSettingWindowDone', (cmd: string, settingWindowId: number) => {
if (cmd === command) { if (cmd === command) {
@ -138,7 +139,6 @@ class ShortKeyHandler {
} }
} }
registerPluginShortKey (pluginName: string) { registerPluginShortKey (pluginName: string) {
const picgo = new PicGo(CONFIG_PATH)
const plugin = picgo.pluginLoader.getPlugin(pluginName) const plugin = picgo.pluginLoader.getPlugin(pluginName)
if (plugin && plugin.commands) { if (plugin && plugin.commands) {
if (typeof plugin.commands !== 'function') { if (typeof plugin.commands !== 'function') {
@ -170,7 +170,7 @@ class ShortKeyHandler {
keyList.forEach(item => { keyList.forEach(item => {
globalShortcut.unregister(item.key) globalShortcut.unregister(item.key)
shortKeyService.unregisterCommand(item.command) shortKeyService.unregisterCommand(item.command)
db.unset('settings.shortKey', item.command) picgo.unsetConfig('settings.shortKey', item.command)
}) })
} }
} }

View File

@ -3,13 +3,8 @@ import db from '#/datastore'
import axios from 'axios' import axios from 'axios'
import pkg from 'root/package.json' import pkg from 'root/package.json'
const version = pkg.version const version = pkg.version
let release = 'https://api.github.com/repos/Molunerfinn/PicGo/releases/latest' let release = 'https://cdn.jsdelivr.net/gh/Molunerfinn/PicGo/package.json'
const downloadUrl = 'https://github.com/Molunerfinn/PicGo/releases/latest' const downloadUrl = 'https://github.com/Molunerfinn/PicGo/releases/latest'
const isDevelopment = process.env.NODE_ENV !== 'production'
if (isDevelopment) {
release = `${release}?access_token=${process.env.GITHUB_TOKEN}`
}
const checkVersion = async () => { const checkVersion = async () => {
let showTip = db.get('settings.showUpdateTip') let showTip = db.get('settings.showUpdateTip')
@ -25,7 +20,7 @@ const checkVersion = async () => {
console.log(err) console.log(err)
} }
if (res.status === 200) { if (res.status === 200) {
const latest = res.data.name const latest = res.data.version
const result = compareVersion2Update(version, latest) const result = compareVersion2Update(version, latest)
if (result) { if (result) {
dialog.showMessageBox({ dialog.showMessageBox({

View File

@ -5,37 +5,15 @@ import {
ipcMain, ipcMain,
WebContents WebContents
} from 'electron' } from 'electron'
import path from 'path'
import dayjs from 'dayjs' import dayjs from 'dayjs'
import PicGoCore from '~/universal/types/picgo' import picgo from '~/main/utils/picgo'
import db from '#/datastore'
// eslint-disable-next-line
const requireFunc = typeof __webpack_require__ === 'function' ? __non_webpack_require__ : require
const PicGo = requireFunc('picgo') as typeof PicGoCore
const STORE_PATH = app.getPath('userData')
const CONFIG_PATH = path.join(STORE_PATH, '/data.json')
const renameURL = process.env.NODE_ENV === 'development' const renameURL = process.env.NODE_ENV === 'development'
? `${(process.env.WEBPACK_DEV_SERVER_URL as string)}#rename-page` ? `${(process.env.WEBPACK_DEV_SERVER_URL as string)}#rename-page`
: `picgo://./index.html#rename-page` : `picgo://./index.html#rename-page`
// type BrowserWindowOptions = { const createRenameWindow = (currentWindow: BrowserWindow) => {
// height: number,
// width: number,
// show: boolean,
// fullscreenable: boolean,
// resizable: boolean,
// vibrancy: string | any,
// webPreferences: {
// nodeIntegration: boolean,
// nodeIntegrationInWorker: boolean,
// backgroundThrottling: boolean
// },
// backgroundColor?: string
// autoHideMenuBar?: boolean
// transparent?: boolean
// }
const createRenameWindow = (win: BrowserWindow) => {
let options: IBrowserWindowOptions = { let options: IBrowserWindowOptions = {
height: 175, height: 175,
width: 300, width: 300,
@ -60,9 +38,9 @@ const createRenameWindow = (win: BrowserWindow) => {
const window = new BrowserWindow(options) const window = new BrowserWindow(options)
window.loadURL(renameURL) window.loadURL(renameURL)
// check if this window is visible // check if this window is visible
if (win.isVisible()) { if (currentWindow && currentWindow.isVisible()) {
// bounds: { x: 821, y: 75, width: 800, height: 450 } // bounds: { x: 821, y: 75, width: 800, height: 450 }
const bounds = win.getBounds() const bounds = currentWindow.getBounds()
const positionX = bounds.x + bounds.width / 2 - 150 const positionX = bounds.x + bounds.width / 2 - 150
let positionY let positionY
// if is the settingWindow // if is the settingWindow
@ -98,27 +76,34 @@ const waitForRename = (window: BrowserWindow, id: number): Promise<string|null>
} }
class Uploader { class Uploader {
private picgo: PicGoCore private webContents: WebContents | null = null
private webContents: WebContents private currentWindow: BrowserWindow | null = null
private img: undefined | string[] constructor () {
constructor (img: IUploadOption, webContents: WebContents, picgo: PicGoCore | undefined = undefined) { this.init()
this.img = img
this.webContents = webContents
this.picgo = picgo || new PicGo(CONFIG_PATH)
} }
upload (): Promise<ImgInfo[]|false> { init () {
const win = BrowserWindow.fromWebContents(this.webContents) picgo.on('notification', message => {
const picgo = this.picgo const notification = new Notification(message)
picgo.config.debug = true notification.show()
// for picgo-core })
picgo.config.PICGO_ENV = 'GUI'
let input = this.img
picgo.on('uploadProgress', progress => {
this.webContents!.send('uploadProgress', progress)
})
picgo.on('beforeTransform', ctx => {
if (db.get('settings.uploadNotification')) {
const notification = new Notification({
title: '上传进度',
body: '正在上传'
})
notification.show()
}
})
picgo.helper.beforeUploadPlugins.register('renameFn', { picgo.helper.beforeUploadPlugins.register('renameFn', {
handle: async ctx => { handle: async ctx => {
const rename = picgo.getConfig('settings.rename') const rename = db.get('settings.rename')
const autoRename = picgo.getConfig('settings.autoRename') const autoRename = db.get('settings.autoRename')
await Promise.all(ctx.output.map(async (item, index) => { await Promise.all(ctx.output.map(async (item, index) => {
let name: undefined | string | null let name: undefined | string | null
let fileName: string | undefined let fileName: string | undefined
@ -128,7 +113,7 @@ class Uploader {
fileName = item.fileName fileName = item.fileName
} }
if (rename) { if (rename) {
const window = createRenameWindow(win) const window = createRenameWindow(this.currentWindow!)
await waitForShow(window.webContents) await waitForShow(window.webContents)
window.webContents.send('rename', fileName, window.webContents.id) window.webContents.send('rename', fileName, window.webContents.id)
name = await waitForRename(window, window.webContents.id) name = await waitForRename(window, window.webContents.id)
@ -137,46 +122,40 @@ class Uploader {
})) }))
} }
}) })
picgo.on('beforeTransform', ctx => {
if (ctx.getConfig('settings.uploadNotification')) {
const notification = new Notification({
title: '上传进度',
body: '正在上传'
})
notification.show()
} }
})
picgo.upload(input) setWebContents (webContents: WebContents) {
this.webContents = webContents
return this
}
picgo.on('notification', message => { upload (img?: IUploadOption): Promise<ImgInfo[]|false> {
const notification = new Notification(message) this.currentWindow = BrowserWindow.fromWebContents(this.webContents!)
notification.show()
})
picgo.on('uploadProgress', progress => { picgo.upload(img)
this.webContents.send('uploadProgress', progress)
})
return new Promise((resolve) => { return new Promise((resolve) => {
picgo.on('finished', ctx => { picgo.once('finished', ctx => {
if (ctx.output.every((item: ImgInfo) => item.imgUrl)) { if (ctx.output.every((item: ImgInfo) => item.imgUrl)) {
resolve(ctx.output) resolve(ctx.output)
} else { } else {
resolve(false) resolve(false)
} }
picgo.removeAllListeners('failed')
}) })
picgo.on('failed', ctx => { picgo.once('failed', ctx => {
setTimeout(() => {
const notification = new Notification({ const notification = new Notification({
title: '上传失败', title: '上传失败',
body: '请检查配置和上传的文件是否符合要求' body: '请检查配置和上传的文件是否符合要求'
}) })
notification.show() notification.show()
}, 500)
picgo.removeAllListeners('finished')
resolve(false) resolve(false)
}) })
}) })
} }
} }
export default Uploader export default new Uploader()

View File

@ -27,7 +27,9 @@ export default {
}, },
methods: { methods: {
choosePicBed (val) { choosePicBed (val) {
this.$db.set('picBed.current', this.type) this.letPicGoSaveData({
'picBed.current': this.type
})
this.$emit('update:choosed', this.type) this.$emit('update:choosed', this.type)
} }
} }

View File

@ -143,7 +143,9 @@ export default class extends Vue {
type: 'radio', type: 'radio',
checked: this.$db.get('picBed.current') === item.type, checked: this.$db.get('picBed.current') === item.type,
click () { click () {
_this.$db.set('picBed.current', item.type) _this.letPicGoSaveData({
'picBed.current': item.type
})
ipcRenderer.send('syncPicBed') ipcRenderer.send('syncPicBed')
} }
} }

View File

@ -371,7 +371,9 @@ export default class extends Vue {
} }
confirmProxy () { confirmProxy () {
this.proxyVisible = false this.proxyVisible = false
db.set('picBed.proxy', this.proxy) this.letPicGoSaveData({
'picBed.proxy': this.proxy
})
const successNotification = new Notification('设置代理', { const successNotification = new Notification('设置代理', {
body: '设置成功' body: '设置成功'
}) })
@ -391,7 +393,9 @@ export default class extends Vue {
} }
return item return item
}) })
db.set('picBed.list', list) this.letPicGoSaveData({
'picBed.list': list
})
ipcRenderer.send('getPicBeds') ipcRenderer.send('getPicBeds')
} }
handleAutoStartChange (val: boolean) { handleAutoStartChange (val: boolean) {
@ -399,10 +403,14 @@ export default class extends Vue {
ipcRenderer.send('autoStart', val) ipcRenderer.send('autoStart', val)
} }
handleRename (val: boolean) { handleRename (val: boolean) {
db.set('settings.rename', val) this.letPicGoSaveData({
'settings.rename': val
})
} }
handleAutoRename (val: boolean) { handleAutoRename (val: boolean) {
db.set('settings.autoRename', val) this.letPicGoSaveData({
'settings.autoRename': val
})
} }
compareVersion2Update (current: string, latest: string) { compareVersion2Update (current: string, latest: string) {
const currentVersion = current.split('.').map(item => parseInt(item)) const currentVersion = current.split('.').map(item => parseInt(item))
@ -447,7 +455,9 @@ export default class extends Vue {
if (this.form.logLevel.length === 0) { if (this.form.logLevel.length === 0) {
return this.$message.error('请选择日志记录等级') return this.$message.error('请选择日志记录等级')
} }
db.set('settings.logLevel', this.form.logLevel) this.letPicGoSaveData({
'settings.logLevel': this.form.logLevel
})
const successNotification = new Notification('设置日志', { const successNotification = new Notification('设置日志', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -199,7 +199,9 @@ export default class extends Vue {
label: '启用插件', label: '启用插件',
enabled: !plugin.enabled, enabled: !plugin.enabled,
click () { click () {
_this.$db.set(`picgoPlugins.picgo-plugin-${plugin.name}`, true) _this.letPicGoSaveData({
[`picgoPlugins.picgo-plugin-${plugin.name}`]: true
})
plugin.enabled = true plugin.enabled = true
_this.getPicBeds() _this.getPicBeds()
} }
@ -207,7 +209,9 @@ export default class extends Vue {
label: '禁用插件', label: '禁用插件',
enabled: plugin.enabled, enabled: plugin.enabled,
click () { click () {
_this.$db.set(`picgoPlugins.picgo-plugin-${plugin.name}`, false) _this.letPicGoSaveData({
[`picgoPlugins.picgo-plugin-${plugin.name}`]: false
})
plugin.enabled = false plugin.enabled = false
_this.getPicBeds() _this.getPicBeds()
if (plugin.config.transformer.name) { if (plugin.config.transformer.name) {
@ -334,9 +338,13 @@ export default class extends Vue {
toggleTransformer (transformer: string) { toggleTransformer (transformer: string) {
let currentTransformer = this.$db.get('picBed.transformer') || 'path' let currentTransformer = this.$db.get('picBed.transformer') || 'path'
if (currentTransformer === transformer) { if (currentTransformer === transformer) {
this.$db.set('picBed.transformer', 'path') this.letPicGoSaveData({
'picBed.transformer': 'path'
})
} else { } else {
this.$db.set('picBed.transformer', transformer) this.letPicGoSaveData({
'picBed.transformer': transformer
})
} }
} }
async handleConfirmConfig () { async handleConfirmConfig () {
@ -345,13 +353,19 @@ export default class extends Vue {
if (result !== false) { if (result !== false) {
switch (this.currentType) { switch (this.currentType) {
case 'plugin': case 'plugin':
this.$db.set(`picgo-plugin-${this.configName}`, result) this.letPicGoSaveData({
[`picgo-plugin-${this.configName}`]: result
})
break break
case 'uploader': case 'uploader':
this.$db.set(`picBed.${this.configName}`, result) this.letPicGoSaveData({
[`picBed.${this.configName}`]: result
})
break break
case 'transformer': case 'transformer':
this.$db.set(`transformer.${this.configName}`, result) this.letPicGoSaveData({
[`transformer.${this.configName}`]: result
})
break break
} }
const successNotification = new Notification('设置结果', { const successNotification = new Notification('设置结果', {
@ -368,7 +382,11 @@ export default class extends Vue {
// this.$http.get(`https://api.npms.io/v2/search?q=${val}`) // this.$http.get(`https://api.npms.io/v2/search?q=${val}`)
this.$http.get(`https://registry.npmjs.com/-/v1/search?text=${val}`) this.$http.get(`https://registry.npmjs.com/-/v1/search?text=${val}`)
.then((res: INPMSearchResult) => { .then((res: INPMSearchResult) => {
this.pluginList = res.data.objects.map((item: INPMSearchResultObject) => { this.pluginList = res.data.objects
.filter((item:INPMSearchResultObject) => {
return item.package.name.includes('picgo-plugin-')
})
.map((item: INPMSearchResultObject) => {
return this.handleSearchResult(item) return this.handleSearchResult(item)
}) })
this.loading = false this.loading = false
@ -404,13 +422,17 @@ export default class extends Vue {
if (item === 'uploader') { if (item === 'uploader') {
const current = this.$db.get('picBed.current') const current = this.$db.get('picBed.current')
if (current === name) { if (current === name) {
this.$db.set('picBed.current', 'smms') this.letPicGoSaveData({
'picBed.current': 'smms'
})
} }
} }
if (item === 'transformer') { if (item === 'transformer') {
const current = this.$db.get('picBed.transformer') const current = this.$db.get('picBed.transformer')
if (current === name) { if (current === name) {
this.$db.set('picBed.transformer', 'path') this.letPicGoSaveData({
'picBed.transformer': 'path'
})
} }
} }
} }
@ -422,6 +444,9 @@ export default class extends Vue {
goAwesomeList () { goAwesomeList () {
remote.shell.openExternal('https://github.com/PicGo/Awesome-PicGo') remote.shell.openExternal('https://github.com/PicGo/Awesome-PicGo')
} }
letPicGoSaveData (data: IObj) {
ipcRenderer.send('picgoSaveData', data)
}
beforeDestroy () { beforeDestroy () {
ipcRenderer.removeAllListeners('pluginList') ipcRenderer.removeAllListeners('pluginList')
ipcRenderer.removeAllListeners('installSuccess') ipcRenderer.removeAllListeners('installSuccess')

View File

@ -166,7 +166,9 @@ export default class extends Vue {
type: 'radio', type: 'radio',
checked: this.$db.get('picBed.current') === item.type, checked: this.$db.get('picBed.current') === item.type,
click () { click () {
_this.$db.set('picBed.current', item.type) _this.letPicGoSaveData({
'picBed.current': item.type
})
ipcRenderer.send('syncPicBed') ipcRenderer.send('syncPicBed')
} }
} }

View File

@ -90,7 +90,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
this.$refs.aliyun.validate((valid) => { this.$refs.aliyun.validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.aliyun', this.form) this.letPicGoSaveData({
'picBed.aliyun': this.form
})
const successNotification = new window.Notification('设置结果', { const successNotification = new window.Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -81,7 +81,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
this.$refs.github.validate((valid) => { this.$refs.github.validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.github', this.form) this.letPicGoSaveData({
'picBed.github': this.form
})
const successNotification = new Notification('设置结果', { const successNotification = new Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -58,7 +58,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
this.$refs.imgur.validate((valid) => { this.$refs.imgur.validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.imgur', this.form) this.letPicGoSaveData({
'picBed.imgur': this.form
})
const successNotification = new Notification('设置结果', { const successNotification = new Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -56,7 +56,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
const result = await this.$refs.configForm.validate() const result = await this.$refs.configForm.validate()
if (result !== false) { if (result !== false) {
this.$db.set(`picBed.${this.type}`, result) this.letPicGoSaveData({
[`picBed.${this.type}`]: result
})
const successNotification = new Notification('设置结果', { const successNotification = new Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })
@ -66,7 +68,9 @@ export default class extends Vue {
} }
} }
setDefaultPicBed (type: string) { setDefaultPicBed (type: string) {
this.$db.set('picBed.current', type) this.letPicGoSaveData({
'picBed.current': type
})
// @ts-ignore mixin // @ts-ignore mixin
this.defaultPicBed = type this.defaultPicBed = type
const successNotification = new Notification('设置默认图床', { const successNotification = new Notification('设置默认图床', {

View File

@ -98,7 +98,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
this.$refs.qiniu.validate((valid) => { this.$refs.qiniu.validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.qiniu', this.form) this.letPicGoSaveData({
'picBed.qiniu': this.form
})
const successNotification = new Notification('设置结果', { const successNotification = new Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -24,7 +24,9 @@ import mixin from '@/utils/ConfirmButtonMixin'
}) })
export default class extends Vue { export default class extends Vue {
confirm () { confirm () {
this.$db.set('picBed.smms', true) this.letPicGoSaveData({
'picBed.smms': true
})
// @ts-ignore mixin // @ts-ignore mixin
this.setDefaultPicBed('smms') this.setDefaultPicBed('smms')
const successNotification = new window.Notification('设置结果', { const successNotification = new window.Notification('设置结果', {

View File

@ -114,7 +114,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
this.$refs.tcyun.validate((valid) => { this.$refs.tcyun.validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.tcyun', this.form) this.letPicGoSaveData({
'picBed.tcyun': this.form
})
const successNotification = new window.Notification('设置结果', { const successNotification = new window.Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -89,7 +89,9 @@ export default class extends Vue {
// @ts-ignore // @ts-ignore
this.$refs.tcyun.validate((valid) => { this.$refs.tcyun.validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.upyun', this.form) this.letPicGoSaveData({
'picBed.upyun': this.form
})
const successNotification = new Notification('设置结果', { const successNotification = new Notification('设置结果', {
body: '设置成功' body: '设置成功'
}) })

View File

@ -91,12 +91,14 @@ export default {
confirm (formName) { confirm (formName) {
this.$refs[formName].validate((valid) => { this.$refs[formName].validate((valid) => {
if (valid) { if (valid) {
this.$db.set('picBed.weibo', { this.letPicGoSaveData({
'picBed.weibo': {
username: this.form.username, username: this.form.username,
password: this.form.password, password: this.form.password,
quality: this.quality, quality: this.quality,
cookie: this.form.cookie, cookie: this.form.cookie,
chooseCookie: this.chooseCookie chooseCookie: this.chooseCookie
}
}) })
const successNotification = new window.Notification('设置结果', { const successNotification = new window.Notification('设置结果', {
body: '设置成功' body: '设置成功'

View File

@ -1,9 +1,12 @@
import { Component, Vue } from 'vue-property-decorator' import { Component, Vue } from 'vue-property-decorator'
import { ipcRenderer } from 'electron'
@Component @Component
export default class extends Vue { export default class extends Vue {
defaultPicBed = this.$db.get('picBed.current') defaultPicBed = this.$db.get('picBed.current')
setDefaultPicBed (type: string) { setDefaultPicBed (type: string) {
this.$db.set('picBed.current', type) this.letPicGoSaveData({
'picBed.current': type
})
this.defaultPicBed = type this.defaultPicBed = type
const successNotification = new Notification('设置默认图床', { const successNotification = new Notification('设置默认图床', {
body: '设置成功' body: '设置成功'

View File

@ -0,0 +1,8 @@
import { Component, Vue } from 'vue-property-decorator'
import { ipcRenderer } from 'electron'
@Component
export default class extends Vue {
letPicGoSaveData (data: IObj) {
ipcRenderer.send('picgoSaveData', data)
}
}

View File

@ -59,7 +59,7 @@ class DB {
return this.read().get(key).insert(value).write() return this.read().get(key).insert(value).write()
} }
unset (key: string, value: any): boolean { unset (key: string, value: any): boolean {
return this.read().get(key).unset(value).write() return this.read().get(key).unset(value).value()
} }
} }

View File

@ -8,5 +8,6 @@ declare module 'vue/types/vue' {
$db: typeof db $db: typeof db
$http: typeof axios $http: typeof axios
$builtInPicBed: string[] $builtInPicBed: string[]
letPicGoSaveData(data: IObj): void
} }
} }

View File

@ -162,7 +162,7 @@ interface IShowInputBoxOption {
type IShowFileExplorerOption = IObj type IShowFileExplorerOption = IObj
type IUploadOption = undefined | string[] type IUploadOption = string[]
interface IShowNotificationOption { interface IShowNotificationOption {
title: string title: string

View File

@ -2870,12 +2870,15 @@ commander@~2.8.1:
dependencies: dependencies:
graceful-readlink ">= 1.0.0" graceful-readlink ">= 1.0.0"
comment-json@^1.1.3: comment-json@^2.3.1:
version "1.1.3" version "2.3.1"
resolved "https://registry.yarnpkg.com/comment-json/-/comment-json-1.1.3.tgz#6986c3330fee0c4c9e00c2398cd61afa5d8f239e" resolved "https://registry.npm.taobao.org/comment-json/download/comment-json-2.3.1.tgz#40f24f573ffece9b9a87806244c330d614a32604"
integrity sha1-aYbDMw/uDEyeAMI5jNYa+l2PI54= integrity sha1-QPJPVz/+zpuah4BiRMMw1hSjJgQ=
dependencies: dependencies:
json-parser "^1.0.0" core-util-is "^1.0.2"
esprima "^4.0.1"
has-own-prop "^2.0.0"
repeat-string "^1.6.1"
commitizen@^4.0.3: commitizen@^4.0.3:
version "4.0.3" version "4.0.3"
@ -3262,7 +3265,7 @@ core-js@^3.3.2, core-js@^3.4.3:
resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.6.0.tgz#2b854e451de1967d1e29896025cdc13a2518d9ea" resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.6.0.tgz#2b854e451de1967d1e29896025cdc13a2518d9ea"
integrity sha512-AHPTNKzyB+YwgDWoSOCaid9PUSEF6781vsfiK8qUz62zRR448/XgK2NtCbpiUGizbep8Lrpt0Du19PpGGZvw3Q== integrity sha512-AHPTNKzyB+YwgDWoSOCaid9PUSEF6781vsfiK8qUz62zRR448/XgK2NtCbpiUGizbep8Lrpt0Du19PpGGZvw3Q==
core-util-is@1.0.2, core-util-is@~1.0.0: core-util-is@1.0.2, core-util-is@^1.0.2, core-util-is@~1.0.0:
version "1.0.2" version "1.0.2"
resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7"
integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac= integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=
@ -4534,12 +4537,7 @@ espree@^5.0.1:
acorn-jsx "^5.0.0" acorn-jsx "^5.0.0"
eslint-visitor-keys "^1.0.0" eslint-visitor-keys "^1.0.0"
esprima@^2.7.0: esprima@^4.0.0, esprima@^4.0.1:
version "2.7.3"
resolved "https://registry.yarnpkg.com/esprima/-/esprima-2.7.3.tgz#96e3b70d5779f6ad49cd032673d1c312767ba581"
integrity sha1-luO3DVd59q1JzQMmc9HDEnZ7pYE=
esprima@^4.0.0:
version "4.0.1" version "4.0.1"
resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71"
integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==
@ -5628,6 +5626,11 @@ has-flag@^4.0.0:
resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
has-own-prop@^2.0.0:
version "2.0.0"
resolved "https://registry.npm.taobao.org/has-own-prop/download/has-own-prop-2.0.0.tgz#f0f95d58f65804f5d218db32563bb85b8e0417af"
integrity sha1-8PldWPZYBPXSGNsyVju4W44EF68=
has-symbol-support-x@^1.4.1: has-symbol-support-x@^1.4.1:
version "1.4.2" version "1.4.2"
resolved "https://registry.yarnpkg.com/has-symbol-support-x/-/has-symbol-support-x-1.4.2.tgz#1409f98bc00247da45da67cee0a36f282ff26455" resolved "https://registry.yarnpkg.com/has-symbol-support-x/-/has-symbol-support-x-1.4.2.tgz#1409f98bc00247da45da67cee0a36f282ff26455"
@ -6661,13 +6664,6 @@ json-parse-better-errors@^1.0.1, json-parse-better-errors@^1.0.2:
resolved "https://registry.yarnpkg.com/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz#bb867cfb3450e69107c131d1c514bab3dc8bcaa9" resolved "https://registry.yarnpkg.com/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz#bb867cfb3450e69107c131d1c514bab3dc8bcaa9"
integrity sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw== integrity sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==
json-parser@^1.0.0:
version "1.1.5"
resolved "https://registry.yarnpkg.com/json-parser/-/json-parser-1.1.5.tgz#e62ec5261d1a6a5fc20e812a320740c6d9005677"
integrity sha1-5i7FJh0aal/CDoEqMgdAxtkAVnc=
dependencies:
esprima "^2.7.0"
json-schema-traverse@^0.4.1: json-schema-traverse@^0.4.1:
version "0.4.1" version "0.4.1"
resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660"
@ -8280,14 +8276,14 @@ performance-now@^2.1.0:
resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b" resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b"
integrity sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns= integrity sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=
picgo@^1.3.7: picgo@^1.4.2:
version "1.3.7" version "1.4.2"
resolved "https://registry.yarnpkg.com/picgo/-/picgo-1.3.7.tgz#b152ec8235062a5638243c20afbafe87bf295c47" resolved "https://registry.npm.taobao.org/picgo/download/picgo-1.4.2.tgz#3a9ca1fb3c7c51c0b94036eac95a842bb022a5dc"
integrity sha512-iJamLDhBy+Io/ePS6G+WT7ncXdmATrDFRFP4RvypqkuIdvPBULhukWyIEbB5hiXaQbYNBK6xx2trxp86YuvyeQ== integrity sha1-Opyh+zx8UcC5QDbqyVqEK7Aipdw=
dependencies: dependencies:
chalk "^2.4.1" chalk "^2.4.1"
commander "^2.17.0" commander "^2.17.0"
comment-json "^1.1.3" comment-json "^2.3.1"
cross-spawn "^6.0.5" cross-spawn "^6.0.5"
dayjs "^1.7.4" dayjs "^1.7.4"
download-git-repo "^1.1.0" download-git-repo "^1.1.0"