我希望tp构建一个WPF(或Windows Forms,如果它可以解决问题),它可以同时显示具有不同凭据的相同网页(它是具有复杂用户操作流程的Web应用程序的测试工具)。
到目前为止,我有一个非常简单的WPF应用程序:
<Window x:Class="TestTool.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow" Height="600" Width="800">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*"/>
<ColumnDefinition Width="*"/>
</Grid.ColumnDefinitions>
<WebBrowser Margin="8" Source="http://theapptotest" />
<WebBrowser Margin="8" Source="http://theapptotest" Grid.Column="1" />
</Grid>
</Window>
这不是应用程序的目标布局,但它可以测试我的目标是否可行。
正如预期的那样,应用程序并排显示相同的Web应用程序。但是,如果我在第一个Web浏览器控件中登录,则第二个也会登录。我想这是因为cookie是为当前用户共享的。
我检查了WebBrowser class documentation但我找不到任何有用的东西。
我如何实现目标?
是否可以将一个网络浏览器与其他网页浏览器“隔离”?
是否可以“模仿”用户控件?像一些应用程序(主要是Web浏览器)正在产生不同的进程,我可以在不同的帐户下“生成”进程并在我的主窗口中显示它吗?
请注意,我不必与网页互动。我只想通过更改应用程序的选项卡(每个用户一个选项卡)来切换用户。
PS:我不会坚持使用Internet Explorer引擎,如果有替代浏览器的解决方案,请告诉我
PS:目标应用程序使用两种身份验证机制。用户可以使用Windows帐户或基于表单的自定义身份验证进行身份验证。如果需要,我只能使用表单身份验证(如果我可以插入一个单独的cookie容器就可以完成这项工作)。
[编辑] 我尝试与其他用户凭据生成Internet Explorer的新实例,然后通过将其父级设置为我的一个Windows窗体主机来附加进程主窗口。但是,这非常不可靠(如果IE的任何现有实例正在运行,则进程经常崩溃,无法工作等。
答案 0 :(得分:2)
好的,在Simon Mourrier previous answer的帮助下,我终于找到了实现目标的方法。
我已经创建了一个自定义用户控件:
Xaml部分:
<UserControl x:Class="WpfApplication1.AppIsolatedView"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
mc:Ignorable="d" SizeChanged="UserControl_SizeChanged"
d:DesignHeight="300" d:DesignWidth="300" Loaded="UserControl_Loaded" Unloaded="UserControl_Unloaded">
<Grid>
<WindowsFormsHost Margin="12" Name="windowsFormsHost1" />
</Grid>
</UserControl>
使用此代码:
using System;
using System.ComponentModel;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Threading;
using System.Windows;
using System.Windows.Controls;
namespace WpfApplication1
{
public partial class AppIsolatedView : UserControl
{
public AppIsolatedView()
{
InitializeComponent();
_panel = new System.Windows.Forms.Panel();
windowsFormsHost1.Child = _panel;
}
string url = "http://theapptotest";
System.Windows.Forms.Panel _panel;
Process _process;
private void UserControl_Loaded(object sender, RoutedEventArgs e)
{
if (IsInDesignModeStatic) return; // Avoid consumer of the control to explode when in design mode
ProcessStartInfo psi = new ProcessStartInfo("iexplore.exe", "-noframemerging -private \"" + url + "\"");
_process = Process.Start(psi);
_process.WaitForInputIdle();
Thread.Sleep(2000);
SetParent(_process.MainWindowHandle, _panel.Handle);
// remove control box
int style = GetWindowLong(_process.MainWindowHandle, GWL_STYLE);
style = style & ~WS_CAPTION & ~WS_THICKFRAME;
SetWindowLong(_process.MainWindowHandle, GWL_STYLE, style);
// resize embedded application & refresh
ResizeEmbeddedApp();
}
private void UserControl_Unloaded(object sender, RoutedEventArgs e)
{
if (_process != null)
{
_process.Refresh();
_process.Close();
}
}
private void ResizeEmbeddedApp()
{
if (_process == null)
return;
SetWindowPos(_process.MainWindowHandle, IntPtr.Zero, 0, 0, (int)_panel.ClientSize.Width, (int)_panel.ClientSize.Height, SWP_NOZORDER | SWP_NOACTIVATE);
}
protected override Size MeasureOverride(Size availableSize)
{
Size size = base.MeasureOverride(availableSize);
ResizeEmbeddedApp();
return size;
}
private static bool? _isInDesignMode;
/// <summary>
/// Gets a value indicating whether the control is in design mode (running in Blend
/// or Visual Studio).
/// </summary>
public static bool IsInDesignModeStatic
{
get
{
if (!_isInDesignMode.HasValue)
{
#if SILVERLIGHT
_isInDesignMode = DesignerProperties.IsInDesignTool;
#else
var prop = DesignerProperties.IsInDesignModeProperty;
_isInDesignMode
= (bool)DependencyPropertyDescriptor
.FromProperty(prop, typeof(FrameworkElement))
.Metadata.DefaultValue;
#endif
}
return _isInDesignMode.Value;
}
}
private void UserControl_SizeChanged(object sender, SizeChangedEventArgs e)
{
ResizeEmbeddedApp();
}
#region pinvoke stuff
[DllImport("user32.dll")]
private static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);
[DllImport("user32.dll", SetLastError = true)]
private static extern int GetWindowLong(IntPtr hWnd, int nIndex);
[DllImport("user32")]
private static extern IntPtr SetParent(IntPtr hWnd, IntPtr hWndParent);
[DllImport("user32")]
private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, int uFlags);
private const int SWP_NOZORDER = 0x0004;
private const int SWP_NOACTIVATE = 0x0010;
private const int GWL_STYLE = -16;
private const int WS_CAPTION = 0x00C00000;
private const int WS_THICKFRAME = 0x00040000;
#endregion
}
}
我不得不作弊:
-noframemerging
时添加IExplore.exe
以避免内部IE会话共享。MainWindowHandle
属性在几毫秒后发生了变化(我猜iexplore有一个引导程序进程连接到现有进程或者生成一个新进程,有新句柄,或类似的东西)。 我仍然需要扩展用户控件以接受输入参数(url是硬编码的),但这个想法就在这里。我只需要专注于应用程序本身。