2017-02-22 41 views
0

У меня есть программа C# WPF .NET 4.6, которая создает файлы HTML, и я хотел бы распечатать их автоматически (без диалога) с известным принтером, отличным от стандартного. Конечно, это включает в себя визуализацию HTML. Поскольку программа создает эти файлы, данные HTML могут поступать из MemoryStream, FileStream или непосредственно из строки.Render and Print HTML для нестандартного принтера

У программы есть настройки, которые позволяют пользователю заранее указать принтер для печати, используя System.Drawing.Printing.PrinterSettings.InstalledPrinters, так как каждому файлу может потребоваться другой принтер. Во время печати имя принтера известно, но, вероятно, будет отличаться от принтера по умолчанию Windows.

Я изучил много других проектов, но они, похоже, не объясняют, что принтер отличается от стандартного. Изменение принтера по умолчанию будет антисоциальным и вызовет мир, связанный с потоками. Это похоже на принятое решение №1, хотя не может быть лучшим решением?


Исследования и решения смотрели на:

Printing the contents of a WPF WebBrowser и Silently print HTML from WPF WebBrowser и corresponding MSDNforum discussions являются недостаточными в качестве функции COM ExecWB только печатает на принтере по умолчанию

MSDN example использует только (?) Print() на WebBrowser, который снова использует принтер по умолчанию.

Итак, я пошел по пути, чтобы изменить параметры принтера. Programmatically changing the destination printer for a WinForms WebBrowser control был задан, но имеет довольно неудовлетворительный ответ, так как он имеет неработающую ссылку, и я не знаю, какие внешние программы работают на компьютере, поэтому я не могу гарантировать Adobe, OpenOffice и т. Д. OP упомянул, что они анализировали ActiveX COM не вдаваясь в подробности. Звучит сложно.

Возможно, я мог бы что-то прописать из письма в RichTextBox, например, this project, и спрятать поле?

Я думал, что Silent print HTML file in C# using WPF находится на хорошей дорожке, однако исходное сообщение имеет номера с жестким кодом для экранов экрана‽ и OP упомянули, что принтер отключил документ. В принятом (и добавленном) ответе снова используется метод настройки принтера по умолчанию ExecWB.

