using NTwain.Data; using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace NTwain { // this contains all cap-related methods prefixed with Cap partial class TwainSource { /// /// Gets the actual supported operations for a capability. /// /// The cap identifier. /// public QuerySupport CapQuerySupport(CapabilityId capId) { QuerySupport retVal = QuerySupport.None; using (TWCapability cap = new TWCapability(capId)) { var rc = _session.DGControl.Capability.QuerySupport(cap); if (rc == ReturnCode.Success) { var read = CapabilityReader.ReadValue(cap); if (read.ContainerType == ContainerType.OneValue) { retVal = read.OneValue.ConvertToEnum(); } } } return retVal; } /// /// Gets the current value for a capability. /// /// The cap id. /// public object CapGetCurrent(CapabilityId capId) { using (TWCapability cap = new TWCapability(capId)) { var rc = _session.DGControl.Capability.GetCurrent(cap); if (rc == ReturnCode.Success) { var read = CapabilityReader.ReadValue(cap); switch (read.ContainerType) { case ContainerType.Enum: if (read.CollectionValues != null) { return read.CollectionValues[read.EnumCurrentIndex]; } break; case ContainerType.OneValue: return read.OneValue; case ContainerType.Range: return read.RangeCurrentValue; case ContainerType.Array: // no source should ever return an array but anyway if (read.CollectionValues != null) { return read.CollectionValues.FirstOrDefault(); } break; } } } return null; } /// /// A general method that tries to get capability values from current . /// /// The capability unique identifier. /// public IList CapGetValues(CapabilityId capabilityId) { var list = new List(); using (TWCapability cap = new TWCapability(capabilityId)) { var rc = _session.DGControl.Capability.Get(cap); if (rc == ReturnCode.Success) { cap.ReadMultiCapValues(list); } } return list; } #region xfer mech /// /// Gets the supported image for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetImageXferMech() { return CapGetValues(CapabilityId.ICapXferMech).CastToEnum(true); } #endregion #region compression /// /// Gets the supported for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetCompression() { return CapGetValues(CapabilityId.ICapCompression).CastToEnum(true); } /// /// Change the image compression for the current source. /// /// The compression. /// public ReturnCode CapSetImageCompression(CompressionType compression) { using (TWCapability compressCap = new TWCapability(CapabilityId.ICapCompression, new TWOneValue { Item = (uint)compression, ItemType = ItemType.UInt16 })) { return _session.DGControl.Capability.Set(compressCap); } } #endregion #region image format /// /// Gets the supported for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetImageFileFormat() { return CapGetValues(CapabilityId.ICapImageFileFormat).CastToEnum(true); } /// /// Change the image format for the current source. /// /// The format. /// public ReturnCode CapSetImageFormat(FileFormat format) { using (TWCapability formatCap = new TWCapability(CapabilityId.ICapImageFileFormat, new TWOneValue { Item = (uint)format, ItemType = ItemType.UInt16 })) { return _session.DGControl.Capability.Set(formatCap); } } #endregion #region pixel type /// /// Gets the supported for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetPixelTypes() { return CapGetValues(CapabilityId.ICapPixelType).CastToEnum(true); } /// /// Change the pixel type for the current source. /// /// The type. /// public ReturnCode CapSetPixelType(PixelType type) { var one = new TWOneValue(); one.Item = (uint)type; one.ItemType = ItemType.UInt16; using (TWCapability dx = new TWCapability(CapabilityId.ICapPixelType, one)) { return _session.DGControl.Capability.Set(dx); } } #endregion #region xfer mech /// /// Gets the supported image for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetImageXferMechs() { return CapGetValues(CapabilityId.ICapXferMech).CastToEnum(true); } /// /// Gets the supported audio for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetAudioXferMechs() { return CapGetValues(CapabilityId.ACapXferMech).CastToEnum(true); } /// /// Change the image xfer type for the current source. /// /// The type. /// public ReturnCode CapSetImageXferMech(XferMech type) { var one = new TWOneValue(); one.Item = (uint)type; one.ItemType = ItemType.UInt16; using (TWCapability dx = new TWCapability(CapabilityId.ICapXferMech, one)) { return _session.DGControl.Capability.Set(dx); } } /// /// Change the audio xfer type for the current source. /// /// The type. /// public ReturnCode CapSetAudioXferMech(XferMech type) { var one = new TWOneValue(); one.Item = (uint)type; one.ItemType = ItemType.UInt16; using (TWCapability dx = new TWCapability(CapabilityId.ACapXferMech, one)) { return _session.DGControl.Capability.Set(dx); } } #endregion #region dpi /// /// Gets the supported DPI values for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetDPIs() { var list = CapGetValues(CapabilityId.ICapXResolution); return list.Select(o => o.ConvertToFix32()).ToList(); } /// /// Change the DPI value for the current source. /// /// The DPI. /// public ReturnCode CapSetDPI(TWFix32 dpi) { return CapSetDPI(dpi, dpi); } /// /// Change the DPI value for the current source. /// /// The x DPI. /// The y DPI. /// public ReturnCode CapSetDPI(TWFix32 xDPI, TWFix32 yDPI) { TWOneValue one = new TWOneValue(); one.Item = (uint)xDPI;// ((uint)dpi) << 16; one.ItemType = ItemType.Fix32; using (TWCapability xres = new TWCapability(CapabilityId.ICapXResolution, one)) { var rc = _session.DGControl.Capability.Set(xres); if (rc == ReturnCode.Success) { one.Item = (uint)yDPI; using (TWCapability yres = new TWCapability(CapabilityId.ICapYResolution, one)) { rc = _session.DGControl.Capability.Set(yres); } } return rc; } } #endregion #region supported paper size /// /// Gets the supported for the current source. /// Only call this at state 4 or higher. /// /// public IList CapGetSupportedSizes() { return CapGetValues(CapabilityId.ICapSupportedSizes).CastToEnum(true); } /// /// Change the supported paper size for the current source. /// /// The size. /// public ReturnCode CapSetSupportedSize(SupportedSize size) { var one = new TWOneValue(); one.Item = (uint)size; one.ItemType = ItemType.UInt16; using (TWCapability xres = new TWCapability(CapabilityId.ICapSupportedSizes, one)) { var rc = _session.DGControl.Capability.Set(xres); return rc; } } #endregion #region onesie flags /// /// Change the auto deskew flag for the current source. /// /// if set to true use it. /// public ReturnCode CapSetAutoDeskew(bool useIt) { var rc = ReturnCode.Failure; if (SupportedCaps.Contains(CapabilityId.ICapAutomaticDeskew)) { if (Identity.ProtocolMajor >= 2) { // if using twain 2.0 will need to use enum instead of onevalue (yuck) TWEnumeration en = new TWEnumeration(); en.ItemList = new object[] { (uint)(useIt ? 1 : 0) }; en.ItemType = ItemType.Bool; using (TWCapability dx = new TWCapability(CapabilityId.ICapAutomaticDeskew, en)) { rc = _session.DGControl.Capability.Set(dx); } } else { TWOneValue one = new TWOneValue(); one.Item = (uint)(useIt ? 1 : 0); one.ItemType = ItemType.Bool; using (TWCapability capValue = new TWCapability(CapabilityId.ICapAutomaticDeskew, one)) { rc = _session.DGControl.Capability.Set(capValue); } } } return rc; } /// /// Change the auto rotate flag for the current source. /// /// if set to true use it. /// public ReturnCode CapSetAutoRotate(bool useIt) { var rc = ReturnCode.Failure; if (SupportedCaps.Contains(CapabilityId.ICapAutomaticRotate)) { if (Identity.ProtocolMajor >= 2) { // if using twain 2.0 will need to use enum instead of onevalue (yuck) TWEnumeration en = new TWEnumeration(); en.ItemList = new object[] { (uint)(useIt ? 1 : 0) }; en.ItemType = ItemType.Bool; using (TWCapability dx = new TWCapability(CapabilityId.ICapAutomaticRotate, en)) { rc = _session.DGControl.Capability.Set(dx); } } else { TWOneValue one = new TWOneValue(); one.Item = (uint)(useIt ? 1 : 0); one.ItemType = ItemType.Bool; using (TWCapability capValue = new TWCapability(CapabilityId.ICapAutomaticRotate, one)) { rc = _session.DGControl.Capability.Set(capValue); } } } return rc; } /// /// Change the auto border detection flag for the current source. /// /// if set to true use it. /// public ReturnCode CapSetBorderDetection(bool useIt) { var rc = ReturnCode.Failure; if (SupportedCaps.Contains(CapabilityId.ICapAutomaticBorderDetection)) { // this goes along with undefinedimagesize so that also // needs to be set if (Identity.ProtocolMajor >= 2) { // if using twain 2.0 will need to use enum instead of onevalue (yuck) TWEnumeration en = new TWEnumeration(); en.ItemList = new object[] { (uint)(useIt ? 1 : 0) }; en.ItemType = ItemType.Bool; using (TWCapability dx = new TWCapability(CapabilityId.ICapUndefinedImageSize, en)) { rc = _session.DGControl.Capability.Set(dx); } using (TWCapability dx = new TWCapability(CapabilityId.ICapAutomaticBorderDetection, en)) { rc = _session.DGControl.Capability.Set(dx); } } else { TWOneValue one = new TWOneValue(); one.Item = (uint)(useIt ? 1 : 0); one.ItemType = ItemType.Bool; using (TWCapability capValue = new TWCapability(CapabilityId.ICapUndefinedImageSize, one)) { rc = _session.DGControl.Capability.Set(capValue); } using (TWCapability capValue = new TWCapability(CapabilityId.ICapAutomaticBorderDetection, one)) { rc = _session.DGControl.Capability.Set(capValue); } } } return rc; } /// /// Change the duplex flag for the current source. /// /// if set to true to use it. /// public ReturnCode CapSetDuplex(bool useIt) { if (Identity.ProtocolMajor >= 2) { // twain 2 likes to use enum :( TWEnumeration en = new TWEnumeration(); en.ItemList = new object[] { (uint)(useIt ? 1 : 0) }; en.ItemType = ItemType.Bool; using (TWCapability dx = new TWCapability(CapabilityId.CapDuplexEnabled, en)) { return _session.DGControl.Capability.Set(dx); } } else { TWOneValue one = new TWOneValue(); one.Item = (uint)(useIt ? 1 : 0); one.ItemType = ItemType.Bool; using (TWCapability dx = new TWCapability(CapabilityId.CapDuplexEnabled, one)) { return _session.DGControl.Capability.Set(dx); } } } /// /// Change the use feeder flag for the current source. /// /// if set to true use it. /// public ReturnCode CapSetFeeder(bool useIt) { var rc = ReturnCode.Failure; if (SupportedCaps.Contains(CapabilityId.CapFeederEnabled)) { if (Identity.ProtocolMajor >= 2) { // if using twain 2.0 will need to use enum instead of onevalue (yuck) TWEnumeration en = new TWEnumeration(); en.ItemList = new object[] { (ushort)(useIt ? 1 : 0) }; en.ItemType = ItemType.Bool; // we will never set feeder off, only autofeed and autoscan // but if it is to SET then enable feeder needs to be set first if (useIt) { using (TWCapability dx = new TWCapability(CapabilityId.CapFeederEnabled, en)) { rc = _session.DGControl.Capability.Set(dx); } } // to really use feeder we must also set autofeed or autoscan, but only // for one of them since setting autoscan also sets autofeed if (SupportedCaps.Contains(CapabilityId.CapAutoScan)) { using (TWCapability dx = new TWCapability(CapabilityId.CapAutoScan, en)) { rc = _session.DGControl.Capability.Set(dx); } } else if (SupportedCaps.Contains(CapabilityId.CapAutoFeed)) { using (TWCapability dx = new TWCapability(CapabilityId.CapAutoFeed, en)) { rc = _session.DGControl.Capability.Set(dx); } } } else { TWOneValue one = new TWOneValue(); one.Item = (uint)(useIt ? 1 : 0); one.ItemType = ItemType.Bool; if (useIt) { using (TWCapability enabled = new TWCapability(CapabilityId.CapFeederEnabled, one)) { rc = _session.DGControl.Capability.Set(enabled); } } // to really use feeder we must also set autofeed or autoscan, but only // for one of them since setting autoscan also sets autofeed if (SupportedCaps.Contains(CapabilityId.CapAutoScan)) { using (TWCapability autoScan = new TWCapability(CapabilityId.CapAutoScan, one)) { rc = _session.DGControl.Capability.Set(autoScan); } } else if (SupportedCaps.Contains(CapabilityId.CapAutoFeed)) { using (TWCapability autoScan = new TWCapability(CapabilityId.CapAutoFeed, one)) { rc = _session.DGControl.Capability.Set(autoScan); } } } } return rc; } #endregion } }