Oracle Forms Interactive Workbook
Oracle Forms Interactive Workbook
Oracle Forms Interactive Workbook
INTERACTIVE WORKBOOK
Oracle Forms
INTERACTIVE WORKBOOK
BAMAN MOTIVALA
Prentice Hall PTR Upper Saddle River, New Jersey 07458 www.phptr.com
for maria.
INTRODUCTION
The Oracle Forms Interactive Workbook presents Oracle Forms in a unique and highly effective format. It challenges you to learn Oracle Forms by using it rather than by simply reading about it. Just as a grammar workbook would teach you about nouns and verbs by first showing you examples and then asking you to write sentences, the Oracle Forms workbook teaches you about Forms, triggers, and items by first showing you examples and then asking you to create these objects yourself.
xi
xii
Introduction
Exercise Answers with Detailed Discussion Self-Review Questions Lab... Test Your Thinking Questions Each Chapter contains interactive Labs that introduce topics about Oracle Forms. The topics are discussed briefly and then explored through Exercises, which are the heart of each Lab. Each Exercise consists of a series of steps that you will follow to perform a specific task, along with questions that are designed to help you discover important things about Forms on your own. The answers to these questions are given at the end of the Exercises, along with more in-depth discussion of the concepts explored. The Exercises are not meant to be closed-book quizzes to test your knowledge. On the contrary, they are intended to act as your guide and walk you through a task. You are encouraged to flip back and forth from the Exercise question section to the Exercise answer section so that if need be, you can read the answers and discussions as you go along. At the end of each Lab is a series of multiple-choice self-review questions. These are meant to be closed-book quizzes of sorts to test that you have absorbed the Lab material. The answers to these questions appear in Appendix A. There are also additional self-review questions at this books companion Web site, found at http://www.phptr.com/phptrinteractive/. (The companion Web site will be explained in the next section of this introduction.) Finally, at the end of each Chapter you will find a Test Your Thinking section, which consists of a series of projects designed to solidify all of the skills you have learned in the Chapter. If you have successfully completed all of the Labs in the Chapter, you should be able to tackle these projects with few problems. There are not always answers to these projects, but where appropriate, you will find guidance and/or solutions at the companion Web site.
The Chapters should be completed in sequence because the material builds on itself as you go along. Additionally, many of the files you create and save in earlier Chapters will be required in later Chapters. In the end, all of the skills you have acquired and files you have created will come together in Chapter 13, Forms Menus, where you will create a menu system to manage your completed and working application.
Introduction
xiii
All of the Exercises and questions are based on a sample database called STUDENT. The files required to create and install the STUDENT schema are downloadable from the Web site. Additionally, many of the Exercises require that you work with pre-created Forms files. For example, in Exercise 1.1.1, you will be required to open and answer questions about a file called EX01_01.fmb. This file and all the rest you will need for the workbook are downloadable from the Web site. The answers to the Test Your Thinking sections will also be found at the Web site. These answers will be textual or in the form of downloadable files. In addition to required files and Test Your Thinking answers, the Web site will have many other features like additional review questions, a message board, and periodically updated information about the book.
You should visit the companion Web site and download the required files before starting the Labs and Exercises.
SOFTWARE
Oracle Developer 6.0 Oracle8 Access to the WWW Windows 95/98 or NT 4.0
xiv
Introduction
ORACLE DEVELOPER 6.0: Oracle Developer 6.0 is Oracles application development tool suite that contains a number of different components. The Oracle Forms Interactive Workbook is concerned only with Oracle Forms. Oracle Forms Version 6.0.5.0.2 was used to create the Exercises, but subsequent versions should be compatible. Since Oracle frequently improves and changes its products, new versions are released all the time. However, the concepts covered in this book are fundamental to the Oracle Forms product and are unlikely to change significantly in the near future. So, even if your version of Oracle Forms is different than the one listed here, you should still be able to make use of this book. ORACLE8: Oracle8 is Oracles RDBMS and its flagship product. You can use either Oracle Personal Edition or Oracle Enterprise Edition. If you use Oracle Enterprise Edition, it can be running on a remote server or locally on your own machine. Oracle 8.0.5 Enterprise Edition running locally was used to create the Exercises for this book, but subsequent versions of Oracle should be compatible. Additionally, you should have access to and be familiar with SQL*Plus. WINDOWS 95/98 OR NT 4.0: The Oracle Forms development environment is available on a number of different operating system platforms, including Microsoft Windows and various flavors of UNIX. The Exercises, screenshots, and examples in this workbook were created using Microsoft Windows NT 4.0 with Service Pack 3. Therefore, it is geared more toward those working in a Windows environment. But, as mentioned before, most of the Forms concepts in this book are rather fundamental and, therefore, apply to all operating systems. So, even if you are developing on a UNIX platform, this book can still be of use to you. If you are using UNIX or another non-Windows OS, keep in mind that the screenshots will not match what you see on your screen and that Appendix B, Windows Registry, does not apply to you. ACCESS TO THE WWW: You will need access to the Internet and WWW so that you can reach the companion Web site, http://www.phptr .com/motivala Here you will find the files that are necessary for completing the Exercises. It is important that you visit this site and download the necessary files before you start working through the Chapters in this book.
Introduction KNOWLEDGE
xv
To complete the Exercises, you should be familiar with relational databases as well as Oracle database concepts. You should be comfortable using SQL to access and manipulate database objects such as tables, constraints, sequences, and so on. You should also be able to write simple PL/SQL procedures that include, among other things, local variables, conditional logic, and cursors. If you are not familiar or comfortable with these subjects, it is recommended that you refer to the other books in the Oracle series. These are listed earlier in this Introduction. Finally, you should be reasonably comfortable with accessing and configuring the Windows Registry. This will be necessary so that Oracle Forms can properly locate all of the files you create. Appendix B, Windows Registry, provides a brief description of the Registry and all of the information you will need to configure it for Oracle Forms.
You should read and complete the tasks in Appendix B before starting the Labs and Exercises.
xvi
Introduction
The schema also has a number of other tables that manage grading for each student in each section.
This icon is used to flag notes or advice from the author to you, the reader. For instance, if there is a particular topic or concept that you really need to understand for the exam, or if theres something that you need to keep in mind while working, you will find it set off from the main text like this. This icon is used to flag tips or especially helpful tricks that will save you time or trouble. For instance, if there is a shortcut for performing a particular task or a method that the author has found useful, you will find it set off from the main text like this. Computers are delicate creatures and can be easily damaged. Likewise, they can be dangerous to work on if youre not careful. This icon is used to flag information and precautions that will not only save you headaches in the long run, they may even save you or your computer from harm. This icon is used to flag passages with a reference to the books companion Web site, which once again is located at http://www .phptr.com/motivala.
ACKNOWLEDGMENTS
Many people were instrumental in helping me complete this project. Im glad to have the opportunity to acknowledge them here. Id like to thank Michael Stowe for serving as technical editor. He provided valuable insight and suggestions, and caught a number of ghastly errors. Also, thanks to Gayle Conarello, Matt Portnoy, Tom Ziek, and Mehli Motivala (my dad) for making many helpful comments after working through the Exercises in the early Chapters of the book. On the publishing side, I am indebted to Ralph Moore and Russ Hall, who deftly handled the developmental process as well as my unceasing questions. Tim Moore of Prentice Hall was also a great help in handling things on the acquisitions side. I would also like to acknowledge the other authors in the Oracle Series for making this a team effort. I am especially indebted to Douglas Scherer, first for offering me the opportunity to be one of the authors, then for acting as coordinator and general problem-solver for the whole series. I am lucky to have two caring families, one in Sweden and one in the U.S., who offered their constant support and encouragement. My mom and dad deserve very special thanks for teaching me to speak and write properly, and for always standing behind me no matter what I choose to do. Finally, Id like to acknowledge the two women in my life. Olivia, my four-legged companion, who took me for long walks when I was tired of thinking about Forms and triggers, and Maria, my fiance. One night in December of 1998, I announced to Maria that I had been asked to write a book about Oracle Forms. I had planned to follow this announcement with a dozen reasons and excuses as to why I couldnt do it. But before I could continue, she had thrown her arms around my neck and showered me with such a torrent of enthusiastic encouragement, congratulations, and questions that it was impossible to introduce any negativity. This project would not even have been started, let alone completed, without her belief in me, and her unyielding optimism.
xvii
xix
CONTENTS
Introduction Acknowledgments About the Author Chapter 1 Concepts and Objects
LAB 1.1 Oracle Forms Concepts 1.1.1 Explain How Oracle Forms Works LAB 1.2 Mandatory Forms Objects 1.2.1 Identify Items and Their Types 1.2.2 Identify Canvases and Frames 1.2.3 Define Base-table Blocks 1.2.4 Understand Modules 1.2.5 Relate the Mandatory Forms Elements CHAPTER 1 Test Your Thinking 2 4 11 17 18 19 20 20 30
xi xvii xix 1
31
32 42 43 53 54 55 57 63
65
66 68 69 70 72 72
vii
viii
Contents
LAB 3.2 The Property Palette 3.2.1 View Properties 3.2.2 Change Properties LAB 3.3 The Layout Editor 3.3.1 Create and Format Objects 3.3.2 Arrange and Size Objects CHAPTER 3 Test Your Thinking
84 86 86 93 95 97 105
107
108 110 112 124
Chapter 5 Items
LAB 5.1 Text Items and Display Items 5.1.1 Create and Define Text Items Without the Wizard 5.1.2 Create and Define Display Items LAB 5.2 Buttons, List Items, Radio Groups, and Check Boxes 5.2.1 Create Buttons 5.2.2 Put Simple Code Behind Buttons 5.2.3 Create List Items 5.2.4 Create Radio Groups 5.2.5 Create Check Boxes CHAPTER 5 Test Your Thinking
125
126 127 131 143 147 148 150 153 155 169
171
172 175 176 178 187 189 192 195 197 213 216 223
Contents
ix 225
226 228 233 248 251 259
261
262 267 269 277 278 287 289 298 299 302 311
313
314 315 323 325 327 336 338 341 348 349 352
353
354 357
Contents
LAB 10.2 PL/SQL Libraries 10.2.1 Create and Attach PL/SQL Libraries 10.2.2 Use Indirect References in Library Code LAB 10.3 Stored PL/SQL Objects 10.3.1 Use Stored PL/SQL Objects CHAPTER 10 Test Your Thinking
381
382 385 387 400
403
404 406 407 414 416 420
421
422 426 435 436 443
C H A P T E R
efore you can begin to create applications, you must understand the basics of how Oracle Forms thinks and the nature of its elements. By learning the fundamentals of Forms behavior, you will understand how a form reacts to input and instructions from your users. You will be introduced to the most common objects in a Forms application, and you will learn how to create and configure these objects. Despite their introductory nature, all of the Exercises in this Chapter have hands-on components so you will learn the basics by getting your hands dirty.
1
go to contents
2 LAB 1.1
LAB
1.1
Oracle Forms belongs to a larger product called Oracle Developer, which has close to 20 individual components. Oracle Forms, Oracle Reports, and Oracle Graphics are the core components of the development environment. Supporting this environment are sub-components and utilities, including a project manager, debugger, database schema builder, and many others. The primary focus of this interactive workbook is Oracle Forms. The purpose of this Lab is to get you acquainted with how Oracle Forms applications (forms) work.
go to contents
3 LAB 1.1
sponds to a system action. Clicking a button, tabbing from one item to another, and opening or closing a window are typical examples of interface events. Validating an item is an example of an internal processing event. It is one of the events that occurs after a user has changed the value of an item (either by entering a new value or changing one queried from the database) and then left that item either by tabbing or navigating with the mouse. The Validate Item event tells Forms to make sure that the value in the item conforms to whatever validation rules have been defined. Events are important because they drive Forms applications. They are also important because they give you, the programmer, a tremendous amount of control over an application because for every event that occurs, you have the opportunity to write code to respond to it. The code objects that respond to events are called triggers. A trigger fires, or numerous triggers fire, whenever an event occurs. For example, if a user wants to close an application window on a Windows platform, he will click the Close button at the upper right-hand corner of the window. This is an event. In response to this event, Forms fires the WHENWINDOW-CLOSED trigger. You, the programmer, have written code inside this trigger to tell the application what to do. You could have the entire application close along with the window, or you could flash a message reminding the user to save his work if he intends to quit. Basically, you can have the application do anything in response to an event.
ITEMS
The Forms interface is made up of items. Buttons, text fields (called text items or display items in Forms), check boxes, and radio groups are typical examples of items. Items are used to present information from the database (base-table items) or to act as controls (non-base-table items). In other Forms books, you may see base-table items referred to as data items and non-base-table items referred to as control items. Most item types, like display items, are flexible so that they can be used as base-table items or non-base-table items.
I FOR EXAMPLE:
You may create a display item to present information pulled directly from the database, such as a students name or address. This would be considered a base-table item because the display item is based on a column in the database. Or, you may create a display item to present the number of students enrolled in a certain section. This would be consid-
go to contents
4 LAB 1.1
I FOR EXAMPLE:
You have created a base-table item in a form called ZIP that is based on the zip column in the STUDENT table of the STUDENT schema. There is also a ZIPCODE table in the schema that stores all of the valid Zip Codes. When the user enters or changes values in the ZIP item in the form, you want to validate the value she has entered by checking that it exists in the ZIPCODE table. The following would occur: 1. 2. 3. 4. 5. The user changes the ZIP items value from 10011 to 07652. The user presses the TAB key, which is an interface event. The interface event causes a number of internal processing events to occur. One of them is the Validate Item event. The Validate Item event fires the WHEN-VALIDATE-ITEM trigger. The code in the WHEN-VALIDATE-ITEM trigger validates the value in ZIP.
It can be said, with only a hint of drama, that this series of occurrences represents the essence of a Forms application. In the Exercises that follow, you will explore the Form Builder to discover more about items, events, and triggers.
Open the Form Builder. You will see a dialog titled Welcome to the Form Builder. Ignore this dialog and simply click the Cancel button. The earlier section titled About the Companion Web Site explained how to download required files from the companion Web site at http://www.phptr.com/phptrinteractive. The sample database and some Oracle Forms files available at the site are
go to contents
Lab 1.1: Oracle Forms Concepts required to complete the Exercises in this Chapter and in almost all of the Chapters that follow. Please read the entire Introduction now and then visit the Web site to learn more about what you must download.
5 LAB 1.1
Also note that in many of the Exercises, reference is made to the audit columns. These columns are in each table in the STUDENT schema and are named CREATED_BY, CREATED_DATE, MODIFIED_BY, and MODIFIED_DATE. If you are unfamiliar with the STUDENT schema, please read the Introduction before continuing. You will not be able to insert new records into the tables in the STUDENT schema until after you have completed Chapter 6, Triggers & Built-ins. This is not a problem because you will not be required to insert records before then. From the Main Menu, select File | Open and open the file named EX01_01.fmb, which should be in the \guest\forms\exercises directory on your local machine. In the upper left-hand corner, you will see a window titled Object Navigator. Here, you will find a hierarchical tree that lists all of the objects in a particular form. The first node on the list is Forms, under which you will see that there is a Forms object named EX01_01. Looking down the list, you will see nodes called Triggers, Alerts, Attached Libraries, and so on. Continue looking down the list until you come to the node named Canvases. Click the small plus sign to the left of the word Canvases to expand this node. The Object Navigator behaves like the Windows Explorer with regard to expanding and collapsing nodes. Once the Canvases node is expanded, you will see a small, colorful icon with the word STUDENT next to it. Double-click this canvas icon. You have just opened the Layout Editor and are getting a WYSIWYG view of a form. Refer to Figure 1.1 to see how the Form Builder should look after taking these steps.
go to contents
6 LAB 1.1
Figure 1.1 I The Form Builder with the Object Navigators Canvases node expanded and the Layout Editor open. a) What type of Forms object is the button labeled EXIT? In Forms terminology, what is the result when a user clicks this button?
b) If the EX01_01.fmb form were running, describe what would happen behind the scenes in Forms if a user clicked the EXIT button?
c) By expanding objects in the Object Navigator, try to locate the code associated with this button. What is the code called? Does this particular piece of code have a name? What language is it written in?
go to contents
7 LAB 1.1
On the bottom right-hand side of the screen, there is a check box labeled Enroll Now. Right now there is no trigger associated with this item. So, if a user tries to change the value of the check box, no code will respond. d) Based on the information provided in this Exercise, try to guess the name of the trigger you would use if a user were to change the value in the check box. If you feel up to it, explore the Form Builder and try to find the correct name to see if you are right.
a) What type of Forms object is the button labeled EXIT? In Forms terminology, what is the result when a user clicks this button? Answer: The button is an item. The result of clicking this button is an interface event. All Forms applications are driven by responses to interface events and internal processing events. When a user clicks something, this results in an interface event. When a user moves the cursor from one item to another, this results in an event. When a user inserts, updates, deletes, or queries records from the database, these are also events. b) If the EX01_01.fmb form were running, describe what would happen behind the scenes in Forms if a user clicked the EXIT button? Answer: A Forms trigger would fire in response to this event. If the trigger code were written properly, the application would exit. c) By expanding objects in the Object Navigator, try to locate the code associated with this button. What is the code called? Does this particular piece of code have a name? What language is it written in? Answer: The code is called a trigger. This particular trigger is called WHEN-BUTTONPRESSED. The WHEN-BUTTON-PRESSED trigger is written in PL/SQL as are all triggers, procedures, and functions in Oracle Forms. If you were unable to locate the WHEN-BUTTON-PRESSED trigger in the Object Navigator, take the following steps and then look at Figure 1.2:
go to contents
8 LAB 1.1
Figure 1.2 I The Object Navigator with nodes expanded to show the EXIT buttons WHEN-BUTTON-PRESSED trigger.
1) Look in the Object Navigator for the Data Blocks node and expand it. Remember, to expand a node, you simply click the small plus sign next to the node name. Expand the data block named STUDENT. Expand the Item named EXIT. Expand the Triggers node.
2) 3) 4)
For almost every conceivable interface event and system event, there exists a corresponding Forms trigger. For the user event of clicking a button, you have already seen the WHEN-BUTTON-PRESSED trigger. You will learn about other user events and their corresponding triggers in Chapter 6, Triggers & Built-ins. As a Forms programmer, you put your own code inside a trigger. This gives you the power to have Forms respond to any event in any way that you would like.
go to contents
9 LAB 1.1
d) Based on the information provided in this Exercise, try to guess the name of the trigger you would use if a user were to change the value in the check box. If you feel up to it, explore the Form Builder and try to find the correct name to see if you are right. Answer: The name of the trigger is WHEN-CHECKBOX-CHANGED. Until you become very comfortable with Forms, you will find yourself wanting to respond to an event, but not knowing the name of the corresponding trigger. In Exercise 1.1.1.d, you knew you needed the application to respond to the changing of a check box, but you had to guess what the trigger name might be. Of course there is a list of all available triggers, but it is rather long. Being able to make educated guesses will help you sift through the list to find the trigger you need more quickly. You will delve into events and their corresponding triggers in Chapter 6, Triggers & Built-ins, but for now, here are three quick ways to find the name of the trigger you need in the Form Builder: 1) In the Object Navigator, right-click the ENROLL _NOW item and select Smart Triggers. This will give you a short list of trigger suggestions. In the Object Navigator, expand the ENROLL_NOW item to reveal the Triggers node. Double-click on the Triggers node. A window titled Triggers will open with a list of triggers that you can scroll through. Select Help | Form Builder Help Topics from the Main Menu. Select the Index tab and search for Triggers, alphabetical list of.
2)
3)
Forms triggers are quite different from Oracle database triggers. Forms triggers are stored inside an Oracle Forms application, while Oracle database triggers are stored in the database. Forms triggers fire because a user event or system event has occurred in a Forms application. Oracle database triggers fire when there is an attempt to insert, update, or delete data from an Oracle database table.
go to contents
10 LAB 1.1
go to contents
11
LAB
1.2
LAB 1.2
Identify Items and Their Types Identify Canvases and Frames Define Base-table Blocks Understand Modules Relate the Mandatory Forms Elements
In this Lab, you will be introduced to the five mandatory Forms objects: items, canvases, windows, blocks, and modules. Items, canvases, and windows are physical interface objects, while blocks and modules are logical container objects. In the text and Exercises, you will focus on understanding the individual roles of these objects and how they relate to each other.
MORE ON ITEMS
Items are the interface objects (buttons, text items) that allow Forms users to interact with Forms applications. Interface object is a fancy term for an object whose purpose is rather simple. Put plainly, items allow applications and users to communicate. An application can communicate with a user by presenting database data in a text item. A user can communicate with an application by clicking a button item. Items are defined by their properties. Properties include physical attributes such as Height, Width, X Position and Y Position, and so on; ex-
go to contents
12
LAB 1.2
There are 15 different item types in Forms. You will explore six of the most important types in the Exercises.
CANVASES
For items to be visible to users, they must be positioned on canvases. Similar to a painters canvas, a Forms canvas is the surface on which you po-
go to contents
13
sition, size, and color different objects. The parallel between programmer and painter should end there, for, as a programmer, it is wise to rein in your creative energy with regard to object positioning and color schemes. A user should be able to work with an application without being distracted by bright colors or acrobatic widgets. The best-designed user interfaces are those that the user hardly notices. Layout, positioning, coloring, and so on are done in the Layout Editor the tool in the Form Builder that gives you a WYSIWYG view of the canvas and its items. Canvases, like items, have properties that can be viewed and manipulated in the Property Palette at design-time or programmatically at run-time. Graphical objects called frames are contained in canvases, and although frames are not mandatory Forms objects, they are worth mentioning here because they can be rather helpful in controlling the positioning of a group of items. You can put a group of items in a frame and then set properties for that frame which affect the entire group. Figure 1.3 shows a typical canvas in the Layout Editor with two frames and items of varying types.
LAB 1.2
Figure 1.3 I The Layout Editor with a view of the COURSE canvas, COURSE and SECTION frames, and multiple items. go to contents
14
WINDOWS
Windows are the physical containers of canvases. This also makes them the ultimate physical containers of all the visual objects on canvases, such as items and graphics. As with other objects in Forms, windows have properties that can be changed at design-time and run-time. The windows you create for your Forms applications are similar to the windows you have seen in typical Microsoft Windows applications. They have titles, icons, and sizable borders. They can be opened and closed manually by the user or programmatically by the application.
LAB 1.2
I FOR EXAMPLE:
The form in Figure 1.4 shows the layout of items from the INSTRUCTOR base-table block, which is based on the INSTRUCTOR table. The INSTRUCTOR block contains items based on most of the columns in the INSTRUCTOR table, like INSTRUCTOR_ID, SALUTATION, FIRST_NAME, and so on. It also contains CITY and STATE, which are non-base-table items. There are no columns in the INSTRUCTOR table for city and state values. These are retrieved from the ZIPCODE table using trigger code. If you were looking at the Object Navigator, you would see that CITY and STATE are included in the INSTRUCTOR block. They were put there because of their function, which is to provide more information about the INSTRUCTOR record.
go to contents
15
LAB 1.2
Figure 1.4 I Items in the INSTRUCTOR block on a canvas. CITY and STATE are non-base-table display items in the INSTRUCTOR block.
Typically, base-table blocks are based on database tables or views. It is possible, however, to base a block on an Oracle stored procedure. This is an advanced topic that will not be covered in this interactive workbook. For more information, refer to the on-line help in the Form Builders help topics. Non-base-table blocks are not based on any database object, nor are any of their items. They typically contain non-base-table items such as buttons. Sometimes they contain display items that show non-database information, such as the time or perhaps the users name. All blocks, whether they are base-table or non-base-table blocks, are regarded as logical because they do not have physical properties like X Position, Y Position, Height, Width, and so on. They serve a purely functional purpose in that they allow you to group items without regard to physical location. Therefore, you can easily perform programmatic operations on a block no matter where the items in the block are physically positioned.
I FOR EXAMPLE:
Figure 1.5 shows a screenshot of a running form. You can see that there are two windows, one called Student and one called Record History. By clicking the Record History button, the user has the option to view the records history information. This form has one block that is based on the STUDENT table and includes all of its columns as items. Some of the
go to contents
16
LAB 1.2
Figure 1.5 I Items can be logically contained by one block, but physically positioned across multiple canvases and windows.
items are on a canvas in the Student window, while the rest of the items are on a canvas in the Record History window. Forms will still automatically coordinate and manage the querying and record status of the block even though its items are not positioned together physically.
MODULES
Although forms module is the proper term, these objects are commonly referred to simply as forms. These terms will be used interchangeably throughout this workbook. Modules are logical containers of all the objects in a form, which means they serve as the highest-level object in the hierarchy of a single form. In Figure 1.5, what you can see are the physical objects that are contained within a forms module called COURSE.fmb. A typical application is made up of a group of modules. Depending on the complexity of the application, there could be tens or hundreds of modules.
go to contents
17
LAB 1.2
Open the Form Builder. If you have any forms or modules open, close them by going to the Main Menu and selecting File | Close. Now, open the form file titled EX01_02.fmb, which should be in your \guest\forms\exercises directory. If you cannot find it, read the information on how to download this file and other files from the companion Web site. a) Locate the items in EX01_02.fmb in the Object Navigator. What purpose do items serve in a Forms application?
From the Main Menu, select Tools | Layout Editor. The Layout Editor will open, and you will see the layout for a canvas named COURSE_SECTION. b) There are six different types of items on the COURSE_SECTION canvas. Identify one item for each of these six types. Can you briefly describe each type (e.g., COURSE_NO is a text item. Text items are used . . . )?
c) Which items have properties associated with them? Which items in the COURSE block have triggers associated with them, and what are the names of these triggers?
go to contents
18
Lab 1.2: Mandatory Forms Objects d) What are two ways a Forms programmer can manipulate and control the look, feel, and behavior of items?
c) What is the value of the SECTION frames Layout Data Block property?
d) Are there any triggers associated with this canvas? How do you know?
go to contents
19
1.2.3
As in the previous two Exercises, we will continue working in the Form Builder with EX01_02.fmb. a) What are blocks and what purposes do they serve?
LAB 1.2
b) Name the blocks in EX01_02.fmb. Are they non-base-table blocks or base-table blocks?
c) Explore the Object Navigator to determine if these blocks are related. What is the name of the relation?
d) Can a Forms user see or interact with a block? Why or why not?
e) Why does Forms need blocks at all? Why cant items simply be grouped by canvas?
f) Which block property would you change if you wanted to control the order of the records returned?
go to contents
20 1.2.4
UNDERSTAND MODULES
Once again, you will complete this Exercise in the Form Builder using EX01_02.fmb.
LAB 1.2
a) Make a list of the objects that are owned or contained by a form. Are there any instances of these objects in EX01_02.fmb? What are they called (e.g., Alerts - DEMO_OBJECTS; Triggers - ON-)?
From the Main Menu, select File | New | Form and observe what happens. b) What did Forms name the new module? Did the Form Builder create any default objects for this form? If so, what kinds of objects are they and what are they called?
1.2.5
go to contents
21
LAB 1.2
e) Which objects can have triggers associated with them in the Object Navigator?
f) Of the objects youve learned so far, which are visible to the Forms user?
a) Locate the items in EX01_02.fmb in the Object Navigator. What purpose do items serve in a Forms application? Answer: Items are the interface objects that allow a Forms user to interact with a Forms application. To locate the items, take the following steps in the Object Navigator:
go to contents
22
LAB 1.2
3.
b) There are six different types of items on the COURSE_SECTION canvas. Identify one item for each of these six types. Can you briefly describe each type? Answer: COURSE_NO is a text item. Text items are used to display database data or derived values in characters, numbers, or a mixture of both. Users can navigate to a text item and edit its value. COST is a radio group. Radio groups are used to present a series of mutually exclusive choices to the user. One value is always selected. SECTION_ID is a display item. Display items are used to display database data or derived values in characters, numbers, or a mixture of both. Users cannot navigate to a display item and edit its value. LOCATION is a list item. List items are used to display a drop-down list of choices to the user. AUTHORIZATION is a check box. Check boxes have two states, checked and unchecked. Very often they are used when the value must be either Yes/No, On/Off, Accept/Decline, etc. LIST is a push button. Push buttons are used for many different things, but usually they initiate some kind of action like a query, save, exit, show list, and so forth. Push buttons are usually referred to simply as buttons. As described in the Lab, there are three ways to determine an items type: you can look in the Property Palette, the Object Navigator, or the Layout Editor. The Property Palette will give you the surest answer because it shows the actual name of the item type, while the Object Navigator and Layout Editor show graphical representations of the items type.
As you may have already realized, the Object Navigator, Layout Editor, and Property Palette are synchronized. So, if you select an item, or any object for that matter, in the Object Navigator, its property list will appear in the Property Palette and it will be selected in the Layout Editor. By the same token, if you select any object in the Layout Editor, its properties will also appear.
There are other item types in Oracle Forms, but the six identified are the most common. There are some rules of thumb for choosing item types,
go to contents
23
but in general, item types and their uses are pretty straightforward. In Chapter 5, Items, you will learn how to create, edit, and manipulate different items. You will also learn when it may be appropriate to use one item type over another. c) Which items have properties associated with them? Which items in the COURSE block have triggers associated with them, and what are the names of these triggers? Answer: All items have properties associated with them. AUTHORIZATION has a WHEN-CHECKBOX-CHANGED. Not only do all items have properties, but all other types of objects created in Forms have properties also. d) What are two ways a Forms programmer can manipulate and control the look, feel, and behavior of items? Answer: A programmer can manipulate and control items at design-time by manually editing them in the Layout Editor or by adjusting their properties in the Property Palette. Items can also be manipulated at run-time by changing their properties programmatically with triggers or procedures. Using the Property Palette and Layout Editor at design-time are obvious and rather easy ways to manipulate non-base-table items. You will learn more about how to use these tools in Chapter 3, The Development Environment. Controlling items programmatically at run-time is a bit more complicated, but also extremely powerful. For example, you may want to have the application change the color of an item depending on its value. Lets say that you want to color the SECTION_ID red for all classes that are overbooked, meaning the enrollment is greater than the capacity. You would do this programmatically by writing a trigger that uses Forms built-ins to check enrollment and capacity, then change or SET the color of the SECTION_ID for the overbooked courses. You will learn how to use triggers and built-ins to change properties at run-time in Chapter 6, Triggers & Built-ins.
LAB 1.2
1.2.2
ANSWERS
a) Identify the name and canvas type of each canvas in form EX01_02.fmb. What purpose do canvases serve in a Forms application? Answer: The canvas name is COURSE_SECTION. It has a canvas type of content. Canvases are the physical surfaces upon which items are positioned, sized, and colored. By now you should be accustomed to looking in the Object Navigator and Property Palette for answers. Both would have told you that the canvas name is COURSE_SECTION.
go to contents
24
LAB 1.2
You will see that, after dragging the frame, the PREREQUISITE item was automatically repositioned back inside it. There will be cases in which you will want to intentionally drag items outside their frames or otherwise manually position items yourself. To prevent a frame from automatically updating the layout, set the frames Update Layout property to Locked. d) Are there any triggers associated with this canvas? How do you know? Answer: No, there are no triggers associated with this canvas. In the Object Navigator, there is no Trigger node under Canvases.
go to contents
25
Canvases never have triggers directly associated with them. The only objects that can have triggers associated with them are items, blocks, and modules. Item, block, or module triggers can affect canvases, but triggers cannot belong to, be contained by, or be owned by canvases.
1.2.3
ANSWERS
Answer: Blocks are logical containers of like items. Blocks fall into one of two types, base-table or non-base-table. A base-table block is based on a table and its items are based on the columns in that table. In a non-base-table block, the items are not based on any objects in the database.
LAB 1.2
b) Name the blocks in EX01_02.fmb. Are they non-base-table blocks or basetable blocks? Answer: There are two blocks named COURSE and SECTION. They are both basetable blocks. c) Explore the Object Navigator to determine if these blocks are related. What is the name of the relation? Answer: Yes, they are related. The relation is named COURSE_SECTION. The relation object is under the COURSE blocks Relations node. Relations are similar to joins in SQL queries in that they allow you to join or create a relationship between two blocks. In Forms, this is called a master-detail relation. In the form EX01_02.fmb, the relation COURSE_ SECTION has COURSE as the master block and SECTION as the detail block. For each COURSE record returned, Forms will return and display the corresponding SECTION records. Master-detail forms are very common, and you will learn more about them in Chapter 4, Master-Detail Forms. d) Can a Forms user see or interact with a block? Why or why not? Answer: No, a user cannot see or interact with a block. Blocks are logical containers, not physical ones, so they are not displayed to the user. Examples of physical objects in Oracle Forms are items, canvases, windows, lists of values, and alerts. These objects are physical because they have physical properties such as Height and Width, which make them visible to the user. Blocks do not have any physical properties to make them visible. Blocks do not need physical properties because their purpose is functional and perhaps sometimes organizational.
go to contents
26
LAB 1.2
f)
Which block property would you change if you wanted to control the order of the records returned? Answer: You would change the Order By property.
1.2.4
ANSWERS
a) Make a list of the objects that are owned or contained by a form. Are there any instances of these objects in EX01_02.fmb? What are they called? Answer: The list is displayed in the following table:
Object
Triggers Alerts Attached Libraries Data Blocks Canvases Editors LOVS Object Groups Parameters Popup Menus Program Units
Instances
ON-CLEAR-DETAILS WHEN-NEW-FORM-INSTANCE DEMO_OBJECTS None COURSE SECTION COURSE_SECTION None None None None None CHECK_PACKAGE_FAILURE CLEAR_ALL_MASTER_DETAILS DEMO_ALERT QUERY_MASTER_DETAIL TABLE_ITEM_PROMPT_ALIGNMENT None None None COURSE_INFORMATION
go to contents
27
From the Main Menu, select File | New | Form and observe what happens. b) What did Forms name the new module? Did the Form Builder create any default objects for this form? If so, what kinds of objects are they and what are they called? Answer: Forms named the module MODULE2 (the number may be different for you). Forms created a window called WINDOW1. Whenever you create a module in Forms, it is named MODULE# by default. This is also true for how Forms initially names all objects you create. Whenever you create a new form or module, Forms creates WINDOW1 by default. A single module can have multiple windows. c) Can you assign a menu to a form (module)? How do you know? Answer: Yes you can. Each module object has a Menu Module property. So far you have only explored single forms. A typical Forms application will be made up of many forms. It is common to employ a menu system to control all of the forms that make up a single application. Menus are built in the Form Builder and then assigned to forms using the Menu Module property. In Chapter 13, Forms Menus, you will build your own menu using the Form Builder and a tool within it called the Menu Editor.
LAB 1.2
1.2.5
ANSWERS
a) Which Forms objects contain items? Can an item stand by itself or be selfcontained? Answer: Blocks contain items. No, an item cannot stand by itself. It must be contained by a block. b) Where do you put an item if you want it to be visible to a Forms user? Answer: You put an item on a canvas if you want it to be visible to a Forms user. c) How do blocks relate to canvases? Answer: Items from a block can be positioned on a canvas, but there is no direct relationship between blocks and canvases themselves. This means that at no time do you specify that a block is on a certain canvas. Nor at any time do you specify that a canvas belongs to or is owned by a certain block.
go to contents
28
LAB 1.2
Remember, form and module are interchangeable terms, so the latter part of this question was a bit deceiving. However, it does help to illustrate that a module (form) is a single application component that cannot be contained or owned by any other application component. It can, however, be associated with other form or module components as you will learn later on. e) Which objects can have triggers associated with them in the Object Navigator? Answer: Forms, blocks, and items can have triggers associated with them in the Object Navigator. Other objects, such as alerts and canvases, can be controlled and manipulated by triggers, but only modules, blocks, and items can have triggers associated with them. Later, as you work more with triggers, you will find that they are sometimes referred to as form-level triggers, block-level triggers, or item-level triggers. f) Of the objects youve learned so far, which are visible to the Forms user? Answer: Canvases, frames, windows, and items are visible to the user. In passing, you have learned about alerts and lists of values (LOVs), which are also visible to the user.
go to contents
29
LAB 1.2
go to contents
30
CHAPTER
go to contents
C H A P T E R
t is now time to try your hand at creating some simple applications. The Form Builder has a vast and versatile interface that you will continue to uncover throughout the rest of this book and throughout your relationship with the tool. Lab 2.1 will bring you through the wizards that provide an extremely friendly environment in which to start your acquaintance with the Form Builder. Once you have used the wizards to create some simple applications, you will want to know what to do with the files that store them. Lab 2.2 will lead you through some of the files you will encounter when using the Form Builder and illustrate how you can run and compile them.
31
go to contents
32 LAB 2.1
LAB
2.1
Use the Data Block and Layout Wizards Reenter the Wizards
Having sat patiently through two introductory Labs, you are now ready to start creating objects and simple applications. Despite your limited experience with the Form Builder interface, you will be able to quickly and easily create a block and assign its items to a canvas by employing the Data Block and Layout Wizards. The initial part of this Lab will walk you through the pages of the wizards. This will be interactive and will require that you read and work in the Form Builder simultaneously. The Exercises will push you a bit further in using the wizards and test the skills you learned in the Lab.
Throughout this Lab and the Exercises, try to think about the concepts you have already learned. What are the mandatory Forms objects that you are creating with the wizards? Can you see how they are related? Can you identify possible events in your new applications and anticipate which triggers you might use to respond to them?
go to contents
33 LAB 2.1
THE WIZARDS
The Data Block Wizard and Layout Wizard are usually used in sequence, meaning you walk through the Data Block Wizard to create a block, then you go straight to the Layout Wizard to create a canvas and frame. Each screen in a wizard is referred to as a page. The creators of Oracle Forms have given each page a specific name, and it is these names that will be used here. Each page will be described briefly in the Lab and will be accompanied by a screenshot. You should have the Form Builder open and follow along with each step in the Lab. You will start by using the Data Block Wizard to create a block based on the STUDENT table.
Each wizard has a Welcome! page. These will not be discussed in the Lab.
To set the environment, take the following steps: 1) 2) 3) 4) Open the Form Builder. From the Main Menu, select File | New | Form to create a new form. From the Main Menu, select File | Connect to connect to the database. From the Main Menu, select Tools | Data Block Wizard to open the Data Block Wizard.
go to contents
34 LAB 2.1
Figure 2.1 I The Data Block Wizards type page. TYPE PAGE
Here you have one taskto select the type of database object your block will be based upon. Throughout most of this book, you will base your blocks on tables. So, here you should leave this page set to Table or View (the default) and click the Next button.
35 LAB 2.1
There are two ways to choose the base table. You can either type the table name directly into the Table or view field, or you can click the Browse button to choose the table name from a list. If you click Browse, Forms will peer into the databases data dictionary and present you with a Tables window with a list of tables and views like the one in Figure 2.3. Double-click the STUDENT table to select it. This brings you back to the table page, which will now look like Figure 2.4. Now you are on to the second task on this page, which is to select the database columns to include as items in this block. You do not have to include all of them, but it is wise to always include the primary-key columns and foreign-key columns. To select columns, simply move them from the Available Columns text list to the Database Items text list. There are two ways to do this: 1) 2) Use the arrow buttons positioned between the two text lists. Double-click individual items.
go to contents
36 LAB 2.1
Figure 2.4 I The Database Wizards table page with the STUDENT table selected.
Windows multi-select functions also work here so that you can move more than one column at a time. Move all of the columns in the STUDENT table to the Database Items list. Check Enforce data integrity for your STUDENT block. The table page should now resemble Figure 2.5. Click the Next button to move on.
go to contents
37 LAB 2.1
CANVAS PAGE
Here you have two or three tasks that you will accomplish by making selections from three list items: 1) 2) 3) Choose the canvas. Choose the canvas type. Choose the tab page.
go to contents
38 LAB 2.1
As you can see, this page is nearly identical in layout to the Data Block Wizards table page. It behaves the same way as well, so you already know how to move items from the Available Items list to the Displayed Items list. Use whichever method you prefer to move all of the items to the Displayed Items list. It is possible and common to display only a few of the items that are available. It all depends on how you want your canvas to look. Your next task is to order the items. The Layout Wizard will lay the items out on the screen in the order in which they appear in the Displayed
go to contents
39 LAB 2.1
Your next task is to set the item type for each item. To do this, take the following steps: 1) 2) Select an item in the Displayed Items list. Change its type using the Item Type drop-down list.
Change the following items to type Display Item: CREATED_BY. CREATED_DATE. MODIFIED_BY. MODIFIED_DATE.
Once you have completed this task, click the Next button to move on to the items page.
go to contents
40 LAB 2.1
To complete this task, simply position the cursor on the value that youd like to change and edit it. Make the following adjustments: 1) 2) Change the Width of CREATED_BY to 117. Change the Width of MODIFIED_BY to 117.
After you have made these changes, click the Next button to continue to the style page.
STYLE PAGE
No screenshot is necessary here as this is a rather simple page. Your task is to decide how you would like the items to be laid out. In form style, the items are laid out so that the screen will resemble a paper-based form. In tabular style, the items are laid out in a grid. Look at the images on the lefthand side of the wizard page to see examples of how each style will look. Select the Form style. Click the Next button to move on to the records page.
RECORDS PAGE
Here you have four tasks: 1) Choose a Frame Title.
go to contents
41 LAB 2.1
Because this is a new form and because you have chosen Form as the layout style, you will only have to complete the first task on this page. You will leave the rest as their default values. Type the name Student into the Frame Title field; this will be displayed on the canvas. Because the items will be laid out in form style, it is common to display only one record. Therefore, there is no need to change the value in the Records Displayed field. For the same reason, there is no need to change the value in the Distance Between Records field or the Display Scrollbar check box. In this Labs Exercises, you will have to create some forms with a tabular style, which lends itself to displaying multiple records at one time with a scrollbar. Click the Next button to move on to the finish page. You could also simply click the Finish button here as the finish page is purely informational.
FINISH PAGE
No screenshot has been included, as there are no tasks on this page for you to complete. It is purely an informational page. Click the Finish button and the Layout Wizard will close and your new canvas will be displayed in the Layout Editor. It should resemble Figure 2.10.
42 LAB 2.1
Explore the Form Builder and find at least four different ways to access the Data Block Wizard. a) What are the four ways you discovered to open the Data Block Wizard?
Open the R_WIZARD.fmb form, which you created in the previous Lab. Expand the Data Blocks node, expand the STUDENT block, and then expand the STUDENT blocks trigger node. b) What is the name of this trigger?
Double-click the icon to the right of the trigger to open the PL/SQL Editor and view the trigger code. c) What function will this trigger code perform?
go to contents
43 LAB 2.1
d) Where did the Data Block Wizard get the information needed to write this trigger?
e) What did you do to make the Data Block Wizard create this trigger?
f) Have any triggers been created for other objects in this block? Fully expand the block and its items to find out.
View the properties for the STUDENT_ID item. g) What are the values for the Data Type and Required properties?
h) How did the Data Block Wizard know to set them this way?
2.1.2
Select File | Close from the Main Menu to close the R_WIZARD.fmb form. Select File | New | Form from the Main Menu to create a new form. Use the wizards to quickly create a form based on the SECTION table. In the Data Block Wizard, include all of the SECTION tables columns in the block except CREATED_BY and MODIFIED_BY. Enforce data integrity should be unchecked. go to contents
44 LAB 2.1
In the Layout Wizard, display all of the items. Do not make any changes to Prompt, Width, or Height on the items page. Choose Form as the layout style. Display one record and do not include a scrollbar. Click the Finish button when you are done so that control returns to the Form Builder. Use the Name property in the Property Palette to rename the canvas SECTION and its frame SECTION. Select File | Save from the Main Menu and save the form to \guest\forms\exercises as R_SECTION.fmb. a) Try to reenter the Data Block Wizard for the SECTION block using the Object Navigator. How did you do it?
b) How does the reentered Data Block Wizard look different from the wizard you are used to using?
c) Which tab page should you select to add CREATED_BY and MODIFIED_BY to the block? How did you add them?
Click the Finish button. d) Have CREATED_BY and MODIFIED_BY been added to the block? How about the canvas?
Select the SECTION canvas in the Object Navigator and right-click to open the Layout Wizard.
go to contents
45 LAB 2.1
e) Does it look like you have reentered the Layout Wizard? How can you tell that you havent?
f) Which object should you select in the Object Navigator to reenter the Layout Wizard for the items in the SECTION block? Why? Hint: This is a physical object that helps you group and arrange items.
Position the reentered wizard in the upper right-hand corner of the screen so that you get a partial (or whole, depending on the size of your monitor) view of the Layout Editor. Select the Data Block tab page in the wizard and move CREATED_BY and MODIFIED_BY to the Displayed Items list. Click the Apply button. g) Between which items did they appear on the canvas? How can you use the wizard to position them so that they appear after CAPACITY and before CREATED_DATE?
Change the style of the form to Tabular and display five records. Click the Apply button to view the changes. h) What steps did you take to accomplish this? List the tab page names as well as a brief description of what you did on each page.
After you apply the changes and exit the wizard, the layout will have changed and will not look very pleasing. Ignore this. The purpose of this Exercise is to experiment with reentering the wizard. Look and feel is not important here.
go to contents
46 LAB 2.1
a) What are the four ways you discovered to open the Data Block Wizard?
4) 5)
6)
It is certainly not necessary to memorize these six ways now. However, as you become more familiar with the Form Builder, you will become anxious to speed your programming by learning different ways to accomplish the same tasks. As a beginner, it is wise to use the toolbars because they are visual and intuitive. Right-clicking is also very easy because you will be presented with a descriptive, context-sensitive list of choices. b) What is the name of this trigger? Answer: The name of this trigger is KEY-DELREC. Double-click the icon to the right of the trigger to open the PL/SQL Editor and view its code. c) What function will this trigger code perform? Answer: It will prevent the user from deleting STUDENT records if ENROLLMENT records exist.
go to contents
47 LAB 2.1
Answer: It queried the databases data dictionary to see if there were any integrity constraints defined for the table and its columns. e) What did you do to make the Data Block Wizard create this trigger? Answer: You should have checked Enforce data integrity on the Data Block Wizards table page. f) Have any triggers been created for other objects in this block? Fully expand the block and its items to find out. Answer: Yes, WHEN-VALIDATE-ITEM triggers have been created for many of the items in the block. As you have learned through these Exercises, checking Enforce data integrity results in the Form Builder writing a series of triggers in the form to enforce the integrity constraints that are stored at the database level. Had there been check constraints against the STUDENT table, triggers to enforce them would have been written as well. By checking Enforce data integrity, you are helping to insure that the data going to the database is valid.
I FOR EXAMPLE:
Re-open the R_WIZARD.fmb form if you have already closed it. Double-click the icon next to the WHEN-VALIDATE-ITEM trigger for the STUDENT.ZIP item. Note that the code has two sections. If you study the code for the first section, you can see that it is based on the STU_ZIP_FK foreign-key constraint. The trigger code is analogous to the constraint in that it will compare the ZIP value that a user has entered in the form with the ZIP values in the ZIPCODE table. If it does not find the value in the ZIPCODE table, the trigger will return an error. This corresponds to what you know about foreign-key constraints; a ZIP value cannot be inserted or updated into the STUDENT table unless that value already exists in the ZIPCODE table. Well, you say, the database would have done this anyway: thats what the constraint is for. True, but the database would not have checked until the user had tried to commit the record. The user would have tabbed out of the ZIP item and would have had to navigate back to fix it. But, it is unlikely that the user would know which item to fix because the database would not have returned an intelligible error message. By putting the validation logic at the item level within the application itself, the user is alerted to his mistake immediately and is given a message he can understand.
go to contents
48 LAB 2.1
I FOR EXAMPLE:
By default, the wizard uses the column names in the database to create the prompts for the items. In the sample STUDENT schema, most of the column names are in near-plain English, so in most cases, there really isnt any need to change the default prompts that the wizard assigns. In many database systems, column and table names will not be in plain English, or they may be prefixed with some kind of system codes like ST_STUD_NAM. In cases like these, it is wise to use the items page of the Layout Wizard to change the prompts. You can certainly change the prompts later, but if you do it on the items page, the Layout Wizard will create the default layout with these prompts in mind. That is, it will adjust the layout automatically depending on the lengths of the prompts. If you choose to set the prompts manually in the Layout Editor, which is certainly possible, you may have to manually adjust more than just the prompt name to keep the layout neat and organized. This could cost you some time and considerable effort. The length of an item is expressed in points. Forms makes each item approximately five points wide for each character. So, if a column in the database is set to be VARCHAR(5), Forms will make its corresponding item approximately 25 points wide. These values are set for each module in the Coordinate System property. To view the Coordinate System settings, take the following steps: 1) 2) Open R_WIZARD.fmb in the Form Builder. In the Object Navigator, select the R_WIZARD module and view its properties.
go to contents
49 LAB 2.1
Figure 2.11 I The Coordinate Info dialog displaying the coordinate system information for the form.
3) 4) 5) Scroll down in the Property Palette window until you see the property named Coordinate System. Select the Coordinate System property and click the More button. You will see a window like the one in Figure 2.11.
You can reset the coordinate values here if you wish. The Coordinate System pop-up list lets you choose between Real and Character coordinates. Real gives you a finer grain of control over the positioning of items, while Character positions items by character cell. What you select here depends on how you plan to deploy the application. The Forms help system provides a small chart that explains which Coordinate System setting to choose for some of the more common deployment strategies. If you choose the Real coordinate system you can choose which real unit youd like to use and the size of each character cell. The Character Cell values determine the size of each character cell, expressed in the Real Unit you have chosen. All of this is quite a mouthful and not worth worrying about now. It is certainly possible that at some point, depending on the deployment strategy, you will have to reset these values. For now you should understand the concepts behind the different coordinate systems, but stick with the default values.
2.1.2
ANSWERS
a) Try to reenter the Data Block Wizard for the SECTION block using the Object Navigator. How did you do it? Answer: You should have selected the SECTION block, then right-clicked and selected the Data Block Wizard.
go to contents
50 LAB 2.1
go to contents
51 LAB 2.1
Which object should you select in the Object Navigator to reenter the Layout Wizard for the items in the SECTION block? Why? Hint: This is a physical object that helps you group and arrange items. Answer: You should select the SECTION frame.
When the Layout Wizard creates and positions items on canvases, it also lays them out within frames. As described before, these frames are graphical objects that belong to canvases. They make it easier to control the layout of multiple groups of items.
I FOR EXAMPLE:
In Chapter 4, Master Detail Forms, you will learn how to create forms, which will allow you, among other things, to have two groups of items on a single canvas. One of those groups will be laid out in form style, while the other will be laid out in tabular style. Each group of items will be in a frame. Having two frames will let you reenter the Layout Wizard for each group (frame) individually. Select the Data Block tab page in the Wizard and move CREATED_BY and MODIFIED_BY to the Displayed Items list. Click the Apply button. g) Between which items did they appear on the canvas? How can you use the wizard to position them so that they appear after CAPACITY and before CREATED_DATE? Answer: It depends on which item was selected in the Displayed Items list when you moved CREATED_BY and MODIFIED_BY over. You can drag the items up and down in the Displayed Items list to position them. Change the style of the form to Tabular and display five records. Click the Apply button to view the changes. h) What steps did you take to accomplish this? List the tab page names as well as a brief description of what you did on each page. Answer: On the style page, you must select the radio button labeled Tabular. On the rows page, you should set Records Displayed to 5 and check the check box labeled Display Scrollbar. Reentering the wizard can be fast and efficient for adding items to the Displayed Items list or changing the layout style from Form to Tabular, or vice versa. However, it can sometimes be more trouble than it is worth. In the example, it stretched the layout far beyond the width of the canvas. As your proficiency with the Form Builder improves, you may find it easier to make adjustments manually in the Layout Editor rather than reentering the Layout Wizard.
go to contents
52 LAB 2.1
go to contents
53
LAB
2.2
LAB 2.2
Differentiate Between Source and Executable Files Compile Binary Files into Executable Files Run Executable Files
In the previous three Labs, you learned how Forms works, what its mandatory objects are, and how to create those objects quickly and easily using wizards. With this knowledge, you have been able to create some simple applications. Now you will learn how Forms stores the applications you create, how it readies those applications for deployment, and how it runs those readied files.
go to contents
54
RUNNING
It is common to want to test your forms as you are building them. You finish the wizard, make a few changes to the layout, add a couple of triggers, and then you want to test the form and see how it looks and operates. You can do this by running the form directly from the Form Builder. You do not have to be in the Form Builder to run a form. Along with the Form Builder and Form Compiler, another utility called the Forms Runtime is available. In the Exercises, you will get the opportunity to run forms from the Form Builder, as well as from the Forms Runtime.
go to contents
Lab 2.2: Oracle Forms Files b) Which file do you work with in the Form Builder?
55
c) Do both files get distributed to application users? If not, which one does?
LAB 2.2
2.2.2
Open EX02_02.fmb in the Form Builder. Select File | Administration | Compile File from the Main Menu. a) How can you tell that a form compiled successfully?
b) What can you assume was created and written to the filesystem?
Expand the Triggers node for the module and double-click the icon for the WHEN-NEW-FORM-INSTANCE trigger. Delete the semi-colon after the word Null. Compile the file again using the steps for Question a. c) Did the form compile successfully? How do you know?
Replace the semi-colon and compile the form a third time. Select Program from the Main Menu and view the compilation options you see.
go to contents
56
Lab 2.2: Oracle Forms Files d) These options are for compilation, but they do not create .fmx files. What do you think they compile?
LAB 2.2
From the Windows Start menu, locate and run the Form Compiler. You will find it just below the Form Builder in the menu system. Once the Form Compiler has opened, use its Browse button to locate the file EX02_02. e) Which should you compile, the .fmx file or .fmb file?
Enter your User ID, Password, and Database alias and click the OK button. f) What happened?
g) Why can you develop your applications on a Windows NT workstation but then deploy them to a UNIX server?
h) What steps would you take to accomplish 2.2.2.g? Assume that the UNIX machine is running Solaris 2.6.
go to contents
57
2.2.3
Open EX02_02.fmb in the Form Builder. a) Which button on the Object Navigator should you use to run the form?
LAB 2.2
b) What has to be happening by default when you run a module from the Form Builder?
From the Windows Start menu, locate and run the Forms Runtime. You will find it just below the Form Compiler in the menu system. Once the Forms Runtime is open, use the Browse button to locate the file EX02_02. c) Which should you run, the .fmx file or the .fmb file?
A running form can be in one of three modes: Normal, Enter Query, or Fetch mode. d) Which button on the default toolbar can you use to put the form into Enter Query mode?
go to contents
58
a) What are the differences between .fmb files and .fmx files?
2.2.2
ANSWERS
Answer: The message Module Built Successfully will appear in the Form Builders hint line.
You can also tell that it has compiled successfully when no error messages are reported. b) What can you assume was created and written to the filesystem? Answer: You can assume that an .fmx file was created since the result of a successful compilation is an executable. When the Form Builder creates .fmx files, it names them after the .fmb files. So, if you look in the filesystem, you will now find a file called EX02_02.fmx. Compile the file again using the steps for Question a. c) Did the form compile successfully? How do you know? Answer: No, a Compilation Errors window opened.
go to contents
59
Note that the message in the window is very descriptive. It tells you where in the form it encountered the error and supplies an error number. Answer: Under the Program selection from the Main Menu, there are options for Compile and Compile Selection. d) These options are for compilation, but they do not create .fmx files. What do you think they compile? Answer: They compile the PL/SQL in the form, but do not create an .fmx file. As your forms get more complicated, you will add more PL/SQL objects such as triggers, program units, and PL/SQL libraries. You can use the Compile and Compile Selection commands to compile these PL/SQL objects. e) Which should you compile, the .fmx file or.fmb file? Answer: The .fmb file. The compiled version of the .fmb is saved as an .fmx file. Enter your User ID, Password and Database alias and click the OK button. f) What happened? Answer: Next to nothing. There was a pause, then the Form Compiler closed. If the module compiles successfully, the Form Compiler will not give you any confirmation of success; it will simply close. You can confirm the compilation by looking for the .fmx file in the filesystem. If there are any errors during compilation, they will be displayed in a Forms Compilation Error window and written to a text file. This text file will have the same name as the .fmb file and will have an .err extension. So, in this case, if there had been an error, you would have seen it listed in an error window and would have been able to find an EX02_02.err file in the filesystem. g) Why can you develop your applications on a Windows NT workstation but then deploy them to a UNIX server? Answer: Because .fmb files are platform-independent and therefore portable from platform to platform. The current release of Oracle Developer is supported on Windows 95, 98, and NT, as well as a host of UNIX and other platforms. For a complete list of supported platforms, visit www.oracle.com.
LAB 2.2
go to contents
60
LAB 2.2
Cross-platform development is a large topic, though it is not covered within the scope of this book. However, it is worth noting that it is one of the features that has made Oracle Forms such a popular product.
2.2.3
ANSWERS
Open EX02_02.fmb in the Form Builder. a) Which button on the Object Navigator should you use to run the form? Answer: From the Object Navigators vertical toolbar or the Layout Editors horizontal toolbar, click the Run Form Client/Server button. This button has a traffic light as its icon. b) What has to be happening by default when you run a module from the Form Builder? Answer: By default, the Form Builder has to be compiling the .fmb file and creating an .fmx file.
go to contents
61
Remember, in the Form Builder you are working with .fmb files, which are binary files and therefore not executable. Only executables, .fmx files, can be run. When you click Run, Forms automatically compiles the form before running it. If there are any errors during compilation, they will be displayed to you in a Compilation Errors window. If the form can run despite these errors, it will. If it cant, you will have to fix the errors in the Form Builder before continuing. c) Which should you run, the .fmx file or .fmb file? Answer: You should run the .fmx file, which is the executable. A running form can be in one of three modes: Normal, Enter Query, or Fetch mode. d) Which button on the default toolbar can you use to put the form into Enter Query mode? Answer: The Enter Query button, which has a question mark in front of a cylinder as its icon. When a form first opens, the default behavior is for it to be in Normal mode, which means it is capable of accepting new records or updating existing ones. By putting the form in Enter Query mode, it is set to accept a query by example.
LAB 2.2
I FOR EXAMPLE:
Run the EX02_02 form from the Form Builder. Click the Enter Query button. Type 101 into the INSTRUCTOR ID item. Click the Execute Query button. Note that only the record for Instructor 101 was returned to the form. You can add more than one parameter in Enter Query mode and you can also use wildcards. So, what would you do to get all of the instructors whose Zip Code is 10025 and whose FIRST NAME begins with T? Put the form into Enter Query mode, enter T% in the FIRST NAME item and 10025 in the ZIP item, and click the Execute Query button. There are certain restrictions that apply to writing code in Enter Query mode, which you will learn in later Chapters as you begin to write more complicated logic and utilize built-ins. e) In which mode should the form be to insert records? Answer: The form should be in Normal mode.
go to contents
62
LAB 2.2
Shortcut keys in the Form Builder can be very helpful for running and compiling forms. Use CTRL-R to run the form. Use CTRL-T to compile the form without running it.
go to contents
63
CHAPTER
go to contents
CHAPTER
he Oracle Forms development environment is a simple one to navigate. The three main tools within in it are designed to make development intuitive and easy. The Exercises and Labs in this Chapter will walk you through the basic ins and outs of the Object Navigator, Property Palette, and Layout Editor. Along the way you will also learn more about how to configure the mandatory Forms objects that you learned about in Chapter 1, Concepts and Objects.
Some of the Exercises in this Chapter may seem elementary if you already have experience with a graphical development tool. You should still complete the Chapter since it will teach you many features and functions that are specific to the Form Builder and that may not be in the other tools to which you are accustomed.
Chapters 1, 2, and 3 are meant to prepare you for the more complicated and interesting lessons that lay ahead.
65
go to contents
66 LAB 3.1
LAB
3.1
Open and Identify Objects Create and Delete Objects Drag & Drop and Cut & Paste Objects Run and Save Forms View Database Objects
In the preceding Chapters, you learned about the fundamentals of Forms behavior and Forms objects, as well as how to create simple forms with the help of wizards. In doing so, you were exposed to the Object Navigator. In this Lab, you will take what you have learned about objects and apply it to working with those objects in the Object Navigator. The Object Navigator gives you a hierarchical view of all the objects in a form. It organizes these objects by node and lets you expand or collapse nodes to view objects. Within the Object Navigator, you can create, delete, move, and manipulate objects in other ways. You can name an object in the Object Navigator, but you cannot define it any further. To adjust the specific characteristics of an object, you must use the Property Palette or Layout Editor, which are discussed later in the Chapter. Figure 3.1 shows a screenshot of a single form open in the Object Navigator.
go to contents
67 LAB 3.1
Figure 3.1 I The Object Navigator with nodes expanded to show Data Blocks and Canvases.
Along the left-hand side is the vertical toolbar, and in the center is the hierarchy itself. Note that the Data Blocks and Canvases nodes are expanded to reveal their objects. Also note that below both the Data Blocks and Canvases nodes there are sub-nodes that are collapsed. In this Labs Exercises, you will be exposed to copying and moving objects from node to node and even from form to form within the Object Navigator. This powerful feature paves the way for object reusability, which you will learn about in later Chapters. You may have noticed that Forms is not alone as the highest node in the hierarchy. At the bottom there are five other nodes: Menus, PL/SQL Libraries, Object Libraries, Built-ins, and Database Objects, which are also at the top level of the hierarchy. In this Labs Exercises, you will experiment with the Database Objects node only, but the rest will come in later Chapters.
go to contents
68 LAB 3.1
Many of the Exercises ask you to click a specific button on the toolbar. The buttons have icons that indicate their function. However, if the meaning of an icon is not clear, note that all of the buttons have Tool Tips to further explain their purposes.
Open the Form Builder, but close all Forms modules. Use a button on the Object Navigators toolbar to open EX03_01.fmb. a) Which button did you use to open EX03_01.fmb?
b) Does this form have any alerts? How about LOVs? How could you tell without even touching the mouse or keyboard?
Use the expand and collapse buttons on the Object Navigators vertical toolbar to answer Questions cg c) How many blocks does this form have?
go to contents
Lab 3.1: The Object Navigator d) Are there any block-level triggers? Name them.
69 LAB 3.1
In the top left-hand corner of the Object Navigator window there is a Find feature. Use this to find the object QUERY_MASTER_DETAILS. g) What kind of object is it?
3.1.2
Open form EX03_01.fmb in the Form Builder. Use the buttons on the Object Navigators toolbar to complete the tasks in this Exercise. If you have not already done so, familiarize yourself with all of the buttons in the toolbar by reading their Tool Tips. Create an alert and change its name to NEW_ALERT. a) What did you have to do to change the alerts name?
go to contents
70 LAB 3.1
Lab 3.1: The Object Navigator b) How can you create a block manually? Change its name to CONTROL.
Create two items in the CONTROL block. Name one SAVE and the other EXIT. Select Tools | Property Palette from the Main Menu to open the Property Palette. Change the Item Type property of the SAVE and EXIT items to Push Button. c) Why cant you see these buttons in the Layout Editor?
3.1.3
Open form EX03_01.fmb in the Form Builder as in the previous two Exercises. Drag the button CONTROL.SAVE to the INSTRUCTOR block. Position it after the INSTRUCTOR.ZIP item. a) How did the Object Navigator indicate that you were positioning CONTROL.SAVE after INSTRUCTOR.ZIP?
Using the buttons on the Object Navigators vertical toolbar, copy the button CONTROL.EXIT to the SECTION block. Position it after the SECTION.CAPACITY item. go to contents
71 LAB 3.1
b) Which buttons did you use and how were you able to position the button item properly this time?
Leave EX03_01.fmb open, and then open form EX03_03.fmb. You will be working with both of them. Drag the following objects from EX03_01.fmb to EX03_03.fmb: Form Trigger - WHEN-NEW-FORM-INSTANCE. Alert - DEMO_OBJECTS. Program Unit - DEMO_ALERT. If you have trouble locating any of these objects, use the Object Navigators Find feature. c) What happened as you tried to drag these objects from one form to the other?
d) What do you think the relationship will be between the objects in EX03_01.fmb and EX03_03.fmb if you choose Subclass?
Choose Copy for each object. If you already chose Subclass, simply delete all of the objects you dragged into EX03_03.fmb. Drag them over again, but this time choose Copy. Keep form EX03_03.fmb open for the next Exercise.
go to contents
The purpose of this Exercise is to run a form from the Form Builder. You will also check that the objects from EX03_01.fmb in the previous Exercise were dragged over successfully. Using the buttons in the Object Navigators vertical toolbar, run the form EX03_03.fmb in Client/Server mode. Note the horrible layout. You will be fixing it in Exercise 3.3.1. a) Which button did you choose to run EX03_03.fmb ?
b) What message did you receive when you ran this form?
Use the buttons on the Object Navigators vertical toolbar to save the changes to the form. Make sure you save it to the \guest\forms\exercises directory.
3.1.5
Open the Form Builder. It is not necessary to have any forms open, but it will not hurt if you do. Make sure you are connected to the database. If you are unsure, reconnect from the Main Menu by selecting File | Connect. Open and view the STUDENT schema under the Database Objects node. a) What types of objects are visible to you? (Give the object types like Synonyms, Rollback Segments, and so forth.)
go to contents
Lab 3.1: The Object Navigator c) Can you add columns to the STUDENT table?
73 LAB 3.1
go to contents
74 LAB 3.1
Of course, you could have done this by simply clicking the small plus sign to the left of the Data Blocks node. Or, you could have even double-clicked the text of the node. However, it is good to become familiar with all of the GUI features of the Object Navigator and Form Builder. d) Are there any block-level triggers? Name them. Answer: Yes, there are three block-level triggers: POST-QUERY, ON-POPULATEDETAILS, and ON-CHECK-DELETE-MASTER. If you had trouble with this, take the following steps: 1) 2) Select the Data Blocks node. Click the Expand All button on the Object Navigators toolbar. The Expand All button has two black plus signs as its icon.
In this case, it is best to use the Object Navigators toolbar buttons instead of going straight for the node or the objects themselves. The Expand All button expands all of the blocks and all of the items as well, giving you a complete view. In this case, it lets you answer the question in two steps instead of three, four, or maybe more. e) Do both blocks have items? Answer: Yes, they both have items. If you used the Expand All button as described in 3.2.2.d, you would not have had to take any steps to find this out. f) Are there any item-level triggers? Name them. Answer: No, there are no item-level triggers. If you used the Expand All button as described in 3.2.2.d, you would not have had to take any steps to find this out. You may have been skeptical about the value of Expand All in Exercise 3.2.2.d. One click, two clicks, whats the difference? Here, the value of Expand All should be quite clear. There will often be times when you will want to see all of the trigger objects for all of your items. To click through each item individually would be tedious and annoying. While Expand All is convenient, it can also make the Object Navigator a bit messy. Luckily there are two other buttons, Collapse and Collapse All, on the Object Navigator that reduce what has been expanded to make the Navigator more readable. g) What kind of object is it? Answer: QUERY_MASTER_DETAILS is a Program Unit.
go to contents
75 LAB 3.1
Position the cursor in the Find text field, which is in the top right-hand corner of the Object Navigator. Start typing QUER You dont have to type the whole thing. In fact, it should have been found after you typed the first character. The Form Builder will automatically begin searching as soon you begin typing.
The Find feature is helpful as your forms get more involved, and it is especially helpful for finding triggers.
3.1.2
ANSWERS
Create an alert and change its name to NEW_ALERT. a) What did you have to do to change the alerts name? Answer: To change the name of an object in the Object Navigator, you have to first select it, then click it again to put it into an editable mode. This is a rather simple concept, but it does deserve some discussion for it can lead to some frustration if you are a beginner. There are three states for objects in the Navigator: deselected, selected, and name-editable. Again, this follows the same behavior as the Windows 95/98/NT Explorer. In this Exercise, you are focusing on the Object Navigator. Later on, you will learn how you could have changed the object name in the Property Palette instead. Both achieve the same result; the one you choose depends on where you are in the Form Builder at the time, or which method you prefer. If you had trouble with this, take the following steps: 1) 2) 3) 4) 5) Select the Alerts node in the Object Navigator. Click the Create button on the Object Navigators toolbar. The Create button has a large green plus sign (+) as its icon. Select the alert that was just created. It should be called ALERT14 (although the number may be different for you). Click it again so that a blue box appears around the text ALERT14. Change the name to NEW_ALERT.
go to contents
76 LAB 3.1
3) 4)
In Step 1, you selected the Data Blocks node and then clicked Create. Doing so in Forms positions your new block first in the list. If you had selected the INSTRUCTOR block, it would have put CONTROL second on the list. When a form is running, the user will be able to navigate through the form by tabbing from item to item. Forms sets the default navigation order based on how the items are positioned in the Object Navigator. So, the first item in the first block listed in the Object Navigator will be navigated to first, then the next, and so on.
I FOR EXAMPLE:
Figure 3.3 shows the Data Blocks node of the Object Navigator for a given form. Figure 3.4 shows the Layout Editor for the same form. Note the positions of the items in both the Object Navigator and Layout Editor. They are in a different order, arent they?
go to contents
77 LAB 3.1
78 LAB 3.1
Whenever you create items in the Object Navigator, they will not be visible until you go to Property Palette and set the items Canvas property. You will also have to set the items X Position and Y Position properties to position them properly on the canvas. If youd like to position these buttons on the canvas, take the following steps: 1) 2) 3) 4) Select both buttons with CTRL + click. In the Property Palette, change their Canvas property to INSTRUCTOR_SECTION. Open the Layout Editor and you will see that they are now positioned in the upper left-hand corner of the canvas. Drag them to positions below the SECTION frame.
You must have noticed also that you cannot specify an items type in the Object Navigator. By default, the Object Navigator creates new items as text items, then it is up to you to change the properties accordingly. Delete NEW_ALERT. d) Which button did you use to delete it? Answer: The Delete button, which has a red X as its icon. You could also have pressed the DELETE key on your keyboard.
go to contents
79 LAB 3.1
3.1.3
ANSWERS
a) How did the Object Navigator indicate that you were positioning CONTROL.SAVE after INSTRUCTOR.ZIP? Answer: A horizontal black line appeared to indicate where the button item would be positioned. If you had trouble with this, take the following steps: 1) 2) 3) Expand the INSTRUCTOR block and all of the items contained within it. Click and hold the CONTROL.SAVE button. Drag it down into the INSTRUCTOR block until the horizontal black line is below INSTRUCTOR.ZIP.
Note that dragging an object within a form actually moves that object from one place to another. Again, it is important to be aware of positioning in the Object Navigator. By placing the SAVE button after INSTRUCTOR.ZIP, you are indicating that you want the form to navigate from INSTRUCTOR.ZIP to INSTRUCTOR.SAVE (as it now belongs to the INSTRUCTOR block), then to INSTRUCTOR.TELEPHONE. b) Which buttons did you use and how were you able to position the button item properly this time? Answer: You should have used the Copy button, which has two pieces of paper as its icon, and the Paste button, which has a clipboard as its icon. You should have selected SECTION.CAPACITY before pasting to put the item in the proper position. 1) 2) 3) Select CONTROL.EXIT and click the Copy button. Expand the SECTION block and the items below it. Select SECTION.CAPACITY and click the Paste button.
Whenever you are pasting or creating objects in the Object Navigator, they will always be positioned directly below the object that is currently selected. c) What happened as you tried to drag these objects from one form to the other? Answer: As you tried to drop the object, a small alert should have appeared asking you whether you wanted to copy or subclass the object.
go to contents
80 LAB 3.1
3.1.4
ANSWERS
Answer: You should have used the Run button, which has the traffic light as its icon (the traffic light without the globe behind it).
If you had trouble running the form, take the following steps: 1) Put the focus of the Form Builder anywhere within the form EX03_03.fmb. 2) Click the Run button. As you already know, the Run button has a traffic light as its icon. To put the focus of the Form Builder on a certain form in the Object Navigator, you select any object within that forms module. This tells the Form Builder which form you want to work with when you click Run, Close, Compile, Save, and so forth. If you have multiple forms open at one time, which is often the case, getting the proper focus is essential. b) What message did you receive when you ran this form? Answer: An alert saying, A letter of approval from the director must accompany all teaching reassignments should have appeared when you ran the form. If it did not, then you made a mistake when completing the tasks before in Exercises 3.1.3.ce
go to contents
81 LAB 3.1
3.1.5
ANSWERS
a) What types of objects are visible to you? (Give the object types like Synonyms, Rollback Segments, and so forth.) Answer: Stored Program Units, PL/SQL Libraries, Tables, Views, and Types are visible. If you had any trouble locating STUDENTs objects, take the following steps: 1) 2) Expand the Database Objects node in the Object Navigator. Expand the STUDENT schema.
b) Can you rename the STUDENT table to ST_STUDENTS? Answer: No you cannot. See the discussion under Question c for more details. c) Can you add columns to the STUDENT table? Answer: You cannot add columns to a table through the Object Navigator. Tables, views, and columns are read only, so you cannot change them. However, the Database Objects node is a handy way to examine the contents of database tables and views. It shows you table, view, and column names, as well as column data types and lengths. This can save you from having to go to SQL*Plus or another tool to view table descriptions. d) Does the STUDENT table have any triggers? Answer: No. e) Are you able to create and edit database triggers? Answer: Yes you are. You can create and edit any stored PL/SQL object through the Form Builder. To test this out, simply select the Triggers node for the STUDENT table and click the Object Navigators Create button. The database trigger editor will open and you will be able to set the trigger type as well as write the trigger code. You can also write PL/SQL stored procedures, packages, and functions through the Form Builder.
go to contents
82 LAB 3.1
go to contents
83 LAB 3.1
6) Which of the following best describes subclassing? a) ____ When one object is linked to another object in such a way that if changes are made to either object, both objects are updated with these changes b) ____ When you make a copy of an object in one form and place it into another form c) ____ When a source object is linked to a subclassed object in such a way that if changes are ever made to the source, the subclassed object can inherit these changes d) ____ Creating new objects based on old ones 7) While you can drag objects from one form to another, you cannot drag an object from node to node within a form. a) ____ True b) ____ False Quiz answers appear in Appendix A, Section 3.1.
go to contents
84
LAB
LAB 3.2
3.2
In the Object Navigator, you create objects; in the Property Palette, you define objects. The Property Palette displays a list of characteristics (properties) for whichever object is currently selected in a form. The look, feel, and behavior of an object can be defined by its properties.
VIEW PROPERTIES
Figure 3.5 shows a screenshot of the Property Palette displaying the properties for a text item. First be aware that what is shown here is not the entire Property Palette. The window is scrollable, and for an item, the list of properties is over four times as long as the list you see in the figure. Having this many configurable properties gives you a tremendous amount of control over the object. The Property Palette is coordinated with both the Object Navigator and Layout Editor. So, if you select an object in either of these tools, its properties appear in the Property Palette. Within the Property Palette, properties are grouped in categories like General, Physical, Database, and so on. These categories are expandable and collapsible like the nodes in the Object Navigator. Having this expand-and-collapse ability makes it much easier to view properties.
go to contents
85
LAB 3.2
CHANGING PROPERTIES
Forms objects are defined by their properties. So, if you change an objects properties, you are changing its definition. If the changes you make affect an objects physical appearance, those changes will be immediately apparent in the Layout Editor. By the same token, any changes you make graphically in the Layout Editor will be immediately apparent in the Property Palette. You can change properties for one object at a time, or you can do mass changes by selecting multiple objects.
I FOR EXAMPLE:
Lets say you wanted 10 items in your form to be displayed in the font Times New Roman. To make mass property changes to these items, you would take the following steps:
go to contents
86
LAB 3.2
Open form EX03_02.fmb in the Form Builder. a) Explore the Form Builder and list three different ways of opening the Property Palette for the STUDENT.PHONE item.
b) View the properties for the STUDENT block. Now switch to view the properties of the STUDENT canvas. What are the first three nodes in the Property Palette for the block? for the canvas?
View the Bevel property for the STUDENT.PHONE item. Press the F1 key (for MS Windows users) to view help for the Bevel property. c) What is the purpose of the items Bevel property? Besides the F1 key, can you see another way to get a hint about what the property is for?
3.2.2
CHANGE PROPERTIES
Use form EX03_02.fmb as in the previous Exercise. For the item STUDENT.ZIP, change the following properties:
Font Name to Arial. Font Weight to Demibold.
go to contents
87
a) What happened to the small icons to the left of the property names? Why is this helpful?
LAB 3.2
Go back to the properties for STUDENT.ZIP and select the Insert Allowed property. Click the Inherit button on the Property Palettes toolbar. b) What happened?
Stay with the properties of STUDENT.ZIP. Click the Freeze button so that its icon becomes a pinhead rather than a full pin. Select STUDENT.PHONE in the Object Navigator. Right-click to open another Property Palette window. Drag this window to the left. c) What can you see under the Property Palette window you just dragged away?
Use CTRL + click to select STUDENT.ADDRESS, STUDENT.EMPLOYER, and STUDENT.REGISTRATION_DATE. View the Y Position property for each object. d) What are the values of these properties? Can you change them?
e) View the Name property for all of these items. Can you change that property?
go to contents
88
In Question a of this Exercise, you changed the Font Name and Font Weight properties for STUDENT.ZIPCODE. Copy the font-related properties from one of these items and paste them to the remaining items in the STUDENT block.
LAB 3.2
f ) Can you complete this task for all of the items at once? How?
a) Explore the Form Builder and list three different ways of opening the Property Palette for the STUDENT.PHONE item. Answer: There are actually five ways to access the palette from the Form Builder: 1) 2) 3) From the Main Menu, select Tools | Property Palette. In the Object Navigator, select the object whose properties youd like to see and right-click. In the Object Navigator, double-click on the icon to the left of the object whose properties youd like to see (this does not work for canvases). In the Layout Editor, select the object whose properties youd like to see and right-click. Press F4.
4) 5)
Because the Property Palette is a tool you will access frequently, it is helpful to familiarize yourself with all of the methods listed here. b) View the properties for the STUDENT block. Now switch to view the properties of the STUDENT canvas. What are the first three nodes in the Property Palette for the block? for the canvas? Answer: For the block, the first three nodes are General, Navigation, and Records. For the canvas, they are General, Functional, and Physical. This illustrates two important points: first, that there is coordination between the Object Navigator and Property Palette; second, that the properties for each type of object can vary greatly.
go to contents
89
Answer: For an item, setting the Bevel property changes the appearance of the items border. Besides pressing F1, you can find this out by looking at the Property Palettes hint line. The help system has a lot of information to offer about a property. Obviously, you can get a description of the property. In addition, the help system lists restrictions for using the property and the Forms built-in you would have to use to change this property programmatically. A built-in is a pre-written sub-program that you can use for standard application functions. To change an item programmatically, you write a trigger that calls a built-in.
LAB 3.2
I FOR EXAMPLE:
Lets say you want to create a button that changes the foreground color (font color) of the STUDENT_ID item. You would write a WHEN-BUTTONPRESSED trigger that would include the following command: SET_ITEM_PROPERTY('STUDENT.STUDENT_ID', foreground_color, 'red');
The built-in is SET_ITEM_PROPERTY. There are hundreds of built-ins available in Forms, and as you get deeper into this workbook, you will learn more of them.
3.2.2
ANSWERS
a) What happened to the small icons to the left of the property names? Why is this helpful? Answer: They have changed from small dots to small green squares. This is helpful because it shows which properties have been changed and which still have their original value. In later Chapters when you experiment with visual attributes, property classes, and subclassing, you will see even more changes to the icons in the Property Palette. b) What happened? Answer: The property has been returned to its default value.
go to contents
90
LAB 3.2
Figure 3.6 I The Property Palette when more than one object has been selected.
go to contents
91
Not all properties are available for mass changes. The purpose of mass changes is to give more than one object the same value for a certain property. Therefore, Forms will not allow you to do mass changes on the Name property of items because all items within a block must have a unique name. Also, Forms will not allow you to do mass changes to the Subclass Information property. f) Can you complete this task for all of the items at once? How? Answer: Yes you can. Copy the properties for STUDENT.ZIPCODE, then select all of the items in the STUDENT block. Click any property in the Property Palette to put the focus there, then click the Paste Property button. As you can see, there are many ways to change the properties for multiple items at once.
LAB 3.2
go to contents
92
LAB 3.2
go to contents
93
LAB
3.3
The Layout Editor complements the Object Navigator and Property Palette in that it allows you to give faces to the objects you create and configure. Here you will visually position, arrange, size, and color your objects. In the Object Navigator, you can create any object, be it logical or physical. In the Layout Editor, you can only create physical objects that can appear on a canvas, such as items, other canvases, and graphics. Graphics include frames and any other non-item objects like rectangles, circles, lines, and static text. The Layout Editor has three toolbars that provide utility, formatting, and create functions. Figure 3.7 shows the Layout Editor and its toolbars.
go to contents
94
LAB 3.3
Figure 3.7 I The Layout Editor and its three toolbars. The Forms help system does not use these names for the toolbars. They are used in this text for the sake of the explanations and Exercises.
A brief introduction to the toolbars will provide a good starting point for understanding the Layout Editor.
UTILITY TOOLBAR
From the Utility toolbar, you can open, save, and run forms, as well as cut and paste and so on. As these functions are shared with the Object Navigator, there is no reason to explain them in detail. The Utility toolbar also lets you coordinate which block or canvas you are working with, as well as gain access to the wizards.
FORMATTING TOOLBAR
The first half of the Formatting toolbar is for formatting text and is similar in look and function to the equivalent toolbar you see in word proces-
go to contents
95
sors. It also provides tools for positioning and arranging text. The more tidy a form is, the easier it will be on the users eyes. The Exercises will walk you through the Formatting toolbar so that you can quickly and easily position objects in an orderly manner.
LAB 3.3
You will explore each of these functions in this Labs Exercises. The most powerful feature of the Tool Palette is that it provides an alternative to creating items in the Object Navigator.
Open EX03_03.fmb in the Form Builder. Open the Layout Editor and Property Palette. The objective of this set of Exercises is to learn the functions of the Layout Editor by cleaning up EX03_03.fmb. Use the Layout Editor to create a button item and position it anywhere within the Student frame. Change its name to EXIT and change its label to Exit. a) Which toolbar did you use to create the EXIT button?
go to contents
96
In the Object Navigator, create a block manually and name it CONTROL. Drag EXIT from the STUDENT block into the CONTROL block. Your objective is to create two more buttons for the CONTROL block. Before doing so, answer the following questions. c) Which feature on the Utility toolbar can you use to ensure that these new buttons are assigned to the CONTROL block?
LAB 3.3
Select the Button tool on the Tool Palette by clicking it once. The button will appear inset as if it has been pressed. Now, double-click it until an icon appears. d) What icon has appeared on the Button tool?
Drag the mouse pointer into the canvas area on the Layout Editor and click once in one spot, then click again in another spot. e) What has pinning the Button tool allowed you to do?
f) How do you think you can get out of the Pinning mode?
Rename and label these new buttons Save and Print. Arrange them neatly below the STUDENT frame. Using the Layout Editors horizontal toolbar, change the font of the STUDENT_ID item to Ms Sans Serif, 8 pt.
go to contents
97
g) Can you repeat the font change you just made by selecting all of the remaining items and prompts on the canvas?
There are three buttons at the bottom of the Tool Palette that deal with color. h) What are the Tool Tips for each of these buttons, and which property do you think each corresponds to?
LAB 3.3
i) Can you change the background color for all of the text items to white? Be sure to do this simultaneously for all items. Do not include the button items.
j) How about the foreground color? How would you change the foreground color for all of the items so that they are blue?
k) Have the prompt background colors changed, too? Can you change their background colors to gray without affecting the text items?
3.3.2
Select all of the text items on the canvas. Do not include the buttons. Use the Layout Editors Formatting toolbar to arrange all of the text items on the canvas so that they are flush-left (meaning their left-hand edges are all in a line). Hint: You may have trouble aligning if you select both the item and its prompt. go to contents
98
Lab 3.3: The Layout Editor a) Which button on the Formatting toolbar did you choose?
There are some Layout Editor functions that are not available on the toolbars. From the Main Menu, select Arrange | Size Objects to open the Size Objects dialog.
LAB 3.3
b) Which options would you select to give the objects a height of 14 points?
Again, have only the text items selected and from the Main Menu, select Arrange | Align objects. c) Which options should you choose to stack the objects vertically?
Select the STUDENT frame and change its Update Layout property to Automatically. Move three or four of the items so that they are misaligned by dragging them out of their current positions. Select the STUDENT frame in the Layout Editor. Click the Update Layout button on the Utility toolbar. d) What has the Update Layout button done?
go to contents
99
f) What must the frames Update Layout property be set to for it to work?
LAB 3.3
2. 3. 4.
For graphical objects such as items, stacked canvases, and frames, it is common to use the Layout Editor for creation. This way, you get to create, position, and size the object all at once. In this question, you single-clicked to create the button, which gave it the default size. Had you clicked and held the mouse button, you would have been able to drag out the size of the button as you were creating it. Create another button in EX03_03.fmb and try to size it upon creation. The upper left-hand corner of an object is what indicates its position. What this means is that if the X Position and Y Position properties are set to 10, 10, the objects upper left-hand corner is at the coordinates 10, 10. This is true for items, frames, canvases, windows, and all other graphical objects.
go to contents
100
LAB 3.3
c)
Which feature on the Utility toolbar can you use to ensure that these new buttons are assigned to the CONTROL block? Answer: On the Utility toolbar, use the drop-down list labeled Block.
When you begin to work with multi-block forms, it will be common for you to have objects from more than one block on the same canvas. When creating additional objects for these blocks in the Layout Editor, it will be important that you adjust the Block drop-down list accordingly. But, if you do forget to adjust it and you end up creating an item in the wrong block, you can use the Object Navigator to drag the item to the correct block. d) What icon has appeared on the Button tool? Answer: A small pin icon. e) What has pinning the Button tool allowed you to do? Answer: Pinning allows you to create more than one button at once. Pinning works for all the objects that you can create from the Tool Palette, including graphical objects and canvases. It was extremely helpful in this case, when you needed to create multiple buttons, and it will come in handy in future Exercises when you will need to quickly create more than one display item. f) How do you think you can get out of the Pinning mode? Answer: Click the Select tool in the upper right-hand corner of the Tool Palette, or any other tool in the Tool Palette. g) Can you repeat the font change you just made by selecting all of the remaining items and prompts on the canvas? Answer: Yes you can.
go to contents
101
Selecting deserves a bit of attention. The Windows select functions work here, so you could CTRL + click items individually to select them. However, in this case, because you are selecting many items, it is best to rubber-band them. To do so, you would follow these steps: 1. 2. 3. Position the mouse pointer above and to the left of all the items. Click and drag so that you are creating a rectangular band. Stretch this band around all of the items and release to select them. The entire object must be within the band for it to be included in the selection.
LAB 3.3
You can now start formatting and setting properties for all of the items and objects that you have selected. You can also move them as a group. If you were to single-click and drag one of the selected objects, all of the other objects would move with it. To perform an operation on everything on a canvas, go to the Main Menu and select Edit | Select All. h) What are the Tool Tips for each of these buttons, and which property do you think each corresponds to? Answer: Listed from top to bottom: Fill Color corresponds to Background Color, Line Color corresponds to Edge Foreground color, and Text Color refers to Foreground Color. Ignore Line Color since it only applies to frame edges and other graphical objects. Fill Color and Text Color are the tools you will use most frequently since they are what will alter the color of the text in a text or display item and the color of, obviously, its background. i) Can you change the background color for all of the text items to white? Be sure to do this simultaneously for all items. Do not include the button items. Answer: Yes you can. If you had trouble with this, try these simple steps: 1) 2) Select all of the items, but not their prompts. The best way to do this is to rubber-band everything then deselect the prompts. Click the Fill button and choose white to change the background color. The Fill button has a paint can as its icon.
go to contents
102
j)
LAB 3.3
3.3.2
ANSWERS
Answer: The button with Align Left as its Tool Tip.
As you experienced in some of the questions in Exercise 3.2.3, it is often necessary to deselect the prompts when performing group operations like coloring or arranging. In this case, you do not want the prompts to be included in the aligning function. You only want to align the text items and the prompts will follow. b) Which options would you select to give the objects a height of 14 points? Answer: Under Width, select No Change. Under Height, select Custom, and then type 14 into the text item. Under Units, select Points. The Size Objects window lets you enter custom measurements as you did in Question b, and it also lets you size objects based on other objects. c) Which options should you choose to stack the objects vertically? Answer: Under Align, select Each Other. Under Horizontally, select None. Under Vertically, select Stack. d) What has the Update Layout button done? Answer: The Update Layout button has automatically arranged the items on the canvas. e) Does this button update the layout of a canvas or frame? Answer: Update Layout only works for items within a frame. Other items on the canvas that are outside the frame will not be adjusted when you click the Update Layout button.
go to contents
103
What must the frames Update Layout property be set to for it to work? Answer: The Update Layout property must be set to either Automatically or Manually.
Once you have completed the Exercises in this Lab, the items in your canvas should be arranged and formatted as they are in Figure 3.8.
LAB 3.3
Figure 3.8 I EX03_03.fmb after its layout has been cleaned up.
go to contents
104
LAB 3.3
go to contents
105
CHAPTER
Figure 3.9 I A block based on the INSTRUCTOR table with items arranged. go to contents
C H A P T E R
MASTER-DETAIL FORMS
CHAPTER OBJECTIVES
In this Chapter, you will learn about: Master-Detail Forms Page 108
n the previous Chapters, you created and worked with forms that had only one base-table block. In this Chapter, you will create a form with multiple base-table blocks and establish a relationship between them. The relationship will, among other things, allow your users to issue a query in the master block, which will cause the form to issue a corresponding query in the detail block. You will create the master and detail blocks using wizards. This will automatically create an object called a relation. You will adjust the properties of the relation to change the behavior of the form.
107
go to contents
LAB
4.1
MASTER-DETAIL FORMS
LAB OBJECTIVES
After this Lab, you will be able to:
In the STUDENT schema, as in almost all schemas in relational databases, there are tables that are related. If the relationship is a primary-foreign key relationship, then one table can be considered the parent and the other can be considered the child.
I FOR EXAMPLE:
In the STUDENT database, there is a table containing data about students and a table containing data about enrollments. There is a primary-foreign key relationship between the STUDENT and ENROLLMENT tables that tells you that for each student record there may be zero or many enrollment records. In Forms, you can work with this relationship using a master-detail form like the one in Figure 4.1. A master-detail form is powerful in that it allows you to relate two blocks in the same way that two tables are related in a database. Thus, the users can see the data from both tables in a meaningful way. For instance, when they query on a certain student record, they will see only that students corresponding enrollment records. Not only will they be allowed to see the records, but they will be able to insert, update, and delete records in these blocks as well.
go to contents
Figure 4.1 I A master-detail form showing a Student and her associated Enrollments.
The form in Figure 4.1 was created using wizards and then manually edited in the Layout Editor. Note that the STUDENT items that belong to the master block are laid out in Form style with only one record displayed. The ENROLLMENT items that belong to the detail block are laid out in Tabular style with five records displayed. This single-record block to multi-record block layout style is typical for a master-detail form. The wizards can help tremendously in creating master and detail blocks. They create blocks as they normally do, but also help you create and configure the objects that will coordinate the master and detail block. There are a number of objects that work together to coordinate the processing of master and detail blocks. The main object is called a relation. Its job is to hold the join condition that relates the blocks. The relation object has a number of properties that control how the master and detail blocks are coordinated. The creation of the relation object also initiates the creation of some triggers and program units. These triggers are written by the Form Builder for you.
go to contents
In the following Exercises, you will create a master-detail form based on the INSTRUCTOR and SECTION tables. a) Which table will the master block be based on? the detail? Why?
Create the master block and its canvas using the wizards. Include all of the columns as items in the blocks, but do not display the audit columns on the canvas. Leave Enforce data integrity unchecked. Lay the items out in Form style. b) Did you have to do anything to indicate that this was to be a master block?
Start to create the detail block using the wizards. Leave Enforce data integrity unchecked. c) Have you encountered a new wizard page? What will it help you do?
Auto-join data blocks should be checked. Click the Create Relationship button. d) What is this List of Values dialog showing you? What happens after you click the OK button?
go to contents
e) Because the wizard has already written the join condition for you, do you need to change the values in the Detail Item and Master Item list items?
Click the Next button and continue on to the Layout Wizard. The SECTION block should appear directly below the items in the INSTRUCTOR block. f) Which canvas should you choose for the SECTION block? Is it necessary to create a new one?
Continue through the Layout Wizard until you get to the items page. g) Which of these items already appears in the INSTRUCTOR frame? Is it necessary to display it again in the SECTION frame?
Continue through the Layout Wizard. Select Tabular as the display style and display five records. Click the Finish button when you are done. h) If the layout of the SECTION items is not so pleasing, how can you make quick changes?
Run the form and issue a query. i) Have the items in both frames been populated?
go to contents
Lab 4.1: Master-Detail Forms j) What happens to the records in the SECTION frame when you scroll to the next record in the INSTRUCTOR frame?
4.1.2
Use R_INS_SEC.fmb for all of the Exercises in this section. a) Which block is listed first in the Object Navigator? What would be the problem if SECTION were listed first?
b) Has a relation object been created? What is it called? Which of the blocks owns the relation?
c) What is the value of the SECTION.INSTRUCTOR_ID items Copy Value from Item property? Why has the Form Builder done this?
d) What form-level triggers have been created? What block-level triggers have been created?
go to contents
e) Judging from their names, what do you think these triggers do?
Expand the Object Navigator so that you can see the triggers under the INSTRUCTOR block. Change the INSTRUCTOR_SECTION relations Delete Record Behavior property to Cascading. f) What has happened to the triggers under the INSTRUCTOR block?
Double-click the new trigger that has been created and view its code. g) Judging from the code you see and what you know about the term cascading from SQL, what will happen if you run the form and delete a master record?
Run the form. Navigate to SECTION_ID in the SECTION frame and issue a query. h) Were the SECTION records returned to the form? How about the INSTRUCTOR records?
Exit the form and return to the Form Builder. Change the INSTRUCTOR_ SECTION relations Prevent Masterless Operations property to Yes. Run the form. Navigate to SECTION_ID in the SECTION frame and issue a query.
go to contents
Lab 4.1: Master-Detail Forms i) Were the SECTION records returned to the form this time? What does the Prevent Masterless Operations property do?
Exit the form and return to the Form Builder. Change the INSTRUCTOR_ SECTION relations Deferred property to Yes and the Automatic Query property to No. Run the form and issue a query. j) Were the items in the SECTION frame populated?
Navigate to SECTION_ID in the SECTION frame and click the Execute Query button on the toolbar. l) What happened after you issued the second query?
Exit the form and return to the Form Builder. Change the INSTRUCTOR_ SECTION relations Automatic Query property to Yes. Run the form and issue a query. n) Will you have to issue a second query this time? If not, what should you do to populate the items in the SECTION frame?
go to contents
a) Which table will the master block be based on? the detail? Why? Answer: The master block will be based on the INSTRUCTOR table and the detail block will be based on the SECTION table. There is a one-to-many relationship between the INSTRUCTOR and SECTION tables. INSTRUCTOR_ID is the primary key in the INSTRUCTOR table and the foreign key in the SECTION table. A master-detail form is used to establish and display a one-to-many (primary key-to-foreign key) relationship between blocks. The primary-key items are always in the master block and the foreign-key items are always in the detail block. In this Exercise, you built a form that displays one instructor and one or many of the sections that this instructor teaches. Therefore, the master block will be based on the INSTRUCTOR table because it is on the one side of the relationship. The detail block will be based on the SECTION table because it is on the many side of the relationship.
Master-detail relationships in forms can also be based on REF columns, which are a type of object column that can be used if your database contains object tables. Object tables are not within the scope of this book, so you will not learn about REF columns here. Refer to the Oracle Forms Reference Manual for more details.
Create the master block and its canvas using the wizards. b) Did you have to do anything to indicate that this was to be a master block? Answer: No.
go to contents
go to contents
self. You would select the items that make up the logical join and the wizard would use those items to write the relation objects join condition. f) Which canvas should you choose for the SECTION block? Is it necessary to create a new one? Answer: You would choose INSTRUCTOR if you had already renamed it, or CANVAS# if it still had the default name. And you wouldnt have to create a new canvas because you want the items in both blocks to appear on the same canvas. g) Which of these items already appears in the INSTRUCTOR frame? Is it necessary to display it again in the SECTION frame? Answer: INSTRUCTOR_ID already appears in the INSTRUCTOR frame. No, it is not necessary to display it again. In most master-detail forms, it would be redundant to display the join item, in this case INSTRUCTOR_ID, in both frames. h) If the layout of the SECTION items is not so pleasing, how can you make quick changes? Answer: Reenter the Layout Wizard and adjust some of the column widths. Run the form and issue a query. i) Have the items in both frames been populated? Answer: Yes they have. j) What happens to the records in the SECTION frame when you scroll to the next record in the INSTRUCTOR frame? Answer: They are coordinated. If you go to the next record in the INSTRUCTOR frame, you see its corresponding records appear in the SECTION frame.
4.1.2
ANSWERS
Use R_INS_SEC.fmb for all of the Exercises in this section. a) Which block is listed first in the Object Navigator? What would be the problem if SECTION were listed first? Answer: It depends on what object or node you had selected in the Object Navigator when you clicked the Create button. If SECTION were listed first, the default navigation would be wrong.
go to contents
go to contents
Answer: The Copy Value from Item property is what coordinates the population of the detail block. When the Form Builder issues a query in the master block, it needs to return corresponding rows to the detail block. This ensures that when a record is queried in the master block, the corresponding records will be brought back to the detail block. d) What form-level triggers have been created? Answer: An ON-CLEAR-DETAILS trigger has been created. A description of this trigger will follow Question e. What block-level triggers have been created? Answer: The ON-POPULATE-DETAILS and ON-CHECK-DELETE-MASTER triggers have been created. A description of these triggers will follow Question e. e) Judging from their names, what do you think these triggers do? Answer: They help the relation coordinate the population of records in the master and detail blocks and manage the deletion of records. These three triggers, along with three program units (CHECK_PACKAGE_ FAILURE, CLEAR_ALL_MASTER_DETAILS, and QUERY_MASTER_DETAILS), are created automatically whenever a relation object is created. This is true whether you use the wizards to create the relation or do it manually. The ON-CLEAR-DETAILS and ON-POPULATE-DETAILS triggers work together to ensure that the records in the detail block correspond to those in the master block. This means that if you are looking at Instructor ID 101 in the master block, then in the detail block you should only see the sections Instructor 101 has taught. The ON-CLEAR-DETAILS trigger fires whenever the user goes from one record in the master block to another. This could be done by scrolling to the next record with a button on the toolbar or by issuing an entirely new query. This trigger simply calls the CLEAR_ALL_MASTER_DETAILS procedure that flushes the records from any of this master blocks detail blocks. So, in simple terms, it clears out the detail block.
go to contents
3)
Run the form. Navigate to SECTION_ID in the SECTION frame and issue a query.
go to contents
Were the SECTION records returned to the form this time? What does the Prevent Masterless Operations property do? Answer: No, the SECTION records were not returned this time. The Prevent Masterless Operations property does not allow you to perform operations on the detail block if a master record is not in the master block.
Specifically, it will not allow the user to query or insert records into the detail block unless there is a record present in the master block. j) Were the items in the SECTION frame populated? Answer: No they were not. k) What then does the Deferred property defer? Answer: It defers or holds off on executing the query for the detail block. In the previous questions, the master block was queried and populated immediately after the detail block was queried and populated. With the Deferred property set to Yes, the form waits for more actions from the user before querying and populating the detail block. l) What happened after you issued the second query? Answer: The detail records were returned to the form. m) How could this be useful? Answer: This can be useful if the user wants to query by example for the detail block, or if the user wants to insert detail records without querying existing records.
I FOR EXAMPLE:
What if a user wanted to see the sections that Instructor ID 101 taught that had a capacity of 15? By setting Deferred to Yes and Auto Query to No, the user can retrieve the Instructor 101 record first, then retrieve all of the sections that had a capacity of 15. Take the following steps to test this: 1) Check that the INSTRUCTOR_SECTION relations Deferred property is still set to Yes and Auto Query is set to No. Run the form. Click Enter Query button on the toolbar and enter 101 in the INSTRUCTOR_ID item.
2)
go to contents
5)
n) Will you have to issue a second query this time? If not, what should you do to populate the items in the SECTION frame? Answer: No, simply navigate to the SECTION frame to populate the items within. In this case, the query and subsequent population of the detail records is still deferred. However, as soon as the user navigates to an item in the detail block, the query is issued and it gets populated. The user does not have to enter more query criteria or explicitly click the Execute_Query button. o) How could this be useful? Answer: This could be useful if the user wants to view the master records before deciding whether or not to view the details.
I FOR EXAMPLE:
Assume that the SECTION block will cause a long-running query. The user may not always want to wait for the query to complete. Here, he can query an instructor record and view it to decide whether or not it is necessary to also see the section records. For the user, this operation is fast and flexible. Fast because the return of the instructor record is not hindered by the slowness of the query for the section records. Flexible because he has a choice of whether or not to view the section records at all.
go to contents
4) Which of the following is not true of the data items in a detail block? a) ____ They are base-table items b) ____ They are copied from the master block c) ____ They can appear on the same canvas as the items in the master block d) ____ It is common to give them a Tabular style layout 5) Which of the following is true about the relation object? a) ____ It is owned by the detail block b) ____ It is owned by the master block c) ____ One of its properties can affect the way records are deleted d) ____ a & c e) ____ b & c 6) What will happen when the Deferred property of a relation is set to No? a) ____ The master block is populated after the detail block b) ____ The detail block is populated along with the master c) ____ The detail block is disabled d) ____ The master and detail blocks are put into Enter Query mode. 7) What will happen when the Prevent Masterless property is set to Yes? a) ____ The detail records will be deleted along with corresponding master records b) ____ The detail blocks operations are put into Normal mode c) ____ The detail block cannot be queried or inserted into unless a master record is present d) ____ The master operations are prevented from coordinating the detail processing Quiz answers appear in Appendix A, Section 4.1.
go to contents
124
CHAPTER
go to contents
C H A P T E R
ITEMS
CHAPTER OBJECTS
In this Chapter, you will learn about: Text Items and Display Items Buttons, List Items, Radio Groups, and Check Boxes Page 126 Page 143
early all of the interaction between users and your forms will take place through items. In this Chapter, you will learn more about how to create items of various types and how to set their properties. You will create database items for items based on columns in a database and nondatabase items to initiate some kind of action or to represent data or information that is not based on a column in a database. Throughout the Exercises that deal with items, you will encounter questions and answers that provide brief tips about GUI design techniques. This is by no means intended to be an exhaustive coverage of the topic. In fact, in many of the Exercises you will be encouraged to ignore layout and aesthetics in favor of concentrating on creating Forms objects. Once you have mastered the basic functions of Oracle Forms, it is recommended that you read the Designing Visually Effective Applications section in the Oracle Forms on-line manuals, or purchase a separate book dedicated to the subject of GUI design such as GUI Design Essentials by Susan Weinschenk, Pamela Jamar, and Sarah Yeo (John Wiley & Sons, 1997).
125
go to contents
LAB
5.1
Create and Define Text Items Without the Wizard Create and Define Display Items
In Chapters 1 through 4, you created and defined some text and display items using wizards. In this Lab, you will go a few steps further by exploring their uses and properties in more detail Display items and text items are fairly similar and share many of the same properties. The biggest difference between the two is that a user can navigate to a text item and change its value. This is not possible with a display item. As its name implies, it merely displays information. Items, especially text and display items, have by far the most properties of all the objects in Forms. It would be impractical and unnecessary to discuss all of them here. Impractical because of the sheer number and unnecessary because so many of the property names are self-explanatory. However, there are several properties in the Functional, Data, and Database property categories that are worth exploring and that you will experiment with in the Exercises.
TEXT ITEMS
Text items are usually database items, meaning they are commonly based on columns in a database. The Data Block Wizard is the easiest tool to use to create these types of items. As you have seen in previous Chapters, it
go to contents
automatically sets properties so that the form knows which database column the text item is based on. Text items can serve as non-database items as well. That is, they do not always have to be based on a column in a database.
DISPLAY ITEMS
As described in Chapter 1, Concepts and Objects, display items can be either database items or non-database items. In this Labs Exercises, the display item you create will be a non-database item. You will use its properties to configure it to perform a calculation. In other situations, you will use display items to display data from other tables.
I FOR EXAMPLE:
Assume you have a block based on the ENROLLMENT table that includes the STUDENT_ID column. Along with the enrollment information, youd also like to display the students last name. You could create a display item to hold the last name and use a trigger to fetch the value from the database. This will be covered in Chapter 6, Triggers & Built-ins. Keep in mind that not all non-database display items have to display the results of calculations or values fetched from other tables. Often, display items are used to provide simple information to the user such as the time, date, or perhaps the name of the database to which the user is connected.
go to contents
Use the Object Navigator to create an item in the COURSE block. Name it COST and position it between DESCRIPTION and PREREQUISITE in the COURSE block. a) Is COST positioned on the COURSE canvas? What one property should you change to place it there? What values has Forms assigned for X Position and Y Position?
b) Instead of manually dragging COST to position it between DESCRIPTION and PREREQUISITE, which of the Layout Editors functions can you use to position it automatically?
Now that you have created and positioned the text item, begin exploring its properties. c) Did you have to adjust the Item Type property when you created this item? What does this tell you about the default behavior of creating items in the Object Navigator?
d) How would the Item Type property have been set if you had created COST in the Layout Editor?
Change the COST items Enabled property to No. Run the form and execute a query.
go to contents
e) How does the appearance of the value in the COST item differ from that of the other items? Can you enter the item via the keyboard or with the mouse? When might you want to use this function?
Exit the form and return to the Form Builder. Change the COST items Enabled property back to Yes. Stay in the Functional category and look at the Multi Line and Word Wrap properties. f) Judging from the names of these properties and the description in the hint line, would it be appropriate to set Multi Line to Yes for COST? Which other item in the COURSE block might it apply to and why?
You will need to use the Database Objects node to answer parts of the following questions. g) What is the data type of the course.cost column? What is the value of the COURSE.COST items Data Type property? What does this tell you about the default behavior of creating items outside of the wizard?
Change the COST items Data Type property to Number. For the next question, you will work with the COURSE_NO item. Select COURSE_NO in the Object Navigator and set its Initial Value property to:
:SEQUENCE.COURSE_NO_SEQ.NEXTVAL
go to contents
Return to the properties for the COST item. Select Format Mask in the Property Palette and press the F1 key on your keyboard. Scroll down in the help screen until you see a section titled Numbers. i) What should you put in the Format Mask property to format COST so that it is displayed like this: $1,195?
Save this form as R_COURSE.fmb as you will be using it again. k) What are some of the things about the layout of the form in Figure 5.1 that make it attractive and easy to read?
go to contents
5.1.2
Use the wizards to quickly create a new master-detail form based on the ENROLLMENT and GRADE tables. Include the audit columns in the blocks, but do not display them on the canvas. For both blocks, Enforce data integrity should be unchecked. Choose Form as the layout style for the ENROLLMENT block. Choose Tabular as the layout style for the GRADE block and display five records. Adjust the widths of the items in the GRADE block so that they fit neatly on the canvas. Name the canvas ENRO_GRAD and the frames ENROLL and GRADE, respectively. In this Exercise, you will create a display item that displays the average grade for each enrollment. Use the Layout Editors Tool Palette to create a display item in the GRADE block and name it GRADE_AVG. Position it below the NUMERIC_GRADE column of items. a) How many GRADE_AVG items are displayed? Remember, GRADE_AVG must belong to the GRADE block.
b) Which property can you change so that the GRADE_AVG item is displayed only once, but the rest of the items in the GRADE block are displayed five times?
go to contents
Lab 5.1: Text Items and Display Items c) Which of the Prompt properties should you change so that the GRADE_AVERAGE prompt is positioned like the prompt for ENROLLMENT.SECTION_ID?
You are going to make GRADE_AVG a calculated item and have it display the average NUMERIC_GRADE. d) Which category of properties will you work with for GRADE_AVG to make it a calculated item?
You will be using a pre-written function to calculate the average of the grades. e) How do you think you should set the Calculation Mode and Summary Function properties?
f) Which item in which block will you be summarizing? Set the associated properties accordingly.
g) What should the data type of the GRADE_AVG item be? Why?
go to contents
Lab 5.1: Text Items and Display Items h) Why do you think you had to set this property?
go to contents
I FOR EXAMPLE:
The STUDENT schema contains SEQUENCES that generate values for columns such as student.student_id and instructor.instructor_ id. When you display these sequence-generated values in forms, you will not want to allow the user to insert or update their values. By setting the Enabled property to No, the user will be able to view the value of student _id or instructor_id, but not change it. Wont a display item provide the same functionality? In a way, yes, in that it will also display information in an item but will prevent the user
go to contents
from accessing it. However, a display item prevents access during Enter Query mode, as well as Normal mode. This means that if STUDENT_ID is a display item, a user cannot use Enter Query mode to search for a student whose ID is 101. However, a text item with Enabled set to Yes is accessible during Enter Query mode. This will allow a user to enter query criteria for the item. Note that there is no Enabled property for display items. f) Judging from the names of these properties and the description in the hint line, would it be appropriate to set Multi Line to Yes for COST? Which other item in the COURSE block might it apply to and why? Answer: No, it would not really be appropriate for COST. It might apply to DESCRIPTION. As their names imply, the Multi Line and Wrap Style properties allow you to create items that can display more than one line. Because a DESCRIPTION can be rather lengthy, it may be appropriate to set its Multi Line property to Yes. If you set Multi Line to Yes, you must remember to manually adjust the Height property of the item if you want more than one line to be visible to the user. The Multi Line and Word Wrap properties are often used when displaying address items. g) What is the data type of the course.cost column? What is the value of the COURSE.COST items Data Type property? What does this tell you about the default behavior of creating items outside of the wizard? Answer: The course.cost column has Number as its data type. The COURSE.COST items Data Type property is set to Char. When you create items outside of the wizard, the item does not inherit any properties from the database and sets every items Data Type property to Char. In this case, the mismatch of data types did not prevent the form from running and functioning properly. However, it is wise to adjust the items Data Type property to match the data type of its base column. In Question i, you will work with format masks to format the appearance of the COST item. If you had left the Data Type as Char, you would have encountered problems when trying to create a format mask. h) How will this affect the COURSE_NO item? Answer: When the user is creating new records, the form will populate the COURSE_NO item with the next value in the sequence. In many applications, sequences will exist in the database and will be used to populate column values. Forms can make use of database sequences by setting the Initial Value property using the following syntax:
go to contents
In the STUDENT schema, there is a sequence called COURSE_NO_SEQ that you can use for COURSE_NO values. There are also sequences for the INSTRUCTOR_ID, SECTION_ID, and STUDENT_ID columns. i) What should you put in the Format Mask property to format COST so that it is displayed like this: $1,195? Answer: You should use $9,999. Format Mask is a powerful and flexible property in that it lets you display information in a format that is different from how the information is stored in the database. You can use format masks to format the display of currencies, Social Security numbers, telephone numbers, product codes, dates, character strings, etc. Format masks can also be used to validate how values are entered into an item.
I FOR EXAMPLE:
Set the COURSE.CREATED_DATE items Canvas property to COURSE. Do not be concerned with where it is positioned on the canvas. Set the COURSE.CREATED_DATE items Format Mask property to DY-DD-MM-YY. Set its Width property to 90. Run the form and issue a query. Note that the date was returned and displayed as indicated in the Format Mask property. In the running form, change the CREATED_DATE value to 12-MAR-99. Tab out of the item and look at the running forms hint line. Note that the error message is indicating the proper format mask. So, not only has the format mask affected the display of the item, but it will also prevent users from entering data in invalid formats. j) Is COST set to be a database item? Answer: Yes it is. The Database Item property is set to Yes. The Database Item property tells Forms that this item is based on a column in the database. Forms will include this items name in whatever SQL statements it issues to the database. Take the following simple steps to get a feel for how the properties you set for an item can affect how Forms builds queries for blocks. 1) 2) Set Database Item to No. Run the form and issue a query.
go to contents
What happened? Note that the COST value was not returned. This is because Forms ignored this item when it issued its SELECT statement to the database. It assumed that COST was a non-database item. 1) 2) 3) Set Database Item to Yes. In the Object Navigator, change the name of COST to V_COST. Run the form and issue a query.
What happened? You got an Unable to Perform Query error because Forms included V_COST in the SQL statement it issued to the database. Because V_COST is not a column in the COURSE table, the database returned an error. 1) 2) Set the Column Name property to COST. Run the form and issue a query.
Note that this time the query worked. The item name V_COST was overridden by the items Column Name property. When you create data items manually, it is wise to set the Column Name property appropriately, even if you name the item after its base column. Note that the other data items in the block that were created by the wizard have this property set. k) What are some of the things about the layout of the form in Figure 5.1 that make it attractive and easy to read? Answer: Read the discussion below. When using an application, it is important that the user can accomplish her tasks quickly and easily without being distracted by the interface. The user should be able to navigate from item to item quickly and smoothly and read and understand the information on the form easily.
I FOR EXAMPLE:
In Figure 5.1, there is plenty of space between each item so that the form does not appear crowded. The items are sized similarly so that their righthand edges are nearly flush. If they were all sized differently, the righthand edge of all the items would not be smooth and would create a distracting, jagged edge. The font is uniform across all of the items and is rather plain. Fancy fonts with serifs are attractive, but they are rather difficult to read and should not be used in Forms applications.
go to contents
5.1.2 ANSWERS
a) How many GRADE_AVG items are displayed? Remember, GRADE_AVG must belong to the GRADE block. Answer: Five GRADE_AVG items are displayed. When the Layout Wizard sets the Number of Records Displayed, it does it for all items in the block being created. Then, all of the items in the block inherit this value. Even though you created GRADE_AVG outside the wizard, it is still inheriting this property from the GRADE block. If only one GRADE_AVG item is displayed, then you have created it in the ENROLL block by accident. What probably happened is that you did not set the value of the Block drop-down properly in the Layout Editors Utility toolbar. If you created GRADE_AVG in the ENROLL block, delete it and try it again using the proper tools in the Layout Editor. b) Which property can you change so that the GRADE_AVG item is displayed only once, but the rest of the items in the GRADE block are displayed five times? Answer: You can change the GRADE_AVG items Number of Items Displayed property to 1. As you can see, by setting an individual items Number of Items Displayed property, you can override the block-level value for that specific item, but you will not affect the other items in the block.
go to contents
The properties under the Prompt category in the Property Palette allow you to set the position of the prompt relative to its item. You can attach the prompt to any of the items four edges, set how far the prompt should be from the item, set how the prompt should be aligned to the item, and so on. At design-time, you can set the values for Prompt Alignment Offset and Prompt Attachment Offset in the Property Palette and Layout Editor. However, Prompt Attachment Edge, Prompt Alignment, Prompt Justification, and the rest of the properties in the Prompt category can only be changed in the Property Palette. The beauty of these Prompt properties is that they will not change if you reposition an item. What this means is that if you drag an item from one position on the canvas to another, its prompt will be dragged along with it. Not only will it accompany the item across the canvas, but its position relative to the item will stay the same. This saves you from having to reposition the prompt every time you reposition an item. d) Which category of properties will you work with for GRADE_AVG to make it a calculated item? Answer: Calculation. e) How do you think you should set the Calculation Mode and Summary Function properties? Answer: Calculation Mode should be set to Summary and Summary Function should be set to AVG. A calculated item gets its value from either an existing summary function, like AVG or SUM, or from a formula. Summary functions are convenient because like pre-existing database functions, the mathematical expression is already written for you. In this example, since you wish to calculate the average of the grades for each enrollment, the Calculation Mode property should be set to Summary and the Summary Function property should be set to AVG.
go to contents
I FOR EXAMPLE:
Assume you created an item called NO_OF_ENROLL that contained the number of students enrolled in a given section. You included NO_OF_ENROLL in a SECTION block. Now you want to create another item called SEATS_LEFT in the SECTION block. In this item you want to display the number of seats remaining in the section. You would calculate this by subtracting SECTION.NO_OF_ENROLL from SECTION.CAPACITY. In the properties for the SEATS_LEFT item, you would set Calculation Mode to Formula and you would write the following expression in the Formula property: :SECTION.CAPACITY - :SECTION.NO_OF_ENROLL At run-time, the results of this formulaic expression would be displayed in the SEATS_LEFT item. f) Which item in which block will you be summarizing? Set the associated properties accordingly. Answer: You will be summarizing GRADE.NUMERIC_GRADE. Therefore, Summarized Block should be set to Grade and Summarized Item should be set to Numeric Grade. g) What should the data type of the GRADE_AVG item be? Why? Answer: The Data Type property should be set to NUMBER since the calculation will produce a number value. If you leave GRADE_AVG as CHAR, the form will return an error. h) Why do you think you had to set this property? Answer: So that the average is computed for all of the records in the querys result set. By default, Forms does not always return all of the records in a result set to the form at once. Each block has a Query Array Size property, which determines how many records will be fetched from the database at a time. The default value of this property is set by the Number of Records Displayed property. What this means is that by default, the number of records returned to a block is equal to the number of records displayed on
go to contents
the canvas. For instance, in the GRADE block, only 5 records are displayed on the canvas, so the Query Array Size property is also set to 5. What if 10 or 20 records are part of the result set? Forms would not be able to calculate the average correctly because not all of the values would be returned. Therefore, when creating summary items, it is necessary to set the Query All Records property to Yes. By doing so, the Query Array Size property is overridden, all records are returned to the form, and the average is computed accurately. In this example, it is safe to return all of the records to the form because the result sets are rather small. However, there may be cases in which the result set could be rather large. This would mean that setting Query All Records to Yes could possibly degrade the performance of the application. In these cases, you would set the properties for the GRADE block a bit differently. First, you would set Query All Records to No. Then, under the Advanced DML category in the Property Palette, you would set the Pre Compute Summaries property to Yes. This will not return all of the records to the block at once. Instead, the number of records returned to the block will be based on the number set by the Query Array Size property. But wont this mean that the average for the calculated item will be incorrect? Not in this case, because Forms will issue a second query to figure out the average. This way, the average is being computed by the database and then returned to the form. The value will be correct and the form will not have to fetch all of the rows from the database.
go to contents
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
143
LAB
5.2
LAB 5.2
Create Buttons Put Simple Code Behind Buttons Create List Items Create Radio Groups Create Check Boxes
The items in this Lab are different from text items and display items in that they dont simply display text to communicate their value. Instead they employ different combinations of text and graphics to display information or provide a function. As you may have noticed, there are many other item types available in Forms such as images, sounds, ActiveX Controls, and so on. In this Lab and in the rest of this book, you will focus on buttons, list items, radio groups, and check boxes. These, along with text and display items, are by far the most common Forms item types. For a complete list of all the different item types, simply look at the Item Type property in the Property Palette and refer to the help system for details on how to implement them.
go to contents
144
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
BUTTONS
Buttons give users opportunities to make a form do something. This could be something simple like saying OK, or it could be something more involved like executing a query or even opening another form. Creating and positioning buttons in Forms is easy. Whats challenging is writing the code that goes behind them. As you already know, each button should have a WHEN-BUTTON-PRESSED trigger associated with it so that it can respond to the Button Pressed event. In the Exercises, you will use the help system to locate Forms built-ins to put behind your buttons.
LAB 5.2
I FOR EXAMPLE:
The list item in Figure 5.2 shows three choices for the SECTION tables LOCATION column. The labels are Lecture Hall One, Lecture Hall Two, and Lab One. But, they may not necessarily be this way in the database. The column values could be L210, L500 and L510, respectively. The list has been configured so that the user sees values that are more meaningful to them. What this illustrates is that radio groups, list items, and check boxes allow you to display information in any way youd like, regardless of how the values are stored in the database. When you configure these items, you define the values youd like to display, along with how they should be represented in the database. These three types of items can also handle data that they are not designed to expect.
Figure 5.2 I The list item elements are not necessarily how the information is stored in the database.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
145
I FOR EXAMPLE:
At design-time, the Location list item in Figure 5.2 could be configured to handle the database values L210, L500, and L510. When the user issues a query, the list item would expect that one of these three values would be returned to the form. But, what would happen if the user issued a query and the database returned the value L999? The list item is not expecting this value and would not know what to do with it. Luckily there is a list item property called Mapping of Other Values that can be set to handle this situation. Radio groups and check boxes have a similar property. In the Exercises in this Lab, you will explore the Mapping of Other Values property in more detail.
LAB 5.2
LIST ITEMS
List items are exactly what their name impliesa short list of values that the user can choose from. List items can serve as either database or non-database items. The list item you create in the Exercise will be based on the COST column in COURSE table. You can also use list items for selection lists that are not bound to columns in the database but instead act as controls. The Form Builder makes use of list items in this way in the Layout Editors Utility toolbar as shown in Figure 5.3.
RADIO GROUPS
Radio groups are logical containers of radio buttons. Radio buttons are the small circles that appear on the screen which the user can click to select values. Radio group selections are mutually exclusive. That is, selection of one radio button deselects the previously selected button. The radio group you will build in the Exercises will be a database item. However, radio groups can also be non-database items.
I FOR EXAMPLE:
You often see radio groups used in the Form Builder itself to let the user make choices. Figure 5.4 shows the New Data Block dialog with a radio group with two radio buttons.
go to contents
146
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
LAB 5.2
Figure 5.4 I The New Data Block dialog with a radio group.
In this case, the radio group is acting as a control, much like a button item acts as a control. It gives a user the opportunity to choose how the application should behave. If you were to employ such a radio group in one of your Forms applications, you would have to write triggers to change the behavior of the form depending on which radio button was selected.
CHECK BOXES
Check boxes are useful for storing Yes/No, True/False, and On/Off-type values. Like radio groups and list items, they can also serve as both database items and non-database items. In the Form Builder, there are multiple examples of check boxes as non-database items. Figure 5.5 shows a check box on the welcome page of the Data Block Wizard.
Figure 5.5 I The welcome page of the Data Block Wizard, which includes a check box.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
147
LAB 5.2
b) What should you set on the Layout Editors toolbar to make sure that these buttons are assigned to the CONTROL block?
148
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes c) Edit the properties so that the text on the button matches its name in the Object Navigator. Which property did you use?
LAB 5.2
Run the form. d) Which item has the Forms Runtime navigated to (where is the cursor)? Why is Forms doing this?
Exit the running form and return to the Form Builder. e) What should you do to prevent the button from being navigated to first?
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
149
b) What is the title of the window that has just opened and what should you do in it?
You will need to use a Forms built-in to exit the form. Use the help system to determine which one you need. From the Form Builders Main Menu, select Help | Form Builder Help Topics. Click the Index tab and type the word Exit into the first field. c) What is the name of the built-in you should use to exit the form?
LAB 5.2
Type the code for the built-in into the PL/SQL Editor. Put a semi-colon after the built-in so that it compiles correctly. Click the PL/SQL Editors Compile button. Close the PL/SQL Editor. Run the form and test the Exit button. Create the same trigger for the Clear button using the CLEAR_BLOCK built-in as the trigger code. Click the PL/SQL Editors Compile button. The Save button on this form is not for saving files, but for saving changes a user has made by inserting or updating records to the database. With this in mind, answer the following questions. d) What is the command for saving updated or inserted records to the database?
Based on the answer to the previous question, search the help system again. e) Which built-in should you use to save the changes a user has made?
go to contents
150
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
Type the code for the built-in into the PL/SQL Editor and click Compile. Close the PL/SQL Editor. Run the form and issue a query. Test the Save button by making a change to one of the records returned. Test the Clear button as well. Save the R_COURSE.fmb form for future Exercises.
LAB 5.2
Read the answers and discussions for Exercises 5.2.1 and 5.2.2 before continuing to the next Exercise.
Change the COURSE.COST items Item Type property to List Item. Use the Freeze/Unfreeze button you learned about in Chapter 3, The Development Environment, to compare the Functional properties of COURSE.COST to those of COURSE.COURSE_NO. b) What list-related properties have been added to the Functional category of COURSE.COST?
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
151
c) What happens when you click the Elements in List property and then click the More button?
d) If list elements are what appear to the user, what are list item values used for?
LAB 5.2
e) Can a list element be different from its list item value? When and why might you want them to be different?
Assume that there are only three possible costs for courses in the STUDENT database (1095, 1195, 1595). Use these values to populate the List Element and List Item Value fields. f) Can you format list elements with dollar signs and commas?
g) Which Data property would you set so that a list item would start at 1595 during an insert?
Run the form and query Course Number 100. h) Can you change the COST to 2000?
go to contents
152
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
Exit the form and return to the Form Builder. Change the COST items List Type property to Combo Box. Run the form and query Course Number 100 again. i) Can you change the COST to 2000 now?
LAB 5.2
j) Can you change the COST to $2,000 and save the change? Why not?
Exit the form and return to the Form Builder. Change the COST items List Type property to Tlist. k) Does COST still look like a list item? What does it look like now?
Change the COST items Height property to 34. Run the form. l) What has happened to COST? How many of its elements are visible?
m) If there were 15 values for the COST item, would it still be appropriate to use a list item? What if there were 20? 30?
Read the answers and discussions for Exercise 5.2.3 before continuing to the next Exercise.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
153
LAB 5.2
In this Exercise, you will use the COST item again, but this time you will implement it as a radio group item. Change the COST items Item Type property to Radio Group. a) Is COST still visible on the COURSE canvas? Is it still in the COURSE block? Why isnt it still visible on the canvas?
b) What are the nodes under the COST item in the Object Navigator?
c) Can you create radio buttons in the Object Navigator? Why might it be easier to create them in the Layout Editor?
Select the Radio Button tool from the Layout Editors Tool Palette and create three radio buttons. Use the pinning feature you learned in Chapter 3, The Development Environment, to make this easier. Update the layout when you are done. Change the background color of the buttons to match the canvas. Readjust the position of the Save, Clear, and Exit buttons if they are in the way.
go to contents
154
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes d) What else did you have to decide in addition to selecting the position of the radio buttons?
LAB 5.2
Assume that there are only three possible costs for courses in the STUDENT database (1095, 1195, 1595). Rename the radio buttons in the Object Navigator. e) Are you able to rename them 1095, 1195, 1595? Why?
f) Which radio button property should you change so that $1,095, $1,195, and $1,595 appear next to each radio button respectively?
g) Which radio button property should you use to indicate its value?
Assume that a query returns a value to the COST radio group that is not one of its pre-defined values. h) What Functional radio group property can handle this situation?
i) Does the COST radio group have a property for labeling or titling the group?
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
155
j) Can you create a graphic object to indicate to the users that this group of radio buttons describes the courses cost? Which ones can you use?
LAB 5.2
k) If there were 6 values for the COST item, would it still be appropriate to use a radio group? What if there were 10? 20?
b) If the allowable values for DROP_LOWEST are Y and N (Yes and No), what should you enter for Value When Checked? Value When Unchecked? Why?
go to contents
156
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes c) What are the three values of the Check Box Mapping of Other Values property? Give a brief description of what you think each means.
LAB 5.2
Set the DROP_LOWEST items Check Box Mapping of Other Values property to Not Allowed. Run the form. d) What was the error you received just before the Forms Runtime started? If you were unable to see the error message, minimize the Forms Runtime and navigate to the Form Builder.
Set the DROP_LOWEST items Check Box Mapping of Other Values property to Checked and run the form to confirm that the error has been corrected. Set the DROP_LOWEST items Width property to 90. Set the DROP_LOWEST items Label property to Drop Lowest Grade. e) What is the difference between the DROP_LOWEST items Prompt property and its Label property?
f) Are there any check box-related triggers? Check the Smart Triggers to find your answer.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
157
LAB 5.2
I FOR EXAMPLE:
Select the Save button in R_COURSE.fmb and locate its Iconic property. Change Iconic to Yes, and change Icon Name to Save. Look at the Save button in the Layout Editor and note how the text has been replaced by an icon. If you were to use this icon, you would obviously have to adjust the size and position of the button to make it look better. But for now, simply leave it as it is. When you installed Oracle Developer, a number of sample icon files were installed along with it. They should be in your \ORACLE_HOME\tools\ devdem60\bin\icon directory. The Form Builder and Forms Runtime are able to find these icon files because an entry in the Windows Registry points to this directory. For more information about the Registry, see Appendix B. d) Which item has the Forms Runtime navigated to (where is the cursor)? Why is Forms doing this? Answer: Forms has navigated to the Save button because it is listed first in the Object Navigator.
go to contents
158
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
Exit the running form and return to the Form Builder. e) What should you do to prevent the button from being navigated to first? Answer: You should reposition the CONTROL block after the COURSE block.
LAB 5.2
This will prevent the form from navigating to these buttons before it navigates to the enterable items in the COURSE block. You should also set the Mouse Navigate button to No. Users will still be able to click the buttons, but the cursor will not rest on a button; it will return to the previous item. If the cursor can rest on a button, it may prevent the user from performing some operations. A form, for instance, will not allow a user to issue a query if the cursor is resting on a button.
5.2.2 ANSWERS
a) Which trigger should you create to respond to the event of a user clicking this button? Answer: The WHEN-BUTTON-PRESSED trigger. b) What is the title of the window that has just opened and what should you do in it? Answer: The PL/SQL Editor. The PL/SQL Editor is the tool you will use to write trigger and procedure code in Oracle Forms. It will be discussed in more detail in Chapter 6, Triggers & Built-ins. c) What is the name of the built-in you should use to exit the form? Answer: You should use the EXIT_FORM built-in.
Figure 5.7 I The PL/SQL Editor with the EXIT_FORM built-in. Note the semi-colon following the built-in.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
159
Forms built-ins, as stated before, will be discussed in Chapter 6. For now, simply type the built-in into the PL/SQL Editor and click the Compile button. If you are having problems, refer to Figure 5.7 to see how the code should look in the PL/SQL Editor. d) What is the command for saving updated or inserted records to the database? Answer: You commit updated or inserted records to the database. e) Which built-in should you use to save the changes a user has made? Answer: The COMMIT_FORM built-in. Type the code for the built-in into the PL/SQL Editor and click Compile. Close the PL/SQL Editor. Run the form and issue a query. Test the Save button by making a change to one of the records returned. Test the Clear button as well. Save the R_COURSE.fmb form for future Exercises.
LAB 5.2
5.2.3
ANSWERS
a) Which text item(s) in the COURSE block would be better displayed as a list item(s)? Answer: COURSE.COST might be better displayed as a list item because it has only three allowable values. b) What list-related properties have been added to the Functional category of COURSE.COST? Answer: List, List Style, and Mapping of Other Values have been added to the Functional category. c) What happens when you click the Elements in List property and then click the More button? Answer: You are presented with a List Elements dialog box. This is the main tool for configuring a list item. d) If list elements are what appear to the user, what are list item values used for? Answer: The list item values are how each item in the list is represented in the database. Each element in a list is represented in two ways: to the user as a list element and to the database as a list item value. Every list element must have a corresponding list item value. The List Elements dialog box lets you set both of these values.
go to contents
160
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
LAB 5.2
Figure 5.8 I The completed List Elements dialog box for COURSE.COST.
I FOR EXAMPLE:
For the list item COURSE.COST, you will have three list elements: 1095, 1195, and 1595. The list elements are what the user will see in the list item at run-time. These list elements will have corresponding list item values of 1095, 1195, and 1595, respectively. The list item values are what will be inserted or updated to the database when the user makes a change. Figure 5.8 shows the completed List Elements dialog box for COURSE.COST. Note that in Figure 5.8, the 1195 list element is selected, so its corresponding value appears in the List Item Value field. The values in both fields are the same now, but as you move through the Exercises, they will change. e) Can a list element be different from its list item value? When and why might you want them to be different? Answer: Yes. The ability to use list items to represent elements one way and then send and retrieve their values from the database in another is what makes list items convenient and powerful.
I FOR EXAMPLE:
The GRADE_TYPE table contains a GRADE_TYPE_CODE column. The values in this column are FI for final, QZ for quiz, HW for homework, and so on. If you were to create a list item based on the GRADE_TYPE_CODE column, it would make sense to represent the values differently from how they are stored in the database. For example, you would have Final as a list element with FI as its corresponding list item value. Refer to Figure 5.9 to see how this would look in the List Elements dialog box.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
161
LAB 5.2
Figure 5.9 I List elements with different list item values. Note that Final is selected.
The Functional category has the Mapping of Other Values property. As discussed in the Lab text, its purpose is to handle values that are returned from the database that dont match any of those entered as list item values. That is, if a value that is not one of the list item values is returned from the database, it will be mapped to the value indicated in the Mapping of Other Values property.
I FOR EXAMPLE:
Assume that in the GRADE_TYPE_CODE example, FI, QZ, and HW are the only values entered as list item values. Also assume that the Mapping of Other Values property is set to QZ. What would appear in the list item if the user queried a record in the GRADE_TYPE table that had TEST as the value for GRADE_TYPE_CODE? TEST does not match any of the list item values. But, since Mapping of Other Values was set to QZ, the list item would display QZs corresponding list element. Then, when this record is resaved to the database, the column will be updated with the QZ value. f) Can you format list elements with dollar signs and commas? Answer: Yes you can. This is the same as changing FI to Final in the answer discussion for Question e. The list element does not have to match its list item value and can contain any combination of characters. g) Which Data property would you set so that a list item would start at 1595 during an insert? Answer: You would set the Initial Value property to 1595.
go to contents
162
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
If the user is creating a new record, the value in the list item will initially be set to 1595. The value in the Initial Value property must be one of the list item values that were entered in the List Elements dialog. h) Can you change the COST to 2000?
LAB 5.2
i)
Answer: No you cant. There is no way to add values that are not on the list. Can you change the COST to 2000 now? Answer: Yes you can. There are three types of list items: poplists, Tlists, and combo boxes. In Questions a through g, you worked with poplists, which are the default type. They restrict the user from selecting only from the list elements that are displayed. A combo box is nearly identical to a poplist in look and functionality except that it allows users to enter values that are not defined on the list. For example, in this case you were able to change the COST to 2000, even though it was one of the list elements. j) Can you change the COST to $2,000 and save the change? Why not? Answer: No, because COST is a numeric column in the database and will not accept the dollar sign. k) Does COST still look like a list item? What does it look like now? Answer: No, it looks like a regular text item with a miniature scrollbar. l) What has happened to COST? How many of its elements are visible? Answer: All of its elements are visible in a list. m) If there were 15 values for the COST item, would it still be appropriate to use a list item? What if there were 20? 30? Answer: Yes, it would still be appropriate with 15 values, but not 20 or 30. A good rule of thumb is to limit the number of values in a list item to 15. If the list grows much larger than 15, you should consider using a List of Values object, which you will learn about in Chapter 7, LOVs and Alerts. For more information on the design concerns for list items, refer to the Oracle Developer 6.0 On-line Manuals in the section titled Designing Effective GUI Applications.
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
163
5.2.4 ANSWERS
a) Is COST still visible on the COURSE canvas? Is it still in the COURSE block? Why isnt it still visible on the canvas? Answer: No, it is not visible on the COURSE canvas. Yes, it is still in the COURSE block. A radio group is a logical container of radio buttons. Therefore, the group itself is not represented on the canvas because it has no physical properties. For this radio group, you will create radio buttons that have physical properties and are visible on the canvas. The radio buttons serve as a visual representation of the choices in the radio group. In this Exercise, you have already created a radio group called COURSE.COST. Soon you will create three radio buttons to represent the values 1095, 1195, and 1595. b) What are the nodes under the COST item in the Object Navigator? Answer: The Triggers and Radio Buttons nodes are under the COST item. The Radio Button node is positioned here in the Object Navigators hierarchy so that the Form Builder will know which radio buttons are logically contained by which radio groups. c) Can you create radio buttons in the Object Navigator? Why might it be easier to create them in the Layout Editor? Answer: Yes, you can create them in the Object Navigator. However, it would be easier to create them in the Layout Editor so that you can position them as you create them. d) What else did you have to decide in addition to selecting the position of the radio buttons? Answer: You had to select which radio group they would be assigned to by using the Radio Groups dialog. In this case, you only had one radio group created so the choice was simple. If you had created more than one radio group in this form, they would have been listed here. Also note that you could have created a new radio group using this dialog. e) Are you able to rename them 1095, 1195, 1595? Why? Answer: No, because Forms will not accept numbers as names of objects.
LAB 5.2
go to contents
164
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
You must prefix the numbers with something like COST_ or RB_ if you would like the numbers to be included in a radio buttons name. f) Which radio button property should you change so that $1,095, $1,195, and $1,595 appear next to each radio button respectively? Answer: You should use the Label property. Radio buttons are similar to button items in that they have both Label and Prompt properties. Again, it is customary to use Label for radio buttons rather than Prompt. g) Which radio button property should you use to indicate its value? Answer: You should use the Radio Button Value property. h) What Functional radio group property can handle this situation? Answer: The Mapping of Other Values property. If the database returns a value that does not match one of the values for one of the radio buttons, the form will return an error. However, if the Mapping of Other Values property is set, the unmatched value will be handled by the form.
LAB 5.2
I FOR EXAMPLE:
Assume that the radio button values are set to 1095, 1195, and 1595. Also assume that the Mapping of Other Values property is set to 1095. If the database returns a value of 9999 to the form, then the 1095 radio button will be selected as is indicated in the Mapping of Other Values property. When the user commits changes to this record, the COST column will be updated to 1095. i) Does the COST radio group have a property for labeling or titling the group? ANSWER: No, it does not. j) Can you create a graphic object to indicate to the users that this group of radio buttons describes the courses cost? Which ones can you use? Answer: You can use text or a frame.
Figure 5.10 I The Cost radio group with a textual label. go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
165
LAB 5.2
5.2.5
ANSWERS
a) Which of the DROP_LOWEST items Functional properties allow you to set the value that will be inserted or updated to the database? Answer: The Value When Checked and Value When Unchecked properties. The check box has two states, checked and unchecked, that correspond to the values Yes/No, True/False, and On/Off. In the checked state, the user will see a small check mark in the check box. In the unchecked state, the check box will be empty. b) If the allowable values for DROP_LOWEST are Y and N (Yes and No), what should you enter for Value When Checked? Value When Unchecked? Why?
go to contents
166
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
Answer: Y for Value When Checked, N for Value When Unchecked. A Forms check box is analogous to the check boxes you have seen on paper-based forms. On paper-based forms, you often see check boxes for things like Add me to your mailing list, Insured? Criminal Record?, and so on, which require that you check the box if you meet the condition stated. The behavior is the same in Forms. The Value When Checked property typically corresponds to the positive or affirmative state, while Value When Unchecked corresponds to the negative state. So in this case, by checking DROP_LOWEST, the user is indicating Yes, the lowest grade for this type should be dropped. Forms will insert a Y into the DROP_LOWEST column for this record. c) What are the three values of the Check Box Mapping of Other Values property? Give a brief description of what you think each means. Answer: Checked, Unchecked, and Not Allowed. The Check Box Mapping of Other Values is similar to the Mapping of Other Values function in list items and radio groups. It is used to determine how to handle those values returned from the database that do not match either of the values you have assigned to the check box in the Value When Checked or Value When Unchecked properties.
LAB 5.2
I FOR EXAMPLE:
Assume that for the DROP_LOWEST check box, the Value When Checked property and Value When Unchecked property are set to Y and N respectively. Also assume that a user using a different application has inserted an M for Maybe into the DROP_LOWEST column for one of the records in the GRADE_TYPE_WEIGHT table. When a user queries this record, the form will behave in the following ways depending on how the Check Box Mapping of Other Values property is set: 1) 2) 3) CheckedThe check box will be set to the checked state when M is returned. UncheckedThe check box will be set to the unchecked state when M is returned. Not AllowedThe form will not be able to retrieve the record. An FRM-40301 error will be returned and appear with a message in the hint line.
In Situations 1 and 2, the form will accept the M value and let the user continue processing the record even though it does not match either of the values specified in the Value When Checked or Value When
go to contents
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
167
Unchecked properties. When the record is saved, the DROP_LOWEST column will be updated with the value in the Check Box Mapping of Other Values property. In Situation 3, the form will reject the M value and not allow the user to process the record. d) What was the error you received just before the Forms Runtime started? If you were unable to see the error message, minimize the Forms Runtime and navigate to the Form Builder. Answer: The message reads FRM-30188: No initial value given, and other values are not allowed (item GRADE_TYPE_WEIGHT .DROP_LOWEST). If you decide to set Check Box Mapping of Other Values to Not Allowed, then you must specify an Initial Value for the check box. This Initial Value must match one of the values specified in the Value When Checked or Value When Unchecked properties. e) What is the difference between the DROP_LOWEST items Prompt property and its Label property? Answer: The Prompt propertys value appears to the left of the item, while the Label propertys value appears to the right. As with button items and radio buttons, it is customary to use the Label property rather than the Prompt property. f) Are there any check box-related triggers? Check the Smart Triggers to find your answer. Answer: Yes, there is a WHEN-CHECKBOX-CHANGED trigger. The following is optional. To test the behavior of the check box as it responds to a WHEN-CHECKBOX-CHANGED trigger, take the following steps: 1) 2) 3) Right-click on the DROP_LOWEST item. Navigate to Smart Triggers and select the WHEN-CHECKBOXCHANGED trigger. The PL/SQL Editor will open. Enter the following code:
LAB 5.2
IF GRADE_TYPE_WEIGHT.DROP_LOWEST = 'Y' THEN MESSAGE('The lowest grade will be dropped'); ELSE MESSAGE('The lowest grade will not be dropped'); END IF;
go to contents
168
Lab 5.2: Buttons, List Items, Radio Groups, and Check Boxes
4) 5) Run the form and issue a query. Toggle DROP_LOWEST and watch the Forms Runtimes status line.
go to contents
169
CHAPTER
go to contents
C H A P T E R
racle Forms applications come equipped with a significant amount of default processing. That is, when events occur, there is always default code that responds. Quite often this default processing is not enough to give the application the functionality your users require. The problem is that you cannot directly access and edit the code for the default processing to make it do what you want. This is why you write triggers: to complement, augment, or replace this default processing. In this Chapter, you will delve a bit deeper into trigger and event concepts. You will learn how and when triggers fire, and how they fire in relation to each other. You will also write triggers of your own. PL/SQL language, syntax and structure will not be discussed here, since it is assumed that you have already had ample experience writing database stored procedures, functions, and triggers.
171
go to contents
LAB
6.1
TRIGGER BASICS
LAB OBJECTIVES
After this Lab, you will be able to:
Use PL/SQL and SQL in Triggers Understand Trigger Scope Categorize Triggers
In the simplest terms, a trigger contains PL/SQL code that responds to Forms events. You have already been exposed to many triggers and the events that fire them in the previous Labs and Exercises, so you have a general idea of how they work. One of the advantages of using Oracle Forms and an Oracle database together is that the PL/SQL programming language is used in both of them. So, if you have already written packages and procedures for the database, then you already know how to write triggers in Forms. But, before you begin writing triggers, it is necessary to understand when they fire and how they are organized.
TRIGGER SCOPE
Triggers are always attached to other objects. The level of the object in the Forms hierarchy helps determine the scope of the attached trigger(s). The ON-POPULATE-DETAILS trigger in Figure 6.1 is defined at the block level. It is attached to the COURSE block in the Object Navigator and will only fire in response to events within the scope of the COURSE block. Triggers can be attached to items and forms as well. Triggers at the item level fire in response to events within the scope of their respective items. Form-level triggers fire in response to events within the scope of the form.
go to contents
Certain trigger types can be attached at the item level, the block level, or the form level. You can attach a trigger at higher levels of the Forms hierarchy to increase its scope.
I FOR EXAMPLE:
In previous Exercises, you attached WHEN-BUTTON-PRESSED triggers to individual items. You created a button item called EXIT and attached a trigger directly to that item which fired code to exit the form. The triggers scope was limited to the EXIT button. That is, the trigger only fired in response to the Button Pressed event of the EXIT item (button). It is possible to attach a WHEN-BUTTON-PRESSED trigger at the block or form level. What does this mean? Assume you have a CONTROL block with five buttons and you assign a WHEN-BUTTON-PRESSED trigger at the block level. When will the trigger fire? It will fire in response to a Button Pressed event for any of the buttons in the block. What does this mean? It means the scope of the trigger is now at the block level rather than only at the item level.
CATEGORIES OF TRIGGERS
The Forms help system categorizes triggers in two ways: by name and by functional category. Understanding the two methods of categorization will help you understand when and why certain triggers fire, which will
go to contents
You would choose the appropriate trigger from one of these categories depending on what you want your own trigger code to do and how you want Forms to handle its own default processing.
I FOR EXAMPLE:
Assume you wanted to write some code to respond to the Commit Transactions event which fires each time a form tries to insert a record. There are a number of insert-related triggers to choose from, including ON-INSERT, PRE-INSERT, and POST-INSERT. Do you want to replace Forms default insert processing and write all of the insert logic yourself? In that case, use an ON-INSERT trigger. Do you want to fire some of your own logic just before Forms executes its default insert processing? In that case, you would use a PRE-INSERT trigger. Or, you might want to use a POST-INSERT trigger to fire just after Forms has completed its default processing.
go to contents
Query triggers, which respond to events regarding queries. Validation triggers, which respond to events regarding the validation of items and records. Transactional triggers, which respond to events regarding inserting, updating, and committing of records. Key triggers, which respond to Key Press events.
Each trigger falls into both a named and a functional trigger category.
I FOR EXAMPLE:
The ON-UPDATE trigger falls into both the On event trigger named category and the Transactional functional trigger category.
b) Look at the SQL statement that defines the c_val_zip cursor. How are the block and item expressed?
c) How would you write an SQL statement to select DESCRIPTION from the COURSE table into a DESCRIPTION display item in a SECTION block? The DESCRIPTION value you select should correspond to the COURSE_NO value that is currently in the form.
go to contents
Look at the STUDENT.EXIT buttons WHEN-BUTTON-PRESSED trigger. d) Is this still PL/SQL? Why or why not?
b) For EX06_01.fmb, at which levels within the Forms hierarchy are there WHEN-VALIDATE-ITEM triggers?
Run the form. Type Mr. into the SALUTATION item and then press the TAB key. Look at the status line for the Forms Runtime. c) Which of the WHEN-VALIDATE-ITEM triggers fired? What does this tell you about the firing order of triggers?
go to contents
d) Are there any WHEN-VALIDATE-ITEM triggers at the item level for FIRST_NAME and LAST_NAME?
Go back to the Forms Runtime. Take the following two actions and watch the status line after each. Type Joe into the FIRST_NAME item and press the TAB key. Type Smith into the LAST_NAME item and press the TAB key. e) Which trigger fired? Why do you think this happened?
Exit and close the Forms Runtime and return to the Form Builder. View the properties for the SALUTATION items WHEN-VALIDATE-ITEM trigger. Change the Execution Hierarchy property to Before. Run the form, type Mr. in the SALUTATION item and press the TAB key. Read the alert message, then click the OK button. Take note of the status line. f) How has changing the Execution Hierarchy property affected the form?
Select the Triggers node under the STUDENT.SALUTATION item and click the Create button in the Object Navigator. g) Can you create a PRE-FORM trigger here? Why not?
go to contents
b) Is it mandatory that you write triggers to respond to each event? What happens if you dont?
In the Exercises for Lab 6.2, you will create display items called CITY and STATE for a block based on the STUDENT table. You will also write a trigger to populate the CITY and STATE items with values that correspond to the value that has been fetched into the ZIP item. c) When should the trigger you write populate these items? Before or after the query is issued?
d) Based on your answer to Question c, what trigger should you create? At what level should you attach it to the form? Search the help system if you are having trouble with this question.
go to contents
Also in the Exercises for Lab 6.2, you will write another trigger to check that the value a user has entered into the ZIP item is valid in that it exists in the ZIPCODE table. e) Which trigger should you create? What named and functional categories does this trigger belong to?
In the Exercises for Lab 6.3, you will write two triggers to set the values for the audit columns. These triggers will assign values to CREATED_BY, CREATED_DATE, MODIFIED_BY, and MODIFIED_DATE so that they can be inserted or updated to the database. f) Should these triggers fire before or after the inserts and updates are issued?
g) Based on your answer to Question f, which triggers should you choose and what are their functional and named categories?
go to contents
I FOR EXAMPLE:
The WHEN-VALIDATE-ITEM could have been written a little differently, in which case, it would have had to include an exception handler. Instead of using a cursor, a simple SQL statement could have been used to fetch rows from the database. Therefore, the trigger could no longer use the cursor attribute %NOTFOUND to detect invalid records. It would have to include an exception instead. The code would look like this: BEGIN SELECT city, state INTO :STUDENT.CITY, :STUDENT.STATE FROM zipcode WHERE zip = :STUDENT.ZIP; EXCEPTION WHEN NO_DATA_FOUND THEN MESSAGE('Zipcode does not exist in Zipcode table.'); RAISE FORM_TRIGGER_FAILURE; END;
This trigger is using the pre-defined NO_DATA_FOUND exception to handle instances in which no rows are returned for the SELECTINTO statement. You are not restricted to pre-defined exceptions, however. You can create your own user-defined exceptions in Forms triggers just as you have in standard PL/SQL stored procedures. b) Look at the SQL statement that defines the c_val_zip cursor. How are the block and item expressed? Answer: The block and item are expressed as : STUDENT.ZIP. Whenever you want to refer to an item and its block in an SQL statement in a trigger, you must express the reference using the following syntax: :block.item
go to contents
Note that this applies to the SELECTINTO section of the SQL statement as well as the WHERE clause. c) How would you write an SQL statement to select DESCRIPTION from the COURSE table into a DESCRIPTION display item in a SECTION block? The DESCRIPTION value you select should correspond to the COURSE_NO value that is currently in the form. Answer: See description below. SELECT description INTO :SECTION.DESCRIPTION FROM course WHERE course.course_no = :SECTION.COURSE_NO.
Look at the STUDENT.EXIT buttons WHEN-BUTTON-PRESSED trigger. d) Is this still PL/SQL? Why or why not? Answer: Yes it is. The EXIT_FORM statement is a Forms built-in. Even though there are no BEGIN or END statements listed here, this is still PL/SQL. If there is nothing to declare in the DECLARE statement, then it is not mandatory that you include a BEGIN and an END statement. You can simply issue a series of PL/SQL executable commands. The STUDENT.EXIT buttons WHENBUTTON-PRESSED trigger has only one line, which is a simple call to a Forms built-in. It is possible to have more involved PL/SQL triggers that still do not include a BEGIN or an END statement.
6.1.2 ANSWERS
Have form EX06_01.fmb open in the Form Builder. STUDENT.SALUTATION and STUDENT.ZIP both have WHEN-VALIDATEITEM triggers. a) When an item needs to be validated, which of these triggers will fire? Will Forms simply fire both? Answer: The trigger that is attached to the item that is being validated will fire. The trigger that fires is determined by the scope of the event. The Validate Item event will occur at a specific item. Therefore, only that items
go to contents
go to contents
The block-level WHEN-VALIDATE-ITEM trigger fired in both of these cases because there was no WHEN-VALIDATE-ITEM trigger at the item level. The Validate Item event occurred, so Forms went searching for WHEN-VALIDATE-ITEM triggers. It found none at the item level, so it continued to the block level. It found one there and fired it. This can be very useful if there is logic that youd like to execute for every item in the block. Exit and close the Forms Runtime and return to the Form Builder. View the properties for the SALUTATION items WHEN-VALIDATE-ITEM trigger. Change the Execution Hierarchy property to Before. Run the form. Type Mr. in the SALUTATION item and press the TAB key. Read the alert message, then click the OK button. Take note of the status line. f) How has changing the Execution Hierarchy property affected the form? Answer: Now both the item-level and block-level WHEN-VALIDATE-ITEM triggers have fired. The default firing order has been changed so that the block-level trigger is now fired as well, instead of being ignored. The item-level trigger is fired first, then the block-level trigger. As you can tell, the Execution Hierarchy property lets you determine the order in which like triggers at different levels in the Forms hierarchy should be fired. By selecting Before, you indicated that you wanted the lowest level WHEN-VALIDATE-ITEM trigger to fire before any WHEN-VALIDATE-ITEM triggers at higher levels. You can imagine what would have happened if you had set Execution Hierarchy to After. What would happen if you were to set the block-level WHEN-VALIDATEITEMs Execution Hierarchy property to Before? When the Validate Item event occurred for SALUTATION, all three of this forms WHEN-VALIDATE-ITEM triggers would fire in the following order: item, block, form. Select the Triggers node under the STUDENT.SALUTATION item and click the Create button in the Object Navigator. g) Can you create a PRE-FORM trigger here? Why not? Answer: No you cannot. A PRE-FORM trigger is form-specific and therefore cannot be defined at the item level. That is, not all trigger types can be defined at multiple levels in the Forms hierarchy.
go to contents
6.1.3 ANSWERS
a) What trigger would you create to replace the default delete processing? What named category does this fall under? What functional category does it fall under? Answer: You would create the ON-DELETE trigger. This is an On event trigger that falls into the Transactional category. Since the ON-DELETE trigger replaces the way Forms would normally delete a record, it is considered a Transactional trigger. b) Is it mandatory that you write triggers to respond to each event? What happens if you dont? Answer: No it is not. Forms default processing handles the event. Form EX06_01.fmb has a number of triggers that respond to events like Validate Item, Button Pressed, and so on. But, think back to the forms you created in earlier Chapters. You did not write any triggers to respond to the Validate Item events, yet the events still occurred when you changed an items value and then navigated to another item. Forms looked for triggers to respond to the events, but when it found none, it simply executed the default processing. c) When should the trigger you write populate these items? Before or after the query is issued? Answer: The trigger should populate these items after the query is issued and the results have been returned to the form. d) Based on your answer to Question c, what trigger should you create? At what level should you attach it to the form? Search the help system if you are having trouble with this question. Answer: You should create a POST-QUERY trigger and attach it at the block level. POST-QUERY is a Post event trigger that belongs to the Query group of triggers. As its name implies, the POST-QUERY trigger will fire each time a record is returned to the block. POST-QUERY triggers can be attached at the block or form level, but not at the item level. If you attach a POST-QUERY trigger to a STUDENT block, for example, it will only fire when records are fetched to the STUDENT block. If you attach a POST-QUERY trigger at the form level of a form that has a STUDENT and an ENROLLMENT block, it will fire whenever a record is fetched into either block.
go to contents
e) Which trigger should you create? What named and functional categories does this trigger belong to? Answer: You should create a WHEN-VALIDATE-ITEM trigger. This is a When event trigger that belongs to the Validation functional category. f) Should these triggers fire before or after the inserts and updates are issued? Answer: These triggers should fire before the inserts and updates are issued. g) Based on your answer to Question f, which triggers should you choose and what are their functional and named categories? Answer: You should choose PRE-INSERT and PRE-UPDATE triggers. These are Pre event triggers that belong to the Transactional functional category.
4) Which of the following are true about a POST-TEXT-ITEM trigger? a) ____ It is a When event trigger b) ____ It is a navigational trigger c) ____ It will replace Forms default processing d) ____ None of the above
go to contents
go to contents
187
LAB
6.2
LAB 6.2
Create Query Triggers Create Validation Triggers Create Transactional Triggers Create Key Triggers
There are hundreds of triggers in Forms, and multiple ways to use each trigger. In this Lab, you will learn to write some commonly used triggers. The code you write will be specific to the objects in the STUDENT application, but can serve as templates for triggers you write in your own applications.
QUERY TRIGGERS
The POST-QUERY is often used to populate non-base table display items in a block. These non-base table display items are sometimes referred to as lookup items, and are used to make one or more of the base-table items more meaningful.
I FOR EXAMPLE:
Assume you have created a block based on the SECTION table. You are displaying all of its columns, including COURSE_NO. To make each record more meaningful, youd like to display the courses description as well. However, the DESCRIPTION column resides in the COURSE table, so you cant include it as a base-table item in the SECTION block. You must,
go to contents
188
LAB 6.2
The form would fetch a section record into the block and then fire the POST-QUERY trigger. The trigger would then fetch the corresponding course.description from the database and place it into the DESCRIPTION display item.
VALIDATION TRIGGERS
There are two Validation triggers that are commonly used in Forms: WHEN-VALIDATE-ITEM and WHEN-VALIDATE-RECORD. Each serves to validate data entered by a user. In the Exercises, you will write some simple Validation triggers to confirm that: 1) 2) Values entered by a user adhere to the business rules. Values entered into foreign-key items exist in the parent table.
I FOR EXAMPLE:
Assume there is a business rule in the STUDENT application that states that no class can cost more than $5,000. Whenever a user enters a value into a COST item, you want the form to confirm that the value theyve entered adheres to the rule. You could do so by writing a WHEN-VALIDATE-ITEM trigger that contains the following code and attach it to the COST item: IF :SECTION.COST > 5000 THEN MESSAGE('Course costs must be less than $5,000.'); RAISE FORM_TRIGGER_FAILURE; END IF; The WHEN-VALIDATE-ITEM trigger will fire when both of the following two conditions have been met: 1) 2) The user has changed the value in the item. The user has navigated out of the item.
If the user enters a value greater than 5000 and navigates out of the item, the Validate Item event will occur and the WHEN-VALIDATE-ITEM trigger
go to contents
189
will fire. Since validation has failed, the user will receive a message and processing will stop. Validation triggers can also be used to check that values entered into foreign-key items exist in the parent table.
I FOR EXAMPLE:
Assume you have a form based on the ENROLLMENT table. You want the application to confirm that the value entered for SECTION_ID exists in the SECTION table. If it doesnt, the INSERT or UPDATE statement will be rejected by the database. If the database is going to reject it anyway, which is essentially validation, then why repeat the code in the application? For one thing, it makes the application a bit more user-friendly. The user will be alerted to his mistake immediately rather than later at the time of the insert. It also makes it easier to process the error. You respond to and handle validation item-by-item rather than by trying to process the error message returned by the database, which might not always be meaningful.
LAB 6.2
TRANSACTIONAL TRIGGERS
There are a number of Transactional triggers used to augment or replace Forms default transaction processing. The ON-POPULATE-DETAILS and ONCHECK-DELETE-MASTER master-detail triggers are considered Transactional triggers. PRE-CHANGE, POST-FORMS-COMMIT, POST-DATABASE-COMMIT, and many other transaction-related triggers allow you to write your own processing logic in and around Forms-level and database-level transactions. In this Lab, you will experiment with two: the PRE-INSERT and PREUPDATE triggers. You will use these to set values for the audit columns.
KEY TRIGGERS
Key triggers fire whenever a user presses a corresponding key on the keyboard. If a user presses the down arrow, or the down key, then the KEYDOWN trigger will fire. Key triggers can be used if you want to change or replace default key processing.
190
LAB 6.2
on the canvas. Lay the items out in Form style. Create two display items in the STUDENT block and name them CITY and STATE. Position them after STUDENT.ZIP in the block and just to the right of STUDENT.ZIP on the canvas. Size and align them so that they are arranged neatly, but do not be overly concerned with the look of the form. Use the code below to answer Questions ad.
DECLARE CURSOR c_city_state IS SELECT city, state FROM zipcode WHERE zip = :STUDENT.ZIP; BEGIN OPEN c_city_state; FETCH c_city_state INTO :STUDENT.CITY, :STUDENT.STATE; CLOSE c_city_state; END;
b) How will the POST-QUERY trigger know which record to fetch from the database?
c) Which items are being populated? Which line of code populates these items?
go to contents
191
Create a POST-QUERY trigger and attach it to the object that was your answer for Question d. Type the code above into the PL/SQL Editor and click the Compile button. e) Were there any errors?
LAB 6.2
When the POST-QUERY trigger compiles correctly, run the form and issue a query. f) Were the CITY and STATE items populated?
If not, look at the Forms Runtimes status line for error messages. Select Help | Display Errors from the Forms Runtimes Main Menu to see more details. g) What was the error? Why did this happen? What should you do to the display items to correct this?
Exit the form and fix the mistake. Run the form again and issue a query to test the POST-QUERY trigger. h) Did it populate CITY and STATE this time? What happens when you scroll from record to record?
i) If you were to create a new form based on the ENROLLMENT table, what are some display items you could create and populate with a POST-QUERY trigger?
go to contents
192
Lab 6.2: Creating Triggers of Various Types j) What would the code for the trigger be?
LAB 6.2
a) What variable are you declaring to help check the validity of the ZIPCODE value? What is its data type?
go to contents
193
b) Which line of code assigns a value to this variable? What cursor attribute are you using to assign the value?
c) What will be the value of the v_invalid variable if the cursor fails to fetch a row? What will this mean about the value the user has entered?
LAB 6.2
d) What two commands will the trigger issue if the value is invalid?
e) Why is the trigger fetching values into the CITY and STATE columns if the purpose is to validate the ZIP item? Wont the columns be populated by the POST-QUERY trigger?
Create a WHEN-VALIDATE-ITEM trigger for the STUDENT.ZIP item and enter the code above. Compile the trigger. Run the form and issue a query. Change the value in the ZIP item to 123 and press the TAB key. f) Has the WHEN-VALIDATE-ITEM trigger fired? What two things about the forms behavior indicate that it has?
g) What Forms object could you attach to this item to help the user choose a valid Zip Code?
go to contents
194
LAB 6.2
Use the wizards to quickly create a form based on the SECTION table. Enforce data integrity on the wizards table page should be unchecked. Include the audit columns in the block, but do not display them on the canvas. Lay the items out in Form style. Assume that there is a building called L5 on the STUDENT campus. The rooms in this building can only seat 15 students or less. In the SECTION tables LOCATION column, all of the rooms in the L5 building are named L501, L502, and so on. When users are inserting or updating section records, you want to prevent them from making the CAPACITY greater than 15 for any room in the L5 building. h) Could you write a Validation trigger to enforce this rule? What would the code be? Write your answer on paper first. The trigger code should not be overcomplicated. You should be able to do it in three simple statements.
i) If you use a WHEN-VALIDATE-ITEM trigger, which item could the trigger code be attached to? Do not create the trigger, simply write down your answer.
j) If your answer to Question i was CAPACITY, when will the WHEN-VALIDATE-ITEM trigger fire?
k) If the user inserted a new row, what would happen if the user set CAPACITY to 25 first and then set LOCATION to L501? Would the validation take place? Why not?
go to contents
195
l) Which Validation trigger could you create to make sure that the trigger fires for each record? Which object should you attach it to? Create the trigger, enter the code, and test the form.
LAB 6.2
You do not need to save this form as you will not need it in future Exercises.
In this Exercise, you will write Transactional triggers to set the values for the audit columns. a) Why do you have to write a trigger to set these values? Why not make the user input these values?
go to contents
196
Lab 6.2: Creating Triggers of Various Types c) What two pieces of information will you need to get from the system to assign values for the audit columns?
LAB 6.2
Create the PRE-INSERT trigger at the form level and enter the code above. Run the form and try to insert a new record.
go to contents
Lab 6.2: Creating Triggers of Various Types g) Did the trigger work? How do you know?
197
h) What trigger should you create to set MODIFIED_BY and MODIFIED_DATE every time a record is changed?
LAB 6.2
j) Could you reuse these triggers exactly as they are for forms with SECTION blocks? STUDENT blocks? Any block?
198
Run the form and press the F8 key on the keyboard to test the trigger. Exit the form after you have tested the trigger to return to the Form Builder.
LAB 6.2
a) Did the Key trigger respond when the key was pressed? Why didnt the form execute a query?
Add the EXECUTE_QUERY statement to the end of the KEY-EXEQRY trigger. The trigger should now be as follows:
MESSAGE('You have pressed the F8 key to execute a query'); EXECUTE_QUERY;
Create a WHEN-BUTTON-PRESSED trigger for the ZIPCODE.EXECUTE_ QUERY button. Add the following statement:
EXECUTE_QUERY;
Run the form and press the F8 key on the keyboard to confirm that the query has been executed. Now click the Execute Query button. b) Was the message text issued along with the query? Why not?
Run the form and test the Execute Query button. c) What function did the DO_KEY built-in provide?
go to contents
Lab 6.2: Creating Triggers of Various Types d) What function will this trigger perform?
199
Run the form. Click the Enter Query button on the toolbar to put the form into Enter_Query mode. Issue a query for the Zip Code 06605. Click the Remove Record button on the toolbar. e) Did the KEY-DELREC trigger fire? What built-in must the trigger associated with the Remove Record button use to make this happen?
LAB 6.2
go to contents
200
LAB 6.2
Figure 6.2 I The PL/SQL Editor showing a successfully compiled POST-QUERY trigger.
go to contents
201
You have already worked with the PL/SQL Editor in previous Chapters, and have probably found that it is a rather simple, yet handy, tool for writing code. Now that your triggers are becoming more complicated, it is worth a bit more exploration.
LAB 6.2
I FOR EXAMPLE:
If there had been a mistake in your code, the PL/SQL Editor may have looked like Figure 6.3. Note that the gray area below the trigger code lists error messages. In this case, the ZIPCODE table was misspelled, so the Form Builder could not find it in the database.
go to contents
202
LAB 6.2
f)
Were the CITY and STATE items populated? Answer: No they were not.
g) What was the error? Why did this happen? What should you do to the display items to correct this? Answer: The error was FRM-40505 Unable to Perform Query. If you looked at the Show Errors window, you would have been able to see the error in the SELECT statement. CITY and STATE are not base-table items in this block and should not be included in the query. They are non-basetable items, so their Database Item properties should have been set to No. It is important to remember that, although CITY and STATE are being populated by values from the database, they are not base-table items in this block. Run the form and issue a query to test the POST-QUERY trigger. h) Did it populate CITY and STATE this time? What happens when you scroll from record to record? Answer: Yes, the CITY and STATE values change to correspond with the value in the ZIP item. i) If you were to create a new form based on the ENROLLMENT table, what are some display items you could create and populate with a POST-QUERY trigger? Answer: You could create display items to show the students LAST_NAME and FIRST_NAME and perhaps the COURSE_NO. In these examples, you are using the POST-QUERY trigger to provide lookup values to make the form data more meaningful to the user. In the STUDENT form created above, the ZIP item was much more meaningful to the user when the CITY and STATE values were supplied along with it. In the case of a form based on the ENROLLMENT table, what values from other tables might make the enrollment information more meaningful? The LAST_NAME and FIRST_NAME would make STUDENT_ID more meaningful. And perhaps the COURSE_NO would make the SECTION_ID more
go to contents
203
meaningful. You could include even more, like the DESCRIPTION of the course, the INSTRUCTOR_ID, and so on. In the following question, write a trigger that would populate LAST_NAME, FIRST_NAME, and COURSE_NO. j) What would the code for the trigger be? Answer: See below. DECLARE CURSOR c_student_name is SELECT first_name, last_name FROM student WHERE student_id = :ENROLLMENT.STUDENT_ID; CURSOR c_course_no is SELECT course_no FROM section WHERE section_id = :ENROLLMENT.SECTION_ID; BEGIN OPEN c_student_name; FETCH c_student_name INTO :ENROLLMENT.LAST_NAME, :ENROLLMENT.FIRST_NAME; CLOSE c_student_name; OPEN c_course_no; FETCH c_course_no INTO :ENROLLMENT.COURSE_NO; CLOSE c_course_no; END; Note that, in this case, the trigger had more than one cursor.
LAB 6.2
6.2.2
ANSWERS
a) What variable are you declaring to help check the validity of the ZIPCODE value? What is its data type? Answer: The variable is v_invalid and its data type is Boolean. b) Which line of code assigns a value to this variable? What cursor attribute are you using to assign the value? Answer: See below.
go to contents
204
LAB 6.2
go to contents
205
items while the user is editing items and tabbing around the form because there has not been a query. f) Has the WHEN-VALIDATE-ITEM trigger fired? What two things about the forms behavior indicate that it has? Answer: The message appeared on the hint line, and it is not possible to navigate from the ZIP item. g) What Forms object could you attach to this item to help the user choose a valid Zip Code? Answer: An LOV. If the user is having trouble entering valid values, it might be helpful to provide an LOV for them to select from. h) Could you write a Validation trigger to enforce this rule? What would the code be? The trigger code should not be overcomplicated. You should be able to do it in three simple statements. Answer: See the code below. IF :SECTION.LOCATION LIKE 'L5%' AND :SECTION.CAPACITY > 15 THEN MESSAGE('Capacity must be less than 15 for sections in the L5 building.'); RAISE FORM_TRIGGER_FAILURE; END IF; Note that the trigger will check two of the items in the form and if both conditions are met, it will issue a message to the user and RAISE FORM_TRIGGER_FAILURE to halt processing. i) If you use a WHEN-VALIDATE-ITEM trigger, which item should the trigger code be attached to? Answer: It could be attached to the CAPACITY or LOCATION item. j) If your answer to Question i was CAPACITY, when will the WHEN-VALIDATE-ITEM trigger fire? Answer: It will fire after you change the CAPACITY value and then navigate out of the item.
LAB 6.2
go to contents
206
LAB 6.2
Because the trigger is attached to the CAPACITY column, it will fire only after the user has changed CAPACITY and navigated out of it. It will not fire when the user changes and navigates out of LOCATION. Therefore, if the user changes the CAPACITY item first, then changes the LOCATION item second, the trigger will not fire. This goes back to trigger scope; a trigger will fire only within its scope, which in this case is within the validation of CAPACITY. So, what could you do? You could copy the trigger and attach it to both items, but that would not be very elegant. You could put the WHEN-VALIDATE-ITEM trigger at the block level, but then it would fire for every item, which would be inelegant and inefficient. Or, you could put a different trigger at the block level. Read the next question for more details on what you could do. l) Which Validation trigger could you create to make sure that the trigger fires for each record? Which object should you attach it to? Create the trigger, enter the code, and test the form. Answer: You could create a WHEN-VALIDATE-RECORD trigger and attach it to the block. The WHEN-VALIDATE-RECORD trigger will fire once for the entire record when the Validate Record event occurs. So, no matter what order a user enters or changes the values in the block, the WHEN-VALIDATE-RECORD trigger will fire and catch any invalid values.
6.2.3 ANSWERS
a) Why do you have to write a trigger to set these values? Why not make the user input these values? Answer: These are audit columns and should be maintained by the system. The purpose of these columns is to keep a strict record of when, and by whom, each record was updated or changed. If the audit trail is being kept for security reasons, as well as for record-keeping reasons, then it does not make sense to allow the user to edit the values.
go to contents
207
The most secure method would be to populate these columns with database triggers. But, for the purpose of these Exercises, you will have the form populate them. b) Should this trigger be assigned to the form or block level? Answer: PRE-INSERT triggers can be set at either the form or block level. The code you will write in this Exercise will be block-independent. That is, the block names will not be hard-coded into the trigger so that they can apply to any base-table block in the form. Therefore, if you put the triggers at the form level, their scope will be for all base-table blocks. In this application, all of the base-table blocks will contain the audit items. c) What two pieces of information will you need to get from the system to assign values for the audit columns? Answer: You will need to get the users name and the date. d) Which built-in is being used to get the users name? Answer: The GET_APPLICATION_PROPERTY built-in. e) How is the value of v_block assigned? Answer: v_block is assigned using the :SYSTEM.CURSOR_BLOCK system variable. System variables hold internal information about the form. SYSTEM .CURSOR_BLOCK holds the value of the current navigation block. There are many other system variables that you can reference to get all sorts of internal information like the name of the current item, if the form is in Enter Query mode or Normal mode, the current position of the mouse, and so on. In this case, you want to get the name of the current block so that you can set the values for the audit items appropriately. f) What parameters are being passed to the COPY built-in? Answer: A value and a block.item name. The COPY built-in takes a value and copies it somewhere else. Here, you are copying the value in the variable into one of the audit items. You could also use the COPY built-in to copy the value in one variable into another variable.
LAB 6.2
go to contents
208
LAB 6.2
While this method would work, it is not block-independent since you had to hard-code the block name into the statement. g) Did the trigger work? How do you know? Answer: Yes, a Transaction complete one record applied and saved message appeared in the hint line. Also note that the values appeared in the items in the form. This, however, is not an indication that the insert succeeded in the database. This merely indicates that the values were successfully populated in the items. The PRE-INSERT trigger fires before an insert. The code you have written in the PRE-INSERT only assigns values to items in the form; it does nothing to make sure that those values are inserted to the database. Once the PRE-INSERT trigger has completed, Forms continues with its default insert processing. Forms writes an INSERT statement that includes every data item in the block and sends it off to the database. h) What trigger should you create to set MODIFIED_BY and MODIFIED_DATE every time a record is changed? Answer: You should use a PRE-UPDATE trigger. i) What will the code be for this trigger? Answer: See the code below. DECLARE v_block VARCHAR2(30); v_username VARCHAR2(30); v_date DATE; BEGIN v_username := GET_APPLICATION_PROPERTY(USERNAME); v_date := SYSDATE; v_block := :SYSTEM.CURSOR_BLOCK; COPY(v_date, v_block||'.MODIFIED_DATE'); COPY(v_username, v_block||'.MODIFIED_BY'); END;
go to contents
209
Note that the only difference is that the CREATED_BY and CREATED_DATE items are not being populated here. j) Could you reuse these triggers exactly as they are for forms with SECTION blocks? STUDENT blocks? Any block? Answer: Yes you could. Since you didnt hard-code the block names into the triggers, you have made them portable across forms.
LAB 6.2
6.2.4
ANSWERS
a) Did the Key trigger respond when the key was pressed? Why didnt the form execute a query? Answer: Yes, the trigger responded. The MESSAGE statement that was written to the trigger was successfully executed. However, the form didnt execute a query. Key triggers, like On triggers, replace Forms default processing, so the original default processing will not occur. To augment the default processing of a keystroke, you must remember to manually enter the necessary code.
I FOR EXAMPLE:
For the KEY-EXEQRY trigger to reproduce the default processing, the code must be as follows: MESSAGE('You have pressed the F8 key to execute a query'); EXECUTE_QUERY; b) Was the message text issued along with the query? Why not? Answer: No it was not. The WHEN-BUTTON-PRESSED trigger is not aware of the code that is in the KEY-EXEQRY trigger, so of course it will not fire the MESSAGE statement. The problem here is that the application will behave differently if the user executes a query by pressing the F8 key on the keyboard, or if they click the Execute Query button on the screen. In almost all cases, you
go to contents
210
LAB 6.2
Change the code in the WHEN-BUTTON-PRESSED trigger to the following: DO_KEY('EXECUTE_QUERY'); Run the form and test the Execute Query button. c) What function did the DO_KEY built-in provide? Answer: The DO_KEY built-in fired the KEY-EXEQRY trigger. When executed, DO_KEY fires the Key trigger associated with the built-in it has accepted as its parameter.
I FOR EXAMPLE:
If you issue the statement DO_KEY('COMMIT_FORM'); the KEY-COMMIT trigger will fire. If you issue the statement DO_KEY('ENTER_QUERY'); the KEY-ENTQRY trigger will fire. Study the code for the ZIPCODE blocks KEY-DELREC trigger. d) What function will this trigger perform? Answer: See the discussion below. The form will prevent the current record from being marked for deletion if that record has child records in another table. Run the form. Click the Enter Query button on the toolbar to put the form into Enter Query mode. Issue a query for the Zip Code 06605. Click the Remove Record button on the toolbar.
go to contents
211
e) Did the KEY-DELREC trigger fire? What built-in must the trigger associated with the Remove Record button use to make this happen? Answer: Yes, the DO_KEY built-in was used. This example illustrates the usefulness of the DO_KEY built-in. The default processing for the deletion of a record has been overwritten and replaced with a KEY-DELREC trigger. By using the DO_KEY built-in behind the toolbar buttons, the application is ensuring that any Key trigger logic will be fired. You will use the DO_KEY built-in again when you create your own toolbar in Chapter 8, Canvases and Windows.
LAB 6.2
go to contents
212
LAB 6.2
6) What is FORM_TRIGGER_FAILURE? a) ____ A built-in to respond to the failure of an event b) ____ A built-in you can use to crash the operating system c) ____ A built-in exception to help you handle errors d) ____ An event you can respond to with the ON-ERROR trigger Quiz answers appear in Appendix A, Section 6.2.
go to contents
213
LAB
6.3
FORMS BUILT-INS
LAB OBJECTIVES
After this Lab, you will be able to: LAB 6.3
The Forms built-ins are a set of PL/SQL functions and procedures that perform standard application functions. You have already used built-ins like EXIT_FORM and COMMIT_FORM in previous Labs. In these cases, you simply typed the built-ins name, and in doing so, accepted its default functionality. But, like the PL/SQL functions and procedures you have written yourself, most built-ins can accept parameters. The parameters you pass a built-in will affect its behavior.
I FOR EXAMPLE:
When you used the EXIT_FORM built-in, you didnt pass it any parameters. The code looked like this: EXIT_FORM; However, the EXIT_FORM built-in can also accept parameters that affect what the form does when it exits. It might look like this: EXIT_FORM('DO_COMMIT'); By passing EXIT_FORM the DO_COMMIT parameter, you are telling the form to validate and commit any outstanding changes in the form as well as exit the form.
go to contents
214
GET_ BUILT-INS
There are a number of built-ins that are prefixed with the word GET_. You used the GET_APPLICATION_PROPERTY in Lab 6.1 to get the users name and assign it to the CREATED_BY and MODIFIED_BY items. The code looked like this: :COURSE.CREATED_BY := GET_APPLICATION_PROPERTY(USERNAME); This specific built-in is used to get information about the application. There are other GET_ built-ins that you can use to get properties about other Forms objects such as items, blocks, canvases, and so on. It is also possible, and quite common, to assign the results of a GET_ built-in to a variable.
LAB 6.3
I FOR EXAMPLE:
If you wanted to assign the user name to a variable called v_username, the code would look like this: DECLARE v_user_name VARCHAR2(50); BEGIN v_user_name := GET_APPLICATION_PROPERTY(USERNAME); END;
SET_ BUILT-INS
There is another group of built-ins that are prefixed with the word SET_. As you can imagine, you use them to set certain values.
I FOR EXAMPLE:
You can use the SET_BLOCK_PROPERTY built-in to set properties about a block at run-time. The following two statements set the ORDER BY clause and the WHERE clause for a block called SECTION:
go to contents
215
SET_BLOCK_PROPERTY('SECTION', DEFAULT_WHERE, 'INSTRUCTOR_ID = 101'); SET_BLOCK_PROPERTY('SECTION', ORDER_BY, 'SECTION_ID); The SET_ built-ins are accepting three parameters: the name of the object to be adjusted, the name of the property to be set, and the value to give that property. There are SET_ built-ins for other objects in Forms, like windows, items, canvases, and so on.
FIND_ BUILT-INS
In both cases above, you used the SECTION objects name in the SET_ statements. While this is correct, it is slightly inefficient since Forms must use resources to look up the object by name. Every object in Forms is assigned a unique object ID at run-time. Since you are referring to the SECTION block more than once, it is better to refer to it by its ID in the built-ins so that Forms can look up the object more efficiently. To get an objects ID, you must use one of the FIND_ built-ins. In this case, since you are working with a block, you would use the FIND_BLOCK built-in. You would use it to find the ID of SECTION, and then use that ID in the subsequent SET_ statements. The code would look like this: DECLARE v_block_id BLOCK; BEGIN v_block_id := FIND_BLOCK('SECTION'); SET_BLOCK_PROPERTY(v_block_id, DEFAULT_WHERE, 'INSTRUCTOR_ID = 101'); SET_BLOCK_PROPERTY(v_block_id, ORDER_BY, 'SECTION_ID'); END; The variable v_block_id is used to hold the result of the FIND_BLOCK built-in. Then, v_block_id is used in the subsequent SET_ statements to identify the object. Now, the SECTION object is only referenced by name once, in the FIND_BLOCK statement. The SET_ statements use the object ID for the SECTION block, which makes the code much more efficient.
LAB 6.3
go to contents
216
In this Exercise, you will use built-ins to manipulate the properties of a window at run-time.
LAB 6.3
Open the form EX06_03.fmb in the Form Builder. Create a WHEN-NEWFORM-INSTANCE trigger at the form level. a) How can you use the GET_APPLICATION_PROPERTY built-in to get the name of the current form module?
b) What built-in can you use to size the window MAINWIN to 200, 200? Try to do this using only one built-in statement.
c) How can you use the same built-in from Question b to set the title for MAINWIN? The title should be as follows: 'This is form... <FORM NAME> ' At the end of the title, you should insert the result of the GET_APPLICATION_PROPERTY built-in from Question a.
Run the form and test the built-ins. d) What can you do to refer to MAINWIN more efficiently than by using its object name? Change the code in the WHEN-NEW-FORMINSTANCE trigger to do this.
go to contents
Lab 6.3: Forms Built-Ins e) Would this trigger fire properly if MAINWIN did not exist?
217
Create a PRE-RECORD trigger for the ZIPCODE block and give it the following code:
GO_ITEM('ZIPCODE.CITY');
Run the form and issue a query. f) What was the error you received in the hint line?
LAB 6.3
a) How can you use the GET_APPLICATION_PROPERTY built-in to get the name of the current form module? Answer: See the discussion below. GET_APPLICATION_PROPERTY is a built-in function that returns a value, so in this case, its result should be assigned to a variable as is shown in the code below: DECLARE v_form_name VARCHAR2(50); BEGIN v_form_name := GET_APPLICATION_PROPERTY(CURRENT_FORM_NAME); END; There are many GET_ built-ins that allow you to get properties for objects or information from the system. You can use GET_ITEM_PROPERTY, GET_CANVAS_PROPERTY, and GET_BLOCK_PROPERTY as well to find the current value of a property and then act on it. In this Exercise, you will
go to contents
218
LAB 6.3
go to contents
219
help files often to confirm that you are using the proper syntax and that you are passing parameters using the proper data types. c) How can you use the same built-in from Question b to set the title for MAINWIN? The title should be as follows:
'This is form . . . <FORM NAME>' At the end of the title, you should insert the result of the GET_ APPLICATION_PROPERTY built-in from Question a.
Answer: See the discussion below. Again, you would use the SET_WINDOW_PROPERTY to set the title. This statement would be as follows: SET_WINDOW_PROPERTY('MAINWIN', TITLE, 'This is form '||v_form_name); The syntax is the same as in the SET_WINDOW_PROPERTY statement that you used to set the window size. What is different are the values that you are passing to the built-in. Note that you passed the v_form_name variable into the built-in. This is very common in that it keeps you from hard-coding values into the built-in. Run the form and test the built-ins. d) What can you do to refer to MAINWIN more efficiently than by using its object name? Change the code in the WHEN-NEW-FORM-INSTANCE trigger to do this. Answer: You can refer to it by object ID using the FIND_WINDOW built-in. The FIND_WINDOW built-in will get the windows object ID for you. Then you can use the item ID in the SET_ built-ins instead of the item name. As you learned in the Lab text, it takes fewer resources to refer to an object by its ID than to refer to it by name, so whenever possible, it is more efficient to refer to an object by its ID. Just like GET_ and SET_, there are FIND_ built-ins for virtually every object in Forms, all of which will return an objects system ID. To use the FIND_ built-ins, you must employ variables of specific types.
LAB 6.3
I FOR EXAMPLE:
When you declare a variable to hold the system ID of a block, the variable must be of type BLOCK. So, if you want to FIND_ the ID of the ZIP-
go to contents
220
LAB 6.3
DECLARE v_form_name VARCHAR2(50); v_window_id WINDOW; BEGIN v_form_name := GET_APPLICATION_PROPERTY(CURRENT_FORM_NAME); v_window_id := FIND_WINDOW('MAINWIN'); SET_WINDOW_PROPERTY(v_window_id, WINDOW_SIZE, 200, 200); SET_WINDOW_PROPERTY(v_window_id, TITLE, 'This is form '||v_form_name); END; Note that a new variable called v_window_id has been created to hold MAINWINs object ID. v_window_id is set using the FIND_WINDOW builtin. Then, the SET_ statements v_window_id is used to refer to the window instead of its name. Note that the v_window_id variable is not in single quotes in the built-in statements. e) Would this trigger fire properly if MAINWIN did not exist? Answer: No it would not. The FIND_WINDOW built-in accepts the MAINWIN window objects name. If this window did not exist in the form, you would see the following error when the trigger tried to fire: FRM-41052 Cannot find window. Invalid id. This type of error is not limited to the FIND_WINDOW built-in. You would receive similar errors if any built-in tried to refer to an object that did not
go to contents
221
exist in the form. To guard against this, it is wise to write the trigger so that it can alert you or the user to the absence of an object.
I FOR EXAMPLE:
DECLARE v_form_name VARCHAR2(50); v_window_id WINDOW; BEGIN v_form_name := GET_APPLICATION_PROPERTY(CURRENT_FORM_NAME); v_window_id := FIND_WINDOW('MAINWIN'); IF ID_NULL(v_window_id) THEN MESSAGE('MAINWIN doesnotexist.Error in WHEN-NEW-FORM-INSTANCE trigger'); RAISE FORM_TRIGGER_FAILURE; END IF; SET_WINDOW_PROPERTY(v_window_id, WINDOW_SIZE, 200, 200); SET_WINDOW_PROPERTY(v_window_id, TITLE, 'This is form '||v_form_name); END; The ID_NULL built-in evaluates whether or not the value v_window_id is null. If it is null, it means the MAINWIN window does not exist. Create a PRE-RECORD trigger for the ZIPCODE block and give it the following code: GO_ITEM('ZIPCODE.CITY'); Run the form and issue a query. f) What was the error you received in the hint line? Built-ins that have to do with navigation are deemed restricted. That is, they cannot be used in Navigational triggers. The PRE-RECORD trigger is a Navigational trigger, as is PRE-TEXT-ITEM, POST-BLOCK, POST-QUERY, and many others. You are unable to use restricted built-ins like GO_ITEM, GO_BLOCK, and so on in these triggers. As you have already seen, the help system will indicate whether a built-in is restricted or not under the Built-in Type heading. The help for triggers will also indicate what types of built-ins they can include under the Legal Commands heading.
LAB 6.3
go to contents
222
LAB 6.3
go to contents
223
CHAPTER
go to contents
C H A P T E R
ists of values (LOVs) and alerts are visual objects with which users can interact. They are different from items in that they appear in their own windows and are not positioned on a canvas. An LOV is used to present a list of values from which a user can choose to populate items on a form. LOVs can be created manually or by using a wizard. In the Exercises in this Chapter, you will create an LOV and learn about the built-ins that you can use to display it. Alerts are used to present an important message to the user. Alerts also have buttons so that the user can respond to the message that is being displayed. In the Exercises in this Chapter, you will learn how to create an alert and use the built-ins that you can use to display it.
225
go to contents
LAB
7.1
In Chapter 5, Items, you used list items to present the user with a list of choices. LOVs also present lists of choices to the user, but the lists can be much longer, and they can display multiple columns instead of just one.
I FOR EXAMPLE:
Figure 7.1 shows an LOV called Sections. It has three columns that display the section_id, course_no, and description values. The LOV was displayed when the user entered the SECTION_ID item and clicked the List of Values button. To choose an item, the user can select the item in the list and click OK, or simply double-click the item. Once selected, some of the column values from the LOV will be returned to the form to populate text and display items. In this case, the section_id from the LOV value will be returned to the SECTION_ID item in the form. LOVs serve a number of purposes. They make data entry easier, and they ensure the validity of data. The SECTION table has 78 section_ids. It would be unreasonable to expect users to memorize all of them along with their corresponding course numbers and descriptions. The LOV provides a quick and easy way for the user to reference the available
go to contents
go to contents
b) As you go through and answer the questions on the wizard pages, what kinds of design-time operations will the wizard perform to create your list of values? Do not over-think this question; the answers are rather straightforward.
The LOV Wizard has 9 pages that you will walk through here. The headings in bold indicate the names of the pages according to the Forms help system. The headings will be followed by questions that refer to that page and the function it performs. After each set of questions, there will be a short statement describing how the page should be configured before clicking the Next button.
go to contents
Please note that the LOV Wizard has a Welcome and a Finish page that will not be discussed here. SOURCE PAGE c) Are there any record group objects in the R_STUDENT.fmb form?
d) If there were existing record groups, would you be able to use one of them here as the source of your LOV? How?
Click the Next button to continue. SQL QUERY PAGE The SQL statement will be the basis of the record group, which will in turn be the source of the LOV. The LOV should display the ZIP, CITY, and STATE columns from the ZIPCODE table. The list should be ordered by STATE and then CITY. e) What does the Build SQL Query button do? Do not use it to build the query, simply click the button to see what it does. Click Cancel when you are done.
f) Can you import an existing SQL statement? Where will Forms look for it? Do not import an SQL statement. Simply click the button to see what it does. Click Cancel when you are done.
go to contents
Lab 7.1: Lists of Values (LOVs) g) What should the SQL statement be for this record group? Type it into the SQL Statement field.
h) How can you confirm that the SQL you have written is correct?
Click the Next button to continue. COLUMN SELECTION PAGE Again, your LOV should display ZIP, CITY, and STATE. j) Which columns from the record group should you include in the LOV?
Include the columns and click the Next button to continue. COLUMN DISPLAY PAGE Adjust the widths of the columns if youd like, but remember that you can do this later after you have tested the LOV. Adjust the horizontal scrollbar so that you can see the Return Value field. k) When the user makes a selection in the LOV, which of the LOVs columns should be returned to the form? ZIP, CITY, or STATE? In this case, returned means taken from the LOV and put into an item in the form.
go to contents
l) Which item in which block should this LOV column be returned to?
Make sure the proper return item is specified, and click the Next button to continue. LOV DISPLAY PAGE There will be no questions for this page. Simply follow the instructions below. Title the LOV Zip Codes. Leave the Height and Width coordinates as they are. Select Yes, let Forms position the LOV automatically and click the Next button. ADVANCED OPTIONS PAGE There are 227 rows in the ZIPCODE table. n) What would be the difference in behavior in leaving Retrieve Number of Rows at 20, or in switching it to 227?
o) How often do state names, city names, and their Zip Codes change? Do you think it is necessary for the form to issue the record group query to refresh the LOV every time it is displayed?
Leave Retrieve Number of rows at 20. Uncheck Refresh record group data before displaying the LOV. Uncheck Let the user filter records before selecting them. Click the Next button to continue. go to contents
So far, you have created a record group and created and configured an LOV. p) How can you indicate that you want the LOV to be attached to the STUDENT.ZIP item?
Click the Next button to continue to the Finish page. Click the Finish button to return to the Form Builder. Rename the LOV ZIP_LIST. Run the form and issue a query. Slowly tab from item to item and keep your eye on the Forms Runtimes status line. q) What happens when you tab into the ZIP item?
Every operating system has a List of Values key. When the user presses this key, the LOV will open. On Windows operating systems, it is the F9 key. r) What happens when you press F9?
Note that the CITY column is a bit too wide and that the STATE column is not visible. s) What can you do to make the LOV easier to read?
go to contents
Set the LOVs Automatic Position property to No. Set the LOVs X Position and Y Position properties to 0, 0. Run the form, issue a query, and display the LOV. b) Where is the LOV displayed? Are the X and Y positions relative to the canvas, the window, or the entire screen?
Exit the form and return to the Form Builder. c) What is the property that will display the LOV as soon as the user navigates to the STUDENT.ZIP item? Do not set this property. Simply write its name as your answer.
Set Auto Skip to Yes. Run the form and issue a query. Tab to the ZIP item and display the LOV. Select any Zip Code value from the list and click the OK button. d) Which item did the form navigate to after it closed the LOV? Why might Auto Skip be convenient for your users?
go to contents
Look at the properties for the STUDENT.ZIP item and note the ones under the LOV category. Assume that the Validate from List property is set to Yes. e) What will happen if a user manually enters a value into STUDENT.ZIP that is not in the ZIPCODE table?
So far, you have been displaying the LOV by pressing the F9 key on your keyboard. f) What type of Forms item could you use to display the LOV?
Create a button and name it SHOW_LIST. Label it List. Make sure it is listed as the last item in the STUDENT block in the Object Navigator. Set its Mouse Navigate property to No. Create a WHEN-BUTTON-PRESSED trigger. Type the following code into the PL/SQL editor:
LIST_VALUES;
Run the form and issue a query. Keep the cursor in the STUDENT_ID item. Click the SHOW_LIST button. g) Was the LOV displayed?
Navigate to ZIP and click the SHOW_LIST button again. h) Did it work this time? What does this tell you about the way the LIST_VALUES built-in displays LOVs?
go to contents
Exit the form and return to the Form Builder. Select the WHEN-BUTTONPRESSED trigger in the Object Navigator and right-click to open the PL/SQL Editor. Erase the code that is there and replace it with the following. Click the Compile button when you are finished.
DECLARE v_lov boolean; BEGIN v_lov := SHOW_LOV(ZIP_LIST); END;
Run the form and issue a query. Keep the cursor in the STUDENT_ID item. Click the SHOW_LIST button. i) Was the LOV displayed? What does this tell you about the way the SHOW_LOV built-in displays LOVs?
go to contents
SOURCE PAGE
c) Are there any record group objects in the R_STUDENT.fmb form? Answer: No, because you havent done anything to create one. A record group is like a database table in that it is made up of columns and rows. However, record groups belong to a form and are stored locally on a users machine rather than in a database instance. The groups structure can be defined in three ways: 1) 2) 3) By a query at design-time. Statically at design-time. Programmatically at run-time.
The LOV you create in this Exercise will have a record group based on a query. d) If there were existing record groups, would you be able to use one of them here as the source of your LOV? How? Answer: Yes you would by selecting Existing Record Group on the Source Page and choosing a record group from the list item. There are two reasons for basing an LOV on an existing record group. First, it allows you to have multiple LOVs based on the same record group. Second, it allows you to base the LOV on a static record group. A static record group is not based on a query. Instead, its source is an array of static values that you define at design-time. The wizard only creates record groups based on queries; it cannot help you define static record groups. You would have to create and define the static record group manually and then reference it from the LOV Wizards Source Page. WHAT THE WIZARD WILL DO ON THIS PAGE: Since you selected New Record Group based on a query, the wizard will create a record group object along with the LOV object. The next few wizard pages will define some of the properties for the record group.
go to contents
e) What does the Build SQL Query button do? Do not use it to build the query, simply click the button to see what it does. Click Cancel when you are done. Answer: It opens the Query Builder. The Query Builder is a graphical tool for building SQL statements. It allows you to point and click at tables and their columns to specify the syntax of the SQL statement. It is useful for complicated statements that access multiple tables and columns, or require many joins. It is also helpful for those with poor SQL skills. The Query Builder is unnecessary for this LOV because the SQL statement is rather simple and can be typed in quickly. f) Can you import an existing SQL statement? Where will Forms look for it? Do not import an SQL statement. Simply click the button to see what it does. Click Cancel when you are done. Answer: Yes you can. Forms will look for it in the filesystem. g) What should the SQL statement be for this record group? Type it into the SQL Statement field. Answer: The SQL statement should be: SELECT zip, city, state FROM zipcode ORDER BY state, city
h) How can you confirm that the SQL you have written is correct? Answer: You can click the Check Syntax button. i) Should you put a semi-colon after the query? Answer: No you should not. If you put a semi-colon after the query, the wizard will return an Invalid Character, Invalid SQL Statement error. WHAT THE WIZARD WILL DO ON THIS PAGE: The wizard will take the query you have written here and store it the record groups Record Group Query property. You can edit the query later either by reentering the wizard or by adjusting the SQL Query Statement property directly for the record group.
go to contents
I FOR EXAMPLE:
Assume you build a record group called STUDENT_REC based on the following query: SELECT student_id, first_name, last_name FROM student ORDER BY last_name; There is a requirement that the application must have two LOVs: one that displays a students student ID, last name, and first name, and another LOV that displays only the students last name and first name. The columns for both LOVs exist in the STUDENT_REC record group, so you can use STUDENT_REC for both LOVs. The wizards Column Selection page displays all of the columns included in the record group and allows you to indicate which columns each LOV should include. WHAT THE WIZARD WILL DO ON THIS PAGE: The wizard will begin to populate the new LOVs Column Mapping Properties property. This property has multiple values, which are displayed and configured in the LOV Column Mapping dialog shown in Figure 7.2. This property is one of the most important for an LOV because it not only defines which columns will be displayed, but also each columns return item (which item on the form the LOV column will populate) and the width and title for the LOV column.
go to contents
Figure 7.2 I The Column Mapping Properties propertys LOV Column Mapping dialog.
The wizard configures column display, return items, column width, and the title for an LOV over a series of pages. Once completed, it gathers all of the information you have entered in the separate pages and configures the Column Mapping Properties property. If you were to build or edit an LOV manually, you would configure these same settings yourself in the Form Builder by clicking the Column Mapping Properties property for the LOV to open the LOV Column Mappings dialog.
go to contents
go to contents
o) How often do state names, city names, and their Zip Codes change? Do you think it is necessary for the form to issue the record group query to refresh the LOV every time it is displayed? Answer: They dont change very often and are unlikely to change during a users session. Therefore, it is not necessary to refresh the LOV each time it is displayed. If Refresh record group data before displaying LOV is checked, the record groups query will be issued each time the user displays the LOV. This will require resources from the database as well as resulting in increased traffic on the network. The values in the ZIPCODE table are fairly static and will not be changed or updated very often. Therefore, it is not necessary to refresh the record group each time the LOV is displayed. However, what if you had created an LOV on student_id, first_name, and last_name? It is likely that the data in the STUDENT table will change during the course of a users session, so it would be wise to refresh the LOV each time it is displayed despite the database and network resources required. WHAT THE WIZARD WILL DO ON THIS PAGE: The wizard will set the Record Group Fetch Size property for the record group. It will also set the Automatic Refresh property for the LOV. It can also set the Filter Before Display property for the LOV. If set to Yes, this property will present the user with a small filter window to let them filter the values returned to the LOV.
I FOR EXAMPLE:
Assume the user wishes to display the Zip Codes LOV, but they only want to see Zip Codes that begin with 07. If Filter Before Display is set to Yes, they will be presented with the dialog shown in Figure 7.3. Once they enter their filter criteria, the record group query will be adjusted and then issued. When the LOV opens, it will not include any records that are outside the filter criteria. This will not only make the list more focused for the user, but it will lessen the amount of rows that have to be fetched from the database.
ITEMS PAGE
So far you have created a record group and created and configured an LOV. p) How can you indicate that you want the LOV to be attached to the STUDENT .ZIP item? Answer: Move STUDENT.ZIP from the Return items column to the Assigned items column.
go to contents
go to contents
Both of these changes can be done either by reentering the wizard or by adjusting the properties directly in the Property Palette. It is probably easiest to make the CITY column a bit narrower by setting it to 85 in the Property Palette.
7.1.2
ANSWERS
a) When you displayed the LOV in the previous questions, where was it displayed on the screen? Which of the LOVs properties determined that? Answer: It was displayed just below STUDENT.ZIP, which is the item it is attached to. The LOVs Automatic Position property determined this. The Automatic Position property will relieve you from having to position the LOV yourself. It always positions the LOV just below the item that the LOV is attached to. b) Where is the LOV displayed? Are the X and Y positions relative to the canvas, the window, or the entire screen? Answer: It was displayed in the upper right-hand corner. The X Position and Y Position properties are relative to the entire screen. Since the position properties are relative to the entire screen, you can position the LOV outside the canvas and window that contain the current items, or even outside the entire Forms Runtime MDI window. This can be helpful if you dont want the LOV to cover any of the items or objects on the form. The item that an LOV is attached to also has LOV X Position and LOV Y Position properties. The item-level position properties override the LOV position properties. This can be helpful if the LOV is reused on multiple forms and you dont want the application to determine the position of the LOV automatically. c) What is the property that will display the LOV as soon as the user navigates to the STUDENT.ZIP item? Do not set this property. Simply write its name as your answer. Answer: The LOVs Automatic Display property will display the LOV as soon as the user navigates into it. Set Auto Skip to Yes. Run the form and issue a query. Tab to the ZIP item and display the LOV. Select any Zip Code value from the list and click the OK button.
go to contents
go to contents
While this is acceptable behavior, it creates a small problem. The user must use the mouse to click the button to open the LOV. He cannot simply press the ENTER key to open the LOV, which may be more convenient during data entry. You can get around this by adjusting the WHENBUTTON-PRESSED trigger slightly.
I FOR EXAMPLE:
Change the trigger code so that it now looks like this: GO_ITEM('STUDENT.ZIP'); LIST_VALUES; Run the form and issue a query. Tab through the items until the cursor rests on the SHOW_LIST button and press the ENTER key. The LOV should be displayed. The GO_ITEM built-in causes the form to navigate to the STUDENT.ZIP item so that the LIST_VALUES built-in functions properly. Select the WHEN-BUTTON-PRESSED trigger in the Object Navigator and right-click to open the PL/SQL Editor. Erase the code that is there and replace it with the following code. Click the Compile button when you are finished. DECLARE v_lov boolean; BEGIN v_lov := SHOW_LOV('ZIP_LIST'); END; i) Was the LOV displayed? What does this tell you about the way the SHOW_LOV built-in displays LOVs? Answer: See discussion below. The SHOW_LOV built-in is also used to display LOVs, but its functionality is slightly different from that of LIST_VALUES. It does not require that the LOV be attached to an item and it allows you to call an LOV explicitly by name. Note that the parameter for SHOW_LOV is ZIP_LIST, the name of the LOV. Another interesting feature is that SHOW_LOV is a Boolean function that will return TRUE if the user picks a value from the LOV and FALSE if they press the LOVs Cancel button. This can be useful to augment processing depending on the users behavior.
go to contents
I FOR EXAMPLE:
You can have the form issue messages depending on whether or not the user selects a value from the LOV. Edit the code for the WHEN-BUTTONPRESSED trigger so that it appears as it does below: DECLARE v_lov boolean; BEGIN IF NOT SHOW_LOV('ZIP_LIST') THEN MESSAGE ('The user cancelled the LOV'); ELSE MESSAGE ('The user selected from the LOV'); END IF; END; Run the form and issue a query. Experiment with selecting from the LOV and simply canceling it. Note the different messages that appear on the status line. The commands you put into the THEN and ELSE expressions can be as simple as these or more complicated, depending on how you want the form to react.
go to contents
8) What value will the SHOW_LOV built-in return? a) ____ The value selected from the LOV b) ____ TRUE if the user selects from the LOV, FALSE if they cancel the LOV c) ____ The position of the LOV d) ____ The LOVs object ID Quiz answers appear in Appendix A, Section 7.1.
go to contents
248
LAB
LAB 7.2
7.2
ALERTS
LAB OBJECTIVES
After this Lab, you will be able to:
Alerts are windows that contain messages and buttons. They provide a convenient way to send a message to alert the user that something has caused an error, something they have done will have certain consequences, or that something has happened that they should simply know about. In previous Exercises and Chapters, you used the MESSAGE built-in to send messages to the user. It was convenient and easy, but had two limitations: there was no guarantee that the user would see the message, and there was no way for the user to respond to the message. Alerts overcome these two limitations in that they appear in the middle of the screen and they require that the user react to them. They can also be configured to let the user make a decision as to how to proceed in response to the alerts message.
I FOR EXAMPLE:
In Chapter 6, Triggers & Built-ins you used the MESSAGE built-in to send a message to the user reading, This Zip Code does not exist in the database. Please re-enter another. There is a chance that the user may miss this type of message since it is a single line of text that appears at the bottom left-hand corner of the MDI windows hint line. You could use an alert instead to display the same message as shown in Figure 7.4. Alert objects are displayed in the center of the users screen so they are easy to see. Additionally, the user is not able to return to the form and continue processing until she presses the OK button to respond to the
go to contents
249
LAB 7.2
CREATE ALERTS
Alerts are created and configured using the Object Navigator and Property Palette. They are fairly simple objects with only seven Functional properties. You will explore these in the Exercises. Like other visual objects, they have Color and Font properties as well. Since an alert is a separate window, you do not position it on a canvas. Therefore, it is not visible in the Layout Editor, or anywhere else in the Form Builder at design-time. The only way to see an alert is to run the form and display it.
DISPLAY ALERTS
Alerts are displayed using built-ins in much the same way LOVs are displayed with the SHOW_LOV built-in. To display an alert, you would use the following built-in function:
go to contents
250
LAB 7.2
DECLARE v_alert_button NUMBER; BEGIN v_alert_button := SHOW_ALERT('ALERT1'); END; In the example above, there is no code to respond to the buttons: the alert is simply being displayed. This code could be used to display an alert that has only one button like the one pictured in Figure 7.4. When the button is clicked, the alert will close and control will be returned to the form. When you create alerts that have more than one button, you need to write code to respond to the button that has been pressed. The SHOW_ALERT built-in returns a number that corresponds to the value of the alert button that was pressed. The three-button alert in Figure 7.5 would require that code be written so that the form could respond differently for each alert button. It would look something like this: DECLARE v_alert_button NUMBER; BEGIN v_alert_button := SHOW_ALERT('ALERT1'); IF v_alert_button = ALERT_BUTTON1 THEN --code to let the user enter another value ELSIF v_alert_button := ALERT_BUTTON2 THEN --code to display an LOV for zipcodes ELSIF v_alert_button := ALERT_BUTTON3 THEN --code to cancel END IF; END; The constants ALERT_BUTTON1, ALERT_BUTTON2, and ALERT_BUTTON3 correspond to the buttons in the alert. As with LOVs, and other objects, you could use the FIND_ALERT built-in to get the alerts object ID.
go to contents
251
LAB 7.2
b) How do you think you can change the alerts Button Label 2 property so that it will not be displayed to the user?
Change the Message property so that it reads You pressed the Exit button. Create a WHEN-BUTTON-PRESSED trigger for the ZIPCODE .EXIT button c) What code could you write behind the WHEN-BUTTONPRESSED trigger to display this alert? For this question, your code does not have to respond to the alert buttons.
Run the form and click the Exit button. Note the alerts icon. You will need to know it to respond to Question e.
go to contents
252
Lab 7.2: Alerts d) Can you return to the form without responding to the alert?
LAB 7.2
Click the alerts OK button, then exit the form using the toolbars Exit button to return to the Form Builder. e) What icon was displayed inside the alert? Which of the alerts properties determined the icon?
Change the Message property to Are you sure you want to exit the form? Change the value of Button Label 1 to Yes and the value of Button Label 2 to No. f) How should you change the code in the WHEN-BUTTON-PRESSED trigger so that the form will exit if the user chooses Button 1 from the alert? If the user chooses Button 2, send a message to the hint line that reads You have chosen to continue.
Run the form and test the alert. Exit the form and return to the Form Builder. Set the Default Alert Button property to Button 2. Run the form and display the alert. g) How did the Default Alert Button property affect the alert?
Exit the form and return to the Form Builder. h) What built-in could you use if you needed to know an alerts object ID? Can you set any of the alerts properties programmatically?
go to contents
253
i) How could you change this form so that the alert appears whether the user clicks the ZIPCODE.EXIT button or the Exit button on the toolbar? Do not make these changes to the form, simply write your answer below.
LAB 7.2
go to contents
254
LAB 7.2
I FOR EXAMPLE:
To set a buttons label to Yes at run-time, you would use the following statement: SET_ALERT_BUTTON_PROPERTY('EXIT_ALERT', ALERT_BUTTON1, LABEL 'Yes'); Change the Message property so that it reads You pressed the Exit button. Create a WHEN-BUTTON-PRESSED trigger for the ZIPCODE.EXIT button. c) What code could you write behind the WHEN-BUTTON-PRESSED trigger to display this alert? For this question, your code does not have to respond to the alert buttons. Answer: See the discussion below. The code would be as follows: DECLARE v_alert_button NUMBER; BEGIN v_alert_button := SHOW_ALERT('EXIT_ALERT'); END; In this example, all you are doing is showing the alert in response to the Button Pressed event. There is no need to evaluate which alert button was pressed since there is only one button on the alert. This technique is useful if you simply want to display an important message to a user.
I FOR EXAMPLE:
In Chapter 10, Reusable Code, you will use an alert to tell the user that validation of an item has failed. No decision will be required of the user, so the alert will have only one button labeled OK. The code to call the
go to contents
255
alert will be similar to the code above since there will only be a need to display the alert, not evaluate any buttons. Run the form and click the Exit button. Note the alerts icon. You will need to know it to respond to Question e. d) Can you return to the form without responding to the alert? Answer: No you cannot. The alert is displayed in what is called a modal window. Modal means that the user cannot leave the window until she has exited it. You will learn more about modality in Chapter 8, Canvases and Windows. In the case of alerts, users exit them by selecting one of their alert buttons. This makes alerts very useful when you want to require users to respond to a question or a message before allowing them to continue. Click the alerts OK button, then exit the form using the toolbars Exit button to return to the Form Builder. e) What icon was displayed inside the alert? Which of the alerts properties determined the icon? Answer: A red stop light. The Alert Style property. The three alert styles, Stop, Caution, and Note, let you communicate the urgency of an alerts message to the user. Stop alerts should be used for the most urgent messages. Caution alerts can be used for those that are rather urgent and require the user to make a choice. In this Exercise, the message you will present to the user, Are you sure you want to exit the form?, should have Caution as its Alert Style. Note-style alerts are for those messages that are important, but do not require a response. They will usually only have one button labeled OK. The alert you will use in Chapter 10, Reusable Code, will have Note as its Alert Style. Change the Message property to Are you sure you want to exit the form? Change the value of Button Label 1 to Yes and the value of Button Label 2 to No. Change the Alert Style to Caution. f) How should you change the code in the WHEN-BUTTON-PRESSED trigger so that the form will exit if the user chooses Button 1 from the alert? If the user chooses Button 2, send a message to the hint line that reads You have chosen to continue. Answer: See the discussion below.
LAB 7.2
go to contents
256
LAB 7.2
I FOR EXAMPLE:
If you can predict that your users will normally select Button 1, then you should set the Default Alert Button property to Button 1. That way, when the alert is displayed, the user can simply press ENTER to respond to the alert and close it. Exit the form and return to the Form Builder.
go to contents
257
h) What built-in could you use if you needed to know an alerts object ID? Can you set any of the alerts properties programmatically? Answer: You could use FIND_ALERT and SET_ALERT_PROPERTY. FIND_ALERT will find the alerts object ID, which you can then use in the SHOW_ALERT built-in or the SET_ALERT_PROPERTY built-in. You can use the SET_ALERT_PROPERTY built-in to set the title of the alert and the alerts message text programmatically.
LAB 7.2
I FOR EXAMPLE:
The code to set an alerts title and message text at run-time could look like this:
DECLARE v_alert_id v_alert_button BEGIN v_alert_id := FIND_ALERT('EXIT_ALERT'); SET_ALERT_PROPERTY(v_alert_id, TITLE, 'Warning'); SET_ALERT_PROPERTY(v_alert_id, ALERT_MESSAGE_TEXT, 'Are you sure you want to exit the form?'); v_alert_button := SHOW_ALERT(v_alert_id); ... END; NUMBER; ALERT;
Note that, as always, you could also include the ID_NULL built-in to evaluate whether or not the alert object exists. i) How could you change this form so that the alert appears whether the user clicks the ZIPCODE.EXIT button or the Exit button on the toolbar? Do not make these changes to the form, simply write your answer below. Answer: You could use a KEY-EXIT trigger to show the alert. If you moved the code to show the alert to a KEY-EXIT trigger, the alert would be shown whenever the user pressed the Exit key on the keyboard. It would also fire whenever the user clicked the Exit button on the toolbar. To get the alert to be shown when the user clicks the Exit button on the canvas, you would also have to change the code in the WHEN-BUTTONPRESSED trigger to the following: DO_KEY('EXIT_FORM'); This way, when the user clicks the button, it fires the KEY-EXIT trigger as well.
go to contents
258
LAB 7.2
go to contents
259
CHAPTER
go to contents
CHAPTER
o far, your forms have been limited to the windows and canvases that Forms has created for you. As your forms get more complex, you will want to add additional windows and canvases and display them in response to interface or internal processing events. This Chapter will cover the fundamentals of canvas and window objects, as well as a variety of methods for displaying them.
261
go to contents
LAB
8.1
For a canvas and its items to be visible, it must be assigned to a window. By the same token, for a window to be visible, it must contain at least one canvas. Therefore, canvases and windows rely on each other to make themselves and other Forms objects appear to the user. The forms you have created so far have had one window and one canvas. The window (WINDOW1) was created by default when the new form was created. The canvas was created by the Layout Wizard and automatically assigned to WINDOW1. A single form can contain multiple instances of both canvases and windows, which you can create and define yourself.
WINDOWS
Like any other physical object in Forms, windows have properties that determine their size and position. Under the Functional property category, they have a series of window-specific properties that determine whether or not the window can be minimized, resized, closed, and so on. There are two styles of windows in Forms: document and dialog. Document windows are commonly used for data entry and query screens; they are the main windows in an application. Dialog windows are used to deliver messages, display lists, or present the user with a specific task. On Microsoft Windows operating systems, there is a third style of window
go to contents
go to contents
Figure 8.2 I A dialog window outside the boundaries of the MDI window.
I FOR EXAMPLE:
In Figure 8.2, the LOV titled Dialog Window is modal. The user must either select from the LOV or click the Cancel button before going back to the window titled Document Window. The user cannot navigate to and from other windows. Modal windows give you the opportunity to present the user with a message or task and force them to respond to it before moving on. If a window is modeless, the user can navigate to another window and leave the current window open and accessible. Modeless windows let you give your users the convenience of being able to switch from window to window. Document windows are usually modeless (Modal property set to No) and dialog windows are usually modal (Modal property set to Yes). The WINDOW1 object created by the Form Builder is a modeless document window.
go to contents
There are a variety of methods for displaying and hiding windows at runtime. You can explicitly open windows by using window built-ins, you can set their properties, or you can simply navigate to them. SHOW_WINDOW is a window-specific built-in that can be used to open all types of windows, be they modal or modeless, document or dialog.
I FOR EXAMPLE:
You could open a window named MAINWIN with the following statement: SHOW_WINDOW('MAINWIN'); Depending on the modality of MAINWIN and how its properties are set, the SHOW_WINDOW statement can produce different results in how the window is opened. It may display the windows content canvas or not. It may try to navigate to an item in the window or not. In the Exercises, you will experiment with some of the different outcomes of SHOW_WINDOW. You can also open a window simply by navigating to an item in that window, or to a block that contains an item in the window. Keep in mind that for an item to be in a window it must be assigned to a canvas that is assigned to that window. If navigation has moved to a certain item and that item is not displayed or is not visible, Forms will make it visible. It will, among other things, display the canvas and window that item is assigned to. So by using built-ins to navigate to an item, you can open/display canvases and windows.
I FOR EXAMPLE:
Assume you have the following code in a trigger: GO_ITEM('BLOCK1.ITEM1'); Also assume that ITEM1 is on CANVAS1, which is assigned to MAINWIN. When Forms navigates to ITEM1, it must display it. So, if ITEM1 is not visible when the GO_ITEM built-in is executed, Forms will automatically show the canvas and window to make ITEM1 visible to the user. There are numerous combinations of property settings and programmatic or navigational methods that affect the opening and closing of windows
go to contents
CANVASES
So far, you have worked with content canvases to display and position the items in your form. Content canvases are the most common canvas type because every window must have one as its main canvas. In the Exercises in the rest of this Chapter, you will experiment with two other types of canvases: stacked and toolbar. Also, in the Test Your Thinking section, you will attempt to build a form with a tabbed canvas. In this Lab, you will focus on some concepts that are common to all types of canvases.
go to contents
edges of the canvas, but the center of the canvas, the part with the painting, is visible. Think of the part that is visible as the viewport. The size of the canvas and the size of its viewport are determined by properties. Figure 8.3 shows what a canvas and its viewport look like in the Layout Editor. The viewport in Figure 8.3 is represented by the thin black line that surrounds the items and the frame. The canvas is represented by the larger rectangular area. The canvas ends where the surface of the Layout Editor appears to have raised dots. On a painters canvas, the painting is rarely ever behind the frame, or in any way hidden from the viewer. This is not necessarily the case with Forms canvases. It is common to place some items outside the viewport so that they are not initially visible. Then, as the user scrolls or navigates the form, the viewport can move to expose different parts of the canvas. In the Test Your Thinking section of this Chapter, you will be asked to create a canvas whose viewport will move to show and hide items and create a scrolling effect.
DISPLAYING CANVASES
The rules for displaying and hiding canvases are similar to those for windows. A canvas can be displayed either by using built-ins to open it explicitly or by navigating to an item on the canvas. If you choose to display a canvas by navigating to it, you could use the same GO_ITEM built-in explained in the example above. If you choose to display a canvas explicitly, you could use the SHOW_VIEW built-in and pass it the canvas name. The code would look like this for a canvas named CANVAS1: SHOW_VIEW('CANVAS1');
go to contents
Click the Alert button. b) Can you drag the alert out of the MDI window? What does this tell you about its window style?
c) Can you navigate to the form without closing the alert? What does this tell you about its modality?
Click the OK button to close the alert. Click the Window button. d) What style is this window? Is it modal?
e) Have any properties been set for the MDI window? How can you tell?
Exit the running form and return to the Form Builder. Open the WHEN-BUTTON-PRESSED trigger for the CONTROL.SHOW_HIST button. f) Is this built-in opening the window explicitly?
Comment out the current code in the SHOW_HIST buttons WHEN-BUTTON-PRESSED trigger and then add the code shown below:
--GO_ITEM('CONTROL.HIDEWIN); SHOW_WINDOW('SECONDWIN');
go to contents
Lab 8.1: Canvas and Window Concepts Run the form and click the Window button.
g) Did the window open? Are there any items visible? What does this mean about opening a window with navigation vs. doing it programmatically?
Reenter the SHOW_HIST buttons WHEN-BUTTON-PRESSED trigger and change it back so it appears as it does below:
GO_ITEM('CONTROL.HIDEWIN); --SHOW_WINDOW('SECONDWIN');
h) Which built-ins are being used in the HIDE_HIST buttons WHEN-BUTTON-PRESSED trigger?
Comment out HIDE_WINDOW('SECONDWIN'); in the HIDE_HIST buttons WHEN-BUTTON-PRESSED trigger. Run the form and press the Window button. Now click the Close button. i) Did navigation return to the ZIP item on the MAINWIN? Did the SECONDWIN close? What does this tell you about navigation in and out of modeless windows?
go to contents
Lab 8.1: Canvas and Window Concepts b) Which window is the ZIPCODE canvas assigned to? Which window is the HISTORY canvas assigned to?
Open the STACKED buttons WHEN-BUTTON-PRESSED trigger. d) How is the HISTORY canvas being displayed?
Run the form and test the Stacked Canvas button to see how the stacked canvas will behave.
go to contents
forth between them, just like in MSWord and other typical windowing applications. Click the Alert button. b) Can you drag the alert out of the MDI window? What does this tell you about its window style? Answer: Yes you can. It is a dialog window. Alerts are good examples of simple dialog windows since they allow the user to read the alert message and then drag it completely out of the way to view the form before responding. c) Can you navigate to the form without closing the alert? What does this tell you about its modality? Answer: No you cannot. It is a modal window. Dialog-style windows are often implemented as modal. They are windows that require some sort of response from the user before regular processing can continue. The alert in EX08_01.fmb will not let the user continue until they have clicked the OK button. In this case, the alert was merely informational. The response required can be seen as an OK, I read the alert. But other alerts and modal dialogs require that the user make a decision or complete some tasks.
I FOR EXAMPLE:
Figure 8.4 shows the LOV Column Mapping dialog from the properties of an LOV. This modal dialog requires that the user enter some information before continuing.
I FOR EXAMPLE:
The statements in the PRE-FORM trigger look like this: SET_WINDOW_PROPERTY(FORMS_MDI_WINDOW, WIDTH, 550); FORMS_MDI_WINDOW is a Forms constant. It is not possible to refer to the MDI window by name in single quotes or with an object ID. Exit the running form and return to the Form Builder. Open the WHENBUTTON-PRESSED trigger for the CONTROL.SHOW_HIST button. f) Is this built-in opening the window explicitly? Answer: No, the window is being opened by navigation. This is a fairly straightforward method for opening a window and its canvas. Once you have told Forms which item it should navigate to, it does the rest of the work by opening the window and the appropriate canvas.
go to contents
Comment out the current code in the SHOW_HIST buttons WHEN-BUTTON-PRESSED trigger and then add the code shown below: --GO_ITEM('CONTROL.CLOSE_HIST); SHOW_WINDOW('SECONDWIN'); Run the form and click the Window button. g) Did the window open? Are there any items visible? What does this mean about opening a window with navigation vs. doing it programmatically? Answer: Yes, the window opened. No, there are no items visible. In Question f, navigation automatically opened the window and canvas. Here you can see that opening the window does not automatically display the canvas or its items. To get the canvas and its items to appear, you would have had to add another line of code to the trigger or adjust one of the windows properties.
I FOR EXAMPLE:
To get the canvas and its items to open along with the window, you could change the code to the following: SHOW_WINDOW('SECONDWIN'); SHOW_VIEW('SECONDCAN'); In this case, you are explicitly opening both the window and its canvas. The other option is to use properties to manipulate the window and canvas. In that case, you would set the SECONDWIN windows Primary Canvas property to SECONDCAN. By doing this, SECONDCAN will open automatically when SECONDWIN is opened. Reenter the SHOW_HIST buttons WHEN-BUTTON-PRESSED trigger and change it back so it appears as it does below: GO_ITEM('CONTROL.HIDE_HIST); --SHOW_WINDOW('SECONDWIN'); h) Which built-ins are being used in the HIDE_HIST buttons WHEN-BUTTONPRESSED trigger? Answer: The GO_ITEM built-in, which is navigational, and the HIDE_WINDOW builtin, which explicitly closes the window.
go to contents
8.1.2 ANSWERS
Open the EX08_02.fmb form in the Form Builder.
a) How many canvases are there and what are their names and types? Answer: There are two canvases: ZIPCODE and HISTORY. ZIPCODE is a content canvas and HISTORY is a stacked canvas. b) Which window is the ZIPCODE canvas assigned to? Which window is the HISTORY canvas assigned to? Answer: Both canvases are assigned to MAINWIN. A canvas must be assigned to a window to be visible. A stacked canvas must always be stacked on a content canvas to be visible. Therefore, both canvases are assigned to the same window. It is technically possible to display a stacked canvas in a window without a content canvas, but it is not recommended.
go to contents
Canvases, be they stacked or content, do not have to be explicitly assigned to windows at design-time. Instead, they can be assigned programmatically at run-time. c) How big is the HISTORY canvas? How big is its viewport? Answer: The Height and Width properties of the HISTORY canvas are 200, 300. Its view, or viewport, is 122,200. The properties that govern the size of the actual canvas are under the Physical category and are called Height and Width. The properties that govern the size of the viewport are under Viewport category and are called Viewport Height and Viewport Width. In this case, as in many, the canvas is far larger than the viewport. It is quite common to place items on the canvas but not include them in the viewport. Open the STACKED buttons WHEN-BUTTON-PRESSED trigger. d) How is the HISTORY canvas being displayed? Answer: It is being displayed programmatically using the SET_VIEW_PROPERTY built-in. The same rules that apply to windows apply to canvases. It is easiest to display a canvas using navigation, but it is possible and sometimes necessary to display a canvas programmatically. In this case, it was necessary to display the HISTORY canvas programmatically because it does not contain any navigable items. As an alternate and equally effective method to using the SET_ built-in, the SHOW_VIEW and HIDE_VIEW built-ins would have worked equally well here also.
go to contents
go to contents
277
LAB
8.2
LAB 8.2
While content canvases have viewports, they do not have properties to set the size of their viewports. The size of a content canvas viewport is the same as the size of the window that it occupies.
I FOR EXAMPLE:
Assume that CANVAS1 is the content canvas for WINDOW1. If WINDOW1s Height and Width properties are set to 200, 200, then this will be the size of CANVAS1s viewport. To set the size of a content canvas viewport, you must either: 1) 2) Use the Height and Width properties of the window it is assigned to. Adjust it visually in the Layout Editor.
In the Exercise for this Lab, you are going to walk through the process of creating a content canvas and window. Your objective will be to duplicate the functionality of the form EX08_01.fmb. First you are going to create a basic form using the wizards. Then you will create a second canvas and window. You will use these to display the values of the audit columns. You will then create a button to show this window and experiment with showing it and hiding it with navigation.
go to contents
278
Rename the canvas HISTORY. b) Do you want HISTORY to be a content canvas for MAINWIN? Why not?
c) What steps should you take to create a window and assign HISTORY as its content canvas?
Rename the window HISTWIN. Users should be able to drag HISTWIN outside the MDI window, and they should not be able to access other windows while it is open. d) Which of the HISTWIN windows properties should you set to make the above happen?
go to contents
279
e) Should you assign the audit items to the HISTORY canvas or to the HISTWIN window? Whats the easiest way to do this?
LAB 8.2
g) What should the item type for the audit columns be since you dont want users to be able to navigate to them or change their values?
Arrange the audit items and their prompts on the HISTORY canvas so that the layout is similar to that in Figure 8.5. Create a non-base table block and name it CONTROL. Create two buttons in the CONTROL block and name them SHOW_HIST and HIDE_HIST. Position SHOW_ HIST on the STUDENT canvas in the lower right-hand corner. Position HIDE_ HIST on the HISTORY canvas below the items in the center. Set the Key Board Navigable and Mouse Navigable properties for both buttons to No.
Figure 8.5 I The audit items arranged on the HISTORY canvas. go to contents
280
Lab 8.2: Content Canvases and Windows h) What navigation built-in could you use to open the HISTWIN window and its canvas?
LAB 8.2
i) Could you pass this built-in CREATED_BY as its parameter? Why not?
Create a WHEN-BUTTON-PRESSED trigger for the SHOW_HIST button with the following code:
GO_ITEM('CONTROL.HIDE_HIST');
Create another WHEN-BUTTON-PRESSED trigger for the HIDE_HIST button with the following code:
GO_ITEM('STUDENT.STUDENT_ID); HIDE_WINDOW('HISTWIN');
Run the form and test the buttons. k) Which objects X and Y Positions and Height and Width properties should you adjust to position and size the history information a bit better?
Try a Height of 130 and a Width of 190 to size the object. Adjust the X and Y Positions to taste, but try not to obscure any of the items on the STUDENT canvas.
go to contents
Lab 8.2: Content Canvases and Windows Run the form to test the size and positioning properties.
281
Navigate to STUDENT.ADDRESS and click the SHOW_HIST button. Now click the CLOSE_HIST button. Note that navigation did not return to STUDENT.ADDRESS, but to STUDENT.ZIP. l) When the Button Pressed event occurs, you want to capture the value of the current item so that you can navigate back to it when the HISTWIN window closes. What system variable can you use to do this?
LAB 8.2
Change the code in the SHOW_HIST button to use a global variable. The code should now look like this:
:global.last_item := :SYSTEM.CURSOR_ITEM; GO_ITEM('CONTROL.HIDE_HIST');
m) How can you use the global variable in the WHEN-BUTTONPRESSED trigger for HIDE_HIST so that it will return to the item that called it?
n) Will these windows, canvases, buttons, and triggers be useful in other forms that use the audit columns? Will the triggers work as they are written now?
go to contents
282
go to contents
283
Remember, items are assigned to canvases, and then canvases are assigned to windows. f) Can you view both canvases at the same time? How? Answer: Yes you can. You can open two instances of the Layout Editor. You can have as many instances of the Layout Editor open as youd like, which can make comparing canvases, especially stacked canvases, easier. You can also toggle from canvas to canvas using the Canvas list item on the Layout Editors Utility toolbar. In Lab 8.3, you will learn about another Layout Editor feature that allows you to view a content canvas with all of its stacked canvases stacked on top of it. g) What should the item type for the audit columns be since you dont want users to be able to navigate to them or change their values? Answer: The item type should be display item. h) What navigation built-in could you use to open the HISTWIN window and its canvas? Answer: You could use the GO_ITEM built-in. You could also have used the GO_BLOCK built-in to navigate to the CONTROL block. HIDE_HIST is the first item in the block, so navigation would have succeeded. i) Could you pass this built-in CREATED_BY as its parameter? Why not? Answer: No, because it is a display item, it is not navigable. The GO_ITEM built-in is only valid for items that are navigable. Display items are not navigable, so the navigation would have failed if you had tried to pass CREATED_BY into the GO_ITEM built-in. j) What is the only navigable item on the HISTORY canvas? Answer: The HIDE_HIST button. Even though the Keyboard Navigable property was set to No, SHOW_ HIST is considered a navigable item and the GO_ITEM built-in will succeed. Interestingly, as you learned in the previous question, this does not apply to display items since by definition they are non-navigable and have no Keyboard Navigable property.
LAB 8.2
go to contents
284
LAB 8.2
k) Which objects X and Y Positions and Height and Width properties should you adjust to position and size the history information a bit better? Answer: The HISTWIN windows properties. Since the HISTORY canvas is a content canvas, its sizing properties are ignored by the Form Builder and do not affect the size or position of the canvas at run-time. This is because the content canvas inhabits the entire surface of its window. Therefore, the size and position properties that you set for the window, in this case HISTWIN, are what dictate the size and positioning of the canvas. l) When the Button Pressed event occurs, you want to capture the value of the current item so that you can navigate back to it when the HISTWIN window closes. What system variable can you use to do this? Answer: :SYSTEM.CURSOR_ITEM. m) How can you use the global variable in the WHEN-BUTTON-PRESSED trigger for HIDE_HIST so that it will return to the item that called it? Answer: See discussion below. The code should be: GO_ITEM(:global.last_item); HIDE_WINDOW('HISTWIN'); The global variable created and assigned in the SHOW_HIST trigger is referenced in the HIDE_HIST trigger so that the form will know which item to return to. Global variables are user-defined variables that, like other variables, can hold values that need to be referenced later on. However, global variables are special in four ways: 1) 2) They dont have to be declared in a DECLARE statement of a PL/SQL block. They are assigned as they are created.
go to contents
285
They can be referenced outside the PL/SQL block they were created in. If multiple forms are open, they can all reference the global variable.
I FOR EXAMPLE:
In this case, the global variable is not declared in a DECLARE section of a PL/SQL block; it is assigned a value as it is created. It is created in one trigger and referenced in another. n) Will these windows, canvases, buttons, and triggers be useful in other forms that use the audit columns? Will the triggers work as they are written now? Answer: Yes they will. All of the objects that create the Record History window could be copied immediately to another form that has the audit columns in its data blocks. A little work would be required to assign and position the audit items on the HISTORY canvas, but once done, the objects would be completely usable. Why does this matter? Now that you have designed a way to let the user view the contents of the audit columns, you can apply it to every form you create that includes the audit columns. This way, you wont have to redo it for every form. In Chapter 9, Reusable Objects, you will learn a way to group these objects so that you can move them from form to form quickly and easily.
LAB 8.2
go to contents
286
LAB 8.2
4) Windows cannot be closed with navigation. a) ____ True b) ____ False Quiz answers appear in Appendix A, Section 8.2.
go to contents
287
LAB
8.3
STACKED CANVASES
LAB OBJECTIVES
After this Lab, you will be able to: LAB 8.3
Stacked canvases are never the sole canvas in a window. They are always stacked on top of other canvases, and partially or completely obscure those canvases when displayed at run-time. To stack a stacked canvas, you must position it relative to the content canvas that it is stacked upon. Figure 8.6 shows both a stacked and content canvas in the Layout Editor.
Figure 8.6 I A stacked canvas positioned on a content canvas in the Layout Editor. go to contents
288
LAB 8.3
The Viewport X Position on Canvas and Viewport Y Position on Canvas properties determine where the viewport will be positioned on the stacked canvas itself. A stacked canvas has two other properties called Viewport X Position and Viewport Y Position. These position the canvas relative to the content canvas.
I FOR EXAMPLE:
If the Viewport X Position of a stacked canvas is set to 40 (in points), and its Viewport Y Position is set to 178, that means that its upper left-hand corner will be at those coordinates on the content canvas as illustrated in Figure 8.7. In form EX08_02.fmb, you saw how a stacked canvas could be shown and hidden programmatically. In this example, there is only one stacked canvas, but there could have been many stacked on top of one another to simulate pages. A wizard can also be built using multiple stacked canvases.
I FOR EXAMPLE:
Assume you are going to create a wizard object to add student records and then add enrollment records for those students. The wizard will have three or more wizard pages to step the user through adding a students name, address, employment information, enrollment information, and so on. These pages will be implemented as stacked canvases and will be positioned on top of one another. The underlying content canvas will contain the wizards Back, Next, and Finish buttons. You will use navigation or SHOW_VIEW and HIDE_VIEW built-ins to switch from page to page.
go to contents
289
LAB 8.3
go to contents
290
Lab 8.3: Stacked Canvases a) What are three Form Builder tools you could use to create the stacked canvas?
The stacked canvas will ultimately lie on top of the content canvas.
LAB 8.3
b) With this in mind, which of the tools that you listed in Question a might be the best for creating and simultaneously positioning the stacked canvas?
Create the stacked canvas using the Stacked Canvas tool from the Layout Editors Tool Palette. Make sure you completely cover the ENROLLMENT items on the content canvas, but do not obscure or cover the frame. Name the new stacked canvas INSTRUCTOR. c) Are you able to see both the content canvas and stacked canvas in the Layout Editor? Is part of the content canvas obscured?
Select View | Stacked Views from the Main Menu. When the Stacked/ Tab Canvases dialog opens, de-select INSTRUCTOR and click the OK button. d) Are you still able to see both canvases? What does this tell you about the View | Stacked Views feature?
Select View | Stacked Views from the Main Menu again. When the Stacked/Tab Canvases dialog opens, re-select INSTRUCTOR and click the OK button so that INSTRUCTOR is visible on top of SECTION.
go to contents
291
In the Object Navigator, click the icon next to INSTRUCTOR to open another Layout Editor window. e) How can you position the items in the INSTRUCTOR block on the INSTRUCTOR canvas?
Run the form and issue a query. The Change View button will not work yet, so use Block | Next Block from the Main Menu to test the stacked canvas. Exit the form and return to the Form Builder. f) Which properties can you adjust to make the stacked canvas and its items look better? For example, isnt the stacked canvas set a bit lower than the content canvas?
LAB 8.3
g) Which of the stacked canvas properties can you change so that the canvas is not visible when the form opens?
Open the WHEN-BUTTON-PRESSED trigger for the CHANGEVIEW button. h) What trigger code could you write to change the canvas Visible property? Your trigger should first evaluate whether or not the canvas is visible. If the canvas is not visible, use a built-in to show it. If it is visible, use a built-in to hide it. Use conditional logic and GET_ and SET_ built-ins to accomplish this.
Run the form, issue a query, and test the Change View button.
go to contents
292
Lab 8.3: Stacked Canvases i) What are two other built-ins you could use to hide and show the canvas? These should not be GET_ or SET_ built-ins, and they should not cause navigation.
LAB 8.3
Run the form, issue a query, and test the Change View button. Exit the form and return to the Form Builder. Replace the built-ins you used for Question i with GO_BLOCK('INSTRUCTOR') and GO_BLOCK('ENROLLMENT') statements. Run the form, issue a query, and test the Change View button. j) How is the behavior of the form different now that you are using navigational built-ins instead of the built-ins you used in Questions h and i?
go to contents
293
By using the Layout Editor, you will be able to treat the stacked canvas like an item. You will be able to position the cursor where you would like the canvas to begin and stretch out the canvas until it reaches the size you desire. Create the stacked canvas using the Stacked Canvas tool from the Layout Editors Tool Palette. Make sure you completely cover the ENROLLMENT items on the content canvas, but do not obscure or cover the frame. c) Are you able to see both the content canvas and stacked canvas in the Layout Editor? Is part of the content canvas obscured? Answer: Yes, you can see both and yes, part of the content canvas is obscured. Select View | Stacked Views from the Main Menu. When the Stacked/Tab Canvases dialog opens, de-select INSTRUCTOR and click the OK button. d) Are you still able to see both canvases? What does this tell you about the View | Stacked Views feature? Answer: No, you cannot see both. The View | Stacked Views feature lets you view the size and position of the stacked canvases on their content canvases. The stacked canvas and its objects are accessible in this mode so that you can select and adjust their positions. In the previous Lab, you learned that you can have two instances of the Layout Editor open. This can also be helpful in that it lets you see the stacked canvas on its own. Additionally, the Stacked Views mode shows only the stacked canvas viewport. Select View | Stacked Views from the Main Menu again. When the Stacked/Tab Canvases dialog opens, re-select INSTRUCTOR and click the OK button so that INSTRUCTOR is visible on top of SECTION. In the Object Navigator, click the icon next to INSTRUCTOR to open another Layout Editor window. e) How can you position the items in the INSTRUCTOR block on the INSTRUCTOR canvas? Answer: Select all of the items in the Object Navigator and use the Property Palette to set their Canvas property to INSTRUCTOR. Note that it makes no difference which instance of the Layout Editor you use to manipulate the items on the INSTRUCTOR stacked canvas. You can
LAB 8.3
go to contents
294
LAB 8.3
Which properties can you adjust to make the stacked canvas and its items look better? For example, isnt the stacked canvas set a bit lower than the content canvas? Answer: You can set the Bevel property to None and re-arrange the items and their prompts.
g) Which of the stacked canvas properties can you change so that it is not visible when the form opens? Answer: You can set the INSTRUCTOR canvas Visible property to No. As you noticed for the INSTRUCTOR canvas, if the Visible property is set to Yes and the form has not navigated to an item on a canvas that will cover INSTRUCTOR, then INSTRUCTOR will be visible. The layering of INSTRUCTOR on top of SECTION determines the stacking order of the canvases. Like default navigation, the stacking order of canvases is determined by the order of the canvases in the Object Navigator. Since INSTRUCTOR is listed after the SECTION content canvas, it appears on top. If there had been other canvases after INSTRUCTOR, those would have appeared on top.
I FOR EXAMPLE:
Figure 8.8 shows the Object Navigator with two more stacked canvases added to this form. Note that CANVAS_3 and CANVAS_4 come after SECTION and INSTRUCTOR. Figure 8.9 shows these same canvases at run-time, just after the form has been opened. Note that because CANVAS_4 is the last canvas listed in the Navigator, it is displayed on top at run-time. The default stacking order of canvases is similar to default navigation in that it can be overridden programmatically at run-time. Open the WHEN-BUTTON-PRESSED trigger for the CHANGEVIEW button.
go to contents
295
LAB 8.3
296
LAB 8.3
The SHOW_VIEW and HIDE_VIEW built-ins would work just as well and can simply replace the SET_VIEW_PROPERTY statements. The code would be as follows: DECLARE v_visible VARCHAR2(50); BEGIN v_visible := GET VIEW PROPERTY('INSTRUCTOR', VISIBLE); IF v_visible = 'TRUE' THEN HIDE_VIEW('INSTRUCTOR'); ELSE SHOW_VIEW('INSTRUCTOR'); END IF; END; Run the form, issue a query, and test the Change View button. Exit the form and return to the Form Builder. Replace the built-ins you used for Question i with GO_BLOCK('INSTRUCTOR') and GO_BLOCK('ENROLLMENT') statements. Run the form, issue a query, and test the Change View button. j) How is the behavior of the form different now that you are using navigational built-ins instead of the built-ins you used in Questions h and i? Answer: The cursor moves to the first item in the block that is being navigated to. The GO_BLOCK built-in is similar to GO_ITEM in that it forces navigation. The form automatically navigated to the first item in the block being referenced in the built-in.
go to contents
297
LAB 8.3
go to contents
298
LAB
8.4
TOOLBAR CANVASES
LAB OBJECTIVES
After this Lab, you will be able to: LAB 8.4
A toolbar canvas properly positions toolbar items in a window. By specifying a canvas as either a vertical or a horizontal toolbar, Forms will position it horizontally along the top of a window or vertically along the left-hand edge of the window. Toolbars, like all canvases, are made visible by assigning them to windows. Unlike other canvas types, toolbar canvases can be assigned to the MDI window. You will experiment with assigning a toolbar to both a normal window and the MDI window in the Exercises. Since you are creating a toolbar, you must also create a group of items to place on it. In the Exercises below, you will create buttons. However, it is possible to include other item types on a toolbar. For example, the Utility toolbar in the Layout Editor includes two list items. In the Exercises, you will create a simple toolbar that mimics some of the functionality of the default toolbar that Forms provides for all forms. Once the toolbar is created and configured, you will learn how to reuse the toolbar by copying it from one form to another.
go to contents
299
This simple toolbar canvas will have five buttons: SAVE, EXIT, ENTER_QUERY, EXECUTE_QUERY, and CANCEL_QUERY. b) What type of object will you have to create to logically contain these items?
LAB 8.4
Create a block and name it TOOLBAR. Create five buttons. Name them SAVE, EXIT, ENTER_QUERY, EXECUTE_QUERY, and CANCEL_QUERY. They should appear in this order in the block. Assign the buttons to the TOOLBAR canvas. c) Do typical toolbars have text labels or iconic labels? Which property can you adjust to make your toolbar buttons typical?
Use the icons supplied by Oracle Developer. The item names are in CAPS and the corresponding icon names are in lower-case as follows: SAVE save, EXIT exit, ENTER_QUERY entqry, EXECUTE_QUERY exeqry, and CANCEL_ QUERY canqry.
go to contents
300
Lab 8.4: Toolbar Canvases d) Should the user be able to navigate to these buttons with the mouse or keyboard? Which properties should you set to ensure that they cant?
Adjust the size properties of the buttons and set Height to 19 and Width to 19. If you havent done so already, open the Layout Editor to view the TOOLBAR canvas. e) What Layout Editor feature can you use to quickly position all of the buttons next to each other? Note that if you used the Object Navigator to create the buttons, they may be currently positioned on top of each other. This should not pose a problem for you.
LAB 8.4
Adjust the Height of the TOOLBAR canvas to 21 to make the buttons fit better. f) What property can you use to give each button a Tool Tip?
Temporarily set the SAVE buttons Keyboard Navigable property to Yes. This will allow you to run the form and view the toolbar without having to create any other blocks or navigable items. You will switch it back later. Run the form. g) Which object is the toolbar attached to? Which of the TOOLBAR canvas properties indicate this?
go to contents
301
h) Can you assign the toolbar to the module? How do you think this will affect the position of the toolbar?
Assign the TOOLBAR to the module and run the form again. i) Where is the toolbar now? Is the original default toolbar still there?
Change the modules Menu Module property from DEFAULT&SMARTBAR to DEFAULT. Run the form. Create a WHEN-BUTTON-PRESSED trigger for the EXIT button with the following code:
DO_KEY('EXIT_FORM')
LAB 8.4
Use the wizards to quickly create a block based on the ZIPCODE table. Include all of the items in the block. Enforce data integrity should be unchecked. Stop when you get to the Layout Wizards canvas page. k) Should you position the ZIPCODE items on the TOOLBAR canvas? If not, what type of canvas should you create for them?
Continue creating the canvas for the ZIPCODE block. When you are finished, change the SAVE buttons Keyboard Navigable property back to No. Run the form and test that the toolbar appears properly.
go to contents
302
Lab 8.4: Toolbar Canvases l) What built-ins can you use to respond to the rest of the buttons? Create a WHEN-BUTTON-PRESSED trigger for each button and use DO_KEY statements with the built-ins.
m) Does it make sense to have the SAVE button enabled during Enter Query mode? Why not?
LAB 8.4
8.4.2
In this Exercise, you will copy the toolbar and its objects to the R_CONCANV.fmb form. But, before you do, you will add another button to the toolbar. Open both R_TOOLBAR.fmb and R_CONCANV.fmb in the Form Builder. a) What item on the R_CONCANV.fmb form could be better implemented as a toolbar item?
Minimize all of the objects for R_CONCANV.fmb for the time being. Create another button on the TOOLBAR canvas in R_TOOLBAR.fmb and position it to the right of Cancel Query. Name it SHOW_HIST. Adjust its size, navigable, and icon properties to match the other buttons on the TOOLBAR canvas. Set its Icon Filename property to srch_frw and set its Tool Tip to Record History. go to contents
303
This button will be used to open the HISTWIN window in the R_CONCANV.fmb form. b) What code should you put in the WHEN-BUTTON-PRESSED trigger for the SHOW-HIST button?
Add the code and compile the trigger. c) Which objects should you drag from R_TOOLBAR.fmb to R_CONCANV.fmb so that it too can have a functioning toolbar? Copy the objects; do not subclass them.
LAB 8.4
d) What should you do to make this toolbar available to the form? What else should you do to make it the only toolbar in the form?
Run the form and test all of the toolbar buttons. e) Was it convenient to have to drag these objects one at a time into R_CONCANV.fmb?
Save the changes to R_TOOLBAR.fmb. Do NOT save the changes to R_CONCANV.fmb. If you already have, simply delete all of the TOOLBAR objects from R_CONCANV.fmb and re-save it.
go to contents
304
LAB 8.4
A toolbar is very similar to the forms you created for data entry and querying. It requires items that must be positioned on a canvas and contained in a block. The main difference is that there are no data items on a toolbar, only non-data items. c) Do typical toolbars have text labels or iconic labels? Which property can you adjust to make your toolbar buttons typical? Answer: Typical toolbars have iconic labels. You should adjust the Iconic property. d) Should the user be able to navigate to these buttons with the mouse or keyboard? Which properties should you set to ensure that they cant? Answer: No, set both Keyboard Navigable and Mouse Navigable to No. In certain cases, you have seen that it is often necessary for a user to be able to navigate, at least with the keyboard, to a button. This is especially true if they are doing data entry and would like to use the ENTER key to initiate the Button Pressed event. However, it is never necessary to navigate to a button on a toolbar. e) What Layout Editor feature can you use to quickly position all of the buttons next to each other? Note that if you used the Object Navigator to create the buttons, they may be currently positioned on top of each other. This should not pose a problem for you. Answer: You can use Arrange | Align Objects, Stack Horizontally. Stacking the buttons will put them next to each other as shown in Figure 8.10. In many applications, you may have noticed that the toolbar buttons are spaced or separated into groups to make the toolbar more understandable.
go to contents
305
I FOR EXAMPLE:
You may wish to organize your toolbar so that the buttons are spaced by function. You could have two groups, one for the SAVE and EXIT buttons, and one for the QUERY buttons. The groups could be separated by blank space and even graphic objects as in Figure 8.11. The graphic object is a line with its Bevel property set to Inset. f) What property can you use to give each button a Tool Tip? Answer: You can use the Tool Tips property. While the button icons are usually quite descriptive, it is always helpful to include Tool Tips, especially for new users. Run the form. g) Which object is the toolbar attached to? Which of the TOOLBAR canvas properties indicate this? Answer: The TOOLBAR canvas is attached to MAINWIN. The canvas Window property indicates this. As you know, all canvases must be assigned to a window to be visible. The TOOLBAR canvas was attached by default to MAINWIN when you created it. Interestingly, all windows have a Horizontal Toolbar and Vertical Toolbar property. So, you could assign the toolbar canvas at the window level as well. There are no other canvases in this form yet, but if there were, how do you think TOOLBAR would be positioned relative to those canvases?
LAB 8.4
I FOR EXAMPLE:
Assume you have created a block based on the STUDENT table and have positioned all of its items on a content canvas called STUDENT. Both the STUDENT and TOOLBAR canvases are assigned to MAINWIN. How will the
306
LAB 8.4
i)
Answer: Yes, using the forms Horizontal Toolbar property. It will assign the TOOLBAR canvas to the MDI window. Assign TOOLBAR to the module and run the form again. Where is the toolbar now? Is the original default toolbar still there? Answer: It is attached to the MDI window. Yes, the original, default toolbar is still there. Multiple horizontal toolbars are extremely common. Applications like MS Word can have as many as six horizontal toolbars. Even the Layout Editor has two horizontal toolbars. In this case, the toolbars have functionality in common, so it will not be necessary to have both. j) What happened to the default toolbar? Answer: The default toolbar has disappeared. The TOOLBAR you have created is now the main toolbar for the application. The &SMARTBAR value in the Menu Module property assigns a menu toolbar, also known as a smart bar, to the default Forms menu. The default menu and smart bar are objects that Forms attaches to every form at runtime by default. As you can see, you can override these default settings and use your own toolbars. In Chapter 13, Forms Menus, you will learn how to override the defaults to create your own menus. Use the wizards to quickly create a block based on the ZIPCODE table. Stop when you get to the Layout Wizards canvas page.
go to contents
307
k) Should you position the ZIPCODE items on the TOOLBAR canvas? If not, what type of canvas should you create for them? Answer: No, you should not position them on the TOOLBAR canvas. You should create a new content canvas for them. l) What built-ins can you use to respond to the rest of the buttons? Create a WHEN-BUTTON-PRESSED trigger for each button and use DO_KEY statements with the built-ins. Answer: See the discussion below. Use the following statements for each button: 1) 2) 3) 4) 5) SAVE - DO_KEY('COMMIT_FORM'); EXIT - DO_KEY('EXIT_FORM'); ENTER_QUERY - DO_KEY('ENTER_QUERY'); EXECUTE_QUERY - DO_KEY('EXECUTE_QUERY'); CANCEL_QUERY - DO_KEY('EXIT_FORM');
LAB 8.4
Note that the CANCEL_QUERY button also uses the EXIT_FORM built-in because EXIT_FORM switches the form from Enter Query mode back to Normal mode. m) Does it make sense to have the SAVE button enabled during Enter Query mode? Why not? Answer: No, because you cannot commit records during ENTER_QUERY mode. In fact, there are two other buttons that should not be enabled in Enter Query mode. The EXIT_FORM button should not be enabled because you should cancel the query before being able to exit the form. The ENTER_ QUERY button should not be enabled either because you are in Enter Query mode already, which means there is no need to click the button again. By the same token, the CANCEL_QUERY button should be disabled when the form is not in Enter Query mode. Disabling and enabling buttons on a toolbar make an application much more user-friendly. The status of the buttons tells a user what she can and cant do at any given time.
308
LAB 8.4
8.4.2
ANSWERS
a) What item on the R_CONCANV.fmb form could be better implemented as a toolbar item? Answer: The SHOW_HIST button. Remember that eventually you will want both the TOOLBAR canvas and HISTORY canvas on all of your forms. Therefore, it makes sense to add the SHOW_HIST button to the toolbar. This button will be used to open the HISTWIN window in the R_CONCANV.fmb form. b) What code should you put in the WHEN-BUTTON-PRESSED trigger for the SHOW-HIST button? Answer: See the discussion below.
go to contents
309
You will use the GO_ITEM built-in and add it to the trigger as shown below: :global.last_item := :SYSTEM.CURSOR_ITEM; GO_ITEM('CONTROL.HIDE_HIST'); Add the code and compile the trigger. c) Which objects should you drag from R_TOOLBAR.fmb to R_CONCANV.fmb so that it too can have a functioning toolbar? Copy the objects; do not subclass them. Answer: See the discussion below. You should copy the following objects: 1) The TOOLBAR block. 2) The TOOLBAR canvas. These two objects work together to make the toolbar visible and functional. d) What should you do to make this toolbar available to the form? What else should you do to make it the only toolbar in the form? Answer: Set the modules Horizontal Toolbar property to TOOLBAR. Remove &SMARTBAR from the modules Menu Module property. Run the form and test all of the toolbar buttons. e) Was it convenient to have to drag these objects one at a time into R_CONCANV.fmb? Answer: Not really. You have created the toolbar to be a reusable object, but reusing it is a bit of a nuisance. In the next Chapter, you will create two more objects to include as part of the toolbar, so reusing it as you have here would become even more tedious. In the next Chapter, you will learn how to package the toolbar objects together so that you can move them as a single entity from form to form quickly and easily.
LAB 8.4
Save the changes to R_TOOLBAR.fmb. Do NOT save the changes to R_CONCANV.fmb. If you already have, simply delete all of the TOOLBAR objects and re-save it.
go to contents
310
LAB 8.4
go to contents
311
CHAPTER
Identify the built-ins to correspond to these buttons and add the code to the WHEN-BUTTON-PRESSED trigger. Save the form as R_TOOLBAR.fmb. 2) Implement the toolbar in R_TOOLBAR as a vertical toolbar. Save this form as R_TOOLBAR_V.fmb. 3) Use the wizards to create a form with tabbed canvases. This was not covered in the Lab sections of this Chapter, but you should be able to do it based on your knowledge of canvases. The form will be a master-detail-detail form based on the INSTRUCTOR, SECTION, and ENROLLMENT tables. The items from each block should appear on a different tab page. Arrange and align the items as you see fit. 4) Create a scrolling stacked canvas. First you must do some minor setup. Create a master-detail form based on the COURSE and SECTION tables. Include the audit columns in the blocks, but do not display them on the canvas. Add a display item called INSTRUCTOR_NAME to the SECTION block and populate it with a POSTQUERY trigger. IMPORTANT: First use the wizards to position all of the items on a content canvas, as you would if you were creating a simple master-detail form. Do not try to create a stacked canvas using the wizard. Once you have finished with the wizard and the items are positioned on the content canvas, use the Layout Editor to create a stacked canvas. Then, reassign some of the items in the SECTION block to this canvas. It should look like Figure 8.14. You will have to move the items to this canvas by adjusting their properties.
go to contents
312
Figure 8.14 I Stacked canvas with some of the items from the SECTION block.
Note that, while the items are on the canvas, not all of them are within view. This is very important to simulate the scrolling effect. Also note that there is one item from the SECTION block that is not on the stacked canvas. Which item is it? Which canvas do you think it is on? Use the Stacked Views feature to align and arrange the items. When the form is running, it should look like Figure 8.15.
go to contents
C H A P T E R
REUSABLE OBJECTS
CHAPTER OBJECTIVES
In this Chapter, you will learn about: Subclassing Visual Attributes and Property Classes Object Groups and Object Libraries Template Forms Page 314 Page 323 Page 336 Page 348
eusing objects makes development easier and faster. If you create an object in one form and duplicate it in others, you save the time and hassle of having to recreate the same object over and over again. If the duplicates are linked to the source object so that they will change as their source changes, you have saved yourself time and hassle again by being able to maintain and update all of the objects at once rather than one by one. Forms provides a number of facilities that make it easy and convenient to reuse objects. In this Chapter, you will learn about subclassing, which creates links between reusable objects and enables the inheritance of properties. There are a number of different ways to make use of subclassing. You can subclass entire objects and you can create groups of properties called property classes and subclass or apply them to objects. You can also store reusable objects in object groups, object libraries, and template forms to organize them and make access to them easier.
313
go to contents
LAB
9.1
SUBCLASSING
LAB OBJECTIVES
After this Lab, you will be able to:
Subclass Objects
To create one object based on another object, you can use either the Copy feature or Subclass feature. Copying an object creates an exact duplicate of an object that is independent of its source. The copied version of the object has no relationship or link to the source version. Each object can be edited or even deleted without affecting the other. Subclassing, on the other hand, allows you to create an exact duplicate of an object and maintain a link between the source version and the subclassed (duplicate) version. The subclassed version inherits all of the properties of the source object. If the source is ever changed or edited, the subclassed version will inherit the changes.
I FOR EXAMPLE:
In Lab 8.4, you created a non-base-table block called TOOLBAR that contained several button items. To reuse this TOOLBAR block, you could subclass it to other forms. By subclassing, exact duplicates of the TOOLBAR block will be created in the other forms. Since TOOLBAR contained several buttons, they will be duplicated as well. All of the properties for TOOLBAR and its buttons will be inherited by the subclassed versions of the objects. The link established between the source and the subclassed objects will ensure that if you add another button to the source TOOLBAR or change some of its properties, these changes will be propagated to the subclassed versions of the TOOLBAR. The subclassed TOOLBAR does not have to remain an exact duplicate of the source. Its properties can be changed and manipulated. You are free
go to contents
to change the properties that have not been inherited, and you can also override the values of properties that have been inherited. Despite the changes to the subclassed object, the link will remain with the source and it will continue to inherit changes. Any Forms object can be subclassed. In the previous example, by subclassing a block, you also subclassed its items and their triggers. By the same token, if you subclass a canvas, its frames and other graphic objects are subclassed as well. There are two simple steps for subclassing objects in Forms: 1) 2) Drag the source object from its form and drop it in the target form. Select Subclass when prompted by an alert.
In the Exercises, you will experiment with this and another method for subclassing objects. You will also explore how the Form Builder indicates which objects have been subclassed and which of their properties have been inherited.
go to contents
Lab 9.1: Subclassing b) How does the Object Navigator indicate that there are subclassed objects in R_TARGET.fmb?
d) How can you tell which of the subclassed objects properties are being inherited?
In the form EX09_01.fmb, change the PRE-FORM triggers Execution Hierarchy property to Before. e) Has this change been inherited by the subclassed PRE-FORM trigger in R_TARGET.fmb? How can this be useful?
Compile R_TARGET.fmb, then save it and close it. In EX09_01.fmb, change the MAINWINs Title property to "Changed while R_TARGET was closed." Save and close EX09_01.fmb. Re-open R_TARGET.fmb. f) Has the change to the Title property been propagated to the subclassed version of MAINWIN?
go to contents
g) If you were to use the Forms Runtime to run the previously compiled version of R_TARGET.fmx, would the change have been propagated there?
Delete MAINWIN and the PRE-FORM trigger from R_TARGET.fmb. You should still have a window object named WINDOW1 in R_TARGET.fmb. If you do not, create one now. Re-open EX09_01.fmb in the Form Builder. h) How can you make the WINDOW1 object in R_TARGET.fmb a subclassed version of MAINWIN in EX09_01.fmb without using the drag-and-drop technique?
i) Would you still have been able to do this if EX09_01.fmb had been closed?
go to contents
Figure 9.1 I The Subclass Information dialog for the MAINWIN object. go to contents
I FOR EXAMPLE:
In R_TARGET.fmb, note that MAINWINs Close Allowed property is being inherited. Change Close Allowed to Yes. Note that the black check mark is replaced with a black check mark with a red x over it, indicating that the inherited value has been overridden. Now click the Inherit button on the Property Palettes toolbar. What has happened? Here, the Inherit button re-establishes the link between the property and its inherited value, not its default value.
go to contents
go to contents
would use the Subclass Information property if you wanted to make an existing object a subclassed version of a source object. In this case, a new version of the object is not created. Instead, the existing object inherits all of the properties in the source object that have been changed from their defaults. If the same properties have previously been changed in an existing object, they will be overridden by the source objects properties. Also, note that the existing object keeps its original name after it has been subclassed. i) Would you still have been able to do this if EX09_01.fmb had been closed? Answer: No you would not have. The form that holds the source objects must be open when you perform the subclassing. This applies to the drag-and-drop method as well as using the Subclass Information dialog.
Save R_TARGET.fmb as you will need it in the Exercises for Lab 9.2.
go to contents
go to contents
323
LAB
9.2
LAB 9.2
Create and Apply Named Visual Attributes Create and Apply Property Classes
In Lab 9.1, you subclassed entire objects to promote inheritance. In this Lab, you will use visual attributes and property classes to promote the inheritance of a group of properties rather than an entire object.
VISUAL ATTRIBUTES
The visual attribute object contains all of the properties that determine font and color. These properties are useful in that they let you create and apply look and feel standards to multiple objects throughout your applications. A visual attribute called STD_TEXT_ITEMS might have its properties set as those in Figure 9.3. If you were to apply this visual attribute to an item, it would inherit these property values, making its Foreground Color black, Font Name Arial, and so on. Visual attributes can be applied to physical objects like items, frames, canvases, windows, LOVs, and alerts. They cannot be applied to logical objects like blocks or record groups since those objects have no color or font properties. Although the block object has a visual attribute group property, it is ignored by the Form Builder.
go to contents
324
LAB 9.2
PROPERTY CLASSES
Property classes are similar in principle to visual attributes, but they have a few more features that make them a bit more powerful. 1) 2) 3) Any property can be included in a property class, not only font and color properties. Property classes do not have to be in the same form as the objects they are applied to. Property classes can have triggers attached to them.
Visual attributes have a standard set of properties that cannot be changed. Property classes have no standard properties; you can add and delete whichever properties you like from a property class. This makes them more flexible in that you can include properties from the Functional, Database, Record, and other property categories. Since a property class can include any property, its use is not limited to physical objects, as is the case with visual attributes. A property class can be applied to any object, including logical objects like blocks, record groups, and triggers.
go to contents
325
LAB 9.2
I FOR EXAMPLE:
You could create a property class called STD_QUERY_BLOCK to set standards for blocks that are query-only. The property class might look like that in Figure 9.4.
go to contents
326
Look at the ZIPCODE.ZIP item in the Layout Editor to confirm that STD_TEXT_ITEM has been applied. c) How can you apply STD_TEXT_ITEM to all of the items in the ZIPCODE block? Do it in as few steps as possible.
LAB 9.2
Arrange the Form Builder so that both the Layout Editor and Property Palette are open, but do not overlap. View the properties for STD_TEXT_ITEM. d) What happens to the text items if you change the STD_TEXT_ ITEMs Foreground Color property to blue? to green? to black?
Before you go on, make sure that the Foreground Color of STD_TEXT_ITEM is set to black. Create a new visual attribute and name it STD_PROMPT. e) Which of STD_PROMPTs properties will let you define it specifically for prompts?
Set STD_PROMPTs Foreground Color to black, its Font Name to Arial, and its Font Weight to Bold. f) Can you set the Visual Attribute Group property for ZIPCODE.ZIP to STD_PROMPT? Which of ZIPCODE.ZIPs properties must you use to apply STD_PROMPT?
327
g) Can you reuse STD_TEXT_ITEM and STD_PROMPT with the R_COURSE.fmb form that you created in Exercise 5.1.1? How? Do not actually take these steps. Simply write your answer.
LAB 9.2
Close R_TARGET.fmb. You will not need it anymore.
9.2.2
In this Exercise, you will create a property class that will be used to format the audit items. This property class will help you reuse the record history objects that you created in the R_CONCANV.fmb form. Open the R_CONCANV.fmb form. Select the CREATED_BY item and view its properties. a) Take a moment to consider the characteristics of all of the audit items and their relation to the Record History window. Which of their properties would you want to apply to all audit items on subsequent forms so that they could be a part of the Record History window?
Select all of these properties in the Property Palette. Click the Property Class button on the Property Palettes toolbar. b) What object has been created? What are the properties of this new object?
328
Lab 9.2: Visual Attributes and Property Classes c) How can you apply the AUDIT_ITEMS property class to the CREATED_BY item? Can you apply AUDIT_ITEMS to all of the audit items simultaneously?
LAB 9.2
d) If you make a change or add another property to AUDIT_ITEMS, do you think it will be propagated to the audit items that are inheriting from the property class?
Save R_CONCANV.fmb and keep it open in the Form Builder. The next few questions will guide you through using the property class in another form. Quickly create a form based on the COURSE table. Include the audit columns in the block, but not on the canvas. Leave Enforce data integrity unchecked. Name the form R_TARGET2. R_TARGET2.fmb is a form you will use to practice subclassing. e) Can you apply AUDIT_ITEMS to the audit items in R_TARGET2? Do you have to create an instance of AUDIT_ITEMS in R_TARGET2 first?
f) If you make a change to AUDIT_ITEMS in the source form, R_CONCANV, do you think it will be propagated to the audit items in R_TARGET2?
Save R_CONCANV.fmb and close it. Save R_TARGET2.fmb and close it as you will need it in the Exercise in Lab 9.3.
In the following questions, you will create another property class that will help you reuse the toolbar. This property class will be based on the module object in the R_TOOLBAR.fmb form. go to contents
329
Open R_TOOLBAR.fmb. Select the R_TOOLBAR module and view its properties. g) Which of the module objects properties assign the toolbar to the module?
LAB 9.2
h) Which of the module objects properties indicate the menu module that should be used? How should the property be set so that the default smart bar is not displayed?
Create a property class in the Object Navigator and name it TOOLBAR_MODULE. i) Which of the Property Palettes toolbar buttons can you use to add properties to the class? Do not use the Copy button.
Use this function to add the properties from your answers to Questions g and h to the TOOLBAR_MODULE property class. Also include the Console Window property and set its value to MAINWIN. In Lab 9.3 you will use this property class and the other objects to reuse the toolbar in other forms. Save the R_TOOLBAR.fmb form.
go to contents
330
LAB 9.2
The behavior of visual attributes is somewhat similar to subclassing in that properties from one object are inherited by another. However, there is no subclassed relationship here. If you look at the Subclass Information property for ZIPCODE.ZIP, you will not see any reference to the visual attribute. The objects visual attribute properties are based on a named visual attribute. Or, put another way, the visual attribute is being applied to the object. c) How can you apply STD_TEXT_ITEM to all of the items in the ZIPCODE block? Do it in as few steps as possible. Answer: You can select all of the items in the block and set the Visual Attribute property for them all. This feature makes visual attributes a convenient way to standardize the look of a group of objects in a form. In this case, if the form had had two or three base-table blocks with multiple items in each, you could have formatted all of their visual attributes simultaneously. d) What happens to the text items if you change the STD_TEXT_ITEMs Foreground Color property to blue? to green? to black? Answer: The items foreground color changes. Since the visual attribute properties are being inherited by the items, any change to the visual attribute is immediately propagated to those items. Before you go on, make sure that the Foreground Color of STD_TEXT_ ITEM is set to black. Create a new visual attribute and name it STD_PROMPT. e) Which of STD_PROMPTs properties will let you define it specifically for prompts? Answer: The Visual Attribute Type property. There are three values for the Visual Attribute Type property: 1) Common, which for text and display items applies to the item itself and its prompt. Common visual attributes can also be used for other interface objects like canvases, windows, and LOVs.
go to contents
331
3) f)
Prompt, which applies only to the prompt properties of an item. A visual attribute of type Prompt will override any properties set by a visual attribute of type Common. Title, which applies to frame titles.
Can you set the Visual Attribute Group property for ZIPCODE.ZIP to STD_PROMPT? Which of ZIPCODE.ZIPs properties must you use to apply STD_PROMPT? Answer: No you cannot. You must use the Prompt Visual Attribute Group property.
LAB 9.2
g) Can you reuse STD_TEXT_ITEM and STD_PROMPT with the R_COURSE.fmb form that you created in Exercise 5.1.1? How? Do not actually take these steps. Simply write your answer. Answer: Yes you can. You must copy or subclass it to the form. An instance of a visual attribute must be in a form for it to be available to the objects in that form. It can either be copied or subclassed to the form. So, in this case, if STD_TEXT_ITEM and STD_PROMPT were subclassed to the R_COURSE.fmb form, they could be applied to its items. Also, both visual attributes would behave like any other subclassed object. If any changes were made to the source instances of STD_TEXT_ITEM and STD_PROMPT, they would be propagated to the subclassed instances of the visual attributes in R_COURSE.fmb. These changes would then be propagated to all of the items in R_COURSE.fmb that the visual attributes were applied to. This feature gives you the ability to easily enforce look and feel standards across your application.
9.2.2 ANSWERS
a) Take a moment to consider the characteristics of the audit items and their relation to the Record History window. Which of their properties would you want to apply to all audit items on subsequent forms so that they could be a part of the Record History window? Answer: See the discussion below. The purpose of this property class is to size, position, and configure the audit items for the HISTORY canvas so that they can be a part of the Record History window. This way, each time you create a form and wish to include the audit items in the Record History window, you will simply have to apply this property class. The properties and values you should include are as follows:
go to contents
332
Item Type
1) 2) 3) 4) 5) 6) Canvas X Position Width Height Background Color Foreground Color
Display Item
HISTORY 93 75 14 gray black
LAB 9.2
You may want to adjust the value of X Position to suit the size of your HISTORY canvas. Unfortunately, since all of the items will have different Y Position values, you cannot add it to this property class. One workaround would be to create four property classes with four different Y Position settings. b) What object has been created? What are the properties of this new object? Answer: A new property class with a default name has been created. It contains only the properties that were selected. To create a property class with multiple properties, this is the easiest way to do it. Not only were the properties added to the class, but their values were copied as well. You could also create the property class in the Object Navigator and then copy and paste the properties from the source item into the property class. c) How can you apply the AUDIT_ITEMS property class to the CREATED_BY item? Can you apply AUDIT_ITEMS to all of the audit items simultaneously? Answer: You use the Subclass Information dialog. No, you cannot apply AUDIT_ITEMS to more than one item at a time. In Lab 9.1, you used the Subclass Information dialog to subclass objects. Here you will use it to subclass property classes. In Figure 9.5 you can see that it looks only slightly different from the dialog you saw earlier. Note that the Property Class radio button at the top of the dialog has been selected. As you did previously, you simply select the module name and property class to subclass. Unfortunately, unlike visual attributes, you cannot apply a property class to more than one object at a time.
go to contents
333
LAB 9.2
Figure 9.5 I The Subclass Information dialog for property classes.
d) If you make a change or add another property to AUDIT_ITEMS, do you think it will be propagated to the audit items that are inheriting from the property class? Answer: Yes. Again, the basic rules of subclassing apply here. Any change made to the source AUDIT_ITEMS property class will be propagated to subclassed versions. e) Can you apply AUDIT_ITEMS to the audit items in R_TARGET2? Do you have to create an instance of AUDIT_ITEMS in R_TARGET2 first? Answer: Yes you can apply it. No, you do not have to create an instance of AUDIT_ITEMS in R_TARGET2. Property classes, unlike visual attributes, have a subclassed relationship with the object to which they are applied. If you apply a property class to an object, you will see a reference to it in the Subclass Information dialog. The object will also have a red arrow in front of it in the Object Navigator to further indicate that there is a subclassed relationship. Because of this, the property class does not have to be in the same form as the object it is applied to. It can be subclassed from other modules just like any other object by using the Subclass Information dialog. This is different from the behavior of visual attributes, which have to be in the same form as the objects they are applied to. f) If you make a change to AUDIT_ITEMS in the source form, R_CONCANV, do you think it will be propagated to the audit items in R_TARGET2? Answer: Yes. Basic subclassing rules apply here too.
go to contents
334
LAB 9.2
go to contents
335
5) Which of the following is not true about property classes? a) _____ They can be applied to any object b) _____ They can contain visual as well as other properties c) _____ They cannot be applied to more than one item in the same form d) _____ Properties can be added or deleted from them 6) Which of the following cannot be done to properties in a property class? a) _____ Override them at the subclassed object level b) _____ Change them after the property class has been applied c) _____ Copy or paste them from another object d) _____ None of the above 7) How can you apply a property class to an object? a) _____ By dragging and dropping it on top of the object b) _____ By setting the property class Target Objects property c) _____ By setting the target objects Subclass Information property d) _____ By making it available to the target form 8) Which takes precedence over the properties inherited from a property class? a) _____ Properties overridden at the object level b) _____ Form-level properties c) _____ Properties inherited from a visual attribute d) _____ a & c Quiz answers appear in Appendix A, Section 9.2.
LAB 9.2
go to contents
336
LAB
9.3
LAB 9.3
Create and Reuse Object Groups Create and Utilize Object Libraries
Object groups and object libraries serve as containers to make reusing objects, either through subclassing or copying, much easier. An object group is an object within a forms module, while an object library is a module unto itself.
OBJECT GROUPS
In previous Labs, you created a group of objects that comprised a toolbar and a group of objects that comprised the Record History window. To reuse these groups of objects in another form, you had to drag each object one by one from the source form to the target form. An object group is a single object that allows you to store references to the two objects that comprise the toolbar. When you want to reuse the toolbar in another form, you simply drag the object group from the source form to the target and select Copy or Subclass. All of the objects are brought over to the target form in one simple step. This will become an extremely convenient feature when you begin to create reusable objects that are made up of five or more individual objects. Object groups are logical containers that are only visible in the Object Navigator. The object group for the toolbar might look something like that in Figure 9.6
go to contents
337
OBJECT LIBRARIES
As you create more reusable objects, you will want to store them in a central place so that you can access them easily. In the previous Labs, all of your objects were spread out across three or four forms. If you wanted to reuse a number of these objects, you had to open each form and then drag and drop each of the objects. An object library is a module, stored in a separate file, that can be opened and configured in the Form Builder. Library modules have an .olb extension. You can open object library modules in the LIBRARY window, view their objects, and copy or subclass their objects into forms modules. A typical object library would resemble that in Figure 9.7. The object library can hold any type of object, including triggers, blocks, items, and even object groups. The tabs are user-defined and serve to make the library more organized. You can have as many or as few tabs as youd like and name them as you please.
LAB 9.3
Figure 9.7 I A typical object library module in the LIBRARY window. go to contents
338
LAB 9.3
Open R_TARGET2.fmb. Start dragging the AUDIT_POPULATION object group from R_TRANS.fmb into R_TARGET2.fmb. b) Onto which two nodes in R_TARGET2.fmb can you drag and drop AUDIT_POPULATION? Make sure you subclass the object group.
c) Where were the trigger objects positioned? How will this be helpful when you create an object group for all of the objects that make up the Record History window?
go to contents
339
d) Do the subclassed objects list the object group as their source? If not, what do they list and why?
e) If you were to change either the PRE-INSERT or PRE-UPDATE trigger in the R_TRANS.fmb form, do you think the changes would be propagated to R_TARGET2.fmb?
LAB 9.3
Close R_TARGET2.fmb without saving. You will not need it anymore.
Collapse all the nodes in the R_TRANS.fmb form so that the Object Navigator is easier to read. Open R_TOOLBAR.fmb. Create an object group called TOOLBAR. f) Which objects should you drag into the TOOLBAR object group if you want it to contain all of the objects that make up the toolbar?
Drag the objects you listed for Question f into the TOOLBAR object group. Collapse all the nodes in the R_TOOLBAR.fmb form so that the Object Navigator is easier to read. Open R_CONCANV.fmb. Delete the SHOW_HIST button from the CONTROL block as you will not need it anymore. Create an object group called RECORD_HISTORY. g) Which objects should you drag into the RECORD_HISTORY object group if you want it to contain all of the objects that make up the Record History window?
go to contents
340
Drag the objects you listed for Question g into the RECORD_HISTORY object group. Save R_TRANS.fmb, R_TOOLBAR.fmb, and R_CONCANV.fmb to preserve the addition of the object groups.
LAB 9.3
Create a new form in the Object Navigator. Subclass the AUDIT_POPULATION, TOOLBAR, and RECORD_HISTORY object groups into the new form. h) Which property class should you apply to this new form to ensure that the toolbar is positioned on the MDI window?
i) How should the windows be ordered in the Object Navigator? Which window can you delete?
Quickly create a block based on the ZIPCODE table. Enforce data integrity should be unchecked. Include the audit columns in the block, but not on the canvas. Make sure you create a new canvas in the Layout Wizard. j) Which property class should you apply to the audit items?
Open the HISTORY canvas. Arrange the audit items so that they are all visible and aligned. Run the form and test that the toolbar works, the Record History window opens, and that the audit columns are populated when you insert and update records. Close this new form without saving. You will not need it anymore.
go to contents
341
9.3.2
Open R_TOOLBAR.fmb, R_TRANS.fmb, and R_CONCANV.fmb in the Form Builder. Collapse all of their nodes to make the Object Navigator easier to read. Locate the Object Library node in the Object Navigator. Create an object library and name it LIBRARY. Double-click the icon next to LIBRARY to open it. a) How many default library tabs were created for LIBRARY? How can you rename and label them? Use the following name and label combinations: OBJGROUP - Object Groups and SINGLEOBJ Single Objects.
LAB 9.3
b) How can you add the AUDIT_ITEMS property class from the R_CONCANV.fmb form to the Single Objects tab of LIBRARY?
c) Were you asked to subclass or copy AUDIT_ITEMS into LIBRARY? Why not?
Collapse all of the nodes for R_CONCANV.fmb. Save LIBRARY.olb since you cannot reuse any of its objects until it has been saved. Create a new form, but do not create any blocks or objects. d) How can you reuse the AUDIT_ITEMS property class that is stored in the LIBRARY in this new form? Choose Subclass when reusing.
go to contents
342
Lab 9.3: Object Groups and Object Libraries e) In the new form, what is the value of the Subclass Information property for AUDIT_ITEMS? Is there any reference to R_CONCANV.fmb?
LAB 9.3
Drag AUDIT_ITEMS into the new form again so that you can edit it. Choose Copy instead of Subclass. Change AUDIT_ITEMSs Comment property to "This object has been changed." Drag AUDIT_ITEMS back to LIBRARY. g) How should you respond to the alert you are presented with if you want LIBRARY to have the new changes?
h) If AUDIT_ITEMS were previously subclassed to other forms, would the changes to the Comment property be inherited by those forms?
Drag the AUDIT_POPULATION, TOOLBAR, and RECORD_HISTORY object groups into the Object Group tab of LIBRARY and save it. Close the forms R_TRANS.fmb, R_TOOLBAR.fmb, and R_CONCANV.fmb.
go to contents
343
i) If you were to subclass these object groups from LIBRARY to other forms, do you think the subclassing behavior would be the same for them as it was for the AUDIT_ITEMS property class?
LAB 9.3
go to contents
344
LAB 9.3
g) Which objects should you drag into the RECORD_HISTORY object group if you want it to contain all of the objects that make up the Record History window? Answer: See the list below. 1) 2) 3) 4) CONTROL block. HISTORY canvas. AUDIT_ITEMS property class. HISTWIN window.
The CONTROL block has also brought the HIDE_HIST button and its trigger into the object group. By the same token, if the HISTORY canvas had had any frames or other graphic objects, those would have been included as well. h) Which property class should you apply to this new form to ensure that the toolbar is positioned on the MDI window? Answer: The TOOLBAR_MODULE property class. i) How should the windows be ordered in the Object Navigator? Which window can you delete? Answer: The Layout Wizard will assign new canvases to the window that is listed first in the Object Navigator. Therefore, MAINWIN should be listed first. You can delete WINDOW1.
go to contents
345
Since MAINWIN is going to be the window for the canvas that holds the data items, you want it to be listed first in the Object Navigator. This way, the canvas for the data items will not be assigned to HISTWIN. j) Which property class should you apply to the audit items? Answer: You should apply the AUDIT_ITEMS property class.
9.3.2
a) How many default library tabs were created for LIBRARY? How can you rename and label them? Use the following name and label combinations: OBJGROUP - Object Groups and SINGLEOBJ - Single Objects. Answer: Two tabs were created by default. Library tabs have properties that you can use to set the name and label. You can add as many library tabs to an object library as youd like. In this Exercise, you will put Object Groups on one tab and Single Objects on another. You could organize the library differently with tabs for Items, Blocks, Physical Objects, Logical Objects, or whatever. b) How can you add the AUDIT_ITEMS property class from the R_CONCANV.fmb form to the Single Objects tab of LIBRARY? Answer: You can drag it into the object library. c) Were you asked to subclass or copy AUDIT_ITEMS into LIBRARY? Why not? Answer: No you were not. Object libraries hold actual instances of objects, not just pointers to them. Also, since the object library is supposed to serve as the source for all of your objects, you would not want it to contain subclassed versions of objects. If you choose to use object libraries, they should be the source that all of your subclassed objects come from. d) How can you reuse the AUDIT_ITEMS property class that is stored in the LIBRARY in this new form? Choose Subclass when reusing. Answer: You can drag it from the object library into the form. e) In the new form, what is the value of the Subclass Information property for AUDIT_ITEMS? Is there any reference to R_CONCANV.fmb? Answer: The value is LIBRARY. AUDIT_ITEMS, which has been subclassed from the object library, not the R_CONCANV.fmb form.
go to contents
346
LAB 9.3
go to contents
347
3) Why are object groups useful? a) _____ You can package related objects together b) _____ You can copy or subclass a group of related objects to another form in a single step c) _____ If subclassed, you can make changes to the objects in the object group that are propagated to target objects d) _____ All of the above 4) How are objects stored in an object library? a) _____ As pointers to objects in other modules b) _____ As actual instances of objects c) _____ As subclassed instances d) _____ b & c 5) How can you use an object library? a) _____ To organize the objects you wish to subclass or copy to other forms b) _____ As a central source for all of your Forms objects c) _____ As any area to edit and update reusable objects d) _____ a & b 6) What can you do with objects in an object library? a) _____ Copy them to other forms b) _____ Subclass them to other forms c) _____ Organize them any way youd like d) _____ All of the above Quiz answers appear in Appendix A, Section 9.3.
LAB 9.3
go to contents
348
LAB
9.4
TEMPLATE FORMS
LAB OBJECTIVES
LAB 9.4 After this Lab, you will be able to: Create and Use Template Forms
In the previous Labs in this Chapter, you learned how to reuse objects through subclassing, object groups, and object libraries. In each case, you were referencing an object in one form or library module and subclassing or copying it to another. This allowed you to pick and choose which source objects to include in target forms that either already existed or that were just being created. In each case, you had to drag and drop or otherwise manually reference the objects into the target forms. Sometimes you will want to create entirely new forms that already contain a standard set of objects. When this is true, then you can create a template form. The template form would already contain all of these objects and would be what you would use as the starting point for your new forms. Using a template form will save you the time of having to manually add the objects to each new form you create.
I FOR EXAMPLE:
In the Exercises, you will create a template form to contain all of the objects for the audit population, the toolbar, and the Record History window. When you create forms based on this template, these objects will automatically be added to the new form. Template forms are no different than any other form. They are created, configured, and saved in the Form Builder in same manner as regular .fmb files. When you want to create a new form based on a template form, you select File | New | Form Using Template from the Main Menu. A new form will open with all of the objects that are included in the template.
go to contents
349
LAB 9.4
b) How can you use the Main Menu to create a new form based on the R_TEMPLATE.fmb template form?
c) What has the Form Builder named the new form? Why has it done this?
d) Are all of the objects from the template form in your new form?
Close the new form without saving as you will not need it anymore.
go to contents
350
LAB 9.4
go to contents
351
LAB 9.4
go to contents
352
CHAPTER
go to contents
C H A P T E R
1 0
REUSABLE CODE
CHAPTER OBJECTIVES
In this Chapter, you will learn about: Program Units PL/SQL Libraries Stored PL/SQL Objects Page 354 Page 362 Page 374
o far, all of the PL/SQL logic you have written has been stored in triggers. This has been convenient and powerful because you were able to write custom code to respond to events. One of the limitations of triggers is that it is difficult to reuse them. Of course, you can subclass or copy them, but then you are creating multiple instances of the same code. It is far better to remove the code from a single trigger and store it somewhere where multiple triggers can access it.
In forms modules, this is done by creating program units. In Lab 10.1, you will create a single program unit that all of the WHEN-VALIDATEITEM triggers in a form can reuse. In Lab 10.2, you will store that same program unit in a PL/SQL library module so that triggers in other forms modules can reuse it. You will also learn how to write generic code so that a program unit can be useful in more than one situation.
353
go to contents
LAB
10.1
PROGRAM UNITS
LAB OBJECTIVES
After this Lab, you will be able to:
Program units are Forms objects that contain PL/SQL code. They can be packages, procedures, or functions, and can accept parameters and return values. They are identical in structure and syntax to the packages, procedures, and functions stored in the database, except they are stored, compiled, and run within a Forms application. Program units make the PL/SQL code within forms modules more reusable because they release the code from being bound to a single trigger and the events that the trigger responds to. Once code has been placed in a program unit, it is accessible from any trigger or other program unit in the form.
I FOR EXAMPLE:
Assume you create a button called BUTTON_1 that displays a canvas called CANVAS_1 when pressed. The code in the WHEN-BUTTON-PRESSED trigger could be as follows: SHOW_VIEW('CANVAS_1'); While this will work quite well, the limitation is that the canvas will only be displayed in response to the Button Pressed event for BUTTON_1. Now, assume that you also want to create POST-QUERY and POST-INSERT triggers to display the canvas each time the user issues a query and each time they insert a record. Since you cannot call the code in the WHEN-BUTTONPRESSED trigger, you would have to re-write it in both the POST-QUERY
go to contents
and POST-INSERT triggers. You would then have three triggers with identical code. This would be difficult to maintain and hard to manage, especially if the code were more complicated than a single SHOW_VIEW statement. The alternative is to create a program unit and call it from each of the three triggers. The program unit could be named DISPLAY_CANVAS and it could contain the following code: PROCEDURE DISPLAY_CANVAS IS BEGIN SHOW_VIEW('CANVAS_1'); END; Each trigger would then contain the following statement to call DISPLAY_CANVAS: DISPLAY_CANVAS; The code is now in a central location and accessible to all three triggers in the form. It will also be accessible to subsequent triggers and program units created in the form. Program units also make PL/SQL code more reusable because they can accept parameters. By doing so, the code in the program unit can be made more generic and therefore reusable.
I FOR EXAMPLE:
Assume that your application has more than just CANVAS_1; assume that there is a CANVAS_2, CANVAS_3, and CANVAS_4. Instead of writing individual program units to display each canvas, you could make the code in the DISPLAY_CANVAS procedure more generic. The code could be as follows: PROCEDURE DISPLAY_CANVAS (p_canvas_name VARCHAR2) IS BEGIN SHOW_VIEW(p_canvas_name); END; The procedure DISPLAY_CANVAS now accepts a VARCHAR2 parameter and passes it to the SHOW_VIEW built-in. It is more generic because the canvas name is no longer hard-coded into the program unit. So, it can be used to show any canvas.
go to contents
Figure 10.2 I The PL/SQL Editor with the specification section for a program unit. go to contents
Open the form EX10_01.fmb in the Form Builder. In this Exercise, you will create a simple program unit that changes the message text in an alert. You will show this alert whenever a WHEN-VALIDATEITEM trigger fails to validate a foreign-key item. Note that there is an alert called VAL_ALERT already created in EX10_01 .fmb. It has one button labeled OK. Create a program unit and name it GET_ALERT_TEXT. The program unit will accept a VARCHAR2 parameter called p_item. p_item will be the name or description of the item being validated. a) How should you change the statements in the PL/SQL Editor so that this procedure will accept the p_item parameter?
b) How can you declare three variables? They should be: v_ alert_id of type ALERT, v_message of type VARCHAR2(200), and v_alert_response of type NUMBER.
go to contents
Lab 10.1: Program Units c) How is the p_item parameter being used in this procedure?
Open the WHEN-VALIDATE-ITEM trigger for STUDENT.ZIP. d) How can you change this trigger so that the user sees the alert when validation fails? What value should you pass to the procedure?
Run the form and issue a query. Test the program unit and alert by entering an invalid Zip Code (e.g., 123). Exit the form and return to the Form Builder. e) If you were to reuse this program unit in another form, which other object would you have to copy or subclass with it?
f) If GET_ALERT_TEXT had been part of a package called GENERAL, how do you think you would have called it from the WHENVALIDATE-ITEM trigger?
go to contents
a) How should you change the statements in the PL/SQL Editor so that this procedure will accept the p_item parameter? Answer: See the discussion below. The code so far should be: PROCEDURE GET_ALERT_TEXT (p_item VARCHAR2) IS BEGIN The syntax for creating a program unit is identical to the syntax for creating a stored procedure in the database. The Form Builder automatically writes the specification section in the PL/SQL Editor. In this case, the procedure accepts one parameter called p_item. b) How can you declare three variables? They should be: v_alert_id of type ALERT, v_message of type VARCHAR2(200), and v_alert_response of type NUMBER. Answer: See the discussion below. The code so far should be: PROCEDURE GET_ALERT_TEXT (p_item VARCHAR2) IS v_alert_id ALERT; v_message VARCHAR2(200); v_alert_response NUMBER; BEGIN c) How is the p_item parameter being used in this procedure? Answer: It is being passed to a character string variable, which is then referenced in the SET_ALERT_PROPERTY built-in. p_item will be the description of the item that is being validated by the WHEN-VALIDATE-ITEM trigger. The message will tell the user that the value that has been entered is invalid. In this case, the value will be ZIP or ZIPCODE. p_item is concatenated to a string that will be the message text of the alert. This message text will then be applied to the alert using the SET_ALERT_PROPERTY built-in. d) How can you change this trigger so that the user sees the alert when validation fails? What value should you pass to the procedure? Answer: See the discussion below. You should remove the line: Message('This zipcode does not exist in the database');
go to contents
go to contents
2) Why are program units reusable? a) ____ They are not bound to a single trigger and the events the trigger responds to b) ____ They can be written generically so that they are more widely applicable c) ____ They can be subclassed d) ____ All of the above 3) Which of the following can be a program unit? a) ____ Package specification b) ____ Trigger c) ____ Event d) ____ None of the above 4) Where can program units be called from? a) ____ Properties b) ____ Triggers c) ____ Other program units d) ____ b & c Quiz answers appear in Appendix A, Section 10.1.
go to contents
362
LAB
LAB 10.2
10.2
PL/SQL LIBRARIES
LAB OBJECTIVES
After this Lab, you will be able to:
Create and Attach PL/SQL Libraries Use Indirect References in Library Code
Some program units will be usable in more than one form, in which case, you have a few options for making them accessible to those forms. You could subclass or copy the program units to each form as you learned in Chapter 9, Reusable Objects, or you could store the program units in a PL/SQL Library. PL/SQL Libraries should be used to store program units when these program units will be used by most, or all, of the forms in an application. PL/SQL Libraries do not belong to forms modules. They are separate modules that can be created, edited, and compiled in the Form Builder. They can contain any number of program units, be they packages, procedures, or functions. Figure 10.3 shows a PL/SQL Library called COMMON in the Object Navigator. Note that it contains the DISPLAY_CANVAS program unit. Before a form can execute the objects in a PL/SQL Library, the Library must be attached to the form. This can be done using the Object Navigator. Figure 10.4 shows the COMMON Library attached to a form. PL/SQL Library modules are saved in two file formats: .pll files, which contain the source code and executable code of the library and .plx files, which contain only the executable code. Once a Library is saved as a .pll file, you can use the Form Builder to attach it to as many forms as necessary. Continuing with the canvas example, if every form in your application needed to show canvases, then it would make sense to attach COMMON.pll to all of those forms.
go to contents
363
LAB 10.2
Figure 10.3 I A PL/SQL Library module called COMMON in the Object Navigator.
In addition to reusability, another advantage of PL/SQL Libraries is that they help make your application more lightweight and efficient. When you attach a PL/SQL Library to a form, the Library and its code are not stored within the form; the form simply knows that the Library exists and therefore can reference its subprograms. The forms module becomes significantly smaller because it does not have to store the PL/SQL code itself. In addition, if the same Library is attached to ten forms in your application, only one version of the Library file has to be deployed to the users. This will make the overall size of the application smaller.
364
LAB 10.2
v_student_id := :STUDENT.STUDENT_ID and v_item := :SYSTEM.CURSOR_ITEM; These are known as direct references because you are directly using the name of a bind variable in your code. In the two statements listed above, you are assigning values to variables by making direct references to an item name and a system variable. You are able to successfully compile forms modules that contain statements like these because the compiler is able to locate an item called :STUDENT.STUDENT_ID in the form. However, you would not be able to successfully compile a PL/SQL Library if it contained direct references to bind variables. The Library module would not contain the item called :STUDENT.STUDENT_ID, so it would not be able to locate it and therefore would compile the Library with errors. To make reference to bind variables in Library code, you have to do so indirectly using the COPY and NAME_IN built-ins. In Chapter 6, Triggers & Built-ins, you indirectly referenced the names of the audit items with statements that were similar to the following: COPY(v_date, 'COURSE.CREATED_DATE'); v_date is being copied into the COURSE.CREATED_DATE item, but the item name is being referenced indirectly. The NAME_IN built-in allows you to indirectly reference bind variables and return their values. To indirectly reference the :SYSTEM.CURSOR_ITEM, you would write the following statement: v_item := NAME_IN(':SYSTEM.CURSOR_ITEM');
Open the form P_PUNIT.fmb. In this Exercise, you will put the GET_ ALERT_TEXT program unit into a PL/SQL Library and then attach the Library to the R_PUNIT.fmb form.
go to contents
365
Create a PL/SQL Library using the Object Navigator. Save the Library and name it COMMON.pll. a) How do you think you can add GET_ALERT_TEXT to COMMON .pll?
LAB 10.2
b) What are two ways you can compile COMMON.pll from within the Form Builder?
Compile, save, and close COMMON.pll. Delete GET_ALERT_TEXT from the R_PUNIT.fmb form. c) How can you use the Object Navigator to attach the COMMON.pll library to R_PUNIT.fmb? What dialog has opened?
Browse the filesystem to locate COMMON.pll. Click Attach when COMMON.pll is in the Library field. d) How should you respond to the alert you are presented with? Should you remove the path or keep the path? How will Forms find the .pll if you remove the path?
e) Can you open and edit GET_ALERT_TEXT through the Attached Library node in the Object Navigator?
go to contents
366
Run the form, issue a query, and enter an invalid Zip Code to test that the code in the attached Library can show the alert. Save the changes to form R_PUNIT.fmb.
Open COMMON.pll. Create a program unit and name it INSERT_AUDIT_ ITEMS. It will not have to accept any parameters.
go to contents
367
d) What will the code be for INSERT_AUDIT_ITEMS? Hint: It will be similar to the code for the current PRE-INSERT trigger. How will the code for the PRE-INSERT trigger have to change to call the INSERT_AUDIT_ITEMS program unit?
LAB 10.2
Create another program unit called UPDATE_AUDIT_ITEMS. e) What should the code for UPDATE_AUDIT_ITEMS be? How should you change the PRE-UPDATE trigger?
Save COMMON.pll. Also, make sure that you have changed the code in the PRE-INSERT and PRE-UPDATE triggers so that they call the program units. f) Have the two new procedures been added to the version of COMMON.pll under the Attached Libraries node in the Object Navigator?
g) What should you do with the AUDIT_POPULATION object group so that its version in the object library is current?
h) Can you add COMMON.pll to the object library? How about to your template form?
go to contents
368
All will open the Compile dialog shown in Figure 10.5. The Compile dialog gives you more control over the compilation than by pressing CTRL-T. It will allow you to interrupt and resume compilation using the buttons at the bottom of the window. Also, if there are any compilation errors, you can navigate directly to the offending PL/SQL object by clicking on the error.
go to contents
369
LAB 10.2
Figure 10.5 I The Compile dialog showing compilation errors.
It is important to note that these compilation options apply to forms modules as well as PL/SQL Library modules. Compile, save, and close COMMON.pll. Delete GET_ALERT_TEXT from the R_PUNIT.fmb form. c) How can you use the Object Navigator to attach the COMMON.pll library to R_PUNIT.fmb? What dialog has opened? Answer: Select the Attached Libraries node and click Create. The Attach dialog opens. The Attach dialog lets you search the filesystem, the database, or both for the PL/SQL Library. Unfortunately you cannot drag a PL/SQL Library in the Object Navigator and drop it on a form. d) How should you respond to the alert you are presented with? Should you remove the path or keep the path? How will Forms find the .pll if you remove the path? Answer: You should remove the path. Forms will find the .pll using the FORMSXX_PATH Registry entry. As you learned in Chapter 9, Reusable Objects, whenever Forms has to reference objects or code stored in other modules, it uses the Windows Registry to locate the modules. The Form Builder and Forms Runtime will look to the Registry for pointers to the files instead of using a hard-coded path to find them. This saves you from having to have identical directory structures on every machine that will run or edit this form. e) Can you open and edit GET_ALERT_TEXT through the Attached Library node in the Object Navigator? Answer: No you cannot.
go to contents
370
LAB 10.2
10.2.2
ANSWERS
Answer: Yes, they will be valid.
a) Will the COPY statements be valid in the PL/SQL Library? Why or why not?
The COPY built-in makes indirect references to Forms items and variables. Therefore, it can be used in procedures and functions that are in an attached library. b) Which statement will not be valid in the PL/SQL Library? Answer: v_block := :SYSTEM.CURSOR_BLOCK; will not be valid. Since this statement makes a direct reference to a system variable, it cannot be used in the PL/SQL Library. c) How can you change it so its value is referenced indirectly? Answer: Use the NAME_IN built-in. The NAME_IN built-in will allow you to reference the system variable indirectly and then assign its value to a local variable. The statement will look like this: v_block := NAME_IN(':SYSTEM.CURSOR_BLOCK'); Open COMMON.pll. Create a program unit and name it INSERT_ AUDIT_ITEMS. It will not have to accept any parameters. d) What will the code be for INSERT_AUDIT_ITEMS? Hint: It will be similar to the code for the current PRE-INSERT trigger. How will the code for the PRE-INSERT trigger have to change to call the INSERT_AUDIT_ITEMS program unit? Answer: See the discussion below. The code for INSERT_AUDIT_ITEMS will be fairly similar to the code for the old version of the PRE-INSERT trigger. The only differences will be in
go to contents
371
the specification section and in how :SYSTEM.CURSOR_BLOCK gets referenced. PROCEDURE insert_audit_items IS v_block VARCHAR2(30); v_username VARCHAR2(30); v_date DATE; BEGIN v_username := GET_APPLICATION_PROPERTY(USERNAME); v_date := SYSDATE; v_block := NAME_IN(':SYSTEM.CURSOR_BLOCK'); COPY(v_date, v_block||'.CREATED_DATE'); COPY(v_username, v_block||'.CREATED_BY'); COPY(v_date, v_block||'.MODIFIED_DATE'); COPY(v_username, v_block||'.MODIFIED_BY'); END; The code for the PRE-INSERT trigger will be reduced to the following statement: INSERT_AUDIT_ITEMS; Create another program unit called UPDATE_AUDIT_ITEMS. e) What should the code for UPDATE_AUDIT_ITEMS be? How should you change the PRE-UPDATE trigger? Answer: See the discussion below. PROCEDURE update_audit_items IS v_block VARCHAR2(30); v_username VARCHAR2(30); v_date DATE; BEGIN v_username := GET_APPLICATION_PROPERTY(USERNAME); v_date := SYSDATE; v_block := NAME_IN(':SYSTEM.CURSOR_BLOCK'); COPY(v_date, v_block||'.MODIFIED_DATE'); COPY(v_username, v_block||'.MODIFIED_BY'); END;
LAB 10.2
go to contents
372
LAB 10.2
go to contents
373
LAB 10.2
go to contents
374
LAB
10.3
LAB 10.3
Forms modules can call PL/SQL objects that are stored in the database just as they do PL/SQL objects stored in attached libraries. Databasestored packages, procedures, and functions can be called from a Forms application, and they can also be passed parameters. The syntax for calling a stored PL/SQL object is no different than if you were calling a program unit in a form.
I FOR EXAMPLE:
If you were to call a stored PL/SQL procedure called DELETE_STUDENT and pass it a value in a variable, the code in the Forms application would be as follows: DELETE_STUDENT(v_student_id); Forms will look for a PL/SQL object named DELETE_STUDENT and execute the first instance that it finds. First, Forms will search the forms module for a program unit named DELETE_STUDENT. If it doesnt find one, it will search any attached libraries. If it doesnt find one there either, it will then search the database. Tremendous performance advantages can be gained from storing some of your application logic in the database. If, for example, the PL/SQL objects you have created are data-intensive, then it may be better to store the ob-
go to contents
375
jects in the database. The database will execute the code more quickly and there will be less traffic back and forth across the network. Also, the code will be stored centrally in the database, which will make it accessible to other Forms and non-Forms applications. Many PL/SQL objects can be written to successfully compile and run in both a Forms application and the database. As long as the PL/SQL object does not reference any Forms-specific PL/SQL statements like built-ins or bind variables, you can decide whether to store it in the database or in the form. It is not mandatory that PL/SQL objects be written to be so portable, but it can be helpful when doing performance tests.
I FOR EXAMPLE:
If PL/SQL objects are portable, they can be moved back and forth from a form to the database with no changes to either the object itself or the form that is using it. Therefore, you can easily test an application to see where the code runs best. PL/SQL objects can be moved back and forth between the database and a Forms application by dragging and dropping the objects in the Object Navigator from the forms module to the Database Objects node. This is often referred to as application partitioning, because you are partitioning the application by having some of the PL/SQL logic in the form and some in the database.
LAB 10.3
go to contents
376
Expand the nodes under the COURSE_TAKEN program unit and study the information under the Specification and Referenced By nodes. a) Which trigger calls this function?
LAB 10.3
b) What parameters does the trigger pass to the function? What will the trigger do if the function returns TRUE?
Run the form and issue a query. Scroll through the student records until you find one that has two or three enrollments. Use the LOV for SECTION_ID to add another enrollment. Purposely choose a course that the student has already taken to test the function. Look at the hint line to confirm that the error message appears. Exit the running form and return to the Form Builder. c) How can you use the Object Navigator to make this function a stored database function? Make sure you store it in the STUDENT schema.
d) Do you have to change the code in the trigger that calls this function?
e) What should you do with the version of the function that is stored within the forms module? Why?
Run the form again and test that the function still works. go to contents
377
LAB 10.3
go to contents
378
LAB 10.3
2) Stored PL/SQL objects can accept and return parameters. a) ____ True b) ____ False 3) Which of the following statements would not be valid in both a client Forms PL/SQL object and a stored PL/SQL object? a) ____ SELECT LAST_NAME INTO v_ last_name FROM STUDENT; b) ____ OPEN CURSOR c_student; c) ____ FIND_ITEM(v_item); d) ____ END; 4) In what order does Forms search for PL/SQL objects? a) ____ Forms module, database, attached library b) ____ Database, attached library, forms module c) ____ Forms module, attached library, database d) ____ Database, forms module, attached library Quiz answers appear in Appendix A, Section 10.3.
go to contents
379
CHAPTER
10
* This is optional. Attach COMMON.pll to R_TEMPLATE.fmb only if you plan to create more forms modules for the STUDENT schema. Make sure that the PRE-INSERT and PRE-UPDATE triggers have been updated in LIBRARY.olb so that all of their subclassed versions will be able to call the INSERT_AUDIT_ITEMS and UPDATE_AUDIT_ITEMS program units in COMMON.pll. 2) Edit all of the WHEN-VALIDATE-ITEM triggers in the following forms that could make use of GET_ALERT_TEXT: 1) 2) 3) 4) 5) R_STUDENT.fmb R_COURSE.fmb R_INSTRUCTOR.fmb R_CRSESECT.fmb R_STUDENRL.fmb
Make sure you subclass the VAL_ALERT alert to all of these forms. 3) Create a program unit that enables and disables the buttons on a toolbar depending on the mode of a form. Edit the WHEN-BUTTON-PRESSED triggers for the TOOLBAR buttons so that they call this program unit. Add this program unit to the COMMON.pll library.
go to contents
C H A P T E R
1 1
MULTIPLE-FORM APPLICATIONS
CHAPTER OBJECTIVES
In this Chapter, you will learn about: Calling One Form from Another Page 382
typical application will contain more than one forms module. Users will use one form to accomplish one task, and another form to accomplish another task. If these tasks are related, you may want to have one form call the other. You may even want to pass information from form to from. In this Chapter, you will learn a number of different methods for calling forms. In the examples and Exercises, you will be using one forms module to call another. In Chapter 13, Forms Menus, you will call forms from a menu system. While you are working through the Exercises in this Chapter, keep in mind that you will be reusing these methods in Chapter 13, Forms Menus.
Throughout this Chapter, you will come across the terms calling form and called form. Calling form refers to the form that is invoking another. Called form refers to the form that has been invoked. So, for example, if FORM_A calls FORM_B, then FORM_A is the calling form and FORM_B is the called form.
381
go to contents
LAB
11.1
There are three built-ins that call one form from another: OPEN_FORM, CALL_FORM, and NEW_FORM. To open another form, you simply issue one of these statements along with the forms name. The statement to open a form called COURSE would look like this: OPEN_FORM('COURSE'); Since the full path is not included in the OPEN_FORM statement, the Forms Runtime will look for the COURSE.fmx file in the paths listed in the Registry. It is possible to include the full path in a built-in that calls another form, but it is not recommended because it makes the application less portable. The form name is a mandatory parameter. However, each of the three built-ins can accept other parameters that, depending on how you set them, can affect the behavior and state of the calling form and called form. You will explore these parameters in the Exercises. The built-ins themselves also alter the behavior of the calling and called form.
go to contents
I FOR EXAMPLE:
Assume that FORM_A is the calling form and FORM_B is the called form. The following is a description of what will happen when FORM_A issues different built-ins: OPEN_FORM('FORM_B'); FORM_B will open, and FORM_A will remain active and accessible. The user will be able to navigate between both forms and access items in each. CALL_FORM('FORM_B'); FORM_B will open as a modal form. The user will not be able to leave FORM_B until it has been exited or closed. FORM_A may be visible, but none of its items will be accessible. NEW_FORM('FORM_B'); FORM_A will close and then FORM_B will open. The EXIT_FORM built-in will exit and close the current form. The CLOSE_FORM built-in will also exit and close a form, but it requires a form name as a parameter. CLOSE_FORM('FORM_B');
I FOR EXAMPLE:
Assume that FORM_A calls FORM_B. In a PL/SQL object in FORM_A, you could make the following two statements:
go to contents
I FOR EXAMPLE:
Assume that FORM_A is the calling form and FORM_B is the called form. In FORM_A, you would use the following built-in to create a parameter list: v_plist_id := CREATE_PARAMETER_LIST('forms_params'); v_plist_id is a local variable that would have been declared in the PL/SQL block. forms_params is the name of the parameter list that is being created; its object ID is being stored in v_plist_id. Once you have created the parameter list, you would add parameters and their values to it. ADD_PARAMETER(v_plist_id, 'P_1', text_parameter, v_value); v_plist_id tells the ADD_PARAMETER built-in which parameter list to work with. P_1 is the name of the parameter being added to the list. text_parameter indicates the type of parameter. When passing parameters from form to form, they must be defined as text_parameters. v_value is the value you are assigning to P_1. This would have been declared and assigned earlier and could contain anything: an items current value, a WHERE clause, etc. In the Exercises, the value of the parameter you create will contain a WHERE clause.
go to contents
Now that the code has been written to create a parameter in FORM_A, FORM_B must be prepared to accept this parameter. Here you would use the Object Navigator to create a parameter with the same name as the parameter that will be passed in. So, in this case, the parameter in FORM_B must also be named P_1.
In this Exercise, you will experiment with the three different built-ins for opening one form from another. You will use the form EX11_01 to open the form EX11_02. EX11_01.fmb contains a STUDENT block. EX11_02.fmb is a masterdetail form with a STUDENT block and an ENROLLMENT block. The STUDENT block is query-only. Open EX11_01.fmb and EX11_02.fmb in the Form Builder. a) When EX11_01 is running in the Forms Runtime, will it be able to run EX11_02.fmb? What must you do to EX11_02 so that the calling form can run it?
In EX11_01.fmb, create a WHEN-BUTTON-PRESSED trigger for the STUDENT.ENROLLMENT button. b) What statement should you write in this trigger to open the form EX11_02? You want both forms to be open at the same time and you want both to be accessible to the user.
go to contents
Lab 11.1: Calling One Form from Another c) Can you navigate from form to form with the mouse? What happens if you click the ENROLLMENTS button again?
d) Does the MDI toolbar work for both forms? Do you have to exit both forms explicitly or does clicking the EXIT button close the entire application?
Exit the forms and return to the Form Builder. Add the following parameters to the built-in. Position them just after the form name. Do not put them in single quotes.
NO_ACTIVATE, SESSION
Run the form and test the button. e) What has the NO_ACTIVATE parameter done to the called form? What do you think the SESSION parameter has done?
f) How can you change the code in the WHEN-BUTTON-PRESSED trigger so that EX11_01 will close when EX11_02 is opened? The only parameter in the built-in should be the form name.
Add the following parameters to the built-in. Position them just after the form name. Do not put them in single quotes.
TO_SAVEPOINT, QUERY_ONLY
go to contents
g) What effect will the QUERY_ONLY parameter have on the called form?
Run the form and click the ENROLLMENTS button to test your work. Confirm that the QUERY_ONLY parameter is effective. h) How can you change the code in the WHEN-BUTTON-PRESSED trigger so that EX11_01 will stay open and EX11_02 will open as a modal form?
11.1.2
Open forms EX11_01.fmb and EX11_02.fmb in the Form Builder. In EX11_01.fmb, create a program unit that is a procedure and name it ADD_ENROLLMENTS. This program unit will not accept any parameters. You are going to create the parameter list and call form EX11_02.fmb from this procedure. It is being done here, rather than in the WHEN-BUTTONPRESSED trigger, so that the code will be available to other Forms events rather than just the Button Pressed event. Change the code in the WHEN-BUTTON-PRESSED trigger so that it has the following statement only:
ADD_ENROLLMENTS;
go to contents
Lab 11.1: Calling One Form from Another a) What is the name of the parameter list? Which variable is holding its object ID?
b) What is the name of the parameter being added to the parameter list? What type of parameter is it? What will its value be?
Compile the procedure and run the form. Click the button once. Nothing should happen. Click the button a second time and note the alert messages. Exit the form and return to the Form Builder. c) Can you create a parameter list twice? What code can you add to the beginning of ADD_ENROLLMENTS to check to see if the parameter list called forms_params already exists? Hint: Use a GET_ built-in to get the object ID of the parameter list, then evaluate the ID to see if it is null.
d) If the list exists and the value is not null, what built-in can you use to destroy the parameter list? Search the help system for the answer.
Check the answers section to confirm the code in ADD_ENROLLMENTS before continuing. Open form EX11_02.fmb.
go to contents
e) How can you use the Object Navigator to create a parameter in form EX11_02.fmb? What should you name the parameter?
f) How is the parameter being referred to? Do you have to refer to the parameter list? Do you think the value of the parameter will be available to all triggers and procedures in EX11_02.fmb?
Compile form EX11_02. Run form EX11_01 and click the ENROLLMENTS button to test that the parameter is being passed properly. Exit the running form and return to the Form Builder. In EX11_02, edit the WHEN-NEW-FORM-INSTANCE trigger and enter the following code:
DECLARE v_where VARCHAR2(100); BEGIN IF :PARAMETER.P_1 IS NOT NULL THEN v_where := :PARAMETER.P_1; SET_BLOCK_PROPERTY('STUDENT', DEFAULT_WHERE, v_where); EXECUTE_QUERY; GO_BLOCK(ENROLLMENT); END IF; END;
go to contents
Lab 11.1: Calling One Form from Another g) What do the v_where variable and SET_BLOCK_PROPERTY built-in accomplish?
Compile EX11_02.fmb and run EX11_01.fmb. Issue a query or create a new record. Click the ENROLLMENTS button to test the CALL_FORM and parameter list. i) In EX11_01.fmb, you issue a COMMIT_FORM statement just before you call form EX11_02.fmb. If you were creating a new record in EX11_01.fmb and you did not commit the form, would the query in EX11_02 return any rows? Why not?
Open EX11_01.fmb and EX11_02.fmb in the Form Builder. a) When EX11_01 is running in the Forms Runtime, will it be able to run EX11_02.fmb? What must you do to EX11_02 so that the calling form can run it? Answer: No it will not. You must compile EX11_02.fmb so that the calling form can run it. As you already know, an .fmb file cannot be run. The CALL_FORM, OPEN_FORM, and NEW_FORM built-ins will only run .fmx files.
go to contents
b) What statement should you write in this trigger to open the form EX11_02? You want both forms to be open at the same time and you want both to be accessible to the user. Answer: See the discussion below. You would use the OPEN_FORM built-in. The statement should look like the following: OPEN_FORM('EX11_02'); c) Can you navigate from form to form with the mouse? What happens if you click the ENROLLMENTS button again? Answer: Yes, you can navigate from form to form with the mouse. If you click ENROLLMENTS again, another instance of the EX11_02 form will open. Each time you click the ENROLLMENTS button, you will open another instance of the EX11_02 form. This can be useful if you want to allow users to work on multiple tasks simultaneously. When multiple forms are open, you can navigate between them using navigation built-ins.
I FOR EXAMPLE:
If forms EX11_01 and EX11_02 are open, you can navigate from one to the other with the following statement: GO_FORM('EX11_02'); The GO_FORM built-in is passed a name or object ID to open a specific form. Additionally, there are two other built-ins for navigating from form to form and their syntax is as follows: NEXT_FORM; and PREVIOUS_FORM; Note that these two built-ins do not refer to a specific form by name or object ID. Instead, NEXT_FORM and PREVIOUS_FORM navigate from form to form in the order that the forms were opened. d) Does the MDI toolbar work for both forms? Do you have to exit both forms explicitly or does clicking the EXIT button close the entire application? Answer: The MDI toolbar works for both forms. You must explicitly exit each form.
go to contents
OPEN_FORM('EX11_02',ACTIVATE,NO_SESSION,NO_SHARE_LIBRARY_DATA,'parameter_list_name');
You are not required to include all of the parameters in an OPEN_FORM statement. If you leave any of them out, Forms will simply issue the builtin with the default values for each parameter.
I FOR EXAMPLE:
The following statement: OPEN_FORM('EX11_02'); will produce the same results as the statement: OPEN_FORM('EX11_02', ACTIVATE, NO_SESSION, NO_SHARE_LIBRARY_DATA); This rule applies to the NEW_FORM and CALL_FORM built-ins, as well as to any other built-in that accepts parameters. The on-line help system details the parameters for each built-in and its allowable values, along with the default value. f) How can you change the code in the WHEN-BUTTON-PRESSED trigger so that EX11_01 will close when EX11_02 is opened? The only parameter in the built-in should be the form name. Answer: See the discussion below. You would use the NEW_FORM built-in. The statement would look like the following:
go to contents
NEW_FORM can also accept as many as five parameters, which are as follows: NEW_FORM(form name, rollback mode, query mode, data mode, parameter list); rollback mode lets you indicate how the form should roll back uncommitted changes in the calling form. The rollback mode parameter becomes important in applications with transactions that span multiple forms. It accepts the values TO_SAVEPOINT (the default), NO_ROLLBACK, and FULL_ROLLBACK. The statement for Question g will open the form in Query-Only mode, which will prevent the user from issuing inserts, updates, or deletes. The form cannot be switched to Normal mode and will remain in Query-Only mode throughout the session. h) How can you change the code in the WHEN-BUTTON-PRESSED trigger so that EX11_01 will stay open and EX11_02 will open as a modal form? Answer: See the discussion below. You would use the CALL_FORM built-in. CALL_FORM can accept as many as six parameters, which are as follows:
CALL_FORM(form name, display, switch menu, query mode, data mode, parameter list);
The display parameter indicates how the calling form should be displayed while the called form is active. It accepts one of two values: HIDE or NO_HIDE. If you want the calling form to be hidden while the called form is active, you would issue the following statement: CALL_FORM('EX11_02', HIDE); The switch menu parameter accepts one of two values: NO_REPLACE or DO_REPLACE. In Chapter 8, Canvases and Windows, you learned that each module has a Menu Module property; the menu module assigned here is referred to as the forms default menu. It is possible for forms within the same application to have different menu modules assigned to them at run-time. The switch menu parameter lets you indicate whether or not the called form should use the default menu of the calling form.
go to contents
I FOR EXAMPLE:
At design-time, form EX11_01 was assigned menu module A; form EX11_02 was assigned menu module B. The following statement will cause the called form, EX11_02, to use the menu module of the calling form, EX11_01. CALL_FORM('EX11_02', NO_REPLACE); The following statement will also call EX11_02 from EX11_01. However, in this case, the called form, EX11_02, will use its own default menu. In other words, it will replace the default menu of the calling form, EX11_01. CALL_FORM('EX11_02', DO_REPLACE);
11.1.2
ANSWERS
Answer: The name of the parameter list is forms_params. v_plist is the variable that is holding its object ID.
a) What is the name of the parameter list? Which variable is holding its object ID?
b) What is the name of the parameter being added to the parameter list? What type of parameter is it? What will its value be? Answer: The parameter is called P_1, and it is a text_parameter. The value of the parameter will be the value assigned to the v_where variable. This will contain the character string 'STUDENT_ID = ', concatenated to the current value of :STUDENT.STUDENT_ID. c) Can you create a parameter list twice? What code can you add to the beginning of ADD_ENROLLMENTS to check to see if the parameter list called forms_params already exists? Hint: Use a GET_ built-in to get the object ID of the parameter list, then evaluate the ID to see if it is null. Answer: See the discussion below. Forms cannot create two parameter lists of the same name, which is why you received errors when you clicked the ENROLLMENTS button the second time. Before you create a parameter list, you should check to see if one of the same name already exists and then destroy it if it does. In previous Labs, you used FIND_ built-ins to return the object ID of an object. There is no FIND_PARAMETER_LIST built-in; you must use the following built-ins instead:
go to contents
d) If the list exists and the value is not null, what built-in can you use to destroy the parameter list? Search the help system for the answer. Answer: See the discussion below. The code should be as follows: v_plist_id := GET_PARAMETER_LIST('forms_params'); IF NOT ID_NULL(v_list_id) THEN DESTROY_PARAMETER_LIST(v_list_id); END IF; DESTROY_PARAMETER_LIST erases the parameter list and any parameters and values that may have been added to it. Now you are free to re-create the parameter list and add new parameters to it. In this example, the parameter list is being used to pass a WHERE clause from one form to another. You may want to perform this action multiple times within a users session. Therefore, it will be necessary to destroy the existing parameter list before creating a new one. Check the Answers section to confirm the code in ADD_ENROLLMENTS before continuing. Open form EX11_02.fmb. The code for the procedure should be as follows: PROCEDURE ADD_ENROLLMENTS IS v_plist_id PARAMLIST; v_where VARCHAR2(50); BEGIN v_plist_id := GET_PARAMETER_LIST('forms_params'); IF NOT ID_NULL(v_plist_id) THEN DESTROY_PARAMETER_LIST(v_plist_id); END IF; v_where := 'STUDENT_ID = '||:STUDENT.STUDENT_ID; v_plist_id := CREATE_PARAMETER_LIST('forms_params'); ADD_PARAMETER(v_plist_id, 'P_1', TEXT PARAMETER, v_where); COMMIT_FORM; CALL_FORM('EX11_02', NO_HIDE, NO_REPLACE, NO_QUERY_ONLY, v_plist_id); END;
go to contents
g) What do the v_where variable and SET_BLOCK_PROPERTY built-in accomplish? Answer: They set the STUDENT blocks WHERE clause. In this example, the calling form is passing a WHERE clause to the called form, setting the WHERE clause for a block in the called form, and then executing a query. For forms EX11_01 and EX11_02, the result is that the same record is brought up in the called form and the user is given the opportunity to add detail records to it. One caveat to this example is that, since the WHERE clause has been set, the user will only be able to access enrollment records for this STUDENT. This
go to contents
might be fine for this example, since you are calling this form explicitly to set enrollments for this student. However, if you wanted to give the user the ability to query other student records, you could reset the WHERE clause by adding another SET_BLOCK_PROPERTY statement as follows: DECLARE v_where VARCHAR2(100); BEGIN IF :PARAMETER.P_1 IS NOT NULL THEN v_where := :PARAMETER.P_1; SET_BLOCK_PROPERTY('STUDENT', DEFAULT_WHERE, v_where); EXECUTE_QUERY; SET_BLOCK_PROPERTY('STUDENT', DEFAULT_WHERE, ' '); GO_BLOCK('ENROLLMENT'); END IF; END; The form will execute the query using the WHERE clause that was passed into the form, and will then reset the blocks DEFAULT_WHERE property immediately after the query has been executed. h) Why is the GO_BLOCK('ENROLLMENT'); statement issued? Answer: So that the form will navigate to the ENROLLMENT block. The STUDENT block is query-only, so the user will not be able to change the record. Moreover, the purpose of this example is for the user to add enrollment records for this student, so it makes sense for the application to navigate to the first item in the ENROLLMENT block immediately. i) In EX11_01.fmb, you issue a COMMIT_FORM statement just before you call form EX11_02.fmb. If you were creating a new record in EX11_01.fmb and you did not commit the form, would the query in EX11_02 return any rows? Why not? Answer: No it would not. The changes to form EX11_01 would not have been applied or committed to the database. The COMMIT_FORM statement, just before calling form EX11_02, applies and saves all the changes to form EX11_01. Therefore, there were no transactions that spanned both forms. However, there may be cases in which you want to spread transactions across multiple forms. To do so, you would take advantage of Forms commit processing functionality.
go to contents
I FOR EXAMPLE:
The COMMIT_FORM statement applies and saves changes to the database in one step. Once you have committed changes with COMMIT_FORM, they cannot be rolled back. The POST built-in, on the other hand, applies changes to the database but does not commit them. These changes can then be committed or rolled back later on. In this example, you could replace the COMMIT_FORM; statement with a POST; statement. The form will then post the changes and open form EX11_02. When a COMMIT_FORM is issued in form EX11_02, all of the changes, including the ones posted in EX11_01, will be committed to the database.
go to contents
go to contents
400
CHAPTER
11
401
4)
Call the procedure that shows the alert from a KEY-COMMIT trigger so that whenever a user attempts to save a change, they are prompted with the alert. You can do any one of the following: leave the button you created in Question 1 as it is, change the WHEN-BUTTON-PRESSED code so that it calls the alert procedure, or delete the button.
4) Repeat the steps for Challenge Question 3 for the R_COURSE.fmb and R_CRSESECT.fmb forms.
go to contents
C H A P T E R
1 2
n many applications, you will want to display database data in report format as well as in data entry screens. The Oracle Forms and Oracle Reports products have been tightly integrated so that forms modules can use built-ins to call reports modules and even pass them parameters. In this Chapter, you will learn two different methods for calling reports modules from forms modules. You will also learn how to pass userdefined parameters to influence a reports result set and how to pass system parameters to control how a report will be executed. Since Oracle Reports is not within the scope of this interactive workbook, you will not be expected to create your own reports modules. Instead, you will use the pre-created reports modules that you downloaded from the companion Web site along with the forms modules you have been using for previous Exercises. These three reports modules should be in your \guest\forms\exercises directory. They are called STUDENT.rdf, STUDENRL.rdf, and INSTSECT.rdf.
403
go to contents
LAB
12.1
Run an Oracle Report with RUN_PRODUCT Run an Oracle Report with RUN_REPORT_ OBJECT
Before you learn the methods for calling reports from forms, it is best to get a brief overview of the Oracle Reports product and its modules. In the simplest of terms, Oracle reports modules are SQL queries with formatting instructions. The two basic elements of a reports module are its query and its layout. The query specifies the data to be returned and the layout specifies how this result set should be formatted. Reports can be laid out in a number of different styles and can include color and graphics. In addition to the query and layout, you can also include triggers, program units, and attached libraries in reports modules. The Report Builder is used to design and build reports modules. It has a similar interface to the Form Builder in that it is a graphical development environment with an Object Navigator, Property Palette, Layout Editor, and PL/SQL Editor. Reports modules can be run by themselves, using the Reports Runtime, or they can be called from forms. In the Exercises in this Chapter, you will call reports from forms. There are two Forms built-ins that can be used to call reports modules: RUN_PRODUCT and RUN_REPORT_OBJECT. They achieve almost the same result, but are implemented differently.
go to contents
RUN_PRODUCT
The RUN_PRODUCT built-in is rather flexible in that it can be used to call any of the three types of Oracle Developer modules: forms, reports, and graphics. In this Lab, you will use RUN_PRODUCT to call reports modules only. It is simple, as well, because it takes just one single statement to run a report. It can be called from a trigger or a program unit and does not require that any additional objects be created in the calling form. The syntax for RUN_PRODUCT is:
RUN_PRODUCT(product, module, commode, execmode, location, parameter_list, display);
A call to RUN_PRODUCT to run a report called COURSE would look like this:
RUN_PRODUCT(REPORTS, 'COURSE', ASYNCHRONOUS, RUNTIME, FILESYSTEM, 'NULL', NULL);
The built-in accepts a number of parameters, the two most important being the name of the product to be called and the name of the module to be run. You will explore the meanings of the rest of the parameters in the Exercises.
RUN_PRODUCT_OBJECT
This built-in is not as flexible as RUN_PRODUCT because it applies to reports modules only. Therefore, it cannot be used to call graphics modules or forms modules. It is also not as simple because it requires more work in the Form Builder at design-time. However, it is a more powerful method since it provides tighter integration between the calling form and the called report.
I FOR EXAMPLE:
Assume you have a forms module called FORM_A. The syntax to run a report called COURSE from FORM_A using RUN_REPORT_OBJECT would be as follows: DECLARE v_repobj_id REPORT_OBJECT; v_repsrv_id VARCHAR2(100); BEGIN v_repobj_id := FIND_REPORT_OBJECT(COURSE); v_repsrv_id := RUN_REPORT_OBJECT(v_repobj_id); END;
go to contents
A reports module called STUDENT.rdf has been created for you already. You will not need to enter the Report Builder or in any way manipulate this reports module. You will simply be running it from within a form. However, you must make sure that the reports module (STUDENT.rdf) is in a directory that is pointed to by the REPORTXX_PATH Registry entry. If you have questions about the Windows Registry, refer to Appendix B. Open the form EX12_01.fmb. Ignore the data items for now; they will become important in Exercise 12.2.1. Create a WHEN-BUTTON-PRESSED trigger for the STUDENT.REPORTS button. Put the following statement in the trigger:
RUN_PRODUCT(REPORTS,'STUDENT',ASYNCHRONOUS,RUNTIME,FILESYSTEM,'NULL',NULL);
Run the form and put it in Enter Query mode. Click the Reports button.
go to contents
a) Can you navigate back and forth from the calling form to the reports module? What do you think the behavior would have been if commode had been set to SYNCHRONOUS?
b) Can you see the report and interact with it in the previewer? What do you think the behavior would have been if execmode had been set to BATCH?
c) Where will the Reports Runtime look for the reports module?
Exit the report and form and return to the Form Builder.
12.1.2
Open the form EX12_01.fmb in the Form Builder. Use the Reports node in the Object Navigator to create a report object based on an existing report called STUDENT. a) How should you respond to the New Report dialog to base the report object on an existing report named STUDENT?
go to contents
Lab 12.1: Running Oracle Reports from Forms b) The RUN_REPORT_OBJECT built-in does not accept parameters like commode, execmode, etc. Where do you think you can set them instead?
c) Which property can you set so that the report output gets sent to the screen?
Set the Execution Mode property to Runtime and the Communication Mode property to Asynchronous. d) What would the code be for the WHEN-BUTTON-PRESSED trigger to open this report with the RUN_REPORT_OBJECT built-in?
Run the form and test the report. Exit the form and return to the Form Builder. e) What built-in do you think you could use to set the properties of the report object at run-time?
Open the form EX12_01.fmb. Ignore the data items for now; they will become important in later Exercises. Create a WHEN-BUTTON-PRESSED trigger for the STUDENT.REPORTS button. Put the following statement in the trigger:
go to contents
Run the form and put it in Enter Query mode. Click the Reports button. a) Can you navigate back and forth from the calling form to the reports module? What do you think the behavior would have been if commode had been set to SYNCHRONOUS? Answer: Yes you can. The reports module would have been modal. The commode parameter determines whether or not the user should be able to return to the calling module before the called module has been exited. When set to ASYNCHRONOUS, the user can navigate back and forth between both modules. When set to SYNCHRONOUS, the user must exit the called module, in this case the report, before returning to the calling module. ASYNCHRONOUS mode can be useful in situations where a report is expected to take a long time to complete. The user can run the report and continue working in a forms module while they wait for the report to complete. b) Can you see the report and interact with it in the previewer? What do you think the behavior would have been if execmode had been set to BATCH? Answer: It would have run the report in the background. BATCH mode is useful when the user does not want to view the report onscreen. Instead, they want to send the report results to a file or printer. c) Where will the Reports Runtime look for the reports module? Answer: It will look in the filesystem. As you have learned in previous Chapters, Oracle Developer lets you choose between storing modules in the filesystem or in the database. The location parameter indicates where the Reports Runtime should look to find the module to be run. In this case, since you have chosen filesystem and you have not included the full path with the report name, the Reports Runtime will search the directories indicated in the Windows Registry for the report file. d) Is the form passing the report any parameters? Answer: No it is not.
go to contents
12.1.2
ANSWERS
Open the form EX12_01.fmb in the Form Builder. Use the Reports node in the Object Navigator to create a report object based on an existing report called STUDENT. a) How should you respond to the New Report dialog to base the report object on an existing report named STUDENT? Answer: You should select the Use Existing Report File radio button and enter the name of the reports module in the Filename field. It is not necessary to include the entire path in the Filename field. The Reports Runtime will be able to find the reports module as long as it is stored in a directory that the Windows Registry points to. The STUDENT report is now represented as an object in the form EX12_ 01.fmb. The reports module itself is not stored here, nor are any of the report objects. However, having this pointer to the report contained in the forms module makes it much easier to control the running of the report. Incidentally, if you were to select Create New Report File, the Form Builder would launch the Report Builder and you would be able to create a new reports module. b) The RUN_REPORT_OBJECT built-in does not accept parameters like commode, execmode, etc. Where do you think you can set them instead? Answer: They can be set using the properties for the report object. The report object has additional properties for other SYSTEM.PARAMETERS, like Destination Name and Destination Format. If the report is being sent to a file or printer, these properties determine how it should be named and how it should be formatted.
go to contents
Destination Name and Destination Format can also be set using the RUN_PRODUCT built-in, but they would not be set using properties. They would have to be added to a parameter list, which requires coding. Set the Execution Mode property to Runtime and the Communication Mode property to Asynchronous. c) Which property can you set so that the report output gets sent to the screen? Answer: The Report Destination Type property. This property tells the Reports Runtime where to send the report. By choosing Screen, it sends the report to the previewer so that the user can view the report results. As their names imply, the values File and Printer would send the report output to a file or a printer, respectively. When selecting these options, you must also set the Destination Name property so that the report knows where to send the report. d) What would the code be for the WHEN-BUTTON-PRESSED trigger to open this report with the RUN_REPORT_OBJECT built-in? Answer: See the discussion below. The code would be as follows: DECLARE v_repobj_id REPORT_OBJECT; v_repins_id VARCHAR2(100); BEGIN v_repobj_id := FIND_REPORT_OBJECT('STUDENT'); v_repins_id := RUN_REPORT_OBJECT(v_repobj_id); END; As discussed in the Lab text, the FIND_REPORT_OBJECT built-in returns the object ID of the report object. RUN_REPORT_OBJECT runs the report and also returns a value. This value is the unique ID that the Reports engine has assigned for this job. This Reports engine could be on the users machine or on a remote server somewhere in the network. You can use the ID returned by RUN_REPORT_OBJECT to get information about the status of a specific report job.
I FOR EXAMPLE:
REPORT_OBJECT_STATUS(v_repins_id); would return a VARCHAR2 value, indicating whether this job has finished, has been canceled, is still running, etc. This can be very helpful, es-
go to contents
I FOR EXAMPLE:
You could present the user with a control item like a radio group that would let them choose whether to display the report on-screen or send it to a printer. The code might be as follows: IF :block.radio_group = 'SCREEN' THEN SET_REPORT_OBJECT_PROPERTY('STUDENT', REPORT_DESTYPE, SCREEN); ELSIF :block.radio_group = 'PRINTER' THEN ...
go to contents
4) Which of the following built-in statements is correct? a) ____ RUN_REPORT_OBJECT('STUDENT'); b) ____ RUN_PRODUCT(REPORTS, 'STUDENT', SYNCHRONOUS, RUNTIME, FILESYSTEM, 'NULL', NULL); c) ____ RUN_PRODUCT_OBJECT(v_repobj_id); d) ____ RUN_PRODUCT(REPORTS, 'STUDENT', SYNCHRONOUS, RUNTIME, FILESYSTEM, NULL, NULL); Quiz answers appear in Appendix A, Section 12.1.
go to contents
414
LAB
LAB 12.2
12.2
Reports modules can be passed parameters using parameter lists. In fact, the methods for creating and using parameter lists are no different from what you learned in Chapter 11, Multiple-Form Applications. You first must create a parameter list in a form, add parameters to it, and then pass it to the called reports module. Both the RUN_PRODUCT_ OBJECT and RUN_PRODUCT built-ins accept parameter list names or parameter list object IDs.
I FOR EXAMPLE:
If you had created a parameter list and stored its object ID in a variable called v_plist_id, the syntax to call a report named SECTIONS would be as follows:
RUN_PRODUCT(REPORTS, 'SECTIONS', ASYNCHRONOUS, RUNTIME, FILESYSTEM, v_plist_id, NULL);
go to contents
415
For the report to accept the parameter, it must be created in the Report Builder at design-time. This requirement is identical to that when passing parameters from form to form using CALL_FORM, etc.
I FOR EXAMPLE:
If a parameter called P_1 had been defined in the reports module at design-time, the ADD_PARAMETER statement in the forms module would look like this: ADD_PARAMETER(v_plist_id, 'P_1', text_parameter, v_value);
LAB 12.2
go to contents
416
LAB 12.2
In this Exercise, you will pass the value of the forms modules current STUDENT_ID to a report called STUDENRL. The report will use this value in its WHERE clause to show the enrollment information of the current student. Open the form EX12_01.fmb in the Form Builder. The name of the parameter in the STUDENRL report is P_1. a) What would the code be to create a parameter list and then add the current STUDENT_ID to the list as a parameter? Remember to add code statements to destroy the parameter list if it already exists. Also remember to add a parameter to suppress the reports modules parameter form.
Run the form and test the report. Issue a query for STUDENT_ID 248 to be certain that you are working with a student with enrollment records. c) How would you have to change the WHEN-BUTTON-PRESSED trigger so that it would use RUN_REPORT_OBJECT instead of RUN_PRODUCT? Remember to add another report object to the form for the STUDENRL report.
go to contents
417
Open the form EX12_01.fmb in the Form Builder. The name of the parameter in the STUDENRL report is P_1. a) What would the code be to create a parameter list and then add the current STUDENT_ID to the list as a parameter? Remember to add code statements to destroy the parameter list if it already exists. Also remember to add a parameter to suppress the reports modules parameter form. Answer: See the discussion below. The code would be as follows: DECLARE v_plist_id PARAMLIST; BEGIN v_plist_id := GET_PARAMETER_LIST('rep_params'); IF NOT ID_NULL(v_plist_id) THEN DESTROY_PARAMETER_LIST(v_plist_id); END IF; v_plist_id := CREATE_PARAMETER_LIST('rep_params'); ADD_PARAMETER(v_plist_id, 'P_1', text_parameter, :STUDENT.STUDENT_ID); ADD_PARAMETER(v_plist_id, 'PARAMFORM', text_parameter, 'NO'); b) What would the RUN_PRODUCT syntax to call STUDENRL be? Answer: See the discussion below. The code would be as follows:
RUN_PRODUCT(REPORTS, 'STUDENRL', ASYNCHRONOUS, RUNTIME, FILESYSTEM, v_plist_id, NULL);
c)
How would you have to change the WHEN-BUTTON-PRESSED trigger so that it would use RUN_REPORT_OBJECT instead of RUN_PRODUCT? Remember to add another report object to the form for the STUDENRL report. Answer: See the discussion below.
go to contents
418
LAB 12.2
DECLARE v_plist_id PARAMLIST; v_repobj_id REPORT_OBJECT; v_repins_id VARCHAR2(100); BEGIN v_plist_id := GET_PARAMETER_LIST('rep_params'); IF NOT ID_NULL(v_plist_id) THEN DESTROY_PARAMETER_LIST(v_plist_id); END IF; v_plist_id := CREATE_PARAMETER_LIST('rep_params'); ADD_PARAMETER(v_plist_id, 'P_1', text_parameter, :STUDENT.STUDENT_ID); ADD_PARAMETER(v_plist_id, 'PARAMFORM', text_parameter, 'NO'); v_repobj_id := FIND_REPORT_OBJECT('STUDENRL'); v_repins_id := RUN_REPORT_OBJECT(v_repins_id, v_plist_id); END;
go to contents
419
RUN_REPORT_OBJECT(v_repobj_id, v_plist_id, SYNCHRONOUS); a) b) c) d) ____ The built-in should be RUN_PRODUCT_OBJECT ____ The report object should be referenced by module name ____ RUN_REPORT_OBJECT does not accept parameter list IDs ____ SYNCHRONOUS should be set with properties, not through the built-in
LAB 12.2
go to contents
420
CHAPTER
12
go to contents
C H A P T E R
1 3
FORMS MENUS
CHAPTER OBJECTIVES
In this Chapter, you will learn about: Menu Modules Menu Security Page 422 Page 435
ow that you have created a number of forms modules, it is time to learn how to tie them all together in an application with a menu system. So far, all of your forms modules have been run with the DEFAULT menu that is provided for each form. It has some basic but very useful functions for editing, navigation, querying, and so on. In this Chapter, you will learn how to create your own custom menus that go a little beyond the DEFAULT menu in functionality. In Chapter 11, Multiple-Form Applications, you learned several built-ins for calling one form from another. You will use these same built-ins to call forms from a menu system. This will let you unite many of the forms you have created in previous Labs in a single application. You will also learn how to use menu modules to implement security features to control access to the application.
421
go to contents
LAB
13.1
MENU MODULES
LAB OBJECTIVES
After this Lab, you will be able to:
MENU OBJECTS
Menu modules and their objects can be created in the Form Builder. There are four main components of a menu module: the menu module itself, the Main Menu, individual menus, and menu items. Figure 13.1 puts these objects in the context of a running form.
go to contents
The menu module itself, like a forms module, is not visible to the user. It is the logical container object that owns all of the other menu objects. The Main Menu is the horizontal bar across the top that contains the labels INDIVIDUAL_MENU_1, INDIVIDUAL_MENU_2, and Window. Each of these labels is an individual menu that contains a group of menu items. The menu items for INDIVIDUAL_MENU_2 are being displayed. The menu items are the most important objects in the menu module because they are what the users select to initiate actions.
MENU ITEMS
There are five types of menu items: plain, magic, check, radio, and separator. Plain menu items are the ones you will use most. They display text labels and have PL/SQL code behind them.
I FOR EXAMPLE:
A plain menu item labeled Save might have the following code behind it: DO_KEY('COMMIT_FORM'); As you can imagine, when the user selects this menu item, Forms executes the DO_KEY built-in and saves any changes. Magic items are perhaps the most convenient menu items because some of them already have code associated with them; therefore, you do not have to write any code yourself. They can be used to perform functions that are common to most applications.
I FOR EXAMPLE:
A magic item of type Copy has the code for copying text already associated with it. Therefore, at design-time, all you have to do is indicate that you want a certain menu item to be a magic item and Forms will associate the proper code with it. Check and radio menu items allow you to create Boolean menu items that are often used to configure the state of the application at run-time.
I FOR EXAMPLES:
The Form Builder uses check and radio menu items under View in the Main Menu. These items allow you to select the view style of the Object Navigator and whether or not to display PL/SQL objects. The View menu is shown in Figure 13.2.
go to contents
Figure 13.2 I The View menu in the Form Builder showing radio and check menu item types.
In your applications, you could create a check menu item called Display Toolbar that would hide or display a toolbar. Separators are the only menu items that do not have any commands associated with them. They are simply dummy items that you can use to separate menu items to make your menus more readable. In Figure 13.2, there is a separator between Visual View and Show PL/SQL only.
go to contents
The purpose of this Exercise is to create a menu module and experiment with some of the different types of menu items. You will begin by examining the DEFAULT menu provided by Oracle Forms. Then you will create and configure your own custom menu. Open the form EX13_01.fmb in the Form Builder. Run the form. a) What is the second individual menu in this menu module and what are its menu items? Can you guess their menu item types?
Close the form and return to the Form Builder. Use the Object Navigator to create a menu module. Name it MAIN and save it. Right-click the menu module and open the Menu Editor. b) What objects have been created automatically?
c) How can you use the Menu Editors toolbar to create a menu item for this individual menu?
In the Menu Editor, name the individual menu File and the menu item Exit. Look at the properties for each menu object. d) What is the label property for the Exit menu item?
go to contents
e) How can you make the menu item Exit a magic item that will quit the application?
Compile the menu module using CTRL-T. f) How can you attach the R_MAIN menu module to EX13_01? Do you have to use its full path or just the module name?
Run the EX13_01.fmb form and test the menu. Exit the form and return to configuring the menu in the Form Builder. g) How can you use the Menu Editors toolbar to create another individual menu with two menu items? Make sure you create this new individual menu in the Main Menu at the same level as File.
Name the individual menu Registration and the menu items Register Students and Enroll Students. Register Students and Enroll Students will be used to open the R_STUDENT and R_STUDENRL forms, respectively. h) What should the menu item type be for Register Students and Enroll Students? Which property can you use to enter code to call them? For both menu items, use the CALL_FORM command and pass it only the forms modules name.
Compile the menu module. Run the EX13_01.fmb form and test the menu. Exit the form and return to configuring the menu in the Form Builder.
go to contents
Compile the menu module. Run the EX13_01.fmb form and test the menu items under Registration. i) What is different about the menu for each form? What has caused this difference?
Exit the forms and return to the Form Builder. Both forms should use the R_MAIN menu to change the menu item command for Register Students to:
CALL_FORM('R_STUDENT, HIDE, NO_REPLACE);
j) If you were to create another individual menu with menu items to call reports modules, what types of menu items would these be? What code would you put behind them to call the reports modules? Do not take these steps in the Menu Editor; simply answer the question.
go to contents
Open the form EX13_01.fmb in the Form Builder. Run the form. a) What is the second individual menu in this menu module and what are its menu items? Can you guess their menu item types? Answer: The second individual menu is Edit. Its menu items are Cut, Copy, Paste, Error, and Display List. Cut, Copy, and Paste are magic items. Error and Display List are plain menu items with code behind them. The menu you are exploring here is the DEFAULT menu that is attached to every newly created form. It is usually accompanied by the SMARTBAR, which you read a little bit about in Chapter 8, Canvases and Windows. The SMARTBAR has been removed for this Exercise. The DEFAULT menu cannot be edited directly. You cannot make changes to a menu module called DEFAULT.mmb in the Form Builder. You must either use DEFAULT as-is or attach a different menu module to the form. There may be some features within DEFAULT that you want to reuse. For that reason, a menu module called file menudef.mmb, which is identical in function to the DEFAULT menu, was installed along with your Oracle Developer files. You should be able to find it in the \ORACLE_HOME\ DEVDEM60\DEMO\FORMS directory. You edit this module directly or reuse some of its objects in other modules. For example, you may want to copy and paste the EDIT individual menu from menudef.mmb into your custom menu modules.
go to contents
Figure 13.5 I A menu module in the Object Navigator with one Main Menu and one menu item.
to the MENUCANV canvas and its Width and Height properties are set to 0,0 so that it is not visible to the user. It is not mandatory to have a functionless default menu form as your starting point. It is also common to attach a menu to a form that is always opened first by the user.
I FOR EXAMPLE:
In the Student application, if it were the case that users began each session by entering or updating records in the STUDENT table, then it may have made sense to attach the menu module to R_STUDENT.fmb. This would then be the forms module that would be the starting point of the application. b) What objects have been created automatically? Answer: In the Object Navigator, there is a Main Menu called MENU1 and an individual menu called ITEM2. In the Menu Editor, there is only an individual menu labeled New Item. Figures 13.5 and 13.6 show what you should be seeing now in the Object Navigator and Menu Editor.
Figure 13.6 I A menu module in the Menu Editor with one individual menu. go to contents
How can you use the Menu Editors toolbar to create a menu item for this individual menu? Answer: Click the Create Down button on the Menu Editors toolbar.
d) What is the label property for the Exit menu item? Answer: The labels have been changed to File and Exit. Note that changes have been made in the Object Navigator as well. This illustrates that it is much easier to label menu objects using the Menu Editor because it sets the name property as well. If need be, you can give menu objects two-word labels and Forms will automatically insert an underscore when it sets the objects name. For example, if you type Save Form into the label in the Menu Editor, Forms will set the name of this object to SAVE_FORM. Hot keys can be set for the labels of menu items as well. The user can press the hot key associated with a menu item instead of clicking it.
I FOR EXAMPLE:
The hot key for the Exit menu item is E. If the user selects the individual menu File and presses E, the menu module will respond as if the user clicked Exit with the mouse. The default behavior is that the first letter of the label is designated as the hot key for all menu items. This can be changed by inserting an ampersand into the label just before the letter that you wish to make the hot key. So, if you wanted x to be the hot key for Exit, you would write the label as follows: E&xit e) How can you make the menu item Exit a magic item that will quit the application? Answer: You set its Menu Item Type property to Magic and its Magic Item property to Quit. The Quit, Cut, Copy, Paste, Clear, and Window magic items already have code behind them that makes them easy and convenient to use. In this case, by selecting Quit as the magic item, you do not have to write any PL/SQL to exit or close the current form, as Forms will automatically execute the code built into the magic item. f) How can you attach the R_MAIN menu module to EX13_01? Do you have to use its full path or just the module name? Answer: You set the EX13_01 module-level Menu Module property to R_MAIN. No, you do not have to include the full path.
go to contents
Figure 13.7 I The Menu Editor with a new individual menu and two new menu items. go to contents
The switch menu parameter of the CALL_FORM built-in determines which menu should be attached to the called form. Both R_STUDENT and R_STUDENRL have DEFAULT as the setting for their Menu Module property. The calling form is EX13_01, which has R_MAIN set for its Menu Module property. DO_REPLACE in the call to R_STUDENT instructed the form to replace the menu of the calling form with the menu of the called form. Therefore, R_STUDENT (the called form) displayed DEFAULT, its own menu. NO_REPLACE in the call to R_STUDENRL instructed the form not to replace the menu of the calling form with the menu of the called form. Therefore, R_STUDENRL (the called form) displayed R_MAIN, the menu attached to EX13_01 (the calling form). NO_REPLACE is the default value of the CALL_FORM statement. Exit the forms and return to the Form Builder. Both forms should use the R_MAIN menu, so change the menu item command for Register Students to: CALL_FORM('R_STUDENT', HIDE, NO_REPLACE); j) If you were to create another individual menu with menu items to call reports modules, what types of menu items would these be? What code would you put behind them to call the reports modules? Do not take these steps in the Menu Editor; simply answer the question. Answer: You could use either RUN_PRODUCT or RUN_PRODUCT_OBJECT. Both built-ins would work equally well. However, to use RUN_PRODUCT_ OBJECT, you would have to create a report object in EX13_01.fmb.
go to contents
go to contents
435
LAB
13.2
LAB 13.2
MENU SECURITY
LAB OBJECTIVES
After this Lab, you will be able to:
There may be instances in which you will want to restrict access to parts of your application from certain users. You will want to give some users access to all of the modules in an application, but then provide other users access to only a few modules in the application. Instead of creating two or more completely different menu systems for each group of users, you can create a single menu system and implement menu security by granting and restricting access to individual menus and menu items. The menu security system is integrated with the database, meaning that you can base menu item access on database roles.
I FOR EXAMPLE:
Assume that in the database you have created two roles: one called STUDENT_USERS and one called OFFICE_USERS, with various users assigned to each role. STUDENT_USERS can only query the COURSE and SECTION tables to get information about courses. OFFICE_USERS can perform all DML operations on all of the tables in the STUDENT schema. Also assume that you have created an application with a number of forms modules, one of which is a query-only master-detail form showing course and section data. When creating the menu module, you can set properties for the menu items so that the STUDENT_USERS role will only be able to access the menu item that calls the course and section master-detail form. You would use the same properties to indicate that the OFFICE_USERS role will be able to access all menu items in the application.
go to contents
436
LAB 13.2
Both steps are taken using the properties that you will explore in the Exercises.
To practice menu security, you will need to create some sample database users and database roles. Before attempting the Exercises, you must complete the following two steps:
1) Run scripts to create sample database users and roles. 2) Build Forms schema objects and grant the sample database users access to them.
Take the following steps to use these files to create users and roles:
1) Start an SQL*Plus session and connect as the SYSTEM user. You may have to ask your database administrator for help if you do not know the SYSTEM password. 2) Run the dba.sql file. 3) Wait for the script to finish. The script will issue some drop statements, which will result in error messages. Ignore these error messages. 4) When the script has finished, log into the STUDENT schema by issuing the following command from the SQL prompt. If learn is not the password you are using, then substitute the correct one:
go to contents
437
These scripts have created two users: reg_user/reg_user and adm_user/ adm_user. The scripts have also created two roles: ADMINISTRATOR and REGISTRAR. The ADMINISTRATOR role has been granted to adm_user. The REGISTRAR role has been granted to reg_user.
LAB 13.2
An SQL*Plus session will open and you will be required to enter the SYSTEM password and database connect string. The script will run and create all of the Forms schema objects. Next, you must take the following steps to grant your users access to these objects.
1) From the Windows Start menu, navigate to the Oracle Developer 6.0 Admin menu item again. 2) Select Oracle Developer Grant.
An SQL*Plus session will open and you will be required to enter the SYSTEM password and database connect string. You will then be prompted for the user name to which you wish to grant access.
3) Enter the user name STUDENT each time you are prompted for a user name. 4) Repeat Steps 1 through 3 for ADM_USER and REG_USER.
go to contents
438
You are now ready to use the Forms menu security system. If you ever want additional users to have access to the security system, you must repeat Steps 1 through 3 for each user.
LAB 13.2
Open the modules EX13_01.fmb and R_MAIN.mmb in the Form Builder. Select R_MAIN and view its properties. a) What happens when you click the Module Roles property? What values should you enter?
b) Which property should you set to indicate that you would like to activate menu security?
Select both the File and Registration individual menus. You can do this in either the Object Navigator or Menu Editor. c) What happens if you click the Item Roles property? How can you indicate that both roles should have access to both items?
Select the Exit menu item. Give both roles access to this menu item. Select the Register Students menu item. Give both roles access to this menu item. d) How can you set the properties so that the ADMINISTRATOR role will have access to Enroll Students but the REGISTRAR role will not have access to Enroll Students?
go to contents
439
Press CTRL-J and re-connect as REG_USER/REG_USER. Run form EX13_ 01.fmb. e) Can you select the Enroll Students menu item? Why not?
LAB 13.2
Exit the form and return to the Form Builder. f) Which property could you set so that REG_USER would not be able to see Enroll Students without proper access? Do not do this in the Form Builder. Simply write down your answer.
Reset the properties for Enroll Students so that both ADMINISTRATOR and REGISTRAR have access to it. Save R_MAIN.mmb and EX13_01.fmb.
Open the modules EX13_01.fmb and R_MAIN.mmb in the Form Builder. Select R_MAIN and view its properties. a) What happens when you click the Module Roles property? What values should you enter? Answer: The Menu Module Roles dialog opens. You should enter the ADMINISTRATOR and REGISTRAR roles. The Menu Module Roles dialog shown in Figure 13.8 is used to give certain database roles access to the menu system. This property gives certain roles overall access to the menu module. Later, the role names entered here will be used to grant or deny access to individual menus and menu items.
go to contents
440
LAB 13.2
Figure 13.8 I The Menu Module Roles dialog populated with role names.
b) Which property should you set to indicate that you would like to activate menu security? Answer: The Use Security property. The Use Security property enables or disables the menu security system. If it is set to No, menu security will not be implemented, even if it has been completely configured for all of the menu items in the module. Select both the File and Registration individual menus. You can do this in either the Object Navigator or Menu Editor. c) What happens if you click the Item Roles property? How can you indicate that both roles should have access to both items? Answer: The Menu Item Roles dialog will open. The Menu Item Roles dialog shown in Figure 13.9 displays the names of the roles that were entered in the Menu Module Roles dialog. By selecting ADMINISTRATOR and REGISTRAR, you are giving both of these roles access to the File and Registration individual menus. What is important to note is that by selecting multiple items in the Object Navigator or Menu Module, you were able to assign roles to multiple items simultaneously. This can be a big time saver when you have many individual menus or menu items that need to be accessed by the same roles. One caveat is that, while you were able to set the roles for more than one item at a time, you are not able to view the roles for more than one item at a time. If you wish to see which roles have already been assigned to a menu item, you must view its Item Roles property individually.
go to contents
441
LAB 13.2
Figure 13.9 I The Menu Item Roles dialog displaying available roles.
Select the Exit menu item. Give both roles access to this menu item. Select the Register Students menu item. Give both roles access to this menu item. d) How can you set the properties so that the ADMINISTRATOR role will have access to Enroll Students but the REGISTRAR will not have access to Enroll Students? Answer: Select the Item Roles property for Enroll Students. Select ADMINISTRATOR in the dialog but not REGISTRAR. By not selecting REGISTRAR for Enroll Students, you are denying access to this menu item to any user who has been granted the REGISTRAR role. e) Can you select the Enroll Students menu item? Why not? Answer: No, it is grayed out. REG_USER has been granted the REGISTRAR role, so it has been denied access to this menu item. If you were to log in as ADM_USER, which has been granted the ADMINISTRATOR role, the Enroll Students menu item would be enabled. f) Which property could you set so that REG_USER would not be able to see Enroll Students without proper access? Do not do this in the Form Builder. Simply write down your answer. Answer: Display Without Privilege. If the user does not have access to a certain menu item, and that items Display Without Privilege property is set to No, then the item will be hidden completely instead of grayed out.
go to contents
442
LAB 13.2
go to contents
443
CHAPTER
13
Use CALL_FORM to call all forms modules. The called forms should use the calling forms menu. The called form should be hidden while the calling form is active. Use RUN_PRODUCT to call reports modules. The report should be displayed on the screen. Control should return to the calling form immediately. Do not pass any parameters or suppress the reports modules parameter form.
444
2) Implement security for the menu. The ADMINISTRATOR role should have access to all of the individual menus and menu items. The REGISTRAR role should have access to all of the individual menus and menu items, except Administration and Add Instructors.
go to contents
A P P E N D I X
Answer
d b a
Comments
PL/SQL is the language used in Forms triggers and program units. The direct result of clicking a button is the Button Pressed event, which is an interface event. If there is a WHENBUTTON-PRESSED trigger associated with the button, then this trigger will fire to respond to the Button Pressed event.
4) 5)
d c
Question
1)
Answer
d
Comments
Blocks are logical containers of items. They are not physical objects because they do not have properties that would make them visible to the user. Modules, like blocks, are logical containers because they do not have properties that would make them visible to the user.
2)
3) 4) 5) 6)
a a d a
Frames are graphic objects that are owned by canvases. All of the objects in a form are owned by the forms module. So, while Answers b, c, and d are also true, a is a more complete answer.
445
go to contents
446
CHAPTER 2
Question
1)
Answer
d
Comments
The Layout Wizard will help you create the first cut or rough draft of a canvas. You are free to edit this canvas by reentering the Layout Wizard or by using the Layout Editor and Property Palette. The main function of the Data Block Wizard is to create blocks and associate them with tables in the database. It also helps you create items in blocks and associate them with columns in the database. Reentering the Layout Wizard will allow you to change the prompt value for an item as well as edit many other item and canvas properties. While they are not listed as answers for Question 3, it is also possible to edit item prompts in the Property Palette and Layout Editor. Since neither the Data Block Wizard or Layout Wizard have anything to do with the module level of a form, they do not affect the module name.
2)
3)
4)
Question
1) 2) 3) 4)
Answer
d c e c
Comments
Answer c is incorrect because icons are stored in .ico files, not as modules. Program Compile All will compile all of the PL/SQL in a form, but it will not create an .fmx file.
CHAPTER 3
Question
1) 2)
Answer
c d
Comments
go to contents
447
Only object names can be changed in the Object Navigator. All other properties must be changed in either the Property Palette or Layout Editor.
4) 5) 6) 7)
a b c b
Question
1) 2) 3)
Answer
c c d
Comments
Mass changes to properties can be done by selecting multiple objects in the Object Navigator and then changing their properties. It is not possible to delete item properties as Answer b suggests. However, in Chapter 9, Reusable Objects, you will learn how to delete properties from objects called property classes.
Question
1) 2) 3) 4) 5) 6)
Answer
c a b e b d
Comments
CHAPTER 4
Question
1) 2) 3) 4) 5)
Answer
b d c b e
Comments
The relation object is a logical object, so it cannot be displayed on a canvas.
6)
The relation object is always owned by the master block. The Delete Record Behavior property affects the way records are deleted. The Deferred property indicates whether or not the form should defer (wait) execution of the query for the detail block. If it is set to No, the query will not be deferred and the detail block will be populated along with the master block.
7)
go to contents
448
Answer
c c a d
Comments
It also lets you control the format in which information is entered into the form. Items can only be seen by the user if their Visible property is set to True and they are positioned on a canvas that is also visible.
5) 6)
b d
Question
1)
Answer
d
Comments
Answer b is incorrect because while buttons have a Prompt property, it should not be used to communicate their function.
2) 3) 4) 5) 6)
b a a d d
CHAPTER 6
Question
1) 2)
Answer
a a
Comments
Answer c is incorrect because the item-level WHENBUTTON-PRESSED trigger with Override set to Yes will cause the form to override or ignore WHEN-BUTTONPRESSED triggers that are higher up in the hierarchy. Many item-specific triggers can be set at the block and form level as well to increase their scope. Answer c is incorrect because Post triggers do not replace default processing. They fire after the default processing has occurred. Answer c is incorrect because When triggers can also respond to internal processing events as demonstrated with the Validate Item event and WHEN-VALIDATE-ITEM trigger.
3) 4)
d b
5)
6) 7) 8)
b b b
WHEN-NEW-FORM-INSTANCE triggers can be created at the form and block level, but not the item level.
go to contents
449
Question
1)
Answer
b
Comments
Answer c is incorrect because POST-QUERY triggers can only be attached at the block or form level. Moreover, no trigger can be attached to a record group or any other object that is not an item, block, or form. None of the other answers are true because POST-QUERY triggers cannot be attached to items. Although it was not mentioned in the Lab text, you probably noticed that the PL/SQL Editor formats your code to make it easier to read by indenting and coloring key words.
2) 3)
b d
4) 5) 6)
d b c
Question
1) 2) 3) 4) 5)
Answer
c d c d b
Comments
Answer a is only true if the built-in is restricted and the trigger fires because navigation has occurred.
6)
If you want to pass an item name to a built-in, you must use the following syntax: 'BLOCK.ITEM'. If you include the colon and exclude the single quotes as in Answer b, the built-in will not compile. If you answered a, you are half right. Up until now, you have only used FIND_ built-ins to get object IDs. However, later in the book, you will also use a GET_ built-in to get the object ID of a parameter list.
CHAPTER 7
Question
1) 2) 3) 4) 5) 6) 7) 8)
Answer
d b d d b b a b
Comments
go to contents
450
Question
1) 2) 3) 4) 5) 6)
Answer
c c a c d c
Comments
CHAPTER 8
Question
1) 2) 3) 4) 5) 6) 7)
Answer
c c d b a d b
Comments
Answer c is incorrect because content is a type of canvas. Note that the answer says well-suited and not mandatory. Dialog windows are usually modal, although they do not have to be.
Answer c is incorrect because the MDI window does not have a Visible property.
Question
1) 2) 3) 4)
Answer
a d c b
Comments
Two content canvases can be assigned to the same window, but they cannot appear at the same time.
A window can be closed with navigation if its Hide on Exit property is set to Yes.
Question
1) 2) 3) 4) 5)
Answer
a a a d d
Comments
You can position items on a stacked canvas as you did in the Exercises for Lab 7.3. Stacked canvases are always positioned relative to a content canvas, not relative to a window.
go to contents
451
Question
1) 2) 3) 4)
Answer
c d d d
Comments
Toolbars can only be attached to window objects or the MDI window.
CHAPTER 9
Question
1) 2) 3)
Answer
d d b
Comments
Answer c is incorrect because copying an object does not create a subclassed version of that object. Answer a is incorrect because the Object Navigator only shows if an object has been subclassed, not what its source is.
4) 5)
c d
Question
1) 2) 3) 4) 5) 6) 7) 8)
Answer
c d a d c d c d
Comments
Visual attributes only apply font and color properties, not physical properties like Canvas. Any object that has font and color properties can inherit from a visual attribute.
Answer c is true because if the same property is set by both a visual attribute and a property class, the visual attributes setting will take precedence.
Question
1) 2) 3) 4)
Answer
a c d b
Comments
Answer a is incorrect because an object group only contains pointers to objects, not actual instances of objects.
go to contents
452
Question
1) 2) 3) 4)
Answer
a d a b
Comments
Answer b is incorrect because object libraries should serve as the source for template forms. Answer b is incorrect because Forms will automatically rename a form that is based on a template form.
CHAPTER 10
Question
1) 2) 3) 4)
Answer
b d a d
Comments
Question
1) 2) 3) 4) 5)
Answer
c b b d d
Comments
Answers a and b are incorrect because PL/SQL Libraries are separate modules.
The statement :CONTROL.CURDATE directly references a block and item name, which would make it invalid in a PL/SQL Library.
Question
1) 2) 3) 4)
Answer
c a c c
Comments
go to contents
453
Answer
d c
Comments
Answer a is incorrect because the NO_ACTIVATE parameter will not make the calling form invisible. It will open the called form, but set the focus to the calling form. Answer a is incorrect because the value of a system variable only applies to a specific form. Parameter lists can contain more than one parameter.
3) 4) 5)
b d b
CHAPTER 12
Question
1)
Answer
d
Comments
Answer a is incorrect because you cannot create reports in the Form Builder. You must do it in the Report Builder. It is only possible to invoke the Report Builder from the Form Builder.
2) 3) 4)
a a b
Answer a is incorrect because RUN_REPORT_OBJECT cannot accept report object names. Answer c is incorrect because RUN_PRODUCT_OBJECT is not a valid built-in. Answer d is incorrect because the single quotes have been left off the NULL parameter list value.
Question
1) 2) 3) 4)
Answer
a d d d
Comments
CHAPTER 13
Question
1) 2)
Answer
d c
Comments
The switch parameter can sometimes be confusing. A good way to remember how to set the switch parameter
go to contents
454
Question
1) 2) 3)
Answer
d c d
Comments
go to contents
A P P E N D I X
455
go to contents
456
Expand the HKEY_LOCAL_MACHINE folder to see its sub-folders. Working with the folders under HKEY_LOCAL_MACHINE will let you change Registry values for the local machine. So, any Windows user that logs into this machine will be affected by the Registry values set here. Expand the SOFTWARE node and scroll down so that you can see the folder named ORACLE. Select the ORACLE folder, but do not expand it. The Registry Editor should now appear as it does in Figure B.2.
go to contents
457
Figure B.2 I The Registry Editor with the ORACLE folder selected.
The hierarchy in the left pane of the window is displaying a list of Registry keys. HKEY_LOCAL_MACHINE and SOFTWARE are both Registry keys. The list in the right pane of the window is showing a list of Registry values. Scroll down in the right pane until you see a Registry value called FORMS60_PATH.
FORMS60_PATH
The FORMS60_PATH Registry value is a Forms environment variable that contains a list of directory paths. Whenever the Forms Runtime, Form Compiler, or Form Builder needs to find or reference a forms module, it looks to FORMS60_PATH to see which directory path that module might be stored in.
I FOR EXAMPLE:
Chapter 11, Multiple-Forms Applications, includes an example in which you run one form from another using the following command: OPEN_FORM('STUDENT'); Note that this command does not include the path to the STUDENT module file. It simply uses the modules name. For Forms to open this module, it must search the directory paths listed in the FORMS60_PATH Registry entry until it finds the module STUDENT.fmx. If it cannot find STUDENT, you would receive the following error: FRM-40010: Cannot read form STUDENT
go to contents
458
If C: is not your drive letter, please be sure to substitute the drive letter you are using in the tasks that follow.
This is also the path you will use to save all of the files you will create throughout the Exercises. Therefore, you must add this path to the FORMS60_PATH Registry value.
At this time, you should only edit the FORMS60_PATH Registry value. Changes to other Registry values could make your system unstable. If you are interested in exploring the Registry further, it is strongly recommended that you first read the manuals for your operating system.
Take the following steps to edit the Registry: 1) 2) Double-click the FORMS60_PATH value to open the Edit String dialog as shown in Figure B.3. Use the cursor keys to navigate to the very front of the string. Be careful not to delete the entries that are already there. If you accidentally delete the existing entries, simply click the Cancel button and then re-open the Edit String dialog.
Figure B.3 I The Edit String dialog for editing Registry values. go to contents
459
4)
Make sure you include the semi-colon to separate your new entry from the one that comes after it. Again, if C: is not your drive letter, be sure to substitute the appropriate one. Click the OK button.
The Registry has been updated, so now Forms will be able to reference any modules that are stored in C:\guest\forms\exercises.
REPORTS60_PATH
The REPORTS60_PATH Registry value is the environment variable that stores directory paths that Oracle Reports uses to search for modules. You will not be building Oracle Reports in this book, but you will be calling them from forms. Therefore, you will need to adjust the REPORTS60_ PATH as well. Take the following steps: 1) 2) Double-click the REPORTS60_PATH value to open the Edit String dialog. Repeat Steps 24 above.
REGISTRY ADVANTAGES
The beauty of environment variables and how the Registry uses them is that you are saved from having to hard-code the paths to your files into your applications. The modules are able to find each other using the environment variables in the Registry without having to know the exact path. Therefore, you do not have to create identical directory structures on every machine that uses your application.
I FOR EXAMPLE:
Assume you had used the following command to open a form: OPEN_FORM('C:\applications\forms\modules\student.fmx'); By doing this, you would force every user and every developer who wanted to work with this form to have the same directory structure on their machines. This would seriously complicate the deployment and maintenance of your applications. Thus, the Registry is important during development as well.
go to contents
460
I FOR EXAMPLE:
Chapter 9, Reusable Objects, uses subclassing to have an object in one form reference an object in another form to inherit some of its properties. Subclassed objects maintain a link with their source objects. To maintain that link, the module containing the subclassed object uses the FORMS60_ PATH Registry value to find the module that contains its source objects. This becomes even more important if you plan to share modules and objects with other developers. As long as the source and subclassed modules are saved in directory paths that FORMS60_PATH can find, you can easily share with other developers and still maintain the links established by subclassing.
go to contents
INDEX
A Advanced options page, LOV Wizard, 231, 24041 Alerts, 25, 28, 24858, 271 creating, 249, 25153 displaying, 24953 as examples of simple dialog windows, 271 with one button, 249 with two buttons, 249 Alert Style property, 255 Attach dialog, 369 Attached Libraries node, Object Navigator, 372 AUDIT_ITEMS property class, 34546 Automatic Display property, 243 Automatic Position property, 243 Automatic Refresh property, 241 B Background Color property, 12 Basetable blocks, 1416, 19, 107 Bevel property, 89 Binary files, 5354 compiling into executable files, 5556 Blocklevel triggers, 28 Blocks, 2526, 27 and canvases, 27 and items, 27 physical properties, 25 Boolean menu items, 423 Builtins, defined, 89 Buttons, 3, 144, 15758 creating, 14748 labels, 254 putting simple code behind for, 14850 Button tool, 134 C Calculation Mode property, 13940 CALL_FORM builtin, 39293 Canvases, 1113, 1718, 2425, 93, 262, 266312 and blocks, 27 content canvases, 27786 creating, 27885 displaying, 267 properties, 13 stacked canvases, 28797 toolbar canvases, 298310 and triggers, 25 understanding, 26970 viewports (views), 26667 Canvases node, 67 Canvas page, Layout Wizard, 3738 Canvas property, 133 Canvas viewports, 26667 Character coordinate system, 49 Check boxes, 3, 22, 14446, 16568 compared to check boxes in paperbased forms, 166 creating, 15556 states, 165 Check Box Mapping of Other Values property, 166 Check menu items, 423 CLEAR_ALL_MASTER_DETAILS, 26, 119 CloseAllowed property, MAINWIN, 319 Collapse All button, Object Navigator, 74 Collapse button, Object Navigator, 74 Column display page, LOV Wizard, 23031, 23940 Column Name property, 137 Column selection page, LOV Wizard, 230, 23839 COMMIT_FORM, 213 COMMIT_FORM statement, 39798 Companion Web site, xiii Compilation Errors window, 59, 61 Compile dialog, 36869 Compile option, 59, 368 Compile Selection option, 368 Content canvases/windows, 266, 27786 creating, 27885 Conventions, xvi Coordinate System settings, 49 COPY builtin, 2078 COURSE.fmb, 16 COURSE_INFORMATION, 26 COURSE_SECTION, 26 Create button, Object Navigator, 81, 183
461
go to contents
462
Index
Fill Color, 101 Filter Before Display property, 241 FIND_ALERT builtin, 250, 257 FIND_BLOCK builtin, 215 FIND_ builtins, 215, 21921, 394 FIND_WINDOW builtin, 22021 Finish page, 38 Data Block Wizard, 38 Layout Wizard, 4142 Firing, of triggers, 3 .fmb files, 53, 5860 .fmx files, 5354, 5860, 320 Foreground Color, 101 Format Mask property, 136 Formatting toolbar, Layout Editor, 9495, 97 Form Builder, 2, 47, 1718, 27, 32, 46, 5354, 58, 75, 119, 14849, 157, 177, 252, 320, 404, 42425 View menu, 42324 Form Compiler, 54, 59, 320 Formlevel triggers, 28 FORMPL/SQL EditorTRIGGER FAILURE exception, 204 Forms, use of term, 16 Forms builtins, 21323 defined, 89 FIND_ builtins, 215, 21921, 394 GET_ builtins, 214, 21718 SET_ builtins, 21415, 21819 using, 21617 Forms help system, 94, 17374 FORMS_MDI_WINDOW, 272 Forms menus, 42144 menu modules, 42134 menu security, 43543 Forms module, use of term, 16 Forms Runtime, 54, 177, 425 Forms triggers, 9 FORM_TRIGGER FAILURE, 2045 Frames: and canvases, 18, 24 defined, 13 Freeze/Unfreeze button, Property Palette, 90 G GET_APPLICATION_PROPERTY builtin, 21619 GET_BLOCK_PROPERTY, 217 GET_ builtins, 214, 21718 GET_CANVAS_PROPERTY, 217 GET_ITEM_PROPERTY, 217 Global variables, 38283 GO_BLOCK builtin, 296 GO_BLOCK trigger, 221
CREATED_BY, 209 CREATED_DATE, 209 D Database Item property, 13637, 202 Database Objects node, Object Navigator, 81 Data block page, Layout Wizard, 3839 Data Blocks node, Object Navigator, 67, 7374, 76 Data Block Wizard, 32, 3337, 4243, 4650, 126, 146, 197 finish page, 38 opening, 33, 42, 46 pages, 33 table page, 3437 type page, 34 using, 4243 Data Type property, 135 Default Alert Button property, 256 Default menu form, 42933 DEFAULT.mmb, 429 DEFAULT_WHERE property, 397 Delete Record Behavior property, 120 DEMO_OBJECTS, 26 Development environment, 6583 Layout Editor, 93104 Object Navigator, 6683 Property Palette, 8492 Dialog windows, 26264, 271 Direct references, 364 Display items, 3, 22, 127 creating/defining, 131133, 138141 creating/defining without the wizard, 12730, 13339 Display Without Privilege property, 441 Document windows, 26264, 270 DO_KEY builtin, 21011 E Edge Foreground, 101 Enter Query button, 122 Enter Query mode, Form Builder, 57, 61 .Err extension, 59 Events, 23 and items/triggers, 4 Executable files, 5354, 57 Execute Query button, 122, 197 Execution Hierarchy property, 183 EXIT_FORM, 213 Expand All button, Object Navigator, 74 Expand button, Object Navigator, 74 F Fetch mode, Form Builder, 57, 61 Fill button, 1012
go to contents
Index
GO_FORM builtin, 391 GO_ITEM builtin, 265, 267, 27374 GO_ITEM trigger, 221, 283 Graphics, 93 GUI Design Essentials (Weinschenk/Jamar/Yeo), 125 H HIDE_HIST trigger, 284 HIDE_VIEW builtin, 275, 296 HIDE_WINDOW builtin, 27374 Horizontal Toolbar property, 334 I ID_NULL builtin, 221, 257 Indirect references in library code, 364 Initial Value property, 13536, 162, 167 Interface event, 2 Interface object, 11 Internal processing event, 23 Itemlevel triggers, 28 Item Roles property, 440 Items, 34, 1112, 1718, 27, 12569 and blocks, 27 buttons, 144, 15758 creating, 14748 utting simple code behind, 14850 check boxes, 14446, 16568 creating, 15556 controlling programmatically at runtime, 23 display items, 127 creating/defining, 13133, 13841 and events/triggers, 4 identifying type of, 22 list items, 14445 creating, 15052 nullcanvas, 133 properties, 1112 radio groups, 14445, 16365 creating, 15355 text items, 12627 creating/defining without the wizard, 12730, 13338 Items page: Layout Wizard, 3940 LOV Wizard, 232, 24143 Item Type property, 143, 153 Item types, choosing, 2223 K Keyboard Navigable property, 283 KEYDELREC trigger, 19799, 21011 KEYEXEQRY trigger, 19798, 20910 KEYEXIT trigger, 257 Key triggers, 174, 175, 189, 20911 creating, 19799 Knowledge requirements, xv
463
L Label property, 157 Layout Data Block property, 24 Layout Editor, 13, 22, 23, 41, 50, 66, 77, 85, 93104, 109, 134, 147, 153, 249, 267, 287, 292, 404 arranging/sizing objects, 9799 Button tool, 99100 creating/formatting objects, 9597 Formatting toolbar, 9495, 97 Run Form Client/Server button, 60 Size Objects window, 102 toolbars, 9394 Tool Palette, 95 Update Layout button, 1023 Utility toolbar, 94, 283 Layout Wizard, 32, 33, 3742, 4446, 48, 5051, 111, 138, 262, 292 canvas page, 3738 data block page, 3839 Data Block tab page, 51 finish page, 4142 items page, 3940 pages, 37 records page, 4041 style page, 40 Line Color, 101 List Elements dialog box, 15960, 162 List items, 22, 14445 creating, 15052 Lists of values (LOVs), 28, 22647 Automatic Display property, 243 Automatic Position property, 243 creating, 22832 displaying, 23335 LIST_VALUES builtin, 24445 LOV Wizard, 22829, 23543 advanced options page, 231, 24041 column display page, 23031, 23940 column selection page, 230, 23839 items page, 232, 24143 LOV display page, 231 source page, 229, 236 SQL query page, 22930, 237 and record groups, 22728 SHOW_LOV builtin, 24546 LIST_VALUES builtin, 24445 List of Values property, 242 LOV Column Mapping dialog, 271 LOV display page, LOV Wizard, 231
go to contents
464
Index
defined, 435 forms schema objects, building, 43739 implementing, 43639 sample database users/roles, running script to create, 43637 MESSAGE builtin, 209, 248 .mmb files, 58, 425 .mmx files, 425 Modal windows, 26364 MODIFIED_BY, 208 MODIFIED_DATE, 208 Modules, 16, 20 Mouse Navigate button, 158 Multi Line property, 135 Multiple Document Interface (MDI) window, 263, 272 Multipleforms, 381402 calling form compared to called form, 382 form name, 382 global variables, 38283 opening, 38587 parameter list, 38485 creating/passing to form, 38798 parameters, 384 passing values to called forms, 38385 N Name of Records Displayed property, 138 NEW_FORM builtin, 39293 New Program Unit dialog box, 356, 368 NEXT_FORM builtin, 391 Nonbasetable blocks, 1416 Normal mode, Form Builder, 57, 61 Nullcanvas item, 133 Number of Items Displayed property, 138 Number of Records Displayed property, 14041 O Object groups, 33637 creating/reusing, 33840 Object libraries, 33738 creating and utilizing, 34143 and LIBRARY window, 337 Object Navigator, 59, 22, 2324, 46, 50, 6683, 88, 118, 128, 148, 197, 292, 29495, 336, 356, 369, 404, 424, 440 Collapse All button, 74 Collapse button, 74 Create button, 81, 183 creating/deleting objecs, 6970 cutting and pasting objects, 7071 database objects, viewing, 7273
LOV Wizard, 22829, 23543 advanced options page, 231, 24041 column display page, 23031, 23940 column selection page, 230, 23839 items page, 232, 24143 LOV display page, 231 source page, 229, 236 SQL query page, 22930, 237 M Magic menu items, 423, 42931 Mandatory forms, 1129 basetable blocks, 1416, 19 canvases, 1113, 18, 2425 elements, relating, 17, 2021 items, 1112, 1718 modules, 16, 20 nonbasetable blocks, 1416 windows, 14 Mapping of Other Values property, 161, 164, 166 Master block, and relation object, 118 Masterdetail forms, 25, 10724 Copy Value from Item property, 11819 creating, 11012 defined, 108 Delete Record Behavior property, 120 master block, 11518 table based on, 115 ONCHECKDELETEMASTER trigger, 120 ONCLEARDETAILS trigger, 119 ONPOPULATEDETAILS trigger, 11920, 172 purpose of, 115 relation, 109 relation object, creation of, 109 SECTION block, 11718 wizards, 109, 11516 working with relations and, 11215 Masterdetail relation, 25 menudef.mmb, 429 Menu Editor, 27, 248, 425, 426, 429 Menu Module property, 334 Menu Module Roles dialog, 43940 Menu modules, 42134 attaching to forms, 425 creating/configuring, 42428 default menu form, 42933 menu items, 42324 creating, 42628 types of, 423 menu objects, 42223 Menu security, 43543 configuring, 436
go to contents
Index
Database Objects node, 81 Data Blocks node, 67, 7374, 76 dragging/dropping objects, 7071 Expand All button, 74 Expand button, 74 hierarchy, 67 opening/identifying objects, 6869 Run Form Client/Server button, 60 running/saving forms, 72 vertical toolbar, 67, 68 ONCHECKDELETEMASTER trigger, 119, 120, 189 ONCLEARDETAILS trigger, 26, 119 ONDELETE trigger, 184 ONINSERT trigger, 174 ONPOPULATEDETAILS trigger, 11920, 17273, 189 OPEN_FORM builtin, 391 Oracle Developer 6.0 Online Manuals, 162 Oracle Developer, 2, 53, 59, 157 Oracle forms, 210, 60 button clicks, 7 events, 23 files, 5364 binary, 5354 compiling, 54 compiling binary files into executable files, 5556 executable, 5354, 57 fmb and .fmx files compared, 5455, 58 running, 54 how they work, 47 items, 34, 22 physical objects in, 25 triggers, 23, 9, 2425, 172212 Oracle Graphics, 2 Oracle reports, 2 running from forms, 40413 running with RUN_PRODUCT builtin, 4067 running with RUN_REPORT_OBJECT builtin, 4078 P Parameters: parameters in reports modules, 41516 passing to reports, 41419 Pinning mode, 100 P_item parameter, 35960 Plain menu items, 423, 429 .pll files, 58 PL/SQL, using in triggers, 17576 PL/SQL Editor, 149, 15859, 200, 2013, 356, 404 PL/SQL libraries, 59, 36273
465
advantages of, 36364 creating/attaching, 36466 defined, 362 editing/changing code in, 372 indirect references in library code, 364 using, 36667 modules, saving, 362 POSTBLOCK trigger, 221 POSTDATABASECOMMIT trigger, 189 POSTFORMSCOMMIT trigger, 189 POSTINSERT trigger, 174, 35455 POSTQUERY trigger, 184, 18788, 19091, 19394, 199200, 202, 2045, 221, 354 PRECHANGE trigger, 174 Pre Compute Summaries property, 141 PREFORM trigger, 177, 272 PREINSERT trigger, 174, 189, 196, 2078, 343, 371 PRERECORD trigger, 221 PRETEXTITEM trigger, 221 PREUPDATE trigger, 189, 343, 372 PREVIOUS_FORM builtin, 391 Primary Canvas property, 273 Program units, 35461 creating, 35758 defined, 354 subclassing, 356 syntax for creating, 359 Prompt Alignment Offset property, 139 Prompt Attachment Offset property, 139 Prompt property, 167 Properties, items, 1112 Property classes, 32425, 333 creating/applying, 32729 with multiple properties, 332 Property Palette, 1213, 22, 2324, 49, 66, 78, 8492, 404 accessing, 88 Canvas property, 133 changing properties, 8588 Freeze/Unfreeze button, 90 icons indicating property state, 319 Item Type property, 143, 153 Prompt Alignment Offset property, 139 Prompt Attachment Offset property, 139 property categories, 84 viewing properties, 8486 Visible property, 133 Push buttons, 22 Q Query All Records property, 141
go to contents
466
Index
Size Objects window, 102 Software requirements, xiiixiv Source page, LOV Wizard, 229, 236 SQL, using in triggers, 17576 SQL query page, LOV Wizard, 22930, 237 Stacked canvases, 28797 creating/displaying, 28992 defined, 28788 INSTRUCTOR stacked canvas, 29394 viewport, 288 Stacked Canvas tool, Tool Palette, 293 Stored PL/SQL objects, 37478 calling, 37475 moving between database and Forms application, 375 using, 37577 Style page, Layout Wizard, 40 Subclass Information property, 91, 318, 321, 344 Subclassing, 80, 31422 Subclass Information dialog box, 33233 subclass objects, 31517 Summary Function property, 139 T TABLE_ITEM_PROMPT_ALIGNMENT, 26 Table page, 3537 Data Block Wizard, 3437 Text Color button, 1012 Text fields, 3 Text items, 3, 22, 12627 creating/defining without the wizard, 12730, 13338 Enabled property, 13435 Text Item tool, 134 Toolbar canvases, 298310 creating, 299302 TOOLBAR canvas, 3056, 31415 using in another form, 3023 Tool Palette, 99100, 134, 147 Layout Editor, 95 Stacked Canvas tool, 293 Transactional triggers, 175, 189 creating, 19596 Triggers, 23, 9, 2425, 172212 and canvases, 25 categorizing, 17375, 17879 by function, 17475 by name, 174 and events/items, 4 forms triggers, 9 key triggers, 174, 175, 189, 20911 creating, 19799 On event triggers, 174 Post event triggers, 174
Query Array Size property, 14041 QUERY_MASTER_DETAIL trigger, 26, 11920 Query triggers, 175, 18788 creating, 18992 R Radio Button node, 163 Radio groups, 3, 22, 14445, 16365 creating, 15355 radio buttons in, 165 Radio menu items, 423 Real coordinate system, 49 Record Group Fetch Size property, 241 Record groups, and LOVs, 22728 Record History window, 33132 Records page, Layout Wizard, 4041 Relation object: creation of, 109 and master block, 118 Relations, 25, 107 Report Builder, 404 Reports Runtime, 404, 409 Reusable code, 35361 PL/SQL libraries, 36273 program units, 35461 stored PL/SQL objects, 37478 Reusable objects, 31352 subclassing, 80, 31422 Subclass Information dialog box, 33233 subclass objects, 31517 RUN_PRODUCT builtin, 404, 405 running Oracle reports with, 4067 RUN_REPORT_OBJECT builtin, 404, 4056 running Oracle reports with, 4078 S Sample schema, xvxvi SECONDCAN, 273 SECONDWIN, 27374 Separators, 424 SET_ALERT_PROPERTY builtin, 359 SET_ALERT_PROPTERY builtin, 257 SET_BLOCK_PROPERTY, 218, 397 SET_ builtins, 21415, 21819 SET_CANVAS_PROPERTY, 218 SET_ITEM_PROPERTY builtin, 89, 218 SET_VIEW_PROPERTY builtin, 296 SET_WINDOW_PROPERTY builtin, 218 SHOW_ALERT builtin, 250, 257 Show Errors window, 202 SHOW_HIST trigger, 273, 284 SHOW_LIST button, 245 SHOW_LOV builtin, 24546, 249 SHOW_VIEW builtin, 267, 275, 296, 355
go to contents
Index
Pre event triggers, 174 query triggers, 175, 18788 creating, 18992 scope, 17273, 17677 transactional triggers, 175, 189 creating, 19596 using PL/SQL and SQL in, 17576 validation triggers, 175, 18889 creating, 19295 When event triggers, 174 Type page, 34 Data Block Wizard, 34 U Update Layout property, 24, 98, 1023 Use Security property, 440 Utility toolbar, Layout editor, 94, 283 V VAL_ALERT alert, 357 Validate Item event, 3 Validating an item, 3 Validation triggers, 175, 18889 creating, 19295 Value When Checked property, 16567 Value When Unchecked property, 16567 View menu, Form Builder, 42324 Viewport Height property, 275 Viewports, 26667 Viewport Width property, 275 Visible property, 133 Visual attributes, 323 behavior of, 330 creating/applying, 32527 Visual Attribute Type property, 33031 Common value, 330 Prompt value, 331 Title value, 331
467
W WHENBUTTONPRESSED trigger, 78, 144, 158, 173, 181, 198, 209, 251, 25455, 27273, 28081, 294, 354, 387 WHENCHECKBOXCHANGED trigger, 9, 23, 167 WHENNEWFORMINSTANCE trigger, 26, 55, 216, 220, 389 WHENVALIDATEITEM triggers, 4, 47, 17677, 17983, 18889, 19394, 2046, 353, 35758, 360 WHENVALIDATERECORD trigger, 206 WINDOWCLOSED trigger, 3 Windows, 14, 26269 content windows, 27786 defined, 262 dialog windows, 26264, 271 displaying, 26566 document windows, 26264, 270 modal windows, 26364 Multiple Document Interface (MDI) window, 263, 272 styles of, 262 understanding, 26769 Wizards, 3252, 107, 288 Data Block Wizard, 32, 3337, 4243, 4650 Layout Wizard, 32, 33, 3742, 4446, 48, 5051 reentering, 4345 Wrap Style property, 135 Www.oracle.com, 59 Www.phptr.com/phptrinteractive, 4 WYSIWYG, 1213
go to contents