271 lines
5.6 KiB
Go
271 lines
5.6 KiB
Go
|
// Copyright 2010 The Walk Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style
|
||
|
// license that can be found in the LICENSE file.
|
||
|
|
||
|
// +build windows
|
||
|
|
||
|
package walk
|
||
|
|
||
|
import (
|
||
|
"syscall"
|
||
|
"unsafe"
|
||
|
|
||
|
"github.com/lxn/win"
|
||
|
)
|
||
|
|
||
|
// https://msdn.microsoft.com/en-us/library/windows/desktop/bb760416(v=vs.85).aspx says 80,
|
||
|
// but in reality, that hasn't been enforced for many many Windows versions. So we give it
|
||
|
// 1024 instead.
|
||
|
const maxToolTipTextLen = 1024 // including NUL terminator
|
||
|
|
||
|
type ToolTip struct {
|
||
|
WindowBase
|
||
|
}
|
||
|
|
||
|
func NewToolTip() (*ToolTip, error) {
|
||
|
tt, err := newToolTip(0)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
win.SetWindowPos(tt.hWnd, win.HWND_TOPMOST, 0, 0, 0, 0, win.SWP_NOMOVE|win.SWP_NOSIZE|win.SWP_NOACTIVATE)
|
||
|
|
||
|
return tt, nil
|
||
|
}
|
||
|
|
||
|
func newToolTip(style uint32) (*ToolTip, error) {
|
||
|
tt := new(ToolTip)
|
||
|
|
||
|
if err := InitWindow(
|
||
|
tt,
|
||
|
nil,
|
||
|
"tooltips_class32",
|
||
|
win.WS_DISABLED|win.WS_POPUP|win.TTS_ALWAYSTIP|style,
|
||
|
0); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
succeeded := false
|
||
|
defer func() {
|
||
|
if !succeeded {
|
||
|
tt.Dispose()
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
tt.SendMessage(win.TTM_SETMAXTIPWIDTH, 0, 300)
|
||
|
|
||
|
succeeded = true
|
||
|
|
||
|
return tt, nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) Title() string {
|
||
|
var gt win.TTGETTITLE
|
||
|
|
||
|
buf := make([]uint16, 100)
|
||
|
|
||
|
gt.DwSize = uint32(unsafe.Sizeof(gt))
|
||
|
gt.Cch = uint32(len(buf))
|
||
|
gt.PszTitle = &buf[0]
|
||
|
|
||
|
tt.SendMessage(win.TTM_GETTITLE, 0, uintptr(unsafe.Pointer(>)))
|
||
|
|
||
|
return syscall.UTF16ToString(buf)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) SetTitle(title string) error {
|
||
|
return tt.setTitle(title, win.TTI_NONE)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) SetInfoTitle(title string) error {
|
||
|
return tt.setTitle(title, win.TTI_INFO)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) SetWarningTitle(title string) error {
|
||
|
return tt.setTitle(title, win.TTI_WARNING)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) SetErrorTitle(title string) error {
|
||
|
return tt.setTitle(title, win.TTI_ERROR)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) setTitle(title string, icon uintptr) error {
|
||
|
if len(title) > 99 {
|
||
|
title = title[:99]
|
||
|
}
|
||
|
|
||
|
if win.FALSE == tt.SendMessage(win.TTM_SETTITLE, icon, uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(title)))) {
|
||
|
return newError("TTM_SETTITLE failed")
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) track(tool Widget) error {
|
||
|
form := tool.Form()
|
||
|
if form == nil {
|
||
|
return nil
|
||
|
}
|
||
|
// HACK: We may have to delay this until the form is fully up to avoid glitches.
|
||
|
if !form.AsFormBase().started {
|
||
|
form.Starting().Once(func() {
|
||
|
tt.track(tool)
|
||
|
})
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
ti := tt.toolInfo(tool.Handle())
|
||
|
if ti == nil {
|
||
|
return newError("unknown tool")
|
||
|
}
|
||
|
|
||
|
tt.SendMessage(win.TTM_TRACKACTIVATE, 1, uintptr(unsafe.Pointer(ti)))
|
||
|
|
||
|
b := tool.BoundsPixels()
|
||
|
|
||
|
p := Point{0, b.Y + b.Height}.toPOINT()
|
||
|
if form.RightToLeftLayout() {
|
||
|
p.X = int32(b.X - b.Width/2)
|
||
|
} else {
|
||
|
p.X = int32(b.X + b.Width/2)
|
||
|
}
|
||
|
|
||
|
win.ClientToScreen(tool.Parent().Handle(), &p)
|
||
|
|
||
|
tt.SendMessage(win.TTM_TRACKPOSITION, 0, uintptr(win.MAKELONG(uint16(p.X), uint16(p.Y))))
|
||
|
|
||
|
var insertAfterHWND win.HWND
|
||
|
if form := tool.Form(); form != nil && win.GetForegroundWindow() == form.Handle() {
|
||
|
insertAfterHWND = win.HWND_TOP
|
||
|
} else {
|
||
|
insertAfterHWND = tool.Handle()
|
||
|
}
|
||
|
win.SetWindowPos(tt.hWnd, insertAfterHWND, 0, 0, 0, 0, win.SWP_NOMOVE|win.SWP_NOSIZE|win.SWP_NOACTIVATE)
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) untrack(tool Widget) error {
|
||
|
ti := tt.toolInfo(tool.Handle())
|
||
|
if ti == nil {
|
||
|
return newError("unknown tool")
|
||
|
}
|
||
|
|
||
|
tt.SendMessage(win.TTM_TRACKACTIVATE, 0, uintptr(unsafe.Pointer(ti)))
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) AddTool(tool Widget) error {
|
||
|
return tt.addTool(tt.hwndForTool(tool), false)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) addTrackedTool(tool Widget) error {
|
||
|
return tt.addTool(tt.hwndForTool(tool), true)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) addTool(hwnd win.HWND, track bool) error {
|
||
|
if hwnd == 0 {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
var ti win.TOOLINFO
|
||
|
ti.CbSize = uint32(unsafe.Sizeof(ti))
|
||
|
ti.Hwnd = hwnd
|
||
|
ti.UFlags = win.TTF_IDISHWND
|
||
|
if track {
|
||
|
ti.UFlags |= win.TTF_TRACK
|
||
|
} else {
|
||
|
ti.UFlags |= win.TTF_SUBCLASS
|
||
|
}
|
||
|
ti.UId = uintptr(hwnd)
|
||
|
|
||
|
if win.FALSE == tt.SendMessage(win.TTM_ADDTOOL, 0, uintptr(unsafe.Pointer(&ti))) {
|
||
|
return newError("TTM_ADDTOOL failed")
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) RemoveTool(tool Widget) error {
|
||
|
return tt.removeTool(tt.hwndForTool(tool))
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) removeTool(hwnd win.HWND) error {
|
||
|
var ti win.TOOLINFO
|
||
|
ti.CbSize = uint32(unsafe.Sizeof(ti))
|
||
|
ti.Hwnd = hwnd
|
||
|
ti.UId = uintptr(hwnd)
|
||
|
|
||
|
tt.SendMessage(win.TTM_DELTOOL, 0, uintptr(unsafe.Pointer(&ti)))
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) Text(tool Widget) string {
|
||
|
return tt.text(tt.hwndForTool(tool))
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) text(hwnd win.HWND) string {
|
||
|
ti := tt.toolInfo(hwnd)
|
||
|
if ti == nil {
|
||
|
return ""
|
||
|
}
|
||
|
|
||
|
return win.UTF16PtrToString(ti.LpszText)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) SetText(tool Widget, text string) error {
|
||
|
return tt.setText(tt.hwndForTool(tool), text)
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) setText(hwnd win.HWND, text string) error {
|
||
|
ti := tt.toolInfo(hwnd)
|
||
|
if ti == nil {
|
||
|
return newError("unknown tool")
|
||
|
}
|
||
|
|
||
|
n := 0
|
||
|
for i, r := range text {
|
||
|
if r < 0x10000 {
|
||
|
n++
|
||
|
} else {
|
||
|
n += 2 // surrogate pair
|
||
|
}
|
||
|
if n >= maxToolTipTextLen {
|
||
|
text = text[:i]
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ti.LpszText = syscall.StringToUTF16Ptr(text)
|
||
|
|
||
|
tt.SendMessage(win.TTM_SETTOOLINFO, 0, uintptr(unsafe.Pointer(ti)))
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tt *ToolTip) toolInfo(hwnd win.HWND) *win.TOOLINFO {
|
||
|
var ti win.TOOLINFO
|
||
|
var buf [maxToolTipTextLen]uint16
|
||
|
|
||
|
ti.CbSize = uint32(unsafe.Sizeof(ti))
|
||
|
ti.Hwnd = hwnd
|
||
|
ti.UId = uintptr(hwnd)
|
||
|
ti.LpszText = &buf[0]
|
||
|
|
||
|
if win.FALSE == tt.SendMessage(win.TTM_GETTOOLINFO, 0, uintptr(unsafe.Pointer(&ti))) {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
return &ti
|
||
|
}
|
||
|
|
||
|
func (*ToolTip) hwndForTool(tool Widget) win.HWND {
|
||
|
if hftt, ok := tool.(interface{ handleForToolTip() win.HWND }); ok {
|
||
|
return hftt.handleForToolTip()
|
||
|
}
|
||
|
|
||
|
return tool.Handle()
|
||
|
}
|