execCommand("Print", false, IDon'tUnderstandThisArgument) также показал обещание, как и его ответ был обновлен MSDN answer, но FileStream отправляется на принтер не позволяет HTML, ни делает DocumentStream от WebBrowser появляется на работе (принтер печатает один пустую страницу).

How do I programatically change printer settings with the WebBrowser control? имеет очень похожие требования ко мне, за исключением изменения реестра как решения.

Помимо исследования, как другие сделали это, я также пытался печатать МОФ WebBrowser непосредственно, как это Visual контроль: (?, Потому что WebBrowser не видно)

public static bool Print(string printer, Visual objToPrint) 
    { 
     if (string.IsNullOrEmpty(printer)) 
     { 
     return false; 
     } 

     var dlg = new PrintDialog 
     { 
     PrintQueue = new PrintServer().GetPrintQueue(printer) 
     }; 

     dlg.PrintTicket.CopyCount = 1; 
     dlg.PrintTicket.PageOrientation = PageOrientation.Portrait; 
     dlg.PrintTicket.PagesPerSheet = 1; 

     dlg.PrintVisual(objToPrint, "Print description"); 
     return true; 
    } 

однако это ничего не печатает , И попробовал PrintDocument как обновленный MSDN article предложил:

public static async Task<bool> PrintHTMLAsync(string printer, string html) 
    { 
     bool result; 
     using (var webBrowser = new System.Windows.Forms.WebBrowser()) 
     { 
     webBrowser.DocumentCompleted += ((sender, e) => browserReadySemaphore.Release()); 
     byte[] buffer = Encoding.UTF8.GetBytes(html); 
     webBrowser.DocumentStream = new MemoryStream(buffer); 

     // Wait until the page loads. 
     await browserReadySemaphore.WaitAsync(); 

     try 
     { 
      using (PrintDocument pd = new PrintDocument()) 
      { 
      pd.PrinterSettings.PrinterName = printer; 
      pd.PrinterSettings.Collate = false; 
      pd.PrinterSettings.Copies = 1; 
      pd.PrinterSettings.FromPage = 1; 
      pd.PrinterSettings.ToPage = 1; 
      pd.Print(); 
      result = true; 
      } 
     } 
     catch (Exception ex) 
     { 
      result = false; 
      Debug.WriteLine(ex); 
     } 

     return result; 
     } 
    } 

не радость.

Я также использовал PRINT команду DOS:

public static string PerformSilentPrinting(string fileName, string printerName) 
{ 
    try 
    { 
    ProcessStartInfo startInfo = new ProcessStartInfo(fileName) 
    { 
     Arguments = string.Format("/C PRINT /D:\"{0}\" \"{1}\"", printerName, fileName), 
     FileName = "cmd.exe", 
     RedirectStandardOutput = true, 
     UseShellExecute = false, 
     WindowStyle = ProcessWindowStyle.Hidden, 
    }; 

    // Will execute the batch file with the provided arguments 
    Process process = Process.Start(startInfo); 

    // Reads the output   
    return process.StandardOutput.ReadToEnd(); 
    } 
    catch (Exception ex) 
    { 
    return ex.ToString(); 
    } 
} 

но команда печати, кажется, принимают только текстовые файлы.

ответ

0

EDIT: Это решение работает хорошо, если вам нужно распечатать только одну страницу формата A4. Однако он будет печатать только одну страницу и обрезать все, что угодно.


В конце концов я пошел с WinForms WebBrowser, скопирован контроль в Bitmap, и распечатан с помощью PrintDialog, который также в System.Windows.Forms пространства имен.

using Microsoft.Win32; 
using System; 
using System.ComponentModel; 
using System.Drawing; 
using System.Drawing.Printing; 
using System.IO; 
using System.Threading; 
using System.Threading.Tasks; 
using System.Windows.Forms; 

public static class PrintUtility 
{ 
    private static readonly SemaphoreSlim browserReadySemaphore = new SemaphoreSlim(0); 

    // A4 dimensions. 
    private const int DPI = 600; 
    private const int WIDTH = (int)(8.3 * DPI); 
    private const int HEIGHT = (int)(11.7 * DPI); 

    public static void Print(this Image image, string printer, bool showDialog = false) 
    { 
    if (printer == null) 
    { 
     throw new ArgumentNullException("Printer cannot be null.", nameof(printer)); 
    } 

    using (PrintDialog printDialog = new PrintDialog()) 
    { 
     using (PrintDocument printDoc = new PrintDocument()) 
     { 
     printDialog.Document = printDoc; 
     printDialog.Document.DocumentName = "My Document"; 
     printDialog.Document.OriginAtMargins = false; 
     printDialog.PrinterSettings.PrinterName = printer; 

     printDoc.PrintPage += (sender, e) => 
     { 
      // Draw to fill page 
      e.Graphics.DrawImage(image, 0, 0, e.PageSettings.PrintableArea.Width, e.PageSettings.PrintableArea.Height); 

      // Draw to default margins 
      // e.Graphics.DrawImage(image, e.MarginBounds); 
     }; 

     bool doPrint = !showDialog; 
     if (showDialog) 
     { 
      var result = printDialog.ShowDialog(); 
      doPrint = (result == DialogResult.OK); 
     } 

     if (doPrint) 
     { 
      printDoc.Print(); 
     } 
     } 
    } 
    } 

    public static async Task<bool> RenderAndPrintHTMLAsync(string html, string printer) 
    { 
    bool result = false; 

    // Enable HTML5 etc. (assuming we're running IE9+) 
    SetFeatureBrowserFeature("FEATURE_BROWSER_EMULATION", 9000); 

    // Force software rendering 
    SetFeatureBrowserFeature("FEATURE_IVIEWOBJECTDRAW_DMLT9_WITH_GDI", 1); 
    SetFeatureBrowserFeature("FEATURE_GPU_RENDERING", 0); 

    using (var webBrowser = new WebBrowser()) 
    { 
     webBrowser.ScrollBarsEnabled = false; 
     webBrowser.Width = WIDTH; 
     webBrowser.Height = HEIGHT; 
     webBrowser.DocumentCompleted += ((s, e) => browserReadySemaphore.Release()); 
     webBrowser.LoadHTML(html); 

     // Wait until the page loads. 
     await browserReadySemaphore.WaitAsync(); 

     // Save the picture 
     using (var bitmap = webBrowser.ToBitmap()) 
     { 
     bitmap.Save("WebBrowser_Bitmap.bmp"); 
     Print(bitmap, printer); 
     result = true; 
     } 
    } 

    return result; 
    } 

    /// <summary> 
    /// Make a Bitmap from the Control. 
    /// Remember to dispose after. 
    /// </summary> 
    /// <param name="control"></param> 
    /// <returns></returns> 
    public static Bitmap ToBitmap(this Control control) 
    { 
    Bitmap bitmap = new Bitmap(control.Width, control.Height); 
    Rectangle rect = new Rectangle(0, 0, control.Width, control.Height); 
    control.DrawToBitmap(bitmap, new Rectangle(0, 0, control.Width, control.Height)); 
    return bitmap; 
    } 

    /// <summary> 
    /// Required because of a bug where the WebBrowser only loads text once or not at all. 
    /// </summary> 
    /// <param name="webBrowser"></param> 
    /// <param name="htmlToLoad"></param> 
    /// <remarks> 
    /// http://stackoverflow.com/questions/5362591/how-to-display-the-string-html-contents-into-webbrowser-control/23736063#23736063 
    /// </remarks> 
    public static void LoadHTML(this WebBrowser webBrowser, string htmlToLoad) 
    { 
    webBrowser.Document.OpenNew(true); 
    webBrowser.Document.Write(htmlToLoad); 
    webBrowser.Refresh(); 
    } 

    /// <summary> 
    /// WebBrowser Feature Control 
    /// </summary> 
    /// <param name="feature"></param> 
    /// <param name="value"></param> 
    /// <remarks> 
    /// http://stackoverflow.com/questions/21697048/how-to-fix-a-opacity-bug-with-drawtobitmap-on-webbrowser-control/21828265#21828265 
    /// http://msdn.microsoft.com/en-us/library/ie/ee330733(v=vs.85).aspx 
    /// </remarks> 
    private static void SetFeatureBrowserFeature(string feature, uint value) 
    { 
    if (LicenseManager.UsageMode != LicenseUsageMode.Runtime) 
    { 
     return; 
    } 

    var appName = Path.GetFileName(System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName); 
    Registry.SetValue(
     @"HKEY_CURRENT_USER\Software\Microsoft\Internet Explorer\Main\FeatureControl\" + feature, 
     appName, 
     value, 
     RegistryValueKind.DWord); 
    } 
} 

 Смежные вопросы

  • Нет связанных вопросов^_^