我正在使用Visual Studio 2015社区报告查看器版本12在我的C#项目中显示rdlc报告。 这是正常的A4页面报告
它对于Windows XP,Vista,Windows 7在客户端PC上正常工作,但是当Windows 10 64位安装了相同的应用程序时,我将面临以下问题
如您在上图中所见,右侧和底部都有不必要的边距,并且字体大小也减小了。但是当我将报告导出为PDF时,生成的PDF与我设计的报告一样没有问题。
我尝试过什么:
打印类代码
public static class _cWainfoPrintReport
{
private static int m_currentPageIndex;
private static IList<Stream> m_streams;
public static Stream CreateStream(string name,
string fileNameExtension, Encoding encoding,
string mimeType, bool willSeek)
{
Stream stream = new MemoryStream();
m_streams.Add(stream);
return stream;
}
public static void _mExport(LocalReport report, bool print = true, double _pageWightInches = 8.27, double _pageHeightInches = 11.69, double _MarginTopInches = 0.025, double _MarginLeftInches = 0.025, double _MarginRightInches = 0.025, double _MarginBottomInches = 0.025)
{
string deviceInfo =
@"<DeviceInfo> <OutputFormat>EMF</OutputFormat> <PageWidth>" + _pageWightInches + "in</PageWidth> <PageHeight>" + _pageHeightInches + "in</PageHeight> <MarginTop>" + _MarginTopInches + "in</MarginTop> <MarginLeft>" + _MarginLeftInches + "in</MarginLeft> <MarginRight>" + _MarginRightInches + "in</MarginRight> <MarginBottom>" + _MarginBottomInches + "in</MarginBottom> </DeviceInfo>";
Warning[] warnings;
m_streams = new List<Stream>();
report.Render("Image", deviceInfo, CreateStream,
out warnings);
foreach (Stream stream in m_streams)
stream.Position = 0;
if (print)
{
_mPrint(_pageWightInches, _pageHeightInches, _MarginTopInches, _MarginLeftInches, _MarginRightInches, _MarginBottomInches);
}
report.ReleaseSandboxAppDomain();
}
// Handler for PrintPageEvents
public static void _mPrintPage(object sender, PrintPageEventArgs ev)
{
Metafile pageImage = new
Metafile(m_streams[m_currentPageIndex]);
// Adjust rectangular area with printer margins.
Rectangle adjustedRect = new Rectangle(
ev.PageBounds.Left - (int)ev.PageSettings.HardMarginX,
ev.PageBounds.Top - (int)ev.PageSettings.HardMarginY,
ev.PageBounds.Width,
ev.PageBounds.Height);
// Draw a white background for the report
ev.Graphics.FillRectangle(Brushes.White, adjustedRect);
// Draw the report content
ev.Graphics.DrawImage(pageImage, adjustedRect);
// Prepare for the next page. Make sure we haven't hit the end.
m_currentPageIndex++;
ev.HasMorePages = (m_currentPageIndex < m_streams.Count);
}
public static PaperSize CalculatePaperSize(double WidthInCentimeters, double HeightInCentimetres)
{
int Width = int.Parse((Math.Round((WidthInCentimeters * 0.393701) * 100, 0, MidpointRounding.AwayFromZero)).ToString());
int Height = int.Parse((Math.Round((HeightInCentimetres * 0.393701) * 100, 0, MidpointRounding.AwayFromZero)).ToString());
PaperSize NewSize = new PaperSize();
NewSize.RawKind = (int)PaperKind.Custom;
NewSize.Width = Width;
NewSize.Height = Height;
NewSize.PaperName = "Letter";
return NewSize;
}
public static void _mPrint(double _pageWightInches = 8.27, double _pageHeightInches = 11.69, double _MarginTopInches = 0.025, double _MarginLeftInches = 0.025, double _MarginRightInches = 0.025, double _MarginBottomInches = 0.025)
{
if (m_streams == null || m_streams.Count == 0)
throw new Exception("Error: no stream to print.");
PrintDocument printDoc = new PrintDocument();
PaperSize RequiredPaperSize = CalculatePaperSize(_pageWightInches * 2.54, _pageHeightInches * 2.54);
bool FoundMatchingPaperSize = false;
for (int index = 0; index < printDoc.PrinterSettings.PaperSizes.Count; index++)
{
if (printDoc.PrinterSettings.PaperSizes[index].Height == RequiredPaperSize.Height && printDoc.PrinterSettings.PaperSizes[index].Width == RequiredPaperSize.Width)
{
printDoc.PrinterSettings.DefaultPageSettings.PaperSize = printDoc.PrinterSettings.PaperSizes[index];
printDoc.DefaultPageSettings.PaperSize = printDoc.PrinterSettings.PaperSizes[index];
FoundMatchingPaperSize = true;
break;
}
}
if (!printDoc.PrinterSettings.IsValid)
{
throw new Exception("Error: cannot find the default printer.");
}
else
{
printDoc.PrintPage += new PrintPageEventHandler(_mPrintPage);
m_currentPageIndex = 0;
printDoc.Print();
}
}
public static void _mPrintToPrinter(this LocalReport report)
{
_mExport(report);
}
public static void _mDisposePrint()
{
if (m_streams != null)
{
foreach (Stream stream in m_streams)
stream.Close();
m_streams = null;
}
}
}
“打印”按钮上的代码
PrintViewer _PJobEntry = new PrintViewer();
DataTable dt = new DataSet1.MainTableDataTable();
dt.Rows.Add('vales for dataset');
_PJobEntry._RptView.LocalReport.DataSources.Add(new ReportDataSource("DataSet1", dt));
_PJobEntry._RptView.LocalReport.ReportEmbeddedResource = "WAINFOBUSSOLN.Printing.RptSaleInvoice02.rdlc";
_PJobEntry._RptView.SetDisplayMode(DisplayMode.PrintLayout);
_cWainfoPrintReport._mExport(_PJobEntry._RptView.LocalReport, true, 8.27, 11.69, 0.25, 0.25, 0.28, 0.25);
但其印刷也与上述相同
同一应用程序在装有Windows 10 64位的PC上运行,但在具有Windows 10 64位的客户端PC上部署后无法运行
在Client PC .Net 4.0框架上,SQL Server 2008,已安装Windows 10 64位
解决方法。
答案 0 :(得分:2)
我不认为此问题与所安装的任何工具有关,或者仅与Windows 10 64位无关。
相反,这是jdweng和Reza Aghaei提到的DPI问题。更具体地说,是高DPI设备。
不确定是否注意到,但是您上传的两个图像均具有不同的像素,而像素较低的图像则显示了报告的正确呈现。可以说这支持了由于高DPI引起的缩放问题。
现在,关于此的文章,帖子和问题很多。但是,与受害者提供某种正确指导非常接近的一种方法是Microsoft支持门户网站上的一种方法,该方法似乎没有针对此行为的可能解决方案(是,复数)和变通方法(再次是复数)。
您可以在这里找到本文:https://support.microsoft.com/en-au/help/3025083/windows-scaling-issues-for-high-dpi-devices
我相信,我在下面引用的解决方法应该可以帮助您节省时间。
更改应用程序属性
在资源管理器中或在“开始”菜单上,右键单击应用程序名称,选择“属性”,选择“兼容性”选项卡,然后选择“在高DPI设置上禁用显示比例缩放”复选框。
注意: 在Windows 10版本1703和更高版本的Windows中,“在高DPI设置上禁用显示缩放”选项的文本更改为“替代高DPI缩放行为,缩放由:应用程序执行。
答案 1 :(得分:0)
您应该禁用DPI意识,并将您的应用程序设置为DPI感知-即使不是,也可以防止扩展。
在Windows 10的较新版本上,我找到了一个不同的解决方案:您可以将应用程序置于任何DPI感知状态,但是可以在不同的DPI感知状态下打开特定的窗口。
但是,不幸的是,我仍在寻找一种100%可行的状态,但是似乎有两个选择是不错的选择。
更改DPI意识
if (WindowsVersion.WindowsIs10_1809OrNewer) // this you can get yourself, need to look at build number
{
// https://github.com/microsoft/Windows-classic-samples/blob/master/Samples/DPIAwarenessPerWindow/client/DpiAwarenessContext.cpp
_prevHosting = WinAPI.SetThreadDpiHostingBehavior((IntPtr)WinAPI.DPI_HOSTING_BEHAVIOR.BEHAVIOR_MIXED);
_prevDPIContext = WinAPI.SetThreadDpiAwarenessContext((IntPtr)win10DPIAwareness); // WinAPI.ContextDPIAwareness.Context_UnawareGdiScaled);
}
然后,您可以在此线程中打开新对话框,该对话框将包含ReportViewer。
稍后更改
if (WindowsVersion.WindowsIs10_1809OrNewer)
{
WinAPI.SetThreadDpiAwarenessContext(_prevDPIContext);
WinAPI.SetThreadDpiHostingBehavior(_prevHosting);
}
助手
public static bool WindowsIs10_1809OrNewer
{
get { return WindowsIs10OrNewer && Environment.OSVersion.Version.Build >= 17763; }
}
public static bool WindowsIs10OrNewer
{
get { return Environment.OSVersion.Version >= new Version(10, 0); }
}
public enum DPI_HOSTING_BEHAVIOR
{
BEHAVIOR_INVALID = -1,
BEHAVIOR_DEFAULT = 0,
BEHAVIOR_MIXED = 1
}
public enum ContextDPIAwareness
{
Context_Unaware = -1,
Context_SystemAware = -2,
Context_PerMonitorAware = -3,
Context_PerMonitorAwareV2 = -4,
Context_UnawareGdiScaled = -5,
}
您可以尝试使用Context_UnawareGdiScaled(实际上通常可以正确缩放ReportViewer),也可以尝试使用Context_PerMonitorAware禁用缩放。
在RDP会话中我遇到了问题,如果您没有这个问题,那么使用这些选项可能就可以了。
此外,可悲的是,这仅适用于更新的Windows 10版本。由于这些更新不是可选的,因此可能大多数Windows 10用户将使用新版本,并且Windows 8不再受欢迎并且所有其他Windows版本均不受支持,因此许多用户应使用最新的Windows 10版本。
当然还远远不够完美。
顺便说一句,对于Windows的较早版本,我只是通过SetProcessDpiAwareness禁用缩放,没有找到更好的方法。