Я использовал этот код, чтобы правильно получить текущую и максимальную позиции:
const int SB_HORZ = 0x0000;
const int SB_VERT = 0x0001;
const int WM_HSCROLL = 0x0114;
const int WM_VSCROLL = 0x0115;
const int SB_THUMBPOSITION = 4;
private enum ScrollInfoMask : uint
{
SIF_RANGE = 0x1,
SIF_PAGE = 0x2,
SIF_POS = 0x4,
SIF_DISABLENOSCROLL = 0x8,
SIF_TRACKPOS = 0x10,
SIF_ALL = (SIF_RANGE | SIF_PAGE | SIF_POS | SIF_TRACKPOS)
}
[StructLayout(LayoutKind.Sequential)]
struct SCROLLINFO
{
public uint cbSize;
public uint fMask;
public int nMin;
public int nMax;
public uint nPage;
public int nPos;
public int nTrackPos;
}
public int HScrollPosition
{
get
{
return GetScrollPos(Handle, SB_HORZ);
}
set
{
SetScrollPos((IntPtr)Handle, SB_HORZ, value, true);
PostMessageA((IntPtr)Handle, WM_HSCROLL, SB_THUMBPOSITION + 0x10000 * value, 0);
}
}
public int VScrollPosition
{
get
{
return GetScrollPos(Handle, SB_VERT);
}
set
{
SetScrollPos((IntPtr)Handle, SB_VERT, value, true);
PostMessageA((IntPtr)Handle, WM_VSCROLL, SB_THUMBPOSITION + 0x10000 * value, 0);
}
}
public int HScrollPositionMax
{
get
{
SCROLLINFO scrollInfo = new SCROLLINFO();
scrollInfo.fMask = (uint)ScrollInfoMask.SIF_ALL;
scrollInfo.cbSize = (uint)Marshal.SizeOf(scrollInfo);
GetScrollInfo(Handle, SB_HORZ, ref scrollInfo);
return scrollInfo.nMax - (int)scrollInfo.nPage;
}
}
public int VScrollPositionMax
{
get
{
SCROLLINFO scrollInfo = new SCROLLINFO();
scrollInfo.fMask = (uint)ScrollInfoMask.SIF_ALL;
scrollInfo.cbSize = (uint)Marshal.SizeOf(scrollInfo);
GetScrollInfo(Handle, SB_VERT, ref scrollInfo);
return scrollInfo.nMax - (int)scrollInfo.nPage;
}
}
Утомительно получать «правильное» максимальное значение. Вы должны вычесть значение nPage (большое изменение) максимального значения, которое возвращает GetScrollRange, и затем вы можете правильно сравнить его со свойством HScrollPosition, например:
if (_rtb.VScrollPosition == _rtb.VScrollPositionMax)
{
Debug.WriteLine("Scroll is at the bottom most edge");
}