枚举打印机(winapi)

类别:.NET开发 点击:0 评论:0 推荐:

using System;
using System.Runtime.InteropServices;
using System.Runtime.Serialization;

namespace Sample.GetPrinter
{
 class App
 {
  static void Main(string[] args)
  {
   Console.WriteLine("请输入你想搜索类型的序号:");
   Console.WriteLine("1. Default printer(only Win95,Win98,WinME)\n2. Enumerates the locally installed printers;\n3. Enumerates the list of printers to which the user has made previous connections;\n4. Enumerates the printer identified by Name.;\n5. Enumerates network printers and print servers in the computer's domain;\n6. Enumerates printers that have the shared attribute;\n7. Enumerates network printers in the computer's domain;\n======================================================================");
   
   int pt =0;
   try
   {
    pt=Int32.Parse(Console.ReadLine());
   }
   catch(Exception e)
   {
    Console.WriteLine("错误信息: {0}",e.Message);
    return;
   }

   PRINTER_ENUM printerKind = GetPrinterConType(pt);

   PrinterSearcher p = new PrinterSearcher();  

   PrinterSearcher.PrinterInfo[] printers = p.Search(printerKind);

   foreach(PrinterSearcher.PrinterInfo pi in printers)
   {
    Console.WriteLine("=====================================\n打印机名: {0}\n描叙:: {1}\n注释: {2}\n=====================================\n",
     pi.Name, pi.Description, pi.Comment);
   }
  }

  static PRINTER_ENUM GetPrinterConType(int ins)
  {
   switch(ins)
   {
    case 1:
     return PRINTER_ENUM.DEFAULT ;
    case 2:
     return PRINTER_ENUM.LOCAL;
    case 3:
     return PRINTER_ENUM.CONNECTIONS;
    case 4:
     return PRINTER_ENUM.NAME;
    case 5:
     return PRINTER_ENUM.REMOTE;
    case 6:
     return PRINTER_ENUM.SHARED;
    case 7:
     return PRINTER_ENUM.NETWORK;
    default:
     return PRINTER_ENUM.LOCAL ;
   }
  }
 }


 #region 打印机位置状态枚举 PRINTER_ENUM
 public enum PRINTER_ENUM
 {
  DEFAULT = 0x01,
  LOCAL = 0x02,
  CONNECTIONS = 0x04,
  NAME = 0x08,
  REMOTE = 0x10,
  SHARED = 0x20,
  NETWORK = 0x40
 }
 #endregion

 #region 异常派生 EnumPrinterException
 [Serializable]
 public class EnumPrinterException: ApplicationException
 {
  public EnumPrinterException() { }
  public EnumPrinterException (string message): base(message) { }
  public EnumPrinterException (string message, Exception inner):
   base(message, inner) {}
  protected EnumPrinterException (SerializationInfo info,
   StreamingContext context) : base(info, context)
  { }
 }
 #endregion

 //加上这个属性可以按导出到非托管对像的顺序排序
 [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
 public class PRINTER_INFO_1
 {
  public int flags;
  public IntPtr pDescription;
  public IntPtr pName;
  public IntPtr pComment;
 }

 public class PrinterSearcher
 {
  #region Search
  public PrinterInfo[] Search(PRINTER_ENUM printerKind)
  {
   PrinterInfo[] pInfo = new PrinterInfo[0];

   uint iNeeded = 0, iReturned = 0, iSize = 0;
   IntPtr printers = IntPtr.Zero;

     
   if (!EnumPrinters(printerKind, String.Empty,  1,  printers,  0, ref iNeeded, ref iReturned) )
   {
    //返回由上一个非托管函数返回的错误代码,该函数是使用设置了 -
    //DllImport属性中SetLastError=true 标志的平台调用来调用的

    int err = Marshal.GetLastWin32Error();
    if (err != Win32Error.ERROR_INSUFFICIENT_BUFFER)
     ThrowEnumPrinterException();
   }

   iSize = iNeeded;

   if (iNeeded != 0)
   {
    try
    {
     //使用AllocHGlobal分配非托管内块
     printers = Marshal.AllocHGlobal(new IntPtr(iSize));
     
     //如果调用不成功抛出异常
     if (!EnumPrinters(printerKind, String.Empty,1,printers, iSize, ref iNeeded, ref iReturned) )
     {
      ThrowEnumPrinterException();
     }
     else
     {
      pInfo = GetPrinterInfoFromMemory(printers, iReturned);
     }
    }
    finally
    {
     //释放分配的内存块
     if (printers != IntPtr.Zero)
      Marshal.FreeHGlobal(printers);
    }
   }
   return pInfo;
  }
  #endregion

  #region PrinterInfo
  public struct PrinterInfo
  {
   public string Name;
   public string Description;
   public string Comment;
  }

  
  public sealed class Win32Error
  {
   private Win32Error() {}
   public const int ERROR_INSUFFICIENT_BUFFER = 122;
  }
  #endregion

  #region EnumPrinters
  [DllImport("winspool.drv", SetLastError = true, CharSet = CharSet.Auto)]
  [return: MarshalAs(UnmanagedType.Bool)]
  private static extern bool EnumPrinters ([MarshalAs(UnmanagedType.U4)] PRINTER_ENUM flags,
   [MarshalAs(UnmanagedType.LPStr)] string sName,
   uint iLevel,
   IntPtr pPrinterDesc,
   uint iSize,
   [MarshalAs(UnmanagedType.U4)] ref uint iNeeded,
   [MarshalAs(UnmanagedType.U4)] ref uint iReturned
   );
  #endregion

  #region GetPrinterInfoFromMemory
  private PrinterInfo[] GetPrinterInfoFromMemory(IntPtr prInfo, uint numPrinters)
  {
   PRINTER_INFO_1 pi = new PRINTER_INFO_1();
         
   PrinterInfo[] pInfo = new PrinterInfo[numPrinters];

   for(int i = 0; i < numPrinters; i++)
   {
    //把数据从非托管内存传送到到托管内存
    Marshal.PtrToStructure(prInfo, pi);
         
    pInfo[i].Name = Marshal.PtrToStringAuto(pi.pName);
    pInfo[i].Description = Marshal.PtrToStringAuto(pi.pDescription);
    pInfo[i].Comment = Marshal.PtrToStringAuto(pi.pComment);

    prInfo =  new IntPtr(prInfo.ToInt32() + Marshal.SizeOf(typeof(PRINTER_INFO_1)));
   }
   return pInfo;
  }

  private void ThrowEnumPrinterException()
  {
   throw new EnumPrinterException(string.Format("LastErrorCode: {0}",
    Marshal.GetLastWin32Error()));
  }
  #endregion
 }
}

本文地址:http://com.8s8s.com/it/it42339.htm