Skip to content

Commit

Permalink
Add settings for clipboard plugin
Browse files Browse the repository at this point in the history
  • Loading branch information
qianlifeng committed Nov 25, 2023
1 parent 1f2530c commit 972ea03
Show file tree
Hide file tree
Showing 8 changed files with 204 additions and 39 deletions.
3 changes: 2 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -14,4 +14,5 @@ Plugins/Wox.Plugin.Chatgpt/dist/
Wox/resource/ui/react/
Wox/resource/ui/electron/
Wox.UI.React/dist/
node_modules/
node_modules/
__pycache__/
21 changes: 21 additions & 0 deletions Wox.UI.React/src/components/tools/WoxImage.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,27 @@ import { WoxImageTypeEnum } from "../../enums/WoxImageTypeEnum.ts"
import { WOXMESSAGE } from "../../entity/WoxMessage.typings"
import styled from "styled-components"

export function parseWoxImage(data: string): WOXMESSAGE.WoxImage {
const img = {} as WOXMESSAGE.WoxImage
if (data.startsWith("base64:")) {
img.ImageType = WoxImageTypeEnum.WoxImageTypeBase64.code
img.ImageData = data.slice(7)
return img
}
if (data.startsWith("svg:")) {
img.ImageType = WoxImageTypeEnum.WoxImageTypeSvg.code
img.ImageData = data.slice(4)
return img
}
if (data.startsWith("url:")) {
img.ImageType = WoxImageTypeEnum.WoxImageTypeUrl.code
img.ImageData = data.slice(4)
return img
}

return img
}

