After five releases, YaST2 is now smart enough to make reasonable proposals for (near) every installation setting, thus it is no longer necessary to ask the user that many questions during installation: Most users simply hit the [next] button anyway.
Hence, YaST2 now collects all the individual proposals from its submodules and presents them for confirmation right away. The user can change each individual setting, but he is no longer required to go through all the steps just to change some simple things. The only that (currently) really has to be queried is the installation language - this cannot reasonably be guessed (yet?).
The new YaST2 installation includes the following steps:
(Minimal) hardware probing - no user interaction required
Language selection - user picks installation language
Installation proposal - very much like the old installation summary just before the real installation started, only this time the user CAN change settings by clicking into the summary (or via a separate menu as a fallback).
Create / format partitions according to proposal / user selection - no user interaction required
Install software packages from CD / DVD / other installation media
After this, all that is remained left is basic system configuration like:
X11
Network interface(s)
Network services
Additional hardware (printer, sound card, scanner, ...)
YaST2 installation modules should cooperate with the main program in a consistent API. General usage:
inst_proposal (main program) creates empty dialog with RichText widget
inst_proposal calls each sub-module in turn to make proposal
user may choose to change individual settings (i.e., clicks on a hyperlink)
inst_proposal starts that module's sub-workflow which runs independently. After this, inst_proposal tells all subsequent (all?) modules to check their states and return whether a change of their proposal is necessary after the user interaction.
main program calls each sub-module to write the settings to the system
Each submodule provides a function dispatcher that can be called with 'CallFunction()'. The function to be called is passed as a parameter to this dispatcher. Parameters to the function are passed as another parameter in a map. The result of each call is a map, the contents of which depend on the function called.
The reason for this additional overhead is to provide extensibility and reusability for the installation workflow: A list of submodules to be called is read from file. This requires, however, that no explicit 'mod::func()' calls are used in 'inst_proposal.ycp'. Rather, the list contains the name of the submodule. Since each submodule is required to provide an identical API, this is sufficient.
Example 9.2. Proposal Example
Proposal is to call
input_devices (keyboard, mouse)
partitioning
software_selection
boot_loader
timezone
inst_proposal calls
map result = CallFunction (input_devices( "MakeProposal", $[ "force_reset" : false, "language_changed": false ] ) ); map result = CallFunction (partitioning ( "MakeProposal", $[ "force_reset" : false, "language_changed": false ] ) ); ... |
If the user clicks on the hyperlink on "input_devices" in the proposal display, inst_proposal calls:
map result = CallFunction (input_devices( "AskUser", $[ "has_next": true ] ) ); |
![]() | Note |
---|---|
If any parameter is marked as "optional", it should only be specified if it contains a meaningful value. Don't add it with a 'nil' value. |
The dispatcher provides the following functions:
/** * Module: proposal_dummy.ycp * * $Id: dummy_proposal.ycp,v 1.1 2004/02/27 02:37:39 nashif Exp $ * * Author: Stefan Hundhammer <sh@suse.de> * * Purpose: Proposal function dispatcher - dummy version. * Use this as a template for other proposal dispatchers. * Don't forget to replace all fixed values with real values! * * See also file proposal-API.txt for details. */ { textdomain "installation"; string func = (string) WFM::Args(0); map param = (map) WFM::Args(1); map ret = $[]; if ( func == "MakeProposal" ) { boolean force_reset = param["force_reset" ]:false; boolean language_changed = param["language_changed"]:false; // call some function that makes a proposal here: // // DummyMod::MakeProposal( force_reset ); // Fill return map ret = $[ "raw_proposal" : [ "proposal item #1", "proposal item #2", "proposal item #3" ], "warning" : "This is just a dummy proposal!", "warning_level" : `blocker ]; } else if ( func == "AskUser" ) { boolean has_next = param["has_next"]:false; // call some function that displays a user dialog // or a sequence of dialogs here: // // sequence = DummyMod::AskUser( has_next ); // Fill return map ret = $[ "workflow_sequence": `next ]; } else if ( func == "Description" ) { // Fill return map. // // Static values do just nicely here, no need to call a function. ret = $[ // this is a heading "rich_text_title" : _( "Dummy" ), // this is a menu entry "menu_title" : _( "&Dummy" ), "id" : "dummy_stuff" ]; } else if ( func == "Write" ) { // Fill return map. // ret = $[ "success" : true ]; } return ret; } |