diff --git a/NTwain.Net35/NTwain.Net35.csproj b/NTwain.Net35/NTwain.Net35.csproj index 4d6bd4b..f54b7a1 100644 --- a/NTwain.Net35/NTwain.Net35.csproj +++ b/NTwain.Net35/NTwain.Net35.csproj @@ -153,6 +153,9 @@ Properties\VersionInfo.cs + + ProtocolVersions.cs + SourceEnableMode.cs diff --git a/NTwain/CapWrapper.cs b/NTwain/CapWrapper.cs index c706bfe..98f5907 100644 --- a/NTwain/CapWrapper.cs +++ b/NTwain/CapWrapper.cs @@ -15,10 +15,11 @@ namespace NTwain /// The TWAIN type of the value. public class CapWrapper : NTwain.ICapWrapper { - ICapControl _source; + IDataSource _source; Func _getConvertRoutine; Func _setCustomRoutine; Func _setOneValueFunc; + bool _readOnly; /// /// Initializes a new instance of the class. @@ -31,16 +32,18 @@ namespace NTwain /// or /// getConversionRoutine /// - public CapWrapper(ICapControl source, CapabilityId capability, - Func getConversionRoutine) + public CapWrapper(IDataSource source, CapabilityId capability, + Func getConversionRoutine, bool readOnly) { if (source == null) { throw new ArgumentNullException("source"); } if (getConversionRoutine == null) { throw new ArgumentNullException("getConversionRoutine"); } _source = source; _getConvertRoutine = getConversionRoutine; + _readOnly = readOnly; Capability = capability; - SupportedActions = source.CapQuerySupport(capability); + + CheckSupports(); } @@ -58,7 +61,7 @@ namespace NTwain /// or /// setValueProvider /// - public CapWrapper(ICapControl source, CapabilityId capability, + public CapWrapper(IDataSource source, CapabilityId capability, Func getConversionRoutine, Func setValueProvider) { @@ -70,7 +73,8 @@ namespace NTwain _getConvertRoutine = getConversionRoutine; _setOneValueFunc = setValueProvider; Capability = capability; - SupportedActions = source.CapQuerySupport(capability); + + CheckSupports(); } /// @@ -87,7 +91,7 @@ namespace NTwain /// or /// setValueRoutine /// - public CapWrapper(ICapControl source, CapabilityId capability, + public CapWrapper(IDataSource source, CapabilityId capability, Func getConversionRoutine, Func setValueRoutine) { @@ -99,7 +103,51 @@ namespace NTwain _getConvertRoutine = getConversionRoutine; _setCustomRoutine = setValueRoutine; Capability = capability; - SupportedActions = source.CapQuerySupport(capability); + + CheckSupports(); + } + + private void CheckSupports() + { + if (!_supports.HasValue && _source.IsOpen) + { + var srcVersion = _source.ProtocolVersion; + if (srcVersion >= ProtocolVersions.GetMinimumVersion(Capability)) + { + _supports = _source.CapQuerySupport(Capability); + + if (!_supports.HasValue) + { + // lame source, have to guess using a get call + using (TWCapability cap = new TWCapability(Capability)) + { + var rc = _source.DGControl.Capability.Get(cap); + if (rc == ReturnCode.Success) + { + // assume can do common things + if (_readOnly) + { + _supports = QuerySupports.Get | QuerySupports.GetCurrent | QuerySupports.GetDefault; + } + else + { + _supports = QuerySupports.Get | QuerySupports.GetCurrent | QuerySupports.GetDefault | + QuerySupports.Set | QuerySupports.Reset | QuerySupports.SetConstraint; + } + + } + else + { + _supports = QuerySupports.None; + } + } + } + } + else + { + _supports = QuerySupports.None; + } + } } @@ -118,13 +166,22 @@ namespace NTwain /// public CapabilityId Capability { get; private set; } + QuerySupports? _supports; + /// /// Gets the supported actions. /// /// /// The supported actions. /// - public QuerySupports SupportedActions { get; private set; } + public QuerySupports SupportedActions + { + get + { + CheckSupports(); + return _supports.GetValueOrDefault(); + } + } /// /// Gets a value indicating whether this capability is supported. @@ -242,11 +299,7 @@ namespace NTwain /// public IList Get() { - if (CanGet) - { - return _source.CapGet(Capability).Select(o => _getConvertRoutine(o)).ToList(); - } - return new List(); + return _source.CapGet(Capability).Select(o => _getConvertRoutine(o)).ToList(); } /// @@ -256,22 +309,20 @@ namespace NTwain public string GetLabel() { object value = null; - if (CanGetLabel) - { - using (TWCapability cap = new TWCapability(Capability)) - { - var rc = _source.DGControl.Capability.GetLabel(cap); - if (rc == ReturnCode.Success) - { - var read = CapabilityReader.ReadValue(cap); - switch (read.ContainerType) - { - case ContainerType.OneValue: - // most likely not correct - value = read.OneValue; - break; - } + using (TWCapability cap = new TWCapability(Capability)) + { + var rc = _source.DGControl.Capability.GetLabel(cap); + if (rc == ReturnCode.Success) + { + var read = CapabilityReader.ReadValue(cap); + + switch (read.ContainerType) + { + case ContainerType.OneValue: + // most likely not correct + value = read.OneValue; + break; } } } @@ -285,22 +336,19 @@ namespace NTwain public string GetHelp() { object value = null; - if (CanGetHelp) + using (TWCapability cap = new TWCapability(Capability)) { - using (TWCapability cap = new TWCapability(Capability)) + var rc = _source.DGControl.Capability.GetHelp(cap); + if (rc == ReturnCode.Success) { - var rc = _source.DGControl.Capability.GetHelp(cap); - if (rc == ReturnCode.Success) - { - var read = CapabilityReader.ReadValue(cap); + var read = CapabilityReader.ReadValue(cap); - switch (read.ContainerType) - { - case ContainerType.OneValue: - // most likely not correct - value = read.OneValue; - break; - } + switch (read.ContainerType) + { + case ContainerType.OneValue: + // most likely not correct + value = read.OneValue; + break; } } } @@ -314,15 +362,13 @@ namespace NTwain public IList GetLabelEnum() { var list = new List(); - if (CanGetLabelEnum) + + using (TWCapability cap = new TWCapability(Capability)) { - using (TWCapability cap = new TWCapability(Capability)) + var rc = _source.DGControl.Capability.GetLabelEnum(cap); + if (rc == ReturnCode.Success) { - var rc = _source.DGControl.Capability.GetLabelEnum(cap); - if (rc == ReturnCode.Success) - { - CapabilityReader.ReadValue(cap).PopulateFromCapValues(list); - } + CapabilityReader.ReadValue(cap).PopulateFromCapValues(list); } } return list.Select(o => o.ToString()).ToList(); @@ -350,24 +396,22 @@ namespace NTwain public ReturnCode Set(TValue value) { ReturnCode rc = ReturnCode.Failure; - if (CanSet) + + if (_setCustomRoutine != null) { - if (_setCustomRoutine != null) + rc = _setCustomRoutine(value); + } + else if (_setOneValueFunc != null) + { + using (var cap = new TWCapability(Capability, _setOneValueFunc(value))) { - rc = _setCustomRoutine(value); - } - else if (_setOneValueFunc != null) - { - using (var cap = new TWCapability(Capability, _setOneValueFunc(value))) - { - rc = _source.DGControl.Capability.Set(cap); - } - } - else - { - throw new InvalidOperationException("Simple Set() is not defined for this capability."); + rc = _source.DGControl.Capability.Set(cap); } } + else + { + throw new InvalidOperationException("Simple Set() is not defined for this capability."); + } return rc; } @@ -379,12 +423,9 @@ namespace NTwain public ReturnCode Set(TWArray value) { ReturnCode rc = ReturnCode.Failure; - if (CanSet) + using (var cap = new TWCapability(Capability, value)) { - using (var cap = new TWCapability(Capability, value)) - { - rc = _source.DGControl.Capability.Set(cap); - } + rc = _source.DGControl.Capability.Set(cap); } return rc; } @@ -397,12 +438,9 @@ namespace NTwain public ReturnCode Set(TWEnumeration value) { ReturnCode rc = ReturnCode.Failure; - if (CanSet) + using (var cap = new TWCapability(Capability, value)) { - using (var cap = new TWCapability(Capability, value)) - { - rc = _source.DGControl.Capability.Set(cap); - } + rc = _source.DGControl.Capability.Set(cap); } return rc; } @@ -415,12 +453,9 @@ namespace NTwain public ReturnCode SetConstraint(TWOneValue value) { ReturnCode rc = ReturnCode.Failure; - if (CanSetConstraint) + using (var cap = new TWCapability(Capability, value)) { - using (var cap = new TWCapability(Capability, value)) - { - rc = _source.DGControl.Capability.SetConstraint(cap); - } + rc = _source.DGControl.Capability.SetConstraint(cap); } return rc; } @@ -433,12 +468,9 @@ namespace NTwain public ReturnCode SetConstraint(TWEnumeration value) { ReturnCode rc = ReturnCode.Failure; - if (CanSetConstraint) + using (var cap = new TWCapability(Capability, value)) { - using (var cap = new TWCapability(Capability, value)) - { - rc = _source.DGControl.Capability.SetConstraint(cap); - } + rc = _source.DGControl.Capability.SetConstraint(cap); } return rc; } @@ -451,12 +483,9 @@ namespace NTwain public ReturnCode SetConstraint(TWRange value) { ReturnCode rc = ReturnCode.Failure; - if (CanSetConstraint) + using (var cap = new TWCapability(Capability, value)) { - using (var cap = new TWCapability(Capability, value)) - { - rc = _source.DGControl.Capability.SetConstraint(cap); - } + rc = _source.DGControl.Capability.SetConstraint(cap); } return rc; } diff --git a/NTwain/DataSource.Caps.cs b/NTwain/DataSource.Caps.cs index 0dc68ec..fe6d8c3 100644 --- a/NTwain/DataSource.Caps.cs +++ b/NTwain/DataSource.Caps.cs @@ -17,13 +17,13 @@ namespace NTwain #region low-level cap stuff /// - /// Gets the actual supported operations for a capability. + /// Gets the actual supported operations for a capability. This is not supported by all sources. /// /// The capability id. /// - public QuerySupports CapQuerySupport(CapabilityId capabilityId) + public QuerySupports? CapQuerySupport(CapabilityId capabilityId) { - QuerySupports retVal = QuerySupports.None; + QuerySupports? retVal = null; using (TWCapability cap = new TWCapability(capabilityId)) { var rc = _session.DGControl.Capability.QuerySupport(cap); @@ -549,7 +549,7 @@ namespace NTwain { get { - return _physicalWidth ?? (_physicalWidth = new CapWrapper(this, CapabilityId.ICapPhysicalWidth, ValueExtensions.ConvertToFix32)); + return _physicalWidth ?? (_physicalWidth = new CapWrapper(this, CapabilityId.ICapPhysicalWidth, ValueExtensions.ConvertToFix32, true)); } } @@ -565,7 +565,7 @@ namespace NTwain { get { - return _physicalHeight ?? (_physicalHeight = new CapWrapper(this, CapabilityId.ICapPhysicalHeight, ValueExtensions.ConvertToFix32)); + return _physicalHeight ?? (_physicalHeight = new CapWrapper(this, CapabilityId.ICapPhysicalHeight, ValueExtensions.ConvertToFix32, true)); } } @@ -604,7 +604,7 @@ namespace NTwain { get { - return _nativeXRes ?? (_nativeXRes = new CapWrapper(this, CapabilityId.ICapXNativeResolution, ValueExtensions.ConvertToFix32)); + return _nativeXRes ?? (_nativeXRes = new CapWrapper(this, CapabilityId.ICapXNativeResolution, ValueExtensions.ConvertToFix32, true)); } } @@ -620,7 +620,7 @@ namespace NTwain { get { - return _nativeYRes ?? (_nativeYRes = new CapWrapper(this, CapabilityId.ICapYNativeResolution, ValueExtensions.ConvertToFix32)); + return _nativeYRes ?? (_nativeYRes = new CapWrapper(this, CapabilityId.ICapYNativeResolution, ValueExtensions.ConvertToFix32, true)); } } @@ -1101,7 +1101,7 @@ namespace NTwain { get { - return _minHeight ?? (_minHeight = new CapWrapper(this, CapabilityId.ICapMinimumHeight, ValueExtensions.ConvertToFix32)); + return _minHeight ?? (_minHeight = new CapWrapper(this, CapabilityId.ICapMinimumHeight, ValueExtensions.ConvertToFix32, true)); } } @@ -1117,7 +1117,7 @@ namespace NTwain { get { - return _minWidth ?? (_minWidth = new CapWrapper(this, CapabilityId.ICapMinimumWidth, ValueExtensions.ConvertToFix32)); + return _minWidth ?? (_minWidth = new CapWrapper(this, CapabilityId.ICapMinimumWidth, ValueExtensions.ConvertToFix32, true)); } } @@ -1196,7 +1196,7 @@ namespace NTwain { get { - return _barcodeType ?? (_barcodeType = new CapWrapper(this, CapabilityId.ICapSupportedBarcodeTypes, ValueExtensions.ConvertToEnum)); + return _barcodeType ?? (_barcodeType = new CapWrapper(this, CapabilityId.ICapSupportedBarcodeTypes, ValueExtensions.ConvertToEnum, true)); } } @@ -1359,7 +1359,7 @@ namespace NTwain { get { - return _patchcodeType ?? (_patchcodeType = new CapWrapper(this, CapabilityId.ICapSupportedPatchCodeTypes, ValueExtensions.ConvertToEnum)); + return _patchcodeType ?? (_patchcodeType = new CapWrapper(this, CapabilityId.ICapSupportedPatchCodeTypes, ValueExtensions.ConvertToEnum, true)); } } @@ -1726,7 +1726,7 @@ namespace NTwain { get { - return _cropUseFrame ?? (_cropUseFrame = new CapWrapper(this, CapabilityId.ICapAutomaticCropUsesFrame, ValueExtensions.ConvertToEnum)); + return _cropUseFrame ?? (_cropUseFrame = new CapWrapper(this, CapabilityId.ICapAutomaticCropUsesFrame, ValueExtensions.ConvertToEnum, true)); } } @@ -1868,7 +1868,7 @@ namespace NTwain { get { - return _supportedExtInfo ?? (_supportedExtInfo = new CapWrapper(this, CapabilityId.ICapSupportedExtImageInfo, ValueExtensions.ConvertToEnum)); + return _supportedExtInfo ?? (_supportedExtInfo = new CapWrapper(this, CapabilityId.ICapSupportedExtImageInfo, ValueExtensions.ConvertToEnum, true)); } } @@ -2026,7 +2026,7 @@ namespace NTwain { get { - return _feederLoaded ?? (_feederLoaded = new CapWrapper(this, CapabilityId.CapFeederLoaded, ValueExtensions.ConvertToEnum)); + return _feederLoaded ?? (_feederLoaded = new CapWrapper(this, CapabilityId.CapFeederLoaded, ValueExtensions.ConvertToEnum, true)); } } @@ -2035,7 +2035,7 @@ namespace NTwain private CapWrapper _supportedCaps; /// - /// Gets the supported caps for the current source. + /// Gets the supported caps for the current source. This is not supported by all sources. /// /// /// The supported caps. @@ -2044,7 +2044,7 @@ namespace NTwain { get { - return _supportedCaps ?? (_supportedCaps = new CapWrapper(this, CapabilityId.CapSupportedCaps, value => value.ConvertToEnum(false))); + return _supportedCaps ?? (_supportedCaps = new CapWrapper(this, CapabilityId.CapSupportedCaps, value => value.ConvertToEnum(false), true)); } } @@ -2186,7 +2186,7 @@ namespace NTwain { get { - return _paperDetectable ?? (_paperDetectable = new CapWrapper(this, CapabilityId.CapPaperDetectable, ValueExtensions.ConvertToEnum)); + return _paperDetectable ?? (_paperDetectable = new CapWrapper(this, CapabilityId.CapPaperDetectable, ValueExtensions.ConvertToEnum, true)); } } @@ -2202,7 +2202,7 @@ namespace NTwain { get { - return _uiControllable ?? (_uiControllable = new CapWrapper(this, CapabilityId.CapUIControllable, ValueExtensions.ConvertToEnum)); + return _uiControllable ?? (_uiControllable = new CapWrapper(this, CapabilityId.CapUIControllable, ValueExtensions.ConvertToEnum, true)); } } @@ -2218,7 +2218,7 @@ namespace NTwain { get { - return _devOnline ?? (_devOnline = new CapWrapper(this, CapabilityId.CapDeviceOnline, ValueExtensions.ConvertToEnum)); + return _devOnline ?? (_devOnline = new CapWrapper(this, CapabilityId.CapDeviceOnline, ValueExtensions.ConvertToEnum, true)); } } @@ -2276,7 +2276,7 @@ namespace NTwain { get { - return _duplex ?? (_duplex = new CapWrapper(this, CapabilityId.CapDuplex, ValueExtensions.ConvertToEnum)); + return _duplex ?? (_duplex = new CapWrapper(this, CapabilityId.CapDuplex, ValueExtensions.ConvertToEnum, true)); } } @@ -2313,7 +2313,7 @@ namespace NTwain { get { - return _dsUIonly ?? (_dsUIonly = new CapWrapper(this, CapabilityId.CapEnableDSUIOnly, ValueExtensions.ConvertToEnum)); + return _dsUIonly ?? (_dsUIonly = new CapWrapper(this, CapabilityId.CapEnableDSUIOnly, ValueExtensions.ConvertToEnum, true)); } } @@ -2329,7 +2329,7 @@ namespace NTwain { get { - return _dsData ?? (_dsData = new CapWrapper(this, CapabilityId.CapCustomDSData, ValueExtensions.ConvertToEnum)); + return _dsData ?? (_dsData = new CapWrapper(this, CapabilityId.CapCustomDSData, ValueExtensions.ConvertToEnum, true)); } } @@ -2536,7 +2536,7 @@ namespace NTwain { get { - return _powerSup ?? (_powerSup = new CapWrapper(this, CapabilityId.CapPowerSupply, ValueExtensions.ConvertToEnum)); + return _powerSup ?? (_powerSup = new CapWrapper(this, CapabilityId.CapPowerSupply, ValueExtensions.ConvertToEnum, true)); } } @@ -2552,7 +2552,7 @@ namespace NTwain { get { - return _camPreviewUI ?? (_camPreviewUI = new CapWrapper(this, CapabilityId.CapCameraPreviewUI, ValueExtensions.ConvertToEnum)); + return _camPreviewUI ?? (_camPreviewUI = new CapWrapper(this, CapabilityId.CapCameraPreviewUI, ValueExtensions.ConvertToEnum, true)); } } @@ -2589,7 +2589,7 @@ namespace NTwain { get { - return _serialNo ?? (_serialNo = new CapWrapper(this, CapabilityId.CapSerialNumber, value => { return value.ToString(); })); + return _serialNo ?? (_serialNo = new CapWrapper(this, CapabilityId.CapSerialNumber, value => { return value.ToString(); }, true)); } } @@ -2754,7 +2754,7 @@ namespace NTwain { get { - return _reacuireAllow ?? (_reacuireAllow = new CapWrapper(this, CapabilityId.CapReacquireAllowed, ValueExtensions.ConvertToEnum)); + return _reacuireAllow ?? (_reacuireAllow = new CapWrapper(this, CapabilityId.CapReacquireAllowed, ValueExtensions.ConvertToEnum, true)); } } @@ -2770,7 +2770,7 @@ namespace NTwain { get { - return _battMinutes ?? (_battMinutes = new CapWrapper(this, CapabilityId.CapBatteryMinutes, ValueExtensions.ConvertToEnum)); + return _battMinutes ?? (_battMinutes = new CapWrapper(this, CapabilityId.CapBatteryMinutes, ValueExtensions.ConvertToEnum, true)); } } @@ -2786,7 +2786,7 @@ namespace NTwain { get { - return _battPercent ?? (_battPercent = new CapWrapper(this, CapabilityId.CapBatteryPercentage, ValueExtensions.ConvertToEnum)); + return _battPercent ?? (_battPercent = new CapWrapper(this, CapabilityId.CapBatteryPercentage, ValueExtensions.ConvertToEnum, true)); } } @@ -2970,7 +2970,7 @@ namespace NTwain { get { - return _custGuid ?? (_custGuid = new CapWrapper(this, CapabilityId.CapCustomInterfaceGuid, value => { return value.ToString(); })); + return _custGuid ?? (_custGuid = new CapWrapper(this, CapabilityId.CapCustomInterfaceGuid, value => { return value.ToString(); }, true)); } } @@ -2986,7 +2986,7 @@ namespace NTwain { get { - return _supportedCapsUnique ?? (_supportedCapsUnique = new CapWrapper(this, CapabilityId.CapSupportedCapsSegmentUnique, value => value.ConvertToEnum(false))); + return _supportedCapsUnique ?? (_supportedCapsUnique = new CapWrapper(this, CapabilityId.CapSupportedCapsSegmentUnique, value => value.ConvertToEnum(false), true)); } } @@ -3002,7 +3002,7 @@ namespace NTwain { get { - return _supportedDat ?? (_supportedDat = new CapWrapper(this, CapabilityId.CapSupportedDATs, ValueExtensions.ConvertToEnum)); + return _supportedDat ?? (_supportedDat = new CapWrapper(this, CapabilityId.CapSupportedDATs, ValueExtensions.ConvertToEnum, true)); } } @@ -3207,7 +3207,7 @@ namespace NTwain { get { - return _printFontStyle ?? (_printFontStyle = new CapWrapper(this, CapabilityId.CapPrinterFontStyle, ValueExtensions.ConvertToEnum)); + return _printFontStyle ?? (_printFontStyle = new CapWrapper(this, CapabilityId.CapPrinterFontStyle, ValueExtensions.ConvertToEnum, false)); } } @@ -3288,7 +3288,7 @@ namespace NTwain { get { - return _printIdxTrig ?? (_printIdxTrig = new CapWrapper(this, CapabilityId.CapPrinterIndexTrigger, ValueExtensions.ConvertToEnum)); + return _printIdxTrig ?? (_printIdxTrig = new CapWrapper(this, CapabilityId.CapPrinterIndexTrigger, ValueExtensions.ConvertToEnum, false)); } } @@ -3304,7 +3304,7 @@ namespace NTwain { get { - return _printPreview ?? (_printPreview = new CapWrapper(this, CapabilityId.CapPrinterStringPreview, value => { return value.ToString(); })); + return _printPreview ?? (_printPreview = new CapWrapper(this, CapabilityId.CapPrinterStringPreview, value => { return value.ToString(); }, true)); } } diff --git a/NTwain/DataSource.cs b/NTwain/DataSource.cs index 9355a2e..716393a 100644 --- a/NTwain/DataSource.cs +++ b/NTwain/DataSource.cs @@ -150,22 +150,31 @@ namespace NTwain /// public Version ProtocolVersion { get; private set; } + /// + /// Gets a value indicating whether this data source has been opened. + /// + /// + /// true if this data source is open; otherwise, false. + /// + public bool IsOpen { get { return _session.IsSourceOpen; } } + static readonly CapabilityId[] _emptyCapList = new CapabilityId[0]; private IList _supportedCapsList; /// - /// Gets the list of supported caps for this source. + /// Gets the list of supported caps for this source. /// /// /// The supported caps. /// + [Obsolete("Use CapSupportedCaps.Get() instead.")] public IList SupportedCaps { get { if (_supportedCapsList == null && _session.State > 3) { - _supportedCapsList = CapSupportedCaps.Get();// CapGet(CapabilityId.CapSupportedCaps).CastToEnum(false); + _supportedCapsList = CapSupportedCaps.Get(); } return _supportedCapsList ?? _emptyCapList; } diff --git a/NTwain/DataTransferredEventArgs.cs b/NTwain/DataTransferredEventArgs.cs index 383c02b..cb61889 100644 --- a/NTwain/DataTransferredEventArgs.cs +++ b/NTwain/DataTransferredEventArgs.cs @@ -111,7 +111,7 @@ namespace NTwain /// public IEnumerable GetExtImageInfo(params ExtendedImageInfo[] infoIds) { - if (infoIds != null && infoIds.Length > 0 && DataSource.SupportedCaps.Contains(CapabilityId.ICapExtImageInfo)) + if (infoIds != null && infoIds.Length > 0)// && DataSource.SupportedCaps.Contains(CapabilityId.ICapExtImageInfo)) { var request = new TWExtImageInfo { NumInfos = (uint)infoIds.Length }; if (infoIds.Length > request.Info.Length) diff --git a/NTwain/ICapControl.cs b/NTwain/ICapControl.cs index 105ebba..c74a42b 100644 --- a/NTwain/ICapControl.cs +++ b/NTwain/ICapControl.cs @@ -35,7 +35,7 @@ namespace NTwain /// /// The capability identifier. /// - QuerySupports CapQuerySupport(CapabilityId capabilityId); + QuerySupports? CapQuerySupport(CapabilityId capabilityId); /// /// Resets the current value to power-on default. diff --git a/NTwain/ICapWrapper.cs b/NTwain/ICapWrapper.cs index fc712fd..26c041d 100644 --- a/NTwain/ICapWrapper.cs +++ b/NTwain/ICapWrapper.cs @@ -128,20 +128,12 @@ namespace NTwain IList GetLabelEnum(); /// - /// Gets a value indicating whether this capability is supported. + /// Gets a value indicating whether this capability is at all supported. /// /// /// true if this capability is supported; otherwise, false. /// bool IsSupported { get; } - - /// - /// Gets the supported actions. - /// - /// - /// The supported actions. - /// - QuerySupports SupportedActions { get; } } /// diff --git a/NTwain/IDataSource.cs b/NTwain/IDataSource.cs index a3f9875..34b8f8b 100644 --- a/NTwain/IDataSource.cs +++ b/NTwain/IDataSource.cs @@ -6,7 +6,7 @@ namespace NTwain /// /// Represents a TWAIN data source. /// - public interface IDataSource + public interface IDataSource : ICapControl { /// /// Gets the source's product name. @@ -64,6 +64,14 @@ namespace NTwain /// TWVersion Version { get; } + /// + /// Gets a value indicating whether this data source has been opened. + /// + /// + /// true if this data source is open; otherwise, false. + /// + bool IsOpen { get; } + /// /// Opens the source for capability negotiation. /// diff --git a/NTwain/NTwain.csproj b/NTwain/NTwain.csproj index b3e67f6..e94e8c9 100644 --- a/NTwain/NTwain.csproj +++ b/NTwain/NTwain.csproj @@ -88,6 +88,7 @@ + diff --git a/NTwain/Properties/VersionInfo.cs b/NTwain/Properties/VersionInfo.cs index 9c94307..02d78e9 100644 --- a/NTwain/Properties/VersionInfo.cs +++ b/NTwain/Properties/VersionInfo.cs @@ -23,7 +23,7 @@ namespace NTwain /// /// The build release version number. /// - public const string Build = "3.2.6"; // change this for each nuget release + public const string Build = "3.2.7"; // change this for each nuget release } diff --git a/NTwain/ProtocolVersions.cs b/NTwain/ProtocolVersions.cs new file mode 100644 index 0000000..d39a137 --- /dev/null +++ b/NTwain/ProtocolVersions.cs @@ -0,0 +1,209 @@ +using NTwain.Data; +using System; +using System.Collections.Generic; + +namespace NTwain +{ + /// + /// Contains the minimum TWAIN protocol version for various things. + /// + public static class ProtocolVersions + { + internal static readonly Version v10 = new Version(1, 0); + internal static readonly Version v11 = new Version(1, 1); + internal static readonly Version v15 = new Version(1, 5); + internal static readonly Version v16 = new Version(1, 6); + internal static readonly Version v17 = new Version(1, 7); + internal static readonly Version v18 = new Version(1, 8); + internal static readonly Version v19 = new Version(1, 9); + internal static readonly Version v20 = new Version(2, 0); + internal static readonly Version v21 = new Version(2, 1); + internal static readonly Version v22 = new Version(2, 2); + internal static readonly Version v23 = new Version(2, 3); + + + static readonly Dictionary __capMinVersions = new Dictionary + { + { CapabilityId.ACapXferMech, v18 }, + + { CapabilityId.CapAlarms, v18 }, + { CapabilityId.CapAlarmVolume, v18 }, + { CapabilityId.CapAuthor, v10 }, + { CapabilityId.CapAutoFeed, v10 }, + { CapabilityId.CapAutomaticCapture, v18 }, + { CapabilityId.CapAutomaticSenseMedium, v21 }, + { CapabilityId.CapAutoScan, v16 }, + { CapabilityId.CapBatteryMinutes, v18 }, + { CapabilityId.CapBatteryPercentage, v18 }, + { CapabilityId.CapCameraEnabled, v20 }, + { CapabilityId.CapCameraOrder, v20 }, + { CapabilityId.CapCameraPreviewUI, v18 }, + { CapabilityId.CapCameraSide, v19 }, + { CapabilityId.CapCaption, v10 }, + { CapabilityId.CapClearBuffers, v18 }, + { CapabilityId.CapClearPage, v10 }, + { CapabilityId.CapCustomDSData, v17 }, + { CapabilityId.CapCustomInterfaceGuid, v21 }, + { CapabilityId.CapDeviceEvent, v18 }, + { CapabilityId.CapDeviceOnline, v16 }, + { CapabilityId.CapDeviceTimeDate, v18 }, + { CapabilityId.CapDoubleFeedDetection, v22 }, + { CapabilityId.CapDoubleFeedDetectionLength, v22 }, + { CapabilityId.CapDoubleFeedDetectionResponse, v22 }, + { CapabilityId.CapDoubleFeedDetectionSensitivity, v22 }, + { CapabilityId.CapDuplex, v17 }, + { CapabilityId.CapDuplexEnabled, v17 }, + { CapabilityId.CapEnableDSUIOnly, v17 }, + { CapabilityId.CapEndorser, v17 }, + { CapabilityId.CapExtendedCaps, v10 }, + { CapabilityId.CapFeederAlignment, v18 }, + { CapabilityId.CapFeederEnabled, v10 }, + { CapabilityId.CapFeederLoaded, v10 }, + { CapabilityId.CapFeederOrder, v18 }, + { CapabilityId.CapFeederPocket, v20 }, + { CapabilityId.CapFeederPrep, v20}, + { CapabilityId.CapFeedPage, v10 }, + { CapabilityId.CapIndicators, v11 }, + { CapabilityId.CapIndicatorsMode, v22 }, + { CapabilityId.CapJobControl, v17 }, + { CapabilityId.CapLanguage, v18 }, + { CapabilityId.CapMaxBatchBuffers, v18 }, + { CapabilityId.CapMicrEnabled, v20 }, + { CapabilityId.CapPaperDetectable, v16 }, + { CapabilityId.CapPaperHandling, v22 }, + { CapabilityId.CapPowerSaveTime, v18 }, + { CapabilityId.CapPowerSupply, v18 }, + { CapabilityId.CapPrinter, v18 }, + { CapabilityId.CapPrinterEnabled, v18 }, + { CapabilityId.CapPrinterCharRotation, v23 }, + { CapabilityId.CapPrinterFontStyle, v23 }, + { CapabilityId.CapPrinterIndex, v18 }, + { CapabilityId.CapPrinterIndexLeadChar, v23 }, + { CapabilityId.CapPrinterIndexMaxValue, v23 }, + { CapabilityId.CapPrinterIndexNumDigits, v23 }, + { CapabilityId.CapPrinterIndexStep, v23 }, + { CapabilityId.CapPrinterIndexTrigger, v23 }, + { CapabilityId.CapPrinterMode, v18 }, + { CapabilityId.CapPrinterString, v18 }, + { CapabilityId.CapPrinterStringPreview, v23 }, + { CapabilityId.CapPrinterSuffix, v18 }, + { CapabilityId.CapPrinterVerticalOffset, v22 }, + { CapabilityId.CapReacquireAllowed, v18 }, + { CapabilityId.CapRewindPage, v10 }, + { CapabilityId.CapSegmented, v19 }, + { CapabilityId.CapSerialNumber, v18 }, + { CapabilityId.CapSupportedCaps, v10 }, + { CapabilityId.CapSupportedCapsSegmentUnique, v22 }, + { CapabilityId.CapSupportedDATs, v22 }, + { CapabilityId.CapTimeBeforeFirstCapture, v18 }, + { CapabilityId.CapTimeBetweenCaptures, v18 }, + { CapabilityId.CapTimeDate, v10 }, + { CapabilityId.CapThumbnailsEnabled, v17 }, + { CapabilityId.CapUIControllable, v16 }, + { CapabilityId.CapXferCount, v10 }, + + { CapabilityId.ICapAutoBright, v10 }, + { CapabilityId.ICapAutoDiscardBlankPages, v20 }, + { CapabilityId.ICapAutomaticBorderDetection, v18 }, + { CapabilityId.ICapAutomaticColorEnabled, v21 }, + { CapabilityId.ICapAutomaticColorNonColorPixelType, v18 }, + { CapabilityId.ICapAutomaticCropUsesFrame, v21 }, + { CapabilityId.ICapAutomaticDeskew, v18 }, + { CapabilityId.ICapAutomaticLengthDetection, v21 }, + { CapabilityId.ICapAutomaticRotate, v18 }, + { CapabilityId.ICapAutoSize, v20 }, + { CapabilityId.ICapBarcodeDetectionEnabled, v18 }, + { CapabilityId.ICapBarcodeMaxRetries, v18 }, + { CapabilityId.ICapBarcodeMaxSearchPriorities, v18 }, + { CapabilityId.ICapBarcodeSearchMode, v18 }, + { CapabilityId.ICapBarcodeSearchPriorities, v18 }, + { CapabilityId.ICapBarcodeTimeout, v18 }, + { CapabilityId.ICapBitDepth, v10 }, + { CapabilityId.ICapBitDepthReduction, v15 }, + { CapabilityId.ICapBitOrder, v10 }, + { CapabilityId.ICapBitOrderCodes, v10 }, + { CapabilityId.ICapBrightness, v10 }, + { CapabilityId.ICapCCITTKFactor, v10 }, + { CapabilityId.ICapColorManagementEnabled, v21 }, + { CapabilityId.ICapCompression, v10 }, + { CapabilityId.ICapContrast, v10 }, + { CapabilityId.ICapCustHalftone, v10 }, + { CapabilityId.ICapExposureTime, v10 }, + { CapabilityId.ICapExtImageInfo, v17 }, + { CapabilityId.ICapFeederType, v19 }, + { CapabilityId.ICapFilmType, v22 }, + { CapabilityId.ICapFilter, v10 }, + { CapabilityId.ICapFlashUsed, v16 }, // maybe + { CapabilityId.ICapFlashUsed2, v18 }, + { CapabilityId.ICapFlipRotation, v18 }, + { CapabilityId.ICapFrames, v10 }, + { CapabilityId.ICapGamma, v10 }, + { CapabilityId.ICapHalftones, v10 }, + { CapabilityId.ICapHighlight, v10 }, + { CapabilityId.ICapICCProfile, v19 }, + { CapabilityId.ICapImageDataSet, v17 }, + { CapabilityId.ICapImageFileFormat, v10 }, + { CapabilityId.ICapImageFilter, v18 }, + { CapabilityId.ICapImageMerge, v21 }, + { CapabilityId.ICapImageMergeHeightThreshold, v21 }, + { CapabilityId.ICapJpegPixelType, v10 }, + { CapabilityId.ICapJpegQuality, v19 }, + { CapabilityId.ICapJpegSubsampling, v22 }, + { CapabilityId.ICapLampState, v10 }, + { CapabilityId.ICapLightPath, v10 }, + { CapabilityId.ICapLightSource, v10 }, + { CapabilityId.ICapMaxFrames, v10 }, + { CapabilityId.ICapMinimumHeight, v17 }, + { CapabilityId.ICapMinimumWidth, v17 }, + { CapabilityId.ICapMirror, v22 }, + { CapabilityId.ICapNoiseFilter, v18 }, + { CapabilityId.ICapOrientation, v10 }, + { CapabilityId.ICapOverScan, v18 }, + { CapabilityId.ICapPatchCodeDetectionEnabled, v18 }, + { CapabilityId.ICapPatchCodeMaxRetries, v18 }, + { CapabilityId.ICapPatchCodeMaxSearchPriorities, v18 }, + { CapabilityId.ICapPatchCodeSearchMode, v18 }, + { CapabilityId.ICapPatchCodeSearchPriorities, v18 }, + { CapabilityId.ICapPatchCodeTimeout, v18 }, + { CapabilityId.ICapPhysicalHeight, v10 }, + { CapabilityId.ICapPhysicalWidth, v10 }, + { CapabilityId.ICapPixelFlavor, v10 }, + { CapabilityId.ICapPixelFlavorCodes, v10 }, + { CapabilityId.ICapPixelType, v10 }, + { CapabilityId.ICapPlanarChunky, v10 }, + { CapabilityId.ICapRotation, v10 }, + { CapabilityId.ICapShadow, v10 }, + { CapabilityId.ICapSupportedBarcodeTypes, v18 }, + { CapabilityId.ICapSupportedExtImageInfo, v21 }, + { CapabilityId.ICapSupportedPatchCodeTypes, v18 }, + { CapabilityId.ICapSupportedSizes, v10 }, + { CapabilityId.ICapThreshold, v10 }, + { CapabilityId.ICapTiles, v10 }, + { CapabilityId.ICapTimeFill, v10 }, + { CapabilityId.ICapUndefinedImageSize, v16 }, + { CapabilityId.ICapUnits, v10 }, + { CapabilityId.ICapXferMech, v10 }, + { CapabilityId.ICapXNativeResolution, v10 }, + { CapabilityId.ICapXResolution, v10 }, + { CapabilityId.ICapXScaling, v10 }, + { CapabilityId.ICapYNativeResolution, v10 }, + { CapabilityId.ICapYResolution, v10 }, + { CapabilityId.ICapYScaling, v10 }, + { CapabilityId.ICapZoomFactor, v18 }, + }; + + /// + /// Gets the minimum TWAIN protocol version for a capability. + /// + /// The capability type. + /// + public static Version GetMinimumVersion(CapabilityId id) + { + if (__capMinVersions.ContainsKey(id)) + { + return __capMinVersions[id]; + } + return v10; + } + } +} diff --git a/Tests/Tester.WPF/Tester.WPF.csproj b/Tests/Tester.WPF/Tester.WPF.csproj index f745f83..fb7d17d 100644 --- a/Tests/Tester.WPF/Tester.WPF.csproj +++ b/Tests/Tester.WPF/Tester.WPF.csproj @@ -70,7 +70,7 @@ False - ..\..\packages\ModernWPF.1.2.9.1\lib\net40-Client\ModernWPF.dll + ..\..\packages\ModernWPF.1.2.9.4\lib\net40-Client\ModernWPF.dll ..\..\packages\ModernWPF.Mvvm.0.7.0\lib\net40-Client\ModernWPF.Mvvm.dll diff --git a/Tests/Tester.WPF/ViewModels/CapVM.cs b/Tests/Tester.WPF/ViewModels/CapVM.cs index 76f2095..8d10470 100644 --- a/Tests/Tester.WPF/ViewModels/CapVM.cs +++ b/Tests/Tester.WPF/ViewModels/CapVM.cs @@ -21,7 +21,7 @@ namespace Tester.WPF { _ds = ds; Cap = cap; - Supports = ds.CapQuerySupport(cap); + var capName = cap.ToString(); var wrapProperty = ds.GetType().GetProperty(capName); @@ -33,6 +33,22 @@ namespace Tester.WPF _getCurrentMethod = wrapperType.GetMethod("GetCurrent"); _setMethod = wrapperType.GetMethods().FirstOrDefault(m => m.Name == "Set"); } + + var supportTest = ds.CapQuerySupport(cap); + if (supportTest.HasValue) + { + Supports = supportTest.Value; + } + else + { + if (_wrapper != null) + { + var wrapperType = _wrapper.GetType(); + QuerySupports? supports = (QuerySupports?)wrapperType.GetProperty("SupportedActions").GetGetMethod().Invoke(_wrapper, null); + Supports = supports.GetValueOrDefault(); + } + } + } public IEnumerable Get() diff --git a/Tests/Tester.WPF/ViewModels/DataSourceVM.cs b/Tests/Tester.WPF/ViewModels/DataSourceVM.cs index 0f75562..90a2579 100644 --- a/Tests/Tester.WPF/ViewModels/DataSourceVM.cs +++ b/Tests/Tester.WPF/ViewModels/DataSourceVM.cs @@ -49,7 +49,7 @@ namespace Tester.WPF //rc = DGControl.Status.Get(dsId, ref stat); if (rc == ReturnCode.Success) { - foreach (var c in DS.SupportedCaps.Select(o => new CapVM(DS, o))) + foreach (var c in DS.CapSupportedCaps.Get().Select(o => new CapVM(DS, o))) { Caps.Add(c); } diff --git a/Tests/Tester.WPF/ViewModels/TwainVM.cs b/Tests/Tester.WPF/ViewModels/TwainVM.cs index ee90526..26675b5 100644 --- a/Tests/Tester.WPF/ViewModels/TwainVM.cs +++ b/Tests/Tester.WPF/ViewModels/TwainVM.cs @@ -35,6 +35,7 @@ namespace Tester.WPF _session.TransferReady += _session_TransferReady; _session.DataTransferred += _session_DataTransferred; _session.SourceDisabled += _session_SourceDisabled; + _session.StateChanged += (s, e) => { RaisePropertyChanged(() => State); }; } TwainSession _session; diff --git a/Tests/Tester.WPF/packages.config b/Tests/Tester.WPF/packages.config index b52fe3a..249fd39 100644 --- a/Tests/Tester.WPF/packages.config +++ b/Tests/Tester.WPF/packages.config @@ -4,7 +4,7 @@ - + \ No newline at end of file diff --git a/Tests/Tester.Winform/TestForm.cs b/Tests/Tester.Winform/TestForm.cs index 4ca48ff..767aea2 100644 --- a/Tests/Tester.Winform/TestForm.cs +++ b/Tests/Tester.Winform/TestForm.cs @@ -70,6 +70,10 @@ namespace Tester.Winform { Debug.WriteLine("State changed to " + _twain.State + " on thread " + Thread.CurrentThread.ManagedThreadId); }; + _twain.TransferError += (s, e) => + { + Debug.WriteLine("Got xfer error on thread " + Thread.CurrentThread.ManagedThreadId); + }; _twain.DataTransferred += (s, e) => { Debug.WriteLine("Transferred data event on thread " + Thread.CurrentThread.ManagedThreadId); @@ -223,7 +227,7 @@ namespace Tester.Winform _stopScan = false; - if (_twain.CurrentSource.SupportedCaps.Contains(CapabilityId.CapUIControllable)) + if (_twain.CurrentSource.CapUIControllable.IsSupported)//.SupportedCaps.Contains(CapabilityId.CapUIControllable)) { // hide scanner ui if possible if (_twain.CurrentSource.Enable(SourceEnableMode.NoUI, false, this.Handle) == ReturnCode.Success) @@ -292,8 +296,10 @@ namespace Tester.Winform private void LoadSourceCaps() { var src = _twain.CurrentSource; - var caps = src.SupportedCaps; _loadingCaps = true; + + var test = src.SupportedCaps; + if (groupDepth.Enabled = src.ICapPixelType.IsSupported) { LoadDepth(src.ICapPixelType); @@ -311,7 +317,7 @@ namespace Tester.Winform { LoadPaperSize(src.ICapSupportedSizes); } - btnAllSettings.Enabled = caps.Contains(CapabilityId.CapEnableDSUIOnly); + btnAllSettings.Enabled = src.CapEnableDSUIOnly.IsSupported; _loadingCaps = false; }