export default (props: { img: WOXMESSAGE.WoxImage; height: number; width: number }) => {
return (
<Style width={props.width} height={props.height}>
Expand Down
3 changes: 2 additions & 1 deletion Wox.UI.React/src/components/tools/WoxPreview.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,14 +5,15 @@ import Markdown from "react-markdown"
import styled from "styled-components"
import { WoxThemeHelper } from "../../utils/WoxThemeHelper.ts"
import { Theme } from "../../entity/Theme.typings"
import WoxImage, { parseWoxImage } from "./WoxImage.tsx"

export default (props: { preview: WOXMESSAGE.WoxPreview; resultSingleItemHeight: number }) => {
return (
<Style theme={WoxThemeHelper.getInstance().getTheme()}>
<WoxScrollbar scrollbarProps={{ autoHeight: true, autoHeightMin: 0, autoHeightMax: props.resultSingleItemHeight * 8 + 10 }}>
<div className={"wox-query-result-preview-content"}>
{props.preview.PreviewType === WoxPreviewTypeEnum.WoxPreviewTypeText.code && <p className={"wox-query-result-preview-text"}>{props.preview.PreviewData}</p>}
{props.preview.PreviewType === WoxPreviewTypeEnum.WoxPreviewTypeImage.code && <img className={"wox-query-result-preview-image"} src={props.preview.PreviewData} />}
{props.preview.PreviewType === WoxPreviewTypeEnum.WoxPreviewTypeImage.code && <WoxImage img={parseWoxImage(props.preview.PreviewData)} height={360} width={360}></WoxImage>}
{props.preview.PreviewType === WoxPreviewTypeEnum.WoxPreviewTypeMarkdown.code && <Markdown>{props.preview.PreviewData}</Markdown>}
{props.preview.PreviewType === WoxPreviewTypeEnum.WoxPreviewTypeUrl.code && <iframe className={"wox-query-result-preview-url"} src={props.preview.PreviewData}></iframe>}
</div>
Expand Down
10 changes: 10 additions & 0 deletions Wox/plugin/image.go
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,16 @@ func NewWoxImageBase64(data string) WoxImage {
}
}

func NewWoxImage(image image.Image) (WoxImage, error) {
buf := new(bytes.Buffer)
encodeErr := png.Encode(buf, image)
if encodeErr != nil {
return WoxImage{}, fmt.Errorf("failed to encode image: %s", encodeErr.Error())
}

return NewWoxImageBase64(fmt.Sprintf("data:image/png;base64,%s", base64.StdEncoding.EncodeToString(buf.Bytes()))), nil
}

func NewWoxImageUrl(url string) WoxImage {
return WoxImage{
ImageType: WoxImageTypeUrl,
Expand Down
179 changes: 153 additions & 26 deletions Wox/plugin/system/clipboard.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,21 +6,27 @@ import (
"encoding/base64"
"encoding/json"
"fmt"
"github.com/disintegration/imaging"
"github.com/google/uuid"
"github.com/samber/lo"
"image/png"
"os"
"slices"
"strconv"
"strings"
"time"
"wox/plugin"
"wox/setting"
"wox/util"
"wox/util/clipboard"
"wox/util/keyboard"
"wox/util/window"
)

var clipboardIcon = plugin.NewWoxImageBase64(``)
var isKeepTextHistorySettingKey = "is_keep_text_history"
var textHistoryDaysSettingKey = "text_history_days"
var isKeepImageHistorySettingKey = "is_keep_image_history"
var imageHistoryDaysSettingKey = "image_history_days"

func init() {
plugin.AllSystemPlugin = append(plugin.AllSystemPlugin, &ClipboardPlugin{
Expand Down Expand Up @@ -144,6 +150,43 @@ func (c *ClipboardPlugin) GetMetadata() plugin.Metadata {
"Macos",
"Linux",
},
SettingDefinitions: []setting.PluginSettingDefinitionItem{
{
Type: setting.PluginSettingDefinitionTypeCheckBox,
Value: setting.PluginSettingValueCheckBox{
Key: isKeepTextHistorySettingKey,
Label: "i18n:plugin_clipboard_keep_text_history",
DefaultValue: "true",
},
},
{
Type: setting.PluginSettingDefinitionTypeTextBox,
Value: setting.PluginSettingValueTextBox{
Key: textHistoryDaysSettingKey,
Suffix: "i18n:plugin_clipboard_days",
DefaultValue: "90",
},
},
{
Type: setting.PluginSettingDefinitionTypeNewLine,
},
{
Type: setting.PluginSettingDefinitionTypeCheckBox,
Value: setting.PluginSettingValueCheckBox{
Key: isKeepImageHistorySettingKey,
Label: "i18n:plugin_clipboard_keep_image_history",
DefaultValue: "true",
},
},
{
Type: setting.PluginSettingDefinitionTypeTextBox,
Value: setting.PluginSettingValueTextBox{
Key: imageHistoryDaysSettingKey,
Suffix: "i18n:plugin_clipboard_days",
DefaultValue: "3",
},
},
},
}
}

Expand All @@ -157,6 +200,13 @@ func (c *ClipboardPlugin) Init(ctx context.Context, initParams plugin.InitParams
return
}

if data.GetType() == clipboard.ClipboardTypeText && !c.isKeepTextHistory(ctx) {
return
}
if data.GetType() == clipboard.ClipboardTypeImage && !c.isKeepImageHistory(ctx) {
return
}

icon := c.getDefaultTextIcon()

if data.GetType() == clipboard.ClipboardTypeText {
Expand Down Expand Up @@ -326,12 +376,28 @@ func (c *ClipboardPlugin) convertClipboardData(ctx context.Context, history Clip

if history.Data.GetType() == clipboard.ClipboardTypeImage {
historyData := history.Data.(*clipboard.ImageData)
compressedPreviewImg := imaging.Resize(historyData.Image, 400, 0, imaging.Lanczos)
compressedIconImg := imaging.Resize(historyData.Image, 40, 0, imaging.Lanczos)
previewImage, err := plugin.NewWoxImage(compressedPreviewImg)
if err != nil {
previewImage = c.getDefaultTextIcon()
}
iconImage, iconErr := plugin.NewWoxImage(compressedIconImg)
if iconErr != nil {
iconImage = plugin.NewWoxImageBase64(``)
}

return plugin.QueryResult{
Title: fmt.Sprintf("Image (%d*%d)", historyData.Image.Bounds().Dx(), historyData.Image.Bounds().Dy()),
Icon: plugin.NewWoxImageBase64(``),
Icon: iconImage,
Preview: plugin.WoxPreview{
PreviewType: plugin.WoxPreviewTypeImage,
PreviewData: "image",
PreviewData: previewImage.String(),
PreviewProperties: map[string]string{
"i18n:plugin_clipboard_copy_date": util.FormatTimestamp(history.Timestamp),
"i18n:plugin_clipboard_image_width": fmt.Sprintf("%d", historyData.Image.Bounds().Dx()),
"i18n:plugin_clipboard_image_height": fmt.Sprintf("%d", historyData.Image.Bounds().Dy()),
},
},
Score: 0,
Actions: []plugin.QueryResultAction{
Expand Down Expand Up @@ -381,43 +447,48 @@ func (c *ClipboardPlugin) getActiveWindowIcon(ctx context.Context) (plugin.WoxIm
}

func (c *ClipboardPlugin) saveHistory(ctx context.Context) {
// only save text history
histories := lo.Filter(c.history, func(item ClipboardHistory, index int) bool {
return item.Data.GetType() == clipboard.ClipboardTypeText
})
startTimestamp := util.GetSystemTimestamp()

// only save last 5000 history, but keep all favorite history
if len(histories) > c.maxHistoryCount+100 {
var favoriteHistories []ClipboardHistory
var normalHistories []ClipboardHistory
for i := len(histories) - 1; i >= 0; i-- {
if histories[i].IsFavorite {
favoriteHistories = append(favoriteHistories, histories[i])
} else {
normalHistories = append(normalHistories, histories[i])
}
var favoriteHistories []ClipboardHistory
var normalHistories []ClipboardHistory
for i := len(c.history) - 1; i >= 0; i-- {
if c.history[i].Data == nil {
continue
}

if len(normalHistories) > c.maxHistoryCount {
normalHistories = normalHistories[:c.maxHistoryCount]
if c.history[i].IsFavorite {
favoriteHistories = append(favoriteHistories, c.history[i])
continue
}

histories = append(favoriteHistories, normalHistories...)
if c.history[i].Data.GetType() == clipboard.ClipboardTypeText {
if util.GetSystemTimestamp()-c.history[i].Timestamp > int64(c.getTextHistoryDays(ctx))*24*60*60*1000 {
continue
}
}
if c.history[i].Data.GetType() == clipboard.ClipboardTypeImage {
if util.GetSystemTimestamp()-c.history[i].Timestamp > int64(c.getImageHistoryDays(ctx))*24*60*60*1000 {
continue
}
}

// sort history by timestamp asc
slices.SortStableFunc(histories, func(i, j ClipboardHistory) int {
return int(i.Timestamp - j.Timestamp)
})
normalHistories = append(normalHistories, c.history[i])
}

histories := append(favoriteHistories, normalHistories...)

// sort history by timestamp asc
slices.SortStableFunc(histories, func(i, j ClipboardHistory) int {
return int(i.Timestamp - j.Timestamp)
})

historyJson, marshalErr := json.Marshal(histories)
if marshalErr != nil {
c.api.Log(ctx, fmt.Sprintf("marshal clipboard text history failed, err=%s", marshalErr.Error()))
return
}

c.api.SaveSetting(ctx, "history", string(historyJson), false)
c.api.Log(ctx, fmt.Sprintf("save clipboard text history, count=%d", len(c.history)))
c.api.Log(ctx, fmt.Sprintf("save clipboard history, count:%d, cost:%dms", len(c.history), util.GetSystemTimestamp()-startTimestamp))
}

func (c *ClipboardPlugin) loadHistory(ctx context.Context) {
Expand All @@ -444,3 +515,59 @@ func (c *ClipboardPlugin) loadHistory(ctx context.Context) {
func (c *ClipboardPlugin) getDefaultTextIcon() plugin.WoxImage {
return plugin.NewWoxImageSvg(`<svg xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" width="48" height="48" viewBox="0 0 48 48"><path fill="#90CAF9" d="M40 45L8 45 8 3 30 3 40 13z"></path><path fill="#E1F5FE" d="M38.5 14L29 14 29 4.5z"></path><path fill="#1976D2" d="M16 21H33V23H16zM16 25H29V27H16zM16 29H33V31H16zM16 33H29V35H16z"></path></svg>`)
}

func (c *ClipboardPlugin) isKeepTextHistory(ctx context.Context) bool {
isKeepTextHistory := c.api.GetSetting(ctx, isKeepTextHistorySettingKey)
if isKeepTextHistory == "" {
return true
}

isKeepTextHistoryBool, err := strconv.ParseBool(isKeepTextHistory)
if err != nil {
return true
}

return isKeepTextHistoryBool
}

func (c *ClipboardPlugin) getTextHistoryDays(ctx context.Context) int {
textHistoryDays := c.api.GetSetting(ctx, textHistoryDaysSettingKey)
if textHistoryDays == "" {
return 90
}

textHistoryDaysInt, err := strconv.Atoi(textHistoryDays)
if err != nil {
return 90
}

return textHistoryDaysInt
}

func (c *ClipboardPlugin) isKeepImageHistory(ctx context.Context) bool {
isKeepImageHistory := c.api.GetSetting(ctx, isKeepImageHistorySettingKey)
if isKeepImageHistory == "" {
return true
}

isKeepImageHistoryBool, err := strconv.ParseBool(isKeepImageHistory)
if err != nil {
return true
}

return isKeepImageHistoryBool
}

func (c *ClipboardPlugin) getImageHistoryDays(ctx context.Context) int {
imageHistoryDays := c.api.GetSetting(ctx, imageHistoryDaysSettingKey)
if imageHistoryDays == "" {
return 3
}

imageHistoryDaysInt, err := strconv.Atoi(imageHistoryDays)
if err != nil {
return 3
}

return imageHistoryDaysInt
}
2 changes: 2 additions & 0 deletions Wox/resource/lang/en_US.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@
"plugin_browser_bookmark_open_in_browser": "Open in browser",
"plugin_clipboard_copy_date": "Copy date",
"plugin_clipboard_copy_characters": "Copy characters",
"plugin_clipboard_image_width": "Image width",
"plugin_clipboard_image_height": "Image height",
"plugin_indicator_activate_plugin": "Activate %s plugin",
"plugin_sys_empty_trash": "Empty Trash",
"plugin_sys_lock_computer": "Lock PC",
Expand Down
2 changes: 2 additions & 0 deletions Wox/resource/lang/zh_CN.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@
"plugin_browser_bookmark_open_in_browser": "在浏览器中打开",
"plugin_clipboard_copy_date": "复制日期",
"plugin_clipboard_copy_characters": "复制字符",
"plugin_clipboard_image_width": "图片宽度",
"plugin_clipboard_image_height": "图片高度",
"plugin_indicator_activate_plugin": "激活%s插件",
"plugin_sys_empty_trash": "清空回收站",
"plugin_sys_lock_computer": "锁定电脑",
Expand Down
23 changes: 12 additions & 11 deletions Wox/util/clipboard/clipboard.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package clipboard

import (
"bytes"
"encoding/base64"
"encoding/json"
"errors"
"fmt"
Expand Down Expand Up @@ -198,26 +199,26 @@ func (i *ImageData) MarshalJSON() ([]byte, error) {
return nil, err
}

var mapData = make(map[string]string)
mapData["type"] = string(i.GetType())
mapData["image"] = string(buf.Bytes())
return json.Marshal(mapData)
return json.Marshal(base64.StdEncoding.EncodeToString(buf.Bytes()))
}

func (i *ImageData) UnmarshalJSON(data []byte) error {
var mapData = make(map[string]string)
err := json.Unmarshal(data, &mapData)
if err != nil {
return err
var base64ImgData string
unmarshalErr := json.Unmarshal(data, &base64ImgData)
if unmarshalErr != nil {
return unmarshalErr
}

imageBytes := []byte(mapData["image"])
imageReader := bytes.NewReader(imageBytes)
img, _, err := image.Decode(imageReader)
decodeBytes, err := base64.StdEncoding.DecodeString(base64ImgData)
if err != nil {
return err
}

img, decodeErr := png.Decode(bytes.NewReader(decodeBytes))
if decodeErr != nil {
return decodeErr
}

i.Image = img
return nil
}

0 comments on commit 972ea03

Please sign in to comment.