1 Star 0 Fork 0

橙子/lxnWalk

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
form.go 18.03 KB
一键复制 编辑 原始数据 按行查看 历史
橙子 提交于 2020-08-13 10:03 . .
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819
// Copyright 2012 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 (
"fmt"
"strconv"
"sync"
"syscall"
"time"
"unsafe"
"github.com/xuchengzhi/win"
)
type CloseReason byte
const (
CloseReasonUnknown CloseReason = iota
CloseReasonUser
)
var (
syncFuncs struct {
m sync.Mutex
funcs []func()
}
syncMsgId uint32
taskbarButtonCreatedMsgId uint32
)
func init() {
syncMsgId = win.RegisterWindowMessage(syscall.StringToUTF16Ptr("WalkSync"))
taskbarButtonCreatedMsgId = win.RegisterWindowMessage(syscall.StringToUTF16Ptr("TaskbarButtonCreated"))
}
func synchronize(f func()) {
syncFuncs.m.Lock()
syncFuncs.funcs = append(syncFuncs.funcs, f)
syncFuncs.m.Unlock()
}
func runSynchronized() {
// Clear the list of callbacks first to avoid deadlock
// if a callback itself calls Synchronize()...
syncFuncs.m.Lock()
funcs := syncFuncs.funcs
syncFuncs.funcs = nil
syncFuncs.m.Unlock()
for _, f := range funcs {
f()
}
}
type Form interface {
Container
AsFormBase() *FormBase
Run() int
Starting() *Event
Closing() *CloseEvent
Activating() *Event
Deactivating() *Event
Activate() error
Show()
Hide()
Title() string
SetTitle(title string) error
TitleChanged() *Event
Icon() Image
SetIcon(icon Image) error
IconChanged() *Event
Owner() Form
SetOwner(owner Form) error
ProgressIndicator() *ProgressIndicator
// RightToLeftLayout returns whether coordinates on the x axis of the
// Form increase from right to left.
RightToLeftLayout() bool
// SetRightToLeftLayout sets whether coordinates on the x axis of the
// Form increase from right to left.
SetRightToLeftLayout(rtl bool) error
}
type FormBase struct {
WindowBase
clientComposite *Composite
owner Form
closingPublisher CloseEventPublisher
activatingPublisher EventPublisher
deactivatingPublisher EventPublisher
startingPublisher EventPublisher
titleChangedPublisher EventPublisher
iconChangedPublisher EventPublisher
progressIndicator *ProgressIndicator
icon Image
prevFocusHWnd win.HWND
proposedSize Size
isInRestoreState bool
started bool
didSetFocus bool
closeReason CloseReason
}
func (fb *FormBase) init(form Form) error {
var err error
if fb.clientComposite, err = NewComposite(form); err != nil {
return err
}
fb.clientComposite.SetName("clientComposite")
fb.clientComposite.background = nil
fb.clientComposite.children.observer = form.AsFormBase()
fb.MustRegisterProperty("Icon", NewProperty(
func() interface{} {
return fb.Icon()
},
func(v interface{}) error {
var icon *Icon
switch val := v.(type) {
case *Icon:
icon = val
case int:
var err error
if icon, err = Resources.Icon(strconv.Itoa(val)); err != nil {
return err
}
case string:
var err error
if icon, err = Resources.Icon(val); err != nil {
return err
}
default:
return ErrInvalidType
}
fb.SetIcon(icon)
return nil
},
fb.iconChangedPublisher.Event()))
fb.MustRegisterProperty("Title", NewProperty(
func() interface{} {
return fb.Title()
},
func(v interface{}) error {
return fb.SetTitle(assertStringOr(v, ""))
},
fb.titleChangedPublisher.Event()))
version := win.GetVersion()
if (version&0xFF) > 6 || ((version&0xFF) == 6 && (version&0xFF00>>8) > 0) {
win.ChangeWindowMessageFilterEx(fb.hWnd, taskbarButtonCreatedMsgId, win.MSGFLT_ALLOW, nil)
}
return nil
}
func (fb *FormBase) AsContainerBase() *ContainerBase {
return fb.clientComposite.AsContainerBase()
}
func (fb *FormBase) AsFormBase() *FormBase {
return fb
}
func (fb *FormBase) LayoutFlags() LayoutFlags {
return ShrinkableHorz | ShrinkableVert | GrowableHorz | GrowableVert | GreedyHorz | GreedyVert
}
func (fb *FormBase) SizeHint() Size {
return fb.dialogBaseUnitsToPixels(Size{252, 218})
}
func (fb *FormBase) Children() *WidgetList {
if fb.clientComposite == nil {
return nil
}
return fb.clientComposite.Children()
}
func (fb *FormBase) Layout() Layout {
if fb.clientComposite == nil {
return nil
}
return fb.clientComposite.Layout()
}
func (fb *FormBase) SetLayout(value Layout) error {
if fb.clientComposite == nil {
return newError("clientComposite not initialized")
}
return fb.clientComposite.SetLayout(value)
}
func (fb *FormBase) SetBoundsPixels(bounds Rectangle) error {
if layout := fb.Layout(); layout != nil {
minSize := fb.sizeFromClientSizePixels(layout.MinSizeForSize(bounds.Size()))
if bounds.Width < minSize.Width {
bounds.Width = minSize.Width
}
if bounds.Height < minSize.Height {
bounds.Height = minSize.Height
}
}
if err := fb.WindowBase.SetBoundsPixels(bounds); err != nil {
return err
}
walkDescendants(fb, func(wnd Window) bool {
if container, ok := wnd.(Container); ok {
if layout := container.Layout(); layout != nil {
layout.Update(false)
}
}
return true
})
return nil
}
func (fb *FormBase) fixedSize() bool {
return !fb.hasStyleBits(win.WS_THICKFRAME)
}
func (fb *FormBase) DataBinder() *DataBinder {
return fb.clientComposite.DataBinder()
}
func (fb *FormBase) SetDataBinder(db *DataBinder) {
fb.clientComposite.SetDataBinder(db)
}
func (fb *FormBase) Suspended() bool {
if fb.clientComposite == nil {
return false
}
return fb.clientComposite.Suspended()
}
func (fb *FormBase) SetSuspended(suspended bool) {
fb.clientComposite.SetSuspended(suspended)
}
func (fb *FormBase) MouseDown() *MouseEvent {
return fb.clientComposite.MouseDown()
}
func (fb *FormBase) MouseMove() *MouseEvent {
return fb.clientComposite.MouseMove()
}
func (fb *FormBase) MouseUp() *MouseEvent {
return fb.clientComposite.MouseUp()
}
func (fb *FormBase) onInsertingWidget(index int, widget Widget) error {
return fb.clientComposite.onInsertingWidget(index, widget)
}
func (fb *FormBase) onInsertedWidget(index int, widget Widget) error {
err := fb.clientComposite.onInsertedWidget(index, widget)
if err == nil {
if layout := fb.Layout(); layout != nil && !fb.Suspended() {
minClientSize := fb.Layout().MinSize()
clientSize := fb.clientComposite.SizePixels()
if clientSize.Width < minClientSize.Width || clientSize.Height < minClientSize.Height {
fb.SetClientSizePixels(minClientSize)
}
}
}
return err
}
func (fb *FormBase) onRemovingWidget(index int, widget Widget) error {
return fb.clientComposite.onRemovingWidget(index, widget)
}
func (fb *FormBase) onRemovedWidget(index int, widget Widget) error {
return fb.clientComposite.onRemovedWidget(index, widget)
}
func (fb *FormBase) onClearingWidgets() error {
return fb.clientComposite.onClearingWidgets()
}
func (fb *FormBase) onClearedWidgets() error {
return fb.clientComposite.onClearedWidgets()
}
func (fb *FormBase) ContextMenu() *Menu {
return fb.clientComposite.ContextMenu()
}
func (fb *FormBase) SetContextMenu(contextMenu *Menu) {
fb.clientComposite.SetContextMenu(contextMenu)
}
func (fb *FormBase) applyEnabled(enabled bool) {
fb.WindowBase.applyEnabled(enabled)
fb.clientComposite.applyEnabled(enabled)
}
func (fb *FormBase) applyFont(font *Font) {
fb.WindowBase.applyFont(font)
fb.clientComposite.applyFont(font)
}
func (fb *FormBase) ApplySysColors() {
fb.WindowBase.ApplySysColors()
fb.clientComposite.ApplySysColors()
}
func (fb *FormBase) Background() Brush {
return fb.clientComposite.Background()
}
func (fb *FormBase) SetBackground(background Brush) {
fb.clientComposite.SetBackground(background)
}
func (fb *FormBase) Title() string {
return fb.text()
}
func (fb *FormBase) SetTitle(value string) error {
return fb.setText(value)
}
func (fb *FormBase) TitleChanged() *Event {
return fb.titleChangedPublisher.Event()
}
// RightToLeftLayout returns whether coordinates on the x axis of the
// FormBase increase from right to left.
func (fb *FormBase) RightToLeftLayout() bool {
return fb.hasExtendedStyleBits(win.WS_EX_LAYOUTRTL)
}
// SetRightToLeftLayout sets whether coordinates on the x axis of the
// FormBase increase from right to left.
func (fb *FormBase) SetRightToLeftLayout(rtl bool) error {
return fb.ensureExtendedStyleBits(win.WS_EX_LAYOUTRTL, rtl)
}
func (fb *FormBase) Run() int {
if fb.owner != nil {
win.EnableWindow(fb.owner.Handle(), false)
invalidateDescendentBorders := func() {
walkDescendants(fb.owner, func(wnd Window) bool {
if widget, ok := wnd.(Widget); ok {
widget.AsWidgetBase().invalidateBorderInParent()
}
return true
})
}
invalidateDescendentBorders()
defer invalidateDescendentBorders()
}
if layout := fb.Layout(); layout != nil {
layout.Update(false)
}
fb.clientComposite.focusFirstCandidateDescendant()
fb.started = true
fb.startingPublisher.Publish()
fb.SetBoundsPixels(fb.BoundsPixels())
return fb.mainLoop()
}
func (fb *FormBase) handleKeyDown(msg *win.MSG) bool {
ret := false
if key, mods := Key(msg.WParam), ModifiersDown(); key == KeyTab && (mods&ModControl) != 0 {
doTabbing := func(tw *TabWidget) {
index := tw.CurrentIndex()
if (mods & ModShift) != 0 {
index--
if index < 0 {
index = tw.Pages().Len() - 1
}
} else {
index++
if index >= tw.Pages().Len() {
index = 0
}
}
tw.SetCurrentIndex(index)
}
hwnd := win.GetFocus()
LOOP:
for hwnd != 0 {
window := windowFromHandle(hwnd)
switch widget := window.(type) {
case nil:
case *TabWidget:
doTabbing(widget)
return true
case Widget:
default:
break LOOP
}
hwnd = win.GetParent(hwnd)
}
walkDescendants(fb.window, func(w Window) bool {
if tw, ok := w.(*TabWidget); ok {
doTabbing(tw)
ret = true
return false
}
return true
})
if ret {
return true
}
}
walkDescendants(fb.window, func(w Window) bool {
if webView, ok := w.(*WebView); ok {
webViewHWnd := webView.Handle()
if webViewHWnd == msg.HWnd || win.IsChild(webViewHWnd, msg.HWnd) {
_ret := webView.translateAccelerator(msg)
if _ret {
ret = _ret
}
}
}
return true
})
return ret
}
func (fb *FormBase) Starting() *Event {
return fb.startingPublisher.Event()
}
func (fb *FormBase) Activating() *Event {
return fb.activatingPublisher.Event()
}
func (fb *FormBase) Deactivating() *Event {
return fb.deactivatingPublisher.Event()
}
func (fb *FormBase) Activate() error {
if hwndPrevActive := win.SetActiveWindow(fb.hWnd); hwndPrevActive == 0 {
return lastError("SetActiveWindow")
}
return nil
}
func (fb *FormBase) Owner() Form {
return fb.owner
}
func (fb *FormBase) SetOwner(value Form) error {
fb.owner = value
var ownerHWnd win.HWND
if value != nil {
ownerHWnd = value.Handle()
}
win.SetLastError(0)
if 0 == win.SetWindowLong(
fb.hWnd,
win.GWL_HWNDPARENT,
int32(ownerHWnd)) && win.GetLastError() != 0 {
return lastError("SetWindowLong")
}
return nil
}
func (fb *FormBase) Icon() Image {
return fb.icon
}
func (fb *FormBase) SetIcon(icon Image) error {
var hIconSmall, hIconBig uintptr
if icon != nil {
smallIcon, err := iconCache.Icon(icon, fb.DPI())
if err != nil {
return err
}
hIconSmall = uintptr(smallIcon.handleForDPI(fb.DPI()))
bigDPI := int(48.0 / float64(icon.Size().Width) * 96.0)
bigIcon, err := iconCache.Icon(icon, bigDPI)
if err != nil {
return err
}
hIconBig = uintptr(bigIcon.handleForDPI(bigDPI))
}
fb.SendMessage(win.WM_SETICON, 0, hIconSmall)
fb.SendMessage(win.WM_SETICON, 1, hIconBig)
fb.icon = icon
fb.iconChangedPublisher.Publish()
return nil
}
func (fb *FormBase) IconChanged() *Event {
return fb.iconChangedPublisher.Event()
}
func (fb *FormBase) Hide() {
fb.window.SetVisible(false)
}
func (fb *FormBase) Show() {
fb.proposedSize = fb.minSize
if p, ok := fb.window.(Persistable); ok && p.Persistent() && appSingleton.settings != nil {
p.RestoreState()
}
fb.window.SetVisible(true)
}
func (fb *FormBase) close() error {
if p, ok := fb.window.(Persistable); ok && p.Persistent() && appSingleton.settings != nil {
p.SaveState()
}
fb.window.Dispose()
return nil
}
func (fb *FormBase) Close() error {
fb.SendMessage(win.WM_CLOSE, 0, 0)
return nil
}
func (fb *FormBase) Persistent() bool {
return fb.clientComposite.persistent
}
func (fb *FormBase) SetPersistent(value bool) {
fb.clientComposite.persistent = value
}
func (fb *FormBase) SaveState() error {
if err := fb.clientComposite.SaveState(); err != nil {
return err
}
var wp win.WINDOWPLACEMENT
wp.Length = uint32(unsafe.Sizeof(wp))
if !win.GetWindowPlacement(fb.hWnd, &wp) {
return lastError("GetWindowPlacement")
}
state := fmt.Sprint(
wp.Flags, wp.ShowCmd,
wp.PtMinPosition.X, wp.PtMinPosition.Y,
wp.PtMaxPosition.X, wp.PtMaxPosition.Y,
wp.RcNormalPosition.Left, wp.RcNormalPosition.Top,
wp.RcNormalPosition.Right, wp.RcNormalPosition.Bottom)
if err := fb.WriteState(state); err != nil {
return err
}
return nil
}
func (fb *FormBase) RestoreState() error {
if fb.isInRestoreState {
return nil
}
fb.isInRestoreState = true
defer func() {
fb.isInRestoreState = false
}()
state, err := fb.ReadState()
if err != nil {
return err
}
if state == "" {
return nil
}
var wp win.WINDOWPLACEMENT
if _, err := fmt.Sscan(state,
&wp.Flags, &wp.ShowCmd,
&wp.PtMinPosition.X, &wp.PtMinPosition.Y,
&wp.PtMaxPosition.X, &wp.PtMaxPosition.Y,
&wp.RcNormalPosition.Left, &wp.RcNormalPosition.Top,
&wp.RcNormalPosition.Right, &wp.RcNormalPosition.Bottom); err != nil {
return err
}
wp.Length = uint32(unsafe.Sizeof(wp))
if layout := fb.Layout(); layout != nil && fb.fixedSize() {
minSize := fb.sizeFromClientSizePixels(layout.MinSize())
wp.RcNormalPosition.Right = wp.RcNormalPosition.Left + int32(minSize.Width) - 1
wp.RcNormalPosition.Bottom = wp.RcNormalPosition.Top + int32(minSize.Height) - 1
}
if !win.SetWindowPlacement(fb.hWnd, &wp) {
return lastError("SetWindowPlacement")
}
return fb.clientComposite.RestoreState()
}
func (fb *FormBase) Closing() *CloseEvent {
return fb.closingPublisher.Event()
}
func (fb *FormBase) ProgressIndicator() *ProgressIndicator {
return fb.progressIndicator
}
func (fb *FormBase) WndProc(hwnd win.HWND, msg uint32, wParam, lParam uintptr) uintptr {
switch msg {
case win.WM_ACTIVATE:
switch win.LOWORD(uint32(wParam)) {
case win.WA_ACTIVE, win.WA_CLICKACTIVE:
if fb.prevFocusHWnd != 0 {
win.SetFocus(fb.prevFocusHWnd)
}
appSingleton.activeForm = fb.window.(Form)
fb.activatingPublisher.Publish()
case win.WA_INACTIVE:
fb.prevFocusHWnd = win.GetFocus()
appSingleton.activeForm = nil
fb.deactivatingPublisher.Publish()
}
return 0
case win.WM_CLOSE:
fb.closeReason = CloseReasonUnknown
var canceled bool
fb.closingPublisher.Publish(&canceled, fb.closeReason)
if !canceled {
if fb.owner != nil {
win.EnableWindow(fb.owner.Handle(), true)
if !win.SetWindowPos(fb.owner.Handle(), win.HWND_NOTOPMOST, 0, 0, 0, 0, win.SWP_NOMOVE|win.SWP_NOSIZE|win.SWP_SHOWWINDOW) {
lastError("SetWindowPos")
}
}
fb.close()
}
return 0
case win.WM_COMMAND:
return fb.clientComposite.WndProc(hwnd, msg, wParam, lParam)
case win.WM_GETMINMAXINFO:
if fb.Suspended() || fb.proposedSize == (Size{}) {
break
}
mmi := (*win.MINMAXINFO)(unsafe.Pointer(lParam))
var min Size
if layout := fb.clientComposite.layout; layout != nil {
size := fb.clientSizeFromSizePixels(fb.proposedSize)
min = fb.sizeFromClientSizePixels(layout.MinSizeForSize(size))
if fb.proposedSize.Width < min.Width {
min = fb.sizeFromClientSizePixels(layout.MinSizeForSize(min))
}
}
mmi.PtMinTrackSize = win.POINT{
int32(maxi(min.Width, fb.minSize.Width)),
int32(maxi(min.Height, fb.minSize.Height)),
}
return 0
case win.WM_NOTIFY:
return fb.clientComposite.WndProc(hwnd, msg, wParam, lParam)
case win.WM_SETTEXT:
fb.titleChangedPublisher.Publish()
case win.WM_SIZING:
rc := (*win.RECT)(unsafe.Pointer(lParam))
fb.proposedSize = rectangleFromRECT(*rc).Size()
fb.clientComposite.SetBoundsPixels(fb.window.ClientBoundsPixels())
case win.WM_SIZE:
fb.clientComposite.SetBoundsPixels(fb.window.ClientBoundsPixels())
case win.WM_SHOWWINDOW:
if wParam == win.FALSE {
fb.didSetFocus = false
}
case win.WM_PAINT:
if !fb.didSetFocus && fb.Visible() {
fb.didSetFocus = true
fb.clientComposite.focusFirstCandidateDescendant()
}
case win.WM_SYSCOLORCHANGE:
fb.ApplySysColors()
case win.WM_DPICHANGED:
wasSuspended := fb.Suspended()
fb.SetSuspended(true)
defer fb.SetSuspended(wasSuspended)
dpi := int(win.HIWORD(uint32(wParam)))
seenInApplyFontToDescendantsDuringDPIChange = make(map[*WindowBase]bool)
seenInApplyDPIToDescendantsDuringDPIChange = make(map[*WindowBase]bool)
defer func() {
seenInApplyFontToDescendantsDuringDPIChange = nil
seenInApplyDPIToDescendantsDuringDPIChange = nil
}()
fb.clientComposite.ApplyDPI(dpi)
fb.ApplyDPI(dpi)
if fb.progressIndicator != nil {
fb.progressIndicator.SetOverlayIcon(fb.progressIndicator.overlayIcon, fb.progressIndicator.overlayIconDescription)
}
applyDPIToDescendants(fb.window, dpi)
rc := (*win.RECT)(unsafe.Pointer(lParam))
fb.window.SetBoundsPixels(rectangleFromRECT(*rc))
fb.SetIcon(fb.icon)
time.AfterFunc(time.Second, func() {
if fb.hWnd == 0 {
return
}
fb.Synchronize(func() {
for ni := range notifyIcons {
// We do this on all NotifyIcons, not just ones attached to this form or descendents, because
// the notify icon might be on a different screen, and since it can't get notifications itself
// we hope that one of the forms did for it. We also have to delay it by a second, because the
// tray usually gets resized sometime after us. This is a nasty hack!
ni.applyDPI()
}
})
})
case win.WM_SYSCOMMAND:
if wParam == win.SC_CLOSE {
fb.closeReason = CloseReasonUser
}
case taskbarButtonCreatedMsgId:
version := win.GetVersion()
major := version & 0xFF
minor := version & 0xFF00 >> 8
// Check that the OS is Win 7 or later (Win 7 is v6.1).
if fb.progressIndicator == nil && (major > 6 || (major == 6 && minor > 0)) {
fb.progressIndicator, _ = newTaskbarList3(fb.hWnd)
}
}
return fb.WindowBase.WndProc(hwnd, msg, wParam, lParam)
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/xuchengzhi/lxnWalk.git
git@gitee.com:xuchengzhi/lxnWalk.git
xuchengzhi
lxnWalk
lxnWalk
6b3f71bcaf88

搜索帮助

D67c1975 1850385 1daf7b77 1850385