У меня есть большое WPF-решение, работающее в течение 2 лет. Теперь мы запускаем автоматическую среду сборки для этого решения, когда произошло самое странное.
В 50% наших сборках я получаю эту ошибку:
Исключение: невозможно выполнить листинг объекта тип 'System.Windows.Controls.StackPanel' печатать 'System.Windows.Controls.Border. Ошибка в объекте 'System.Windows.Controls.StackPanel' в файле разметки...
Это кажется достаточно простым. Проблема в том, что мой код выглядит следующим образом:
<UserControl x:Class="SiSM.Episode.Mishap.SpecializationList" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:Converters="clr-namespace:Utils.Converters;assembly=Utils" ...>
<Border x:Name="root" BorderThickness="0.5">
<StackPanel x:Name="stackPanelRoot" VerticalAlignment="Stretch">
<Grid>
...
</Grid>
<StackPanel>
...
</StackPanel>
<ScrollViewer>
...
</ScrollViewer>
</StackPanel>
</Border>
</UserControl>
Ошибка здесь, потому что если я переключу стекную панель для док-панели, сообщение об ошибке изменилось на док-панель.
Моя среда сборки следующая:
Скопируйте код в папку сборки:
private void CopyCode(string sourceDir, string destinationDir) {
foreach (string dirPath in Directory.GetDirectories(sourceDir, "*", SearchOption.AllDirectories)) {
if (!dirPath.Contains(".svn") && !dirPath.Contains(@"\bin") && !dirPath.Contains(@"\obj")) {
Directory.CreateDirectory(dirPath.Replace(sourceDir, destinationDir));
}
}
foreach (string newPath in Directory.GetFiles(sourceDir, "*.*", SearchOption.AllDirectories)) {
if (!newPath.Contains(".svn") && !newPath.Contains(@"\bin") && !newPath.Contains(@"\obj")) {
string dest = newPath.Replace(sourceDir, destinationDir);
File.Copy(newPath, dest);
}
}
Worker.ReportProgress(5, "Copy done");
}
И построим решение:
private void Compile(string buildConfiguration) {
Engine engine = new Engine();
FileLogger logger = new FileLogger { Parameters = @"logfile=C:\builds\build.log" };
engine.RegisterLogger(logger);
BuildPropertyGroup bpg = new BuildPropertyGroup();
bpg.SetProperty("Configuration", buildConfiguration, true);
engine.GlobalProperties = bpg;
var project = new Project(engine);
project.Load(ProjectFilePath);
bool success = engine.BuildProject(project);
engine.UnregisterAllLoggers();
}
Что-то не так, или есть какая-то известная проблема с WPF и механизмом сборки Microsoft?
Изменить 1
Я обнаружил ошибку. Если я впервые запускаю автоматическое приложение для сборки, это всегда удастся, но если я запустил его через несколько секунд, вы получите вышеуказанную ошибку. Так что, возможно, что-то я забыл закрыть, создавая ошибку.
Я добавил engine.Shutdown();
в конце метода компиляции, но это не устранило проблему.
Изменить 2
Благодаря предложению @swiszcz, только что нашел самое странное. Файл SpecializationList.g.cs(в папке obj) изменяется между первой и второй строкой
Первая сборка
void System.Windows.Markup.IComponentConnector.Connect(int connectionId, object target) {
switch (connectionId)
{
case 1:
this.stackPanelRoot = ((System.Windows.Controls.StackPanel)(target));
return;
case 2:
#line 63 "..\..\..\Mishap\SpecializationList.xaml"
((System.Windows.Controls.Button)(target)).Click += new System.Windows.RoutedEventHandler(this.buttonShowGlobalView_Click);
...
Второй сборник
void System.Windows.Markup.IComponentConnector.Connect(int connectionId, object target) {
switch (connectionId)
{
case 2:
this.stackPanelRoot = ((System.Windows.Controls.StackPanel)(target));
return;
case 3:
...
Он увеличивает значение 1 в состоянии переключения, а во второй сборке он не может преобразовать кнопку (случай 2) в StackPanel (случай 1).