Search code examples
c#wpfvisual-studio

how to copy paste a WPF Window (clone) and not have errors


I am using .Net 4.5.2 with WPF and C# 5.0. I created a Window in a WPF project. I would like to copy paste this window in the solution explorer, making a second window, and rename it to a new name.

When I do this, the new (copied) window's InitializeComponent() method always gives me an error. How does one cleanly copy a window (and it's code, etc.) in the solution explorer?

This question was answered partially here: Copy pasting WPF window gives error however the answer did not solve my issue.

My approach (that does not work):

  1. Create a window and title it WindowTest
  2. In the solution explorer, select WindowTest and copy, then paste it into the same project
  3. Rename the new copied Window to WindowTestCopy
  4. In WindowTestCopy, change the x:class property in xaml to be WindowTestCopy instead of WindowTest
  5. Open the code behind in WindowTestCopy, and change any references to WindowTest to WindowTestCopy
  6. Compile

Expected: no errors, the copy (clone) operation is successful

Actual: compile error "Cannot access non-static method 'InitializeComponent' in static context".

I have only this one error. Obviously InitializeComponent() is becoming an ambiguous reference, but it isn't clear to me how to make manual edits to the code to fix this. I wish that VS or Resharper would automatically assist me with this.

UPDATE

WindowTest contains two userControls that I had not mentioned previously. After the copy/paste occurs, for some reason the following xaml elements became malformed within WindowTestCopy:

xmlns:userControls....(ellided) xmlns:userControls....(ellided)

By deleting these, Resharper determined that the userControl objects were missing xmlns reference tags and asked me if I wanted to import them automatically. I selected yes. After Resharper added the missing xmlns reference tags I was able to compile (all errors disappeared).

I do not have an explanation for why this happened. In my steps to reproduce, I do not edit the xaml and it should therefore be identical to the originating xaml. This is curious behavior, but at least there is a workaround as stated.


Solution

  • Someone else actually posted and then deleted had the correct idea, but brief on why.

    The issue you were running into is due to a duplication of a class within the same project... even though in the solution explorer you can easily copy/paste a WPF form or any other class, user control class, etc. When copied, it suffixes the file names to " - Copy", and if you compile right away will fail. Reason being? In the class itself the class name and window name is the same. So, in the project you have TWO instances of the class and it is choking on that hoping for you to resolve the duplicates. I ran into this exact same thing early on in my WPF development.

    So, now that I've explained WHY it is failing, here is what you would need to do, to correct the issue. I created a window in my sample project called "FirstWindowSample", so two files are created FirstWindowSample.xaml and FirstWindowSample.xaml.cs. The .cs version is the code-behind of the window (or user control class, the principle is EXACTLY the same.

    If you look in the "FirstWindowSample.xaml" code (the visual-design version), the code will look something like...

    <Window x:Class="YourProject.FirstWindowSample"
            xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
            xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
            Title="FirstWindowSample" Height="300" Width="300">
        <Grid>
    
        </Grid>
    </Window>
    

    Notice in the first line

       x:Class="YourProject.FirstWindowSample"
    

    You need to change the ".FirstWindowSample" to whatever you want for your new form such as

       x:Class="YourProject.SecondWindowSample"
    

    Next, change over to the CODE-BEHIND file "FirstWindowSample.xaml.cs".

    namespace YourProject
    {
        /// <summary>
        /// Interaction logic for FirstWindowSample.xaml
        /// </summary>
        public partial class FirstWindowSample : Window
        {
            public FirstWindowSample()
            {
                InitializeComponent();
            }
        }
    }
    

    This is the hard-reference of the class name and its corresponding constructor. Just change these to match the new window name you are expecting (such as sample SecondWindowSample)

    namespace YourProject
    {
        /// <summary>
        /// Interaction logic for SecondWindowSample.xaml
        /// </summary>
        public partial class SecondWindowSample : Window
        {
            public SecondWindowSample()
            {
                InitializeComponent();
            }
        }
    }
    

    At the end your project will now have a completely different class name that is not duplicated and cause compile errors. The FIRST time you run into this, yeah... a pain to find out. Once you get it, the next 2-3 times are the "oh-yeah" refresher... After that you sort of do it without thinking about it.