mirror of
https://github.com/soukoku/ntwain.git
synced 2025-04-05 20:59:23 +08:00
Experiment with a winform-based msg pump.
This commit is contained in:
parent
24d4471e0f
commit
502013ee57
@ -18,6 +18,8 @@ namespace WinFormSample
|
||||
{
|
||||
public partial class Form1 : Form
|
||||
{
|
||||
bool useDiyPump = true;
|
||||
MessagePumpThread pump;
|
||||
TwainAppSession twain;
|
||||
readonly string saveFolder;
|
||||
readonly Stopwatch watch = new();
|
||||
@ -34,9 +36,10 @@ namespace WinFormSample
|
||||
var libVer = FileVersionInfo.GetVersionInfo(typeof(TwainAppSession).Assembly.Location).ProductVersion;
|
||||
Text += $"{(TWPlatform.Is32bit ? " 32bit" : " 64bit")} on NTwain {libVer}";
|
||||
|
||||
pump = new MessagePumpThread();
|
||||
TWPlatform.PreferLegacyDSM = false;
|
||||
|
||||
twain = new TwainAppSession(SynchronizationContext.Current!, Assembly.GetExecutingAssembly().Location);
|
||||
twain = new TwainAppSession(Assembly.GetExecutingAssembly().Location);
|
||||
twain.StateChanged += Twain_StateChanged;
|
||||
twain.DefaultSourceChanged += Twain_DefaultSourceChanged;
|
||||
twain.CurrentSourceChanged += Twain_CurrentSourceChanged;
|
||||
@ -62,11 +65,14 @@ namespace WinFormSample
|
||||
|
||||
private void Twain_SourceDisabled(TwainAppSession sender, TW_IDENTITY_LEGACY e)
|
||||
{
|
||||
if (watch.IsRunning)
|
||||
BeginInvoke(() =>
|
||||
{
|
||||
watch.Stop();
|
||||
MessageBox.Show($"Took {watch.Elapsed} to finish that transfer.");
|
||||
}
|
||||
if (watch.IsRunning)
|
||||
{
|
||||
watch.Stop();
|
||||
MessageBox.Show($"Took {watch.Elapsed} to finish that transfer.");
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void SystemEvents_SessionSwitch(object sender, SessionSwitchEventArgs e)
|
||||
@ -85,18 +91,31 @@ namespace WinFormSample
|
||||
{
|
||||
base.OnHandleCreated(e);
|
||||
|
||||
|
||||
var hwnd = this.Handle;
|
||||
var rc = twain.OpenDSM(hwnd);
|
||||
twain.AddWinformFilter();
|
||||
Debug.WriteLine($"OpenDSM={rc}");
|
||||
if (useDiyPump)
|
||||
{
|
||||
_ = pump.AttachAsync(twain);
|
||||
}
|
||||
else
|
||||
{
|
||||
var hwnd = this.Handle;
|
||||
var rc = twain.OpenDSM(hwnd, SynchronizationContext.Current!);
|
||||
twain.AddWinformFilter();
|
||||
Debug.WriteLine($"OpenDSM={rc}");
|
||||
}
|
||||
}
|
||||
|
||||
protected override void OnClosing(CancelEventArgs e)
|
||||
{
|
||||
var finalState = twain.TryStepdown(STATE.S2);
|
||||
Debug.WriteLine($"Stepdown result state={finalState}");
|
||||
twain.RemoveWinformFilter();
|
||||
if (useDiyPump)
|
||||
{
|
||||
pump.Detatch();
|
||||
}
|
||||
else
|
||||
{
|
||||
var finalState = twain.TryStepdown(STATE.S2);
|
||||
Debug.WriteLine($"Stepdown result state={finalState}");
|
||||
twain.RemoveWinformFilter();
|
||||
}
|
||||
base.OnClosing(e);
|
||||
}
|
||||
|
||||
@ -214,39 +233,42 @@ namespace WinFormSample
|
||||
|
||||
private void Twain_DefaultSourceChanged(TwainAppSession sender, TW_IDENTITY_LEGACY ds)
|
||||
{
|
||||
lblDefault.Text = ds.ProductName;
|
||||
BeginInvoke(() => lblDefault.Text = ds.ProductName);
|
||||
}
|
||||
|
||||
private void Twain_StateChanged(TwainAppSession sender, STATE state)
|
||||
{
|
||||
Invoke(() => lblState.Text = state.ToString());
|
||||
BeginInvoke(() => lblState.Text = state.ToString());
|
||||
}
|
||||
|
||||
private void Twain_CurrentSourceChanged(TwainAppSession sender, TW_IDENTITY_LEGACY ds)
|
||||
{
|
||||
lblCurrent.Text = ds.ToString();
|
||||
if (twain.State == STATE.S4)
|
||||
BeginInvoke(() =>
|
||||
{
|
||||
LoadCapInfoList();
|
||||
lblCurrent.Text = ds.ToString();
|
||||
if (twain.State == STATE.S4)
|
||||
{
|
||||
LoadCapInfoList();
|
||||
|
||||
// never seen a driver support these but here it is to test it
|
||||
if (twain.GetCapLabel(CAP.ICAP_SUPPORTEDSIZES, out string? test).RC == TWRC.SUCCESS)
|
||||
{
|
||||
Debug.WriteLine($"Supported sizes label from ds = {test}");
|
||||
// never seen a driver support these but here it is to test it
|
||||
if (twain.GetCapLabel(CAP.ICAP_SUPPORTEDSIZES, out string? test).RC == TWRC.SUCCESS)
|
||||
{
|
||||
Debug.WriteLine($"Supported sizes label from ds = {test}");
|
||||
}
|
||||
if (twain.GetCapHelp(CAP.ICAP_SUPPORTEDSIZES, out string? test2).RC == TWRC.SUCCESS)
|
||||
{
|
||||
Debug.WriteLine($"Supported sizes help from ds = {test2}");
|
||||
}
|
||||
if (twain.GetCapLabelEnum(CAP.ICAP_SUPPORTEDSIZES, out IList<string>? test3).RC == TWRC.SUCCESS && test3 != null)
|
||||
{
|
||||
Debug.WriteLine($"Supported sizes label enum from ds = {string.Join(Environment.NewLine, test3)}");
|
||||
}
|
||||
}
|
||||
if (twain.GetCapHelp(CAP.ICAP_SUPPORTEDSIZES, out string? test2).RC == TWRC.SUCCESS)
|
||||
else
|
||||
{
|
||||
Debug.WriteLine($"Supported sizes help from ds = {test2}");
|
||||
capListView.Items.Clear();
|
||||
}
|
||||
if (twain.GetCapLabelEnum(CAP.ICAP_SUPPORTEDSIZES, out IList<string>? test3).RC == TWRC.SUCCESS && test3 != null)
|
||||
{
|
||||
Debug.WriteLine($"Supported sizes label enum from ds = {string.Join(Environment.NewLine, test3)}");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
capListView.Items.Clear();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void LoadCapInfoList()
|
||||
|
@ -16,14 +16,14 @@ namespace WinFormSample
|
||||
[STAThread]
|
||||
static void Main()
|
||||
{
|
||||
if (DsmLoader.TryUseCustomDSM())
|
||||
{
|
||||
Debug.WriteLine("Using our own dsm now :)");
|
||||
}
|
||||
else
|
||||
{
|
||||
Debug.WriteLine("Will attempt to use default dsm :(");
|
||||
}
|
||||
//if (DsmLoader.TryUseCustomDSM())
|
||||
//{
|
||||
// Debug.WriteLine("Using our own dsm now :)");
|
||||
//}
|
||||
//else
|
||||
//{
|
||||
// Debug.WriteLine("Will attempt to use default dsm :(");
|
||||
//}
|
||||
|
||||
// To customize application configuration such as set high DPI settings or default font,
|
||||
// see https://aka.ms/applicationconfiguration.
|
||||
|
@ -1,6 +1,7 @@
|
||||
using NTwain.Data;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Reflection;
|
||||
@ -8,7 +9,7 @@ using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace WinFormSample
|
||||
namespace NTwain.DSM
|
||||
{
|
||||
/// <summary>
|
||||
/// For demoing loading dsm from custom path in case
|
||||
@ -19,15 +20,26 @@ namespace WinFormSample
|
||||
{
|
||||
static IntPtr __dllPtr;
|
||||
|
||||
public static bool TryUseCustomDSM()
|
||||
public static bool TryLoadCustomDSM()
|
||||
{
|
||||
if (__dllPtr == IntPtr.Zero)
|
||||
{
|
||||
var curFile = Assembly.GetExecutingAssembly().Location;
|
||||
|
||||
var dll = Path.Combine(
|
||||
Path.GetDirectoryName(Environment.ProcessPath ?? Assembly.GetExecutingAssembly().Location)!,
|
||||
Path.GetDirectoryName(curFile)!,
|
||||
$@"runtimes\win-{(TWPlatform.Is32bit ? "x86" : "x64")}\native\TWAINDSM.dll");
|
||||
|
||||
__dllPtr = LoadLibraryW(dll);
|
||||
|
||||
if (__dllPtr != IntPtr.Zero)
|
||||
{
|
||||
Debug.WriteLine("Using our own dsm now :)");
|
||||
}
|
||||
else
|
||||
{
|
||||
Debug.WriteLine("Will attempt to use default dsm :(");
|
||||
}
|
||||
}
|
||||
return __dllPtr != IntPtr.Zero;
|
||||
}
|
@ -660,7 +660,7 @@ namespace NTwain.Data
|
||||
/// <summary>
|
||||
/// A simplified check on whether this has valid data from DSM.
|
||||
/// </summary>
|
||||
public bool HasValue => Id == 0 && ProtocolMajor == 0 && ProtocolMinor == 0;
|
||||
public bool HasValue => Id != 0 && ProtocolMajor != 0 && ProtocolMinor != 0;
|
||||
|
||||
public override string ToString()
|
||||
{
|
||||
|
100
src/NTwain/MessagePumpThread.cs
Normal file
100
src/NTwain/MessagePumpThread.cs
Normal file
@ -0,0 +1,100 @@
|
||||
#if WINDOWS || NETFRAMEWORK
|
||||
using NTwain.Data;
|
||||
using System;
|
||||
using System.Diagnostics;
|
||||
using System.Threading;
|
||||
using System.Threading.Tasks;
|
||||
using System.Windows.Forms;
|
||||
|
||||
namespace NTwain
|
||||
{
|
||||
/// <summary>
|
||||
/// For use under Windows to host a message pump in non-winform/wpf apps.
|
||||
/// This is highly experimental.
|
||||
/// </summary>
|
||||
public class MessagePumpThread
|
||||
{
|
||||
DummyForm? _dummyForm;
|
||||
TwainAppSession? _twain;
|
||||
|
||||
public bool IsRunning => _dummyForm != null && _dummyForm.IsHandleCreated;
|
||||
|
||||
/// <summary>
|
||||
/// Starts the thread, attaches a twain session to it,
|
||||
/// and opens the DSM.
|
||||
/// </summary>
|
||||
/// <param name="twain"></param>
|
||||
/// <returns></returns>
|
||||
/// <exception cref="InvalidOperationException"></exception>
|
||||
public async Task<STS> AttachAsync(TwainAppSession twain)
|
||||
{
|
||||
if (twain.State > STATE.S2) throw new InvalidOperationException("Cannot attach to an opened TWAIN session.");
|
||||
if (_twain != null || _dummyForm != null) throw new InvalidOperationException("Already attached previously.");
|
||||
|
||||
Thread t = new(RunMessagePump);
|
||||
t.IsBackground = true;
|
||||
t.SetApartmentState(ApartmentState.STA);
|
||||
t.Start();
|
||||
|
||||
while (_dummyForm == null || !_dummyForm.IsHandleCreated)
|
||||
{
|
||||
await Task.Delay(100);
|
||||
}
|
||||
|
||||
STS sts = default;
|
||||
ManualResetEventSlim evt = new();
|
||||
_dummyForm.BeginInvoke(() =>
|
||||
{
|
||||
sts = twain.OpenDSM(_dummyForm.Handle, SynchronizationContext.Current!);
|
||||
if (sts.IsSuccess)
|
||||
{
|
||||
twain.AddWinformFilter();
|
||||
_twain = twain;
|
||||
}
|
||||
evt.Set();
|
||||
});
|
||||
evt.Wait();
|
||||
return sts;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Detatches a previously attached session and stops the thread.
|
||||
/// </summary>
|
||||
public void Detatch()
|
||||
{
|
||||
if (_dummyForm != null && _twain != null)
|
||||
{
|
||||
_dummyForm.BeginInvoke(_dummyForm.Close);
|
||||
}
|
||||
}
|
||||
|
||||
void RunMessagePump()
|
||||
{
|
||||
Debug.WriteLine("TWAIN pump thread starting");
|
||||
_dummyForm = new DummyForm();
|
||||
_dummyForm.FormClosed += (s, e) =>
|
||||
{
|
||||
if (_twain != null)
|
||||
{
|
||||
_twain.TryStepdown(STATE.S4);
|
||||
_twain.RemoveWinformFilter();
|
||||
_twain.CloseDSM();
|
||||
_twain = null;
|
||||
}
|
||||
_dummyForm = null;
|
||||
};
|
||||
Application.Run(_dummyForm);
|
||||
Debug.WriteLine("TWAIN pump thread ended");
|
||||
}
|
||||
|
||||
class DummyForm : Form
|
||||
{
|
||||
public DummyForm()
|
||||
{
|
||||
ShowInTaskbar = false;
|
||||
WindowState = FormWindowState.Minimized;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
#endif
|
@ -21,6 +21,18 @@
|
||||
<ItemGroup Condition=" '$(TargetFramework)' != 'net462'">
|
||||
<PackageReference Include="System.Text.Encoding.CodePages" Version="7.0.0" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Remove="runtimes\win-x64\native\TWAINDSM.dll" />
|
||||
<None Remove="runtimes\win-x86\native\TWAINDSM.dll" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Content Include="runtimes\win-x64\native\TWAINDSM.dll">
|
||||
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
||||
</Content>
|
||||
<Content Include="runtimes\win-x86\native\TWAINDSM.dll">
|
||||
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
||||
</Content>
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<None Update="DSM\DSMGenerator.dummy">
|
||||
|
@ -55,5 +55,43 @@ namespace NTwain.Native
|
||||
GPTR = GMEM_FIXED | GMEM_ZEROINIT,
|
||||
GHND = GMEM_MOVEABLE | GMEM_ZEROINIT
|
||||
}
|
||||
|
||||
// [Flags]
|
||||
// public enum PEEK_MESSAGE_REMOVE_TYPE : uint
|
||||
// {
|
||||
// PM_NOREMOVE = 0x00000000,
|
||||
// PM_REMOVE = 0x00000001,
|
||||
// PM_NOYIELD = 0x00000002,
|
||||
// PM_QS_INPUT = 0x04070000,
|
||||
// PM_QS_POSTMESSAGE = 0x00980000,
|
||||
// PM_QS_PAINT = 0x00200000,
|
||||
// PM_QS_SENDMESSAGE = 0x00400000,
|
||||
// }
|
||||
|
||||
//#if NET7_0_OR_GREATER
|
||||
// [LibraryImport("USER32.dll")]
|
||||
// public static partial int PeekMessageW(ref WIN_MESSAGE lpMsg, IntPtr hWnd, uint wMsgFilterMin, uint wMsgFilterMax, PEEK_MESSAGE_REMOVE_TYPE wRemoveMsg);
|
||||
|
||||
// [LibraryImport("USER32.dll", SetLastError = true)]
|
||||
// public static partial int GetMessageW(ref WIN_MESSAGE lpMsg, IntPtr hWnd, uint wMsgFilterMin, uint wMsgFilterMax);
|
||||
|
||||
// [LibraryImport("USER32.dll")]
|
||||
// public static partial int TranslateMessage(ref WIN_MESSAGE lpMsg);
|
||||
|
||||
// [LibraryImport("USER32.dll")]
|
||||
// public static partial nint DispatchMessageW(ref WIN_MESSAGE lpMsg);
|
||||
//#else
|
||||
// [DllImport("USER32.dll")]
|
||||
// public static extern int PeekMessageW(ref WIN_MESSAGE lpMsg, IntPtr hWnd, uint wMsgFilterMin, uint wMsgFilterMax, PEEK_MESSAGE_REMOVE_TYPE wRemoveMsg);
|
||||
|
||||
// [DllImport("USER32.dll", SetLastError = true)]
|
||||
// public static extern int GetMessageW(ref WIN_MESSAGE lpMsg, IntPtr hWnd, uint wMsgFilterMin, uint wMsgFilterMax);
|
||||
|
||||
// [DllImport("USER32.dll")]
|
||||
// public static extern int TranslateMessage(ref WIN_MESSAGE lpMsg);
|
||||
|
||||
// [DllImport("USER32.dll")]
|
||||
// public static extern nint DispatchMessageW(ref WIN_MESSAGE lpMsg);
|
||||
//#endif
|
||||
}
|
||||
}
|
||||
|
@ -112,24 +112,17 @@ namespace NTwain
|
||||
if (!_inTransfer)
|
||||
{
|
||||
// this should be done on ui thread (or same one that enabled the ds)
|
||||
_uiThreadMarshaller.Post(obj =>
|
||||
{
|
||||
((TwainAppSession)obj!).DisableSource();
|
||||
}, this);
|
||||
DisableSource();
|
||||
}
|
||||
break;
|
||||
case MSG.DEVICEEVENT:
|
||||
if (DeviceEvent != null && DGControl.DeviceEvent.Get(ref _appIdentity, ref _currentDS, out TW_DEVICEEVENT de) == TWRC.SUCCESS)
|
||||
{
|
||||
_uiThreadMarshaller.Post(obj =>
|
||||
try
|
||||
{
|
||||
try
|
||||
{
|
||||
var twain = (TwainAppSession)obj!;
|
||||
twain.DeviceEvent!.Invoke(twain, de);
|
||||
}
|
||||
catch { }
|
||||
}, this);
|
||||
DeviceEvent.Invoke(this, de);
|
||||
}
|
||||
catch { }
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
@ -57,14 +57,11 @@ namespace NTwain
|
||||
if (_state != value)
|
||||
{
|
||||
_state = value;
|
||||
_uiThreadMarshaller.Send(obj =>
|
||||
try
|
||||
{
|
||||
try
|
||||
{
|
||||
((TwainAppSession)obj!).StateChanged?.Invoke(this, value);
|
||||
}
|
||||
catch { }
|
||||
}, this);
|
||||
StateChanged?.Invoke(this, value);
|
||||
}
|
||||
catch { }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -105,6 +105,18 @@ namespace NTwain
|
||||
/// Otherwise capturing will begin after this.</param>
|
||||
/// <returns></returns>
|
||||
public STS EnableSource(bool showUI, bool uiOnly)
|
||||
{
|
||||
if (_pumpThreadMarshaller == null) return EnableSourceReal(showUI, uiOnly);
|
||||
|
||||
var sts = default(STS);
|
||||
_pumpThreadMarshaller.Send(_ =>
|
||||
{
|
||||
sts = EnableSourceReal(showUI, uiOnly);
|
||||
}, null);
|
||||
return sts;
|
||||
}
|
||||
|
||||
private STS EnableSourceReal(bool showUI, bool uiOnly)
|
||||
{
|
||||
if (State > STATE.S4)
|
||||
{
|
||||
@ -136,6 +148,17 @@ namespace NTwain
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public STS DisableSource()
|
||||
{
|
||||
if (_pumpThreadMarshaller == null) return DisableSourceReal();
|
||||
|
||||
var sts = default(STS);
|
||||
_pumpThreadMarshaller.Send(_ =>
|
||||
{
|
||||
sts = DisableSourceReal();
|
||||
}, null);
|
||||
return sts;
|
||||
}
|
||||
STS DisableSourceReal()
|
||||
{
|
||||
_closeDsRequested = true;
|
||||
var rc = DGControl.UserInterface.DisableDS(ref _appIdentity, ref _currentDS, ref _userInterface);
|
||||
|
@ -67,18 +67,11 @@ namespace NTwain
|
||||
do
|
||||
{
|
||||
var readyArgs = new TransferReadyEventArgs(pending.Count, (TWEJ)pending.EOJ);
|
||||
if (TransferReady != null)
|
||||
try
|
||||
{
|
||||
_uiThreadMarshaller.Send(_ =>
|
||||
{
|
||||
try
|
||||
{
|
||||
|
||||
TransferReady.Invoke(this, readyArgs);
|
||||
}
|
||||
catch { } // don't let consumer kill the loop if they have exception
|
||||
}, null);
|
||||
TransferReady?.Invoke(this, readyArgs);
|
||||
}
|
||||
catch { } // don't let consumer kill the loop if they have exception
|
||||
|
||||
if (readyArgs.Cancel == CancelType.EndNow || _closeDsRequested)
|
||||
{
|
||||
@ -150,27 +143,21 @@ namespace NTwain
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_uiThreadMarshaller.Send(obj =>
|
||||
try
|
||||
{
|
||||
try
|
||||
{
|
||||
var twain = ((TwainAppSession)obj!);
|
||||
twain.TransferError?.Invoke(twain, new TransferErrorEventArgs(ex));
|
||||
}
|
||||
catch { }
|
||||
}, this);
|
||||
TransferError?.Invoke(this, new TransferErrorEventArgs(ex));
|
||||
}
|
||||
catch { }
|
||||
}
|
||||
}
|
||||
} while (sts.RC == TWRC.SUCCESS && pending.Count != 0);
|
||||
}
|
||||
|
||||
HandleXferCode(ref sts, ref pending);
|
||||
|
||||
if (State >= STATE.S5)
|
||||
{
|
||||
_uiThreadMarshaller.Send(obj =>
|
||||
{
|
||||
((TwainAppSession)obj!).DisableSource();
|
||||
}, this);
|
||||
DisableSource();
|
||||
}
|
||||
_inTransfer = false;
|
||||
}
|
||||
@ -185,11 +172,11 @@ namespace NTwain
|
||||
break;
|
||||
case TWRC.CANCEL:
|
||||
// might eventually have option to cancel this or all like transfer ready
|
||||
_uiThreadMarshaller.Send(obj =>
|
||||
try
|
||||
{
|
||||
var twain = ((TwainAppSession)obj!);
|
||||
twain.TransferCanceled?.Invoke(twain, new TransferCanceledEventArgs());
|
||||
}, this);
|
||||
TransferCanceled?.Invoke(this, new TransferCanceledEventArgs());
|
||||
}
|
||||
catch { }
|
||||
sts = WrapInSTS(DGControl.PendingXfers.EndXfer(ref _appIdentity, ref _currentDS, ref pending));
|
||||
sts = WrapInSTS(DGControl.PendingXfers.Reset(ref _appIdentity, ref _currentDS, ref pending));
|
||||
if (sts.RC == TWRC.SUCCESS) State = STATE.S5;
|
||||
|
@ -19,26 +19,24 @@ namespace NTwain
|
||||
/// <summary>
|
||||
/// Creates TWAIN session with app info derived an executable file.
|
||||
/// </summary>
|
||||
/// <param name="uiThreadMarshaller"></param>
|
||||
/// <param name="exeFilePath"></param>
|
||||
/// <param name="appLanguage"></param>
|
||||
/// <param name="appCountry"></param>
|
||||
public TwainAppSession(SynchronizationContext uiThreadMarshaller,
|
||||
public TwainAppSession(
|
||||
string exeFilePath,
|
||||
TWLG appLanguage = TWLG.ENGLISH_USA, TWCY appCountry = TWCY.USA) :
|
||||
this(uiThreadMarshaller, FileVersionInfo.GetVersionInfo(exeFilePath), appLanguage, appCountry)
|
||||
this(FileVersionInfo.GetVersionInfo(exeFilePath), appLanguage, appCountry)
|
||||
{ }
|
||||
/// <summary>
|
||||
/// Creates TWAIN session with app info derived from a <see cref="FileVersionInfo"/> object.
|
||||
/// </summary>
|
||||
/// <param name="uiThreadMarshaller"></param>
|
||||
/// <param name="appInfo"></param>
|
||||
/// <param name="appLanguage"></param>
|
||||
/// <param name="appCountry"></param>
|
||||
public TwainAppSession(SynchronizationContext uiThreadMarshaller,
|
||||
public TwainAppSession(
|
||||
FileVersionInfo appInfo,
|
||||
TWLG appLanguage = TWLG.ENGLISH_USA, TWCY appCountry = TWCY.USA) :
|
||||
this(uiThreadMarshaller,
|
||||
this(
|
||||
appInfo.CompanyName ?? "",
|
||||
appInfo.ProductName ?? "",
|
||||
appInfo.ProductName ?? "",
|
||||
@ -48,7 +46,6 @@ namespace NTwain
|
||||
/// <summary>
|
||||
/// Creates TWAIN session with explicit app info.
|
||||
/// </summary>
|
||||
/// <param name="uiThreadMarshaller"></param>
|
||||
/// <param name="companyName"></param>
|
||||
/// <param name="productFamily"></param>
|
||||
/// <param name="productName"></param>
|
||||
@ -57,12 +54,15 @@ namespace NTwain
|
||||
/// <param name="appLanguage"></param>
|
||||
/// <param name="appCountry"></param>
|
||||
/// <param name="supportedTypes"></param>
|
||||
public TwainAppSession(SynchronizationContext uiThreadMarshaller,
|
||||
public TwainAppSession(
|
||||
string companyName, string productFamily, string productName,
|
||||
Version productVersion, string productDescription = "",
|
||||
TWLG appLanguage = TWLG.ENGLISH_USA, TWCY appCountry = TWCY.USA,
|
||||
DG supportedTypes = DG.IMAGE)
|
||||
{
|
||||
#if WINDOWS || NETFRAMEWORK
|
||||
DSM.DsmLoader.TryLoadCustomDSM();
|
||||
#endif
|
||||
// todo: find a better place for this
|
||||
if (!__encodingRegistered)
|
||||
{
|
||||
@ -93,7 +93,6 @@ namespace NTwain
|
||||
_legacyCallbackDelegate = LegacyCallbackHandler;
|
||||
_osxCallbackDelegate = OSXCallbackHandler;
|
||||
|
||||
_uiThreadMarshaller = uiThreadMarshaller;
|
||||
StartTransferThread();
|
||||
}
|
||||
|
||||
@ -104,7 +103,7 @@ namespace NTwain
|
||||
#endif
|
||||
// test threads a bit
|
||||
//readonly BlockingCollection<MSG> _bgPendingMsgs = new();
|
||||
private readonly SynchronizationContext _uiThreadMarshaller;
|
||||
SynchronizationContext? _pumpThreadMarshaller;
|
||||
bool _closeDsRequested;
|
||||
bool _inTransfer;
|
||||
readonly AutoResetEvent _xferReady = new(false);
|
||||
@ -176,13 +175,15 @@ namespace NTwain
|
||||
/// Loads and opens the TWAIN data source manager.
|
||||
/// </summary>
|
||||
/// <param name="hwnd">Required if on Windows.</param>
|
||||
/// <param name="uiThreadMarshaller">Context for TWAIN to invoke certain actions on the thread that the hwnd lives on.</param>
|
||||
/// <returns></returns>
|
||||
public STS OpenDSM(IntPtr hwnd)
|
||||
public STS OpenDSM(IntPtr hwnd, SynchronizationContext uiThreadMarshaller)
|
||||
{
|
||||
var rc = DGControl.Parent.OpenDSM(ref _appIdentity, hwnd);
|
||||
if (rc == TWRC.SUCCESS)
|
||||
{
|
||||
_hwnd = hwnd;
|
||||
_pumpThreadMarshaller = uiThreadMarshaller;
|
||||
State = STATE.S3;
|
||||
// get default source
|
||||
if (DGControl.Identity.GetDefault(ref _appIdentity, out TW_IDENTITY_LEGACY ds) == TWRC.SUCCESS)
|
||||
@ -223,6 +224,7 @@ namespace NTwain
|
||||
}
|
||||
catch { }
|
||||
_hwnd = IntPtr.Zero;
|
||||
_pumpThreadMarshaller = null;
|
||||
}
|
||||
return WrapInSTS(rc, true);
|
||||
}
|
||||
|
BIN
src/NTwain/runtimes/win-x64/native/TWAINDSM.dll
Normal file
BIN
src/NTwain/runtimes/win-x64/native/TWAINDSM.dll
Normal file
Binary file not shown.
BIN
src/NTwain/runtimes/win-x86/native/TWAINDSM.dll
Normal file
BIN
src/NTwain/runtimes/win-x86/native/TWAINDSM.dll
Normal file
Binary file not shown.
Loading…
Reference in New Issue
Block a user