|
При выполнении в лабораторной работе упражнения №1 , а именно при выполнении нижеследующего кода: using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; using Microsoft.Xna.Framework.Graphics;
namespace Application1 { public partial class MainForm : Form { // Объявим поле графического устройства для видимости в методах GraphicsDevice device;
public MainForm() { InitializeComponent();
// Подпишемся на событие Load формы this.Load += new EventHandler(MainForm_Load);
// Попишемся на событие FormClosed формы this.FormClosed += new FormClosedEventHandler(MainForm_FormClosed); }
void MainForm_FormClosed(object sender, FormClosedEventArgs e) { // Удаляем (освобождаем) устройство device.Dispose(); // На всякий случай присваиваем ссылке на устройство значение null device = null; }
void MainForm_Load(object sender, EventArgs e) { // Создаем объект представления для настройки графического устройства PresentationParameters presentParams = new PresentationParameters(); // Настраиваем объект представления через его свойства presentParams.IsFullScreen = false; // Включаем оконный режим presentParams.BackBufferCount = 1; // Включаем задний буфер // для двойной буферизации // Переключение переднего и заднего буферов // должно осуществляться с максимальной эффективностью presentParams.SwapEffect = SwapEffect.Discard; // Устанавливаем размеры заднего буфера по клиентской области окна формы presentParams.BackBufferWidth = this.ClientSize.Width; presentParams.BackBufferHeight = this.ClientSize.Height;
// Создадим графическое устройство с заданными настройками device = new GraphicsDevice(GraphicsAdapter.DefaultAdapter, DeviceType.Hardware, this.Handle, presentParams); }
protected override void OnPaint(PaintEventArgs e) { device.Clear(Microsoft.Xna.Framework.Graphics.Color.CornflowerBlue);
base.OnPaint(e); } } } Выбрасывается исключение: Невозможно загрузить файл или сборку "Microsoft.Xna.Framework, Version=3.0.0.0, Culture=neutral, PublicKeyToken=6d5c3888ef60e27d" или один из зависимых от них компонентов. Не удается найти указанный файл. Делаю все пунктуально. В чем может быть проблема? |
Привязка WPF к реляционным данным
Вкладка Page4. Привязка интерфейса WPF к объекту данных с помощью класса ObjectDataProvider
До сих пор мы пользовались свойством DataContext для назначения источника привязки. WPF имеет еще один механизм назначения источника - через ресурс с помощью класса ObjectDataProvider. Класс позволяет создать ресурс, ссылающийся на коллекцию данных, возвращенную методом стороннего объекта, а потом привязать интерфейсный элемент или его свойство к этому ресурсу. Класс ObjectDataProvider может использоваться для привязки любых объектов, но мы рассмотрим его применение на предыдущем примере. Это позволит не делать подготовительную работу и не отвлекаться на новый код, а сосредоточиться только на нюансах привязки.
-
Добавьте в файле Window1.xaml к открывающему дескриптору окна Window строку подключения пространства имен приложения
<Window x:Class="DataBindingRelation.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Height="300" Width="300"
Title="WPF и типизированный DataSet"
MinHeight="300"
MinWidth="300"
WindowStartupLocation="CenterScreen"
xmlns:local="clr-namespace:DataBindingRelation"
>Конечно, подключение пространства имен можно поместить и в открывающий дескриптор вкладки Page4
<TabItem Header="Page4" xmlns:local="clr-namespace:DataBindingRelation">
но тогда его придется дублировать в каждой новой вкладке, где будут использоваться объекты процедурного кода. А можно поместить и во вкладке, и в окне, но мы этого делать не будем.
-
Добавьте в контейнер TabControl файла Window1.xaml разметку для новой вкладки Page4
<!-- Привязка к объекту с помощью класса ObjectDataProvider -->
<TabItem Header="Page4">
<TabItem.Resources>
<ObjectDataProvider
x:Key="myObjectProvider"
ObjectType="{x:Type local:MyObject}"
MethodName="GetCollection" />
</TabItem.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition />
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="2*" />
<ColumnDefinition Width="1.5*" />
<ColumnDefinition Width="2*" />
</Grid.ColumnDefinitions>
<TextBlock Grid.Column="0" HorizontalAlignment="Center">ContactName</TextBlock>
<ListBox Grid.Row="1" Grid.Column="0"
Margin="0,0,0,3"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ItemsSource="{Binding Source={StaticResource myObjectProvider},
Path=Customers}"
DisplayMemberPath="ContactName"
IsSynchronizedWithCurrentItem="True"
/>
<TextBlock Grid.Column="1" HorizontalAlignment="Center">OrderDate</TextBlock>
<ListBox Grid.Row="1" Grid.Column="1"
Margin="0,0,0,3"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ItemsSource="{Binding Source={StaticResource myObjectProvider},
Path=Customers/CustomersOrders}"
DisplayMemberPath="OrderDate"
IsSynchronizedWithCurrentItem="True"
/>
<TextBlock Grid.Column="2" HorizontalAlignment="Center">ProductName</TextBlock>
<ListBox Grid.Row="1" Grid.Column="2"
Margin="0,0,0,3"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ItemsSource="{Binding Source={StaticResource myObjectProvider},
Path=Customers/CustomersOrders/Orders_Order_Details_Extended}"
DisplayMemberPath="ProductName"
/>
</Grid>
<Grid Grid.Row="1">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition />
</Grid.ColumnDefinitions>
<Grid.RowDefinitions>
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<TextBlock Grid.Row="0" Margin="0,0,5,0">ShipName:</TextBlock>
<TextBox Grid.Row="0" Grid.Column="1"
Text="{Binding Source={StaticResource myObjectProvider},
Path=Customers/CustomersOrders/ShipName, Mode=OneWay}"
Focusable="False"
Margin="5,0,0,0"
/>
<TextBlock Grid.Row="1">ShipAddress:</TextBlock>
<TextBox Grid.Row="1" Grid.Column="1"
Text="{Binding Source={StaticResource myObjectProvider},
Path=Customers/CustomersOrders/ShipAddress, Mode=OneWay}"
Focusable="False"
Margin="5,0,0,0"
/>
<TextBlock Grid.Row="2">ShipCity:</TextBlock>
<TextBox Grid.Row="2" Grid.Column="1"
Text="{Binding Source={StaticResource myObjectProvider},
Path=Customers/CustomersOrders/ShipCity, Mode=OneWay}"
Focusable="False"
Margin="5,0,0,0"
/>
<TextBlock Grid.Row="3">ShipCountry:</TextBlock>
<TextBox Grid.Row="3" Grid.Column="1"
Text="{Binding Source={StaticResource myObjectProvider},
Path=Customers/CustomersOrders/ShipCountry, Mode=OneWay}"
Focusable="False"
Margin="5,0,0,0"
/>
</Grid>
</Grid>
</TabItem>-
Добавьте в файл Window1.xaml.cs после класса Window1 новый класс с именем MyObject с единственным методом GetCollection(), возвращающим реляционный набор данных
namespace DataBindingRelation
{
public partial class Window1 : Window
{
public Window1()
{
InitializeComponent();
Page1();
Page2();
Page3();
}
.........................................................
}
// Вкладка Page4. Объект с методом, возвращающим коллекцию привязки
class MyObject
{
public DataSet GetCollection()
{
// Извлекаем в поле строку соединения из файла App.config
String connectionString = System.Configuration.
ConfigurationManager.ConnectionStrings["MyNorthwind"].ConnectionString;
DataSet dataSet = new DataSet();// Создаем множественный набор данных
// Заполняем множественный набор данных из БД
using (OleDbConnection conn = new OleDbConnection(connectionString))
{
OleDbCommand selectCommand = conn.CreateCommand();
OleDbDataAdapter adapter = new OleDbDataAdapter(selectCommand);
// Загружает всю таблицу Customers
selectCommand.CommandText = "SELECT * FROM Customers";
adapter.Fill(dataSet, "Customers");
// Загружает всю таблицу Orders
selectCommand.CommandText = "SELECT * FROM Orders";
adapter.Fill(dataSet, "Orders");
// Загружаем представление по SQL-запросу
selectCommand.CommandText = @"SELECT DISTINCTROW [Order Details].OrderID, [Order Details].ProductID,
Products.ProductName,
[Order Details].UnitPrice, [Order Details].Quantity, [Order Details].Discount,
CCur([Order Details].[UnitPrice]*[Quantity]*(1-[Discount])/100)*100 AS ExtendedPrice
FROM Products INNER JOIN [Order Details] ON Products.ProductID = [Order Details].ProductID
ORDER BY [Order Details].OrderID";
adapter.Fill(dataSet, "Order_Details_Extended");
}
// Создание отношения между таблицами Customers и Orders
dataSet.Relations.Add("CustomersOrders",
dataSet.Tables["Customers"].Columns["CustomerID"],
dataSet.Tables["Orders"].Columns["CustomerID"]);
// Создание отношения между таблицами Orders и Order_Details_Extended
// Немного другим способом
DataColumn parentColumn = dataSet.Tables["Orders"].Columns["OrderID"];
DataColumn childColumn = dataSet.Tables["Order_Details_Extended"].Columns["OrderID"];
DataRelation relation = new DataRelation(
"Orders_Order_Details_Extended",
parentColumn, childColumn);
dataSet.Relations.Add(relation);
return dataSet;
}
}
}По сути, метод класса MyObject использует тот же самый код, который мы создавали для вкладки Page3. Здесь можно было бы кое-что подсократить, например, сразу упаковать код вкладки Page3 в метод и его использовать в примере вкладки Page4. Но не станем мелочиться на учебных примерах, а будем последовательны и дисциплинированы.
-
Запустите приложение - опять на вкладке Page4 получилась та же самая функциональность, что и в предыдущих примерах. Чудеса WPF продолжаются!!!
Следует иметь в виду, что данные, поставляемые методом пользовательскому интерфейсу, можно только просматривать, то есть привязка всегда будет только для чтения.
