Nova solução, muito mais simples, usando o AutoHotKey
Estou procurando uma maneira de evitar congelamentos causados por verificações constantes da janela ativa e também uma solução de código geral mais simples. Aqui está o que eu fiz:
#ifWinActive ahk_class Framework::CFrame
WheelUp::
ControlGetFocus, control, A
Loop 3
SendMessage, 0x115, 0, 0, %control%, A
Return
#ifWinActive ahk_class Framework::CFrame
WheelDown::
ControlGetFocus, control, A
Loop 3
SendMessage, 0x115, 1, 0, %control%, A
Return
Altere o Loop 3
para Loop 5
ou quantas linhas desejar para um movimento da roda rolar.
solução antiga usando AutoIt
Observação: esta solução pode causar gagueira e congelamentos ocasionais em CPUs mais antigas ou com carga total da CPU.
Eu peguei um script AutoIt que detecta o cursor dentro da área da lista de notas e, se você rolar para cima ou para baixo, clica nos botões "para cima" e "para baixo" na barra de rolagem. Isso funciona com várias janelas do OneNote. Você não precisa se concentrar na janela para que a rolagem funcione, apenas passe o mouse sobre essa lista:
#include <AutoItConstants.au3>
#include <MsgBoxConstants.au3>
#include <MouseOnEvent.au3> ; get it from https://www.autoitscript.com/forum/topic/64738-mouseonevent-udf/
#include <WinAPI.au3>
#include <Misc.au3>
_MouseSetOnEvent($MOUSE_WHEELSCROLLUP_EVENT, 'MouseWUp')
_MouseSetOnEvent($MOUSE_WHEELSCROLLDOWN_EVENT, 'MouseWDown')
Opt("WinTitleMatchMode", 2) ; 2 - substring mode
Global $clicks = 3 ; how many items to scroll per scroll, change this to 5, 8, 13, 42 or however fast you wanna scroll
Func MouseWUp($iEvent)
If StringInStr(WinGetTitle($hWin), " - OneNote") Then
; check if cursor is within the notes list and not anywhere else
Local $cur = MouseGetPos()
Local $handle = $hWin ;WinGetHandle("[ACTIVE]")
Local $wpos = WinGetPos($handle) ; returna array of x,y,w,h
Local $cpos = ControlGetPos ($handle, "", "[CLASS:NetUIHWND; INSTANCE:5]")
;MsgBox($MB_SYSTEMMODAL, "title", "win2:" & $wpos[2] & "; win3:" & $wpos[3] & "; cp2:" & $cpos[2] & "; cp3:" & $cpos[3])
If ($cur[0] > $wpos[0] + $wpos[2] - $cpos[2]) And ($cur[0] < $wpos[0] + $wpos[2]) Then
Local $x = $cpos[2] - 2
Local $y = 58
ControlClick($handle, "", "[CLASS:NetUIHWND; INSTANCE:5]", "left", $clicks, $x, $y)
EndIf
EndIf
EndFunc
Func MouseWDown($iEvent)
If StringInStr(WinGetTitle($hWin), " - OneNote") Then
Local $cur = MouseGetPos()
Local $handle = $hWin ;WinGetHandle("[ACTIVE]")
Local $wpos = WinGetPos($handle)
Local $cpos = ControlGetPos ($handle, "", "[CLASS:NetUIHWND; INSTANCE:5]")
If ($cur[0] > $wpos[0] + $wpos[2] - $cpos[2]) And ($cur[0] < $wpos[0] + $wpos[2]) Then
Local $x = $cpos[2] - 2
Local $y = $cpos[3] - 8
ControlClick($handle, "", "[CLASS:NetUIHWND; INSTANCE:5]", "left", $clicks, $x, $y)
EndIf
EndIf
EndFunc
; https://stackoverflow.com/a/11270659
Func _WindowFromPoint($iX,$iY)
Local $stInt64,$aRet,$stPoint=DllStructCreate("long;long")
DllStructSetData($stPoint,1,$iX)
DllStructSetData($stPoint,2,$iY)
$stInt64=DllStructCreate("int64",DllStructGetPtr($stPoint))
$aRet=DllCall("user32.dll","hwnd","WindowFromPoint","int64",DllStructGetData($stInt64,1))
If @error Then Return SetError(2,@error,0)
If $aRet[0]=0 Then Return SetError(3,0,0)
Return $aRet[0]
EndFunc
Local $hControl, $hWin, $hOldWin, $aMousePos
$hOldWin = ""
While True ;Not _IsPressed("1B")
$aMousePos = MouseGetPos()
$hControl=_WindowFromPoint($aMousePos[0],$aMousePos[1])
; Since _WindowFromPoint() can return 'sub' windows, or control handles, we should seek the owner window
$hWin=_WinAPI_GetAncestor($hControl,2)
If $hWin <> $hOldWin Then
;TrayTip("Window Info","Window under mouse = " & WinGetTitle($hWin), 1)
$hOldWin = $hWin
EndIf
Sleep(10)
WEnd
Ainda não descobri como evitar que os pergaminhos afetem a própria página da anotação. Se alguém encontrar uma maneira de impedir que o evento de rolagem atinja a área da nota principal de dentro da área da lista de anotações, compartilhe sua solução para que possamos combinar os scripts para obter melhores resultados.