- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
首先,我想指出,我已将此问题作为一个错误提交给 Microsoft,但他们此时不愿修复它。我正在寻找一种解决方法或更好的方法来实现我正在尝试做的事情,因为我们的客户认为这是一个相当重要的问题。
代码
MainWindow.xaml
<Grid x:Name="mainGrid">
<Grid.RowDefinitions>
<RowDefinition Height="Auto"/>
<RowDefinition Height="Auto"/>
<RowDefinition Height="Auto"/>
</Grid.RowDefinitions>
<ListBox ItemsSource="{Binding Images}">
<ListBox.ItemTemplate>
<DataTemplate>
<Image Source="{Binding}"/>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
<Button Content="Print to file" Grid.Row="1" Click="PrintToFile_Click"/>
<Button Content="Print to device" Grid.Row="2" Click="PrintToDevice_Click"/>
</Grid>
MainWindow.xaml.cs
public partial class MainWindow : Window
{
public IList<byte[]> Images { get; set; }
public MainWindow()
{
InitializeComponent();
Assembly currentAssembly = Assembly.GetExecutingAssembly();
this.Images = new List<byte[]>
{
ReadToEnd(currentAssembly.GetManifestResourceStream("PrintingInvestigation.Images.Chrysanthemum.jpg")),
ReadToEnd(currentAssembly.GetManifestResourceStream("PrintingInvestigation.Images.Desert.jpg")),
ReadToEnd(currentAssembly.GetManifestResourceStream("PrintingInvestigation.Images.Hydrangeas.jpg")),
};
this.DataContext = this;
}
public static byte[] ReadToEnd(System.IO.Stream stream)
{
long originalPosition = 0;
if (stream.CanSeek)
{
originalPosition = stream.Position;
stream.Position = 0;
}
try
{
byte[] readBuffer = new byte[4096];
int totalBytesRead = 0;
int bytesRead;
while ((bytesRead = stream.Read(readBuffer, totalBytesRead, readBuffer.Length - totalBytesRead)) > 0)
{
totalBytesRead += bytesRead;
if (totalBytesRead == readBuffer.Length)
{
int nextByte = stream.ReadByte();
if (nextByte != -1)
{
byte[] temp = new byte[readBuffer.Length * 2];
Buffer.BlockCopy(readBuffer, 0, temp, 0, readBuffer.Length);
Buffer.SetByte(temp, totalBytesRead, (byte)nextByte);
readBuffer = temp;
totalBytesRead++;
}
}
}
byte[] buffer = readBuffer;
if (readBuffer.Length != totalBytesRead)
{
buffer = new byte[totalBytesRead];
Buffer.BlockCopy(readBuffer, 0, buffer, 0, totalBytesRead);
}
return buffer;
}
finally
{
if (stream.CanSeek)
{
stream.Position = originalPosition;
}
}
}
private void PrintToDevice_Click(object sender, RoutedEventArgs e)
{
PrintDialog dialog = new PrintDialog();
if (dialog.ShowDialog() == true)
{
Thickness pageMargins;
if (dialog.PrintTicket.PageBorderless.HasValue == true)
{
if (dialog.PrintTicket.PageBorderless.Value == PageBorderless.Borderless)
{
pageMargins = new Thickness(0, 0, 0, 0);
}
else
{
pageMargins = new Thickness(20, 20, 20, 20);
}
}
else
{
pageMargins = new Thickness(20, 20, 20, 20);
}
int dpiX = 300;
int dpiY = 300;
if (dialog.PrintTicket.PageResolution != null &&
dialog.PrintTicket.PageResolution.X.HasValue &&
dialog.PrintTicket.PageResolution.Y.HasValue)
{
dpiX = dialog.PrintTicket.PageResolution.X.Value;
dpiY = dialog.PrintTicket.PageResolution.Y.Value;
}
else
{
dialog.PrintTicket.PageResolution = new PageResolution(dpiX, dpiY);
}
VisualDocumentPaginator paginator = new VisualDocumentPaginator(this.mainGrid, this.mainGrid.ActualWidth);
paginator.PageSize = new Size(dialog.PrintableAreaWidth, dialog.PrintableAreaHeight);
dialog.PrintDocument(paginator, "My first print");
GC.Collect();
}
}
private void PrintToFile_Click(object sender, RoutedEventArgs e)
{
string filePath = this.PrintToFile(null, this.mainGrid, "My first print", this.mainGrid.ActualHeight, this.mainGrid.ActualWidth);
Process.Start(filePath);
}
public string PrintToFile(Visual titleVisual, Visual contentVisual, string title, double bottomMost, double rightMost)
{
string printedFilePath = System.IO.Path.Combine(System.IO.Path.GetTempPath(), string.Format(CultureInfo.InvariantCulture, "{0}.xps", title));
XpsDocument printedDocument = new XpsDocument(printedFilePath, FileAccess.Write, System.IO.Packaging.CompressionOption.SuperFast);
VisualDocumentPaginator paginator = new VisualDocumentPaginator(contentVisual as FrameworkElement, rightMost);
paginator.PageSize = new Size(793.7, 1122.5);
XpsDocumentWriter writer = XpsDocument.CreateXpsDocumentWriter(printedDocument);
writer.Write(paginator, new PrintTicket
{
Collation = Collation.Collated,
CopyCount = 1,
DeviceFontSubstitution = DeviceFontSubstitution.On,
Duplexing = Duplexing.OneSided,
InputBin = InputBin.AutoSelect,
OutputColor = OutputColor.Color,
OutputQuality = OutputQuality.High,
PageMediaSize = new PageMediaSize(PageMediaSizeName.ISOA4),
PageOrientation = PageOrientation.Portrait,
PageResolution = new PageResolution(PageQualitativeResolution.High),
PagesPerSheet = 1,
TrueTypeFontMode = TrueTypeFontMode.Automatic
});
printedDocument.Close();
return printedFilePath;
}
}
VisualDocumentPaginator.cs
public class VisualDocumentPaginator : DocumentPaginator
{
#region Fields
private double desiredWidth;
private FrameworkElement element;
#endregion
#region Properties
public int Columns
{
get
{
return 1;// (int)Math.Ceiling(Element.ActualWidth / PageSize.Width);
}
}
public int Rows
{
get
{
return (int)Math.Ceiling(element.ActualHeight / PageSize.Height);
}
}
#endregion
#region Constructors
public VisualDocumentPaginator(FrameworkElement element, double desiredWidth)
{
this.desiredWidth = desiredWidth;
this.element = element;
}
#endregion
#region DocumentPaginator Members
public override DocumentPage GetPage(int pageNumber)
{
TransformGroup transforms = new TransformGroup();
double scaleRatio = this.PageSize.Width / this.desiredWidth;
int row = (pageNumber / Columns);
double pageHeight = -PageSize.Height * row / scaleRatio;
double pageWidth = -PageSize.Width * (pageNumber % Columns);
transforms.Children.Add(new TranslateTransform(pageWidth, pageHeight));
// Make sure the control is stretched to fit the page size.
if (scaleRatio != double.NaN)
{
ScaleTransform st = new ScaleTransform(scaleRatio, scaleRatio);
transforms.Children.Add(st);
}
element.RenderTransform = transforms;
Size elementSize = new Size(this.desiredWidth, element.ActualHeight);
element.Measure(elementSize);
element.Arrange(new Rect(new Point(0, 0), elementSize));
var page = new DocumentPage(element, this.PageSize, new Rect(), new Rect());
element.RenderTransform = null;
return page;
}
public override bool IsPageCountValid
{
get { return true; }
}
public override int PageCount
{
get
{
return Columns * Rows;
}
}
public override Size PageSize { set; get; }
public override IDocumentPaginatorSource Source
{
get { return null; }
}
#endregion
}
很抱歉发布了所有代码,但它涵盖了我遇到问题的所有领域。如果有帮助,这里是 Microsoft bug report其中附有一个示例项目,可以在其中重现问题。
问题
只有在写入 XPS 文件时才会出现此问题,其中只有第一张图像被打印 3 次,如果单击“打印到设备”按钮,则会打印正确的图像。
我绑定(bind)到 byte[] 的原因是因为我将我的图像保存在本地 SQL CE 数据库中。我们将它们存储在数据库中,因为它们每个只有 2KB 左右,而且我们允许用户将自己的图标导入系统中使用,我们需要一种机制来保证它们不会被意外删除。
注意
我注意到,如果我没有像上面提到的那样绑定(bind)到 byte[],那么我就看不到这个问题。考虑到系统目前不采用将图像存储在数据库中的方法,如果有解决方法,我宁愿坚持使用它,但我并不完全反对更换这些图像的存储机制。
最佳答案
我遇到过类似的问题,第一张图片被复制并替换了所有其他图片。就我而言,打印到设备、XPS 文档或 PDF 文档并不重要,问题仍然存在。
我使用 .NET 程序集反编译器来弄清楚 System.Windows.Xps.XpsDocumentWriter
类如何处理图像,以确定问题是在我的代码中还是在框架的代码中。我发现该框架使用字典将图像等资源导入文档。即使在 XPS 文档中只导入一次图像,文档也可以多次引用它们。
就我而言,我能够确定问题出在 System.Windows.Xps.Serialization.ImageSourceTypeConverter.GetBitmapSourceFromImageTable
方法中。当从 System.Windows.Media.Imaging.BitmapFrame
构建图像时,转换器将使用键在其字典之一中查找资源。在这种情况下,键对应于 BitmapFrame.Decoder.ToString()
方法返回的字符串的哈希码。不幸的是,由于我的图像是从字节数组而不是 URI 构建的,因此解码器的 ToString
方法返回“image”。由于无论图像如何,该字符串始终会生成相同的哈希码,ImageSourceTypeConverter
将认为所有图像都已添加到 XPS 文档中,并将返回要使用的第一个也是唯一一个图像的 Uri .这解释了为什么第一张图像被复制并替换所有其他图像。
我第一次尝试解决这个问题是覆盖 System.Windows.Media.Imaging.BitmapDecoder.ToString()
方法。为此,我尝试将 BitmapFrame
和 BitmapDecoder
包装到我自己的 BitmapFrame
和 BitmapDecoder
中。不幸的是,BitmapDecoder
类包含一个我无法定义的internal abstract
方法。由于我无法创建自己的 BitmapDecoder
,因此我无法实现该解决方法。
如前所述,当 BitmapSource
是 时,
。当它不是 System.Windows.Xps.Serialization.ImageSourceTypeConverter.GetBitmapSourceFromImageTable
方法将在字典中查找哈希码>位图框架BitmapFrame
时,它会根据图像二进制数据生成 CRC 值并在另一个字典中查找它。
在我的例子中,我决定将 System.Windows.Media.ImageSourceConverter
从字节数组生成的 BitmapFrame
包装到另一种类型的 BitmapSource
例如 System.Windows.Media.Imaging.CachedBitmap
。因为我真的不想要缓存位图,所以我创建了 CachedBitmap
将以下选项:
var imageSource = new CachedBitmap( bitmapFrame, BitmapCreateOptions.None, BitmapCacheOption.None );
有了这些选项,CachedBitmap
主要是一个简单的BitmapSource
包装器。就我而言,此解决方法解决了我的问题。
关于c# - XPS 文件中打印的重复图像,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30523815/
有没有办法在使用 WPF XPS 查看器打印 XPS 时设置默认布局方向? 我的固定文档 XPS 的页面方向设置为横向,页面媒体大小的宽度大于其高度,并且在查看器中正确显示为横向。 只是当您点击打印按
我正在将 WPF FlowDocument 保存到文件系统,使用此代码和带有 xps 扩展名的文件名: // Save FlowDocument to file system as XPS docum
我正在研究批量打印解决方案。 要打印的文件将采用各种(典型)格式,例如 PDF、Word、Excel 等。根据我收集(和测试)的结果,打印 XPS 文件是在 Windows 平台上进行的方式。 但是,
使用 Microsoft XPS Document Writer 打印 pdf 文档: string filename = "C:\\1.pdf"; Process process = new Pro
在他的博客条目中 Printing documents to Microsoft XPS Document Writer without user interaction冯远说 If you're p
我们的工具允许导出为 PNG,效果非常好。 现在,我想将导出添加到某种矢量格式。我尝试了 XPS,但结果根本不令人满意。 看看对比http://www.jakubmaly.cz/xps-vs-png.
我正在尝试在 XPS 文档中添加注释。是否可以使用 .Net API 并读取它们?我需要添加嵌入在 XPS 文件中的隐藏文本。 最佳答案 您可以使用 XpsDocument class 将元数据添加到
当我阅读有关 WPF 的书籍时,我看到作者提到 XPS 就像它很重要一样。 Windows 还包括它的 XPS 查看器,我已经看到它被列为 Windows 的“功能”。 但为什么?重点是什么?谁他妈的
XPS 文档的安全性如何?从XPS内部查看后文档中,找到了 Unicode-string 属性。有人可以注入(inject) e.x.吗?将脚本写入 Unicode 字符串属性? XPS 查看器如何处
我喜欢 DocumentViewer 在 WPF 应用程序中显示 XPS 文档。但我还需要编辑 XPS 文档。我找到了用于编辑 XPS 的独立应用程序 (.exe)。我需要的是一个用于查看和编辑 XP
我认为,XPS 文件类似于 PDF 文件,但 XPS 文件的结构是什么?就像 PDF 文件一样? 最佳答案 XPS vs. PDF on CNET 摘录: XPS 文档更像是 ZIP 存档文件,包含在
我正在用 C# 创建一个 asp.net web 应用程序来将一堆 .xps 文件转换为 .pdf 我想验证输入文件是出于安全原因,按其内容类型而不是文件扩展名转换。 最佳答案 According t
我尝试打印到 XPS 打印机(它不是我的默认打印机),但程序打开了一个对话框。我可以跳过对话框吗?这是代码: pSettings = new PrinterSettings()
我将使用 PrintDocument 显示我的 C# WinForms 应用程序的一部分,这几乎完成了,但是我的打印机出现了问题。我使用以下代码捕获表单图像,然后打印该图像,最后我使用 PrintPr
首先,我想指出,我已将此问题作为一个错误提交给 Microsoft,但他们此时不愿修复它。我正在寻找一种解决方法或更好的方法来实现我正在尝试做的事情,因为我们的客户认为这是一个相当重要的问题。 代码
如果用户打印报告,并且他们碰巧正在使用 微软 XPS 打印机 ,我希望默认文件名有意义。 我原以为 XPS 打印机会采用打印作业的名称,并将其用作默认文件名 - 但事实并非如此。 当我打印到该打印机时
我有一个 C# WinForms 应用程序。用户上传一个 .XPS 文件并指定一些打印机设置(份数、纸盒等)。该程序需要以编程方式打印具有这些设置的文档。即,不能有要打印的用户交互。 我可以接近 Sy
我的代码生成一个 saveFileDialog 来保存一个 .xps 文件,而不是实际打印任何东西。我的代码有问题吗?如果没有,我如何配置我的机器才能使此代码正常工作? class Printer {
我需要用 C# 保护 XPS 文档,但我找不到任何库,请帮助我... 谢谢。 最佳答案 我不完全确定“保护 XPS 文档”是什么意思。 如果您的意思是添加签名以便读者可以验证文档是否由您生成,我建议您
我需要拆分现有的 XPS 文档并创建一个只有原始文档的一页的新 XPS 文档。我尝试复制文档并从复制的文档中删除页面,但速度很慢。有没有更有效的方法来做到这一点?请使用 C#。 谢谢。 已解决: pu
我是一名优秀的程序员,十分优秀!