mirror of
https://github.com/soukoku/ntwain.git
synced 2025-04-05 20:59:23 +08:00
Attempt to read cap values.
This commit is contained in:
parent
513dc96f64
commit
ab2947d03b
66
samples/WinForm32/Form1.Designer.cs
generated
66
samples/WinForm32/Form1.Designer.cs
generated
@ -41,9 +41,11 @@
|
||||
lblState = new System.Windows.Forms.Label();
|
||||
label3 = new System.Windows.Forms.Label();
|
||||
btnOpenDef = new System.Windows.Forms.Button();
|
||||
listCaps = new System.Windows.Forms.ListBox();
|
||||
label4 = new System.Windows.Forms.Label();
|
||||
btnStart = new System.Windows.Forms.Button();
|
||||
btnShowSettings = new System.Windows.Forms.Button();
|
||||
btnClose = new System.Windows.Forms.Button();
|
||||
btnStart = new System.Windows.Forms.Button();
|
||||
((System.ComponentModel.ISupportInitialize)splitContainer1).BeginInit();
|
||||
splitContainer1.Panel1.SuspendLayout();
|
||||
splitContainer1.Panel2.SuspendLayout();
|
||||
@ -56,7 +58,7 @@
|
||||
btnSelect.Name = "btnSelect";
|
||||
btnSelect.Size = new System.Drawing.Size(151, 23);
|
||||
btnSelect.TabIndex = 0;
|
||||
btnSelect.Text = "Select default source";
|
||||
btnSelect.Text = "Choose default source";
|
||||
btnSelect.UseVisualStyleBackColor = true;
|
||||
btnSelect.Click += btnSelect_Click;
|
||||
//
|
||||
@ -111,15 +113,14 @@
|
||||
listSources.Anchor = System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom | System.Windows.Forms.AnchorStyles.Left | System.Windows.Forms.AnchorStyles.Right;
|
||||
listSources.FormattingEnabled = true;
|
||||
listSources.ItemHeight = 15;
|
||||
listSources.Location = new System.Drawing.Point(13, 130);
|
||||
listSources.Location = new System.Drawing.Point(13, 170);
|
||||
listSources.Name = "listSources";
|
||||
listSources.Size = new System.Drawing.Size(289, 379);
|
||||
listSources.Size = new System.Drawing.Size(297, 379);
|
||||
listSources.TabIndex = 6;
|
||||
//
|
||||
// btnSetDef
|
||||
//
|
||||
btnSetDef.Anchor = System.Windows.Forms.AnchorStyles.Bottom | System.Windows.Forms.AnchorStyles.Left;
|
||||
btnSetDef.Location = new System.Drawing.Point(12, 515);
|
||||
btnSetDef.Location = new System.Drawing.Point(13, 131);
|
||||
btnSetDef.Name = "btnSetDef";
|
||||
btnSetDef.Size = new System.Drawing.Size(169, 23);
|
||||
btnSetDef.TabIndex = 7;
|
||||
@ -129,10 +130,9 @@
|
||||
//
|
||||
// btnOpen
|
||||
//
|
||||
btnOpen.Anchor = System.Windows.Forms.AnchorStyles.Bottom | System.Windows.Forms.AnchorStyles.Left;
|
||||
btnOpen.Location = new System.Drawing.Point(187, 515);
|
||||
btnOpen.Location = new System.Drawing.Point(188, 131);
|
||||
btnOpen.Name = "btnOpen";
|
||||
btnOpen.Size = new System.Drawing.Size(104, 23);
|
||||
btnOpen.Size = new System.Drawing.Size(114, 23);
|
||||
btnOpen.TabIndex = 8;
|
||||
btnOpen.Text = "Open selected";
|
||||
btnOpen.UseVisualStyleBackColor = true;
|
||||
@ -160,13 +160,15 @@
|
||||
//
|
||||
// splitContainer1.Panel2
|
||||
//
|
||||
splitContainer1.Panel2.Controls.Add(listCaps);
|
||||
splitContainer1.Panel2.Controls.Add(label4);
|
||||
splitContainer1.Panel2.Controls.Add(btnStart);
|
||||
splitContainer1.Panel2.Controls.Add(btnShowSettings);
|
||||
splitContainer1.Panel2.Controls.Add(btnClose);
|
||||
splitContainer1.Panel2.Controls.Add(label2);
|
||||
splitContainer1.Panel2.Controls.Add(lblCurrent);
|
||||
splitContainer1.Size = new System.Drawing.Size(800, 564);
|
||||
splitContainer1.SplitterDistance = 305;
|
||||
splitContainer1.Size = new System.Drawing.Size(1023, 564);
|
||||
splitContainer1.SplitterDistance = 325;
|
||||
splitContainer1.TabIndex = 9;
|
||||
//
|
||||
// lblState
|
||||
@ -197,6 +199,34 @@
|
||||
btnOpenDef.UseVisualStyleBackColor = true;
|
||||
btnOpenDef.Click += btnOpenDef_Click;
|
||||
//
|
||||
// listCaps
|
||||
//
|
||||
listCaps.FormattingEnabled = true;
|
||||
listCaps.ItemHeight = 15;
|
||||
listCaps.Location = new System.Drawing.Point(13, 87);
|
||||
listCaps.Name = "listCaps";
|
||||
listCaps.Size = new System.Drawing.Size(234, 469);
|
||||
listCaps.TabIndex = 8;
|
||||
//
|
||||
// label4
|
||||
//
|
||||
label4.AutoSize = true;
|
||||
label4.Location = new System.Drawing.Point(13, 62);
|
||||
label4.Name = "label4";
|
||||
label4.Size = new System.Drawing.Size(91, 15);
|
||||
label4.TabIndex = 7;
|
||||
label4.Text = "Supported Caps";
|
||||
//
|
||||
// btnStart
|
||||
//
|
||||
btnStart.Location = new System.Drawing.Point(299, 32);
|
||||
btnStart.Name = "btnStart";
|
||||
btnStart.Size = new System.Drawing.Size(132, 23);
|
||||
btnStart.TabIndex = 6;
|
||||
btnStart.Text = "Start acquisition";
|
||||
btnStart.UseVisualStyleBackColor = true;
|
||||
btnStart.Click += btnStart_Click;
|
||||
//
|
||||
// btnShowSettings
|
||||
//
|
||||
btnShowSettings.Location = new System.Drawing.Point(161, 32);
|
||||
@ -217,21 +247,11 @@
|
||||
btnClose.UseVisualStyleBackColor = true;
|
||||
btnClose.Click += btnClose_Click;
|
||||
//
|
||||
// btnStart
|
||||
//
|
||||
btnStart.Location = new System.Drawing.Point(299, 32);
|
||||
btnStart.Name = "btnStart";
|
||||
btnStart.Size = new System.Drawing.Size(132, 23);
|
||||
btnStart.TabIndex = 6;
|
||||
btnStart.Text = "Start acquisition";
|
||||
btnStart.UseVisualStyleBackColor = true;
|
||||
btnStart.Click += btnStart_Click;
|
||||
//
|
||||
// Form1
|
||||
//
|
||||
AutoScaleDimensions = new System.Drawing.SizeF(7F, 15F);
|
||||
AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
|
||||
ClientSize = new System.Drawing.Size(800, 564);
|
||||
ClientSize = new System.Drawing.Size(1023, 564);
|
||||
Controls.Add(splitContainer1);
|
||||
Name = "Form1";
|
||||
Text = "TWAIN Test";
|
||||
@ -262,5 +282,7 @@
|
||||
private System.Windows.Forms.Label label3;
|
||||
private System.Windows.Forms.Button btnShowSettings;
|
||||
private System.Windows.Forms.Button btnStart;
|
||||
private System.Windows.Forms.ListBox listCaps;
|
||||
private System.Windows.Forms.Label label4;
|
||||
}
|
||||
}
|
@ -18,15 +18,6 @@ namespace WinFormSample
|
||||
var libVer = FileVersionInfo.GetVersionInfo(typeof(TwainAppSession).Assembly.Location).FileVersion;
|
||||
Text += $"{(TwainPlatform.Is32bit ? " 32bit" : " 64bit")} on NTwain {libVer}";
|
||||
|
||||
if (DsmLoader.TryUseCustomDSM())
|
||||
{
|
||||
Debug.WriteLine("Using our own dsm now :)");
|
||||
}
|
||||
else
|
||||
{
|
||||
Debug.WriteLine("Will attempt to use default dsm :(");
|
||||
}
|
||||
|
||||
TwainPlatform.PreferLegacyDSM = false;
|
||||
|
||||
twain = new TwainAppSession(new WinformMarshaller(this), Assembly.GetExecutingAssembly().Location);
|
||||
@ -45,6 +36,16 @@ namespace WinFormSample
|
||||
private void Twain_CurrentSourceChanged(TwainAppSession arg1, TW_IDENTITY_LEGACY ds)
|
||||
{
|
||||
lblCurrent.Text = ds.ProductName;
|
||||
if (twain.State == STATE.S4)
|
||||
{
|
||||
var caps = twain.GetAllCaps();
|
||||
foreach (var c in caps)
|
||||
listCaps.Items.Add(c);
|
||||
}
|
||||
else
|
||||
{
|
||||
listCaps.Items.Clear();
|
||||
}
|
||||
}
|
||||
|
||||
private void Twain_DefaultSourceChanged(TwainAppSession arg1, TW_IDENTITY_LEGACY ds)
|
||||
|
@ -1,4 +1,5 @@
|
||||
using System;
|
||||
using System.Diagnostics;
|
||||
using System.Windows.Forms;
|
||||
|
||||
namespace WinFormSample
|
||||
@ -11,6 +12,15 @@ 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 :(");
|
||||
}
|
||||
|
||||
// To customize application configuration such as set high DPI settings or default font,
|
||||
// see https://aka.ms/applicationconfiguration.
|
||||
ApplicationConfiguration.Initialize();
|
||||
|
@ -1,9 +1,7 @@
|
||||
using NTwain;
|
||||
using System;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Globalization;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Runtime.InteropServices.ComTypes;
|
||||
using System.Text;
|
||||
|
||||
namespace NTwain.Data
|
||||
{
|
||||
@ -236,89 +234,89 @@ namespace NTwain.Data
|
||||
//}
|
||||
}
|
||||
|
||||
///// <summary>
|
||||
///// A more dotnet-friendly representation of <see cref="TW_ENUMERATION"/>.
|
||||
///// </summary>
|
||||
///// <typeparam name="TValue"></typeparam>
|
||||
//public class Enumeration<TValue> where TValue : struct
|
||||
//{
|
||||
// public int CurrentIndex;
|
||||
/// <summary>
|
||||
/// A more dotnet-friendly representation of <see cref="TW_ENUMERATION"/>.
|
||||
/// </summary>
|
||||
/// <typeparam name="TValue"></typeparam>
|
||||
public class Enumeration<TValue> where TValue : struct
|
||||
{
|
||||
public int CurrentIndex;
|
||||
|
||||
// public int DefaultIndex;
|
||||
public int DefaultIndex;
|
||||
|
||||
// public TValue[] Items;
|
||||
//}
|
||||
public TValue[]? Items;
|
||||
}
|
||||
|
||||
///// <summary>
|
||||
///// A more dotnet-friendly representation of <see cref="TW_RANGE"/>.
|
||||
///// </summary>
|
||||
///// <typeparam name="TValue"></typeparam>
|
||||
//public partial class Range<TValue> : IEnumerable<TValue> where TValue : struct
|
||||
//{
|
||||
// public TValue MinValue;
|
||||
// public TValue MaxValue;
|
||||
// public TValue StepSize;
|
||||
// public TValue DefaultValue;
|
||||
// public TValue CurrentValue;
|
||||
/// <summary>
|
||||
/// A more dotnet-friendly representation of <see cref="TW_RANGE"/>.
|
||||
/// </summary>
|
||||
/// <typeparam name="TValue"></typeparam>
|
||||
public partial class Range<TValue> : IEnumerable<TValue> where TValue : struct
|
||||
{
|
||||
public TValue MinValue;
|
||||
public TValue MaxValue;
|
||||
public TValue StepSize;
|
||||
public TValue DefaultValue;
|
||||
public TValue CurrentValue;
|
||||
|
||||
// IEnumerator<TValue> IEnumerable<TValue>.GetEnumerator()
|
||||
// {
|
||||
// if (!(MinValue is IConvertible))
|
||||
// throw new NotSupportedException($"The value type {typeof(TValue).Name} is not supported for range enumeration.");
|
||||
IEnumerator<TValue> IEnumerable<TValue>.GetEnumerator()
|
||||
{
|
||||
if (!(MinValue is IConvertible))
|
||||
throw new NotSupportedException($"The value type {typeof(TValue).Name} is not supported for range enumeration.");
|
||||
|
||||
// return new DynamicEnumerator(MinValue, MaxValue, StepSize);
|
||||
// }
|
||||
return new DynamicEnumerator(MinValue, MaxValue, StepSize);
|
||||
}
|
||||
|
||||
// IEnumerator IEnumerable.GetEnumerator()
|
||||
// {
|
||||
// return ((IEnumerable<TValue>)this).GetEnumerator();
|
||||
// }
|
||||
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
|
||||
{
|
||||
return ((IEnumerable<TValue>)this).GetEnumerator();
|
||||
}
|
||||
|
||||
// // dynamic is a cheap hack to sidestep the compiler restrictions if I know TValue is numeric
|
||||
// class DynamicEnumerator : IEnumerator<TValue>
|
||||
// {
|
||||
// private readonly TValue _min;
|
||||
// private readonly TValue _max;
|
||||
// private readonly TValue _step;
|
||||
// private TValue _cur;
|
||||
// bool started = false;
|
||||
// dynamic is a cheap hack to sidestep the compiler restrictions if I know TValue is numeric
|
||||
class DynamicEnumerator : IEnumerator<TValue>
|
||||
{
|
||||
private readonly TValue _min;
|
||||
private readonly TValue _max;
|
||||
private readonly TValue _step;
|
||||
private TValue _cur;
|
||||
bool started = false;
|
||||
|
||||
// public DynamicEnumerator(TValue min, TValue max, TValue step)
|
||||
// {
|
||||
// _min = min;
|
||||
// _max = max;
|
||||
// _step = step;
|
||||
// _cur = min;
|
||||
// }
|
||||
public DynamicEnumerator(TValue min, TValue max, TValue step)
|
||||
{
|
||||
_min = min;
|
||||
_max = max;
|
||||
_step = step;
|
||||
_cur = min;
|
||||
}
|
||||
|
||||
// public TValue Current => _cur;
|
||||
public TValue Current => _cur;
|
||||
|
||||
// object IEnumerator.Current => this.Current;
|
||||
object System.Collections.IEnumerator.Current => this.Current;
|
||||
|
||||
// public void Dispose() { }
|
||||
public void Dispose() { }
|
||||
|
||||
// public bool MoveNext()
|
||||
// {
|
||||
// if (!started)
|
||||
// {
|
||||
// started = true;
|
||||
// return true;
|
||||
// }
|
||||
public bool MoveNext()
|
||||
{
|
||||
if (!started)
|
||||
{
|
||||
started = true;
|
||||
return true;
|
||||
}
|
||||
|
||||
// var next = _cur + (dynamic)_step;
|
||||
// if (next == _cur || next < _min || next > _max) return false;
|
||||
var next = _cur + (dynamic)_step;
|
||||
if (next == _cur || next < _min || next > _max) return false;
|
||||
|
||||
// _cur = next;
|
||||
// return true;
|
||||
// }
|
||||
_cur = next;
|
||||
return true;
|
||||
}
|
||||
|
||||
// public void Reset()
|
||||
// {
|
||||
// _cur = _min;
|
||||
// started = false;
|
||||
// }
|
||||
// }
|
||||
//}
|
||||
public void Reset()
|
||||
{
|
||||
_cur = _min;
|
||||
started = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
partial struct TW_FIX32 : IEquatable<TW_FIX32>, IConvertible
|
||||
{
|
||||
@ -716,7 +714,7 @@ namespace NTwain.Data
|
||||
string? val = null;
|
||||
if (UTF8string != IntPtr.Zero && Size > 0)
|
||||
{
|
||||
val = ValueReader.PtrToStringUTF8(mgr, UTF8string, (int)Size);
|
||||
val = UTF8string.PtrToStringUTF8(mgr, (int)Size);
|
||||
}
|
||||
if (freeMemory) Free(mgr);
|
||||
return val;
|
||||
@ -741,11 +739,6 @@ namespace NTwain.Data
|
||||
mgr.Free(hContainer);
|
||||
hContainer = IntPtr.Zero;
|
||||
}
|
||||
|
||||
public void Read(IMemoryManager mgr, bool freeMemory = true)
|
||||
{
|
||||
if (freeMemory) Free(mgr);
|
||||
}
|
||||
}
|
||||
|
||||
//partial struct TW_DEVICEEVENT
|
||||
|
@ -1,7 +1,6 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Runtime.CompilerServices;
|
||||
using System.Diagnostics;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
|
||||
@ -15,11 +14,11 @@ namespace NTwain.Data
|
||||
/// <summary>
|
||||
/// Reads pointer as UTF8 string.
|
||||
/// </summary>
|
||||
/// <param name="memMgr"></param>
|
||||
/// <param name="data">Pointer to string.</param>
|
||||
/// <param name="memMgr"></param>
|
||||
/// <param name="length">Number of bytes to read.</param>
|
||||
/// <returns></returns>
|
||||
public static unsafe string? PtrToStringUTF8(IMemoryManager memMgr, IntPtr data, int length)
|
||||
public static unsafe string? PtrToStringUTF8(this IntPtr data, IMemoryManager memMgr, int length)
|
||||
{
|
||||
string? val = null;
|
||||
var locked = memMgr.Lock(data);
|
||||
@ -47,244 +46,271 @@ namespace NTwain.Data
|
||||
return val;
|
||||
}
|
||||
|
||||
static T MarshalTo<T>(IntPtr ptr) => Marshal.PtrToStructure<T>(ptr)!;
|
||||
|
||||
|
||||
// most of these are modified from the original TWAIN.CapabilityToCsv()
|
||||
// these contain parts from the original TWAIN.CapabilityToCsv()
|
||||
|
||||
//public static TValue ReadOneValueContainer<TValue>(IMemoryManager memMgr, ref TW_CAPABILITY cap, bool freeMemory = true) where TValue : struct
|
||||
//{
|
||||
// if (cap.hContainer == IntPtr.Zero) return default;
|
||||
/// <summary>
|
||||
/// Reads a one value out of a cap. This can only be done once if memory is freed.
|
||||
/// </summary>
|
||||
/// <typeparam name="TValue"></typeparam>
|
||||
/// <param name="cap"></param>
|
||||
/// <param name="memMgr"></param>
|
||||
/// <param name="freeMemory"></param>
|
||||
/// <returns></returns>
|
||||
public static TValue ReadOneValue<TValue>(this ref TW_CAPABILITY cap, IMemoryManager memMgr, bool freeMemory = true) where TValue : struct
|
||||
{
|
||||
if (cap.ConType != TWON.ONEVALUE || cap.hContainer == IntPtr.Zero) return default;
|
||||
|
||||
// var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
|
||||
// try
|
||||
// {
|
||||
// TWTY itemType;
|
||||
// // Mac has a level of indirection and a different structure (ick)...
|
||||
// if (TwainPlatform.IsMacOSX)
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var onevalue = MarshalTo<TW_ONEVALUE_MACOSX>(lockedPtr);
|
||||
// itemType = (TWTY)onevalue.ItemType;
|
||||
// lockedPtr += Marshal.SizeOf(onevalue);
|
||||
// }
|
||||
// else
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var onevalue = MarshalTo<TW_ONEVALUE>(lockedPtr);
|
||||
// itemType = onevalue.ItemType;
|
||||
// lockedPtr += Marshal.SizeOf(onevalue);
|
||||
// }
|
||||
try
|
||||
{
|
||||
TWTY itemType;
|
||||
// Mac has a level of indirection and a different structure (ick)...
|
||||
if (TwainPlatform.IsMacOSX)
|
||||
{
|
||||
// Crack the container...
|
||||
var onevalue = MarshalTo<TW_ONEVALUE_MACOSX>(lockedPtr);
|
||||
itemType = (TWTY)onevalue.ItemType;
|
||||
lockedPtr += Marshal.SizeOf(onevalue);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Crack the container...
|
||||
var onevalue = MarshalTo<TW_ONEVALUE>(lockedPtr);
|
||||
itemType = onevalue.ItemType;
|
||||
lockedPtr += Marshal.SizeOf(onevalue);
|
||||
}
|
||||
|
||||
// return ReadContainerData<TValue>(lockedPtr, itemType, 0);
|
||||
// }
|
||||
// finally
|
||||
// {
|
||||
// if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
// if (freeMemory) memMgr.Free(ref cap.hContainer);
|
||||
// }
|
||||
//}
|
||||
//public static Enumeration<TValue> ReadEnumerationContainer<TValue>(IMemoryManager memMgr, ref TW_CAPABILITY cap, bool freeMemory = true) where TValue : struct
|
||||
//{
|
||||
// Enumeration<TValue> retVal = new Enumeration<TValue>();
|
||||
return ReadContainerData<TValue>(lockedPtr, itemType, 0);
|
||||
}
|
||||
finally
|
||||
{
|
||||
if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
if (freeMemory)
|
||||
{
|
||||
memMgr.Free(cap.hContainer);
|
||||
cap.hContainer = IntPtr.Zero;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// if (cap.hContainer == IntPtr.Zero) return retVal;
|
||||
public static Enumeration<TValue> ReadEnumeration<TValue>(this ref TW_CAPABILITY cap, IMemoryManager memMgr, bool freeMemory = true) where TValue : struct
|
||||
{
|
||||
Enumeration<TValue> retVal = new();
|
||||
|
||||
// var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
if (cap.ConType != TWON.ENUMERATION || cap.hContainer == IntPtr.Zero) return retVal;
|
||||
|
||||
// try
|
||||
// {
|
||||
// TWTY itemType;
|
||||
// int count = 0;
|
||||
var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
|
||||
// // Mac has a level of indirection and a different structure (ick)...
|
||||
// if (TwainPlatform.IsMacOSX)
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var twenumerationmacosx = MarshalTo<TW_ENUMERATION_MACOSX>(lockedPtr);
|
||||
// itemType = (TWTY)twenumerationmacosx.ItemType;
|
||||
// count = (int)twenumerationmacosx.NumItems;
|
||||
// retVal.DefaultIndex = (int)twenumerationmacosx.DefaultIndex;
|
||||
// retVal.CurrentIndex = (int)twenumerationmacosx.CurrentIndex;
|
||||
// lockedPtr += Marshal.SizeOf(twenumerationmacosx);
|
||||
// }
|
||||
// // Windows or the 2.4+ Linux DSM...
|
||||
// else if (TWAIN.GetPlatform() == Platform.WINDOWS || ((twain.m_blFoundLatestDsm || twain.m_blFoundLatestDsm64) && (twain.m_linuxdsm == TWAIN.LinuxDsm.IsLatestDsm)))
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var twenumeration = MarshalTo<TW_ENUMERATION>(lockedPtr);
|
||||
// itemType = twenumeration.ItemType;
|
||||
// count = (int)twenumeration.NumItems;
|
||||
// retVal.DefaultIndex = (int)twenumeration.DefaultIndex;
|
||||
// retVal.CurrentIndex = (int)twenumeration.CurrentIndex;
|
||||
// lockedPtr += Marshal.SizeOf(twenumeration);
|
||||
// }
|
||||
// // The -2.3 Linux DSM...
|
||||
// else if (twain.m_blFound020302Dsm64bit && (twain.m_linuxdsm == TWAIN.LinuxDsm.Is020302Dsm64bit))
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var twenumerationlinux64 = MarshalTo<TW_ENUMERATION_LINUX64>(lockedPtr);
|
||||
// itemType = twenumerationlinux64.ItemType;
|
||||
// count = (int)twenumerationlinux64.NumItems;
|
||||
// retVal.DefaultIndex = (int)twenumerationlinux64.DefaultIndex;
|
||||
// retVal.CurrentIndex = (int)twenumerationlinux64.CurrentIndex;
|
||||
// lockedPtr += Marshal.SizeOf(twenumerationlinux64);
|
||||
// }
|
||||
// // This shouldn't be possible, but what the hey...
|
||||
// else
|
||||
// {
|
||||
// Log.Error("This is serious, you win a cookie for getting here...");
|
||||
// return retVal;
|
||||
// }
|
||||
try
|
||||
{
|
||||
TWTY itemType;
|
||||
int count = 0;
|
||||
|
||||
// retVal.Items = new TValue[count];
|
||||
// Mac has a level of indirection and a different structure (ick)...
|
||||
if (TwainPlatform.IsMacOSX)
|
||||
{
|
||||
// Crack the container...
|
||||
var twenumerationmacosx = MarshalTo<TW_ENUMERATION_MACOSX>(lockedPtr);
|
||||
itemType = (TWTY)twenumerationmacosx.ItemType;
|
||||
count = (int)twenumerationmacosx.NumItems;
|
||||
retVal.DefaultIndex = (int)twenumerationmacosx.DefaultIndex;
|
||||
retVal.CurrentIndex = (int)twenumerationmacosx.CurrentIndex;
|
||||
lockedPtr += Marshal.SizeOf(twenumerationmacosx);
|
||||
}
|
||||
// Windows or the 2.4+ Linux DSM...
|
||||
else
|
||||
{
|
||||
// Crack the container...
|
||||
var twenumeration = MarshalTo<TW_ENUMERATION>(lockedPtr);
|
||||
itemType = twenumeration.ItemType;
|
||||
count = (int)twenumeration.NumItems;
|
||||
retVal.DefaultIndex = (int)twenumeration.DefaultIndex;
|
||||
retVal.CurrentIndex = (int)twenumeration.CurrentIndex;
|
||||
lockedPtr += Marshal.SizeOf(twenumeration);
|
||||
}
|
||||
// The -2.3 Linux DSM...
|
||||
//else if (twain.m_blFound020302Dsm64bit && (twain.m_linuxdsm == TWAIN.LinuxDsm.Is020302Dsm64bit))
|
||||
//{
|
||||
// // Crack the container...
|
||||
// var twenumerationlinux64 = MarshalTo<TW_ENUMERATION_LINUX64>(lockedPtr);
|
||||
// itemType = twenumerationlinux64.ItemType;
|
||||
// count = (int)twenumerationlinux64.NumItems;
|
||||
// retVal.DefaultIndex = (int)twenumerationlinux64.DefaultIndex;
|
||||
// retVal.CurrentIndex = (int)twenumerationlinux64.CurrentIndex;
|
||||
// lockedPtr += Marshal.SizeOf(twenumerationlinux64);
|
||||
//}
|
||||
// This shouldn't be possible, but what the hey...
|
||||
//else
|
||||
//{
|
||||
// Log.Error("This is serious, you win a cookie for getting here...");
|
||||
// return retVal;
|
||||
//}
|
||||
|
||||
// for (var i = 0; i < count; i++)
|
||||
// {
|
||||
// retVal.Items[i] = ReadContainerData<TValue>(lockedPtr, itemType, i);
|
||||
// }
|
||||
// }
|
||||
// finally
|
||||
// {
|
||||
// if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
// if (freeMemory) memMgr.Free(ref cap.hContainer);
|
||||
// }
|
||||
// return retVal;
|
||||
//}
|
||||
//public static IList<TValue> ReadArrayContainer<TValue>(IMemoryManager memMgr, ref TW_CAPABILITY cap, bool freeMemory = true) where TValue : struct
|
||||
//{
|
||||
// if (cap.hContainer == IntPtr.Zero) return EmptyArray<TValue>.Value;
|
||||
retVal.Items = new TValue[count];
|
||||
|
||||
// var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
for (var i = 0; i < count; i++)
|
||||
{
|
||||
retVal.Items[i] = ReadContainerData<TValue>(lockedPtr, itemType, i);
|
||||
}
|
||||
}
|
||||
finally
|
||||
{
|
||||
if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
if (freeMemory)
|
||||
{
|
||||
memMgr.Free(cap.hContainer);
|
||||
cap.hContainer = IntPtr.Zero;
|
||||
}
|
||||
}
|
||||
return retVal;
|
||||
}
|
||||
|
||||
// try
|
||||
// {
|
||||
// TWTY itemType;
|
||||
// uint count;
|
||||
public static IList<TValue> ReadArray<TValue>(this ref TW_CAPABILITY cap, IMemoryManager memMgr, bool freeMemory = true) where TValue : struct
|
||||
{
|
||||
if (cap.ConType != TWON.ARRAY || cap.hContainer == IntPtr.Zero) return Array.Empty<TValue>();
|
||||
|
||||
// // Mac has a level of indirection and a different structure (ick)...
|
||||
// if (TwainPlatform.IsMacOSX)
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var twarraymacosx = MarshalTo<TW_ARRAY_MACOSX>(lockedPtr);
|
||||
// itemType = (TWTY)twarraymacosx.ItemType;
|
||||
// count = twarraymacosx.NumItems;
|
||||
// lockedPtr += Marshal.SizeOf(twarraymacosx);
|
||||
// }
|
||||
// else
|
||||
// {
|
||||
// // Crack the container...
|
||||
// var twarray = MarshalTo<TW_ARRAY>(lockedPtr);
|
||||
// itemType = twarray.ItemType;
|
||||
// count = twarray.NumItems;
|
||||
// lockedPtr += Marshal.SizeOf(twarray);
|
||||
// }
|
||||
var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
|
||||
// var arr = new TValue[count];
|
||||
// for (var i = 0; i < count; i++)
|
||||
// {
|
||||
// arr[i] = ReadContainerData<TValue>(lockedPtr, itemType, i);
|
||||
// }
|
||||
// return arr;
|
||||
// }
|
||||
// finally
|
||||
// {
|
||||
// if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
// if (freeMemory) memMgr.Free(ref cap.hContainer);
|
||||
// }
|
||||
//}
|
||||
//public static Range<TValue> ReadRangeContainer<TValue>(IMemoryManager memMgr, ref TW_CAPABILITY cap, bool freeMemory = true) where TValue : struct
|
||||
//{
|
||||
// var retVal = new Range<TValue>();
|
||||
try
|
||||
{
|
||||
TWTY itemType;
|
||||
uint count;
|
||||
|
||||
// if (cap.hContainer == IntPtr.Zero) return retVal;
|
||||
// Mac has a level of indirection and a different structure (ick)...
|
||||
if (TwainPlatform.IsMacOSX)
|
||||
{
|
||||
// Crack the container...
|
||||
var twarraymacosx = MarshalTo<TW_ARRAY_MACOSX>(lockedPtr);
|
||||
itemType = (TWTY)twarraymacosx.ItemType;
|
||||
count = twarraymacosx.NumItems;
|
||||
lockedPtr += Marshal.SizeOf(twarraymacosx);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Crack the container...
|
||||
var twarray = MarshalTo<TW_ARRAY>(lockedPtr);
|
||||
itemType = twarray.ItemType;
|
||||
count = twarray.NumItems;
|
||||
lockedPtr += Marshal.SizeOf(twarray);
|
||||
}
|
||||
|
||||
// var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
var arr = new TValue[count];
|
||||
for (var i = 0; i < count; i++)
|
||||
{
|
||||
arr[i] = ReadContainerData<TValue>(lockedPtr, itemType, i);
|
||||
}
|
||||
return arr;
|
||||
}
|
||||
finally
|
||||
{
|
||||
if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
if (freeMemory)
|
||||
{
|
||||
memMgr.Free(cap.hContainer);
|
||||
cap.hContainer = IntPtr.Zero;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// try
|
||||
// {
|
||||
// TW_RANGE twrange = default;
|
||||
// TW_RANGE_FIX32 twrangefix32 = default;
|
||||
public static Range<TValue> ReadRange<TValue>(this ref TW_CAPABILITY cap, IMemoryManager memMgr, bool freeMemory = true) where TValue : struct
|
||||
{
|
||||
var retVal = new Range<TValue>();
|
||||
|
||||
// // Mac has a level of indirection and a different structure (ick)...
|
||||
// if (TwainPlatform.IsMacOSX)
|
||||
// {
|
||||
// var twrangemacosx = MarshalTo<TW_RANGE_MACOSX>(lockedPtr);
|
||||
// var twrangefix32macosx = MarshalTo<TW_RANGE_FIX32_MACOSX>(lockedPtr);
|
||||
// twrange.ItemType = (TWTY)twrangemacosx.ItemType;
|
||||
// twrange.MinValue = twrangemacosx.MinValue;
|
||||
// twrange.MaxValue = twrangemacosx.MaxValue;
|
||||
// twrange.StepSize = twrangemacosx.StepSize;
|
||||
// twrange.DefaultValue = twrangemacosx.DefaultValue;
|
||||
// twrange.CurrentValue = twrangemacosx.CurrentValue;
|
||||
// twrangefix32.ItemType = (TWTY)twrangefix32macosx.ItemType;
|
||||
// twrangefix32.MinValue = twrangefix32macosx.MinValue;
|
||||
// twrangefix32.MaxValue = twrangefix32macosx.MaxValue;
|
||||
// twrangefix32.StepSize = twrangefix32macosx.StepSize;
|
||||
// twrangefix32.DefaultValue = twrangefix32macosx.DefaultValue;
|
||||
// twrangefix32.CurrentValue = twrangefix32macosx.CurrentValue;
|
||||
// }
|
||||
// // Windows or the 2.4+ Linux DSM...
|
||||
// else if (TWAIN.GetPlatform() == Platform.WINDOWS || (twain.m_linuxdsm == TWAIN.LinuxDsm.IsLatestDsm) ||
|
||||
// ((twain.m_blFoundLatestDsm || twain.m_blFoundLatestDsm64) && (twain.m_linuxdsm == TWAIN.LinuxDsm.IsLatestDsm)))
|
||||
// {
|
||||
// twrange = MarshalTo<TW_RANGE>(lockedPtr);
|
||||
// twrangefix32 = MarshalTo<TW_RANGE_FIX32>(lockedPtr);
|
||||
// }
|
||||
// // The -2.3 Linux DSM...
|
||||
// else
|
||||
// {
|
||||
// var twrangelinux64 = MarshalTo<TW_RANGE_LINUX64>(lockedPtr);
|
||||
// var twrangefix32macosx = MarshalTo<TW_RANGE_FIX32_MACOSX>(lockedPtr);
|
||||
// twrange.ItemType = twrangelinux64.ItemType;
|
||||
// twrange.MinValue = (uint)twrangelinux64.MinValue;
|
||||
// twrange.MaxValue = (uint)twrangelinux64.MaxValue;
|
||||
// twrange.StepSize = (uint)twrangelinux64.StepSize;
|
||||
// twrange.DefaultValue = (uint)twrangelinux64.DefaultValue;
|
||||
// twrange.CurrentValue = (uint)twrangelinux64.CurrentValue;
|
||||
// twrangefix32.ItemType = (TWTY)twrangefix32macosx.ItemType;
|
||||
// twrangefix32.MinValue = twrangefix32macosx.MinValue;
|
||||
// twrangefix32.MaxValue = twrangefix32macosx.MaxValue;
|
||||
// twrangefix32.StepSize = twrangefix32macosx.StepSize;
|
||||
// twrangefix32.DefaultValue = twrangefix32macosx.DefaultValue;
|
||||
// twrangefix32.CurrentValue = twrangefix32macosx.CurrentValue;
|
||||
// }
|
||||
if (cap.ConType != TWON.RANGE || cap.hContainer == IntPtr.Zero) return retVal;
|
||||
|
||||
// switch (twrange.ItemType)
|
||||
// {
|
||||
// // use dynamic since I know they fit the type.
|
||||
// case TWTY.FIX32:
|
||||
// retVal.MinValue = (dynamic)twrangefix32.MinValue;
|
||||
// retVal.MaxValue = (dynamic)twrangefix32.MaxValue;
|
||||
// retVal.StepSize = (dynamic)twrangefix32.StepSize;
|
||||
// retVal.CurrentValue = (dynamic)twrangefix32.CurrentValue;
|
||||
// retVal.DefaultValue = (dynamic)twrangefix32.DefaultValue;
|
||||
// break;
|
||||
// case TWTY.INT8:
|
||||
// case TWTY.UINT8:
|
||||
// case TWTY.INT16:
|
||||
// case TWTY.BOOL:
|
||||
// case TWTY.UINT16:
|
||||
// case TWTY.INT32:
|
||||
// case TWTY.UINT32:
|
||||
// retVal.MinValue = (dynamic)twrange.MinValue;
|
||||
// retVal.MaxValue = (dynamic)twrange.MaxValue;
|
||||
// retVal.StepSize = (dynamic)twrange.StepSize;
|
||||
// retVal.CurrentValue = (dynamic)twrange.CurrentValue;
|
||||
// retVal.DefaultValue = (dynamic)twrange.DefaultValue;
|
||||
// break;
|
||||
// default:
|
||||
// throw new NotSupportedException($"The value type {twrange.ItemType} is not supported as range.");
|
||||
var lockedPtr = memMgr.Lock(cap.hContainer);
|
||||
|
||||
// }
|
||||
// return retVal;
|
||||
// }
|
||||
// finally
|
||||
// {
|
||||
// if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
// if (freeMemory) memMgr.Free(ref cap.hContainer);
|
||||
// }
|
||||
//}
|
||||
try
|
||||
{
|
||||
TW_RANGE twrange = default;
|
||||
TW_RANGE_FIX32 twrangefix32 = default;
|
||||
|
||||
// Mac has a level of indirection and a different structure (ick)...
|
||||
if (TwainPlatform.IsMacOSX)
|
||||
{
|
||||
var twrangemacosx = MarshalTo<TW_RANGE_MACOSX>(lockedPtr);
|
||||
var twrangefix32macosx = MarshalTo<TW_RANGE_FIX32_MACOSX>(lockedPtr);
|
||||
twrange.ItemType = (TWTY)twrangemacosx.ItemType;
|
||||
twrange.MinValue = twrangemacosx.MinValue;
|
||||
twrange.MaxValue = twrangemacosx.MaxValue;
|
||||
twrange.StepSize = twrangemacosx.StepSize;
|
||||
twrange.DefaultValue = twrangemacosx.DefaultValue;
|
||||
twrange.CurrentValue = twrangemacosx.CurrentValue;
|
||||
twrangefix32.ItemType = (TWTY)twrangefix32macosx.ItemType;
|
||||
twrangefix32.MinValue = twrangefix32macosx.MinValue;
|
||||
twrangefix32.MaxValue = twrangefix32macosx.MaxValue;
|
||||
twrangefix32.StepSize = twrangefix32macosx.StepSize;
|
||||
twrangefix32.DefaultValue = twrangefix32macosx.DefaultValue;
|
||||
twrangefix32.CurrentValue = twrangefix32macosx.CurrentValue;
|
||||
}
|
||||
// Windows or the 2.4+ Linux DSM...
|
||||
else
|
||||
{
|
||||
twrange = MarshalTo<TW_RANGE>(lockedPtr);
|
||||
twrangefix32 = MarshalTo<TW_RANGE_FIX32>(lockedPtr);
|
||||
}
|
||||
// The -2.3 Linux DSM...
|
||||
//else
|
||||
//{
|
||||
// var twrangelinux64 = MarshalTo<TW_RANGE_LINUX64>(lockedPtr);
|
||||
// var twrangefix32macosx = MarshalTo<TW_RANGE_FIX32_MACOSX>(lockedPtr);
|
||||
// twrange.ItemType = twrangelinux64.ItemType;
|
||||
// twrange.MinValue = (uint)twrangelinux64.MinValue;
|
||||
// twrange.MaxValue = (uint)twrangelinux64.MaxValue;
|
||||
// twrange.StepSize = (uint)twrangelinux64.StepSize;
|
||||
// twrange.DefaultValue = (uint)twrangelinux64.DefaultValue;
|
||||
// twrange.CurrentValue = (uint)twrangelinux64.CurrentValue;
|
||||
// twrangefix32.ItemType = (TWTY)twrangefix32macosx.ItemType;
|
||||
// twrangefix32.MinValue = twrangefix32macosx.MinValue;
|
||||
// twrangefix32.MaxValue = twrangefix32macosx.MaxValue;
|
||||
// twrangefix32.StepSize = twrangefix32macosx.StepSize;
|
||||
// twrangefix32.DefaultValue = twrangefix32macosx.DefaultValue;
|
||||
// twrangefix32.CurrentValue = twrangefix32macosx.CurrentValue;
|
||||
//}
|
||||
|
||||
switch (twrange.ItemType)
|
||||
{
|
||||
// use dynamic since I know they fit the type.
|
||||
case TWTY.FIX32:
|
||||
retVal.MinValue = (dynamic)twrangefix32.MinValue;
|
||||
retVal.MaxValue = (dynamic)twrangefix32.MaxValue;
|
||||
retVal.StepSize = (dynamic)twrangefix32.StepSize;
|
||||
retVal.CurrentValue = (dynamic)twrangefix32.CurrentValue;
|
||||
retVal.DefaultValue = (dynamic)twrangefix32.DefaultValue;
|
||||
break;
|
||||
case TWTY.INT8:
|
||||
case TWTY.UINT8:
|
||||
case TWTY.INT16:
|
||||
case TWTY.BOOL:
|
||||
case TWTY.UINT16:
|
||||
case TWTY.INT32:
|
||||
case TWTY.UINT32:
|
||||
retVal.MinValue = (dynamic)twrange.MinValue;
|
||||
retVal.MaxValue = (dynamic)twrange.MaxValue;
|
||||
retVal.StepSize = (dynamic)twrange.StepSize;
|
||||
retVal.CurrentValue = (dynamic)twrange.CurrentValue;
|
||||
retVal.DefaultValue = (dynamic)twrange.DefaultValue;
|
||||
break;
|
||||
default:
|
||||
throw new NotSupportedException($"The value type {twrange.ItemType} is not supported as range.");
|
||||
|
||||
}
|
||||
return retVal;
|
||||
}
|
||||
finally
|
||||
{
|
||||
if (lockedPtr != IntPtr.Zero) memMgr.Unlock(cap.hContainer);
|
||||
if (freeMemory)
|
||||
{
|
||||
memMgr.Free(cap.hContainer);
|
||||
cap.hContainer = IntPtr.Zero;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
///// <summary>
|
||||
///// Read the one value of a cap as string. Only STR* and HANDLE types are supported.
|
||||
@ -356,99 +382,101 @@ namespace NTwain.Data
|
||||
// }
|
||||
// return null;
|
||||
//}
|
||||
///// <summary>
|
||||
///// Read the container pointer content.
|
||||
///// </summary>
|
||||
///// <param name="intptr">A locked pointer to the container's data pointer. If data is array this is the 0th item.</param>
|
||||
///// <param name="type">The twain type.</param>
|
||||
///// <param name="itemIndex">Index of the item if pointer is array.</param>
|
||||
///// <returns></returns>
|
||||
//static TValue ReadContainerData<TValue>(IntPtr intptr, TWTY type, int itemIndex) where TValue : struct
|
||||
//{
|
||||
// var isEnum = typeof(TValue).IsEnum;
|
||||
/// <summary>
|
||||
/// Read the container pointer content.
|
||||
/// </summary>
|
||||
/// <param name="intptr">A locked pointer to the container's data pointer. If data is array this is the 0th item.</param>
|
||||
/// <param name="type">The twain type.</param>
|
||||
/// <param name="itemIndex">Index of the item if pointer is array.</param>
|
||||
/// <returns></returns>
|
||||
static TValue ReadContainerData<TValue>(IntPtr intptr, TWTY type, int itemIndex) where TValue : struct
|
||||
{
|
||||
var isEnum = typeof(TValue).IsEnum;
|
||||
|
||||
// switch (type)
|
||||
// {
|
||||
// default:
|
||||
// throw new NotSupportedException($"Unsupported item type {type} for reading.");
|
||||
// // TODO: verify if needs to read int32 for small types
|
||||
// case TWTY.INT8:
|
||||
// intptr += 1 * itemIndex;
|
||||
// if (isEnum)
|
||||
// {
|
||||
// return NumericToEnum<sbyte, TValue>(MarshalTo<sbyte>(intptr));
|
||||
// }
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.UINT8:
|
||||
// intptr += 1 * itemIndex;
|
||||
// if (isEnum)
|
||||
// {
|
||||
// return NumericToEnum<byte, TValue>(MarshalTo<byte>(intptr));
|
||||
// }
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.INT16:
|
||||
// intptr += 2 * itemIndex;
|
||||
// if (isEnum)
|
||||
// {
|
||||
// return NumericToEnum<short, TValue>(MarshalTo<short>(intptr));
|
||||
// }
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.BOOL:
|
||||
// case TWTY.UINT16:
|
||||
// intptr += 2 * itemIndex;
|
||||
// if (isEnum)
|
||||
// {
|
||||
// return NumericToEnum<ushort, TValue>(MarshalTo<ushort>(intptr));
|
||||
// }
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.INT32:
|
||||
// intptr += 4 * itemIndex;
|
||||
// if (isEnum)
|
||||
// {
|
||||
// return NumericToEnum<int, TValue>(MarshalTo<int>(intptr));
|
||||
// }
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.UINT32:
|
||||
// intptr += 4 * itemIndex;
|
||||
// if (isEnum)
|
||||
// {
|
||||
// return NumericToEnum<uint, TValue>(MarshalTo<uint>(intptr));
|
||||
// }
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.FIX32:
|
||||
// intptr += 4 * itemIndex;
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.FRAME:
|
||||
// intptr += 16 * itemIndex;
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.STR32:
|
||||
// intptr += TW_STR32.Size * itemIndex;
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.STR64:
|
||||
// intptr += TW_STR64.Size * itemIndex;
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.STR128:
|
||||
// intptr += TW_STR128.Size * itemIndex;
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// case TWTY.STR255:
|
||||
// intptr += TW_STR255.Size * itemIndex;
|
||||
// return MarshalTo<TValue>(intptr);
|
||||
// }
|
||||
//}
|
||||
switch (type)
|
||||
{
|
||||
default:
|
||||
throw new NotSupportedException($"Unsupported item type {type} for reading.");
|
||||
// TODO: verify if needs to read int32 for small types
|
||||
case TWTY.INT8:
|
||||
intptr += 1 * itemIndex;
|
||||
if (isEnum)
|
||||
{
|
||||
return NumericToEnum<sbyte, TValue>(MarshalTo<sbyte>(intptr));
|
||||
}
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.UINT8:
|
||||
intptr += 1 * itemIndex;
|
||||
if (isEnum)
|
||||
{
|
||||
return NumericToEnum<byte, TValue>(MarshalTo<byte>(intptr));
|
||||
}
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.INT16:
|
||||
intptr += 2 * itemIndex;
|
||||
if (isEnum)
|
||||
{
|
||||
return NumericToEnum<short, TValue>(MarshalTo<short>(intptr));
|
||||
}
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.BOOL:
|
||||
case TWTY.UINT16:
|
||||
intptr += 2 * itemIndex;
|
||||
if (isEnum)
|
||||
{
|
||||
var shor = MarshalTo<short>(intptr);
|
||||
if (shor == 0x1125) Debugger.Break();
|
||||
return NumericToEnum<ushort, TValue>(MarshalTo<ushort>(intptr));
|
||||
}
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.INT32:
|
||||
intptr += 4 * itemIndex;
|
||||
if (isEnum)
|
||||
{
|
||||
return NumericToEnum<int, TValue>(MarshalTo<int>(intptr));
|
||||
}
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.UINT32:
|
||||
intptr += 4 * itemIndex;
|
||||
if (isEnum)
|
||||
{
|
||||
return NumericToEnum<uint, TValue>(MarshalTo<uint>(intptr));
|
||||
}
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.FIX32:
|
||||
intptr += 4 * itemIndex;
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.FRAME:
|
||||
intptr += 16 * itemIndex;
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.STR32:
|
||||
intptr += TW_STR32.Size * itemIndex;
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.STR64:
|
||||
intptr += TW_STR64.Size * itemIndex;
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.STR128:
|
||||
intptr += TW_STR128.Size * itemIndex;
|
||||
return MarshalTo<TValue>(intptr);
|
||||
case TWTY.STR255:
|
||||
intptr += TW_STR255.Size * itemIndex;
|
||||
return MarshalTo<TValue>(intptr);
|
||||
}
|
||||
}
|
||||
|
||||
//static TEnum NumericToEnum<TNumber, TEnum>(TNumber num) where TEnum : struct
|
||||
//{
|
||||
// // some caps returns a data type that's not the underlying datatype for the enum
|
||||
// // so best way is to ToString() it and parse it as the enum type.
|
||||
// var str = num.ToString();
|
||||
static TEnum NumericToEnum<TNumber, TEnum>(TNumber num) where TEnum : struct
|
||||
{
|
||||
// TODO: some caps returns a data type that's not the underlying datatype for the enum
|
||||
// so foolproof way is to ToString() it and parse it as the enum type.
|
||||
// this is bad for perf so find better way later
|
||||
var str = num!.ToString();
|
||||
|
||||
// if (Enum.TryParse(str, out TEnum parsed))
|
||||
// {
|
||||
// return parsed;
|
||||
// }
|
||||
// return default;
|
||||
//}
|
||||
if (Enum.TryParse(str, out TEnum parsed))
|
||||
{
|
||||
return parsed;
|
||||
}
|
||||
return default;
|
||||
}
|
||||
|
||||
//static T MarshalTo<T>(IntPtr ptr) => (T)Marshal.PtrToStructure(ptr, typeof(T));
|
||||
}
|
||||
}
|
||||
|
@ -13,11 +13,11 @@ namespace NTwain.Data
|
||||
/// <summary>
|
||||
/// Allocates and copies the string value into a pointer in UTF8 that's null-terminated.
|
||||
/// </summary>
|
||||
/// <param name="memMgr"></param>
|
||||
/// <param name="value"></param>
|
||||
/// <param name="memMgr"></param>
|
||||
/// <param name="finalLength">Final length to use with the pointer (includes the null).</param>
|
||||
/// <returns></returns>
|
||||
public static unsafe IntPtr StringToPtrUTF8(IMemoryManager memMgr, string? value, out uint finalLength)
|
||||
public static unsafe IntPtr StringToPtrUTF8(this string? value, IMemoryManager memMgr, out uint finalLength)
|
||||
{
|
||||
finalLength = 0;
|
||||
if (value == null) return IntPtr.Zero;
|
||||
|
@ -13,7 +13,11 @@
|
||||
<UseWPF>true</UseWPF>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup Condition=" '$(TargetFramework)' == 'net462'">
|
||||
<PackageReference Include="Microsoft.CSharp" Version="4.7.0" />
|
||||
<PackageReference Include="System.Buffers" Version="4.5.1" />
|
||||
</ItemGroup>
|
||||
<ItemGroup Condition=" '$(TargetFramework)' != 'net462'">
|
||||
|
@ -1,7 +1,7 @@
|
||||
using NTwain.Data;
|
||||
using NTwain.Triplets;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
||||
namespace NTwain
|
||||
{
|
||||
@ -9,20 +9,20 @@ namespace NTwain
|
||||
|
||||
partial class TwainAppSession
|
||||
{
|
||||
///// <summary>
|
||||
///// Gets all the supported caps for the current source.
|
||||
///// </summary>
|
||||
///// <returns></returns>
|
||||
//public IEnumerable<CAP> GetAllCaps()
|
||||
//{
|
||||
// // just as a sample of how to read cap values
|
||||
/// <summary>
|
||||
/// Gets all the supported caps for the current source.
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public IList<CAP> GetAllCaps()
|
||||
{
|
||||
// just as a sample of how to read cap values
|
||||
|
||||
// if (GetCapValues(CAP.CAP_SUPPORTEDCAPS, out TW_CAPABILITY value) == TWRC.SUCCESS)
|
||||
// {
|
||||
// value.Read(this);
|
||||
// }
|
||||
// return Enumerable.Empty<CAP>();
|
||||
//}
|
||||
if (GetCapValues(CAP.CAP_SUPPORTEDCAPS, out TW_CAPABILITY value).RC == TWRC.SUCCESS)
|
||||
{
|
||||
return value.ReadArray<CAP>(this);
|
||||
}
|
||||
return Array.Empty<CAP>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a CAP's actual supported operations.
|
||||
@ -32,10 +32,10 @@ namespace NTwain
|
||||
/// <returns></returns>
|
||||
public TWQC QueryCapSupport(CAP cap)
|
||||
{
|
||||
var value = new TW_CAPABILITY(cap);
|
||||
var value = new TW_CAPABILITY(cap) { ConType = TWON.ONEVALUE };
|
||||
if (DGControl.Capability.QuerySupport(ref _appIdentity, ref _currentDS, ref value) == TWRC.SUCCESS)
|
||||
{
|
||||
value.Read(this);
|
||||
return value.ReadOneValue<TWQC>(this);
|
||||
}
|
||||
return TWQC.Unknown;
|
||||
}
|
||||
|
@ -47,9 +47,9 @@ namespace NTwain
|
||||
var rc = DGControl.Identity.OpenDS(ref _appIdentity, ref source);
|
||||
if (rc == TWRC.SUCCESS)
|
||||
{
|
||||
State = STATE.S4;
|
||||
RegisterCallback();
|
||||
CurrentSource = source;
|
||||
State = STATE.S4;
|
||||
}
|
||||
return WrapInSTS(rc);
|
||||
}
|
||||
@ -158,7 +158,7 @@ namespace NTwain
|
||||
{
|
||||
task.SizeOf = (uint)Marshal.SizeOf(typeof(TW_TWAINDIRECT));
|
||||
task.CommunicationManager = communicationManager;
|
||||
task.Send = ValueWriter.StringToPtrUTF8(this, taskJson, out uint length);
|
||||
task.Send = taskJson.StringToPtrUTF8(this, out uint length);
|
||||
task.SendSize = length;
|
||||
|
||||
result.ReturnCode = DGControl.TwainDirect.SetTask(ref _appIdentity, ref _currentDS, ref task);
|
||||
@ -168,7 +168,7 @@ namespace NTwain
|
||||
}
|
||||
else if (result.ReturnCode == TWRC.SUCCESS && task.ReceiveSize > 0 && task.Receive != IntPtr.Zero)
|
||||
{
|
||||
result.ResponseJson = ValueReader.PtrToStringUTF8(this, task.Receive, (int)task.ReceiveSize);
|
||||
result.ResponseJson = task.Receive.PtrToStringUTF8(this, (int)task.ReceiveSize);
|
||||
}
|
||||
}
|
||||
finally
|
||||
|
Loading…
Reference in New Issue
Block a user