Introduction
This article explains in detail the process of creating a DotNetNuke module.
Requirements
To use this tutorial, you need:
- Visual Studio Express (download here)
- SQL Server Express (download here)
- DotNetNuke Starter Kit 4.x (download here)
Note: When you download it, it may change the file extension to ".zip". After downloading, change this to ".vsi", and then you can double-click on the file to install the templates.
Note: Do not attempt to create modules on a production installation of DotNetNuke; trust me, you will invariably screw up the installation and will have to restore the database or re-install.
Setup
- Install Visual Studio Express if you haven't already done so (download here).
- Install SQL Server Express if you haven't already done so (download here).
- Follow the directions here to install the DotNetNuke Starter Kit and to create a DotNetNuke website.
Very important: Click once on the root directory in the Solution Explorer window. Doing so will ensure that the starter module code is created in the proper place.
Select File from the toolbar, then New File:
On the next menu, click once on DotNetNuke Module (under My Templates), select "Visual Basic" from the Language dropdown, type "GuestBook." in the Name box, and click Add.
I interrupt this tutorial for an important warning:
Note: If the files DataProvider.vb, GuestBookController.vb, GuestBookInfo.vb, and SqlDataProvider.vb in the "ModuleName" directory are under "DesktopModules/App_Code" then they are in the wrong place.
Click on the "ModuleName" folder and drag it under the "App_Code" directory that is directly under the main root of the website. You will also have to move the "DesktopModules/ModuleName" folder and drag the "ModuleName" folder so it is under the "DesktopModules" folder that is directly under the main root of the website.
This is the correct way it should look:
Now back to the tutorial:
A very helpful help page comes up and instructs you to rename /App_Code/ModuleName to /App_Code/GuestBook, and rename /DesktopModules/ModuleName to /DesktopModules/GuestBook.
Make the changes by right-clicking the folder name in the Solutions Explorer and selecting Rename from the menu. You have to do this twice. Once for the folder under "App_Code", and then for the folder under "DesktopModules".
Setting Up The Module in DotNetNuke
Again, you should be familiar with this step from Shaun Walker's post. Double-click on "GuestBook.SqlDataProvider".
It will open in the main menu:
Click once on the page, then from the toolbar, select Edit, then Select All (Ctrl +A):
Then, Edit and Copy (Ctrl +C):
From the toolbar, select View, then Output:
Now, from the toolbar, select Build, then Build Web Site:
The website should build without errors.
In the Solution Explorer, right-click on "Default.aspx" and select Set As Start Page:
From the toolbar, select Debug, then Start Without Debugging.
(You may wonder why I am not covering debugging. While creating this tutorial, I ran into permission problems on my own machine. They are easily fixed but I wanted to avoid any unneeded complexity. If you need help debugging, see this post.)
The website should now come up.
Click Login:
Log in as "host". The password (if you haven't already changed it) is also "host":
Click on the Host menu and select SQL:
Paste the script you copied into the window, click the "Run as Script" box, and click Execute:
Watch the progress bar in your web browser because when the script has been run, the text in the window will shift, and then... nothing... (this is version 4.0.2, perhaps we will get a "completed" message some day) :).
Click on the Host menu and select SQL again:
You now have a clear form again:
Switch back to Visual Studio, and in the Solution Explorer, double-click on "01.00.00.SqlDataProvider" in the "DestopModules/GuestBook" directory so that it shows up in the main window:
Click once on the page, then from the toolbar, select Edit, then Select All (Ctrl +A):
Then Edit and Copy (Ctrl +C):
Switch back to the DotNetNuke website, and paste (Ctrl +V) the script you copied into the window, click the "Run as Script" box, and click Execute:
Again, watch the progress bar in your web browser because when the script has been run, the text in the window will shift and then... nothing...
Under the Page Functions menu, click Add:
In the Page Details menu:
- Enter "Guest Book" for Page Name.
- Enter "Guest Book" for Page Title.
- Enter "Guest Book" for Description.
- Click the View Page box next to All Users.
Then, click Update:
From the Module drop-down, select "GuestBook":
Then, click Add:
If you get the "Object reference not set to an instance of an object" error, just click on the "Guest Book" link:
(However, if you are running DotNetNuke on a machine that is lower than a Pentium III with 512 KB RAM, then DotNetNuke will regularly throw this error).
The module should now appear:
The Guest Book Module
We will walk through the construction of the Guest Book module in these steps:
- Data Access Layer (DAL) - We will create the tables, the stored procedures, and alter code in the "SqlDataProvider.vb" and "DataProvider.vb" files.
- Business Logic Layer (BLL) - We will alter code in the "GuestBookController.vb" and "GuestBookInfo.vb" files.
- Presentation Layer (UI) - We will alter code in the "ViewGuestBook.ascx", and in the other files in the "DesktopModules/GuestBook" directory.
Data Access Layer (DAL)
To build the Data Access Layer, we will:
- Create the table.
- Create the stored procedures.
- Alter the SqlDataProvider.vb file. Put the code that calls the stored procedures in the "SqlDataProvider.vb" file.
- Alter the DataProvider.vb file. Put the methods (that are overridden by the "SqlDataProvider.vb" file) in the "DataProvider.vb" file.
And that's it! It's actually not a lot of code to add to the "SqlDataProvider.vb" and "DataProvider.vb" files. Creating the tables and stored procedures is standard database programming that you are probably already used to.
From: "DotNetNuke Module Developers' Guide"
Copyright © 2003-2005 Perpetual Motion Interactive Systems, Inc. All Rights Reserved.
Connect To The Database
If you have created your DotNetNuke website using SQL Server 2005 Express, your database connection should already be set up for you. If you set it up using SQL Server 2000 or SQL Server 2005, then you may have to set it up.
To set up a connection to the SQL Server database, from the toolbar, select View, then Server Explorer.
In the Server Explorer, right-click on Data Connections, and select "Add Connection":
Fill in the connection information, and click OK.
Delete the Sample Database Objects
You now need to delete the table and the stored procedures that the template created so that you can make new ones (that happen to use the same names) that the guestbook will use.
The connection will now show up in the Server Explorer. Click the plus next to the connection to expand it. Next, click the plus icon next to Tables to expand it.
Right click on the table "YourCompany_GuestBook" and select "Delete" and delete the table.
Note: At this point, the module will no longer work in your DotNetNuke web site. It will not work again until you have completed all the steps in this tutorial.
Click the plus icon next to Store Procedures to expand it. One at a time, right-click on the following stored procedures and select "Delete" to delete them. Remember, you have to delete them one at a time.
YourCompany_AddGuestBook
YourCompany_DeleteGuestBook
YourCompany_GetGuestBook
YourCompany_GetGuestBooks
YourCompany_UpdateGuestBook
Create The Table
Log into the DotNetNuke website as Host (if you are not already logged in as Host), and from the Host menu, select SQL.
Paste the following script in the box:
CREATE TABLE [dbo].[YourCompany_GuestBook](
[ID] [int] IDENTITY(1,1) NOT NULL,
[ModuleID] [int] NULL,
[Name] [nvarchar](50) COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
[Email] [nvarchar](50) COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
[Message] [nvarchar](250) COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
[DateEntered] [datetime] NULL,
CONSTRAINT [PK_YourCompany_GuestBook] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (IGNORE_DUP_KEY = OFF) ON [PRIMARY]
) ON [PRIMARY]
Do not select the "Run as Script" box, and click Execute.
(Note: This is a SQL 2005 script. It will not work in SQL 2000. For SQL 2000, you will have to create the table manually as explained in the next two steps.)
Optionally, you can create the table in the table designer in Visual Web Developer. In the Server Explorer, right-click on Tables and select Add New Table:
Design the table in the table designer using the graphics below. Make the "ID" column an "identity" column and also set it as the primary key. Save it as "YourCompany_GuestBook
".
Create the Stored Procedures
Log into the DotNetNuke website as Host (if you are not already logged in as Host), and from the Host menu, select SQL.
Paste the following script in the box:
CREATE PROCEDURE {databaseOwner}
[{objectQualifier}YourCompany_GuestBook_Delete]
(
@ID int
)
AS
DELETE FROM {objectQualifier}YourCompany_GuestBook
WHERE (ID = @ID)
RETURN
GO
CREATE PROCEDURE {databaseOwner}
[{objectQualifier}YourCompany_GuestBook_GetAll]
(
@ModuleID int
)
AS
SELECT ID, ModuleID, Name, Email, Message, DateEntered
FROM {objectQualifier}YourCompany_GuestBook
WHERE (ModuleID = @ModuleID)
order by DateEntered DESC
RETURN
GO
CREATE PROCEDURE {databaseOwner}
[{objectQualifier}YourCompany_GuestBook_Insert]
(
@ModuleID int,
@Name nvarchar(50),
@Email nvarchar(50),
@Message nvarchar(250)
)
AS
INSERT INTO {objectQualifier}YourCompany_GuestBook
(ModuleID, Name, Email, Message, DateEntered)
VALUES (@ModuleID,@Name,@Email,@Message,getdate())
RETURN
GO
CREATE PROCEDURE {databaseOwner}
[{objectQualifier}YourCompany_GuestBook_Update]
(
@ID int,
@Name nvarchar(50),
@Email nvarchar(50),
@Message nvarchar(250),
@DateEntered datetime
)
AS
UPDATE {objectQualifier}YourCompany_GuestBook
SET Name = @Name, Email = @Email,
Message = @Message, DateEntered = @DateEntered
WHERE (ID = @ID)
RETURN
GO
Select the Run as Script box, and click Execute:
Verify that the stored procedures have been created. Switch back to Visual Studio, and select View from the toolbar and Server Explorer:
On the "Server Explorer" window, right-click on Stored Procedures and select Refresh:
Scroll down the list of stored procedures and verify that these stored procedures have been created:
YourCompany_GuestBook_Delete
YourCompany_GuestBook_GetAll
YourCompany_GuestBook_Insert
YourCompany_GuestBook_Update
Alter the "SqlDataProvider.vb" file. In Visual Studio, select View from the toolbar and "Solution Explorer":
In the "Solution Explorer" window, expand the "GuestBook" directory under the "App_code" folder and double-click on the "SqlDataprovider.vb" file:
Replace all the code with:
Imports System
Imports System.Data
Imports System.Data.SqlClient
Imports Microsoft.ApplicationBlocks.Data
Imports DotNetNuke.Common.Utilities
Imports DotNetNuke.Framework.Providers
Namespace YourCompany.Modules.GuestBook
Public Class SqlDataProvider Inherits DataProvider
Private Const ProviderType As String = "data"
Private Const ModuleQualifier As String = ""
Private _providerConfiguration As _
ProviderConfiguration = _
ProviderConfiguration.GetProviderConfiguration(ProviderType)
Private _connectionString As String
Private _providerPath As String
Private _objectQualifier As String
Private _databaseOwner As String
' <summary>
' Constructs new SqlDataProvider instance
' </summary>
Public Sub New()
MyBase.New()
'Read the configuration specific information for this provider
Dim objProvider As Provider = CType(_providerConfiguration._
Providers(_providerConfiguration.DefaultProvider), Provider)
'Read the attributes for this provider
If ((objProvider.Attributes("connectionStringName") <> "") _
AndAlso (System.Configuration.ConfigurationManager._
AppSettings(objProvider.Attributes(_
"connectionStringName")) <> "")) Then
connectionString = System.Configuration._
ConfigurationManager.AppSettings(_
objProvider.Attributes("connectionStringName"))
Else
_connectionString = _
objProvider.Attributes("connectionString")
End If
_providerPath = objProvider.Attributes("providerPath")
_objectQualifier = objProvider.Attributes("objectQualifier")
If ((_objectQualifier <> "") _
AndAlso (_objectQualifier.EndsWith("_") _
= False)) Then
_objectQualifier = (_objectQualifier + "_")
End If
_databaseOwner = objProvider.Attributes("databaseOwner")
If ((_databaseOwner <> "") _
AndAlso (_databaseOwner.EndsWith(_
".") = False)) Then
_databaseOwner = (_databaseOwner + ".")
End If
End Sub
' <summary>
' Gets and sets the connection string
' </summary>
Public ReadOnly Property ConnectionString() As String
Get
Return _connectionString
End Get
End Property
' <summary>
' Gets and sets the Provider path
' </summary>
Public ReadOnly Property ProviderPath() As String
Get
Return _providerPath
End Get
End Property
' <summary>
' Gets and sets the Object qualifier
' </summary>
Public ReadOnly Property ObjectQualifier() As String
Get
Return _objectQualifier
End Get
End Property
' <summary>
' Gets and sets the database ownere
' </summary>
Public ReadOnly Property DatabaseOwner() As String
Get
Return _databaseOwner
End Get
End Property
' --------------------------------------------------
' <summary>
' Gets the fully qualified name
' of the stored procedure
' </summary>
' <param name="name">The name
' of the stored procedure</param>
' <returns>The fully qualified name</returns>
' --------------------------------------------------
Private Function GetFullyQualifiedName(ByVal _
name As String) As String
Return (DatabaseOwner _
+ (ObjectQualifier _
+ (ModuleQualifier + name)))
End Function
' --------------------------------------------------
' <summary>
' Gets the value for the field or DbNull
' if field has "null" value
' </summary>
' <param name="Field">The field to evaluate</param>
' <returns></returns>
' --------------------------------------------------
Private Function GetNull(ByVal Field As Object) As Object
Return Null.GetNull(Field, DBNull.Value)
End Function
Public Overrides Sub YourCompany_GuestBook_Insert(ByVal _
ModuleId As Integer, ByVal Name As String, _
ByVal Email As String, ByVal Message As String)
SqlHelper.ExecuteNonQuery(ConnectionString, _
GetFullyQualifiedName("YourCompany_GuestBook_Insert"), _
ModuleId, Name, Email, Message)
End Sub
Public Overrides Sub _
YourCompany_GuestBook_Delete(ByVal ID As Integer)
SqlHelper.ExecuteNonQuery(ConnectionString, _
GetFullyQualifiedName("YourCompany_GuestBook_Delete"), ID)
End Sub
Public Overrides Function YourCompany_GuestBook_GetAll _
(ByVal ModuleId As Integer) As IDataReader
Return CType(SqlHelper.ExecuteReader(ConnectionString, _
GetFullyQualifiedName("YourCompany_GuestBook_GetAll"), _
ModuleId), IDataReader)
End Function
Public Overrides Sub YourCompany_GuestBook_Update(ByVal ID _
As Integer, ByVal Name As String, ByVal Email As _
String, ByVal Message As String, _
ByVal DateEntered As DateTime)
SqlHelper.ExecuteNonQuery(ConnectionString, _
GetFullyQualifiedName("YourCompany_GuestBook_Update"), _
ID, Name, Email, Message, DateEntered)
End Sub
End Class
End Namespace
You will notice that you will see errors such as:
"'YourCompany_GuestBook_Insert' cannot be declared
'Overrides' because it does not override a sub in a base class"
This is because the "DataProvider.vb" must indicate the methods that the "SqlDataProvider.vb" overrides. When we place those methods in the "DataProvider.vb" file, the errors will go away.
Alter the "DataProvider.vb" file. In the Solution Explorer, in the "GuestBook" directory under the "App_code" folder, double-click on the "DataProvider.vb" file.
Replace all the code with:
Imports System
Imports DotNetNuke
Imports System.Data
Imports DotNetNuke.Framework
Namespace YourCompany.Modules.GuestBook
Public MustInherit Class DataProvider
' singleton reference to the instantiated object
Private Shared objProvider As DataProvider = Nothing
' constructor
Shared Sub New()
CreateProvider()
End Sub
' dynamically create provider
Private Shared Sub CreateProvider()
objProvider = CType(Reflection.CreateObject("data", _
"YourCompany.Modules.GuestBook", _
""), DataProvider)
End Sub
' return the provider
Public Shared Function Instance() As DataProvider
Return objProvider
End Function
Public MustOverride Sub YourCompany_GuestBook_Insert(ByVal _
ModuleId As Integer, ByVal Name As String, _
ByVal Email As String, ByVal Message As String)
Public MustOverride Function _
YourCompany_GuestBook_GetAll(ByVal ModuleId _
As Integer) As IDataReader
Public MustOverride Sub YourCompany_GuestBook_Update(ByVal _
ID As Integer, ByVal Name As String, _
ByVal Email As String, ByVal Message _
As String, ByVal DateEntered As DateTime)
Public MustOverride Sub _
YourCompany_GuestBook_Delete(ByVal ID As Integer)
End Class
End Namespace
Switch back to the "SqlDataprovider.vb" file and notice all the error messages are gone. Now would be a good time to save (Ctrl + Shift + S).
Review
- Data Access Layer (DAL) (Done)
- Business Logic Layer (BLL)
- Presentation Layer (UI)
We are done with the Data Access Layer. We will now program the Business Logic Layer. The Business Logic Layer will get done fast because it is only two files: a simple class file to hold our data, and a "controller" file that will fill the class file with data as well as handle inserting and deleting data.
The Business Logic Layer (BLL)
To build the Business Logic Layer, we will:
- Alter "GuestBookInfo.vb"
- Alter "GuestBookController.vb"
And that's it! This is the step that is usually the hardest for beginners to understand but is really not that complicated. However, ASP.NET 2.0 does provide a major reduction of code over the ASP.NET 1.1 version.
Alter the "GuestBookInfo.vb" file
In Visual Studio, select View from the toolbar and the "Solution Explorer":
In the Solution Explorer window, expand the "GuestBook" directory under the "App_code" folder, and double-click on the "GuestBookInfo.vb" file:
Replace every single line of code in the file with this code:
Imports System
Imports System.Configuration
Imports System.Data
Namespace YourCompany.Modules.GuestBook
Public Class GuestBookInfo
Private _ModuleId As Integer
Private _ID As Integer
Private _Name As String
Private _Email As String
Private _Message As String
Private _DateEntered As DateTime
' initialization
Public Sub New()
MyBase.New()
End Sub
' <summary>
' Gets and sets the Module Id
' </summary>
Public Property ModuleId() As Integer
Get
Return _ModuleId
End Get
Set(ByVal value As Integer)
_ModuleId = value
End Set
End Property
' <summary>
' Gets and sets the Item ID
' </summary>
Public Property ID() As Integer
Get
Return _ID
End Get
Set(ByVal value As Integer)
_ID = value
End Set
End Property
' <summary>
' gets and sets the Name
' </summary>
Public Property Name() As String
Get
Return _Name
End Get
Set(ByVal value As String)
_Name = value
End Set
End Property
' <summary>
' Gets and sets the Email
' </summary>
Public Property Email() As String
Get
Return _Email
End Get
Set(ByVal value As String)
_Email = value
End Set
End Property
' <summary>
' Gets and sets the Message
' </summary>
Public Property Message() As String
Get
Return _Message
End Get
Set(ByVal value As String)
_Message = value
End Set
End Property
' <summary>
' Gets and sets the DateEntered
' </summary>
Public Property DateEntered() As DateTime
Get
Return _DateEntered
End Get
Set(ByVal value As DateTime)
_DateEntered = value
End Set
End Property
End Class
End Namespace
Alter the "GuestBookController.vb" file
In Visual Studio, select View from the toolbar and the "Solution Explorer".
In the Solution Explorer window, expand the "GuestBook" directory under the "App_code" folder, and double-click on the "GuestBookController.vb" file:
Replace every single line of code in the file with this code:
Imports System
Imports System.Collections.Generic
Imports System.Configuration
Imports System.ComponentModel
Imports System.Data
Imports System.Xml
Imports System.Web
Imports DotNetNuke
Imports DotNetNuke.Common
Imports DotNetNuke.Common.Utilities
Imports DotNetNuke.Entities.Modules
Imports DotNetNuke.Services.Search
Namespace YourCompany.Modules.GuestBook
Public Class GuestBookController
<DataObjectMethod(DataObjectMethodType.Insert)> _
Public Shared Sub GuestBook_Insert(_
ByVal objTest As GuestBookInfo)
DataProvider.Instance.YourCompany_GuestBook_Insert(_
objTest.ModuleId, objTest.Name, _
objTest.Email, objTest.Message)
End Sub
<DataObjectMethod(DataObjectMethodType.Delete)> _
Public Shared Sub GuestBook_Delete(ByVal _
objTest As GuestBookInfo)
DataProvider.Instance._
YourCompany_GuestBook_Delete(objTest.ID)
End Sub
<DataObjectMethod(DataObjectMethodType.Select)> _
Public Shared Function GuestBook_GetAll(_
ByVal ModuleId As Integer) _
As List(Of GuestBookInfo)
Return CBO.FillCollection(Of GuestBookInfo)_
(DataProvider.Instance()._
YourCompany_GuestBook_GetAll(ModuleId))
End Function
<DataObjectMethod(DataObjectMethodType.Update)> _
Public Shared Sub GuestBook_Update(ByVal _
objTest As GuestBookInfo)
DataProvider.Instance._
YourCompany_GuestBook_Update(objTest.ID, _
objTest.Name, objTest.Email, _
objTest.Message, objTest.DateEntered)
End Sub
End Class
End Namespace
Review
- Data Access Layer (DAL) (Done)
- Business Logic Layer (BLL) (Done)
- Presentation Layer (UI)
We are done with the Business Logic Layer.
What did we just do?
The "GuestBookInfo.vb" was created. This is just a simple class file. It will hold the data.
The "GuestBookController.vb" was created. This class has four methods:
GuestBook_Insert
- Inserts items into the database. You pass a "
GuestBookInfo
" object to this method. The method then opens up the object and passes the individual parameters (module ID, name, email, message) to the "YourCompany_GuestBook_Insert
" method in the DataProvider.vb file.
- Inserts items into the database. You pass a "
GuestBook_Delete
- Deletes items from the database. You pass a "
GuestBookInfo
" object to this method. The method then opens up the object and passes the individual parameter (ID) to the "YourCompany_GuestBook_Delete
" method in the DataProvider.vb file.
- Deletes items from the database. You pass a "
GuestBook_GetAll
- Gets a recordset from the database. You pass a "
ModuleId
" parameter to this method. The method calls the "YourCompany_GuestBook_GetAll
" method in the DataProvider.vb file and returns a "GuestBookInfo
" object filled with data.
- Gets a recordset from the database. You pass a "
GuestBook_Update
- Updates the database. You pass a "
GuestBookInfo
" object to this method. The method then opens up the object and passes the individual parameters (ID, module ID, name, email, message, date entered) to the "YourCompany_GuestBook_Update
" method in the DataProvider.vb file.
- Updates the database. You pass a "
A Lot Of Stuff Was Deleted
You may notice that a lot of code was deleted out of the "GuestBookController.vb" file. Some of this were the optional interfaces that handle exporting data and searching. These are interfaces that you will want to read about and implement in your modules. However, they are not required, and they are left out for simplicity.
A lot of code was eliminated by using code that is exclusive to ASP.NET 2.0 and incompatible with ASP.NET 1.1. ASP.NET 2.0 really saves you a lot of work and requires less code.
The Presentation Layer (UI)
From this point on, the files that need to be altered reside in the ".../DesktopModules/GuestBook" directory.
To build the Presentation Layer, we will:
- Alter the "localization" (.resx) files.
- Alter "controls" and their "code-behind" files:
- EditGuestBook.ascx
- Settings.ascx
- ViewGuestBook.ascx
And that's it! The module will then be complete.
Alter the Localization Files
Localization allows you to create labels that can have their text changed (for example, to change text from English to Spanish) by simply changing a "resource" file that has a ".resx" extension.
Double click on "EditGuestBook.ascx.resx" to open it.
Change the content so it matches the picture below. The order does not matter. Save and close the file when done.
Double click on "Settings.ascx.resx" to open it.
Change the content so it matches the picture below. The order does not matter. Save and close the file when done.
Double click on "ViewGuestBook.ascx.resx" to open it.
Change the content so it matches the picture below. The order does not matter. Save and close the file when done.
Alter the Controls
The module consists of three controls (and their "code-behind" files):
- EditGuestBook.ascx
- EditGuestBook.ascx.vb
- Settings.ascx
- Settings.ascx.vb
- ViewGuestBook.ascx
- ViewGuestBook.ascx.vb
(A screen shot from the module definition that was created in the DotNetNuke site in the early step of the tutorial.)
Right-click on "EditGuestBook.ascx" and select "View Markup":
Replace all the code with this code (save and close the file when done):
<%@ Control language="VB"
Inherits="YourCompany.Modules.GuestBook.EditGuestBook"
CodeFile="EditGuestBook.ascx.vb"
AutoEventWireup="true"%>
<%@ Register TagPrefix="dnn" TagName="Label"
Src="~/controls/LabelControl.ascx" %>
<dnn:label id="lblContent" runat="server"
controlname="lblContent" suffix=":">
</dnn:label>
<asp:ObjectDataSource ID="ObjectDataSource_Tasks"
runat="server"
DataObjectTypeName="YourCompany.Modules.
GuestBook.GuestBookInfo"
DeleteMethod="GuestBook_Delete"
InsertMethod="GuestBook_Insert"
OldValuesParameterFormatString="original_{0}"
OnInit="Page_Load" SelectMethod="GuestBook_GetAll"
TypeName="YourCompany.Modules.GuestBook.GuestBookController"
UpdateMethod="GuestBook_Update">
<SelectParameters>
<asp:Parameter DefaultValue="00" Name="ModuleId" Type="Int32" />
</SelectParameters>
</asp:ObjectDataSource>
<asp:GridView ID="GridView1" runat="server"
AllowPaging="True" AutoGenerateColumns="False"
DataSourceID="ObjectDataSource_Tasks"
DataKeyNames="ID">
<Columns>
<asp:CommandField ShowDeleteButton="True"
ShowEditButton="True" />
<asp:BoundField DataField="ID"
HeaderText="ID" Visible="False" />
<asp:BoundField DataField="ModuleID"
HeaderText="ModuleID" Visible="False" />
<asp:BoundField DataField="Name"
HeaderText="Name" SortExpression="Name" />
<asp:BoundField DataField="Message"
HeaderText="Message" SortExpression="Message" />
<asp:BoundField DataField="Email" HeaderText="Email" />
<asp:BoundField ApplyFormatInEditMode="True"
DataField="DateEntered" DataFormatString="{0:d}"
HeaderText="Date" HtmlEncode="False"
SortExpression="DateEntered" />
</Columns>
</asp:GridView>
Right-click on "Settings.ascx" and select "View Markup":
Replace all the code with this code (save and close the file when done):
<%@ Control Language="VB" AutoEventWireup="false"
CodeFile="Settings.ascx.vb"
Inherits="YourCompany.Modules.GuestBook.Settings" %>
<%@ Register TagPrefix="dnn" TagName="Label"
Src="~/controls/LabelControl.ascx" %>
<dnn:label id="lblshowform" runat="server"
controlname="txtshowform" suffix=":">
</dnn:label>
<br />
<asp:DropDownList ID="DropDownList1" runat="server"
AutoPostBack="True"
nSelectedIndexChanged="DropDownList1_SelectedIndexChanged">
<asp:ListItem Selected="True">Yes</asp:ListItem>
<asp:ListItem>No</asp:ListItem>
</asp:DropDownList>
Right-click on "ViewGuestBook.ascx" and select "View Markup":
Replace all the code with this code (save and close the file when done):
<%@ Control Language="VB"
Inherits="YourCompany.Modules.GuestBook.ViewGuestBook"
CodeFile="ViewGuestBook.ascx.vb"
AutoEventWireup="true" %>
<%@ Register TagPrefix="dnn" TagName="Label"
Src="~/controls/LabelControl.ascx" %>
<asp:ObjectDataSource ID="ObjectDataSource_Tasks"
runat="server"
DataObjectTypeName="YourCompany.Modules.
GuestBook.GuestBookInfo"
DeleteMethod="GuestBook_Delete"
InsertMethod="GuestBook_Insert"
OldValuesParameterFormatString="original_{0}"
SelectMethod="GuestBook_GetAll"
TypeName="YourCompany.Modules.
GuestBook.GuestBookController"
UpdateMethod="GuestBook_Update"
OnInit="Page_Load">
<SelectParameters>
<asp:Parameter DefaultValue="00"
Name="ModuleId" Type="Int32" />
</SelectParameters>
</asp:ObjectDataSource>
<asp:GridView ID="GridView1" runat="server"
DataSourceID="ObjectDataSource_Tasks"
AutoGenerateColumns="False"
AllowPaging="True" HorizontalAlign="Center">
<Columns>
<asp:BoundField DataField="Name"
HeaderText="Name" SortExpression="Name" />
<asp:BoundField DataField="Message"
HeaderText="Message"
SortExpression="Message" />
<asp:BoundField ApplyFormatInEditMode="True"
DataField="DateEntered" DataFormatString="{0:d}"
HeaderText="Date"
SortExpression="DateEntered"
HtmlEncode="False" />
</Columns>
<EmptyDataTemplate>
There are no entries.
</EmptyDataTemplate>
</asp:GridView>
<br />
<center>
<dnn:Label ID="lblAddMessage" runat="server"
ControlName="lblAddMessage" Suffix=":">
</dnn:Label>
</center>
<br />
<asp:FormView ID="FormView1" runat="server"
DataSourceID="ObjectDataSource_Tasks"
DefaultMode="Insert" HorizontalAlign="Center">
<InsertItemTemplate>
<table cellpadding="2" cellspacing="5"
style="width: 50%" align="center">
<tr>
<td align="right" style="width: 4px">
<asp:Label ID="Label1"
runat="server" Text="Name"></asp:Label></td>
<td style="width: 100px">
<asp:TextBox ID="NameTextBox" runat="server"
Text='<%# Bind("Name") %>'
Width="264px"></asp:TextBox></td>
</tr>
<tr>
<td align="right" style="width: 4px; height: 23px">
<asp:Label ID="Label3" runat="server"
Text="Email"></asp:Label></td>
<td style="width: 100px; height: 23px">
<asp:TextBox ID="EmailTextBox" runat="server"
Text='<%# Bind("Email") %>'
Width="264px"></asp:TextBox></td>
</tr>
<tr>
<td align="right" style="width: 4px; height: 21px">
<asp:Label ID="Label2" runat="server"
Text="Message"></asp:Label></td>
<td style="width: 100px; height: 21px">
<asp:TextBox ID="MessageTextBox"
runat="server" EnableViewState="False"
MaxLength="250" Rows="2"
Text='<%# Bind("Message") %>'
TextMode="MultiLine"
Width="264px"></asp:TextBox></td>
</tr>
<tr>
<td align="right" colspan="2" style="height: 21px">
<asp:Button ID="InsertButton" runat="server"
Text="Submit" CommandName="Insert" /></td>
</tr>
</table>
<br />
</InsertItemTemplate>
</asp:FormView>
Right-click on "EditGuestBook.ascx" and select "View Code":
Replace all the code with this code (save and close the file when done.):
Imports DotNetNuke
Imports System.Web.UI
Imports System.Collections.Generic
Imports System.Reflection
Imports DotNetNuke.Entities.Modules
Namespace YourCompany.Modules.GuestBook
Partial Class EditGuestBook Inherits PortalModuleBase
Protected Sub Page_Load(ByVal sender As Object, _
ByVal e As System.EventArgs)
Try
Catch exc As Exception
Exceptions.ProcessModuleLoadException(Me, exc)
End Try
End Sub
Protected Sub SetModuleId(ByVal sender As Object, _
ByVal e As System.Web.UI.WebControls._
ObjectDataSourceSelectingEventArgs) _
Handles ObjectDataSource_Tasks.Selecting
e.InputParameters("ModuleId") = ModuleId.ToString
End Sub
End Class
End Namespace
Right-click on "Settings.ascx" and select "View Code":
Replace all the code with this code (save and close the file when done):
Imports System
Imports System.Web.UI
Imports DotNetNuke
Imports DotNetNuke.Entities.Modules
Imports DotNetNuke.Services.Exceptions
Namespace YourCompany.Modules.GuestBook
Partial Class Settings Inherits ModuleSettingsBase
Public Overrides Sub LoadSettings()
Try
If (Page.IsPostBack = False) Then
If (Not (CType(TabModuleSettings(_
"showform"), _
String)) Is Nothing) Then
Me.DropDownList1.SelectedValue_
= CType(TabModuleSettings(_
"showform"), String)
End If
End If
Catch exc As Exception
Exceptions.ProcessModuleLoadException(Me, exc)
End Try
End Sub
Protected Sub DropDownList1_SelectedIndexChanged(ByVal_
sender As Object, ByVal e As EventArgs)
Dim objModules As ModuleController = _
New ModuleController
If (Me.DropDownList1.SelectedValue = "Yes") Then
objModules.UpdateTabModuleSetting(_
TabModuleId, "showform", "Yes")
Else
objModules.UpdateTabModuleSetting(_
TabModuleId, "showform", "No")
End If
End Sub
End Class
End Namespace
Right-click on "ViewGuestBook.ascx" and select "View Code":
Replace all the code with this code (save and close the file when done):
Imports DotNetNuke
Imports System.Web.UI
Imports System.Collections.Generic
Imports System.Reflection
Imports DotNetNuke.Entities.Modules
Namespace YourCompany.Modules.GuestBook
Partial Class ViewGuestBook
Inherits Entities.Modules.PortalModuleBase
Implements Entities.Modules.IActionable
Public ReadOnly Property ModuleActions() _
As Entities.Modules.Actions._
ModuleActionCollection Implements _
Entities.Modules.IActionable.ModuleActions
Get
Dim Actions As New Entities.Modules_
.Actions.ModuleActionCollection
Actions.Add(GetNextActionID, _
Localization.GetString(_
Entities.Modules.Actions._
ModuleActionType.EditContent, _
LocalResourceFile), _
Entities.Modules.Actions._
ModuleActionType.EditContent, _
"", "", EditUrl(), False, _
Security.SecurityAccessLevel.Edit,_
True, False)
Return Actions
End Get
End Property
Protected Sub Page_Load(ByVal sender _
As Object, ByVal e As System.EventArgs)
Try
Dim objModules As ModuleController_
= New ModuleController
If Not Page.IsPostBack Then
If (Not (CType(Settings("showform"), _
String)) Is Nothing) Then
If (CType(Settings("showform"), _
String) = "No") Then
' Do not allow messages to be added
FormView1.Visible = False
lblAddMessage.Visible = False
End If
End If
Else
Me.GridView1.DataBind()
End If
Catch ex As Exception
Exceptions.ProcessModuleLoadException(Me, ex)
End Try
End Sub
Protected Sub NewItem(ByVal sender As Object, _
ByVal e As System.Web.UI.WebControls._
FormViewInsertEventArgs) _
Handles FormView1.ItemInserting
e.Values.Item("ID") = 0
e.Values.Item("ModuleId") = ModuleId.ToString()
e.Values.Item("DateEntered") = _
DateTime.Now.ToShortDateString
End Sub
Protected Sub SetModuleID(ByVal sender As _
Object, ByVal e As System.Web.UI._
WebControls.ObjectDataSourceSelectingEventArgs) _
Handles ObjectDataSource_Tasks.Selecting
e.InputParameters("ModuleId") = ModuleId.ToString
End Sub
End Class
End Namespace
Review
- Data Access Layer (DAL) (Done)
- Business Logic Layer (BLL) (Done)
- Presentation Layer (UI) (Done)
Build The Site
From the toolbar, select Build, then "Build Web Site":
The site should build with no errors:
From the toolbar, select Debug, then Start Without Debugging:
The website will come up. Click "Guest Book" on the menu bar:
The module will now show: