有没有用过"""Winamp5 艾子哥经典豪华版"""
那里有一个功能就是在当前用户打开的任何窗口的标题栏的右侧放了四个按钮,非常牛,非常有意思,哪位能指点我作出来,万分感谢如果我没说清楚的话自己下载一个看看就知道了UP有分
UP有分
UP有分

解决方案 »

  1.   

    Option Explicit'*********************
    '* API Declarations  *
    '*********************
    Private Declare Function SetWindowLong Lib "user32" Alias "SetWindowLongA" (ByVal hwnd As Long, ByVal nIndex As Long, ByVal dwNewLong As Long) As Long
    Private Declare Function GetWindowRect Lib "user32" (ByVal hwnd As Long, lpRect As Rect) As Long
    Private Declare Function GetParent Lib "user32" (ByVal hwnd As Long) As Long
    Private Declare Function SetParent Lib "user32" (ByVal hWndChild As Long, ByVal hWndNewParent As Long) As Long
    Private Declare Function SetWindowPos Lib "user32" (ByVal hwnd As Long, ByVal hWndInsertAfter As Long, ByVal x As Long, ByVal y As Long, ByVal cx As Long, ByVal cy As Long, ByVal wFlags As Long) As Long
    Private Declare Function SetWindowsHookEx Lib "user32" Alias "SetWindowsHookExA" (ByVal idHook&, ByVal lpfn&, ByVal hmod&, ByVal dwThreadId&) As Long
    Private Declare Function UnhookWindowsHookEx Lib "user32" (ByVal hHook&) As Long
    Private Declare Function CreateWindowEx Lib "user32" Alias "CreateWindowExA" (ByVal dwExStyle As Long, ByVal lpClassName As String, ByVal lpWindowName As String, ByVal dwStyle As Long, ByVal x As Long, ByVal y As Long, ByVal nWidth As Long, ByVal nHeight As Long, ByVal hWndParent As Long, ByVal hMenu As Long, ByVal hInstance As Long, lpParam As Any) As Long
    Private Declare Function ShowWindow Lib "user32" (ByVal hwnd As Long, ByVal nCmdShow As Long) As Long'*********************
    '* Type Declarations *
    '*********************
    Private Type Rect
        Left As Long
        Top As Long
        Right As Long
        Bottom As Long
    End TypePrivate Type CWPSTRUCT
        lParam As Long
        wParam As Long
        Message As Long
        hwnd As Long
    End Type'*********************
    '* Consts            *
    '*********************
    Const WM_MOVE = &H3
    Const WM_SETCURSOR = &H20
    Const WM_NCPAINT = &H85
    Const WM_COMMAND = &H111Const SWP_FRAMECHANGED = &H20
    Const GWL_EXSTYLE = -20'*********************
    '* Vars              *
    '*********************
    Private WHook&
    Private ButtonHwnd As LongPublic Sub Init()
        'Create the button that is going to be placed in the Titlebar
        ButtonHwnd& = CreateWindowEx(0&, "Button", "i", &H40000000, 50, 50, 14, 14, Form1.hwnd, 0&, App.hInstance, 0&)
        'Show the button cause it磗 invisible
        Call ShowWindow(ButtonHwnd&, 1)
        'Initialize the window hooking for the button
        WHook = SetWindowsHookEx(4, AddressOf HookProc, 0, App.ThreadID)
        Call SetWindowLong(ButtonHwnd&, GWL_EXSTYLE, &H80)
        Call SetParent(ButtonHwnd&, GetParent(Form1.hwnd))
    End SubPublic Sub Terminate()
        'Terminate the window hooking
        Call UnhookWindowsHookEx(WHook)
        Call SetParent(ButtonHwnd&, Form1.hwnd)
    End SubPublic Function HookProc&(ByVal nCode&, ByVal wParam&, Inf As CWPSTRUCT)
        Dim FormRect As Rect
        Static LastParam&
        If Inf.hwnd = GetParent(ButtonHwnd&) Then
            If Inf.Message = WM_COMMAND Then
                Select Case LastParam
                    'If the LastParam is cmdInTitlebar call the Click-Procedure
                    'of the button
                    Case ButtonHwnd&: Call Form1.cmdInTitlebar_Click
                End Select
            ElseIf Inf.Message = WM_SETCURSOR Then
                LastParam = Inf.wParam
            End If
            ElseIf Inf.hwnd = Form1.hwnd Then
            If Inf.Message = WM_NCPAINT Or Inf.Message = WM_MOVE Then
                'Get the size of the Form
                Call GetWindowRect(Form1.hwnd, FormRect)
                'Place the button int the Titlebar
                Call SetWindowPos(ButtonHwnd&, 0, FormRect.Right - 75, FormRect.Top + 6, 17, 14, SWP_FRAMECHANGED)
            End If
        End If
    End Function窗体:
    Public Sub cmdInTitlebar_Click()
        MsgBox "Example created by Druid Developing", vbInformation, "About this program"
    End SubPrivate Sub Form_Load()
        Call Init
    End SubPrivate Sub Form_Unload(Cancel As Integer)
        Call Terminate
    End Sub
      

  2.   


    SetParent Command1.hwnd, 0
    SetWindowLong Command1.hwnd, -8, Me.hwnd
    把command1分离窗口出来,使之能自由移动
    对me做子类处理,捕获WM_MOVE消息,处理command1的位置需要注意的是如果希望command1响应事件,需要把它放在一个容器内
    如frame1,然后对frame1使用上面的那两个函数。
    可以参考
    http://expert.csdn.net/Expert/FAQ/FAQ_Index.asp?id=7691下面是个简单的例子:
    在窗体中放上一个frame,在frame中放上一个commandbutton
    '窗体代码
    Private Sub Command1_Click()
    Debug.Print "Click the title button"
    End SubPrivate Sub Form_Load()
    Frame1.BorderStyle = 0
    Command1.Left = 0
    Command1.Top = 0
    Command1.Width = 240
    Command1.Height = 210
    moveHwnd = Frame1.hwnd
    SetParent moveHwnd, 0
    SetWindowLong moveHwnd, -8, Me.hwnd
    gHW = Me.hwnd
    hook
    End SubPrivate Sub Form_Unload(Cancel As Integer)
    Unhook
    End Sub'模块部分的代码Option Explicit
    Public moveHwnd As LongPublic Const GWL_WNDPROC = -4
    Global lpPrevWndProc As Long
    Global gHW As Long
    Declare Function CallWindowProc Lib "user32" Alias "CallWindowProcA" (ByVal lpPrevWndFunc As Long, ByVal hwnd As Long, ByVal MSG As Long, ByVal wParam As Long, ByVal lParam As Long) As Long
    Declare Function SetWindowLong Lib "user32" Alias "SetWindowLongA" (ByVal hwnd As Long, ByVal nIndex As Long, ByVal dwNewLong As Long) As LongPublic Declare Function SetParent Lib "user32" (ByVal hWndChild As Long, ByVal hWndNewParent As Long) As Long
    Public Const WM_MOVE As Long = &H3
    Public Declare Function MoveWindow Lib "user32" (ByVal hwnd As Long, ByVal x As Long, ByVal y As Long, ByVal nWidth As Long, ByVal nHeight As Long, ByVal bRepaint As Long) As Long
    Public Declare Function GetWindowRect Lib "user32" (ByVal hwnd As Long, lpRect As RECT) As Long
    Public Type RECT
        Left As Long
        Top As Long
        Right As Long
        Bottom As LongEnd Type    
        
    Function WindowProc(ByVal hw As Long, ByVal uMsg As Long, ByVal wParam As Long, ByVal lParam As Long) As LongSelect Case uMsg
        Case WM_MOVE
            Dim t As RECT
            GetWindowRect hw, t
            MoveWindow moveHwnd, t.Right - 72, t.Top + 6, 16, 14, 1
        Case Else 
    End Select
    WindowProc = CallWindowProc(lpPrevWndProc, hw, uMsg, wParam, lParam)
    End Function
        
    Public Sub hook()  
     lpPrevWndProc = SetWindowLong(gHW, GWL_WNDPROC, AddressOf WindowProc)
    End Sub
        
    Public Sub Unhook()
     Dim temp As Long
     temp = SetWindowLong(gHW, GWL_WNDPROC, lpPrevWndProc)
    End Sub
      

  3.   

    >>> hhjjhjhj(大头)我试了你的代码,可以实现,但只能对本窗体实现.我想要它能对当前焦点的任何窗体都能实现怎么办呀
      

  4.   

    >>> hhjjhjhj(大头)我试了你的代码,可以实现,但只能对本窗体实现.我想要它能对当前焦点的任何窗体都能实现怎么办呀
      

  5.   

    //我试了你的代码,可以实现,但只能对本窗体实现.VB可能没有方法,可能要做成全局API钩子
      

  6.   

    wh_jounalrecord钩子获取当前活动窗口,可以的
    不用全局钩子
      

  7.   

    Winamp5 艾子哥经典豪华版...
    其实Winamp5用的标题栏是它自己的,所以可以,系统赋予窗体的就不太容易了
      

  8.   

    >>>Winamp5 艾子哥经典豪华版...
    >>>其实Winamp5用的标题栏是它自己的,所以可以,系统赋予窗体的就不太容易了
    不是的呀只要打开着艾子哥呢,任何当前获得焦点的窗体的标题栏上都有的呀,那些可都不是艾子哥自己的窗体了
      

  9.   

    //难道是在ncpaint中重画?也许,这个例子我看到过。只是忘了在哪儿看见的了。